【QT】一个界面中嵌入其它界面(三)

在 Qt 中,通过 UI 设计代码布局 实现界面 A 中同时显示界面 B 和 C,并精确指定它们的位置,可以通过以下两种方式实现。以下是详细步骤和完整代码:


方法 0:使用 Qt Designer 可视化布局

通过 Qt Designer 拖拽控件并设置布局,直观控制子界面的位置。

步骤
  1. 创建子界面 B 和 C

    • 新建两个自定义 QWidget 类(如 WidgetBWidgetC),并在 Qt Designer 中设计它们的 UI。
    • 保存为 .ui 文件(例如 widgetb.uiwidgetc.ui)。
  2. 提升为自定义控件

    • 在主界面 A 的 .ui 文件中,拖入两个 QGridLayout 控件。
    • 两个控件分别重命名为 WidgetB_glWidgetC_gl
  3. 设置布局或绝对位置

cpp 复制代码
WidgetB* widgetb= new WidgetB(this);
WidgetC* widgetc= new WidgetC(this);

ui->WidgetB_gl->addWidget(widgetb); 
ui->WidgetC_gl->addWidget(widgetc);

方法 1:使用 Qt Designer 可视化布局

通过 Qt Designer 拖拽控件并设置布局,直观控制子界面的位置。

步骤
  1. 创建子界面 B 和 C

    • 新建两个自定义 QWidget 类(如 WidgetBWidgetC),并在 Qt Designer 中设计它们的 UI。
    • 保存为 .ui 文件(例如 widgetb.uiwidgetc.ui)。
  2. 提升为自定义控件

    • 在主界面 A 的 .ui 文件中,拖入两个 QWidget 控件。
    • 右键点击这两个 QWidget,选择 Promote to... ,分别提升为 WidgetBWidgetC
  3. 设置布局或绝对位置

    • 使用布局管理器 :将界面 B 和 C 拖入 QGridLayoutQHBoxLayout/QVBoxLayout,调整行列位置。
    • 绝对位置 :取消布局管理器,直接拖拽调整 WidgetBWidgetC 的位置和大小。

方法 2:纯代码实现布局

通过代码动态创建子界面并添加到指定位置。

完整代码示例
1. 子界面 B 和 C 的定义
cpp 复制代码
// WidgetB.h
#ifndef WIDGETB_H
#define WIDGETB_H

#include <QWidget>
#include <QLabel>
#include <QVBoxLayout>

class WidgetB : public QWidget {
    Q_OBJECT
public:
    explicit WidgetB(QWidget* parent = nullptr);
};

#endif // WIDGETB_H

// WidgetB.cpp
#include "WidgetB.h"

WidgetB::WidgetB(QWidget* parent) : QWidget(parent) {
    QLabel* label = new QLabel("子界面 B", this);
    label->setAlignment(Qt::AlignCenter);
    QVBoxLayout* layout = new QVBoxLayout(this);
    layout->addWidget(label);
    setStyleSheet("background-color: #FFCCCC;"); // 粉色背景
}
cpp 复制代码
// WidgetC.h
#ifndef WIDGETC_H
#define WIDGETC_H

#include <QWidget>
#include <QLabel>
#include <QVBoxLayout>

class WidgetC : public QWidget {
    Q_OBJECT
public:
    explicit WidgetC(QWidget* parent = nullptr);
};

#endif // WIDGETC_H

// WidgetC.cpp
#include "WidgetC.h"

WidgetC::WidgetC(QWidget* parent) : QWidget(parent) {
    QLabel* label = new QLabel("子界面 C", this);
    label->setAlignment(Qt::AlignCenter);
    QVBoxLayout* layout = new QVBoxLayout(this);
    layout->addWidget(label);
    setStyleSheet("background-color: #CCFFCC;"); // 绿色背景
}

2. 主界面 A 的实现
cpp 复制代码
// WidgetA.h
#ifndef WIDGETA_H
#define WIDGETA_H

#include <QWidget>
#include <QGridLayout>
#include "WidgetB.h"
#include "WidgetC.h"

class WidgetA : public QWidget {
    Q_OBJECT
public:
    explicit WidgetA(QWidget* parent = nullptr);
};

#endif // WIDGETA_H

// WidgetA.cpp
#include "WidgetA.h"

