Как определить правильный размер QTableWidget? - PullRequest
29 голосов
/ 07 января 2012

Есть ли какой-либо способ установить "правильный" размер 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 после.Я смотрел на подсказки и политику размера Qt, пока у меня не кружилась голова, и я попробовал setResizeMode(QHeaderView::Fixed) и setStretchLastSection(false).Может я что-то упускаю очевидное?Это с Qt 4.7.4 на CentOS 5, если это имеет значение.Спасибо за любую помощь.

Редактировать : В ответ на DK, , если строка resize(1, 1); отсутствует, есть равно и напротив проблема: окно слишком большое.

И в ответ на Донотало добавление:

tbl->setMaximumWidth(222);
tbl->setMinimumWidth(222);
tbl->setMaximumHeight(88);
tbl->setMinimumHeight(88); 

даст желаемый размер окна (по крайней мере, на моей машине), но не желаемым образом.Как мы должны вычислить «константы» 222 и 88?

И ответ Тон на Qt: Как заставить скрытый виджет рассчитать его макет? , похоже, не работаетздесь: добавление tbl->setAttribute(Qt::WA_DontShowOnScreen); tbl->show(); оставило значение tbl->width() без изменений на 640.

Ответы [ 7 ]

17 голосов
/ 07 января 2012

Тема Как установить точный размер QTableWidget, чтобы избежать использования полос прокрутки?(Архив Qt-интереса, июнь 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 пересчитать геометрию в этом тесте.Мне нужно было применить его ко всему окну.

  • Любые предложения по улучшению будут welome.(Я подожду немного перед тем, как принять собственный ответ, в случае, если найдутся лучшие решения.)

Обновление № 2 :

  • Поток Qt-Interest может быть неправильным (или, по крайней мере, он не согласен с моей версией Qt, работающей моей машины) относительно подробностей о том, как рассчитать размер: +1 для каждой линии сетки не требуется, но необходимо общее +4.

  • Я все еще работаю через layout()->invalidate() против, например, QT: Как просмотретьразмеры виджетов в макете ДО a show () .

Обновление # 3 :

  • Это мой окончательный вариант - но я не очень доволен этим.Любые улучшения будут очень добро пожаловать.

  • Такие вещи, как adjustSize() и layout()->invalidate() и Qt::WA_DontShowOnScreen, похоже, не помогут.

  • Блог Сокращение виджетов Qt до минимально необходимого размера интересно, но использование setSizeConstraint() так же хорошо.

  • The 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, но что-то где-то сломалось.

9 голосов
/ 03 июня 2016

Вот мое (питоническое) решение этой проблемы:

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())

Это дает мне таблицу с размером, идеально охватывающим все строки и столбцы.

4 голосов
/ 15 сентября 2012

Недавно я обнаружил ту же проблему.Я протестировал десятки функций, найденных Google в Qt Center, Qt Project, но ни одна из них не дала мне нужного размера.Я сделал ту функцию, которая, я считаю, вычисляет ближайший размер таблицы.Функция дает правильный вертикальный, но немного больший горизонтальный размер - похоже, он все еще покрывает вертикальную полосу прокрутки.Кроме того, я проверил, превышают ли размеры таблицы размер рабочего стола:

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 голосов
/ 30 сентября 2013

Я имел дело с той же проблемой и подправил предыдущие ответы, чтобы получить правильный рабочий код.

Сначала мы получаем поля содержимого от 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);

Теперь, если w или h слишком велики, чтобы поместиться на окне, мы уменьшаем их до 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();

К сожалению, я не знаю лучшего способаприменения w и h для установки ширины и высоты QTableWidget.Если я использую:

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

, тогда пользователь вообще не сможет изменить размер виджета.Я думаю, что нам нужно использовать какое-то ручное размещение.

Это может показаться немного не по теме, но если вы хотите, чтобы у вашей таблицы было более привлекательное представление, используйте эту команду, чтобы изменить размеры столбцов, чтобы соответствовать их данным:

ui->tableWidget->resizeColumnsToContents();
2 голосов
/ 28 октября 2016

Чтобы получить ширину таблицы, сложите ширину вертикальных и горизонтальных заголовков и удвоенную ширину кадра.То же самое касается высоты:

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

Существует несколько способов их установки в зависимости от ваших обстоятельств.Вероятно, предпочтительным было бы переопределить sizeHint () для возврата ваших значений.Таким образом, размер таблицы остается изменяемым, но другие требования не будут соответствовать вашему размеру.

Ключом к этой работе является использование метода length() в нужных местах вместо просто width() или height().Я не совсем уверен, почему, но здесь все происходит правильно; -)

1 голос
/ 04 августа 2016

очевидно есть - может быть, какая-то ошибка? - в функции horizontalHeader () -> 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;
1 голос
/ 21 апреля 2016

Я не могу комментировать пост Джозефа Кунси / Селдора, поэтому я пишу его здесь: Чтобы получить истинное значение QStyle::PM_TitleBarHeight, вы должны сделать это следующим образом:

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

Это потому, что enum не имеет точных значений. Я думаю, что ваше неправильное понимание того, как это работает, было связано с вводящей в заблуждение документацией Qt , где, по-видимому, перечисление непосредственно представляет значения, в то время как, конечно, это не так (поскольку это зависит от стиля).

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...