如何确定QTableWidget的正确大小?

时间:2012-01-07 01:41:32

标签: c++ qt

是否有任何方式来设置QTableWidget的“正确”大小? (我是新手)这个测试代码只有25行,在两个文件中,文件为Test.h:

#include <QtGui>
class Test : public QMainWindow {
   Q_OBJECT
public:
   Test();
};

和文件Test.cpp:

#include "Test.h"
Test::Test() : QMainWindow() {
   QVBoxLayout *vbox = new QVBoxLayout;
   QPushButton  *btn = new QPushButton("Hello World etc etc etc etc etc");
   QTableWidget *tbl = new QTableWidget(2, 2);
   vbox->addWidget(btn);
   vbox->addWidget(tbl);
   QWidget *w = new QWidget;
   setCentralWidget(w);
   w->setLayout(vbox);
   resize(1, 1);
}

int main(int argc, char *argv[]) {
   QApplication app(argc, argv);
   Test test;
   test.show();
   app.exec();
}

然后命令:

   qmake -project && qmake && make && ./Test

给出了窗口:

Unwanted window

但我们当然想要的更像是:

Wanted window

使用tbl->width()似乎毫无用处,因为它在640之前提供了test.show()的默认值,在195之后提供了不需要的值setResizeMode(QHeaderView::Fixed)。我看了Qt大小提示和政策,直到我的头旋转,我已经尝试setStretchLastSection(false)resize(1, 1);。也许我错过了一些明显的东西?这与CentOS 5上的Qt 4.7.4有关,如果这很重要的话。谢谢你的帮助。

编辑 :为响应DK,如果tbl->setMaximumWidth(222); tbl->setMinimumWidth(222); tbl->setMaximumHeight(88); tbl->setMinimumHeight(88); 不存在,则相等相反问题:窗口太大。

并回应Donotalo,补充道:

222

将给出所需的窗口大小(至少在我的机器上),但不是以所需的方式。我们应该如何计算'常数'88?tbl->setAttribute(Qt::WA_DontShowOnScreen); tbl->show();

Ton对Qt: How to force a hidden widget to calculate its layout?的回答似乎不起作用:添加tbl->width()使{{1}}的值保持不变,为640。

9 个答案:

答案 0 :(得分:17)

Lingfa Yang和Susan Macchia之间的线程How to set a precise size of QTableWidget to prevent from having scroll bars? (Qt-interest Archive, June 2007)似乎解决了我的问题。如果我的测试有效,我会很快发布更多细节。

更新#1 :我的测试现在生成了漂亮的窗口:

successful test window

Test.h完整的测试代码是:

#include "Test.h"

static QSize myGetQTableWidgetSize(QTableWidget *t) {
   int w = t->verticalHeader()->width() + 4; // +4 seems to be needed
   for (int i = 0; i < t->columnCount(); i++)
      w += t->columnWidth(i); // seems to include gridline (on my machine)
   int h = t->horizontalHeader()->height() + 4;
   for (int i = 0; i < t->rowCount(); i++)
      h += t->rowHeight(i);
   return QSize(w, h);
}

static void myRedoGeometry(QWidget *w) {
   const bool vis = w->isVisible();
   const QPoint pos = w->pos();
   w->hide();
   w->show();
   w->setVisible(vis);
   if (vis && !pos.isNull())
      w->move(pos);
}

Test::Test() : QMainWindow() {
   QVBoxLayout *vbox = new QVBoxLayout;
   QPushButton *btn  = new QPushButton("Hello World etc etc etc etc etc");
   QTableWidget *tbl = new QTableWidget(2, 2);
   vbox->addWidget(btn);
   vbox->addWidget(tbl);
   setCentralWidget(new QWidget);
   centralWidget()->setLayout(vbox);
   layout()->setSizeConstraint(QLayout::SetMinimumSize); // or SetFixedSize

   tbl->setVerticalHeaderItem(1, new QTableWidgetItem("two")); // change size
   myRedoGeometry(this);
   tbl->setMaximumSize(myGetQTableWidgetSize(tbl));
   tbl->setMinimumSize(tbl->maximumSize()); // optional
}

int main(int argc, char *argv[]) {
   QApplication app(argc, argv);
   Test test;
   test.show();
   app.exec();
}

一些注意事项:

  • 以上帖子包含verticalScrollBar()->width()似乎是错误的。而且,在我的测试中,如果显示滚动条,则总是默认值为100,或者值为15。

  • 在此测试中,仅将show(); hide();序列应用于QTableWidget不足以强制Qt重新计算几何体。我需要将它应用到整个窗口。

  • 任何改进建议都会很好。 (如果有更好的解决方案,我会在接受自己的答案之前等一下。)

