QT 使用全局缩放进行全分辨率适配(QT_SCALE_FACTOR)-程序员宅基地

技术标签: c++  QT  qt  ui  Windows  

事出有因

因为现在做的一个项目是全程全屏显示的。因此不同屏幕分辨率对程序界面的影响太大。而UI设计的时候又没有过多的考虑自动布局这方面的事。
虽然在刚开始做界面的时候已经尽量利用自动布局来做,但是有些控件提供的图片大小不太合适,在做的时候只能给控件设置固定大小。在测试不同分辨率的时候发现界面展现的效果不同。于是就开始想办法做分辨率适配。

偶然事件

具体忘记当时在搜索什么东西了,大概应该是高分屏适配的问题。在qt的一篇官方文档中发现了这个好东西:

QT_SCALE_FACTOR [numeric] defines a global scale factor for the whole application, including point-sized fonts.

QT_SCALE_FACTOR

它是一个QT的环境变量,看它的字面意思是缩放因子或者叫缩放系数
这东西咋用的呢?看一段示例代码:

qputenv("QT_SCALE_FACTOR", "2.0");

就这么简单。但是有个条件,就是这句代码需要放再QApplication实例初始化之前。
它有什么效果呢?
它是用来控制整个程序界面的所有元素的缩放比例的一个环境变量。上面这句代码的效果就是将整个界面放大为原来的2倍。
来看效果:
无缩放
两倍缩放

怎么用呢

从上面两张图可以看出来,在缩放的时候是整个界面进行缩放的。不管是pt单位还是px单位都进行了缩放 。
那么代码就可以这样写。

	//这个是Windows平台用来获取屏幕宽度的代码,
	//因为在qApplication实例初始化之前,QGuiApplication::screens();无法使用。
	qreal  cx = GetSystemMetrics(SM_CXSCREEN);
	qreal scale = cx / 960;				// 960 = 1920 / 2
	qputenv("QT_SCALE_FACTOR", QString::number(scale).toLatin1());

上面代码大概意思就是通过对比当前屏幕的分辨率和设计分辨率(960×540)来确定程序的整体缩放比例。
这样无论在哪种分辨率情况下界面展现的效果都是一样的。
备注:
为什么设计分辨率是960×540。因为它是1920×1080的一半。而现在设备的常用分辨率基本上都是1920×1080;更重要的是我自己的电脑分辨率就是1092×1080,因此我的界面设计的时候就是按照1920×1080分辨率设计的。这样的话我只需要将界面上所有元素的大小减半就行了。另外一个重要的原因就是缩放系数不能小于1.0,不然会出bug,至于到底会发生什么,自己试试就知道了。

后遗症

通过上面两张图,你会发现这个全局缩放引发了两个问题:

  1. 图片放大的时候有锯齿。
  2. 字体在缩放的时候质量不太好。

因此需要另外两个东西来解决这两个问题。

  1. 调整图片缩放质量
    	//控制图片缩放质量
    QApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
    QCoreApplication::setAttribute(Qt::AA_UseHighDpiPixmaps);
    
  2. 文字质量
    这需要使用另外一个东西”qt.conf“。
    内容如下。
    [Platforms]
    WindowsArguments = fontengine=freetype
    
    将这个文件放入程序的资源中。路径为":/qt/etc/qt.conf"

来看效果:
调整质量

注意事项

缩放系数必须大于等于1.0,不然会出现一些奇怪的问题。而且必须是小数形式的字符串。即使缩放系数经过计算后是整数也要写成小数形式。即1920/960 = 2要写成2.0传递进去。
至于获取屏幕尺寸的问题,有些代码提供了一个思路,就是先初始化一个临时的QApplocation获取到屏幕大小后再释放掉,我不建议这么做,因为在我的项目中这样做引起了一些bug,偶尔会闪退。建议自己根据平台不同多写几行代码。

吐槽

QQ截图另存为jpg的时候画质是真的低。

bug修复

由于Windows本身显示缩放机制的存在,上面用来获取屏幕分辨率的方法获取到的并不是真实数据。改用以下方法获取屏幕分辨率。

	DEVMODE NewDevMode;
	//获取屏幕设置中的分辨率
	EnumDisplaySettings(0, ENUM_CURRENT_SETTINGS, &NewDevMode);
	qreal  cx = NewDevMode.dmPelsWidth;			//当前设置的屏幕宽度
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/u014410266/article/details/107488789

智能推荐

Spring设计模式之——策略模式_spring 策略模式-程序员宅基地

文章浏览阅读470次。这里写自定义目录标题欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题,有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants创建一个自定义列表如何创建一个注脚注释也是必不可少的KaTeX数学公式新的甘特图功能,丰富你的文章UML 图表FLowchart流程图导出与导入导出导入欢迎使用Ma..._spring 策略模式

前后端分离项目的跨域问题_偌依前端跨域问题-程序员宅基地

文章浏览阅读786次。springboot+vue 的前后端分离项目, 尝试解决跨域问题._偌依前端跨域问题

分享6个AI绘画网站_gptai绘画网址-程序员宅基地

