【QT入门】对无边框窗口自定义标题栏并实现拖动和拉伸效果

往期回顾:

【QT入门】 QScrollArea实际运用之滑动Widget设计-CSDN博客

【QT入门】QWidget类的简单介绍-CSDN博客

【QT入门】 鼠标按下和移动事件实现无边框窗口拖动-CSDN博客

【QT入门】对无边框窗口自定义标题栏并实现拖动和拉伸效果

对无边框窗口添加自定义标题栏是很简单的,难的是逻辑实现拖拽窗口边框和实现拉伸效果。特别是在无边框窗口下还要实现拉伸效果,我们分别把自定义标题栏的ui设计和拉伸效果放到两个类去实现。

一、UI界面设计

1、最终效果

2、创建标题栏类 

用一个初始化ui函数,定义6个控件,分别是logo、标题、设置按钮、最小化最大化关闭按钮。

#pragma once
#include <QtWidgets/QWidget>
#include <QLabel>
#include <QPushButton>


class CTitleBar: public QWidget
{
public:
    CTitleBar(QWidget* parent = nullptr);
    ~CTitleBar();

private:
    void initUi();

    void mousePressEvent(QMouseEvent* event) override;

private :
    QLabel* m_pLogo;
    QLabel* m_pTitle;

    QPushButton* m_setBtn;
    QPushButton* m_minBtn;
    QPushButton* m_maxBtn;
    QPushButton* m_closeBtn;
};

3、定义相关控件并布局 

我们需要分别定义实现每个控件,设置大小,最后用一个水平布局,把这些控件都加进去,在中间还加了一个弹簧。

setAttribute(Qt::WA_StyledBackground);

有些时候子窗口颜色会被父类窗口给替换掉,为了避免这种影响,这里通过setAttribute方法禁止父窗口影响子窗口样式。

pHlay->addStretch();//弹簧
 pHlay->setContentsMargins(5, 5, 5, 5);//设置布局和窗口边框的距离

#include "CTitleBar.h"
#include <QHBoxLayout>
#include <qt_windows.h>
#include <QMouseEvent>
#include <QWidget>
#include <QSpacerItem>
#pragma comment(lib, "user32.lib")

CTitleBar::CTitleBar(QWidget* parent)
    : QWidget(parent)
{
    //在窗口关闭时自动删除窗口对象,即在调用close()函数关闭窗口时,窗口对象会被自动删除,释放内存。
    this->setAttribute(Qt::WA_DeleteOnClose);

    initUi();
}

CTitleBar::~CTitleBar()
{
}

void CTitleBar::initUi()
{

    //有些时候子窗口颜色会被父类窗口给替换掉
    //为了避免这种影响,禁止父窗口影响子窗口样式
    setAttribute(Qt::WA_StyledBackground);
    this->setFixedHeight(32 + 5 * 2);

    this->setStyleSheet("background-color:rgb(0,122,122)");

    QLabel* m_pLogo;
    QLabel* m_pTitle;
    QPushButton* m_minBtn;
    QPushButton* m_maxBtn;
    QPushButton* m_closeBtn;

    m_pLogo = new QLabel(this);
    m_pLogo->setFixedSize(32, 32);  

    m_pTitle = new QLabel(this);
    m_pTitle->setText(u8"我是标题");
    m_pTitle->setFixedWidth(120);

    m_setBtn = new QPushButton(this);
    m_setBtn->setFixedSize(32, 32);

    m_minBtn = new QPushButton(this);
    m_minBtn->setFixedSize(32, 32);

    m_maxBtn = new QPushButton(this);
    m_maxBtn->setFixedSize(32, 32);

    m_closeBtn = new QPushButton(this);
    m_closeBtn->setFixedSize(32, 32);

    QHBoxLayout* pHlay = new QHBoxLayout(this);
    pHlay->addWidget(m_pLogo);
    pHlay->addWidget(m_pTitle);
    pHlay->addStretch();//添加了一个弹簧到水平布局pHlay中,弹簧会随着窗口大小变化而比变化

    pHlay->addWidget(m_setBtn);

    pHlay->addWidget(m_minBtn);

    pHlay->addWidget(m_maxBtn);

    pHlay->addWidget(m_closeBtn);

    pHlay->setContentsMargins(5, 5, 5, 5);

}