WidgetA::WidgetA(QWidget* parent) : QWidget(parent) {
    // 创建网格布局
    QGridLayout* gridLayout = new QGridLayout(this);

    // 创建子界面 B 和 C,并指定父对象为 A
    WidgetB* widgetB = new WidgetB(this);
    WidgetC* widgetC = new WidgetC(this);

    // 将 B 和 C 添加到网格布局的指定位置
    gridLayout->addWidget(widgetB, 0, 0);  // 第 0 行,第 0 列(左上角)
    gridLayout->addWidget(widgetC, 1, 1);  // 第 1 行,第 1 列(右下角)

    // 设置布局边距和间距
    gridLayout->setContentsMargins(20, 20, 20, 20);
    gridLayout->setSpacing(15);

    // 设置主界面背景色
    setStyleSheet("background-color: #F0F0F0;");
    resize(600, 400); // 初始窗口大小
}

3. 主函数
cpp 复制代码
// main.cpp
#include <QApplication>
#include "WidgetA.h"

int main(int argc, char* argv[]) {
    QApplication app(argc, argv);
    WidgetA widgetA;
    widgetA.setWindowTitle("多子界面布局示例");
    widgetA.show();
    return app.exec();
}

效果说明

  • 布局管理器控制

    • 子界面 B 位于左上角(第 0 行第 0 列),子界面 C 位于右下角(第 1 行第 1 列)。
    • 窗口缩放时,子界面会按布局比例自动调整位置和大小。
  • 自定义样式

    • 子界面 B 为粉色背景,子界面 C 为绿色背景,主界面为灰色背景,便于观察位置。

扩展:手动指定绝对坐标

如果希望子界面位置固定(不随窗口缩放变化),可以使用 setGeometry

cpp 复制代码
// WidgetA.cpp(替代布局代码)
WidgetA::WidgetA(QWidget* parent) : QWidget(parent) {
    // 创建子界面 B 和 C
    WidgetB* widgetB = new WidgetB(this);
    WidgetC* widgetC = new WidgetC(this);

    // 手动设置位置和大小
    widgetB->setGeometry(50, 50, 200, 150);   // (x, y, width, height)
    widgetC->setGeometry(300, 200, 200, 150);

    // 设置主界面背景色和大小
    setStyleSheet("background-color: #F0F0F0;");
    resize(600, 400);
}

关键点总结

  1. 父子关系 :确保子界面(B 和 C)的父对象为 WidgetA(通过构造函数或 setParent)。
  2. 布局管理器
    • 使用 QGridLayoutQHBoxLayoutQVBoxLayout 实现动态布局。
    • 通过行列索引(如 gridLayout->addWidget(widget, row, col))精确控制位置。
  3. 绝对坐标
    • 使用 setGeometry(x, y, width, height)move() + resize() 固定位置。
    • 需在窗口大小变化事件(resizeEvent)中手动调整子界面位置。
  4. 样式调试 :通过 setStyleSheet 设置背景色,直观观察子界面位置。

扩展场景

  • 动态添加/移除子界面 :通过按钮点击事件调用 addWidget()removeWidget()
  • 交互通信:在子界面中定义信号,主界面中绑定槽函数,实现数据传递。
  • 复杂布局 :嵌套使用多种布局管理器(如将 QHBoxLayout 嵌入 QGridLayout)。

通过上述方法,可以灵活地在 Qt应用中实现多子界面的精确布局。

相关推荐
秦少游在淮海18 分钟前
C++ - string 的使用 #auto #范围for #访问及遍历操作 #容量操作 #修改操作 #其他操作 #非成员函数
开发语言·c++·stl·string·范围for·auto·string 的使用
AAA废品回收站陈师傅24 分钟前
68常用控件_QGroupBox的使用
qt
const54426 分钟前
cpp自学 day2(—>运算符)
开发语言·c++
心扬27 分钟前
python生成器
开发语言·python
明月醉窗台27 分钟前
qt使用笔记二:main.cpp详解
数据库·笔记·qt
阿蒙Amon33 分钟前
06. C#入门系列【自定义类型】:从青铜到王者的进阶之路
开发语言·c#
虾球xz37 分钟前
CppCon 2015 学习:CLANG/C2 for Windows
开发语言·c++·windows·学习
沉到海底去吧Go1 小时前
【图片自动识别改名】识别图片中的文字并批量改名的工具,根据文字对图片批量改名,基于QT和腾讯OCR识别的实现方案
数据库·qt·ocr·图片识别自动改名·图片区域识别改名·pdf识别改名
CodeWithMe1 小时前
【C/C++】namespace + macro混用场景
c语言·开发语言·c++
蓝婷儿1 小时前
6个月Python学习计划 Day 17 - 继承、多态与魔术方法
开发语言·python·学习