Qt 自定义窗口的标题栏,重写鼠标事件实现,隐藏窗口,最大化/最小化窗口,关闭窗口

news2025/1/21 9:24:58

Qt 自定义窗口的标题栏,重写鼠标事件实现,隐藏窗口,最大化/最小化窗口,关闭窗口

1、main.cpp

#include "widget.h"

#include <QApplication>

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    Widget w;
    w.show();
    return a.exec();
}

2、widget.h

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>

QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE

class Widget : public QWidget
{
    Q_OBJECT

public:
    Widget(QWidget *parent = nullptr);
    ~Widget();

private:
    Ui::Widget *ui;

protected:
    // 重写鼠标事件,实现(隐藏窗口,最大化/最小化窗口,关闭窗口)
    void mousePressEvent(QMouseEvent *event) override; // 重写鼠标按下事件
    void mouseMoveEvent(QMouseEvent *event) override; // 重写鼠标移动事件
    void mouseReleaseEvent(QMouseEvent *event) override; // 重写鼠标释放事件

    /* 1、将函数声明为virtual的作用是允许这两个函数在派生类中被重写(override),
     * 因为resizeEvent()和moveEvent()是QWidget类中的虚函数,因此在派生类中也要将其声明为virtual,
     * 这样就可以在派生类中重新定义这两个函数,以实现派生类中对窗口大小改变和窗口移动事件的处理
     */
    virtual void resizeEvent(QResizeEvent *event);
    virtual void moveEvent(QMoveEvent *event);


private:
    QWidget *content;       // 中央控件
    QWidget *titleBar;      // 自定义标题栏

    bool isPressed;         // 是否按下鼠标左键
    QPoint mousePos;        // 鼠标按下时相对于窗口左上角的位置


// 声明标题栏槽函数(隐藏窗口,最大化/最小化窗口,关闭窗口)
private slots:
    void on_tbn_min_clicked();
    void on_tbn_max_clicked();
    void on_tbn_close_clicked();



};
#endif // WIDGET_H

3、widget.cpp

#include "widget.h"
#include "ui_widget.h"

