occ中以鼠标所在位置进行缩放

原理分析

鼠标中心缩放的核心原理是通过视口变换矩阵 调整摄像机位置或投影参数。OpenCASCADE通过StartZoomAtPointZoomAtPoint两个关键方法实现:

  • 视点锚定:将鼠标当前位置作为缩放中心点,避免传统缩放导致的视觉偏移。
  • 模拟拖动:通过计算虚拟位移量(delta)触发动态缩放效果,实际是修改投影矩阵的缩放因子。

关键概念

1. 视口坐标系转换

鼠标坐标(mx, my)需转换为OpenCASCADE内部使用的视口坐标系。OCC视图默认原点在左上角,Y轴向下为正方向。

2. 滚轮刻度处理

高精度滚轮事件angleDelta()返回的数值以120为基本单位。正值表示滚轮向前(放大),负值表示向后(缩小)。

3. 缩放因子计算

固定比例系数zoomFactor(示例为1.1)控制缩放幅度。通过判断滚轮方向决定放大或缩小:

  • 放大:scale = zoomFactor
  • 缩小:scale = 1.0 / zoomFactor

代码实现解析

步骤1:获取鼠标位置

cpp 复制代码
const QPoint pos = theEvent->position().toPoint();
const Standard_Integer mx = pos.x();
const Standard_Integer my = pos.y();

将Qt的浮点坐标转换为整型,适应OCC接口要求。

步骤2:处理滚轮事件

cpp 复制代码
const int steps = theEvent->angleDelta().y() / 120;
if (steps == 0) { theEvent->accept(); return; }

过滤无效事件(如横向滚轮),避免不必要的计算。

步骤3:计算缩放比例

cpp 复制代码
const Standard_Real zoomFactor = 1.1;
const Standard_Real scale = (steps > 0) ? zoomFactor : 1.0 / zoomFactor;

通过三元运算符快速确定缩放方向。

步骤4:执行缩放操作

cpp 复制代码
myView->StartZoomAtPoint(mx, my);
const Standard_Integer delta = (steps > 0) ? -10 : 10;
myView->ZoomAtPoint(mx, my, mx + delta, my + delta);
  • StartZoomAtPoint:锁定当前鼠标位置为缩放中心。
  • ZoomAtPoint:通过虚拟位移触发动态缩放。正负delta控制放大/缩小方向。

扩展优化

非线性缩放

可通过动态调整zoomFactor实现加速缩放效果:

cpp 复制代码
const Standard_Real zoomFactor = 1.0 + 0.1 * abs(steps);

视口边界保护

添加边界检查防止过度缩放:

cpp 复制代码
if (myView->Scale() > MAX_SCALE || myView->Scale() < MIN_SCALE) return;

完整代码

cpp 复制代码
void wheelEvent(QWheelEvent* theEvent)
{
    if (myView.IsNull()) return;

    const QPoint pos = theEvent->position().toPoint();
    const Standard_Integer mx = pos.x();
    const Standard_Integer my = pos.y();

    // 高分辨率滚轮:每120为一刻度
    const int steps = theEvent->angleDelta().y() / 120;
    if (steps == 0) { theEvent->accept(); return; }

    // 使用OpenCASCADE内置的缩放功能
    const Standard_Real zoomFactor = 1.1;
    const Standard_Real scale = (steps > 0) ? zoomFactor : 1.0 / zoomFactor;

    // 以鼠标位置为中心进行缩放
    myView->StartZoomAtPoint(mx, my);

    // 模拟拖动来缩放
    const Standard_Integer delta = (steps > 0) ? -10 : 10;
    myView->ZoomAtPoint(mx, my, mx + delta, my + delta);


    theEvent->accept(); // 已处理,不传递给 QWidget
}
相关推荐
用户805533698031 天前
不止三件套:QObject 属性系统全关键字与运行时反射!
c++·qt
xcyxiner1 天前
DicomViewer (vcpkg Windows和ubuntu编译)7
qt
Quz6 天前
QML Hello World 入门示例
qt
xcyxiner9 天前
DicomViewer (dcmtk读取dcm文件)5
qt
xcyxiner10 天前
DicomViewer (后台线程处理文件)4
qt
xcyxiner10 天前
DicomViewer (添加模型类)3
qt
xcyxiner11 天前
DicomViewer (目录调整) 2
qt
xcyxiner11 天前
dcmtk vtk vtk-dicom(gdcm) 编译(debug) v2
qt
桥田智能13 天前
桥田智能 QT-650S:面向白车身焊装的 800kg 重载快换解决方案
开发语言·qt·系统架构
森G13 天前
75、服务器源码解析---------云视频服务项目
linux·服务器·网络·c++·qt