w
wolkov91Қыр. 18, 2018, 9:38 Т.Ж.

Прокси-модель, содержащая на 1 столбец больше, чем модель-источник.

qt5, model, proxy, QIdentityProxyModel, column, PyQt5

Здравствуйте!

Пытаюсь переопределить QIdentityProxyModel так, чтобы она имела на 1 столбец больше, чем ее модель-источник, но никак не могу добиться успешного результата.

Вот заготовка, очищенная от мусора:


#!/usr/bin/python3
# -*- coding: utf-8 -*-

"""Попытка реализовать прокси-модель, содержащую на 1 столбец больше, чем модель-источник."""
import sys
import typing

from PyQt5.QtCore import *
from PyQt5.QtGui import *
from PyQt5.QtWidgets import *


app = None


class MyProxyModel(QIdentityProxyModel):
    """Прокси-модель, содержащая на 1 столбец больше, чем модель-источник."""

    VALUE = "==="

    def columnCount(self, parent: QModelIndex = None) -> int:
        """Переопределяет соответствующий родительский метод.
        columnCount(self, parent: QModelIndex = QModelIndex()) -> int
        """
        if not self.sourceModel():
            return 0
        return super().columnCount() + 1

    def data(self, proxyIndex: QModelIndex, role: int = None) -> typing.Any:
        """Переопределяет соответствующий родительский метод.
        data(self, index: QModelIndex, role: int = Qt.DisplayRole) -> Any
        """
        assert self.checkIndex(proxyIndex)
        if role is None:
            role = Qt.DisplayRole

        if not self.sourceModel():
            return None

        if role == Qt.DisplayRole:
            if proxyIndex.column() == self.columnCount() - 1:
                # text = ""
                # for column in range(self.columnCount() - 1):
                #     # index = self.index(proxyIndex.row(), column, proxyIndex.parent())
                #     index = self.sibling(proxyIndex.row(), column, proxyIndex)
                #     text += "->" + str(super().data(index))
                # return text
                return self.VALUE

        return super().data(proxyIndex, role)

    def headerData(self, section: int, orientation: int, role: int = None) -> typing.Any:
        """Переопределяет соответствующий родительский метод.
        headerData(self, section: int, orientation: Qt.Orientation, role: int = Qt.DisplayRole) -> Any
        """
        if role is None:
            role = Qt.DisplayRole

        if orientation == Qt.Horizontal and role == Qt.DisplayRole and section == self.columnCount() - 1:
            return "New column"

        return super().headerData(section, orientation, role)

    # def mapFromSource(self, sourceIndex: QModelIndex) -> QModelIndex:
    #     # return super().mapFromSource(sourceIndex)
    #     if not self.sourceModel() or not sourceIndex.isValid():
    #         return QModelIndex()
    #
    #     assert sourceIndex.model() is self.sourceModel()
    #     return self.createIndex(sourceIndex.row(), sourceIndex.column(), sourceIndex.internalPointer())
    #
    # def mapToSource(self, proxyIndex: QModelIndex) -> QModelIndex:
    #     # return super().mapToSource(proxyIndex)
    #     if not self.sourceModel() or not proxyIndex.isValid():
    #         return QModelIndex()
    #
    #     assert proxyIndex.model() is self
    #     return self.sourceModel().createIndex(proxyIndex.row(), proxyIndex.column(), proxyIndex.internalPointer())


def main():
    global app
    app = QApplication(sys.argv)

    sourceModel = QFileSystemModel(app)
    sourceModel.setRootPath(".")

    proxyModel = MyProxyModel(app)
    proxyModel.setSourceModel(sourceModel)

    treeView = QTreeView()
    treeView.setMinimumWidth(800)
    treeView.setModel(proxyModel)
    treeView.setColumnWidth(0, 400)
    # TODO: По прежнему выделяет только ячейку. Надо переопределить MyProxyModel.mapSelection<From/To>Source()!
    treeView.setSelectionBehavior(treeView.SelectRows)
    # treeView.setSelectionMode(treeView.SingleSelection)
    treeView.show()

    def checkProxyModel():
        print("BEGIN checkProxyModel()")

        assert proxyModel.columnCount() == sourceModel.columnCount() + 1

        proxyIndexWithZeroColumn = proxyModel.index(0, 0)
        sourceIndexWithZeroColumn = sourceModel.index(0, 0)
        assert proxyModel.data(proxyIndexWithZeroColumn) == sourceModel.data(sourceIndexWithZeroColumn)

        proxyIndexWithPenultimateColumn = proxyModel.index(0, proxyModel.columnCount() - 2)
        sourceIndexWithLastColumn = sourceModel.index(0, sourceModel.columnCount() - 1)
        assert proxyModel.data(proxyIndexWithPenultimateColumn) == sourceModel.data(sourceIndexWithLastColumn)

        proxyIndexWithLastColumn = proxyModel.index(0, proxyModel.columnCount() - 1)
        assert proxyModel.data(proxyIndexWithLastColumn, Qt.DisplayRole) == proxyModel.VALUE

        print("END checkProxyModel()")

    QTimer.singleShot(0, checkProxyModel)

    sys.exit(app.exec())


