【FastCAE源码阅读3】几何模型显示:从OCC对象到VTK对象

从几何到显示还是比较麻烦的,需要将几何对象转换成渲染对象,涉及几何建模、离散化、图形显示,阅读本文需了解一些基本的OCC、VTK编程

一、几何体显示基本流程

FastCAE几何内核使用的是OCC,显示渲染用的VTK,那么就存在将OCC建模后的几何对象变成VTK支持类型的数据这个过程。

FastCAE在用OCC创建完模型之后,会生成一个Geometry::GeometrySet对象,这个对象包含最原始的OCC中TopoDS_Shape实例对象。创建完成后都会发出信号emit showSet(set);,这个信号会触发将TopoDS_Shape变成VTK显示对象。这个段代码是在void GeometryViewProvider::showGeoSet(Geometry::GeometrySet *set, bool render)函数中实现。

二、几何对象变成显示对象过程

几何体是由点、线、面构成的,进行显示的时候是分别提取这些数据,进行单独显示的。也就是说我们看到的一个立方体等几何对象是由三个VTK显示对象Actor拼起来的。这个函数代码如下:

void GeometryViewProvider::showGeoSet(Geometry::GeometrySet *set, bool render)
{QList<vtkPolyData *> viewPolys = _viewData->transferToPoly(set); // 对几何体进行点、线、面的拆分vtkPolyData *facePoly = viewPolys.at(0); // 面的多边形数据vtkPolyData *edgePoly = viewPolys.at(1); // 边的数据集vtkPolyData *pointPoly = viewPolys.at(2); // 点数据集GeoViewObj viewObj;if (facePoly != nullptr) // 创建显示面的Actor{vtkSmartPointer<vtkActor> actor = vtkSmartPointer<vtkActor>::New();vtkSmartPointer<vtkPolyDataMapper> mapper = vtkSmartPointer<vtkPolyDataMapper>::New();mapper->SetInputData(facePoly);actor->SetMapper(mapper);bool vis = set->isVisible();bool show = Setting::BusAPI::instance()->getGraphOption()->isShowGeoSurface();actor->SetVisibility(show && vis); // 设置是否显示面actor->SetPickable(false);   // 不可鼠标拾取?哪里打开呢?actor->GetProperty()->SetRepresentationToSurface(); // 这个函数不调用也没看到啥影响_preWindow->AppendActor(actor, ModuleBase::D3, false); // 将面添加到场景中viewObj._faceObj = QPair<vtkActor *, vtkPolyData *>(actor, facePoly);}// 创建显示边的Actorif (edgePoly != nullptr) {vtkSmartPointer<vtkActor> actor = vtkSmartPointer<vtkActor>::New();vtkSmartPointer<vtkPolyDataMapper> mapper = vtkSmartPointer<vtkPolyDataMapper>::New();mapper->SetInputData(edgePoly);actor->SetMapper(mapper);bool vis = set->isVisible();bool show = Setting::BusAPI::instance()->getGraphOption()->isShowGeoEdge();actor->SetVisibility(show && vis);actor->SetPickable(false);actor->GetProperty()->SetRepresentationToWireframe();//			actor->GetProperty()->EdgeVisibilityOn();float width = Setting::BusAPI::instance()->getGraphOption()->getGeoCurveWidth();actor->GetProperty()->SetLineWidth(width);_preWindow->AppendActor(actor, ModuleBase::D3, false);viewObj._edgeObj = QPair<vtkActor *, vtkPolyData *>(actor, edgePoly);}// 创建显示点的Actorif (pointPoly != nullptr){vtkSmartPointer<vtkActor> actor = vtkSmartPointer<vtkActor>::New();vtkSmartPointer<vtkPolyDataMapper> mapper = vtkSmartPointer<vtkPolyDataMapper>::New();float size = Setting::BusAPI::instance()->getGraphOption()->getGeoPointSize();mapper->SetInputData(pointPoly);actor->SetMapper(mapper);bool vis = set->isVisible();bool show = Setting::BusAPI::instance()->getGraphOption()->isShowGeoPoint();actor->SetVisibility(show && vis);actor->SetPickable(false);actor->GetProperty()->SetRepresentationToPoints();actor->GetProperty()->SetPointSize(size);_preWindow->AppendActor(actor, ModuleBase::D3, false);viewObj._pointObj = QPair<vtkActor *, vtkPolyData *>(actor, pointPoly);}_geoViewHash.insert(set, viewObj);if (render)_preWindow->resetCamera();
}

三、几何面数据的提取

