[Qt初级] 解决 中QMainWindow和QDockWidget添加布局失败问题

时间:2022-11-06 00:22:03

初接触Qt,使用的教程是陆文周编写的《Qt5开发及实例》一书。

其中有关于QDockWidget、QStackedWidget这些类的介绍和使用实例。

要首先说明的是书上讲的非常的清楚,代码也附得非常明白,这本书还是非常值得看看的。

由于是初学,在新建工程的时候,我没有注意到基类的这个细节,没有按照教程上说的新建一个基于QDailog的类的项目,而是新建了一个基于QMainWindow的。

这就给我后面留下了一个问题,在使用布局的时候始终不起作用。

我先附上代码(Ps:代码没有问题,只是基类不是QDailog)Layout.cpp的构造函数中的代码

#include "layout.h"

Layout::Layout(QWidget *parent, Qt::WFlags flags)
: QMainWindow(parent, flags)
{ //ui.setupUi(this); /* ---- QDockWidget 停靠窗口类
setWindowTitle(tr("DockWin")); QDockWidget *dock = new QDockWidget(tr("DockWin-1"),this);//停靠窗口1,可移动
dock->setFeatures(QDockWidget::DockWidgetMovable);
dock->setAllowedAreas(Qt::LeftDockWidgetArea|Qt::RightDockWidgetArea); addDockWidget(Qt::RightDockWidgetArea,dock);
dock = new QDockWidget("DockWin-2",this);//停靠窗口2,可浮动,可关闭
dock->setFeatures(QDockWidget::DockWidgetClosable|QDockWidget::DockWidgetFloatable);
addDockWidget(Qt::RightDockWidgetArea,dock); dock = new QDockWidget("DockWin-3",this);//停靠窗口3,可浮动,可关闭
dock->setFeatures(QDockWidget::AllDockWidgetFeatures);
addDockWidget(Qt::RightDockWidgetArea,dock);
----- */
//setWindowTitle(tr("StackedWidget"));//QStackedWidget堆栈窗体类
setGeometry(,,,);
//堆栈窗口类
QListWidget *list;
QStackedWidget *stack; QLabel *label1;
QLabel *label2;
QLabel *label3; QWidget *widget1;
QWidget *widget2;
QWidget *widget3; //QWidget *newWidget = new QWidget(this);
//QDockWidget *Widget = new QDockWidget(this);
//QWidget *newWidget = new QWidget;
//Widget->setWidget(newWidget);
//Widget->setTitleBarWidget(newWidget);
//->setCentralWidget(newWidget);
list = new QListWidget(this);
//newWidget->setMinimumWidth(200);
//newWidget->setMinimumHeight(80);
//Widget->setMinimumWidth(200);
//Widget->setMinimumHeight(80);
list->insertItem(,tr("label-1"));
list->insertItem(,tr("label-2"));
list->insertItem(,tr("label-3")); label1 = new QLabel(tr("label - text - 1"));
label2 = new QLabel(tr("label - text - 2"));
label3 = new QLabel(tr("label - text - 3")); stack = new QStackedWidget(this);
stack->addWidget(label1);
stack->addWidget(label2);
stack->addWidget(label3); QHBoxLayout *mainlayout = new QHBoxLayout(this);
mainlayout->setMargin();
mainlayout->setSpacing();
mainlayout->addWidget(list);
mainlayout->addWidget(stack,,Qt::AlignHCenter);
mainlayout->setStretchFactor(list,);
mainlayout->setStretchFactor(stack,);
connect(list,SIGNAL(currentRowChanged(int)),stack,SLOT(setCurrentIndex(int)));
this->setLayout(mainlayout); } Layout::~Layout()
{ }

运行效果是这样的:

[Qt初级] 解决 中QMainWindow和QDockWidget添加布局失败问题

可以很明显的看出来,布局是没有效果的。QLabel全都堆在了一起。

首先,已经说过了,只要新建项目的时候基于QDailog便不会有这个问题了

其次,由于我使用的是VS2008+Qt addin所以编译的时候并没有报错,如果使用的Qt Creater其实编译会提示一个这样的错误的:

QWidget::setLayout: Attempting to set QLayout "" on MainWindow "", which already has a layout

这句话的意思是说,你已经给MainWindow设置过一个布局了,再设置一个就会出错。

或者你就是想要建基于QMainWindow的项目,那么解决方法是什么呢?

可以借助一个QWidget来解决这个问题,请看修改(1),(2)代码后的运行效果:

[Qt初级] 解决 中QMainWindow和QDockWidget添加布局失败问题

可以看到布局已经生效了,但是由于尺寸问题,newWidget并没有充满整个窗口。此时我们只需要加入代码(3)即可:

 #include "layout.h"

 Layout::Layout(QWidget *parent, Qt::WFlags flags)