if __name__ == '__main__':
    main()

В отображении появляется новый столбец с нужным заголовком, но без данных.

Я не понимаю, каким образом мне нужно переопределить методы mapFromSource/mapToSource, чтоб все заработало...  Или же вообще нужно переопределять index(), parent() и прочие?..

Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

Ол саған ұнайды ма? Әлеуметтік желілерде бөлісіңіз!

10
Evgenii Legotckoi
  • Қыр. 18, 2018, 1:56 Т.Қ.

Добрый день.

Вот рабочий вариант

#!/usr/bin/python3
# -*- coding: utf-8 -*-

"""Попытка реализовать прокси-модель, содержащую на 1 столбец больше, чем модель-источник."""
import sys
import typing


from PyQt5.QtCore import *
from PyQt5.QtGui import *
from PyQt5.QtWidgets import *


app = None


class MyProxyModel(QIdentityProxyModel):
    """Прокси-модель, содержащая на 1 столбец больше, чем модель-источник."""

    VALUE = "==="

    def columnCount(self, parent: QModelIndex = None) -> int:
        """Переопределяет соответствующий родительский метод.
        columnCount(self, parent: QModelIndex = QModelIndex()) -> int
        """
        if not self.sourceModel():
            return 0
        return super().columnCount() + 1

    def data(self, proxyIndex: QModelIndex, role: int = None) -> typing.Any:
        """Переопределяет соответствующий родительский метод.
        data(self, index: QModelIndex, role: int = Qt.DisplayRole) -> Any
        """

        print(proxyIndex.column())
        if role is None:
            role = Qt.DisplayRole

        if not self.sourceModel():
            return None

        if role == Qt.DisplayRole:
            if proxyIndex.column() == self.columnCount() - 1:
                return self.VALUE

        return super().data(proxyIndex, role)

    def index(self, row: int, column: int, parent: QModelIndex = QModelIndex()) -> typing.Any:
        if column == self.columnCount(parent) - 1:
            return self.createIndex(row, column)

        return super().index(row, column, parent)

    def headerData(self, section: int, orientation: int, role: int = None) -> typing.Any:
        """Переопределяет соответствующий родительский метод.
        headerData(self, section: int, orientation: Qt.Orientation, role: int = Qt.DisplayRole) -> Any
        """
        if role is None:
            role = Qt.DisplayRole

        if orientation == Qt.Horizontal and role == Qt.DisplayRole and section == self.columnCount() - 1:
            return "New column"

        return super().headerData(section, orientation, role)


def main():
    global app
    app = QApplication(sys.argv)

    sourceModel = QFileSystemModel(app)
    sourceModel.setRootPath(".")

    proxyModel = MyProxyModel(app)
    proxyModel.setSourceModel(sourceModel)

    treeView = QTreeView()
    treeView.setMinimumWidth(800)
    treeView.setModel(proxyModel)
    treeView.setColumnWidth(0, 400)
    # TODO: По прежнему выделяет только ячейку. Надо переопределить MyProxyModel.mapSelection<From/To>Source()!
    treeView.setSelectionBehavior(treeView.SelectRows)
    treeView.show()

    def checkProxyModel():
        print("BEGIN checkProxyModel()")

        assert proxyModel.columnCount() == sourceModel.columnCount() + 1

        proxyIndexWithZeroColumn = proxyModel.index(0, 0)
        sourceIndexWithZeroColumn = sourceModel.index(0, 0)
        assert proxyModel.data(proxyIndexWithZeroColumn) == sourceModel.data(sourceIndexWithZeroColumn)

        proxyIndexWithPenultimateColumn = proxyModel.index(0, proxyModel.columnCount() - 2)
        sourceIndexWithLastColumn = sourceModel.index(0, sourceModel.columnCount() - 1)
        assert proxyModel.data(proxyIndexWithPenultimateColumn) == sourceModel.data(sourceIndexWithLastColumn)

        proxyIndexWithLastColumn = proxyModel.index(0, proxyModel.columnCount() - 1)
        assert proxyModel.data(proxyIndexWithLastColumn, Qt.DisplayRole) == proxyModel.VALUE

        print("END checkProxyModel()")

    QTimer.singleShot(0, checkProxyModel)

    sys.exit(app.exec())


