QScrollArea:自动滚动到新添加的小部件

时间:2012-05-08 19:02:57

标签: qt scrollview qwidget

这不是我第一次想要一个行为如下的滚动区域(想象一下日志或聊天窗口,但是使用简单的QTextBrowser太复杂了):

  • 我想添加多个小部件,这些小部件一个在另一个之下(如放在QVBoxLayout中时)
  • 此布局中的每个小部件应具有固定大小或高度宽度(如简单标签)
  • 滚动区域应自动滚动到最近添加的一个
  • (可选)当剩余空间(尚未启用滚动条)时,内容应与底部对齐

使用QScrollArea:

我过去的尝试是使用QScrollArea在里面使用QVBoxLayout。但这似乎并不像我想象的那么简单:每当我向布局添加一个小部件时,布局都不会立即调整滚动区域内容小部件的大小,从而导致内容的延迟调整。在短时间内,布局中包含的小部件将调整大小,以使总大小等于添加操作之前的总大小,从而导致每个小部件的大小过小。此外,滚动到新添加的窗口小部件是不可能的,直到布局将其大小更正为窗口小部件的新总大小,因此即使是QTimer :: singleShot(0,...)也没有帮助。即使超时为20左右,在某些情况下布局也需要更多时间来调整大小。它不是确定性的,因此远离一个很好的解决方案。

为了获得底部对齐行为,我最初在布局中放置了一个spacer项。一旦没有剩余空间并且启用了滚动,它就不需要任何空间。

使用QListView:

由于我的项目过于复杂,因此需要使用QWidgets。他们无法拥有焦点,也无法选择,因此基于项目的解决方案似乎只是“错误的方式”。此外,这个解决方案听起来太“沉重”。

我简直无法相信没有简单的方法,所以我想我还没有看到它!

1 个答案:

答案 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);
}