OpenSceneGraph几个重要功能节点练习
OpenSceneGraph几个重要功能节点练习
一. 空间变换节点
空间变换中最重要的是坐标系和矩阵运算了。OSG坐标系中使用右手系,Z轴垂直向上,X轴水平向右,Y轴垂直屏幕向里,与OpenGL和DirectX都不同。
相关缩放、旋转和平移主要由osg::Matrix, osg::Vec3, osg::Quat几个类来完成。
局部坐标系向世界坐标系转换规则是:设在局部坐标系下顶点 V 转换成世界坐标系坐标 V':
V' = V * Mn* Mn-1*……* M3* M2* M1* M0
其中M0到Mn一次为各个矩阵变换。从世界坐标系坐标下顶点 V' 转换成局部坐标系 V:
V = V' * M0-1 * M1-1 * M2-1 * M3-1 *……* Mn-1-1 * Mn-1
对于空间变换而言,无论是OpenGL,DirectX还是OSG,一般都会遵守SRT(Scale/Rotate/Translate)的运算顺序来完成符合矩阵的构建:
其公式为:
M =Ms * Mr * Mt
osg::Matrix mt = osg::Matrix::scale( osg::Vex3(sx, sy, sz) ) *
osg::Matrix::rotate( osg::Quat(angle, axis) ) *
osg::Matrix::translate( osg::Vec3(tx, ty, tz) ) ;
代码
#include <osg/Node>
#include <osg/AutoTransform>
#include <osg/MatrixTransform>
#include <osg/PositionAttitudeTransform>
#include <osgDB/ReadFile>
#include <osgViewer/Viewer>
//library for OSG
#pragma comment(lib, "osgd.lib")
#pragma comment(lib, "osgDBd.lib")
#pragma comment(lib, "osgViewerd.lib")
osg::Transform* createAutoTransform(double posX, osg::Node* node)
{
osg::ref_ptr<osg::AutoTransform> at = new osg::AutoTransform();
at->setAutoRotateMode( osg::AutoTransform::ROTATE_TO_SCREEN );
at->setPosition( osg::Vec3(posX, , ) );
at->addChild( node );
return at.release();
}
osg::Transform* createMatrixTransform(double posX, double rotateZ, osg::Node* node)
{
osg::ref_ptr<osg::MatrixTransform> mt = new osg::MatrixTransform();
mt->setMatrix( osg::Matrix::rotate( rotateZ, osg::Z_AXIS) *
osg::Matrix::translate( posX, , ) );
mt->addChild( node );
return mt.release();
}
osg::Transform* createPositionAttitudeTransform(double posX, double rotateZ, osg::Node* node)
{
osg::ref_ptr<osg::PositionAttitudeTransform> pat = new osg::PositionAttitudeTransform();
pat->setAttitude( osg::Quat(rotateZ, osg::Z_AXIS) );
pat->setPosition( osg::Vec3(posX, , ) );
pat->addChild(node);
return pat.release();
}
int main(int argc, char** argv)
{
osg::ArgumentParser argument(&argc, argv);
osg::ref_ptr<osg::Node> node = osgDB::readNodeFiles(argument);
if( !node.get() ) node = osgDB::readNodeFile( "cow.osg" );
osg::ref_ptr<osg::Group> root = new osg::Group();
root->addChild( createAutoTransform(0.0, node) );
root->addChild( createMatrixTransform(-15.0, osg::PI_4, node) );
root->addChild( createPositionAttitudeTransform(15.0, -osg::PI_4, node) );
osgViewer::Viewer viewer;
viewer.setSceneData( root.get() );
return viewer.run();
}
二. 开关节点(osg::Switch)
开关节点Switch的作用是,在场景中某时刻,它的某些子节点被隐藏和忽略,而列外一些节点正常显示并完成相应功能。示例中利用开关节点的更新回调实现子节点的切换:
代码
#include <osg/Switch>
#include <osg/NodeCallback>
#include <osgDB/ReadFile>
#include <osgDB/WriteFile>
#include <osgViewer/Viewer>
#pragma comment(lib, "osgd.lib")
#pragma comment(lib, "osgDBd.lib")
#pragma comment(lib, "osgViewerd.lib")
class CessnaCallback: public osg::NodeCallback
{
public:
virtual void operator()(osg::Node* node, osg::NodeVisitor* nv)
{
osg::Switch* sw = dynamic_cast<osg::Switch*>(node);
if(sw && nv)
{
const osg::FrameStamp* fs = nv->getFrameStamp();
if( fs )
{
if( _fireStartFrame < fs->getFrameNumber() )
{
sw->setValue(, false);
sw->setValue(, true);
}
}
}
traverse(node, nv);
}
private:
;
};
int main(int argc, char** argv)
{
osg::ref_ptr<osg::Switch> root = new osg::Switch();
root->addChild( osgDB::readNodeFile("cessna.osg"), true );
root->addChild( osgDB::readNodeFile("cessnafire.osg"), false );
root->addUpdateCallback(new CessnaCallback() );
osgDB::writeNodeFile( *(root.get()), "Switch.osg" );
osgViewer::Viewer viewer;
viewer.setSceneData( root.get() );
return viewer.run();
}
三. 细节层次节点(osg::LOD)
LOD节点,其基本实现功能是在不影响渲染效果的条件下 ,根据场景对象与观察者的距离,从多个预置方案中选择一种合适的来表达要渲染的物体,从而减轻系统的负担,模型越靠近观察者越精细,而远处只需要较少的多边形类表达。示例如下:
代码
#include <osg/LOD>
#include <osgDB/ReadFile>
#include <osgViewer/Viewer>
#include <osg/Notify>
#pragma comment(lib, "osgd.lib")
#pragma comment(lib, "osgDBd.lib")
#pragma comment(lib, "osgViewerd.lib")
int main(int argc, char** argv)
{
osg::ref_ptr<osg::Node> node = osgDB::readNodeFile( "bunny-high.ive" );
if(!node)
{
osg::notify(osg::NotifySeverity::ALWAYS) << "Cannot open the model bunny!\n" ;
;
}
float r = node->getBound().radius();
osg::ref_ptr<osg::LOD> lod = new osg::LOD();
lod->addChild(node. );
lod->addChild( osgDB::readNodeFile(, r* );
lod->addChild( osgDB::readNodeFile(, FLT_MAX );
osgViewer::Viewer viewer;
viewer.setSceneData( lod.get() );
viewer.run();
}
四. 代理节点(ProxyNode)
ProxyNode代理节点是一种用于动态加载其他模型节点的节点类型。这些节点不会立即被解析和加入场景,而是在场景运行过程中逐步载入。示例:
代码
#include <osg/ProxyNode>
#include <osgDB/ReadFile>
#include <osg/ArgumentParser>
#include <osgViewer/Viewer>
#include <osg/Notify>
#include <iostream>
#pragma comment(lib, "osgd.lib")
#pragma comment(lib, "osgDBd.lib")
#pragma comment(lib, "osgViewerd.lib")
int main(int argc, char** argv)
{
osg::ArgumentParser argument(&argc, argv);
osg::ref_ptr<osg::ProxyNode> pn = new osg::ProxyNode();
unsigned ;
; i < argument.argc(); i++)
{
if( argument.isString(i) )
{
std::cout << num << ": " << argument[i] << "\n" ;
pn->setFileName( num++, argument[i] );
}
}
if( !pn->getNumFileNames() )
pn->setFileName(, "cow.osg" );
osgViewer::Viewer viewer;
viewer.setSceneData( pn.get() );
return viewer.run();
}
OpenSceneGraph几个重要功能节点练习的更多相关文章
- 多功能节点连线绘图控件Nevron Diagram for .NET使用方法及下载地址
Nevron Diagram for .NET是一个功能强大,世界上顶级的.NET图表控件.可扩展的图形报表构架,可以帮您创建功能丰富的Winforms及Webforms图表解决方案.这个产品构建于N ...
- Lua BehaviourTree 各节点说明
项目说明 本行为树的代码使用Lua编写,所有的内容也建立的Lua的基础语法之上 因为公司项目需求,需要一套Lua的行为树代码,所以尝试从饥荒中抽离了行为树相关的代码.绝大多数节点行为与饥荒中相同,不过 ...
- zigbee 路由节点丢失后清除 该节点的残余网络信息
清除脱离网络的 路由节点(stale device)的 残留在各表中以AssociationDevList为例的残余信息. 如图所示拓扑结构中: 路由器1脱离网络后,通过协调器按键操作来 清除 协调 ...
- NC6开发配置流程
1.功能注册 2.菜单注册 3.单据类型管理 4.单据模板初始化 5.查询模板初始化 6.功能节点默认模板设置 7.编码对象注册.编码规则定义
- MongoDB 聚合管道(Aggregation Pipeline)
管道概念 POSIX多线程的使用方式中, 有一种很重要的方式-----流水线(亦称为"管道")方式,"数据元素"流串行地被一组线程按顺序执行.它的使用架构可参考 ...
- 分享在winform下实现模块化插件编程
其实很早之前我就已经了解了在winform下实现插件编程,原理很简单,主要实现思路就是:先定一个插件接口作为插件样式及功能的约定,然后具体的插件就去实现这个插件接口,最后宿主(应用程序本身)就利用反射 ...
- 分享在winform下实现左右布局多窗口界面-续篇
之前的这篇文章<分享在winform下实现左右布局多窗口界面>已经实现了左右布局多窗口界面,今天本来是研究基于winform的插件编程,没想到顺便又找到了另一种实现方案,这种实现方案更简单 ...
- HDFS Federation (读书笔记)
HDFS Federation (读书笔记) HDFS的架构 HDFS包含两个层次:命名空间管理(Namespace) 和 块/存储管理(Block Storage). 命名空间管理(Namespac ...
- etc 安装及使用
键值存储仓库,用于配置共享和服务发现. A highly-available key value store for shared configuration and service discover ...
随机推荐
- Django: 之Model、Cookis、Session
到目前为止,当我们的程序涉及到数据库相关操作时,我们一般都会这么搞: 创建数据库,设计表结构和字段 使用MySQLdb来连接数据库,并编写数据访问层代码 业务逻辑层去调用数据访问层执行数据库操作 im ...
- WiresShark 一站式学习
按照国际惯例,从最基本的说起. 抓取报文: 下载和安装好Wireshark之后,启动Wireshark并且在接口列表中选择接口名,然后开始在此接口上抓包.例如,如果想要在无线网络上抓取流量,点击无线接 ...
- jQuery笔记(1)
jQuery 是一个类库 拥有众多js函数的类库 jQuery 大大简化了js的书写代码,看的舒服,用的爽. jQuery 是一个数组,它能够隐性的遍历. 比如 ${"button" ...
- 程序员 10Tips
理解技术债务 技术债务就像信用卡一样,会有很高的利息率,时间越长,修复所花的代价就越大,程序员对这个要有深刻的认识.同时团队应该培养一种保证设计质量的文化,应当鼓励重构.同时也应当鼓励其它有关代码质量 ...
- HDU 1882 Strange Billboard(状态压缩+转置优化)
状态压缩,我们枚举第一行的所有状态,然后根据第一行去转变下面的行,枚举或者深搜直到最后最后一行,可以判断是不是所有的1都可以全部转换为0,记录所有的解,输出最小的一个就可以. 这里有一个很重要的优化, ...
- ecos的model
表->dbschema->model 虚拟化model机制 在dbschema存在model不存在的情况下 很多mvc结构都这么来 model命名规则 {$app_name}_mdl_{$ ...
- Apache Commons工具集简介(转)
此文为转帖,原帖地址:http://zhoualine.iteye.com/blog/1770014
- 2014非专业知识学习---be smart
非专业部分--构建人生 以书籍和网易公开课为主 (1)理财&投资 基金投资相关,好的书籍? (2)哲学总览 <公正>这个看了大半,需要总结归纳. (必选) 同时结合哲学史,归纳西 ...
- Hibernate3提供的属性的延迟加载功能
Hibernate3增强了对实体属性的延迟加载功能,要实现这个功能,分两个步骤 1.在hbm配置文件上对某个property设置lazy=true <property name=" ...
- Android开发:组播(多播)与广播
近期由于需要编写能够使同一局域网中的Android客户端与PC端进行自动匹配通信功能的程序,学习并试验了JAVA组播与广播的内容,记录一些理解如下: 一.组播(多播) 背景知识:组播使用UDP对一定范 ...