#include <QMouseEvent> // 1、包含QMouseEvent类的头文件
#include <QHBoxLayout>
#include <QLabel>
#include <QPushButton>

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);

    // 2、去除默认的标题栏和边框
    // Qt::FramelessWindowHint:这是一个窗口提示,表示窗口没有边框
    // Qt::WindowMinimizeButtonHint:这是一个窗口提示,表示窗口有最小化按钮
    // Qt::WindowMaximizeButtonHint:这是一个窗口提示,表示窗口有最大化按钮
    // Qt::WindowCloseButtonHint:这是一个窗口提示,表示窗口有关闭按钮
    this->setWindowFlags(Qt::FramelessWindowHint | Qt::WindowMinimizeButtonHint | Qt::WindowMaximizeButtonHint | Qt::WindowCloseButtonHint);

    // 创建content容器,放到Widget窗口上
    content = new QWidget(this);
    // 设置 objectName 属性,QSS 文件中使用 #objectName 来定位到该控件
    content->setObjectName("content");
    // 设置content的大小策略,就是当窗口大小变化时,content怎么调整自己的大小,第一个参数是水平方向上的策略,第二个参数是垂直方向上的策略
    content->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
    // 设置content的位置和大小,参数是x坐标、y坐标、宽度和高度
    content->setGeometry(0, 0, width(), height());
    content->setStyleSheet("background-color:red;");

    // 在content上创建一个垂直布局,下面的2种方法都可以
    // content->setLayout(contentVLayout);
    QVBoxLayout *contentVLayout = new QVBoxLayout(content);

    // 3、自定义标题栏(隐藏窗口,最大化/最小化窗口,关闭窗口)
    // 创建titleBar容器,放到content容器里
    // titleBar = new QWidget(content);
    titleBar = new QWidget();
    titleBar->setObjectName("titleBar");
    titleBar->setStyleSheet("background-color:blue;");
    // 把titleBar添加到contentVLayout布局中
    contentVLayout->addWidget(titleBar);
    // 设置对齐方式为顶部对齐,让titleBar显示在垂直布局中的最上面
    contentVLayout->setAlignment(titleBar, Qt::AlignTop);
    // 设置内容边距为0,让它的子部件紧贴着它的边缘
    // 给contentVLayout这个垂直布局设置内边距为0,内边距是指布局和它的子部件之间的空隙
    // setContentsMargins()方法4个参数,分别表示左边距,上边距,右边距和下边距
    contentVLayout->setContentsMargins(0, 0, 0, 0);
    // 水平方向上可以自动扩展,在垂直方向上保持固定大小
    titleBar->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed);
    // 设置标题栏控件的位置和大小
    titleBar->setGeometry(0, 0, width(), 60);

    // 在titleBar上创建一个水平布局
    QHBoxLayout *titleBarHLayout = new QHBoxLayout(titleBar);
    // Qt::AlignLeft    水平方向靠左
    // Qt::AlignRight   水平方向靠右
    // Qt::AlignHCenter 水平方向居中
    // Qt::AlignTop     垂直方向靠上
    titleBarHLayout->setAlignment(Qt::AlignHCenter);
    // 在titleBarHLayout上添加你想要的控件,比如图标、标题、最小化按钮、最大化按钮、关闭按钮等
    QLabel *iconLabel = new QLabel(titleBar); // 创建图标标签
    iconLabel->setPixmap(QPixmap(":/imagenns/icon.png")); // 设置图标标签的图片
    iconLabel->setScaledContents(true); // 设置图标标签的图片自动缩放
    iconLabel->setFixedSize(20, 20); // 设置图标标签的固定大小
    QLabel *titleLabel = new QLabel(titleBar); // 创建标题标签
    titleLabel->setText("QWidget布局"); // 设置标题标签的文本
    titleLabel->setFont(QFont("Arial", 12, QFont::Bold)); // 设置标题标签的字体
    QPushButton *minButton = new QPushButton(titleBar); // 创建最小化按钮
    minButton->setText("-"); // 设置最小化按钮的文本
    minButton->setFixedSize(20, 20); // 设置最小化按钮的固定大小
    QPushButton *maxButton = new QPushButton(titleBar); // 创建最大化按钮
    maxButton->setText("+"); // 设置最大化按钮的文本
    maxButton->setFixedSize(20, 20); // 设置最大化按钮的固定大小
    QPushButton *closeButton = new QPushButton(titleBar); // 创建关闭按钮
    closeButton->setText("x"); // 设置关闭按钮的文本
    closeButton->setFixedSize(20, 20); // 设置关闭按钮的固定大小
    // 将这些控件添加到水平布局中,并设置合适的间距和弹簧
    titleBarHLayout->addWidget(iconLabel); // 将图标标签添加到水平布局中
    titleBarHLayout->addSpacing(10); // 添加10像素的间距
    titleBarHLayout->addWidget(titleLabel); // 将标题标签添加到水平布局中
    titleBarHLayout->addStretch(); // 添加弹簧,使得后面的按钮靠右对齐
    titleBarHLayout->addWidget(minButton); // 将最小化按钮添加到水平布局中
    titleBarHLayout->addWidget(maxButton); // 将最大化按钮添加到水平布局中
    titleBarHLayout->addWidget(closeButton); // 将关闭按钮添加到水平布局中

    connect (minButton, SIGNAL (clicked ()), this, SLOT (on_tbn_min_clicked ())); // 连接最小化按钮的信号和槽
    connect (maxButton, SIGNAL (clicked ()), this, SLOT (on_tbn_max_clicked ())); // 连接最大化按钮的信号和槽
    connect (closeButton, SIGNAL (clicked ()), this, SLOT (on_tbn_close_clicked ())); // 连接关闭按钮的信号和槽

}

Widget::~Widget()
{
    delete ui;
}



// 最小化按钮的槽函数
void Widget::on_tbn_min_clicked()
{
    // 最小化窗口
    showMinimized();
}

// 最大化按钮的槽函数
void Widget::on_tbn_max_clicked()
{
   if (isMaximized()) {
        // 还原窗口
        showNormal();
        // 切换图标
        qobject_cast<QPushButton *>(sender())->setText("+"); // 使用qobject_cast转换类型,并修改文本

   } else {
        // 最大化窗口
        showMaximized();
        qobject_cast<QPushButton *>(sender())->setText("-");
   }
}

// 关闭按钮的槽函数
void Widget::on_tbn_close_clicked()
{
   close();
}

// 重写鼠标事件,实现(隐藏窗口,最大化/最小化窗口,关闭窗口)
// 重写鼠标按下事件
void Widget::mousePressEvent(QMouseEvent *event)
{
    if (event->button() == Qt::LeftButton) { // 如果是鼠标左键按下
        isPressed = true; // 标记为按下状态
        mousePos = event->pos(); // 记录鼠标相对于窗口左上角的位置
    }
}
// 重写鼠标移动事件
void Widget::mouseMoveEvent(QMouseEvent *event)
{
    if (isPressed) { // 如果是按下状态
        move(event->globalPos() - mousePos); // 移动窗口到鼠标当前位置减去鼠标相对于窗口左上角位置之差处
    }
}
// 重写鼠标释放事件
void Widget::mouseReleaseEvent(QMouseEvent *event)
{
    if (event->button() == Qt::LeftButton) { // 如果是鼠标左键释放
        isPressed = false; // 标记为非按下状态
    }
}

