OSG Clip例程剖析

首先是创建剪切节点的函数代码:

 osg::ref_ptr<osg::Node> decorate_with_clip_node(const osg::ref_ptr<osg::Node>& subgraph)
{
osg::ref_ptr<osg::Group> rootnode = new osg::Group; // create wireframe view of the model so the user can see
// what parts are being culled away.
osg::StateSet* stateset = new osg::StateSet;
//osg::Material* material = new osg::Material;
osg::PolygonMode* polymode = new osg::PolygonMode;
polymode->setMode(osg::PolygonMode::FRONT_AND_BACK,osg::PolygonMode::LINE);
stateset->setAttributeAndModes(polymode,osg::StateAttribute::OVERRIDE|osg::StateAttribute::ON); osg::Group* wireframe_subgraph = new osg::Group;
wireframe_subgraph->setStateSet(stateset);
wireframe_subgraph->addChild(subgraph);
rootnode->addChild(wireframe_subgraph); // more complex approach to managing ClipNode, allowing
// ClipNode node to be transformed independently from the subgraph
// that it is clipping. osg::MatrixTransform* transform= new osg::MatrixTransform; osg::NodeCallback* nc = new osg::AnimationPathCallback(subgraph->getBound().center(),osg::Vec3(0.0f,1.0f,1.0f),osg::inDegrees(45.0f));
transform->setUpdateCallback(nc); osg::ClipNode* clipnode = new osg::ClipNode;
osg::BoundingSphere bs = subgraph->getBound();
bs.radius()*= 0.4f; osg::BoundingBox bb;
bb.expandBy(bs); clipnode->createClipBox(bb);
clipnode->setCullingActive(false); transform->addChild(clipnode);
rootnode->addChild(transform); // create clipped part.
osg::Group* clipped_subgraph = new osg::Group; clipped_subgraph->setStateSet(clipnode->getStateSet());
clipped_subgraph->addChild(subgraph);
rootnode->addChild(clipped_subgraph); return rootnode;
}

  经过梳理发现这段代码主要有两大部分组成:根节点下包含线框子图wireframe_subgraph和裁剪子图clipped_subgraph两部分。他们分别负责绘制线框模型和裁剪模型。

通过将线框子图的状态集设置为osg::PolygonMode::LINE,并将函数的参数osg::Node& subgraph加入到其下作为子节点。

  声明转换矩阵transform,并设置其模拟路径回调函数。声明裁剪节点clipnode,并创建其边界盒(边界正方体),设置边界盒的边长为函数的参数osg::Node& subgraph的边界球半径的0.4倍。将裁剪节点clipnode作为转换矩阵transform的字节点,以使其按照模拟回调函数不断地进行旋转裁剪,显示动态裁剪效果。

  最后将已裁剪的节点clipped_subgraph的状态集设置为裁剪节点clipnode的状态集。并将其加入到根节点。

  利用绘图工具对以上函数的对象关系进行梳理分析如下:

  注意在上图中,裁剪节点并没有包含函数参数subgraph作为其子节点。它仅仅是一个旋转的裁剪盒子。

  下一个函数是简单裁剪函数,与以上函数不同的是,它没有显示模型的线框模式,而是对模型直接进行裁剪。其函数代码如下:

 osg::ref_ptr<osg::Node> simple_decorate_with_clip_node(const osg::ref_ptr<osg::Node>& subgraph)
{
osg::ref_ptr<osg::Group> rootnode = new osg::Group; // more complex approach to managing ClipNode, allowing
// ClipNode node to be transformed independently from the subgraph
// that it is clipping. osg::MatrixTransform* transform= new osg::MatrixTransform; osg::NodeCallback* nc = new osg::AnimationPathCallback(subgraph->getBound().center(),osg::Vec3(0.0f,0.0f,1.0f),osg::inDegrees(45.0f));
transform->setUpdateCallback(nc); osg::ClipNode* clipnode = new osg::ClipNode;
osg::BoundingSphere bs = subgraph->getBound();
bs.radius()*= 0.4f; osg::BoundingBox bb;
bb.expandBy(bs); clipnode->createClipBox(bb);
clipnode->setCullingActive(false); transform->addChild(clipnode);
rootnode->addChild(transform); // create clipped part.
osg::Group* clipped_subgraph = new osg::Group; clipped_subgraph->setStateSet(clipnode->getStateSet());
clipped_subgraph->addChild(subgraph);
rootnode->addChild(clipped_subgraph); return rootnode;
}

  可以看到,简单裁剪函数去掉了第一个函数中5~16行创建线框模型的代码。分别用正常和简单模式运行程序,结果如下:

普通模式

简单模式

总结:

  程序通过函数参数subgraph获得其边界球,并取其边界球半径的0.4倍建立边界盒。将上述边界盒设为裁剪节点clipnode的裁剪盒。建立转换矩阵transform,设置其模拟路径(动态旋转效果),并将clipnode作为其子节点。将裁剪节点clipnode的状态集设置为裁剪后的子图clipped_subgraph的状态集,以完成对子图参数subgrpah的动态裁剪。最后将转换矩阵transform,裁剪后的子图clipped_subgraph,线框模型添加到根节点中,实现最终的裁剪效果。通过设置转换矩阵transform的模拟回调路径,或修改裁剪节点clipnode的大小和形状,可修改动态裁剪的效果。

最新文章

  1. 2016年优秀的java网站分享
  2. 在Salesforce中编写Unit Test
  3. 出现upstream sent too big header while reading response header from upstream错误
  4. 多校赛3- Solve this interesting problem 分类: 比赛 2015-07-29 21:01 8人阅读 评论(0) 收藏
  5. 参加魅族 flyme 互联网编程大赛的一些感受
  6. Codeforces 543C Remembering Strings(DP)
  7. 给EcStore商城会员添加推广返利功能
  8. 徒手用Java来写个Web服务器和框架吧&lt;第三章:Service的实现和注册&gt;
  9. 关于java 定时任务
  10. 【Peaks加强版 BZOJ 3551】你被坑了吗?
  11. idea spring-boot总结
  12. skynet记录6:定时器
  13. map和list循环遍历
  14. js算法初窥07(算法复杂度)
  15. oracle 函数 输入值,查询数据,返回相应处理结果
  16. go 语言字典遍历
  17. elasticdump
  18. Spring Boot--02MVC设置
  19. P2015 二叉苹果树
  20. 【十月のA Letter to 后辈菌】

热门文章

  1. [fw]拦截系统调用
  2. 2. ZooKeeper基础
  3. python数字图像处理(三)边缘检测常用算子
  4. [Luogu2365]任务安排(斜率优化)
  5. 在IDEA中如何将Spring boot项目打包成可执行的jar包并发布到linux服务
  6. HTML基础:用表单写一个简易登录页面
  7. ZYNQ的Linux Linaro系统镜像制作SD卡启动(仅使用mkfs部分,其他部分待看)
  8. bzoj3812 主旋律 容斥+状压 DP
  9. Ubuntu18.04 安装 Idea 2018.2
  10. 【学习笔记】虚树复习记(BZOJ2286 SDOI2011 消耗战)