几何体面、边、点转换为vtkPolyData对象的方式类似,都在函数GeometryViewData::transferToPoly()中,这里只分析面数据的提取。

/* 提取几何表面的多边形数据集 */
vtkPolyData *GeometryViewData::transferFace(Geometry::GeometrySet *gset)
{TopoDS_Shape *shape = gset->getShape(); // 获取OCC的ShapeTopExp_Explorer faceExp(*shape, TopAbs_FACE); // 这个类可访问Shape的拓扑关系QList<Handle(TopoDS_TShape)> tshapelist; // 放置已访问的Face对象,防止重复访问vtkSmartPointer<vtkAppendPolyData> appendFilter = vtkSmartPointer<vtkAppendPolyData>::New();vtkPolyData *polyData = vtkPolyData::New();int beg = 0;for (int index = 0; faceExp.More(); faceExp.Next(), ++index) // 开始遍历面{const TopoDS_Shape &s = faceExp.Current();Handle(TopoDS_TShape) ts = s.TShape();if (tshapelist.contains(ts)) // 防止重复访问,会存在这种情况吗?continue;tshapelist.append(ts);IVtkOCC_Shape::Handle aShapeImpl = new IVtkOCC_Shape(s); // OCC提供IVtkOCC_Shape类vtkSmartPointer<IVtkTools_ShapeDataSource> DS = vtkSmartPointer<IVtkTools_ShapeDataSource>::New(); // OCC提供的VTK数据源DS->SetShape(aShapeImpl); vtkSmartPointer<vtkCleanPolyData> cleanFilter = vtkSmartPointer<vtkCleanPolyData>::New();cleanFilter->SetInputConnection(DS->GetOutputPort());cleanFilter->Update();vtkSmartPointer<vtkPolyData> tpolys = vtkSmartPointer<vtkPolyData>::New();vtkPolyData *tpolydata = cleanFilter->GetOutput();const int np = tpolydata->GetNumberOfPoints(); // 点的数量const int nc = tpolydata->GetNumberOfCells();  // cell的数量vtkPoints *points = vtkPoints::New();for (int i = 0; i < np; i++) //  提取几何点数据{double *coor = tpolydata->GetPoint(i);points->InsertNextPoint(coor);}tpolys->SetPoints(points); // 设置几何点数据vtkCellArray *cells = vtkCellArray::New();for (int i = 0; i < nc; ++i){vtkCell *cell = tpolydata->GetCell(i);vtkIdList *ceid = cell->GetPointIds();if (ceid->GetNumberOfIds() == 3) // 只提取三角形,这里获取的cell包含点、线、三角形,一个立方体最后应该12个三角形{vtkTriangle *triangle = vtkTriangle::New();triangle->DeepCopy(cell); // 有DeepCopy接口cells->InsertNextCell(triangle);}}tpolys->SetPolys(cells); // 设置拓扑多边形数据集的cell// 法线数据?貌似打开与否不影响显示效果vtkSmartPointer<vtkPolyDataNormals> normals = vtkSmartPointer<vtkPolyDataNormals>::New();normals->SetInputData(tpolys);normals->FlipNormalsOn();normals->Update();vtkPolyData *facePoly = normals->GetOutput();const int ncell = facePoly->GetNumberOfCells();if (ncell < 1)continue;GeometryViewObject *obj = new GeometryViewObject(GeometryViewObject::Face, beg, beg + ncell - 1, ts);beg += ncell;appendFilter->AddInputData(facePoly);  // 追加到appendFilter中auto setViewObj = this->getGeosetObj(gset);setViewObj->appendFaceViewObj(index, ts, obj);}appendFilter->Update();polyData->DeepCopy(appendFilter->GetOutput());auto setViewObj = this->getGeosetObj(gset);setViewObj->setFacePoly(polyData); // 保存面多边形数据const int npc = polyData->GetNumberOfCells();if (npc < 1)return nullptr;return polyData;
}

这里要注意几个类:

  1. TopExp_Explorer: OCC提供的遍历几何体拓扑结构的类
  2. IVtkOCC_Shape、IVtkTools_ShapeDataSource:这俩类可以将面变成VTK的cell,包含面上的点、边、三角形
  3. vtkPolyDataNormals: 看类名猜测是生成法线的,但是不用这类显示也没啥问题
  4. vtkAppendPolyData:可以将各个面的数据追加在一起

最后来张图吧:
在这里插入图片描述

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

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

相关文章

【VPX630】青翼 基于KU115 FPGA+C6678 DSP的6U VPX通用超宽带实时信号处理平台

