一、问题背景与重现步骤
现象描述:
在 Qt 应用程序中,主界面包含 QComboBox 控件,并且启用了高 DPI 支持(例如在 main() 中调用 QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling)
)。
当用户将窗口拖动到不同 DPI 的屏幕上,点击 QComboBox 弹出下拉列表时,下拉列表的位置与预期不符,出现明显偏移,从而影响用户体验。
重现步骤:
- 在主界面上添加 QComboBox 控件,并确保应用程序启用了高 DPI 支持。
- 将应用程序窗口从一个 DPI 设置较低的屏幕拖动到 DPI 设置较高的屏幕上。
- 点击 QComboBox 弹出下拉列表,观察下拉列表的位置是否正确。
二、问题原因分析
1. QComboBox 的弹出机制
当 QComboBox 下拉列表弹出时,Qt 内部会创建一个临时窗口(通常是 QComboBoxPrivateContainer 或 QAbstractItemView 的顶级窗口)用于显示选项。
这种设计虽然保证了下拉列表能够独立显示,但也带来了额外的窗口句柄和坐标计算问题。
2. 多屏幕和高 DPI 环境下的坐标转换问题
在不同 DPI 屏幕之间切换时,每个屏幕的缩放因子(devicePixelRatio)不同。Qt 在计算下拉列表的全局坐标时,需要将 QComboBox 的局部坐标转换为全局坐标。如果转换过程中未能正确处理屏幕之间的 DPI 差异,就会导致下拉列表的位置计算出错,从而出现偏移现象。
3. 内部实现 bug 的可能性
Qt 5.15.2 对于多屏幕和高 DPI 的支持已有改进,但在 QComboBox 下拉列表弹出时,内部对 popup 窗口位置计算的逻辑仍可能存在不足,导致在跨屏时出现位置偏移问题。