效果

自定义标题栏

titleBar.h

#ifndef TITLEBAR_H
#define TITLEBAR_H #include <QLabel>
#include <QPushButton> class titleBar : public QWidget
{
Q_OBJECT public:
explicit titleBar(QWidget *parent = nullptr);
~titleBar(); protected: //双击标题栏进行界面的最大化/还原
virtual void mouseDoubleClickEvent(QMouseEvent *event); //进行界面的拖动
virtual void mousePressEvent(QMouseEvent *event); //设置界面标题与图标
virtual bool eventFilter(QObject *obj, QEvent *event); private slots: //进行最小化、最大化/还原、关闭操作
void onClicked(); private: //最大化/还原
void updateMaximize(); private:
QLabel *m_pIconLabel; //标题栏图标
QLabel *m_pTitleLabel; //标题栏标题
QPushButton *m_pMinimizeButton; //最小化按钮
QPushButton *m_pMaximizeButton; //最大化/还原按钮
QPushButton *m_pCloseButton; //关闭按钮
}; #endif // TITLEBAR_H

titleBar.cpp

#include <QLabel>
#include <QPushButton>
#include <QHBoxLayout>
#include <QEvent>
#include <QMouseEvent>
#include <QApplication>
#include "titleBar.h" //调用WIN API需要用到的头文件与库
#ifdef Q_OS_WIN
#pragma comment(lib, "user32.lib")
#include <qt_windows.h>
#endif titleBar::titleBar(QWidget *parent)
: QWidget(parent)
{
setFixedHeight(30); //给成员变量申请内存
m_pIconLabel = new QLabel(this);
m_pTitleLabel = new QLabel(this);
m_pMinimizeButton = new QPushButton(this);
m_pMaximizeButton = new QPushButton(this);
m_pCloseButton = new QPushButton(this); //初始化图标Label
m_pIconLabel->setFixedSize(20, 20);
m_pIconLabel->setScaledContents(true); m_pTitleLabel->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed); //设置按钮的固定大小、图片、取消边框
m_pMinimizeButton->setIconSize(QSize(27,22));
m_pMinimizeButton->setIcon(QIcon("://Images/icon_min.png"));
m_pMinimizeButton->setFlat(true);
//--
m_pMaximizeButton->setIconSize(QSize(27,22));
m_pMaximizeButton->setIcon(QIcon("://Images/icon_max.png"));
m_pMaximizeButton->setFlat(true);
//--
m_pCloseButton->setIconSize(QSize(27,22));
m_pCloseButton->setIcon(QIcon("://Images/icon_close.png"));
m_pCloseButton->setFlat(true); //设置窗口部件的名称
m_pTitleLabel->setObjectName("whiteLabel");
m_pMinimizeButton->setObjectName("minimizeButton");
m_pMaximizeButton->setObjectName("maximizeButton");
m_pCloseButton->setObjectName("closeButton"); //给按钮设置静态tooltip,当鼠标移上去时显示tooltip
m_pMinimizeButton->setToolTip("Minimize");
m_pMaximizeButton->setToolTip("Maximize");
m_pCloseButton->setToolTip("Close"); //标题栏布局
QHBoxLayout *pLayout = new QHBoxLayout(this);
pLayout->addWidget(m_pIconLabel);
pLayout->addSpacing(5);
pLayout->addWidget(m_pTitleLabel);
pLayout->addWidget(m_pMinimizeButton);
pLayout->addWidget(m_pMaximizeButton);
pLayout->addWidget(m_pCloseButton);
pLayout->setSpacing(0);
pLayout->setContentsMargins(5, 0, 5, 0);
setLayout(pLayout); //连接三个按钮的信号槽
connect(m_pMinimizeButton, SIGNAL(clicked(bool)), this, SLOT(onClicked()));
connect(m_pMaximizeButton, SIGNAL(clicked(bool)), this, SLOT(onClicked()));
connect(m_pCloseButton, SIGNAL(clicked(bool)), this, SLOT(onClicked()));
} titleBar::~titleBar()
{ } //双击标题栏进行界面的最大化/还原
void titleBar::mouseDoubleClickEvent(QMouseEvent *event)
{
Q_UNUSED(event); //没有实质性的作用,只是用来允许event可以不使用,用来避免编译器警告 emit m_pMaximizeButton->clicked();
} //进行界面的拖动 [一般情况下,是界面随着标题栏的移动而移动,所以我们将事件写在标题栏中比较合理]
void titleBar::mousePressEvent(QMouseEvent *event)
{
#ifdef Q_OS_WIN
if (ReleaseCapture())
{
QWidget *pWindow = this->window();
if (pWindow->isTopLevel())
{
SendMessage(HWND(pWindow->winId()), WM_SYSCOMMAND, SC_MOVE + HTCAPTION, 0);
}
}
event->ignore();
#else
#endif
} //使用事件过滤器监听标题栏所在的窗体,所以当窗体标题、图标等信息发生改变时,标题栏也应该随之改变
bool titleBar::eventFilter(QObject *obj, QEvent *event)
{
switch ( event->type() ) //判断发生事件的类型
{
case QEvent::WindowTitleChange: //窗口标题改变事件
{
QWidget *pWidget = qobject_cast<QWidget *>(obj); //获得发生事件的窗口对象
if (pWidget)
{
//窗体标题改变,则标题栏标题也随之改变
m_pTitleLabel->setText(pWidget->windowTitle());
return true;
}
}
break; case QEvent::WindowIconChange: //窗口图标改变事件
{
QWidget *pWidget = qobject_cast<QWidget *>(obj);
if (pWidget)
{
//窗体图标改变,则标题栏图标也随之改变
QIcon icon = pWidget->windowIcon();
m_pIconLabel->setPixmap(icon.pixmap(m_pIconLabel->size()));
return true;
}
}
break; case QEvent::Resize:
updateMaximize(); //最大化/还原
return true; default:
return QWidget::eventFilter(obj, event);
} return QWidget::eventFilter(obj, event);
} //进行最小化、最大化/还原、关闭操作
void titleBar::onClicked()
{
//QObject::Sender()返回发送信号的对象的指针,返回类型为QObject *
QPushButton *pButton = qobject_cast<QPushButton *>(sender()); QWidget *pWindow = this->window(); //获得标题栏所在的窗口 if (pWindow->isTopLevel())
{
//判断发送信号的对象使哪个按钮
if (pButton == m_pMinimizeButton)
{
pWindow->showMinimized(); //窗口最小化显示
}
else if (pButton == m_pMaximizeButton)
{
pWindow->isMaximized() ? pWindow->showNormal() : pWindow->showMaximized(); //窗口最大化/还原显示
}
else if (pButton == m_pCloseButton)
{
pWindow->close(); //窗口关闭
}
}
} //最大化/还原
void titleBar::updateMaximize()
{
QWidget *pWindow = this->window(); //获得标题栏所在的窗口 if (pWindow->isTopLevel())
{
bool bMaximize = pWindow->isMaximized(); //判断窗口是不是最大化状态,是则返回true,否则返回false
if (bMaximize)
{
//目前窗口是最大化状态,则最大化/还原的toolTip设置为"Restore"
m_pMaximizeButton->setToolTip(tr("Restore"));
//设置按钮的属性名为"maximizeProperty"
m_pMaximizeButton->setProperty("maximizeProperty", "restore");
}
else
{
//目前窗口是还原状态,则最大化/还原的toolTip设置为"Maximize"
m_pMaximizeButton->setToolTip(tr("Maximize"));
//设置按钮的属性名为"maximizeProperty"
m_pMaximizeButton->setProperty("maximizeProperty", "maximize");
} m_pMaximizeButton->setStyle(QApplication::style());
}
}

