如何确定QTableWidget的正确大小?

Jos*_*sey 30 c++ qt

没有办法设置QTableWidget的"正确"大小?(我是新手)这个测试代码只有25行,在两个文件中,文件Test.h:

#include <QtGui>
class Test : public QMainWindow {
   Q_OBJECT
public:
   Test();
};
Run Code Online (Sandbox Code Playgroud)

和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();
}
Run Code Online (Sandbox Code Playgroud)

然后命令:

   qmake -project && qmake && make && ./Test
Run Code Online (Sandbox Code Playgroud)

给出窗口:

不需要的窗口

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

通缉窗口

使用tbl->width()似乎没用,因为它给出了640之前的默认test.show()值,以及195after 的不需要的值.我已经看过了Qt的大小提示和政策,直到我的头纺,我已经试过setResizeMode(QHeaderView::Fixed)setStretchLastSection(false).也许我错过了一些明显的东西?这与CentOS 5上的Qt 4.7.4有关,如果这很重要的话.感谢您的任何帮助.

编辑:作为对DK的回应,如果该行resize(1, 1);不存在,则存在相同相反的问题:窗口太大.

并回应Donotalo,补充说:

tbl->setMaximumWidth(222);
tbl->setMinimumWidth(222);
tbl->setMaximumHeight(88);
tbl->setMinimumHeight(88); 
Run Code Online (Sandbox Code Playgroud)

将给出所需的窗口大小(至少在我的机器上),但不是以所需的方式.我们应该如何计算'常数' 22288?

而Ton对Qt的回答:如何强制隐藏的小部件来计算其布局?似乎在这里不起作用:在640 tbl->setAttribute(Qt::WA_DontShowOnScreen); tbl->show();处添加tbl->width()保持不变的值.

Jos*_*sey 18

线程如何设置QTableWidget的精确大小以防止滚动条?Lingfa Yang和Susan Macchia之间的(Qt-interest Archive,2007年6月)似乎解决了我的问题.如果我的测试有效,我会在短期内发布更多细节.

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

成功的测试窗口

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();
}
Run Code Online (Sandbox Code Playgroud)

一些说明:

  • 以上线程的包含verticalScrollBar()->width()似乎是错误的.而且,在我的测试中,如果显示滚动条,则它始终是默认值100或值15.

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

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

更新#2:

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

  • 我还在努力,layout()->invalidate()比如QT:如何在show()之前预览布局中小部件的大小.

更新#3:

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

  • 之类的东西adjustSize(),并layout()->invalidate()Qt::WA_DontShowOnScreen似乎没有帮助.

  • Qt小部件缩小到最小所需尺寸的博客很有趣,但使用setSizeConstraint()同样好.

  • setSizeConstraint()move()方法都需要后续变化表,这里没有显示.

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

  • 我认为你应该使用`t-> frameWidth()*2`而不是4 (3认同)

小智 9

这是我(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())
Run Code Online (Sandbox Code Playgroud)

这给了我一个大小完美包装所有行和列的表.


Ali*_*Ali 6

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

首先,我们从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();
Run Code Online (Sandbox Code Playgroud)

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

w += ui->tableWidget->verticalHeader()->width();
h += ui->tableWidget->horizontalHeader()->height();
Run Code Online (Sandbox Code Playgroud)

然后,我们将每列的宽度附加到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);
Run Code Online (Sandbox Code Playgroud)

现在,如果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();
Run Code Online (Sandbox Code Playgroud)

不幸的是,我不知道更好的应用wh设置QTableWidget. 如果我使用:

ui->tableWidget->setMinimumWidth(w);
ui->tableWidget->setMaximumWidth(w);
ui->tableWidget->setMinimumHeight(h);
ui->tableWidget->setMaximumHeight(h);
Run Code Online (Sandbox Code Playgroud)

那么用户将根本无法调整小部件的大小。我认为我们需要使用某种手动布局。

这似乎有点离题,但如果您希望您的表具有更吸引人的视图,请使用此命令调整列的大小以适合其数据:

ui->tableWidget->resizeColumnsToContents();
Run Code Online (Sandbox Code Playgroud)


Cel*_*dor 5

最近,我发现了同样的问题。我测试了谷歌在 Qt 中心、Qt 项目中找到的数十个函数,但没有一个给我合适的大小。我做了那个我相信计算最接近表大小的函数。该函数提供了适当的垂直尺寸,但水平尺寸稍大——看起来它仍然覆盖了垂直滚动条。另外,我检查了表格尺寸是否超过桌面尺寸:

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);
}
Run Code Online (Sandbox Code Playgroud)

一些旁注:

QTableView
我使用 QTableView 但我认为它也可以与 QTableWidget 一起使用。

QVBoxLayout
我非常肯定 Qt 中可用的任何布局都可以在此函数中使用。

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


Zby*_*ler 5

要获得表格的宽度,请将垂直和水平标题的宽度与框架宽度的两倍相加。高度也是如此:

w = verticalHeader()->width() + horizontalHeader()->length() + frameWidth()*2
h = horizontalHeader()->height() + verticalHeader()->length() + frameWidth()*2
Run Code Online (Sandbox Code Playgroud)

有多种方法可以根据您的情况进行设置。首选可能是重写 sizeHint() 以返回您的值。这样,桌子就可以保持可调整大小,但如果没有其他要求,则大小将是您的。

实现此功能的关键是length()在正确的位置使用该方法,而不仅仅是width()or height()。我不太清楚为什么,但在这里它做了正确的事情;-)