在这里自定义窗口是通过继承于QFrame,内部分为上下两部分,上半部分就是标题栏,下面是窗口的具体内容。上下两部分通过布局固定位置。最后窗口的各种鼠标拖动,缩放,等操作通过添加鼠标事件来完成一个窗口的常规功能。当然,这个标题栏可以通过布局的改变把它放到任意地方。

下面是实现的代码: 
#pragma once 
#ifndef CUSTOMWIDGET_H_ 
#define CUSTOMWIDGET_H_

#include <QtWidgets/QFrame>

class QToolButton; 
class CustomWidget : public QFrame 
{

Q_OBJECT

public:

explicit CustomWidget();
~CustomWidget();
void setContentWidget(QWidget *contentWidget);
void setTitleInfo(const QString &titleString, const QPixmap &windowIcon);

public slots:

void slotShowSmall();
void slotShowMaxRestore();
void onMoveWindow();

signals:

void moveWindow();

protected:

void mousePressEvent(QMouseEvent *);
void mouseMoveEvent(QMouseEvent *);
void mouseReleaseEvent(QMouseEvent *);
bool nativeEvent(const QByteArray & eventType, void * message, long * result);
void paintEvent(QPaintEvent *);

private:

bool isMax;
bool isPress;
QPoint startPos;
QPoint clickPos;
QWidget *m_pTitleWidget;//顶栏窗体
QWidget *m_pContentWidget;//内容窗体
QPixmap maxPixmap;
QPixmap restorePixmap;
QToolButton *m_pMaxButton;

};

#endif //!CUSTOMWIDGET_H_

customWidget.cpp:

#include "stdafx.h" 
#include "customwidget.h" 
#include <QtWidgets/QLabel> 
#include <QtWidgets/QToolButton> 
#include <QtWidgets/QHBoxLayout> 
#include <QtCore/QEvent> 
#include <Window.h>

static const int TITLE_HEIGHT = 40;//标题栏高度 
static const int FRAME_BORDER = 2;//边框宽度 
CustomWidget::CustomWidget() 
: m_pContentWidget(0) 
//, m_llLocalWinId(0) 
//, m_llRemoteWinId(0)
 
{

this->setWindowFlags(Qt::FramelessWindowHint);
this->setAttribute(Qt::WA_TranslucentBackground);
this->setMouseTracking(true); isMax = false;
isPress = false;
m_pTitleWidget = new QWidget();

}

CustomWidget::~CustomWidget() 

}

void CustomWidget::setContentWidget(QWidget *contentWidget) 
{

if (contentWidget)
{
m_pContentWidget = contentWidget;
m_pContentWidget->move(m_pTitleWidget->rect().bottomLeft());
m_pContentWidget->show();
resize(m_pTitleWidget->width(), m_pTitleWidget->height() + m_pContentWidget->height());
}

}

void CustomWidget::setTitleInfo(const QString &titleString, const QPixmap &windowIcon) 
{

QLabel *logoLabel = new QLabel();
logoLabel->setPixmap(windowIcon);
logoLabel->setFixedSize(16, 16);
logoLabel->setScaledContents(true); QLabel *titleLabel = new QLabel();
titleLabel->setText(titleString);
QFont titleFont = titleLabel->font();
titleFont.setBold(true);
titleLabel->setFont(titleFont);
titleLabel->setObjectName("whiteLabel");
titleLabel->setStyleSheet("color:white;"); QToolButton *minButton = new QToolButton();
connect(minButton, SIGNAL(clicked()), this, SLOT(slotShowSmall()));
m_pMaxButton = new QToolButton();
connect(m_pMaxButton, SIGNAL(clicked()), this, SLOT(slotShowMaxRestore()));
QToolButton *closeButton = new QToolButton();
connect(closeButton, SIGNAL(clicked()), this, SLOT(close()));
connect(this, SIGNAL(moveWindow()), this, SLOT(onMoveWindow())); QHBoxLayout *titleLayout = new QHBoxLayout();
titleLayout->addWidget(logoLabel);
titleLayout->addWidget(titleLabel);
titleLabel->setContentsMargins(5, 0, 0, 0);
titleLayout->addStretch();
titleLayout->addWidget(minButton, 0, Qt::AlignTop);
titleLayout->addWidget(m_pMaxButton, 0, Qt::AlignTop);
titleLayout->addWidget(closeButton, 0, Qt::AlignTop);
titleLayout->setSpacing(0);
titleLayout->setContentsMargins(5, 0, 0, 0); m_pTitleWidget->setFixedSize(600, TITLE_HEIGHT);
m_pTitleWidget->setLayout(titleLayout);
m_pTitleWidget->installEventFilter(0);
m_pTitleWidget->setAutoFillBackground(true);
QPalette palette;
palette.setColor(QPalette::Background, QColor(31,25,25));
m_pTitleWidget->setPalette(palette);

}

