当前位置: 首页 > news >正文

VTK——angleWidget的3D转换

文章目录

  • 3D空间坐标转换
  • 例程
  • 心得

3D空间坐标转换

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

例程

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. 如果是创建线条组件,需要手动触发一次回调函数,因为线条组件的端点位置是可以直接读取的。
                lineCallback->Execute(lineWidget, vtkCommand::EndInteractionEvent, nullptr);         //手动触发回调

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

std::map<int,vtkSmartPointer<vtkAngleWidget>> ngleWidgets;
http://www.lryc.cn/news/155775.html

相关文章:

  • HDFS 集群动态节点管理
  • postman9.12.汉化版(附有下载链接)
  • mysql与msql2数据驱动
  • 解决微信小程序回调地狱问题
  • cron介绍
  • mkp勒索病毒的介绍和防范,勒索病毒解密,数据恢复
  • 【面试精品】关于面试会遇到的Apache相关的面试题
  • python对文件转md5,用于文件重复过滤
  • mac苹果电脑删除顽固残留软件图标
  • 【jsvue】联合gtp仿写一个简单的vue框架,以此深度学习JavaScript
  • linux centos7 系统之编程:求水仙花数
  • git中的cherry-pick和merge有些区别以及cherry-pick怎么用
  • 【前端】CSS-Flex弹性盒模型布局
  • Android AAPT: error: resource color 异常原因处理
  • C++std::function和std::bind()的概念
  • QT Creator工具介绍及使用
  • python爬虫13:pymysql库
  • 权限管理 ACL、RBAC、ABAC的学习
  • python的re正则表达式
  • 【算法与数据结构】700、LeetCode二叉搜索树中的搜索
  • SpringBoot v2.7.x+ 整合Swagger3入坑记?
  • 说说你了解的 CDC
  • SpingMvc入门
  • JVM的故事——类文件结构
  • springboot自定义表格(动态合并单元格)
  • C++零碎记录(二)
  • 数学建模:回归分析
  • 数据库(一)
  • 【算法与数据结构】106、LeetCode从中序与后序遍历序列构造二叉树
  • kali 安装cpolar内网穿透实现 ssh 远程连接