板卡概述 VPX630是一款基于6U VPX总线架构的高速信号处理平台&#xff0c;该平台采用一片Xilinx的Kintex UltraScale系列FPGA&#xff08;XCKU115&#xff09;作为主处理器&#xff0c;完成复杂的数据采集、回放以及实时信号处理算法。采用一片带有ARM内核的高性能嵌入式处理器…

OpenCV标定演示,及如何生成标定板图片

标定的程序在官方的源码里有&#xff0c; opencv-4.5.5\samples\cpp\tutorial_code\calib3d\camera_calibration 很多小白不知道怎么跑起来&#xff0c;这个也怪OpenCV官方&#xff0c;工作没做完善&#xff0c;其实的default.xml是要自己手动改的&#xff0c;输入的图片也要…

华为OD机试 - 寻找身高相近的小朋友 - 双指针(Java 2023 B卷 100分)

目录 专栏导读一、题目描述二、输入描述三、输出描述四、解题思路五、Java算法源码六、效果展示1、输入2、输出3、说明 华为OD机试 2023B卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试&#xff08;JAVA&#xff09;真题&#xff08;A卷B卷&#…

Apache Pulsar 在腾讯云上的最佳实践

导语 由 StreamNative 主办的 Pulsar Meetup Beijing 2023 在2023年10月14日完美落幕&#xff0c;本次活动大咖云集&#xff0c;来自腾讯、滴滴、华为、智联招聘、RisingWave 和 StreamNative 的行业专家们一起&#xff0c;深入探讨 Pulsar 在生产环境中的最佳应用实践&#x…

【优选算法系列】第一节.栈的简介(1047. 删除字符串中的所有相邻重复项和844. 比较含退格的字符串)

文章目录 前言一、删除字符串中的所有相邻重复项和 1.1 题目描述 1.2 题目解析 1.2.1 算法原理 1.2.2 代码编写二、比较含退格的字符串 2.1 题目描述 2.2 题目解析 2.2.1 算法原理 2.2.2 代码编写总结 前言 …

【数据库】关系数据库管理系统 (RDBMS) 中事务处理的四个特性ACID

给自己一个目标&#xff0c;然后坚持一段时间&#xff0c;总会有收获和感悟&#xff01; 数据库事务是指一组数据库操作&#xff08;例如插入、更新、删除等&#xff09;&#xff0c;被视为一个单独的逻辑操作单元&#xff0c;并且要么全部执行成功&#xff0c;要么全部不执行&…

微信小程序-授权登录(手机号码)

1、WXBizDataCrypt.js-下载地址 2、UNIAPP代码 <template> <view class"work-container"> <view class"login"> <view class"content"> <button class"button_wx&q…

图论08-图的建模-状态的表达与理解 - 倒水问题为例

文章目录 状态的表达例题1题解1 终止条件&#xff1a;有一个数位为42 状态的改变&#xff1a;a表示十位数&#xff0c;b表示个位数3 其他设置 例题2 力扣773 滑动谜题JavaC 状态的表达 例题1 从初始的(x&#xff0c;y)状态&#xff0c;到最后变成&#xff08;4&#xff0c;&am…

乌班图 Linux 系统 Ubuntu 23.10.1 发布更新镜像

Ubuntu 团队在其官网上发布了Ubuntu 23.10.1 版本,这是目前较新的 Ubuntu 23.10(Focal Fossa)操作系统系列的第一个发行版,旨在为社区提供最新的安装媒体。Ubuntu 22.04 LTS(Focal Fossa)操作系统系列于 2022 年 4 月 21 日发布。 Ubuntu 23.10 LTS(长期支持版本)可用…

Android Studio中配置Git

安装Git 在安装Android Studio之前&#xff0c;需要先安装Git。可以从Git官网下载并安装Git&#xff1a;https://git-scm.com/downloads 在Android Studio中配置Git 在Android Studio中&#xff0c;依次点击“File” -> “Settings”&#xff0c;在弹出的窗口中选择“Ver…

el-table表格设置——动态修改表头

(1) 首先是form表单写表单设置按钮&#xff1a; &#xff08;1.1&#xff09;使用el-popover&#xff0c;你需要修改的是this.colOptions&#xff0c;colSelect: <el-popover id"popover" popper-class"planProver" placement"bottom" width&…

推荐彩虹知识商城源码

彩虹知识商城7.0.3小森升级版新增供货商开心学习版&#xff0c;新增邮件提醒功能&#xff0c;支持给用户发订单、结算等邮件通知&#xff0c;支持给管理员发送提现、域名审核等邮件通知&#xff0c;支持设置手续费最低扣除金额&#xff0c;修复了其他一些已知问题。 演示地址&…