if __name__ == '__main__':
    main()

Вам надо было метод index переопределить, чтобы создавался QModelIndex для вашей спец-колонки



    w
    • Қыр. 19, 2018, 6:17 Т.Ж.

    Огромное спасибо за ответ!:)

    Но вариант все же не является рабочим.

    Почти всегда дерево отображается корректно и даже раскрываются новые уровни, если нажать точно на треугольник слева. Но если сменить выделенный элемент, например, щелкнуть мышкой на какую-нибудь ячейку, то приложение падает, и единственным выводом в консоль является:

    Process finished with exit code 139 (interrupted by signal 11: SIGSEGV)

    qVersion() == 5.11.1, OS == Kubuntu 18.04

    Также приложение падает, если вернуть в метод data() строчку

    assert self.checkIndex(proxyIndex)

    Изучение исходников метода index() в QFileSystemModel и QStandardItemModel ничем пока не помогло. :(

      Evgenii Legotckoi
      • Қыр. 19, 2018, 6:22 Т.Ж.

      Падение с assert - это уже питоновские заморочки, дайте туда более корректную проверку, поскольку проверка на None просто не работает адекватно в том случае.

      Что касается установки информации, то возможно, что ещё потребуется перепределить метод setData. Переопределите его и посмотрите через print, что туда прилетает в индексе и роли.

      print(proxyIndex.row(), proxyIndex.column(), role)

        w
        • Қыр. 19, 2018, 6:44 Т.Ж.

        До setData() я еще не добрался, так как пока что даже reed only модель не работает.

        В конкретном данном случае в

        assert self.checkIndex(proxyIndex)

        всегда передается экземпляр QModelIndex. Поэтому это не является проблемой обработки None.
        Да и на падения при смене текущей ячейки это не влияет.

        Я Вам очень благодарен за то, что все равно пытаетесь помочь!

          w
          • Қыр. 19, 2018, 8 Т.Ж.
          • (өңделген)

          Если определить метод index() таким костыльным и затратным образом:

          def index(self, row: int, column: int, parent: QModelIndex = QModelIndex()) -> QModelIndex:
              assert self.checkIndex(parent)
              if column == self.columnCount() - 1:
                  return self.createIndex(row, column, QPersistentModelIndex(parent))
          
              return super().index(row, column, parent)
          

          то приложение уже перестает падать, но строку вьюха выделить все еще не может, так как для модельного индекса из моего столбца internalPointer() хранит данные, отличающиеся от данных для других столбцов.

          Также в этом случае в консоль выводится соответствующее сообщение:

          Can't select indexes from different model or with different parents

            w
            • Қыр. 20, 2018, 5:15 Т.Ж.

            В итоге, эмулировать существование дополнительных столбцов через QIdentityModel на PyQt5(5.11.1) получается только для плоских (табличных, списочных) моделей.

            Для древовидных моделей при создании модельного индекса в internalPointer необходимо хранить некие данные о родителе. Но если установить в него данные можно, то при считывании PyQt не может определить их тип из-за того, что в С++ они объявлены как void*, если считывать через internalPointer(), и qintptr, если считывать через internalId().

            В следствие чего, программа крашится.

              Evgenii Legotckoi
              • Қыр. 21, 2018, 4:25 Т.Ж.

              Попробуйте ещё PySide 2 - это официально поддерживаемый пакет привязок Python к Qt, возможно, что там не будет таких проблем.

                w
                • Қаз. 1, 2018, 6:21 Т.Ж.

                Вот работающий вариант переопределения метода index():

                def index(self, row: int, column: int, parent: QModelIndex = QModelIndex()) -> QModelIndex:
                    """Переопределяет соответствующий родительский метод.
                    index(self, row: int, column: int, parent: QModelIndex = QModelIndex()) -> QModelIndex
                    """
                    if column == self.columnCount() - 1:
                        assert self.checkIndex(parent)
                        return self.createIndex(row, column, super().index(row, 0, parent).internalId())
                
                    return super().index(row, column, parent)
                

                Теперь корректно устанавливаются дынные об элементе-предке и программа больше не крашится.

                Но остается вопрос, как заставить вьюху менять выделение/текущий индекс при щелчке мышкой по ячейкам из нового столбца?..

                  w
                  • Қаз. 1, 2018, 9:30 Т.Ж.

                  Изучение исходников QTreeView и QAbstractItemView показало, что для осуществления обработки нажатий по элементам нового столбца необходимо переопределить метод sibling():

                  def sibling(self, row: int, column: int, index: QModelIndex) -> QModelIndex:
                      """Переопределяет соответствующий родительский метод.
                      index(self, row: int, column: int, parent: QModelIndex = QModelIndex()) -> QModelIndex
                      """
                      if column == self.columnCount() - 1:
                          assert self.checkIndex(index)
                          assert index.isValid()  # Это можно будет проверить в строке выше, когда в PyQt5 реализуют соотв. параметры.
                          return self.createIndex(row, column, super().index(row, 0, index.parent()).internalId())
                      return super().sibling(row, column, index)
                  
                    Evgenii Legotckoi
                    • Қаз. 1, 2018, 9:32 Т.Ж.

                    Полагаю, что вам придётся везде создавать соответствующий валидный индекс для дополнительного столбца, если что-то не работает

                      Пікірлер

                      Тек рұқсаты бар пайдаланушылар ғана пікір қалдыра алады.
                      Кіріңіз немесе Тіркеліңіз
                      OI
                      • Ora Iro
                      • Жел. 24, 2024, 6:38 Т.Ж.

                      C++ - Тест 001. Первая программа и типы данных

                      • Нәтиже:40ұпай,
                      • Бағалау ұпайлары-8
                      AD

                      C++ - Тест 004. Указатели, Массивы и Циклы

                      • Нәтиже:50ұпай,
                      • Бағалау ұпайлары-4
                      m
                      • molni99
                      • Қаз. 26, 2024, 1:37 Т.Ж.

                      C++ - Тест 004. Указатели, Массивы и Циклы

                      • Нәтиже:80ұпай,
                      • Бағалау ұпайлары4
                      Соңғы пікірлер
                      ИМ
                      Игорь МаксимовҚар. 22, 2024, 11:51 Т.Ж.
                      Django - Оқулық 017. Теңшелген Django кіру беті Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
                      Evgenii Legotckoi
                      Evgenii LegotckoiҚаз. 31, 2024, 2:37 Т.Қ.
                      Django - Сабақ 064. Python Markdown кеңейтімін қалай жазуға болады Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
                      A
                      ALO1ZEҚаз. 19, 2024, 8:19 Т.Ж.
                      Qt Creator көмегімен fb3 файл оқу құралы Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
                      ИМ
                      Игорь МаксимовҚаз. 5, 2024, 7:51 Т.Ж.
                      Django - Сабақ 064. Python Markdown кеңейтімін қалай жазуға болады Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                      d
                      dblas5Шілде 5, 2024, 11:02 Т.Ж.
                      QML - Сабақ 016. SQLite деректер қоры және онымен QML Qt-та жұмыс істеу Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                      Енді форумда талқылаңыз
                      Evgenii Legotckoi
                      Evgenii LegotckoiМаусым 24, 2024, 3:11 Т.Қ.
                      добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
                      t
                      tonypeachey1Қар. 15, 2024, 6:04 Т.Ж.
                      google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
                      NSProject
                      NSProjectМаусым 4, 2022, 3:49 Т.Ж.
                      Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…
                      9
                      9AnonimҚаз. 25, 2024, 9:10 Т.Ж.
                      Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…

                      Бізді әлеуметтік желілерде бақылаңыз