QT- QML Layout+anchors 布局+锚点实现窗口部件权重比例分配

布局管理

  • 简单比较两种界面管理
  • 锚点布局实现比例布局
  • 布局管理实现比例布局
    • 循环依赖问题简谈

在日常打螺丝中,我们偶尔会需要实现界面各组件能按比例放置,自适应各种分辨率的需求。我用锚点和布局都实现过相关界面,记录下来两种方式实现的差异。跟大家一起学习。

简单比较两种界面管理

特性布局系统锚点系统
排列方式自动流式布局精确相对定位
比例控制内置权重分配机制需手动计算百分比
嵌套复杂度适合多层嵌套结构适合扁平化结构
动态调整自动响应容器尺寸变化需绑定尺寸信号
典型应用表单布局/工具栏/等间距排列悬浮元素/固定边栏/叠加层

锚点、排列的简单介绍

这两者并不是严格分开使用,经常会混用,比如有时候用锚点分区域,然后用Layout布局控制里面的细节,或者用Layout 控制布局,里面用anchors控制小细节。


锚点布局实现比例布局

QML锚点系统(anchors)提供基于相对关系的布局方式,通过元素间的空间关系实现精准定位。比较直观,使用起来很方便。
常用接口:

  • anchors.left: parent.right ,本组件的左边是父组件的左边,相对定位设置 ,还有right 、top 、bottom等
  • anchors.centerIn: parent ,在父组件里正中央 居中
  • anchors.margins:30, 整体边距控制 或单独设置topMarginbottomMarginleftMarginrightMargin
  • anchors.horizontalCenter: parent.horizontalCenter, 相对父组件水平居中
  • anchors.verticalCenter parent.verticalCenter, 相对父组件垂直居中

实现没有什么约束(或者是我目前的场景没有遇到), 直接根据相对比例,计算即可。一般这样的话,根组件(或者依赖的父组件)要确定好大小,或者成为子组件。

property int ratio : 0.2 // 可以独立成属性,方便修改管理Item {id: containerwidth: 400; height: 300Rectangle {  // 标题(20%)id: headercolor: "lightgreen"anchors {top: parent.topleft: parent.leftright: parent.right}height: container.height * ratio }Rectangle {  // 内容(80%)color: "lavender"anchors {top: header.bottombottom: parent.bottomleft: parent.leftright: parent.right}height: container.height * (1-ratio )}
}

布局管理实现比例布局

所有用布局管理器(如 RowLayout、ColumnLayout)管理的组件都会自动拥有Layout.xxxx 相关的属性,可以设置它们来控制布局实现。我个人比较推荐使用这种方式,当然有不同想法也欢迎评论区讨论
常用接口:

  • Layout.alignment: Qt.AlignTop, 控制对齐方式, 有水平居中Qt.AlignHCenter 左对齐Qt.AlignLeft 等
  • Layout.fillWidth: true, 控制是否填满宽度,高度是fillHeight
  • Layout.margins: 16 ,整体边距控制 或单独设置leftMargin / topMargin
  • Layout.preferredWidth: 50, 设定首选宽度是50, minimumHeight是最小

QT 在Qt 6.2+里提供了一个更直观的比例分配属性 Layout.weight。它不需要手动设置fillXXX,直接在兄弟组件之间设定即可。
但目前(2025年)Qt 5.6.3+ 商用还是会收费,所以本文推荐的做法在 Qt 5.x 和 QtQuick.Layouts 1.x 中是可用的。

用Layout布局管理实现比例分配有三个要点:

  • 按比例分配:同时设置 Layout.fillxxx: true 和 Layout.preferredxxx,这样布局里的组件就会以preferredxxx作为比例因子,在分辨率变时组件相对大小保持不变。
  • 固定像素:仅设置 Layout.preferredxxxx(不设置 fillxxxx)就会变成按给定的值固定属性
  • 避免循环依赖:不要直接绑定 parent.width 或 parent.height,而是通过布局系统的内置机制实现自适应。

什么意思? 给举两个例子就明白了:

