I have the following problem with QT model/view framework. I want to render a widget inside a table view item.
First my thought was to use
void QAbstractItemView::setIndexWidget( const QModelIndex & index, QWidget * widget )
But the documentation for this function explicitly states:
This function should only be used to display static content within the visible area corresponding to an item of data. If you want to display custom dynamic content or implement a custom editor widget, subclass QItemDelegate instead.
So they propose to use delegates here. Well, so far so good. I know that the delegates may be used to create an editor, which may be basically any QT widget. But here is the problem - I don't want this widget to be an editor - I want to render the item with this widget always. And not just "render", I need it to have the exact behavior of the widget.
Now the widget I want to use is a custom widget, which is a container of some other widgets (few check-boxes, few buttons with some layout).
The solution I consider is like this:
This solution seems to work, but it smells bad for me. Can anyone thing about more elegant solution for this problem?
Thanks.
Delegates are in charge of creating editors as well as doing all necessary display. They may use styles to do much of the painting, like drawing a progress bar, or paint manually.
A delegate, however, is not a widget. Unless an editor has been invoked, it doesn't have access to most things a widget would. The two are very different, have different purposes, and accomplish different things.
One of the most troublesome aspect of delegates is that they are static. Unless something in the model triggers an update (or the widget is configured to watch hover events), the delegate won't be used to redraw whatever data is present -- the buffered representation will be drawn to the screen.
You have some control over when an editor is invoked with edit triggers, although you can definitely handle it with some custom code, such as through mouse tracking.
You may call QAbstractItemView.openPersistentEditor(index)
for every cell you need to permanently have a complex widget for. The two keys to make this work:
QStyledItemDelegate.sizeHintChanged.emit(index)
needs to be called anytime the size of the editor widget has changed.QStyledItemDelegate.sizeHint()
can be tricky and tedious (or you can do index.internalPointer().editor_widget.sizeHint()
assuming you saved a reference of the editor to the internal pointer during QStyledItemDelegate.createEditor()
I recently used the openPersistentEditor
method to render tables within tree views, however it should be mentioned that opening editors is costly, so if you have thousands of indexes and they are all loaded at once, it can take a while. There are many ways to mitigate this issue:
fetchMore()
mechanismopenPersistentEditor
incrementally (using a timer, or as they come into view for the first time)openPersistentEditor
when the parent is expanded and closePersistentEditor
when the parent is collapsed, and possibly restrict the use of expand-all on nodes with many children.If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With