这不是我第一次想要一个行为如下的滚动区域(想象一下日志或聊天窗口,但是使用简单的QTextBrowser太复杂了):
使用QScrollArea:
我过去的尝试是使用QScrollArea在里面使用QVBoxLayout。但这似乎并不像我想象的那么简单:每当我向布局添加一个小部件时,布局都不会立即调整滚动区域内容小部件的大小,从而导致内容的延迟调整。在短时间内,布局中包含的小部件将调整大小,以使总大小等于添加操作之前的总大小,从而导致每个小部件的大小过小。此外,滚动到新添加的窗口小部件是不可能的,直到布局将其大小更正为窗口小部件的新总大小,因此即使是QTimer :: singleShot(0,...)也没有帮助。即使超时为20左右,在某些情况下布局也需要更多时间来调整大小。它不是确定性的,因此远离一个很好的解决方案。
为了获得底部对齐行为,我最初在布局中放置了一个spacer项。一旦没有剩余空间并且启用了滚动,它就不需要任何空间。
使用QListView:
由于我的项目过于复杂,因此需要使用QWidgets。他们无法拥有焦点,也无法选择,因此基于项目的解决方案似乎只是“错误的方式”。此外,这个解决方案听起来太“沉重”。
我简直无法相信没有简单的方法,所以我想我还没有看到它!
答案 0 :(得分:1)
QListView
应该没问题。您声称您的商品是静态的,没有与它们互动:没有焦点,没有选择。似乎QWidget
对这些物品来说太过分了。您只需要具有固定大小的东西并且可以绘制自己。这正是Qt模型 - 视图系统中的代表所要求的。只需为您的项目实现一个或多个QAbstractItemDelegate
,并为他们将要呈现的数据提供模型的实现。 QAbstractItemView
已经是QAbstractScrollArea
!
如果你想在委托中绘制HTML,那很容易做到 - 再次,QWidget对于静态显示来说是一种过度杀伤!有一个非常食物的原因,为什么使用QWidget很难“ - API引导您找到正确的解决方案。假设你的模型包含每个项目的html,这里是你如何绘制它。当然,您可以使用sizeHint,并且应该缓存文本文档,理想情况下将其存储在我认为的模型中。
void MyDelegate::paint(QPainter* p, const QStyleOptionViewItem & opt, const QModelIndex & index) const
{
QTextDocument doc;
doc.setHtml(index.data().toString());
doc.drawContents(p, QRect(QPoint(0,0), sizeHint(opt, index)));
}
QSize MyDelegate::sizeHint(const QStyleOptionViewItem &, const QModelIndex &) const
{
return QSize(100, 200);
}