4、添加到主类 

由于在主类的.h文件里已经声明了m_pTitleBar,这里直接定义,创建w窗口,用一个垂直布局加进去就ok。

void _3_ListWidgetPro::initUI()
{
    m_pTitleBar = new CTitleBar(this);

    //m_pTitleBar->setStyleSheet("background-color:blue");

    QWidget* w = new QWidget(this);
    w->setMinimumSize(600, 400);

    QVBoxLayout* pVlay = new QVBoxLayout(this);
    pVlay->addWidget(m_pTitleBar);
    pVlay->addWidget(w);

    pVlay->setContentsMargins(0, 0, 0, 0);
}

二、拖动拉伸的逻辑实现

 1、实现自定义标题栏拖动

我们前面其实已经涉及到这类方法,当时是通过重写mousepress和mousemove方法来实现的

【QT入门】 鼠标按下和移动事件实现无边框窗口拖动-CSDN博客

这里我们换另一种方式来实现:

#include <qt_ windows.h>
#pragma comment(ib, "user32.lib")
void CTitleBar::mousePressEvent(QMouseEvent* event)
{
if (ReleaseCapture())
QWidget* pWindow = this->window();
if (pWindow->is TopLevel())
{
SendMessage(HWND(pWindow->winld(), WM_ SYSCOMMAND, SC_ _MOVE
+ HTCAPTION, 0);
}
evenf->ignore0;
}

首先调用了ReleaseCapture()函数来释放鼠标捕获,然后获取了当前窗口部件的指针。

接着,通过判断当前窗口部件是否为顶层窗口(top-level),如果是顶层窗口,则发送了一个WM_SYSCOMMAND消息给窗口的句柄(HWND),消息参数为SC_MOVE + HTCAPTION。这个消息的作用是模拟用户拖动窗口的操作,通过标题栏拖动来移动整个窗口。

最后,调用了event->ignore()函数来告诉Qt框架忽略这个鼠标事件,这样可以确保鼠标事件不会继续传递给其他事件处理函数。

这里涉及到了一个概念:窗口句柄

1.1窗口句柄

在Qt中,窗口句柄(Window Handle)是一个指向窗口对象的唯一标识符。在Qt中,窗口句柄通常表示为一个指向QWidget对象的指针。每个QWidget对象都有一个窗口句柄,用于在操作系统级别标识和管理窗口。

窗口句柄在Qt中通常用于与底层操作系统进行交互,例如在进行窗口管理、绘制和事件处理时。通过窗口句柄,可以实现与操作系统窗口相关的功能,如设置窗口属性、处理窗口事件等。

1.1.1获取窗口句柄方法:

QWidget::winId()

返回窗口句柄的唯一整数标识符。
QWidget::effectiveWinId()返回窗口句柄的有效整数标识符。
QWidget::internalWinId()返回窗口句柄的内部整数标识符。

2、实现无边框窗口拉伸效果 

这一逻辑实现,我们是直接把边框拉伸功能抽象出来,需要的时候派生这个类就可以了,至于具体逻辑实现,大家看这篇,直接派生CFrameLessWidgetBase类即可。

【QT入门】 设计实现无边框窗口拉伸的公用类-CSDN博客


都看到这里了,点个赞再走呗朋友~

加油吧,预祝大家变得更强!

  • 28
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Qt5提供了自定义标题栏的功能,可以通过QObject::findChild()函数找到标题栏的QWidget对象,然后自定义这个QWidget的样式和行为。实现分屏的方法如下: 1. 创建一个新的QWidget作为窗口标题栏,设置它的布局以及其他需要的控件,如最小化按钮、最大化按钮和关闭按钮。 2. 使用QObject::findChild()函数找到QWidget对象,并在其上设置样式和行为。例如,可以使用QSS来设置样式,使其看起来像一个标准的窗口标题栏,并实现拖动窗口的功能。 3. 为窗口的主体内容创建一个新的QWidget对象,并将其设置为窗口的主要布局。 4. 将窗口的主体内容QWidget对象添加到窗口的布局中,并设置合适的位置和大小。 5. 使用QSplitter将窗口的主体内容分割成多个视图。可以通过addWidget()或setWidget()函数将需要分割的QWidget对象添加到QSplitter中。 6. 将QSplitter对象添加到窗口的布局中。 通过以上步骤,就可以完成Qt5自定义标题栏实现分屏的功能。在自定义标题栏中,可以添加其他需要的控件,如窗口图标、标题等,并实现这些控件的相应功能。同时,使用QSplitter可以将窗口的主体内容分割为多个视图,实现分屏显示不同的内容。 ### 回答2: 要实现Qt5自定义标题栏实现分屏,首先需要创建一个自定义窗口,然后通过设置窗口样式来隐藏系统自带的标题栏。 1. 创建自定义窗口:使用Qt的QWidget或QMainWindow类创建一个新的窗口对象。 2. 设置窗口样式:使用setWindowFlags方法设置窗口的样式,将其设置为自定义标题栏样式。 ```cpp setWindowFlags(Qt::FramelessWindowHint); ``` 这会隐藏系统自带的标题栏。 3. 创建分屏功能:通过添加分屏窗口拖动边界等方式实现分屏功能。 - 添加分屏窗口:可以使用QWidget或QMainWindow类创建新的子窗口对象,并将其添加到主窗口的布局中。 - 拖动边界:可以监听窗口的鼠标事件,并在边界上按下鼠标时,根据鼠标移动的位置动态调整分屏窗口的大小。 4. 实现自定义标题栏:使用QWidget或QMainWindow类创建一个自定义标题栏对象,并将其添加到主窗口的布局中。 ```cpp QWidget* titleBar = new QWidget(this); // 设置标题栏的样式 titleBar->setObjectName("TitleBar"); // 将标题栏添加到主窗口的布局中 layout->addWidget(titleBar); ``` 可以根据需要自定义标题栏的样式,例如设置背景颜色、标题文字等。 5. 实现窗口移动:可以监听标题栏的鼠标事件,捕获鼠标按下、移动和释放的事件,通过计算鼠标的移动距离来实现窗口拖动。 ```cpp void mousePressEvent(QMouseEvent* event) override { if (event->button() == Qt::LeftButton) { m_dragPosition = event->globalPos() - frameGeometry().topLeft(); event->accept(); } } void mouseMoveEvent(QMouseEvent* event) override { if (event->buttons() & Qt::LeftButton) { move(event->globalPos() - m_dragPosition); event->accept(); } } void mouseReleaseEvent(QMouseEvent* event) override { if (event->button() == Qt::LeftButton) { event->accept(); } } ``` 通过以上步骤,就可以实现Qt5自定义标题栏实现分屏的功能。注意,还可以根据需求再增加其他功能,例如最大化、最小化、关闭等按钮。 ### 回答3: 在Qt5中,要实现自定义标题栏实现分屏的功能,可以按照以下步骤进行: 1. 创建一个新的窗口应用程序项目,并在主窗口的.ui文件中添加一个可移动的控件作为标题栏。 2. 在标题栏控件中添加必要的窗口操作按钮,例如最小化、最大化和关闭按钮,并为这些按钮添加相应的槽函数,使其能够实现相应的窗口操作。 3. 设置窗口的样式表,将标题栏的背景颜色设置为透明,使其与窗口内容区分开来。 4. 在标题栏控件中重写鼠标事件,使其能够实现窗口的移动操作。这可以通过重写mousePressEvent、mouseMoveEvent和mouseReleaseEvent函数来实现。 5. 在窗口中添加一个Splitter控件,用于分屏显示内容。可以使用QSplitter类的addWidget函数将窗口中的其他控件添加到Splitter中,从而实现分屏效果。 6. 通过在窗口的resizeEvent函数中动态调整Splitter的尺寸,使其能够适应窗口的大小变化。 7. 可以通过设置Splitter的分割方向和分割比例来控制内容的分屏效果。例如,使用setOrientation函数设置分割方向为水平或垂直,使用setSizes函数设置分割比例。 8. 如果需要在分屏的内容区域中显示不同的内容,可以使用QStackedWidget类将相关的内容添加到堆栈中,并根据需要在堆栈中切换显示的内容。 通过以上步骤,就可以实现Qt5中自定义标题栏实现分屏的功能。这样,用户可以自由移动窗口和操作窗口的按钮,同时还可以在分屏的窗口中显示不同的内容。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值