一,正确按比例分配(低于Qt 6.2) : 
两个chilstool会按照 82 的比例分配,且在窗口分辨率变化的情况下,比例依然不变:
如果想进一步细致控制,加大就行: 78 : 22,只需保证所有子项加起来 =1=10 或者 =100能整除的就行ColumnLayout
{anchors.fill: parentRectangle{  // 标题(20%)color: "white"Layout.fillWidth: trueLayout.fillHeight: trueLayout.preferredHeight: 2  //0.2也行 比例系数 2 / (2+8) /******/}Rectangle{  // 内容(80%)color: "white"Layout.fillWidth: trueLayout.fillHeight: trueLayout.preferredHeight: 8  //0.8也行 比例系数 8 / (2+8)/******/}
}

二,布局的循环分配 : 
父引用子,子引用父,导致计算出错:
这个其实算是我踩的坑,还以为和锚点一样的用法,被fillxxxx字面意义带偏了。记录一下ColumnLayout {anchors.fill: parentRowLayout {id: childlayoutLayout.fillHeight: true Layout.fillWidth: true  // 会先考虑子控件的宽Rectangle {id: chilstoolLayout.fillHeight: trueLayout.preferredWidth: parent.width * 0.8 // 又依赖父控件的宽color: "#ffffff"}Rectangle {id: chilstool2Layout.fillHeight: trueLayout.preferredWidth: parent.width * 0.2color: "#ffffff"}}}


循环依赖问题简谈

QML 的布局系统在计算尺寸时遵循以下步骤:

父布局管理器 子布局组件 子项目 这段fillXXX 不参与,隐式计算为共有 1. 收集尺寸提示 查询 implicitWidth/Height 返回隐式尺寸 报告自身尺寸需求 2. 计算布局分配 按优先级处理: Minimum > Preferred > Maximum fillXXX 就发生在这个阶段 3. 分配最终尺寸 4. 设置子项目几何 5. 子项目内部布局 6. 布局完成信号 父布局管理器 子布局组件 子项目

fillWidth/fillHeight 并不发生在隐式尺寸计算阶段。它类似给父控件声明“我不知道我尺寸如何,但我要占用你剩下的空间” 。 所以在隐式尺寸计算阶段,它的尺寸是未确定的,而在隐式尺寸计算阶段,又需要获知子控件的尺寸,若此时子控件又依赖父控件的尺寸,QML尺寸计算就会陷入循环依赖,引发了错误。
从分析上能获知,想破坏循环依赖,其实就是确定好子控件或父控件的尺寸,或者正确设定比例权重分配,打破循环。

  • 所以日常打螺丝中,注意布局管理器下的fillxxx 和子项的 parent.width * 0.8不能一起出现即可。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/diannao/85635.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java项目OOM排查

排查思路 Java项目出现OOM(Out Of Memory,内存溢出)问题时,排查思路如下: 确认OOM类型: Java Heap Space:堆内存溢出,通常是对象创建过多或内存泄漏。PermGen Space:永久…

vue+threeJs 生成云状特效屏幕

嗨&#xff0c;我是小路。今天主要和大家分享的主题是“vuethreeJs 生成云状特效屏幕”。 动态云状特效示例图 二、实例代码 <!--创建一个动态数字屏幕--> <template><div class"pageBox"><div class"leftBox" ref"lef…

ABAP设计模式之---“高内聚,低耦合(High Cohesion Low Coupling)”

“高内聚、低耦合”是面向对象编程中非常重要的设计原则&#xff0c;它有助于提高代码的可维护性、扩展性和复用性。 1. 初衷&#xff1a;为什么会有这个原则&#xff1f; 在软件开发中&#xff0c;随着业务需求的复杂化&#xff0c;代码难免会变得越来越庞大。如果开发者将一…

Registry和docker有什么关系?

当遇到多个服务器需要同时传docker镜像的时候&#xff0c;一个一个的传效率会非常慢且压力完全在发送方的网络带宽&#xff1b;可以参考git hub&#xff0c;通常我们会用git push将代码传到git hub&#xff0c;如果谁需要代码用git pull就可以拉到自己的机器上&#xff0c;dock…

linux命令 systemctl 和 supervisord 区别及用法解读

目录 基础与背景服务管理范围配置文件和管理方式监控与日志依赖管理适用场景常用命令对照表实际应用场景举例优缺点对比小结参考链接 1. 基础与背景 systemctl 和 supervisord 都是用于管理和控制服务&#xff08;进程&#xff09;的工具&#xff0c;但它们在设计、使用场景和…

(11)java+ selenium->元素定位之By_tag_name

1.简介 继续WebDriver关于元素定位,这篇介绍By ClassName。tagName是DOM结构的一部分,其中页面上的每个元素都是通过输入标签,按钮标签或锚定标签等标签定义的。每个标签都具有多个属性,例如ID,名称,值类等。就其他定位符而言在Selenium中,我们使用了标签的这些属性值来…

2021 RoboCom 世界机器人开发者大赛-高职组(复赛)解题报告 | 珂学家

前言 题解 2021 RoboCom 世界机器人开发者大赛-高职组&#xff08;复赛&#xff09;解题报告。 模拟题为主&#xff0c;包含进制转换等等。 最后一题&#xff0c;是对向量/自定义类型&#xff0c;重定义小于操作符。 7-1 人工智能打招呼 分值: 15分 考察点: 分支判定&…

day42 简单CNN

目录 一、从图像分类任务谈起 二、CNN架构解剖实验室 2.1 卷积层&#xff1a;空间特征的魔法师 2.2 归一化层&#xff1a;加速收敛的隐形推手 2.3 激活函数&#xff1a;非线性的灵魂 三、工程实践避坑指南 3.1 数据增强工程 3.2 调度器工程实战 四、典型问题排查手册 …

Gitee Wiki:以知识管理赋能 DevSecOps,推动关键领域软件自主演进

关键领域软件研发中的知识管理困境 传统文档管理模式问题显著 关键领域软件研发领域&#xff0c;传统文档管理模式问题显著&#xff1a;文档存储无系统&#xff0c;查找困难&#xff0c;降低效率&#xff1b;更新不及时&#xff0c;与实际脱节&#xff0c;误导开发&#xff1…

清理 pycharm 无效解释器

1. 起因&#xff0c; 目的: 经常使用 pycharm 来调试深度学习项目&#xff0c;每次新建虚拟环境&#xff0c;都是显示一堆不存在的名称&#xff0c;删也删不掉。 总觉得很烦&#xff0c;是个痛点。决定深入研究一下。 2. 先看效果 效果是能行&#xff0c;而且清爽多了。 3. …

【ConvLSTM第二期】模拟视频帧的时序建模(Python代码实现)

目录 1 准备工作&#xff1a;python库包安装1.1 安装必要库 案例说明&#xff1a;模拟视频帧的时序建模ConvLSTM概述损失函数说明&#xff08;python全代码&#xff09; 参考 ConvLSTM的原理说明可参见另一博客-【ConvLSTM第一期】ConvLSTM原理。 1 准备工作&#xff1a;pytho…

MySQL DDL操作全解析:从入门到精通,包含索引视图分区表等全操作解析

目录 一、DDL 基础概述 1.1 DDL 定义与作用 1.2 DDL 语句分类 1.3 数据类型与存储引擎 1.3.1 数据类型 1.3.2 存储引擎差异 二、基础 DDL 语句详解 2.1 创建数据库与表 2.1.1 创建数据库 2.1.2 创建表 2.2 修改表结构 2.2.1 添加列 2.2.2 修改列属性 2.2.3 删除列…

设计模式——抽象工厂设计模式(创建型)

摘要 抽象工厂设计模式是一种创建型设计模式&#xff0c;旨在提供一个接口&#xff0c;用于创建一系列相关或依赖的对象&#xff0c;无需指定具体类。它通过抽象工厂、具体工厂、抽象产品和具体产品等组件构建&#xff0c;相比工厂方法模式&#xff0c;能创建一个产品族。该模…

Express教程【006】:使用Express写接口

文章目录 8、使用Express写接口8.1 创建API路由模块8.2 编写GET接口8.3 编写POST接口 8、使用Express写接口 8.1 创建API路由模块 1️⃣新建routes/apiRouter.js路由模块&#xff1a; /*** 路由模块*/ // 1-导入express const express require(express); // 2-创建路由对象…

【iOS(swift)笔记-14】App版本不升级时本地数据库sqlite更新逻辑二

App版本不升级时&#xff0c;又想即时更新本地数据库怎么办&#xff1f; 办法二&#xff1a;从服务器下载最新的sqlite数据替换掉本地的数据&#xff08;注意是数据不是文件&#xff09; 稍加调整&#xff0c; // &#xff01;&#xff01;&#xff01;注意&#xff01;&…

Mac电脑_钥匙串操作选项变灰的情况下如何删除?

Mac电脑_钥匙串操作选项变灰的情况下如何删除&#xff1f; 这时候 可以使用相关的终端命令进行操作。 下面附加文章《Mac电脑_钥匙串操作的终端命令》。 《Mac电脑_钥匙串操作的终端命令》 &#xff08;来源&#xff1a;百度~百度AI 发布时间&#xff1a;2025-06&#xff09;…

对接系统外部服务组件技术方案

概述 当前系统需与多个外部系统对接,然而外部系统稳定性存在不确定性。对接过程中若出现异常,需依靠双方的日志信息来定位问题,但若日志信息不够完整,会极大降低问题定位效率。此外,问题发生后,很大程度上依赖第三方的重试机制,若第三方缺乏完善的重试机制,就需要手动…

WAF绕过,网络层面后门分析,Windows/linux/数据库提权实验

一、WAF绕过文件上传漏洞 win7&#xff1a;10.0.0.168 思路&#xff1a;要想要绕过WAF&#xff0c;第一步是要根据上传的内容找出来被拦截的原因。对于文件上传有三个可以考虑的点&#xff1a;文件后缀名&#xff0c;文件内容&#xff0c;文件类型。 第二步是根据找出来的拦截原…

一文学会c++中的内存管理知识点

文章目录 c/c内存管理c语言动态内存管理c动态内存管理new/delete自定义类型妙用operator new和operator delete malloc/new&#xff0c;free/delete区别 c/c内存管理 int globalVar 1;static int staticGlobalVar 1;void Test(){static int staticVar 1;int localVar 1;in…

深入解析Linux死锁:原理、原因及解决方案

Linux死锁是系统资源管理的致命陷阱&#xff0c;平均每年导致全球数据中心约​​3.7亿小时​​的服务中断。本文深度剖析死锁形成的​​四个必要条件​​和六种典型死锁场景&#xff0c;结合Linux内核源码层级的资源管理机制&#xff0c;揭示文件系统锁、内存分配、多线程同步等…