void CustomWidget::slotShowSmall()//最小化 

this->showMinimized(); 
}

void CustomWidget::slotShowMaxRestore()//还原按钮 
{

if (isMax) {
this->showNormal();
m_pMaxButton->setIcon(maxPixmap);
}
else {
this->showMaximized();
m_pMaxButton->setIcon(restorePixmap);
}
isMax = !isMax;

}

void CustomWidget::onMoveWindow() 

//m_pContentWidget->move(m_pTitleWidget->rect().bottomLeft()); 
}

void CustomWidget::mousePressEvent(QMouseEvent *e) 
{

startPos = e->globalPos();
clickPos = e->pos(); if (e->button() == Qt::LeftButton) {
if (e->type() == QEvent::MouseButtonPress) {
isPress = true;
}
else if (e->type() == QEvent::MouseButtonDblClick && e->pos().y() <= TITLE_HEIGHT) {
this->slotShowMaxRestore();
}
}

}

void CustomWidget::mouseMoveEvent(QMouseEvent *e) 
{

if (isMax || !isPress) {
return;
}
this->move(e->globalPos() - clickPos);
emit moveWindow();

}

void CustomWidget::mouseReleaseEvent(QMouseEvent *) 

isPress = false; 
}

bool CustomWidget::nativeEvent(const QByteArray & eventType, void * message, long * result)//实现拖拽边框的效果 
{

Q_UNUSED(eventType);
const int HIT_BORDER = 5;
const MSG *msg = static_cast<MSG*>(message);
if (msg->message == WM_NCHITTEST) {
int xPos = ((int)(short)LOWORD(msg->lParam)) - this->frameGeometry().x();
int yPos = ((int)(short)HIWORD(msg->lParam)) - this->frameGeometry().y();
if (this->childAt(xPos, yPos) == 0) {
*result = HTCAPTION;
}
else {
return false;
}
if (xPos > 0 && xPos < HIT_BORDER) {
*result = HTLEFT;
}
if (xPos >(this->width() - HIT_BORDER) && xPos < (this->width() - 0)) {
*result = HTRIGHT;
}
if (yPos > 0 && yPos < HIT_BORDER) {
*result = HTTOP;
}
if (yPos >(this->height() - HIT_BORDER) && yPos < (this->height() - 0)) {
*result = HTBOTTOM;
}
if (xPos > 0 && xPos < HIT_BORDER && yPos > 0 && yPos < HIT_BORDER) {
*result = HTTOPLEFT;
}
if (xPos >(this->width() - HIT_BORDER) && xPos < (this->width() - 0) && yPos > 0 && yPos < HIT_BORDER) {
*result = HTTOPRIGHT;
}
if (xPos > 0 && xPos < HIT_BORDER && yPos >(this->height() - HIT_BORDER) && yPos < (this->height() - 0)) {
*result = HTBOTTOMLEFT;
}
if (xPos >(this->width() - HIT_BORDER) && xPos < (this->width() - 0) && yPos >(this->height() - HIT_BORDER) && yPos < (this->height() - 0)) {
*result = HTBOTTOMRIGHT;
}
return true;
}
return false;

}

