我有一个QListView,我使用自定义绘图显示项目。在每个项目(即每个列表行)中,我希望能够显示一些用户可以点击的“超链接”,然后调用某些功能。
我已经尝试查看官方文档(例如Model/View Programming)以及相当多的Google信息,但无法弄清楚如何完成此操作。
我有两个想法,每个都有自己的问题:
我最初的方法是使用QListWidget和.setItemWidget(),其中我有一个带有布局和子窗口小部件的适当窗口小部件。不幸的是,当我的列表增加到数百或数千个项目时,这太慢了。这就是我用委托改为QListView的原因。
答案 0 :(得分:2)
我似乎正在接近一个解决方案。
我可以通过覆盖委托的.editorEvent(event, model, option, index)
来获得对元素的点击。然后,我可以找到event.type()
,index.row()
中点击的行以及event.x()
和event.y()
的实际坐标(因为,如果事件类型是MouseButtonRelease,则事件为一个QMouseEvent)。
从这些方面来说,我认为我可以将坐标与屏幕上的元素相关联,并采取相应的行动。
一旦我有了工作代码,我就会更新这个答案。
修改强>
一个简单的工作示例,使用PySide:
class MyModel(QtGui.QStandardItemModel):
def __init__(self):
super(MyModel, self).__init__()
for i in range(10): self.appendRow(QtGui.QStandardItem("Row %d" % i))
class MyDelegate(QtGui.QStyledItemDelegate):
def __init__(self, parent=None):
super(MyDelegate, self).__init__(parent)
self.links = {}
def makeLinkFunc(self, row, text):
def linkFunc(): print("Clicked on %s in row %d" % (text, row))
return linkFunc
def paint(self, painter, option, index):
painter.save()
textHeight = QtGui.QFontMetrics(painter.font()).height()
painter.drawText(option.rect.x()+2, option.rect.y()+2+textHeight, index.data())
rowLinks = {}
for i in range(3):
text = "Link %d" % (3-i)
linkWidth = QtGui.QFontMetrics(font).width(text)
x = option.rect.right() - (i+1) * (linkWidth + 10)
painter.drawText(x, y, text)
rect = QtCore.QRect(x, y - textHeight, linkWidth, textHeight)
rowLinks[rect] = self.makeLinkFunc(index.row(), text)
self.links[index.row()] = rowLinks
painter.restore()
def sizeHint(self, option, index):
hint = super().sizeHint(option, index)
hint.setHeight(30)
return hint
def editorEvent(self, event, model, option, index):
if event.type() == QtCore.QEvent.MouseButtonRelease:
for rect, link in self.links[index.row()].items():
if rect.contains(event.pos()):
link()
return True
return False
listmodel = MyModel()
listview = QtGui.QListView()
listview.setModel(listmodel)
listview.setItemDelegate(MyDelegate(parent=listview))
listview.setSelectionMode(QtGui.QAbstractItemView.NoSelection)