setHorizo​​ntalHeader()后QTableView排序失败

时间:2017-06-10 13:57:23

标签: python qt model-view-controller pyqt pyside

在通过QTableView设置新的QHeaderView实例后,我遇到了PySide setHorizontalHeader()的奇怪问题。具体来说,似乎视图的排序能力被破坏/不能再被调用。

问题

  • 为什么在设置新的标题视图后分类会中断?
  • 为什么在通过setHorizontalHeader()设置新标头后,标题视图实例上的monkeypatching方法 (请参阅演示代码中的wrap_size_hint()。)
  • 在记录表视图的子项时,为什么总是2 QHeaderView个实例?
  • initialize()实际上有哪些QHeaderView方法?它们在这方面有用吗? The docs aren't quite that helpfull

Demo code (gist)

# -*- coding: utf-8 -*-
from __future__ import unicode_literals, print_function, division

import logging
import sys

from PySide import QtCore
from PySide import QtGui


# configure logging
logging.basicConfig(level=logging.DEBUG)
logger = logging.getLogger(__name__)


def wrap_size_hint(func):
    """Monkeypatch virtual method to increase sizeHint.

    Notes:
        Strangely this does not work when when no custom header is set on the
        table view.
    """

    def wrapped(*args, **kwargs):
        """Just extend sizeHint to increase height."""
        size = func(*args, **kwargs)  # QSize
        height_increase = 60
        height = size.height() + height_increase
        size.setHeight(height)
        return size

    return wrapped


class CustomHeaderView(QtGui.QHeaderView):
    """CustomHeaderView"""


class TestWidget(QtGui.QWidget):

    def __init__(self, use_custom_headerview=True, *args, **kwargs):
        super(TestWidget, self).__init__(*args, **kwargs)

        self.logger = logging.getLogger(type(self).__name__)

        QtGui.QVBoxLayout(self)

        self.setup_mvc(use_custom_headerview)
        self.update_model()
        self.log_children(self.view)

    def setup_mvc(self, use_custom_headerview):
        layout = self.layout()

        # model
        header_labels = "One Two Three Four Five".split()
        self.model = QtGui.QStandardItemModel()
        self.model.setHorizontalHeaderLabels(header_labels)

        # proxymodel
        self.proxymodel = QtGui.QSortFilterProxyModel()
        self.proxymodel.setDynamicSortFilter(True)
        self.proxymodel.setSourceModel(self.model)

        # view
        self.view = QtGui.QTableView()
        if (use_custom_headerview is True):
            self.view.setHorizontalHeader(CustomHeaderView(QtCore.Qt.Horizontal))

        self.view.setSortingEnabled(True)
        self.view.setModel(self.proxymodel)

        # monkeypatch sizeHint to increase size
        headerview = self.view.horizontalHeader()
        headerview.sizeHint = wrap_size_hint(headerview.sizeHint)

        layout.addWidget(self.view)

    def update_model(self):

        for row in range(50):
            items = []

            for column in range(self.model.columnCount()):
                text = "{0} {1}".format(row, column)
                item = QtGui.QStandardItem()
                item.setEditable(True)
                item.setData(text, QtCore.Qt.DisplayRole)
                items.append(item)

            self.model.appendRow(items)

    def log_children(self, widget, tab=-1):
        tab += 1
        msg = "{0}{0}: {1}".format("--"*tab, type(widget).__name__, widget)
        self.logger.info(msg)
        for child in widget.children():
            self.log_children(child, tab)


if (__name__ == "__main__"):

    app = QtGui.QApplication(sys.argv)

    # widget_sorting_fails
    widget_sorting_fails = TestWidget(use_custom_headerview=True)
    widget_sorting_fails.setWindowTitle("Custom headerview")
    widget_sorting_fails.show()

    logger.info("-"*100)

    # widget_sorting_succeeds
    widget_sorting_succeeds = TestWidget(use_custom_headerview=False)
    widget_sorting_succeeds.setWindowTitle("No custom headerview")
    widget_sorting_succeeds.show()

    # start event loop
    sys.exit(app.exec_())

1 个答案:

答案 0 :(得分:0)

好的,发现setClickable(True)可以解决问题。默认情况下,QTableView标题视图似乎处于启用状态,而不是手动创建的实例。