界面

widget.h

#ifndef WIDGET_H
#define WIDGET_H /*************************************************************************************
* 功能:实现自定义窗体的无边框与移动,以及自定义标题栏-用来显示窗体的图标、标题,以及控制窗体最小化、最大化/还原、关闭、缩放(仅支持windows平台)。
*************************************************************************************/ #include <QWidget>
#include <QMouseEvent> namespace Ui {
class Widget;
} class Widget : public QWidget
{
Q_OBJECT public:
explicit Widget(QWidget *parent = nullptr);
~Widget(); bool nativeEvent(const QByteArray &eventType, void *message, long *result); private:
Ui::Widget *ui; int m_nBorderWidth; //m_nBorder表示鼠标位于边框缩放范围的宽度
}; #endif // WIDGET_H

widget.cpp

#include "widget.h"
#include "ui_widget.h"
#include "titlebar.h" //包含“自定义标题栏”头文件
#include <QVBoxLayout> //调用WIN API需要用到的头文件与库 [实现缩放]
#ifdef Q_OS_WIN
#include <qt_windows.h>
#include <Windowsx.h>
#endif Widget::Widget(QWidget *parent) :
QWidget(parent),
ui(new Ui::Widget)
{
ui->setupUi(this); //Qt::FramelessWindowHint设置窗口标志为无边框,而Qt::WindowStaysOnTopHint使窗口位于所有界面之上
this->setWindowFlags(Qt::FramelessWindowHint | Qt::WindowStaysOnTopHint);
//背景透明
//setAttribute(Qt::WA_TranslucentBackground, true); //定义自定义标题栏对象
titleBar *pTitleBar = new titleBar(this);
installEventFilter(pTitleBar); resize(400, 300);
setWindowTitle("Custom Window"); //设置窗口名称,会发生窗口标题栏改变事件,随之自定义标题栏的标题会更新
setWindowIcon(QIcon("://Images/icon.png")); //设置窗口图标,会发生窗口图标改变事件,随之自定义标题栏的图标会更新 //使用调色板设置窗口的背景色
QPalette pal(palette());
pal.setColor(QPalette::Background, QColor(150, 150, 150));
setAutoFillBackground(true);
setPalette(pal); //窗口布局中加入标题栏
QVBoxLayout *pLayout = new QVBoxLayout();
pLayout->addWidget(pTitleBar);
pLayout->addStretch();
pLayout->setSpacing(0);
pLayout->setContentsMargins(0, 0, 0, 0);
setLayout(pLayout); //m_nBorder表示鼠标位于边框缩放范围的宽度,可以设置为5
m_nBorderWidth=5;
} Widget::~Widget()
{
delete ui;
} //nativeEvent主要用于进程间通信-消息传递,使用这种方式后来实现窗体的缩放 [加上了这函数,窗口也能移动了]
bool Widget::nativeEvent(const QByteArray &eventType, void *message, long *result)
{
Q_UNUSED(eventType) MSG *param = static_cast<MSG *>(message); switch (param->message)
{
case WM_NCHITTEST:
{
int nX = GET_X_LPARAM(param->lParam) - this->geometry().x();
int nY = GET_Y_LPARAM(param->lParam) - this->geometry().y(); // 如果鼠标位于子控件上,则不进行处理
if (childAt(nX, nY) != nullptr)
return QWidget::nativeEvent(eventType, message, result); *result = HTCAPTION; // 鼠标区域位于窗体边框,进行缩放
if ((nX > 0) && (nX < m_nBorderWidth))
*result = HTLEFT; if ((nX > this->width() - m_nBorderWidth) && (nX < this->width()))
*result = HTRIGHT; if ((nY > 0) && (nY < m_nBorderWidth))
*result = HTTOP; if ((nY > this->height() - m_nBorderWidth) && (nY < this->height()))
*result = HTBOTTOM; if ((nX > 0) && (nX < m_nBorderWidth) && (nY > 0)
&& (nY < m_nBorderWidth))
*result = HTTOPLEFT; if ((nX > this->width() - m_nBorderWidth) && (nX < this->width())
&& (nY > 0) && (nY < m_nBorderWidth))
*result = HTTOPRIGHT; if ((nX > 0) && (nX < m_nBorderWidth)
&& (nY > this->height() - m_nBorderWidth) && (nY < this->height()))
*result = HTBOTTOMLEFT; if ((nX > this->width() - m_nBorderWidth) && (nX < this->width())
&& (nY > this->height() - m_nBorderWidth) && (nY < this->height()))
*result = HTBOTTOMRIGHT; return true;
}
} return QWidget::nativeEvent(eventType, message, result);
}