: QMainWindow(parent, flags)
{
//ui.setupUi(this);
/* ---- QDockWidget 停靠窗口类
setWindowTitle(tr("DockWin"));
QDockWidget *dock = new QDockWidget(tr("DockWin-1"),this);//停靠窗口1,可移动
dock->setFeatures(QDockWidget::DockWidgetMovable);
dock->setAllowedAreas(Qt::LeftDockWidgetArea|Qt::RightDockWidgetArea); addDockWidget(Qt::RightDockWidgetArea,dock);
dock = new QDockWidget("DockWin-2",this);//停靠窗口2,可浮动,可关闭
dock->setFeatures(QDockWidget::DockWidgetClosable|QDockWidget::DockWidgetFloatable);
addDockWidget(Qt::RightDockWidgetArea,dock); dock = new QDockWidget("DockWin-3",this);//停靠窗口3,可浮动,可关闭
dock->setFeatures(QDockWidget::AllDockWidgetFeatures);
addDockWidget(Qt::RightDockWidgetArea,dock);
----- */
//setWindowTitle(tr("StackedWidget"));//QStackedWidget堆栈窗体类
setGeometry(,,,);
//堆栈窗口类
QListWidget *list;
QStackedWidget *stack;
QLabel *label1;
QLabel *label2;
QLabel *label3;
QWidget *widget1;
QWidget *widget2;
QWidget *widget3;
//QWidget *newWidget = new QWidget(this);
//QDockWidget *Widget = new QDockWidget(this);
QWidget *newWidget = new QWidget(this);//!!!!!添加代码(1)
setCentralWidget(newWidget);//!!!!添加代码(3)
//Widget->setWidget(newWidget);
//Widget->setTitleBarWidget(newWidget);
//->setCentralWidget(newWidget);
list = new QListWidget(this);
//newWidget->setMinimumWidth(200);
//newWidget->setMinimumHeight(80);
//Widget->setMinimumWidth(200);
//Widget->setMinimumHeight(80);
list->insertItem(,tr("label-1"));
list->insertItem(,tr("label-2"));
list->insertItem(,tr("label-3")); label1 = new QLabel(tr("label - text - 1"));
label2 = new QLabel(tr("label - text - 2"));
label3 = new QLabel(tr("label - text - 3")); stack = new QStackedWidget(this);
stack->addWidget(label1);
stack->addWidget(label2);
stack->addWidget(label3); QHBoxLayout *mainlayout = new QHBoxLayout(this);
mainlayout->setMargin();
mainlayout->setSpacing();
mainlayout->addWidget(list);
mainlayout->addWidget(stack,,Qt::AlignHCenter);
mainlayout->setStretchFactor(list,);
mainlayout->setStretchFactor(stack,);
connect(list,SIGNAL(currentRowChanged(int)),stack,SLOT(setCurrentIndex(int)));
newWidget->setLayout(mainlayout);//this->setLayout(mainlayout);!!!!修改代码(2)
}
Layout::~Layout(){
}

运行效果:

[Qt初级] 解决 中QMainWindow和QDockWidget添加布局失败问题

那么在QDockWidget中出现这个问题应该如何处理呢?我们知道QDockWidget中并没有

 setCentralWidget()这个方法。答案是SetWidget()
请看代码:
 #include "layout.h"

 Layout::Layout(QWidget *parent, Qt::WFlags flags)
: QMainWindow(parent, flags)
{
//ui.setupUi(this);
/* ---- QDockWidget 停靠窗口类
setWindowTitle(tr("DockWin"));
QDockWidget *dock = new QDockWidget(tr("DockWin-1"),this);//停靠窗口1,可移动
dock->setFeatures(QDockWidget::DockWidgetMovable);
dock->setAllowedAreas(Qt::LeftDockWidgetArea|Qt::RightDockWidgetArea); addDockWidget(Qt::RightDockWidgetArea,dock);
dock = new QDockWidget("DockWin-2",this);//停靠窗口2,可浮动,可关闭
dock->setFeatures(QDockWidget::DockWidgetClosable|QDockWidget::DockWidgetFloatable);
addDockWidget(Qt::RightDockWidgetArea,dock); dock = new QDockWidget("DockWin-3",this);//停靠窗口3,可浮动,可关闭
dock->setFeatures(QDockWidget::AllDockWidgetFeatures);
addDockWidget(Qt::RightDockWidgetArea,dock);
----- */
//setWindowTitle(tr("StackedWidget"));//QStackedWidget堆栈窗体类
setGeometry(,,,);
//堆栈窗口类
QListWidget *list;
QStackedWidget *stack;
QLabel *label1;
QLabel *label2;
QLabel *label3;
QWidget *widget1;
QWidget *widget2;
QWidget *widget3;
//QWidget *newWidget = new QWidget(this);
QDockWidget *Widget = new QDockWidget(this);//!!添加代码(4)
QWidget *newWidget = new QWidget(this);//!!!!!添加代码(1)
//setCentralWidget(newWidget);//!!!!添加代码(3)
Widget->setWidget(newWidget);//添加代码(5)
//Widget->setTitleBarWidget(newWidget);
//->setCentralWidget(newWidget);
list = new QListWidget(this);
//newWidget->setMinimumWidth(200);
//newWidget->setMinimumHeight(80);
//Widget->setMinimumWidth(200);
//Widget->setMinimumHeight(80);
list->insertItem(,tr("label-1"));
list->insertItem(,tr("label-2"));
list->insertItem(,tr("label-3")); label1 = new QLabel(tr("label - text - 1"));
label2 = new QLabel(tr("label - text - 2"));
label3 = new QLabel(tr("label - text - 3")); stack = new QStackedWidget(this);
stack->addWidget(label1);
stack->addWidget(label2);
stack->addWidget(label3); QHBoxLayout *mainlayout = new QHBoxLayout(this);
mainlayout->setMargin();
mainlayout->setSpacing();
mainlayout->addWidget(list);
mainlayout->addWidget(stack,,Qt::AlignHCenter);
mainlayout->setStretchFactor(list,);
mainlayout->setStretchFactor(stack,);
connect(list,SIGNAL(currentRowChanged(int)),stack,SLOT(setCurrentIndex(int)));
newWidget->setLayout(mainlayout);//this->setLayout(mainlayout);!!!!修改代码(2)
}
Layout::~Layout(){
}

运行效果:

[Qt初级] 解决 中QMainWindow和QDockWidget添加布局失败问题

PS:

  其实QMainWindow和QDailog的基类都是QWidget,真的搞不懂为何在QDailog派生的类中可以直接使用布局类,而在QMainWindow的派生类中却不可以。

  知其然而不知其所以然的感受不太好,望知道的大神解疑。