Qt实现无边框窗口的拖动和缩放
在使用QT创建窗体的时候,为了使窗口美化,通常不使用QT自带的边框。会调用下面函数去除窗体边框。
setWindowFlags(Qt::FramelessWindowHint)
但是有个问题,当去除了QT自带边框后,窗体就变得不能移动了,也不能改变窗口大小了。这确实是个问题,该怎么去解决呢?
1.重写mousePressEvent\ mouseMoveEvent mouseReleaseEvent事件
protected:
void mousePressEvent(QMouseEvent* ev)override;
void mouseReleaseEvent(QMouseEvent* ev)override;
void mouseMoveEvent(QMouseEvent* ev)override;
.h文件
namespace {/* 这里我们将一个窗口划分为9个区域,分别为左上角(1, 1)、中上(1,2)、右上角(1, 3)左中 (2, 1)、 中间(2, 2)、右中 (2, 3)左下角(3, 1)、中下(3,2)、 右下角(3, 3)*/const int kMouseRegionLeft = 5;const int kMouseRegionTop = 5;const int kMouseRegionRight = 5;const int kMouseRegionBottom = 5;}// 鼠标的 活动范围的 枚举
enum MousePosition{// 这个是上面图片划分的区域 1,1 区 就用 11 代表 , 1,2 就用12 代表 以此类推kMousePositionLeftTop = 11,kMousePositionTop = 12,kMousePositionRightTop = 13,kMousePositionLeft = 21,kMousePositionMid = 22,kMousePositionRight = 23,kMousePositionLeftBottom = 31,kMousePositionBottom = 32,kMousePositionRightBottom = 33,};private://根据鼠标的设置鼠标样式,用于拉伸void SetMouseCursor(int x, int y);//判断鼠标的区域,用于拉伸int GetMouseRegion(int x, int y);
private:bool isPress;QPoint windowsLastPs;QPoint mousePs;int mouse_press_region = kMousePositionMid;
.cpp文件
void SPlay::mousePressEvent(QMouseEvent* ev)
{if (ev->button() == Qt::LeftButton) {// 如果是鼠标左键// 获取当前窗口位置,以窗口左上角windowsLastPs = pos();// 获取鼠标在屏幕的位置 就是全局的坐标 以屏幕左上角为坐标系mousePs = ev->globalPos();isPress = true;// 获取鼠标在那个区域mouse_press_region = GetMouseRegion(ev->pos().x(), ev->pos().y());}
}void SPlay::mouseReleaseEvent(QMouseEvent* ev)
{if (ev->button() == Qt::LeftButton) {isPress = false;}setCursor(QCursor{});
}void SPlay::mouseMoveEvent(QMouseEvent* ev)
{// 设置鼠标的形状SetMouseCursor(ev->pos().x(), ev->pos().y());// 计算的鼠标移动偏移量, 就是鼠标全局坐标 - 减去点击时鼠标坐标QPoint point_offset = ev->globalPos() - mousePs;if ((ev->buttons() == Qt::LeftButton) && isPress){if (mouse_press_region == kMousePositionMid){// 如果鼠标是在窗口的中间位置,就是移动窗口move(windowsLastPs + point_offset);}else {// 其他部分 是拉伸窗口// 获取客户区QRect rect = geometry();switch (mouse_press_region){// 左上角case kMousePositionLeftTop:rect.setTopLeft(rect.topLeft() + point_offset);break;case kMousePositionTop:rect.setTop(rect.top() + point_offset.y());break;case kMousePositionRightTop:rect.setTopRight(rect.topRight() + point_offset);break;case kMousePositionRight:rect.setRight(rect.right() + point_offset.x());break;case kMousePositionRightBottom:rect.setBottomRight(rect.bottomRight() + point_offset);break;case kMousePositionBottom:rect.setBottom(rect.bottom() + point_offset.y());break;case kMousePositionLeftBottom:rect.setBottomLeft(rect.bottomLeft() + point_offset);break;case kMousePositionLeft:rect.setLeft(rect.left() + point_offset.x());break;default:break;}setGeometry(rect);mousePs = ev->globalPos();}}
}void SPlay::SetMouseCursor(int x, int y)
{// 鼠标形状对象Qt::CursorShape cursor{};int region = GetMouseRegion(x, y);switch (region){case kMousePositionLeftTop:case kMousePositionRightBottom:cursor = Qt::SizeFDiagCursor; break;case kMousePositionRightTop:case kMousePositionLeftBottom:cursor = Qt::SizeBDiagCursor; break;case kMousePositionLeft:case kMousePositionRight:cursor = Qt::SizeHorCursor; break;case kMousePositionTop:case kMousePositionBottom:cursor = Qt::SizeVerCursor; break;case kMousePositionMid:cursor = Qt::ArrowCursor; break;default:break;}setCursor(cursor);
}int SPlay::GetMouseRegion(int x, int y)
{int region_x = 0, region_y = 0;// 鼠标的X坐标小于 边界5 说明他在最上层区域 第一区域if (x < kMouseRegionLeft){region_x = 1;}else if (x > (this->width()/*窗体宽度*/ - kMouseRegionRight/*边界宽度5*/)) {// 如果鼠标X坐标 大于 最右侧的边界 说明他在第三区域region_x = 3;}else {region_x = 2;}if (y < kMouseRegionTop){// 同理 鼠标Y坐标 小于上层边界5 说明鼠标在第一区域region_y = 1;}else if (y > (this->height() - kMouseRegionBottom)) {// 鼠标Y坐标的 大于 最下面的坐标,鼠标就在 第三区region_y = 3;}else {region_y = 2;}// 最后计算 表示区域的 数值 (x=1, y=1) 计算 = 1*10+1 =11 // x=2,y=3 = 3*10+2 = 32 在图的 3,2 区域return region_y * 10 + region_x;
}
2.QSizeGrip实现
#include <QApplication>
#include <QWidget>
#include <QVBoxLayout>
#include <QSizeGrip>int main(int argc, char *argv[])
{QApplication app(argc, argv);QWidget *widget = new QWidget;QVBoxLayout *layout = new QVBoxLayout(widget);widget->setLayout(layout);// 添加窗口内容// ...// 添加 QSizeGrip 控件QSizeGrip *sizeGrip = new QSizeGrip(widget);layout->addWidget(sizeGrip, 0, Qt::AlignBottom | Qt::AlignRight);// 设置窗口大小和位置widget->resize(400, 300);widget->move(100, 100);widget->show();return app.exec();
}
在这个示例中,我们首先创建一个QWidget对象和一个QVBoxLayout布局对象,并将布局设置为窗口的布局。然后,我们添加窗口的内容到布局中。
接下来,我们创建一个QSizeGrip对象,并将其添加到布局中的右下角。我们使用Qt::AlignBottom | Qt::AlignRight参数将QSizeGrip控件放置在窗口的右下角。
最后,我们设置窗口的大小和位置,并将其显示出来。
通过在无边框窗口中添加QSizeGrip控件,用户可以使用鼠标拖动控件来调整窗口的大小。
参考文章1:QT 创建一个 可移动、可拉伸的无边框窗体_qt 无边框窗口拉伸-CSDN博客
参考文章2:Qt实现无边框窗口实现拉伸的三种方法--附源码_qt 无边框窗口拉伸-CSDN博客
参考小项目:https://github.com/a-mo-xi-wei/frameLessWindow
相关文章:
Qt实现无边框窗口的拖动和缩放
在使用QT创建窗体的时候,为了使窗口美化,通常不使用QT自带的边框。会调用下面函数去除窗体边框。 setWindowFlags(Qt::FramelessWindowHint) 但是有个问题,当去除了QT自带边框后,窗体就变得不能移动了,也不能改变窗口大…...
入门岛2-python实现wordcount并进行云端debug
书生大模型学习 任务: 1.实现一个wordcount函数,统计英文字符串中每个单词出现的次数。返回一个字典,key为单词,value为对应单词出现的次数。 2.Vscode连接InternStudio debug TIPS:记得先去掉标点符号,然后把每个单词…...
c语言-链表1
10 链表 一、链表是什么? -- 数据的一种存储方式 -- 链式存储 (1)线性存储 -- 地址连续 -- 自动开辟,自动释放 -- 默认是线性存储 (2)链式存储 -- 地址不连续…...
你好! Git——企业级开发模型
企业级开发模型(6) 一、删除远程分支,git branch -a (查看所有本地分支与远程分支)还能看到已经删除的分支,怎么解决?二、企业级开发流程2.1 企业级开发流程2.2 系统开发环境 三、Git分支设计模…...
力扣面试150 查找和最小的 K 对数字 最小堆 去重
Problem: 373. 查找和最小的 K 对数字 👨🏫 参考题解 class Solution {public List<List<Integer>> kSmallestPairs(int[] nums1, int[] nums2, int k) {// 创建一个大小为 k 的结果列表,用于存储和最小的 k 个数对List<Li…...
Oceanbase 执行计划
test100 CREATE TABLE `test100` ( `GRNT_CTR_NO` varchar(32) COLLATE utf8mb4_bin NOT NULL COMMENT 担保合同编号, `GRNT_CTR_TYP` varchar(3) COLLATE utf8mb4_bin NOT NULL COMMENT 担保合同类型, `COLC_GRNT_IND` varchar(1) COLLATE utf8mb4_bin DEFAULT NULL …...
精品丨模型关系介绍
PowerBI中的模型关系相信小伙伴们都不会感觉到陌生,因为一份优秀的报表无法离开数据模型的支撑。 对比其它BI类工具而言,白茶认为其建模功能才是最为突出的功能点。 模型关系类型 PowerBI中我们常用的模型关系一共包含5类: 一对一关系(1:1) …...
CentOS7 配置 nginx 和 php 方案
配置方案 一、安装软件二、编写配置文件,连接PHP三、引用文件四、测试 鉴于网上教程错综复杂,写下一这篇文章 本教程只需要三步即可 一、安装软件 yum install -y nginx php php-fpm二、编写配置文件,连接PHP 一般情况下在安装完 nginx 后…...
Promise.all全面解析:使用方法与实战技巧
Promise是JavaScript中处理异步操作的重要机制,它提供了一种优雅的方式来处理异步回调,避免了传统回调地狱的问题。而Promise.all作为Promise的一个静态方法,更是在处理多个异步操作时发挥着关键作用。本文将全面解析Promise.all的使用方法&a…...
NLP从零开始------9文本进阶处理之文本相似度计算
1.文本相似度计算简介 在自然语言处理中,经常会涉及度量两个文本相似度的问题。在诸如对话系统和信息减速等中,度量句子或短语之间的相似度尤为重要。在新闻学传媒中应用文本相似度可以帮助读者快速检索到想要了解的报道。 文本相似度的定义式如下所示&a…...
Electron 在 MAC 上的 build 签名应用配置
Electron 在 MAC 上的 build 签名应用配置涉及多个步骤,包括准备开发者账号、生成证书和配置文件、配置环境变量以及使用适当的工具进行签名和公证。以下是一个详细的配置流程: 一、准备开发者账号 首先,你需要在 Apple 开发者网站 注册并拥有一个开发者账号。这个账号将用…...
15 交换机命令行配置
交换机命令行配置 一、交换机命令行基本配置 (一)配置主机名 Switch>enable Switch#configure terminal Switch(config)#hostname S1(二)查看配置信息 Switch#show running-config Building configuration...Current confi…...
工作流之Flowable与SpringBoot结合
文章目录 1 Flowable1.1 flowable-ui部署运行1.2 绘制流程图1.2.1 绘制1.2.2 绘图细节1.2.3 bpmn文件导入 1.3 后台项目搭建1.3.1 pom.xml1.3.2 数据库表说明 1.4 流程引擎API与服务1.4.1 主要API1.4.2 示例 1 Flowable 1.1 flowable-ui部署运行 flowable-6.6.0 运行 官方dem…...
python实战:数据分析基础知识
当涉及到数据分析和统计建模时,Python 提供了强大的工具和库,如 pandas、numpy、statsmodels 和 matplotlib。本文将以一个实际的案例为例,介绍如何利用这些工具进行回归分析,并通过可视化工具进行结果展示和解释。 1. 背景介绍 …...
Grafana深入讲解
Grafana 深入讲解 目录 概述Grafana 基本概念 2.1 Grafana 简介2.2 Grafana 功能特性2.3 Grafana 架构 Grafana 安装与配置 3.1 安装 Grafana3.2 配置 Grafana3.3 验证 Grafana 安装 Grafana 数据源 4.1 支持的数据源类型4.2 添加数据源4.3 配置 Prometheus 数据源 Grafana 仪…...
002 git
下载 使用git clone命令下载特定分支 打开终端或命令行界面。 使用cd命令切换到你想存放仓库副本的本地目录。 使用以下命令克隆仓库的develop分支到本地(注意替换<仓库URL>为实际的仓库URL): git clone -b develop --single-branch…...
MySQL --- 用户管理
一、用户信息 MySQL中的用户信息,都存储在系统数据库mysql的表user中 user表的结构如下 这里主要介绍以下几个字段 host : 表示这个用户可以从哪个主机登陆,如果是 localhost ,表示只能从本机登陆 user: 用户名 a…...
Linux 错误码
目录 一、概述二、含义三、错误处理函数1、IS_ERR2、strerr、perror 一、概述 在 Linux 系统中,错误码是用来表示操作系统运行过程中发生的错误的数字代码。错误码通常由负数表示,0 表示成功,正数表示警告或其他非致命错误。 为了开发者更好…...
《向量数据库指南》——开源社区与商业化的平衡
开源社区与商业化的平衡 Lynn:我觉得这个说的特别好,因为开发者工具其实有很多,但是事实上真正去做开源的这种社区的,尤其是做的比较大的,其实这样的企业还是比较少的。那么当初在起步的时候就这么坚定的去选择开源,然后这么短的时间能获得这么多产品反馈。其实让我想到那…...
记录一次echarts图表大数据量轮询刷新页面卡死问题的优化
项目场景: 在我们的项目架构中,集成的Echarts图表组件采用了折线图,业务需求即每300毫秒自动更新图表上的数据,并且每一次的数据点数量达到了约700个,折线图刷新的很快,每300毫秒就要刷新数据 问题描述 开发过程中发现在这种数据量请求频率下,大概2个小时左右就会导致…...
nginx——方向代理和负载均衡
目录 1.1 Nginx概述 1.1.1 企业青睐 Nginx 的核心原因 1.1.2 Nginx的作用 1.3 反向代理和负载均衡 1.4 注 1.4.1 代理百度并使用 18090 端口 1.1 Nginx概述 1.1.1 企业青睐 Nginx 的核心原因 Nginx 由俄罗斯开发者打造,具有超高稳定性(资源占用极低…...
实战指南:基于快马平台快速开发并部署班级宠物园应用官方下载门户
最近学校想推广一个班级宠物园的教育应用,需要快速搭建一个官方下载页面。作为技术负责人,我尝试用InsCode(快马)平台来快速实现这个需求,整个过程比想象中顺利很多。 项目规划与结构设计 首先明确页面需要包含的几个核心模块:顶部…...
DAB SG(信号发生器)的频道与频率设置详解
1. DAB SG信号发生器基础入门 第一次接触DAB SG信号发生器时,很多人会被那些专业术语搞得一头雾水。其实说白了,这就是个能模拟DAB广播信号的设备,主要用在广播设备测试、信号覆盖测试等场景。我刚开始用的时候也犯迷糊,后来才发现…...
【HTTP】HTTP协议核心体系:请求方法与状态码全结构化解析(附《思维导图》)
文章目录HTTP协议核心体系:请求方法与状态码全结构化解析一、核心基础概念1.1 HTTP方法的两大核心属性(规范级定义)1.2 HTTP状态码分类规则二、HTTP请求方法2.1 标准核心方法(RFC 7231 定义)2.1.1 只读类方法ÿ…...
Qt串口通信避坑指南:用QSerialPort封装类解决粘包拆包(附源码+实战演示)
Qt串口通信实战:从粘包拆包到高可靠数据帧处理的完整解决方案 在嵌入式开发和工业控制领域,串口通信作为最基础却又最关键的通信方式,其稳定性直接影响整个系统的可靠性。许多开发者在使用Qt的QSerialPort进行串口通信时,都曾遇到…...
从梁模型到软体手指:Abaqus仿真进阶,如何为超弹性材料模型设置Ogden参数?
从梁模型到软体手指:Abaqus仿真进阶,如何为超弹性材料模型设置Ogden参数? 在工程仿真领域,线性分析往往只是起点。当面对像软体机器人手指这样的柔性结构时,传统梁模型已无法准确描述其大变形行为。硅胶等超弹性材料的…...
芯片工程师如何从AI那里“榨出“隐性知识?
大语言模型里藏着很多东西,但大部分人只用到了表面。这些模型在训练时吃进去的不只是教科书和官方文档,还有大量的技术博客、论坛讨论、开源代码、甚至是一些没公开发表的技术报告。这些知识以一种隐性的方式存在于模型参数中,不会主动跳出来…...
Phi-4-mini-reasoning推理服务监控:通过webshell日志诊断部署状态方法
Phi-4-mini-reasoning推理服务监控:通过webshell日志诊断部署状态方法 1. 模型简介 Phi-4-mini-reasoning 是一个基于合成数据构建的轻量级开源模型,专注于高质量、密集推理的数据处理。作为Phi-4模型家族的一员,它经过专门微调以提升数学推…...
Wan2.1 VAE与MySQL联动:构建带用户历史记录的图像生成平台
Wan2.1 VAE与MySQL联动:构建带用户历史记录的图像生成平台 你有没有想过,自己用AI生成的每一张图片,都能被自动保存下来,形成一个专属的创意作品集?今天,我们就来动手搭建一个这样的平台。它不仅能让你用W…...
PyTorch模型调试神器:用TensorBoard+torchsummary快速定位网络结构问题
PyTorch模型调试神器:用TensorBoardtorchsummary快速定位网络结构问题 当你在PyTorch中构建复杂的神经网络时,是否经常遇到以下困扰:模型训练时突然报出维度不匹配的错误,却不知道具体是哪一层出了问题?或者模型参数数…...
