Заголовок QAbstractItemModel для дочерних таблиц

1

Я создал свой собственный QAbstractItemModel с древовидной структурой и несколькими подобными таблицами:

group1
  - order1
     - item1
     - item2
  - order2
     - item3
group2
  - order3
     - item4
     - item5
     - item6

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

Теперь мне, очевидно, нужны заголовки для табличного вида, вот где я несколько застрял. Я попытался реализовать решение из этого вопроса, но он работает неправильно. Это то, что я получил до сих пор:

import sys
from PyQt5 import QtCore, QtWidgets


class Node(object):

    def __init__(self, parent=None, item_count=0):
        self.parent = parent
        self.children = []
        self.item_count = item_count
        if self.parent is not None:
            self.parent.add_child(self)

    def add_child(self, child):
        self.children.append(child)
        child.parent = self

    @property
    def row(self):
        if self.parent is None:
            return 0
        return self.parent.children.index(self)

    @property
    def child_item_count(self):
        if len(self.children):
            return self.children[0].item_count
        return 1

    @property
    def child_count(self):
        return len(self.children)

    def child(self, row):
        try:
            return self.children[row]
        except IndexError:
            return None

    def header_data(self):
        return []


class ItemNode(Node):

    def __init__(self, parent=None, name='', description='', price=''):
        super(ItemNode, self).__init__(parent=parent, item_count=3)
        self.name = name
        self.description = description
        self.price = price

    def data(self, column):
        if column == 0:  return self.name
        if column == 1:  return self.description
        if column == 2:  return self.price
        return None


class OrderNode(Node):

    def __init__(self, parent=None, order_id='', order_date=''):
        super(OrderNode, self).__init__(parent=parent, item_count=2)
        self.order_id = order_id
        self.order_date = order_date

    def data(self, column):
        if column == 0:  return self.order_id
        if column == 1:  return self.order_date
        return None

    def header_data(self):
        return ['Name', 'Description', 'Price']


class GroupNode(Node):

    def __init__(self, parent=None, name='', description=''):
        super(GroupNode, self).__init__(parent=parent, item_count=2)
        self.name = name
        self.description = description

    def data(self, column):
        if column == 0: return self.name
        if column == 1: return self.description
        return None


class ItemModel(QtCore.QAbstractItemModel):

    def __init__(self, root=None):
        if root is None:
            root = Node()
        self.root = root
        super(ItemModel, self).__init__()

    def rowCount(self, parent=None, *args, **kwargs):
        node = self.get_node(parent)
        return node.child_count

    def columnCount(self, parent=None, *args, **kwargs):
        node = self.get_node(parent)
        return node.child_item_count

    def index(self, row, column, parent=None, *args, **kwargs):
        node = self.get_node(parent)
        return self.createIndex(row, column, node.child(row))

    def parent(self, index=None):
        node = self.get_node(index)
        if node.parent is None:
            return QtCore.QModelIndex()
        return self.createIndex(node.parent.row, 0, node.parent)

    def data(self, index, role=QtCore.Qt.DisplayRole):
        node = self.get_node(index)
        if role == QtCore.Qt.DisplayRole:
            return node.data(index.column())
        if role == QtCore.Qt.UserRole + 1:  # horizontal header role
            return node.header_data()

    def get_node(self, index):
        if index and index.isValid():
            node = index.internalPointer()
            if node:
                return node
        return self.root

    def is_order(self, index):
        node = self.get_node(index)
        return isinstance(node, OrderNode)


class ProxyItemModel(QtCore.QSortFilterProxyModel):

    horizontal_header_role = QtCore.Qt.UserRole + 1

    def __init__(self):
        self.root_index = QtCore.QModelIndex()
        super(ProxyItemModel, self).__init__()

    def set_root_index(self, index):
        self.root_index = self.mapToSource(index)
        if self.sourceModel() and self.root_index.internalPointer() is not None:
            self.headerDataChanged.emit(
                QtCore.Qt.Horizontal, 0, self.sourceModel().columnCount(self.root_index)
            )

    def headerData(self, section, orientation, role=QtCore.Qt.DisplayRole):
        if self.sourceModel() and self.root_index.isValid():
            if orientation == QtCore.Qt.Horizontal:
                role = self.horizontal_header_role
            header_list = self.sourceModel().data(self.root_index, role)
            if header_list and 0 <= section < len(header_list):
                return header_list[section]
        return super(ProxyItemModel, self).headerData(section, orientation, role)

    def is_order(self, index):
        source_index = self.mapToSource(index)
        return self.sourceModel().is_order(source_index)