文章浏览阅读1.2k次。官网:https://www.midjourney.com/特点:免费且效果不错Stable Diffusion是一个文本生成图像的模型。现在有人把它做成网站了。可以生成与Dall-E相似质量的图像,但其特征较为简单。虽然有时图像可能与提示不完全匹配,但只要你正确书写提示,就能够生成很棒的图像。更为优秀的是,它是完全免费且开源的。没有基于信用积分的系统,您可以随意生成任意数量的图像!官网:https://stablediffusionweb.com/特点:可利用AI生成海报。_gptai绘画网址

Android应用具有persist属性时如何自升级_persistent 升级安装-程序员宅基地

文章浏览阅读5.6k次,点赞2次,收藏5次。Android系统中,为了某些目的需要保证应用运行时尽量不被系统kill(特别是处于后台时),所以都会给应用增加persist标签,以避免在系统低内存时被系统kill。但是添加了该属性的应用如果需要进行应用自升级(无论是别的应用进行安装升级还是应用自己安装自己进行应用升级),有一些需要特别注意的地方,否则就会导致程序的运行状态错乱。 一般没有persist属_persistent 升级安装

crontab_crontab查看执行情况-程序员宅基地

文章浏览阅读4.6k次。Linux crontab定时任务_crontab查看执行情况

Mac环境变量配置(Java)_mac配置java环境变量-程序员宅基地

文章浏览阅读1.1w次,点赞7次,收藏43次。1.打开终端:2.输入命令:【/usr/libexec/java_home -V】,查看默认的jdk下载地址(绿色下划线的就是jdk默认路径)(注意️:命令行终端是区分大小写的【-v 是不对的,必须是大写 -V】)3.如果是第一次配置环境变量,使用命令:【touch .bash_profile】创建一个.bash_profile隐藏配置文件(如果存在已有配置文件就输入:【open -e .bash_profile】)打开如下4.输入以下命令:(注意️:红色字体是第2步查出来自己jd..._mac配置java环境变量

随便推点

flask_sqlalchemy的session线程安全_flask session_options-程序员宅基地

文章浏览阅读3.4k次,点赞2次,收藏5次。flask_sqlalchemy是如何在多线程中对数据库操作不相互影响参考来源:1.https://www.cnblogs.com/lgjbky/p/9482278.html2.http://www.cnblogs.com/ctztake/p/8277372.html3.https://stackoverflow.com/questions/39480914/why-db-s..._flask session_options

基于 SpringBoot + Vue 的前后端分离游戏资讯平台_前后端咨询-程序员宅基地

文章浏览阅读804次。基于 SpringBoot + Vue 的前后端分离游戏资讯平台,使用 Shiro 进行权限控制,使用 JWT 作为交互 token,使用 Aspectj 进行切面编程,使用 Spring Data Jpa 方便进行数据库操作,使用 Druid 作为数据库连接池,使用 MySQL 作为数据库;前端使用 element-ui 作为组件库,使用 bootstrap-vue 进行响应式编程。......_前后端咨询

【深入理解C++】空类对象所占用的空间大小_c++空类大小-程序员宅基地

文章浏览阅读1.4k次,点赞4次,收藏6次。1.须知 2.空类对象所占用的空间大小 3.一个类继承空类 4.空类作为另一个类的成员_c++空类大小

linux 磁盘 ssd 机械硬盘分区,linux(centos7)上的硬盘种类、结构、磁盘分区方式、分区结构...-程序员宅基地

文章浏览阅读629次。硬盘硬盘机械硬盘、固态硬盘机械硬盘(HDD)Hard Disk Drive,即是传统普通硬盘,主要由:盘片,磁头,盘片转轴及控制电机,磁头控制器,数据转换器,接口,缓存等几个部分组成。机械硬盘中所有的盘片都装在一个旋转轴上,每张盘片之间是平行的,在每个盘片的存储面上有一个磁头,磁头与盘片之间的距离比头发丝的直径还小,所有的磁头联在一个磁头控制器上,由磁头控制器负责各个磁头的运动。磁头可沿盘片的半径..._固态硬盘和机械硬盘怎么给linux分区

Nor Flash 与 Nand Flash_nand flash norflash互换-程序员宅基地

文章浏览阅读781次。NOR和NAND是现在市场上两种主要的非易失闪存技术。历史Intel于1988年首先开发出NOR flash技术,彻底改变了原先由EPROM和EEPROM一统天下的局面。1989年,东芝公司发表了NAND flash结构,强调降低每比特的成本,更高的性能,并且象磁盘一样可以通过接口轻松升级。相“flash存储器”经常可以与相“NOR存储器”互换使用。许多业内人士也搞不清楚NAND闪存_nand flash norflash互换

为何你的 App 在 iPhone 12 上显示异常,而别人的不会?-程序员宅基地

文章浏览阅读235次。Python实战社群Java实战社群长按识别下方二维码,按需求添加扫码关注添加客服进Python社群▲扫码关注添加客服进Java社群▲转自:hite和落雁背景10月14日 iPhone ..._ios12 pro 系统异常