参考

一去丶二三里

Qt之界面(自定义标题栏、无边框、可移动、缩放)的更多相关文章

  1. C#自定义Winform无边框窗体

    C#自定义Winform无边框窗体 在实际项目中,WinForm窗体或者控件不能满足要求,所以就需要自己设计窗体等,当然设计界面可以用的东西很多,例如WPF.或者一些第三方的库等.本例中将采用WinF ...

  2. 【Qt编程】基于Qt的词典开发系列<四>--无边框窗口的缩放与拖动

    在现在,绝大多数软件都向着简洁,时尚发展.就拿有道的单词本和我做的单词本来说,绝大多数用户肯定喜欢我所做的单词本(就单单界面,关于颜色搭配和布局问题,大家就不要在意了). 有道的单词本: 我所做的单词 ...

  3. C#自定义按钮、自定义WinForm无边框窗体、自定义MessageBox窗体

    C#自定义按钮.自定义WinForm无边框窗体.自定义MessageBox窗体 C#自定义Button按钮控件 效果展示 C#自定义Winform无边框窗体 效果展示 C#自定义无边框MessageB ...

  4. qt 5 小练习 创建无边框界面

    我们大家都知道QT5 自带的界面不是那么美观,并且每个软件我们都发现他们的边框是自定义的,所以我决定写一篇这样的博文,也许已经有许许多多篇大牛写的论文了,但我还是想写一篇记录自己的学习QT的历程 首先 ...

  5. Qt编写自定义控件67-通用无边框

    一.前言 在之前的一篇文章中写过一个通用的移动控件,作用就是用来传入任意的widget控件,可以在父类容器中自由移动.本篇文章要写的是一个通用的无边框类,确切的说这不叫控件应该叫组件才对,控件是要看得 ...

  6. 【Qt】Qt之自定义界面(实现无边框、可移动)【转】

    简述 UI设计是指对软件的人机交互.操作逻辑.界面美观的整体设计.好的UI设计不仅是让软件变得有个性.有品位,还要让软件的操作变得舒适简单.自由,充分体现软件的定位和特点. 爱美之心人皆有之.其实软件 ...

  7. Qt之自定义界面(实现无边框、可移动)

    简述 UI设计是指对软件的人机交互.操作逻辑.界面美观的整体设计.好的UI设计不仅是让软件变得有个性.有品位,还要让软件的操作变得舒适简单.自由,充分体现软件的定位和特点. 爱美之心人皆有之.其实软件 ...

  8. Qt:无标题栏无边框程序的拖动和改变大小

    From: http://blog.csdn.net/kfbyj/article/details/9284923 最近做项目遇到的问题,总结下. 有时候我们觉得系统的标题栏和按钮太丑太呆板,想做自己的 ...

  9. Qt无边框窗体-最大化时支持拖拽还原

    目录 一.概述 二.效果展示 三.demo制作 1.设计窗体 2.双击放大 四.拖拽 五.相关文章 原文链接:Markdown模板 一.概述 用Qt进行开发界面时,既想要实现友好的用户交互又想界面漂亮 ...

  10. Qt无边框窗体-模拟模态窗体抖动效果

    目录 一.概述 二.效果展示 三.功能实现 四.相关文章 原文链接:Qt无边框窗体-模拟模态窗体抖动效果 一.概述 用Qt开发windows客户端界面确实是一大利器,兼顾性能的同时,速度相对来说也不错 ...

