Switch from DocketEntry widget to ItemDelegate.
We were using a subclassed QTextEdit widget so that we could capture clicks on anchors. Moving to a delegate removed the widget per row issue. Unfortunately, there was no more click on anchor. Adding a mouseEvent() to the TableView allowed us to translate raw mouse presses to anchor triggers. Fixes: #18
This commit is contained in:
@@ -1,22 +1,22 @@
|
|||||||
from typing import cast
|
from typing import cast
|
||||||
|
|
||||||
from PySide6.QtCore import (
|
from PySide6.QtCore import (
|
||||||
QAbstractItemModel,
|
|
||||||
QDir,
|
QDir,
|
||||||
QFile,
|
QFile,
|
||||||
QModelIndex,
|
QModelIndex,
|
||||||
QObject,
|
|
||||||
QPersistentModelIndex,
|
QPersistentModelIndex,
|
||||||
QPoint,
|
QPoint,
|
||||||
|
QPointF,
|
||||||
|
QRect,
|
||||||
QSize,
|
QSize,
|
||||||
Qt,
|
Qt,
|
||||||
QUrl,
|
QUrl,
|
||||||
Signal,
|
Signal,
|
||||||
Slot,
|
Slot,
|
||||||
)
|
)
|
||||||
from PySide6.QtGui import (
|
from PySide6.QtGui import (
|
||||||
QMouseEvent,
|
QMouseEvent,
|
||||||
QPalette,
|
QPainter,
|
||||||
QTextDocument,
|
QTextDocument,
|
||||||
)
|
)
|
||||||
from PySide6.QtNetwork import (
|
from PySide6.QtNetwork import (
|
||||||
@@ -26,7 +26,7 @@ from PySide6.QtNetwork import (
|
|||||||
)
|
)
|
||||||
from PySide6.QtWidgets import (
|
from PySide6.QtWidgets import (
|
||||||
QAbstractItemView,
|
QAbstractItemView,
|
||||||
QSizePolicy,
|
QStyle,
|
||||||
QStyledItemDelegate,
|
QStyledItemDelegate,
|
||||||
QStyleOptionViewItem,
|
QStyleOptionViewItem,
|
||||||
QTableView,
|
QTableView,
|
||||||
@@ -34,7 +34,7 @@ from PySide6.QtWidgets import (
|
|||||||
QWidget,
|
QWidget,
|
||||||
)
|
)
|
||||||
|
|
||||||
from docketModel import docketModel
|
from lib.utils import QStyleOptionViewItemInit
|
||||||
from pdfView import PDFViewer
|
from pdfView import PDFViewer
|
||||||
|
|
||||||
|
|
||||||
@@ -48,42 +48,33 @@ class docketEntryDelegate(QStyledItemDelegate):
|
|||||||
|
|
||||||
def sizeHint(
|
def sizeHint(
|
||||||
self,
|
self,
|
||||||
_: QStyleOptionViewItem,
|
option: QStyleOptionViewItem,
|
||||||
index: QModelIndex | QPersistentModelIndex,
|
index: QModelIndex | QPersistentModelIndex,
|
||||||
) -> QSize:
|
) -> QSize:
|
||||||
widget = self.view.indexWidget(index)
|
options = cast(QStyleOptionViewItemInit, option)
|
||||||
return widget.sizeHint()
|
self.initStyleOption(options, index)
|
||||||
|
|
||||||
|
|
||||||
class docketEntry(QTextEdit):
|
|
||||||
def __init__(self, parent: QWidget | None = None) -> None:
|
|
||||||
super(docketEntry, self).__init__(parent)
|
|
||||||
self.setSizePolicy(
|
|
||||||
QSizePolicy.Policy.Preferred, QSizePolicy.Policy.Fixed
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
def mousePressEvent(self, e: QMouseEvent) -> None:
|
|
||||||
super().mousePressEvent(e)
|
|
||||||
anchor = self.anchorAt(e.pos())
|
|
||||||
if anchor:
|
|
||||||
obj = cast(QObject, self)
|
|
||||||
while not isinstance(obj, docketTableView) and obj is not None:
|
|
||||||
obj = obj.parent()
|
|
||||||
assert obj is not None
|
|
||||||
index = obj.indexAt(obj.mapFromGlobal(self.mapToGlobal(e.pos())))
|
|
||||||
obj.anchorSignal.emit(index, anchor)
|
|
||||||
return
|
|
||||||
|
|
||||||
def sizeHint(self) -> QSize:
|
|
||||||
size = self.size()
|
|
||||||
doc = QTextDocument()
|
doc = QTextDocument()
|
||||||
doc.setPlainText(self.document().toPlainText())
|
doc.setHtml(options.text)
|
||||||
doc.setTextWidth(size.width())
|
doc.setTextWidth(options.rect.width())
|
||||||
docSize = doc.size()
|
return QSize(int(doc.idealWidth()), int(doc.size().height()))
|
||||||
return QSize(size.width(), int(docSize.height()))
|
|
||||||
|
|
||||||
|
def paint(self, painter: QPainter, option: QStyleOptionViewItem, index: QModelIndex | QPersistentModelIndex) -> None:
|
||||||
|
options = cast(QStyleOptionViewItemInit, option)
|
||||||
|
self.initStyleOption(options, index)
|
||||||
|
painter.save()
|
||||||
|
|
||||||
|
doc = QTextDocument()
|
||||||
|
doc.setHtml(options.text)
|
||||||
|
doc.setTextWidth(options.rect.width())
|
||||||
|
|
||||||
|
options.text = ''
|
||||||
|
options.widget.style().drawControl(QStyle.ControlElement.CE_ItemViewItem, options, painter)
|
||||||
|
painter.translate(options.rect.left(), options.rect.top())
|
||||||
|
clip = QRect(0,0, options.rect.width(), options.rect.height())
|
||||||
|
doc.drawContents(painter, clip)
|
||||||
|
painter.restore()
|
||||||
|
return
|
||||||
|
|
||||||
class docketTableView(QTableView):
|
class docketTableView(QTableView):
|
||||||
|
|
||||||
manager: QNetworkAccessManager
|
manager: QNetworkAccessManager
|
||||||
@@ -99,12 +90,6 @@ class docketTableView(QTableView):
|
|||||||
self.manager.finished.connect(self.getDone)
|
self.manager.finished.connect(self.getDone)
|
||||||
return
|
return
|
||||||
|
|
||||||
def setModel(self, model: QAbstractItemModel | None) -> None:
|
|
||||||
assert model is not None
|
|
||||||
super().setModel(model)
|
|
||||||
self.model().modelReset.connect(self.modelReset)
|
|
||||||
return
|
|
||||||
|
|
||||||
@Slot(QNetworkReply) # type: ignore
|
@Slot(QNetworkReply) # type: ignore
|
||||||
def getDone(self, reply: QNetworkReply) -> None:
|
def getDone(self, reply: QNetworkReply) -> None:
|
||||||
dest = QFile("." + reply.url().path())
|
dest = QFile("." + reply.url().path())
|
||||||
@@ -136,16 +121,33 @@ class docketTableView(QTableView):
|
|||||||
self.manager.get(QNetworkRequest(url))
|
self.manager.get(QNetworkRequest(url))
|
||||||
return
|
return
|
||||||
|
|
||||||
def modelReset(self) -> None:
|
def mousePressEvent(self, event: QMouseEvent) -> None:
|
||||||
model = self.model()
|
#
|
||||||
red = QPalette()
|
# The mouse has been pressed somewere in our rect. We need to translate that to a click
|
||||||
red.setColor(QPalette.ColorRole.Base, Qt.GlobalColor.red)
|
# within the cell with the document. This will allow the document to find anchors.
|
||||||
for row in range(0, model.rowCount()):
|
#
|
||||||
index = model.index(row, docketModel.ColumnNames.text)
|
index = self.indexAt(event.pos())
|
||||||
widget = docketEntry()
|
if not index.isValid():
|
||||||
widget.setHtml(model.data(index, Qt.ItemDataRole.DisplayRole))
|
return
|
||||||
widget.setAutoFillBackground(False)
|
if index.column() != 1:
|
||||||
widget.setReadOnly(True)
|
return
|
||||||
widget.setPalette(red)
|
doc = QTextDocument()
|
||||||
self.setIndexWidget(index, widget)
|
doc.setHtml(index.data(Qt.ItemDataRole.DisplayRole))
|
||||||
|
doc.setTextWidth(self.columnWidth(index.column()))
|
||||||
|
#
|
||||||
|
# We need to map our click position to a position within the cell.
|
||||||
|
#
|
||||||
|
pos = event.position()
|
||||||
|
new_pos = QPointF(pos.x() - self.horizontalScrollBar().value(),
|
||||||
|
pos.y() - self.verticalScrollBar().value())
|
||||||
|
rowHeight = 0
|
||||||
|
for row in range(0, index.row()):
|
||||||
|
rowHeight += self.rowHeight(row)
|
||||||
|
cell_pos = QPointF(new_pos.x() - self.columnWidth(0),
|
||||||
|
new_pos.y() - rowHeight)
|
||||||
|
te = QTextEdit()
|
||||||
|
te.setDocument(doc)
|
||||||
|
anchor = te.anchorAt(cell_pos.toPoint())
|
||||||
|
if anchor:
|
||||||
|
self.anchorSignal.emit(index,anchor)
|
||||||
return
|
return
|
||||||
|
|||||||
Reference in New Issue
Block a user