У меня есть QStandardItemModel, подключенный к QTableView. Один из столбцов в моей модели содержит даты, которые имеют удобный для пользователя displayData и удобный для компьютера userData. Так, например, один QStandardItem может отображать строку типа 22 Nov 2018
, но данные пользователя будут выглядеть как 324586
(секунды с начала эпохи). Однако, когда я сортирую столбец, он, конечно же, сортируется по displayData. Как я могу вместо этого принудительно отсортировать таблицу по userData?
QTableView сортирует данные столбца по UserRole
Ответы (1)
Вам необходимо использовать setSortRole ():
from PyQt5 import QtCore, QtGui, QtWidgets
import random
DATECOLUMN = 1
class MainWindow(QtWidgets.QMainWindow):
def __init__(self, parent=None):
super(MainWindow, self).__init__(parent)
self._tableView = QtWidgets.QTableView()
self.setCentralWidget(self._tableView)
self._model = QtGui.QStandardItemModel(10, 4)
self._tableView.setModel(self._model)
now_second = QtCore.QDateTime.currentDateTime().toSecsSinceEpoch()
for i in range(self._model.rowCount()):
for j in range(self._model.columnCount()):
if j == DATECOLUMN:
t = QtCore.QDateTime.fromSecsSinceEpoch(random.randint(0, now_second))
text = t.toString("dd MMM yyyy")
it = QtGui.QStandardItem(text)
it.setData(t.toSecsSinceEpoch(), QtCore.Qt.UserRole)
else:
it = QtGui.QStandardItem("{}-{}".format(i, j))
self._model.setItem(i, j, it)
self._model.setSortRole(QtCore.Qt.UserRole)
self._model.sort(DATECOLUMN, QtCore.Qt.AscendingOrder)
if __name__ == '__main__':
import sys
app = QtWidgets.QApplication(sys.argv)
w = MainWindow()
w.show()
sys.exit(app.exec_())
Хотя я предпочитаю сохранять QDateTime напрямую и использовать делегата для отображения данных в желаемом формате.
from PyQt5 import QtCore, QtGui, QtWidgets
import random
DATECOLUMN = 1
class DateDelegate(QtWidgets.QStyledItemDelegate):
def displayText(self, value, locale):
return locale.toString(value, "dd MMM yyyy")
class MainWindow(QtWidgets.QMainWindow):
def __init__(self, parent=None):
super(MainWindow, self).__init__(parent)
self._tableView = QtWidgets.QTableView()
self.setCentralWidget(self._tableView)
self._model = QtGui.QStandardItemModel(10, 4)
self._tableView.setModel(self._model)
delegate = DateDelegate(self._tableView)
self._tableView.setItemDelegateForColumn(DATECOLUMN, delegate)
now_second = QtCore.QDateTime.currentDateTime().toSecsSinceEpoch()
for i in range(self._model.rowCount()):
for j in range(self._model.columnCount()):
if j == DATECOLUMN:
t = QtCore.QDateTime.fromSecsSinceEpoch(random.randint(0, now_second))
it = QtGui.QStandardItem()
it.setData(t, QtCore.Qt.DisplayRole)
else:
it = QtGui.QStandardItem("{}-{}".format(i, j))
self._model.setItem(i, j, it)
self._model.sort(DATECOLUMN, QtCore.Qt.AscendingOrder)
if __name__ == '__main__':
import sys
app = QtWidgets.QApplication(sys.argv)
w = MainWindow()
w.show()
sys.exit(app.exec_())
person
eyllanesc
schedule
27.11.2018
А, значит, мне нужно установить его на модель, а не на вид! Все еще привыкаю к тому, где все находится в рамках модели / представления ... Спасибо за очень полный ответ!
- person Spencer; 27.11.2018
@Spencer Я рекомендую вам прочитать: doc.qt.io/qt-5 /model-view-programming.html, чтобы вы кое-что поняли о MVC в Qt. С другой стороны, я предлагаю, чтобы в будущем, когда вы укажете, что что-то не так, вы предоставили минимальный воспроизводимый пример
- person eyllanesc; 27.11.2018