随机推荐

  1. http 连接失败重连机制

    1.我们做web开发时,需要经常使用httpclient来请求http服务,有时为了安全起见,服务提供方会提供多个http地址,这样如果我们请求某个ip出现异常,可以重试其他的ip地址,来尽量保证系统 ...

  2. kvm虚拟化学习笔记(一)之kvm虚拟化环境安装

    平时一直玩RHEL/CentOS/OEL系列的操作,玩虚拟化也是采这一类系统,kvm在RHEL6系列操作系统支持比较好,本文采用采用OEL6.3操作系统,网上所有文章都说KVM比xen简单,我怎么感觉 ...

  3. C#趣味程序---百鸡百钱

    问题:公鸡一仅仅5元,母鸡一仅仅3元,小鸡三仅仅1元.问100元能够买多少仅仅鸡? using System; namespace ConsoleApplication1 { class Progra ...

  4. Jenkins系列之-—04 配置用户和权限控制

    一.安装插件 插件名称:Role-based Authorization Strategy Role Strategy Plugin插件可以对构建的项目进行授权管理,让不同的用户管理不同的项目. 二. ...

  5. 《Java设计模式》之訪问者模式

    訪问者模式是对象的行为模式.訪问者模式的目的是封装一些施加于某种数据结构元素之上的操作.一旦这些操作须要改动的话,接受这个操作的数据结构则能够保持不变. 分派的概念 变量被声明时的类型叫做变量的静态类 ...

  6. Android碎纸机效果

    1.总体思想 活用padding和margin 2.实现过程 public class PopupShredderView extends FrameLayout{ public PopupShred ...

  7. (原创)EasyUI中datagrid的行编辑模式中,找到特定的Editor,并为其添加事件

    有时候在行编辑的时候,一个编辑框的值要根据其它编辑框的值进行变化,那么可以通过在开启编辑时,找到特定的Editor,为其添加事件 // 绑定事件, index为当前编辑行 var editors = ...

  8. hadoop第一个例子WordCount

    hadoop查看自己空间 http://127.0.0.1:50070/dfshealth.jsp import java.io.IOException; import java.util.Strin ...

  9. Java 解析excel2003和2007区别和兼容性问题(POI操作)

    最近在使用POI对excel操作中发现一些问题,2003和2007的区别还是蛮大的: 2007相关的包: poi-3.9.jar poi-examples-3.8.jar poi-excelant-3 ...

  10. #啃underscore源码 一、root对象初始化部分

    最近由于比赛要交了,以及工作室屯了各种项目,实在忙不过来刷题,所以很久没更blog了(良心痛),现在自己的水平还是渣代码堆砌 + 简单的增删改查(悲伤) 所以痛定思痛,决定之后的任务是先补学校课堂的知 ...