更新#2

  • Qt-interest线程可能有误(或者,至少,它不同意运行我的机器的Qt的我的版本)有关如何使用的详细信息计算大小:每个网格线的+1不是必需的,但总需要+4。

  • 我仍在处理layout()->invalidate()与... QT: How to preview sizes of widgets in layout BEFORE a show()

更新#3

  • 这是我的最终版本 - 但我对此并不满意。任何改进都将非常欢迎。

  • adjustSize()layout()->invalidate()以及Qt::WA_DontShowOnScreen等内容似乎无济于事。

  • 博客Shrinking Qt widgets to minimum needed size很有趣,但使用setSizeConstraint()同样出色。

  • 后续更改表格时需要setSizeConstraint()move()方法,此处未显示。

  • 我的测试中有一件奇怪的事情,在CentOS 5上用Qt 4.6.3和4.7.4完成。对于hide(); show();序列,保存/恢复窗口位置。但是大约25%的时间(图案是不规则的),恢复的位置在屏幕上 24像素,可能是窗口标题的高度。大约10%的时间pos()返回的值将 null Qt site对X11说它需要nifty heuristics and clever code,但某些地方似乎有些被打破。

答案 1 :(得分:8)

这是我(pythonic)解决这个问题的方法:

table.setSizePolicy(QtGui.QSizePolicy.Minimum, QtGui.QSizePolicy.Minimum)

table.setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff)
table.setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff)

table.resizeColumnsToContents()
table.setFixedSize(table.horizontalHeader()->length()+table.verticalHeader()->width(), table.verticalHeader()->length()+table.horizontalHeader()->height())

这给了我一张大小完美包装所有行和列的表格。

答案 2 :(得分:4)

最近,我发现了同样的问题。我在Qt Center,Qt Project中测试了Google发现的几十个函数,但没有一个函数给我正确的大小。我做了我认为计算最接近的桌子大小的功能。该功能提供了一个适当的垂直但更大的水平尺寸 - 它看起来仍然覆盖垂直滚动条。另外,我检查了表格尺寸是否超过桌面尺寸:

const QRect MyApp::adjustTableSize(
    QTableView* tv,
    const QVBoxLayout* lay,
    const int& maxRows,
    const int& maxCols) {
// get Desktop size
using std::size_t;
QDesktopWidget desktop;
size_t desW = desktop.screen()->width();
size_t desH = desktop.screen()->height();

int leftM,rightM,topM,bottomM;
lay->getContentsMargins(&leftM,&topM,&rightM,&bottomM);

size_t extraTopHeight = topM + tv->frameWidth();
size_t extraBottomHeight = bottomM + tv->frameWidth();
size_t extraLeftWidth = leftM + tv->frameWidth();
size_t extraRightWidth = rightM + tv->frameWidth();
size_t w = tv->verticalHeader()->width() + extraLeftWidth + extraRightWidth;
size_t h = tv->horizontalHeader()->height() + extraTopHeight + extraBottomHeight;
for(size_t col = 0; col < maxCols; ++col) {
    w += tv->columnWidth(col);
}
for(size_t row = 0; row < maxRows; ++row ) {
    h += tv->rowHeight(row);
}

std::size_t x,y;
if((w - extraLeftWidth - extraRightWidth) > desW) {
    x = 0;
    w = desW - extraLeftWidth - extraRightWidth;
} else
    x = (desW - w)/2;
if(h - extraTopHeight - extraBottomHeight - QStyle::PM_TitleBarHeight > desH) {
    y = extraTopHeight + QStyle::PM_TitleBarHeight;
    h = desH - (extraTopHeight + QStyle::PM_TitleBarHeight + extraBottomHeight);
} else
    y = (desH - h)/2;
return QRect(x,y,w,h);
}

一些附注:

QTableView
我使用了QTableView,但我认为它也适用于QTableWidget。

QVBoxLayout
我很肯定Qt中可用的任何布局都可以用在这个函数中。

希望,这个功能可以帮助任何人。

答案 3 :(得分:3)

我正在处理同样的问题,我调整了之前的答案以获得正确的工作代码。

首先,我们从QTableWidget获取内容边距。如果您只想调整其宽度,则只需要禁用水平滚动条。:

int w = 0, h = 0;
ui->tableWidget->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
ui->tableWidget->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);

w += ui->tableWidget->contentsMargins().left()
        + ui->tableWidget->contentsMargins().right();
h += ui->tableWidget->contentsMargins().top()
        + ui->tableWidget->contentsMargins().bottom();

然后,我们将垂直标题的宽度附加到w。垂直标头是包含行索引的特殊列,默认情况下处于启用状态。我们还将水平标题的高度追加到h。水平标题是包含列标题的特殊行。