/* 如果要实现widget窗口跟随鼠标拖动并自动跟随窗口缩放,
 * 需要在QWidget类中重写resizeEvent()函数和moveEvent()函数
 */
// 在resizeEvent()函数中,重新设置widget窗口的大小和位置
void Widget::resizeEvent(QResizeEvent *event)
{
    QWidget::resizeEvent(event); // 调用父类的resizeEvent()函数
    if(this->content) // 判断是否有widget被添加到Material窗口中
    {
        // 获取Material窗口的大小
        int w = this->width();
        int h = this->height();
        // 设置widget的大小为Material窗口的大小
        this->content->setGeometry(0, 0, w, h);
    }
}
// 在moveEvent()函数中,重新设置widget窗口的位置
void Widget::moveEvent(QMoveEvent *event)
{
    QWidget::moveEvent(event);
    if(this->content)
    {
        this->content->move(0, 0);
    }
}

4、效果展示
请添加图片描述

5、完成

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/539483.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ArcGis教程-画一幅城市的shp地图

怎样使用ArcGis10.6得到这么一幅shp地图呢&#xff1f; 首先打开ArcGis10.6&#xff0c;点击带黄底的小加号&#xff0c;添加底图。 可以选择中国地图彩色版&#xff0c;然后双击&#xff0c;转动鼠标滑轮找到属于自己的城市。 点击-目录&#xff0c;在新建的文件夹里右击-新建…

TS:如何判断联合类型变量的具体类型?

一 表示一个值可以是几种类型之一&#xff1a;联合类型 在TS中我们常会遇到这样一个问题。 一个变量&#xff0c;即可能是这种类型&#xff0c;也可能是那种类型&#xff0c;然后根据传入的类型的不同进行不同的操作。 比如下面这种情况&#xff1a; if (pet.name fish) {p…

三种灰狼优化算法(Grey Wolf Optimization)及仿真实验——附代码Matalb

目录 摘要&#xff1a; 灰狼算法原理&#xff1a; 灰狼算法流程&#xff1a; 改进的灰狼算法&#xff1a; 多目标的灰狼算法&#xff1a; 三种灰狼算法运行效果&#xff1a; &#xff08;1&#xff09;GWO &#xff08;2&#xff09;I-GWO &#xff08;3&#xff09;M…

Windows Server 2016 中文版、英文版下载 (updated May 2023)

Windows Server 2016 中文版、英文版下载 (updated May 2023) Windows Server 2016 Version 1607&#xff0c;2023 年 5 月更新 请访问原文链接&#xff1a;https://sysin.org/blog/windows-server-2016/&#xff0c;查看最新版。原创作品&#xff0c;转载请保留出处。 作者…

4.7 树的实现(上)

树 树&#xff08;Tree&#xff09;是n&#xff08;n≥0&#xff09;个节点的有限集合T&#xff0c;它满足两个条件 &#xff1a; 有且仅有一个特定的称为根&#xff08;Root&#xff09;的节点&#xff1b; 其余的节点可以分为m&#xff08;m≥0&#xff09;个互不相交的有…

电子企业WMS仓储管理系统解决方案

随着科技的飞速发展&#xff0c;电子制造行业对仓储管理系统的需求也越来越高。电子企业需要一种能够规划、执行和优化仓库货物流通的IT解决方案&#xff0c;以实现自动化操作和提高效率。本文将探讨电子企业WMS仓储管理系统解决方案&#xff0c;从需求分析、系统设计、实施与运…

在Windows系统中安装Wireshark(图文)

1.打开Wireshark官网后&#xff0c;点Get Acquainted->Download后进入到下载界面&#xff0c;在Stable Release中选择下载Windows 64位的安装包&#xff0c;单击Windows Installer(64-bit) 下载。 2.双击下载的安装包&#xff0c;如下图&#xff0c;点击Next。 3.点Noted&am…

ELK的安装部署与使用

ELK的安装与使用 安装部署 部署环境&#xff1a;Elasticsearch-7.17.3 Logstash-7.17.3 Kibana-7.17.3 一、安装部署Elasticsearch 解压目录&#xff0c;进入conf目录下编辑elasticsearch.yml文件&#xff0c;输入以下内容并保存 network.host: 127.0.0.1 http.port: 9200…

基于相似加权自组装框架的低质量少样本MRI脑卒中病变分割

