VTK——angleWidget的3D转换

文章目录

3D空间坐标转换

在冠状图、矢状面、横截面等创建的角度组件的三个端点坐标,不能直接用在3D视图中。这是因为2D切片的坐标是基于像素的,而3D空间的坐标可能是基于实际物理尺寸的。
解决方案是使用2D点的坐标、切片的物理位置和方向以及可能的其他参数(例如切片厚度或间距),来计算空间坐标。将计算好的空间坐标p1、p2、p3传入下面的函数中,就可以在3D视图中画出角度组件。

例程

cpp 复制代码
get3DViewAngle(double* p1, double* p2, double* mid, int* indexVal) {
    double modifiedPoints[3][3]; 
    int currentAngleIndex = *indexVal;
    auto iter = threeDActors.find(currentAngleIndex);
    if (iter != threeDActors.end()) {
        for (auto act : iter->second) {
            threeDRenderer->RemoveActor(act);
        }
        threeDActors.erase(iter);
    }

    double* pts[] = {p1, p2, mid};

    for(int i = 0; i < 3; i++) {
        modifiedPoints[i][0] = pts[i][0] * xRange[1] / imageDims[0];
        modifiedPoints[i][1] = pts[i][1] * yRange[1] / imageDims[1];
        modifiedPoints[i][2] = pts[i][2] * zRange[1] / imageDims[2];
    }

    vtkSmartPointer<vtkSphereSource> src1 = vtkSmartPointer<vtkSphereSource>::New();
    src1->SetCenter(modifiedPoints[0]);
    src1->SetRadius(2);
    vtkSmartPointer<vtkPolyDataMapper> map1 = vtkSmartPointer<vtkPolyDataMapper>::New();
    map1->SetInputConnection(src1->GetOutputPort());
    vtkSmartPointer<vtkActor> act1 = vtkSmartPointer<vtkActor>::New();
    act1->SetMapper(map1);

    vtkSmartPointer<vtkSphereSource> src2 = vtkSmartPointer<vtkSphereSource>::New();
    src2->SetCenter(modifiedPoints[1]);
    src2->SetRadius(2);
    vtkSmartPointer<vtkPolyDataMapper> map2 = vtkSmartPointer<vtkPolyDataMapper>::New();
    map2->SetInputConnection(src2->GetOutputPort());
    vtkSmartPointer<vtkActor> act2 = vtkSmartPointer<vtkActor>::New();
    act2->SetMapper(map2);

    vtkSmartPointer<vtkSphereSource> src3 = vtkSmartPointer<vtkSphereSource>::New();
    src3->SetCenter(modifiedPoints[2]);
    src3->SetRadius(2);
    vtkSmartPointer<vtkPolyDataMapper> map3 = vtkSmartPointer<vtkPolyDataMapper>::New();
    map3->SetInputConnection(src3->GetOutputPort());
    vtkSmartPointer<vtkActor> act3 = vtkSmartPointer<vtkActor>::New();
    act3->SetMapper(map3);

    vtkSmartPointer<vtkLineSource> lnSrc1 = vtkSmartPointer<vtkLineSource>::New();
    lnSrc1->SetPoint1(modifiedPoints[0]);
    lnSrc1->SetPoint2(modifiedPoints[2]);
    vtkSmartPointer<vtkPolyDataMapper> lnMap1 = vtkSmartPointer<vtkPolyDataMapper>::New();
    lnMap1->SetInputConnection(lnSrc1->GetOutputPort());
    vtkSmartPointer<vtkActor> lnAct1 = vtkSmartPointer<vtkActor>::New();
    lnAct1->SetMapper(lnMap1);

    vtkSmartPointer<vtkLineSource> lnSrc2 = vtkSmartPointer<vtkLineSource>::New();
    lnSrc2->SetPoint1(modifiedPoints[1]);
    lnSrc2->SetPoint2(modifiedPoints[2]);
    vtkSmartPointer<vtkPolyDataMapper> lnMap2 = vtkSmartPointer<vtkPolyDataMapper>::New();
    lnMap2->SetInputConnection(lnSrc2->GetOutputPort());
    vtkSmartPointer<vtkActor> lnAct2 = vtkSmartPointer<vtkActor>::New();
    lnAct2->SetMapper(lnMap2);

    threeDRenderer->AddActor(act1);
    threeDRenderer->AddActor(act2);
    threeDRenderer->AddActor(act3);
    threeDRenderer->AddActor(lnAct1);
    threeDRenderer->AddActor(lnAct2);
    std::vector<vtkSmartPointer<vtkActor>> acts = {act1, act2, act3, lnAct1, lnAct2};
    threeDActors[currentAngleIndex] = acts;

    for (auto act : acts) {
        threeDRenderer->AddActor(act);
    }
    renderWidget[3]->interactor()->Render();
}


}

心得

  1. 在创建角度组件后,不能直接读取组件的端点坐标来用于计算空间坐标,因为此时的端点坐标还是空。需要等到鼠标左键点击三次后,才会得到完整的端点坐标。这个时候可以写在角度组件的回调函数中,然后使用成员变量来获取创建好的端点坐标。
  2. 在角度组件的回调函数中,可以使用信号槽机制来保证拖动组件后能够实时更新角度组件位置。
  3. 如果是创建线条组件,需要手动触发一次回调函数,因为线条组件的端点位置是可以直接读取的。
cpp 复制代码
                lineCallback->Execute(lineWidget, vtkCommand::EndInteractionEvent, nullptr);         //手动触发回调

4.如果想要保证实时删除2D和3D视图中的角度组件,可以使用map,只需要记住创建的索引就能直接删除组件。这样可以与页面上的组件数量和组件索引独立开来.

cpp 复制代码
std::map<int,vtkSmartPointer<vtkAngleWidget>> ngleWidgets;
相关推荐
Vect__1 小时前
基于线程池从零实现TCP计算器网络服务
c++·网络协议·tcp/ip
naruto_lnq6 小时前
分布式系统安全通信
开发语言·c++·算法
CSDN_RTKLIB8 小时前
【四个场景测试】源文件编码UTF-8 BOM
c++
肉包_5119 小时前
两个数据库互锁,用全局变量互锁会偶发软件卡死
开发语言·数据库·c++
Trouvaille ~10 小时前
【Linux】UDP Socket编程实战(一):Echo Server从零到一
linux·运维·服务器·网络·c++·websocket·udp
HellowAmy10 小时前
我的C++规范 - 线程池
开发语言·c++·代码规范
czy878747510 小时前
const 在 C/C++ 中的全面用法(C/C++ 差异+核心场景+实战示例)
c语言·开发语言·c++
十五年专注C++开发10 小时前
MinHook:Windows 平台下轻量级、高性能的钩子库
c++·windows·钩子技术·minhook
一只小小的芙厨11 小时前
寒假集训笔记·树上背包
c++·笔记·算法·动态规划
以卿a11 小时前
C++(继承)
开发语言·c++·算法