网站建设网页制作软件有哪些/制作一个网站的基本步骤
在QT中如果想要自绘标题和边框,一般步骤是:
1) 在创建窗口前设置Qt::FramelessWindowHint标志,设置该标志后会创建一个无标题、无边框的窗口。
2)在客户区域的顶部创建一个自绘标题栏。
3)给窗口绘制一个背景作为边框。
4)如果想要鼠标拖动效果,可以在WM_NCHITTEST消息中返回HTCAPTION,具体方法百度这里不再详述。
但是这样做会导致一个问题:
在win7系统上,将窗口移动到屏幕边缘会自动排列(在屏幕顶部,左边,右边都会自动排列)的功能失效。
如果你的窗口没有这个功能,只有两种可能:
1)你的窗口不支持"移动到屏幕边缘自动排列"功能。
2)你从系统关闭了此项功能(控制面板\轻松访问\轻松访问中心\使任务更容易被关注\防止将窗口移动到屏幕边缘时自动排列窗口)。
怎么样才能够既能够自绘标题和边框,又能够使用屏幕自动排列功能:
有一个windows消息能够帮助我们,响应WM_NCCALCSIZE消息,直接返回true,就可以使客户区域的大小和窗口大小完全一样,这样就没有了标题栏和边框,我们可以按照上面的一般步骤来自绘标题栏和边框,唯一不同的是不需要设置Qt::FramelessWindowHint标志。
这样做也会有问题:
窗口显示的不完整,特别是在最大化的时候,非常明显。
为什么会显示不完整,这个问题困扰我一整天。我新建了一个win32项目,响应WM_NCCALCSIZE消息,窗口显示完整,应该是QT自己处理的问题,最后不断调试QT源码,终于明白问题所在:
调用堆栈(从下往上看):
QWindowsWindow::frameMarginsDp() 行 1854 C++
QWindowsWindow::frameMargins() 行 188 C++
QWidgetPrivate::updateFrameStrut() 行 11824 C++
QWidget::create(unsigned int window, bool initializeWindow, bool destroyOldWindow) 行 1358 C++
关键函数:
QMargins QWindowsWindow::frameMarginsDp() const
{// Frames are invalidated by style changes (window state, flags).// As they are also required for geometry calculations in resize// event sequences, introduce a dirty flag mechanism to be able// to cache results.if (testFlag(FrameDirty)) {// Always skip calculating style-dependent margins for windows claimed to be frameless.// This allows users to remove the margins by handling WM_NCCALCSIZE with WS_THICKFRAME set// to ensure Areo snap still works (QTBUG-40578).m_data.frame = window()->flags() & Qt::FramelessWindowHint? QMargins(0, 0, 0, 0): QWindowsGeometryHint::frame(style(), exStyle());clearFlag(FrameDirty);}return m_data.frame + m_data.customMargins;
}
注释里面清楚说明这是一个BUG(QTBUG-40578),我们虽然已经让客户区域大小和窗口大小完全一样,但是QT还是认为系统有边框,只有当设置了Qt::FramelessWindowHint标志,才会返回QMargins(0, 0, 0, 0)。
现在又回到了原点,且问题相互矛盾,想要自绘标题和边框必须设置Qt::FramelessWindowHint标志,但是设置Qt::FramelessWindowHint标志后"屏幕边缘自动排列"无效。
首先要搞清楚Qt::FramelessWindowHint标志如何影响窗口,因为它直接导致"屏幕边缘自动排列"无效:
WindowCreationData::fromWindow(const QWindow * w, const QFlags<enum Qt::WindowType> flagsIn, unsigned int creationFlags) 行 519 C++
QWindowsWindowData::create(const QWindow * w, const QWindowsWindowData & parameters, const QString & title) 行 1075 C++
QWindowsIntegration::createWindowData(QWindow * window) 行 316 C++
QWindowsIntegration::createPlatformWindow(QWindow * window) 行 340 C++
QWindowPrivate::create(bool recursive) 行 392 C++
QWindow::create() 行 549 C++
QWidgetPrivate::create_sys(unsigned int window, bool initializeWindow, bool destroyOldWindow) 行 1456 C++
QWidget::create(unsigned int window, bool initializeWindow, bool destroyOldWindow) 行 1321 C++
QWidgetPrivate::createWinId(unsigned int winid) 行 2528 C++
void WindowCreationData::fromWindow(const QWindow *w, const Qt::WindowFlags flagsIn,unsigned creationFlags)
{if (popup || (type == Qt::ToolTip) || (type == Qt::SplashScreen)) {style = WS_POPUP;} else if (topLevel && !desktop) {if (flags & Qt::FramelessWindowHint)style = WS_POPUP; // no borderelse if (flags & Qt::WindowTitleHint)style = WS_OVERLAPPED;elsestyle = 0;} else {style = WS_CHILD;}if (!desktop) {if (topLevel) {if ((type == Qt::Window || dialog || tool)) {if (!(flags & Qt::FramelessWindowHint)) {style |= WS_POPUP;if (flags & Qt::MSWindowsFixedSizeDialogHint) {style |= WS_DLGFRAME;} else {style |= WS_THICKFRAME;}if (flags & Qt::WindowTitleHint)style |= WS_CAPTION; // Contains WS_DLGFRAME}} else {exStyle |= WS_EX_TOOLWINDOW;}}}
}
上面一个是调用堆栈(从下往上看),一个是关键函数(函数中不重要的内容已经删除)。从代码中可以看出,设置Qt::FramelessWindowHint标志会改变窗口样式,从而影响创建的窗口,现在基本已经知道,"屏幕边缘自动排列"功能与窗口样式有关。
新建一个win32窗口程序,不断改变窗口的样式,最后得出结论:只有在窗口拥有WS_MAXIMIZEBOX | WS_THICKFRAME样式时,"屏幕边缘自动排列"功能才有效,最好还要添加WS_CAPTION样式,否则窗口最大化会覆盖任务栏。
原本以为完美结束了,但是不要高兴的太早,经过不断测试,还有几个问题:
1)在任务栏点击窗口时,不能最小化:
只要加上Qt::WindowMinimizeButtonHint标志即可解决该问题。
2)如果有多个显示器,在辅屏上直接显示最大化,窗口显示不完整:
QWindowsWindow::show_sys() 行 1230 C++
QWindowsWindow::setVisible(bool visible) 行 1092 C++
QWindow::setVisible(bool visible) 行 518 C++
QWidgetPrivate::show_sys() 行 7897 C++
QWidgetPrivate::show_helper() 行 7826 C++
QWidget::setVisible(bool visible) 行 8110 C++
QWidget::showMaximized() 行 3154 C++
void QWindowsWindow::show_sys() const
{int sm = SW_SHOWNORMAL;bool fakedMaximize = false;const QWindow *w = window();const Qt::WindowFlags flags = w->flags();const Qt::WindowType type = w->type();if (w->isTopLevel()) {const Qt::WindowState state = w->windowState();if (state & Qt::WindowMinimized) {sm = SW_SHOWMINIMIZED;if (!isVisible())sm = SW_SHOWMINNOACTIVE;} else {updateTransientParent();if (state & Qt::WindowMaximized) {sm = SW_SHOWMAXIMIZED;// Windows will not behave correctly when we try to maximize a window which does not// have minimize nor maximize buttons in the window frame. Windows would then ignore// non-available geometry, and rather maximize the widget to the full screen, minus the// window frame (caption). So, we do a trick here, by adding a maximize button before// maximizing the widget, and then remove the maximize button afterwards.if (flags & Qt::WindowTitleHint &&!(flags & (Qt::WindowMinMaxButtonsHint | Qt::FramelessWindowHint))) {fakedMaximize = TRUE;setStyle(style() | WS_MAXIMIZEBOX);}} // Qt::WindowMaximized} // !Qt::WindowMinimized}if (type == Qt::Popup || type == Qt::ToolTip || type == Qt::Tool || testShowWithoutActivating(w))sm = SW_SHOWNOACTIVATE;if (w->windowState() & Qt::WindowMaximized)setFlag(WithinMaximize); // QTBUG-8361ShowWindow(m_data.hwnd, sm);clearFlag(WithinMaximize);if (fakedMaximize) {setStyle(style() & ~WS_MAXIMIZEBOX);SetWindowPos(m_data.hwnd, 0, 0, 0, 0, 0,SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_NOOWNERZORDER| SWP_FRAMECHANGED);}
}
还是老样子,上面一个是调用堆栈,一个是关键函数,我们可以看到最后QT调用了ShowWindow函数来显示最大化窗口,但是为什么会显示不完整呢?
通常遇到一个复杂的问题,我会新建一个简单的项目来做实验。新建一个win32项目,最开始显示就让它最大化,结果显示正常,证明还是QT自己处理的问题,应该是在ShowWindow之后进行其他的处理,导致窗口显示不完整,最后发现是
处理WM_GETMINMAXINFO消息导致的,接下来我们看看QT如何处理WM_GETMINMAXINFO消息。
QWindowsWindow::getSizeHints(tagMINMAXINFO * mmi) 行 2044 C++
QWindowsContext::windowsProc 行 1015 C++
qWindowsWndProc(HWND__ * hwnd, unsigned int message, unsigned int wParam, long lParam) 行 1271 C++
void QWindowsWindow::getSizeHints(MINMAXINFO *mmi) const
{const QWindowsGeometryHint hint(window(), m_data.customMargins);hint.applyToMinMaxInfo(m_data.hwnd, mmi);if ((testFlag(WithinMaximize) || (window()->windowState() == Qt::WindowMinimized))&& (m_data.flags & Qt::FramelessWindowHint)) {// This block fixes QTBUG-8361: Frameless windows shouldn't cover the// taskbar when maximizedconst QScreen *screen = window()->screen();// Documentation of MINMAXINFO states that it will only work for the primary screenif (screen && screen == QGuiApplication::primaryScreen()) {mmi->ptMaxSize.y = screen->availableGeometry().height();// Width, because you can have the taskbar on the sides too.mmi->ptMaxSize.x = screen->availableGeometry().width();// If you have the taskbar on top, or on the left you don't want it at (0,0):mmi->ptMaxPosition.x = screen->availableGeometry().x();mmi->ptMaxPosition.y = screen->availableGeometry().y();} else if (!screen){qWarning() << "window()->screen() returned a null screen";}}qCDebug(lcQpaWindows) << __FUNCTION__ << window() << *mmi;
}
当程序在辅屏上时,它的screen是辅屏,如果当前screen不等于QGuiApplication::primaryScreen(主屏),则不设置MINMAXINFO结构,但是由于它已经处理了WM_GETMINMAXINFO消息,导致这个消息不会被系统默认的窗口处理函数处理(DefWindowProc),所以才会显示不完整,解决办法是优先响应WM_GETMINMAXINFO消息,让后交给系统默认的窗口处理函数进行处理。
3)最大化后,窗口内容变小,最明显的就是最小化、最大化、关闭按钮变小了:
窗口最大化时,系统会在屏幕上面显示所有的客户区域,此时系统会计算边框的大小,然后超出屏幕范围进行显示,例如边框的宽为8高为8,则系统会在(-8,-8,宽度+8,高度+8)的位置显示窗口,给人的感觉窗口的内容变小了,
除去底部的任务栏,程序最大化可显示的最大宽度是1600*860,而窗口的实际位置是(-8,-8,1608,868)。这样我们可以添加一个QWidget作为主显示窗口,然后在程序最大化时,添加一个外边框,让它向内部缩一点。
最后的解决方案是:
1. 在窗口的构造函数中添加以下代码,改变窗口的样式:
this->setWindowFlags(Qt::FramelessWindowHint | Qt::WindowMinimizeButtonHint);
// QMainWindow透明显示,当设置主显示窗口的外边距时,防止外边距显示出来。
this->setAttribute(Qt::WA_TranslucentBackground, true);HWND hwnd = (HWND)this->winId();
DWORD style = ::GetWindowLong(hwnd, GWL_STYLE);
::SetWindowLong(hwnd, GWL_STYLE, style | WS_MAXIMIZEBOX | WS_THICKFRAME | WS_CAPTION);
2. 重载nativeEvent函数,处理WM_NCHITTEST、WM_NCCALCSIZE和WM_GETMINMAXINFO消息
bool CustomWindow::nativeEvent(const QByteArray &eventType, void *message, long *result)
{MSG* msg = (MSG*)message;switch (msg->message) {case WM_NCHITTEST:{int xPos = GET_X_LPARAM(msg->lParam) - this->frameGeometry().x();int yPos = GET_Y_LPARAM(msg->lParam) - this->frameGeometry().y();if (m_title->isCaption(xPos, yPos)) {*result = HTCAPTION;return true;}}break;
case WM_NCCALCSIZE:return true;case WM_GETMINMAXINFO:{if (::IsZoomed(msg->hwnd)) {// 最大化时会超出屏幕,所以填充边框间距RECT frame = { 0, 0, 0, 0 };AdjustWindowRectEx(&frame, WS_OVERLAPPEDWINDOW, FALSE, 0);frame.left = abs(frame.left);frame.top = abs(frame.bottom);this->setContentsMargins(frame.left, frame.top, frame.right, frame.bottom);}else {this->setContentsMargins(2, 2, 2, 2);}*result = ::DefWindowProc(msg->hwnd, msg->message, msg->wParam, msg->lParam);return true;}break;}return QMainWindow::nativeEvent(eventType, message, result);
}
显示效果:
相关文章:

QT自绘标题和边框
在QT中如果想要自绘标题和边框,一般步骤是: 1) 在创建窗口前设置Qt::FramelessWindowHint标志,设置该标志后会创建一个无标题、无边框的窗口。 2)在客户区域的顶部创建一个自绘标题栏。 3)给窗口绘制一个背…...

数据库浅谈之 LLVM
数据库浅谈之 LLVM HELLO,各位博友好,我是阿呆 🙈🙈🙈 这里是数据库浅谈系列,收录在专栏 DATABASE 中 😜😜😜 本系列阿呆将记录一些数据库领域相关的知识 Ἴ…...

Unable to connect to Redis无法连接到Redis
文章目录项目场景:问题描述原因分析:解决方案:项目场景: 提示:这里简述项目相关背景: 在某个项目中的提交按钮不好用 org.springframework.data.redis.RedisConnectionFailureException: Unable to con…...

Feign、Ribbon、Hystrix
🏆今日学习目标: 🍀Feign、Ribbon、Hystrix ✅创作者:林在闪闪发光 ⏰预计时间:30分钟 🎉个人主页:林在闪闪发光的个人主页 🍁林在闪闪发光的个人社区,欢迎你的加入: 林在…...

SpringCloud - Nacos注册发现
目录 服务注册到Nacos 服务分级存储模型 NacosRule负载均衡 服务实例的权重设置 环境隔离 Nacos与Eureka的对比 添加Nacos配置 微服务配置拉取 配置热更新 多环境配置共享 服务注册到Nacos 1.在父工程引入SpringCloudAlibaba的依赖 2.注释掉order-service和user-ser…...

Socket编程、协议理解
Socket编程、协议理解简单说明Socket编程Socket 常用接口Socket服务端业务编码代码说明文件服务(fileServe)消息服务(msgServe)消息写会(writeResponse)注意事项Socket客户端业务编码代码说明总结RYP协议:基于TCP/IP协议之上的自定义协议Rocky Protocol。 协议用途&…...

Idea集成码云
1:Idea集成码云1.1:IDEA安装码云插件【第一步】Idea 默认不带码云插件, 我们第一步要安装 Gitee 插件。如图所示, 在 Idea 插件商店搜索 Gitee,然后点击右侧的 Install 按钮。安装成功后,重启 Idea。Idea 重…...

并发编程学习篇ReentrantLock设计思想剖析
一、AQS原理剖析 什么是AQS java.util.concurrent包中的大多数同步器实现都是围绕着共同的基础行为,比如 等待队列、条件队列、独占获取、共享获取等而这些行为的抽象就是基于AbstractQueuedSynchronizer(简称AQS)实现的,AQS是一…...

区分效度全流程分析
基本说明 区分效度(又称判别效度、区别效度),其实质也是一种结构效度。区分效度强调本不应该在同一因子的测量项,确实不在同一因子下面。比如说,测量项A和 B分别测量两个属性,应该分属于因子A和因子B中&…...

【华为OD机试模拟题】用 C++ 实现 - 找数字(2023.Q1)
最近更新的博客 华为OD机试 - 入栈出栈(C++) | 附带编码思路 【2023】 华为OD机试 - 箱子之形摆放(C++) | 附带编码思路 【2023】 华为OD机试 - 简易内存池 2(C++) | 附带编码思路 【2023】 华为OD机试 - 第 N 个排列(C++) | 附带编码思路 【2023】 华为OD机试 - 考古…...

从0开始写Vue项目-Vue实现用户数据批量上传和数据导出
从0开始写Vue项目-环境和项目搭建_慕言要努力的博客-CSDN博客从0开始写Vue项目-Vue2集成Element-ui和后台主体框架搭建_慕言要努力的博客-CSDN博客从0开始写Vue项目-Vue页面主体布局和登录、注册页面_慕言要努力的博客-CSDN博客从0开始写Vue项目-SpringBoot整合Mybatis-plus实现…...

企业容器云管理平台选型指南
作者简介 涂家英,SUSE 资深架构师,专注 Cloud-Native 相关产品和解决方案设计,在企业级云原生平台建设领域拥有丰富的经验。 数字时代下的容器云管理平台 数字时代,市场竞争加剧,业务需求日新月异,敏态 IT…...

OpenGL超级宝典学习笔记:着色器存储区块、原子内存操作、内存屏障
前言 本篇在讲什么 本篇为蓝宝书学习笔记 着色器存储区块 原子内存操作 内存屏障 本篇适合什么 适合初学Open的小白 本篇需要什么 对C语法有简单认知 对OpenGL有简单认知 最好是有OpenGL超级宝典蓝宝书 依赖Visual Studio编辑器 本篇的特色 具有全流程的图文教学 重…...

SpringMVC框架知识详解(入门版)
✅作者简介:2022年博客新星 第八。热爱国学的Java后端开发者,修心和技术同步精进。 🍎个人主页:Java Fans的博客 🍊个人信条:不迁怒,不贰过。小知识,大智慧。 💞当前专栏…...

25-动画和过渡
动画和过渡 一、动画 使用css动画样式,配合vue实现动画效果。 编写模板 <template><div><button click"isShow !isShow">显示/隐藏</button><h1 v-show"isShow">你好啊</h1></div> </templa…...

Linux 操作系统原理 — 虚拟内存管理
目录 文章目录 目录虚拟内存技术页式内存管理技术x86_32 CPU 虚拟内存虚拟地址格式与内核页表虚拟内存空间Kernel SpaceUser Spacex86_64 CPU 虚拟内存虚拟地址格式与内核页表(四级页表)虚拟内存空间TLB 缓冲(快表)进程页表虚拟内存技术 虚拟内存技术是操作系统实现的一种…...

保持超低温环境新方法:功耗降至十分之一!
(图片来源:网络)量子比特是量子计算机的主要构建部分,然而热量会导致量子比特容易出错,因此量子系统通常保存在超低温稀释制冷机内,可以将温度保持在绝对零度(−273.15℃)以上。但是…...

论文投稿指南——中文核心期刊推荐(音乐)
【前言】 🚀 想发论文怎么办?手把手教你论文如何投稿!那么,首先要搞懂投稿目标——论文期刊 🎄 在期刊论文的分布中,存在一种普遍现象:即对于某一特定的学科或专业来说,少数期刊所含…...

es-10搜索推荐suggest
搜索推荐:Suggest 概述 搜索一般都会要求具有“搜索推荐”或者叫“搜索补全”的功能,即在用户输入搜索的过程中,进行自动补全或者纠错。以此来提高搜索文档的匹配精准度,进而提升用户的搜索体验,这就是Suggest。 四…...

VMware ESXi 7.0 Update 3k - 领先的裸机 Hypervisor (sysin Custom Image)
VMware ESXi 7.0 Update 3k - 领先的裸机 Hypervisor (sysin Custom Image) VMware ESXi 7.0 Update 3k Standard & All Custom Image for ESXi 7.0 U3k Install CD 请访问原文链接:https://sysin.org/blog/vmware-esxi-7-u3/,查看最新版。原创作品…...

JVM整体分析篇
这里写目录标题JVM的组成部分1.类装载子系统1.1一个类加载到JVM的过程1.2类加载机制1.3为什么设计双亲委派机制1.4怎么打破双亲委派机制2.运行时数据区2.1线程私有及共享2.2JVM内存区结构2.3JVM参数设置经验3.Java对象的生命周期3.1.对象的创建3.2.对象大小的计算(6…...

【Python入门第十七天】Python While 循环
Python 循环 Python 有两个原始的循环命令: while 循环for 循环 while 循环 如果使用 while 循环,只要条件为真,我们就可以执行一组语句。 实例 只要 i 小于 7,打印 i: i 1 while i < 7:print(i)i 1运行实…...

怎样激发读者好奇心?短视频营销之场景化
目录 激发读者好奇心?四个小技巧帮你搞定 1.省略法 2.欲言又止法: 3.问句法:就是用疑问的形式引起别人的好奇。 4.反差法 选择合适的主题。 利用场景化效果 使用滤镜。 如何提高用户的留存率。 1、设置一个有趣的话题。 2、用好道具。 3、多用竖屏。 什…...

【LeetCode】剑指 Offer 14- II. 剪绳子 II p96 -- Java Version
题目链接:https://leetcode.cn/problems/jian-sheng-zi-ii-lcof/ 1. 题目介绍(14- II. 剪绳子 II) 给你一根长度为 n 的绳子,请把绳子剪成整数长度的 m 段(m、n都是整数,n>1并且m>1)&…...

【红黑树】红黑树插入操作相关的细节和疑难拆解分析
本文就红黑树的插入操作进行细致到每一个小步骤的解析。1,成员变量本红黑树使用了三叉链结构,使用的时候尤其要记得处理指向父亲的指针。为何在节点的构造函数中,默认节点的颜色为红色?因为考虑到红黑树的性质(对于每个…...

字符串匹配--strstr函数的模拟实现思路和代码
一,strstr函数 原型: const char * strstr ( const char * str1, const char * str2 );char * strstr ( char * str1, const char * str2 ); strstr是一个字符串匹配函数,在str1中去寻找str2,如果找到,返回str2在…...

【ArcGIS Pro二次开发】(7):地图(Map)的基本操作
地图是ArcGIS Pro中的基础起点,也是大多数工程的基础。主要用于显示表示空间数据的图层。 一、地图(Map)的基本操作示例 1、获取当前地图 var map MapView.Active.Map; 2、获取一级图层 var lys map.Layers; 用于获取地图中的单一图层,以及图层组…...

python 自动化测试 pytest 的使用
pytest 是一款以python为开发语言的第三方测试,主要特点如下: 比自带的 unittest 更简洁高效,兼容 unittest框架 支持参数化 可以更精确的控制要测试的测试用例 丰富的插件,已有300多个各种各样的插件,也可自定义扩…...

闭包(回顾)
概念作用保护作用保存作用优缺点命名空间 概念 闭包(closure)指有权访问另一个函数作用域中变量的函数 — Javacript高级程序设计 p309 简单理解,一个作用域可以访问另一个函数内部的私有变量 // 其中 test就是一个闭包 function fn(){var num 10function test …...

利用好这两个方法,服务型企业缺成本票不再难解决!
现代服务业属于人才密集型和技术型类别,其中囊括了不少技术,知识,智力服务等产业:信息技术,文化创意,营销策划,广告设计,以及咨询,商务和法律服务。 在金税三期完善之前…...