文章目录 Stroke Lesion Segmentation from Low-Quality and Few-Shot MRIs via Similarity-Weighted Self-ensembling Framework摘要本文方法Soft Distribution-aware Updating (SDU) 实验结果 Stroke Lesion Segmentation from Low-Quality and Few-Shot MRIs via Similarity…

蓝桥杯模块学习5——按键

第一章 硬件部分 1.1 电路的组成部分 1.1.1 按键电路 原理图&#xff1a; 功能&#xff1a; &#xff08;1&#xff09; J5&#xff1a;当1和2相接&#xff0c;电路就变成一个4*4的矩阵键盘电路&#xff1b;当2和3相接时&#xff0c;电路变成了一个S4-S7的独立按键&#xf…

平板触控笔要原装的吗?苹果平替笔性价比高的推荐

与苹果的电容笔不同&#xff0c;市场上的电容笔只会给人一种倾斜的压感&#xff0c;并不会像苹果的电容笔那样&#xff0c;可以给人一种重力的压感。不过&#xff0c;如果你不一定要画画&#xff0c;那你就不用花很多钱去买一支苹果的原装电容笔了&#xff0c;只需一支平替电容…

ss命令使用详解

ss是Socket Statistics的缩写。顾名思义&#xff0c;ss命令可以用来获取socket统计信息&#xff0c;它可以显示和netstat类似的内容。ss的优势在于它能够显示更多更详细的有关TCP和连接状态的信息&#xff0c;而且比netstat更快速更高效。 当服务器的socket连接数量变得非常大…

从小白到专家:如何在营销中利用 AI 的力量

欢迎来到营销的未来&#xff0c;时至今日人工智能和人类专业知识以前所未有的方式结合在一起。 认识ChatGPT&#xff0c;这是改变游戏规则的革命性工具。 借助ChatGPT&#xff0c;你最终将能够利用AI的力量做出明智的、数据驱动的决策来满足你的受众需求。 但ChatGPT不仅仅是…

[高光谱]高光谱数据的获取与展示

一、环境准备 需要安装spectral包&#xff0c;这个包专门用于高光谱数据展示。 pip install spectral 二、数据加载 要预先准备原始高光谱的.mat数据和分类数据gt.mat(ground-turth)&#xff1b;然后使用scipy.io中的loadmat(.)将其读入程序。 from scipy.io import loadmat…

JCJC句子改写在线工具上线-202305

JCJC句子改写在线工具上线-202305 字根科技发布了新版JCJC在线句子改写功能。 使用网址&#xff1a; JCJC在线句子改写 新版的在线中文句子改写一共分为三种模式&#xff1a; 严谨模式普通模式休闲模式 上述三种改写模式适用于不同的改写需求&#xff0c;界面展示如下&…

springboot读取和写入csv文件数据

前言 csv格式的表格&#xff0c;和xls以及xlsx格式的表格有一些不同&#xff0c;不能够直接用处理xls的方式处理csv&#xff1b; 以下我将介绍如何读取并写入csv数据 准备工作 要处理csv格式的表格数据&#xff0c;我们首先需要引入pom.xml的依赖 <dependency><art…

Linux如何实现动态IP

Linux系统可以通过DHCP&#xff08;动态主机配置协议&#xff09;来实现动态IP。DHCP是一种自动分配IP地址的协议&#xff0c;它可以自动为网络中的设备分配IP地址、子网掩码、网关等网络参数&#xff0c;从而实现动态IP。 在Linux系统中&#xff0c;可以使用DHCP客户端工具来…

分享个常用的跨境电商数据分析平台

在跨境电商人眼中&#xff0c;适合用在跨境电商数据分析上的大数据分析平台该是怎样的&#xff1f;是效率高、财务指标计算快、业务能随时自助分析&#xff0c;最好是能将平台自身的分析经验分享给跨境电商企业&#xff0c;为企业提供更专业的服务。这样的大数据分析平台虽然少…

【Prometheus】实战Prometheus部署

目录 架构安装部署Exporters 架构 Prometheus的基本原理是通过HTTP周期性抓取被监控组件的状态&#xff0c;任意组件只要提供对应的HTTP接口并符合Prometheus定义的数据格式&#xff0c;就可以介入Prometheus监控 Prometheus Server负载定时在目标上抓取metrics(指标)数据&…

chatgpt-最常报错Access denied

一、问题 Access denied 的原因基本都是由于 IP 地址&#xff0c;例如我们用国内网络直接访问 ChatGPT 官网就会报错 Access denied&#xff08;访问被拒绝&#xff09;&#xff0c;Error reference number 1020&#xff08;错误参考编号&#xff1a;1020&#xff09; 二、原因…