class MainWindow(QtWidgets.QMainWindow):

    def __init__(self, *args, **kwargs):
        super(MainWindow, self).__init__(*args, **kwargs)

        self.central_widget = QtWidgets.QWidget(self)
        self.setCentralWidget(self.central_widget)
        central_layout = QtWidgets.QVBoxLayout(self)
        self.central_widget.setLayout(central_layout)

        self.treeView = QtWidgets.QTreeView(self)
        self.tableView = QtWidgets.QTableView(self)
        central_layout.addWidget(self.treeView)
        central_layout.addWidget(self.tableView)

        self.data_model = ItemModel()
        self.set_up_data_model()
        self.proxy_model = ProxyItemModel()
        self.proxy_model.setSourceModel(self.data_model)

        self.treeView.setModel(self.proxy_model)
        self.treeView.header().hide()
        self.tableView.setModel(self.proxy_model)

        self.treeView.selectionModel().currentChanged.connect(self.set_root_index)
        self.treeView.selectionModel().currentChanged.connect(self.hide_show_table)

        self.show()

    def set_root_index(self, index):
        self.tableView.setRootIndex(index)
        self.proxy_model.set_root_index(index)

    def hide_show_table(self, index):
        if self.proxy_model.is_order(index):
            self.tableView.show()
        else:
            self.tableView.hide()

    def set_up_data_model(self):
        root = self.data_model.root
        group1 = GroupNode(root, name='G1', description='Order Group 1')
        group2 = GroupNode(root, name='G2', description='Order Group 2')
        order1 = OrderNode(group1, order_id='123', order_date='17.10.2018')
        item1 = ItemNode(order1, name='Item 1', price='345')
        item2 = ItemNode(order1, name='Item 2', description='item number 2', price='99')
        order2 = OrderNode(group1, order_id='987')


if __name__ == '__main__':
    app = QtWidgets.QApplication(sys.argv)
    ui = MainWindow()

    sys.exit(app.exec_())

Теперь этот код приводит к тому, что горизонтальный заголовок не отображается вообще. Однако, когда я 0 <= section < len(header_list) в методе headerData с 0 < section < len(header_list), заголовок отображается правильно - кроме первого столбца, который только пуст. Что-то должно происходить с кодом, когда section = 0 но для жизни меня я не могу понять, что это такое. Я посмотрел с отладчиком и увидел, что для section = 0 и role и header_list имеют ожидаемые значения.

Любая помощь будет оценена. Я попытался удалить все, что не нужно, чтобы получить минимальный рабочий пример, но все же получился довольно длинным, хотя, сожалею об этом.

  • 0
    Каковы атрибуты ?, Вы можете предоставить минимальный воспроизводимый пример вашей модели.
  • 0
    @eyllanesc Спасибо за замечание. Я отредактировал вопрос, чтобы включить минимальный рабочий пример, который воспроизводит ошибку. Я запустил код с Python 3.6
Теги:
python-3.x
pyqt5
pyqt

1 ответ

0
Лучший ответ

Вы должны проверить раздел Qt::Horizontal и роль Qt::DisplayRole, в вашем случае вы не выполнили вторую проверку, чтобы она была перезаписана, генерируя эту ошибку.

def headerData(self, section, orientation, role=QtCore.Qt.DisplayRole):
    if self.sourceModel() and self.root_index.isValid():
        if orientation == QtCore.Qt.Horizontal and role == QtCore.Qt.DisplayRole:
            header_list = self.sourceModel().data(self.root_index, self.horizontal_header_role)
            if header_list and 0 <= section < len(header_list):
                return header_list[section]
    return super(ProxyItemModel, self).headerData(section, orientation, role)

Изображение 174551

Ещё вопросы

Сообщество Overcoder
Наверх
Меню