void CustomWidget::paintEvent(QPaintEvent *e) 
{//绘制边框

int border = FRAME_BORDER;
if (this->isMaximized()) {
border = 0;
} QPainter painter(this);
QPainterPath painterPath;
painterPath.setFillRule(Qt::WindingFill);
painterPath.addRect(border, border, this->width() - 2 * border, this->height() - 2 * border);
painter.setRenderHint(QPainter::Antialiasing, true);
painter.fillPath(painterPath, QBrush(Qt::black));
QColor color(200, 200, 200);
for (int i = 0; i<border; i++) {
color.setAlpha((i + 1) * 30);
painter.setPen(color);
painter.drawRect(border - i, border - i, this->width() - (border - i) * 2, this->height() - (border - i) * 2);
} painter.setPen(Qt::NoPen);
painter.setBrush(Qt::black);
// 这里可以在资源中指定一张标题背景图片
//painter.drawPixmap(QRect(border, border, this->width()-2*border, this->height()-2*border), QPixmap(DEFAULT_SKIN));
painter.drawRect(QRect(border, TITLE_HEIGHT, this->width() - 2 * border, this->height() - TITLE_HEIGHT - border));
QFrame::paintEvent(e);


///////////////全文完

http://blog.csdn.net/u011915578/article/details/46924151

QT自定义窗口(模拟MainWindow)的更多相关文章

  1. QT自定义窗口

    qt 中允许自定义窗口控件,使之满足特殊要求, (1)可以修改其显示,自行绘制 (2)可以动态显示 (3)可以添加事件,支持鼠标和键盘操作 自定义控件可以直接在QtDesigner里使用,可以直接加到 ...

  2. Qt自定义窗口部件

    QtDesigner自定义窗口部件有两种方法:改进法(promotion)和插件法(plugin)   改进法   1.改进法之前,要先写好子类化QSpinBox后的HexspinBox.h和Hexs ...

  3. qt 自定义窗口绘制正弦曲线

    circlewidget.h #ifndef CIRCLAWIDGET_H #define CIRCLAWIDGET_H #include <QFrame> #include<QVe ...

  4. qt 自定义窗口显示鼠标划过的轨迹

    鼠标事件分为四种: 1.按下 2.抬起 3.移动 4.双击 鼠标事件继承与QWidget void mouseDoubleClickEvent(QMouseEvent *event) void mou ...

  5. QT笔记之自定义窗口拖拽移动

    1.QT自定义标题栏,拖拽标题栏移动窗口(只能拖拽标题,其他位置无法拖拽) 方法一: 转载:http://blog.sina.com.cn/s/blog_4ba5b45e0102e83h.html . ...

  6. Qt入门(19)——自定义窗口部件

    我们介绍可以画自己的第一个自定义窗口部件.我们也加入了一个有用的键盘接口.我们添加了一个槽:setRange().        void setRange( int minVal, int maxV ...

  7. WPF 使用 WindowChrome,在自定义窗口标题栏的同时最大程度保留原生窗口样式(类似 UWP/Chrome)

    WPF 自定义窗口样式有多种方式,不过基本核心实现都是在修改 Win32 窗口样式.然而,Windows 上的应用就应该有 Windows 应用的样子嘛,在保证自定义的同时也能与其他窗口样式保持一致当 ...

  8. QT5每日一学(二)编写QT多窗口程序

    一.添加主窗口 1.首先打开Qt Creator,新建Qt Widgets Application,项目名称设置为windows,在类信息界面保持基类为QMainWindow.类名为MainWindo ...

  9. WPF.UIShell UIFramework之自定义窗口的深度技术 - 模态闪动(Blink)、窗口四边拖拽支持(WmNCHitTest)、自定义最大化位置和大小(WmGetMinMaxInfo)

    无论是在工作和学习中使用WPF时,我们通常都会接触到CustomControl,今天我们就CustomWindow之后的一些边角技术进行探讨和剖析. 窗口(对话框)模态闪动(Blink) 自定义窗口的 ...

随机推荐

  1. 《Programming WPF》翻译 第7章 5.可视化层编程

    原文:<Programming WPF>翻译 第7章 5.可视化层编程 形状元素能提供一种便利的方式与图形一起工作,在一些情形中,添加表示绘图的元素到UI树中,可能是比它的价值更加麻烦.你 ...

  2. [置顶] Responder一点也不神秘————iOS用户响应者链完全剖析

    这篇文章想跟大家分享的主旨是iOS捕获用户事件的各种情况,以及内部封装的一些特殊事件. 我们先从UIButton谈起,UIButton大家使用的太多了,他特殊的地方就在于其内置的普通Default/高 ...

  3. 导入旧数据需要 使用date插件

    "@version" => "1", "@timestamp" => "2016-09-12T08:31:06.630 ...

  4. Buffer lock

    buffer lock    Oracle 提供非常精确,有效的Row Level Lock机制,多个用户同时修改数据时,为了保护数据. 以块为单位挂起锁的情况不会发生,但这不太正确. 以块为单位的锁 ...

  5. Zigzag Iterator 解答

    Question Given two 1d vectors, implement an iterator to return their elements alternately. For examp ...

  6. OpenStack术语名词及问题调试

    aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAnoAAAEkCAIAAAA3pAtBAAAgAElEQVR4nOy953tUx7o9uCrt2EmtHB

  7. 一个跨域请求的XSS漏洞

    场景回顾 一个表单进行跨域提交的方式有很多,我们使用的采用隐藏iframe,在本域下放一个代理页面,通过服务端配合完成一次完整的请求. 首先,部署proxy.html代理页面.这个页面处理服务端返回的 ...

  8. 【转】ffmpeg中的sws_scale算法性能测试

    经常用到ffmpeg中的sws_scale来进行图像缩放和格式转换,该函数可以使用各种不同算法来对图像进行处理.以前一直很懒,懒得测试和甄别应该使用哪种算法,最近的工作时间,很多时候需要等待别人.忙里 ...

  9. QT 4.7支持中文(QT4.7)(中文)(makeqpf)

    QT 4.7支持中文(QT4.7)(中文)(makeqpf) 摘要: QT4.7.0在移植到开发板上的时候,中文支持是必不可少的,如何让QT支持中文,如何制作QT支持的字体文件,如何使QT UI编辑器 ...

  10. hdu 5256 序列变换(LIS最长上升子序列)

    Problem Description 我们有一个数列A1,A2...An,你现在要求修改数量最少的元素,使得这个数列严格递增.其中无论是修改前还是修改后,每个元素都必须是整数. 请输出最少需要修改多 ...