w += ui->tableWidget->verticalHeader()->width();
h += ui->tableWidget->horizontalHeader()->height();

然后,我们将每列的宽度追加到w,将每行的高度追加到h

for (int i=0; i<ui->tableWidget->columnCount(); ++i)
    w += ui->tableWidget->columnWidth(i);
for (int i=0; i<ui->tableWidget->rowCount(); ++i)
    h += ui->tableWidget->rowHeight(i);

现在,如果wh太大而无法放在窗口上,我们会将其缩小为ui->centralWidget->contentsRect()

if (w > ui->centralWidget->contentsRect().width())
    w = ui->centralWidget->contentsRect().width();
if (h > ui->centralWidget->contentsRect().height())
    h = ui->centralWidget->contentsRect().height();

不幸的是,我不知道如何更好地应用wh来设置QTableWidget的宽度和高度。如果我使用:

ui->tableWidget->setMinimumWidth(w);
ui->tableWidget->setMaximumWidth(w);
ui->tableWidget->setMinimumHeight(h);
ui->tableWidget->setMaximumHeight(h);

然后用户根本无法调整窗口小部件的大小。我认为我们需要使用某种手动布局。

这可能看起来有点偏离主题,但是如果您希望您的表具有更吸引人的视图,请使用此命令调整列的大小以适合其数据:

ui->tableWidget->resizeColumnsToContents();

答案 4 :(得分:2)

要获得表格的宽度,请选择垂直和水平标题的宽度以及帧宽度的两倍。高度也是如此:

w = verticalHeader()->width() + horizontalHeader()->length() + frameWidth()*2
h = horizontalHeader()->height() + verticalHeader()->length() + frameWidth()*2

根据您的具体情况,有多种方法可以设置它们。首选可能是覆盖sizeHint()以返回您的值。这样,桌子可以调整大小,但没有其他要求就是你的尺寸。

实现这一目标的关键是在正确的位置使用length()方法,而不仅仅是width()height()。我不确定为什么,但在这里做的是正确的; - )

答案 5 :(得分:1)

我不能评论Joseph Qunesey的/ Celdor的帖子,所以我在这里写: 要检索QStyle::PM_TitleBarHeight的真值,您应该按如下方式执行:

QStyle *style;
if(tv->style())
{
    style = tv->style();
}
else
{
    style = QApplication::style();
}
int titleBarHeight = style->pixelMetric(QStyle::PM_TitleBarHeight));

这是因为枚举没有分配确切的值。我认为,你对它是如何工作的误解是由于误导Qt documentation,看起来enum直接代表了值,而它当然没有(因为它取决于样式)。

答案 6 :(得分:1)

显然有一些 - 可能是错误的? - 在horizo​​ntalHeader() - &gt; width()函数中; 这个解决方案对我有用:

    //given a QtableWidget* called x
    int previous_width= x->horizontalHeader()->width();//get the width
    //the width is relating to all the columns including the vertical headers 
    int previous_length=x->horizontalHeader()->length();//and the length
    // the length is relating to all the columns excluding the size of the vertical headers
    int vertical_headers_width=(previous_width-previous_length);
    x->resizeColumnsToContents();//now we resize
    x->resizeRowsToContents();
    //if we call again x->horizontalHeader()->width();
    // we get a number that looks to me not correct (maybe is my ignorance here)
    x->setFixedWidth(x->horizontalHeader()->length()+vertical_headers_width);

   //by adding to the length of the data columns the old width_of_verticla_headers_column we get the correct size;

答案 7 :(得分:1)

这是我的解决方案(我隐藏了垂直和水平标题):

m_table->resizeColumnsToContents();
m_table->resizeRowsToContents();

int w = 0; // verticalHeader()->width()
int h = 0; // horizontalHeader()->height()
for ( int i = 0; i < m_table->rowCount(); i++ )
{
    h += m_table->rowHeight( i );
}
for ( int j = 0; j < m_table->columnCount(); j++ )
{
    w += m_table->columnWidth( j );
}
m_table->setFixedSize( w, h );

答案 8 :(得分:1)

这不是标题中所述问题的答案,但我认为这是对潜在问题的巧妙答案。

为了在没有滚动条的情况下以完整尺寸显示 QTableWidget,您必须将其 sizeAdjustPolicy 设置为 QAbstractScrollArea::AdjustToContents(我认为 QAbstractScrollArea::AdjustToContentsOnFirstShow 也可以) .

此外,您可能希望禁用该小部件的滚动条(设置其垂直和水平滚动条的滚动条策略),否则会在显示滚动条时为小部件周围分配额外的空间。

参考: