OpenSceneGraph 笔记–如何导出三角形数据

在OpenSceneGraph开发中,为了方便会经常使用到一些不是三角形片的数据,比如四边形等数据。例如画一个管子用四边形带比用三角形片好计算得多。比如现在我们要画一个由两个平面组成的面,我可以这样做:
    osg::Geode* geode=new osg::Geode;
    osg::Geometry
* polyGeom = new osg::Geometry;
    osg::Vec3 myCoords[]
=
    {
        osg::Vec3(
0,1,0),
        osg::Vec3(
0,0,0),
        osg::Vec3(
1,1,0),
        osg::Vec3(
1,0,0),
        osg::Vec3(
2,1,0),
        osg::Vec3(
2,0,0)
    };

    int numCoords = sizeof(myCoords)/sizeof(osg::Vec3);
    osg::Vec3Array
* vertices = new osg::Vec3Array(numCoords,myCoords);
    polyGeom
->setVertexArray(vertices);
    polyGeom
->addPrimitiveSet(new osg::DrawArrays(osg::PrimitiveSet::QUAD_STRIP,0,numCoords));
    geode
->addDrawable(polyGeom);

这样就用6个点,用OpenGL提供的QUAD_STRIP方式画出了两个平面。
但是如果要把这个平面用于碰撞检测等技术,那么就需要把这六个点所表示的四边形带转换成三角形片才行。这些三角形定点如下:
0 1 0
0 0 0
1 1 0

0 0 0
1 0 0
1 1 0

1 1 0
1 0 0
2 1 0

1 0 0
2 0 0
2 1 0
可以看出两个平面由4个三角形组成,而且都是逆时针排列(朝向一致)。
以前我自己做过转换,但是感觉很麻烦。OpenSceneGraph的Example osggeometry中提供了一个printTriangles函数,它可以打印出一个drawable所有的三角形片,不管最初的数据结构如何:

struct NormalPrint
{
    
void operator() (const osg::Vec3& v1,const osg::Vec3& v2,const osg::Vec3& v3, boolconst 
    {
        osg::Vec3 normal 
= (v2-v1)^(v3-v2);
        normal.normalize();
        std::cout 
<< "\t("<<v1<<") ("<<v2<<") ("<<v3<<""<<") normal ("<<normal<<")"<<std::endl;
    }
};

// decompose Drawable primtives into triangles, print out these triangles and computed normals.
void printTriangles(const std::string& name, osg::Drawable& drawable)
{
    std::cout
<<name<<std::endl;
    
    osg::TriangleFunctor
<NormalPrint> tf;
    drawable.accept(tf);
 
    std::cout
<<std::endl;
}

核心的思想就是利用osg::TriangleFunctor这个模版。这个模版会让你重载()运算符,然后让Drawable去visit它。在这个过程中,所有原始的数据(不管是三角形片的,还是四边形的)都转换成了三角形片数据。
那么如何把三角形数据导出哪?只需要修改一下借助这个思路,将NormalPrint修改成我们需要的就对了。

struct GetVertex
{
    
void operator() (const osg::Vec3& v1,const osg::Vec3& v2,const osg::Vec3& v3, boolconst 
    {
        vertexList
->push_back(v1);
        vertexList
->push_back(v2);
        vertexList
->push_back(v3);
    }

    osg::Vec3Array* vertexList;
    
};

void getTriangles(osg::Drawable& drawable)
{
    osg::TriangleFunctor
<GetVertex> tf;
    tf.vertexList
=new osg::Vec3Array;

    drawable.accept(tf);

    for(osg::Vec3Array::iterator itr=tf.vertexList->begin();
        itr
!=tf.vertexList->end();
        itr
++)
    {
        osg::Vec3 vertex
=*itr;
        std::cout
<<vertex<<std::endl;
    }

    std::cout<<std::endl;
}

以下是完整的示例文件:

//  PrimitiveSet.cpp : 定义控制台应用程序的入口点。
//

#include 
" stdafx.h "
#include 
< iostream >

struct GetVertex
{
    
void operator() (const osg::Vec3& v1,const osg::Vec3& v2,const osg::Vec3& v3, boolconst 
    {
        vertexList
->push_back(v1);
        vertexList
->push_back(v2);
        vertexList
->push_back(v3);
    }

    osg::Vec3Array* vertexList;
    
};

void getTriangles(osg::Drawable& drawable)
{
    osg::TriangleFunctor
<GetVertex> tf;
    tf.vertexList
=new osg::Vec3Array;

    drawable.accept(tf);

    for(osg::Vec3Array::iterator itr=tf.vertexList->begin();
        itr
!=tf.vertexList->end();
        itr
++)
    {
        osg::Vec3 vertex
=*itr;
        std::cout
<<vertex<<std::endl;
    }

    std::cout<<std::endl;
}

osg::Node* createGeode()
{
    osg::Geode
* geode=new osg::Geode;
    osg::Geometry
* polyGeom = new osg::Geometry;
    osg::Vec3 myCoords[]
=
    {
        osg::Vec3(
0,1,0),
        osg::Vec3(
0,0,0),
        osg::Vec3(
1,1,0),
        osg::Vec3(
1,0,0),
        osg::Vec3(
2,1,0),
        osg::Vec3(
2,0,0)
    };

    int numCoords = sizeof(myCoords)/sizeof(osg::Vec3);
    osg::Vec3Array
* vertices = new osg::Vec3Array(numCoords,myCoords);
    polyGeom
->setVertexArray(vertices);
    polyGeom
->addPrimitiveSet(new osg::DrawArrays(osg::PrimitiveSet::QUAD_STRIP,0,numCoords));
    geode
->addDrawable(polyGeom);
    getTriangles(
*polyGeom);
    
return geode;
}

int _tmain(int argc, _TCHAR* argv[])
{
    
//Set up viewer
    osgViewer::Viewer viewer;
    osg::ref_ptr
<osg::GraphicsContext::Traits> traits=new osg::GraphicsContext::Traits;
    traits
->x=200;
    traits
->y=200;
    traits
->width=800;
    traits
->height=600;
    traits
->windowDecoration=true;
    traits
->doubleBuffer=true;
    traits
->sharedContext=0;
    
    osg::ref_ptr
<osg::GraphicsContext> gc=osg::GraphicsContext::createGraphicsContext(traits.get());
    osg::ref_ptr
<osg::Camera> camera=new osg::Camera;
    
//osg::Camera camera=new osg::Camera;
    camera->setGraphicsContext(gc.get());
    camera
->setViewport(new osg::Viewport(0,0,traits->width,traits->height));
    camera
->setDrawBuffer(GL_BACK);
    camera
->setReadBuffer(GL_BACK);
    osgGA::TrackballManipulator
* tm=new osgGA::TrackballManipulator;
    
    viewer.setCameraManipulator(tm);
    
    viewer.addSlave(camera.
get());

    //Set up root node
    osg::ref_ptr<osg::Group> root=new osg::Group;

    root->addChild(createGeode());
    

    //Start show!
    viewer.setSceneData(root.get());
    viewer.realize();

    
    while(!viewer.done())
    {
        viewer.frame();
    }
}

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

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

相关文章

自由与限制

#自由与限制##自由 前端&#xff0c;包括html-css-javascript三大基础技术&#xff0c;都是非常能体验出开放自由的。具体说来&#xff0c;因为这三大技术都是以标准先行&#xff0c;html-css的标准由w3c组织制定&#xff0c;js的标准由ECMA制定&#xff0c;都是只有标准&#…

理解GL_TRIANGLE_STRIP等绘制三角形序列的三种方式

GL_TRIANGLE_STRIP绘制三角形方式很多时候令人疑惑&#xff0c;在这里对其运作机理进行解释。 一般情况下有三种绘制一系列三角形的方式&#xff0c;分别是GL_TRIANGLES、GL_TRIANGLE_STRIP和GL_TRIANGLE_FAN。 如下图所示&#xff1a; GL_TRIANGLES是以每三个顶点绘制一个三…

Swift的笔记和参考

好久没来了&#xff0c;趁着新语言Swift发布&#xff0c;继续钻研中&#xff01; Create Class 创建类 &#xff08;重载效果&#xff09; // Create Class 创建类 class MyClass {// Properties 成员变量init() {// Constructor 构造函数}// Method 成员方法func doIt() {prin…

java正则过滤js_JS/Java正则表达式验证

校验密码&#xff1a;6-15位var pwdreg /^[0-9A-Za-z_,.!#$%^*]{6,15}$/;pwdreg.test(phonenum)校验手机号var mobilereg /^0?1[3|4|5|7|8][0-9]\d{8}$/;mobilereg.test(phonenum)只能输入数字(年龄大小)只能输入中文、英文、数字、空格(微信中&#xff0c;关键字&#xff0…

UE4中的字符串转换

虚幻4学习---UE4中的字符串转换(文章来自于UE4官方文档) String Conversions: FString To FName FString To Int32 Float To FString FArrayReaderPtr To FString TArray<uint8> To FString FString To char * --- ( TCHAR_TO_ANSI() ) FString To TCHAR * FString …

DeDeCMS后台批量修改替换sql语句大全

有时候后台文章内容、标题或者锚文本出错&#xff0c;需要修改批量修改&#xff0c;那么就需要用dedecms的sql语句进行批量修改了。 利用dedecms后台SQL命令行工具批量修改内容&#xff0c;路径和超链接等信息。语句 DEDECMS SQL命令批量替换1.更改文章中的内容update dede_add…

jpa mysql乐观锁_【快学springboot】8.JPA乐观锁OptimisticLocking

介绍当涉及到企业应用程序时&#xff0c;正确地管理对数据库的并发访问是至关重要的。为此&#xff0c;我们可以使用Java Persistence API提供的乐观锁定机制。它导致在同一时间对同一数据进行多次更新不会相互干扰。为了使用OptimisticLocking&#xff0c;我们需要一个实体(En…

UE4 连接自定义服务器

UE4 dedicated server是一个很好的游戏服务器,但是对于大厅和聊天服务器来说,我们要自行开发,通过UE4的socket组建很方便的和其他服务器建立连接 创建Socket FSocket* Socket ISocketSubsystem::Get(PLATFORM_SOCKETSUBSYSTEM)->CreateSocket(NAME_Stream, TEXT("de…

学习笔记-java编程-交通灯管理器设计流程。

先抛出需求&#xff1a; 异步随机生成按照各个路线行驶的车辆。例如&#xff1a; 由南向而来去往北向的车辆 ---- 直行车辆 由西向而来去往南向的车辆 ---- 右转车辆 由东向而来去往南向的车辆 ---- 左转车辆 。。。 信号灯忽略黄灯&#xff0c;只考虑红灯和绿灯。应考虑左转车…

php mysql 库存变负数_php解决秒杀并发入库导致的库存负数

我们知道数据库处理sql是一条条处理的&#xff0c;假设购买商品的流程是这样的&#xff1a;sql1:查询商品库存1 if(库存数量 > 0)2 {3 //生成订单4 //库存-15 >当没有并发时&#xff0c;上面的流程看起来是如此完美&#xff0c;假设同时两个人下单&#xff0c;而库存只有…

Codeforces Testing Round #10 A. Forgotten Episode

水题&#xff0c;注意数据范围 #include <iostream> using namespace std;int main(){long long n,a;cin >> n;long long sum (n*(n1))>>1;for(int i 0 ; i < n-1 ; i){cin >>a;sum -a;}cout<<sum<<endl; } 转载于:https://www.cnblo…

UE4--多线程的实现方式

首先查阅了WIKI中能找到Rama大神的两篇文章&#xff0c;讲了两个开线程的方式&#xff1a; https://wiki.unrealengine.com/Multi-Threading:_Task_Graph_System https://wiki.unrealengine.com/Multi-Threading:_How_to_Create_Threads_in_UE4 TaskGraph与FRunnable的比较 一…

python if else格式_Python if else条件语句详解

我们看到的代码都是顺序执行的&#xff0c;也就是先执行第1条语句&#xff0c;然后是第2条、第3条……一直到最后一条语句&#xff0c;这称为顺序结构。但是对于很多情况&#xff0c;顺序结构的代码是远远不够的&#xff0c;比如一个程序限制了只能成年人使用&#xff0c;儿童因…

UE4 HTC VIVE - 番外篇 - 局域网联机(一)

--------------------引擎环境配置文件修改与项目在线模式启动修改--------------------1&#xff09;我们就直接用默认名创建一个第三人称项目Paste_Image.png2&#xff09;右键资源栏&#xff0c;创建一个新的C类创建一个C类选择不继承任何UE提供的基类3&#xff09;打开【解…

UIImageView 圆角

UIImageView *image [[UIImageView alloc]init]; image.layer.cornerRadius 10.0;//如果想变圆形&#xff0c;设置这个值为宽度的一半 image.layer.masksToBounds YES;转载于:https://www.cnblogs.com/qianyindichang/p/3779823.html

java 参数 string_java(String和StringBuffer分别作为参数传递)

package com.day8.StringBuffer;public class Demo7StringBuffer {/***常见对象(String和StringBuffer分别作为参数传递)* A:形式参数问题* String作为参数传递* StringBuffer作为参数传递* B:案例演示* String和StringBuffer分别作为参数传递问题*( 基本数据类型的值传递&…

UE4 HTC VIVE - 番外篇 - 局域网联机(二)

开始之前先说一下网游中服务器与客户端的大致关系&#xff1a;网络游戏中各段关系图客户端职责&#xff1a;1&#xff09;接收玩家的输入翻译得到【玩家指令】上传服务器&#xff1b;2&#xff09;接收服务器下发的【游戏指令】并将其实现服务器职责&#xff1a;1&#xff09;接…

启动文件、简单的消息框

C中打开文件的方法。 1.system&#xff08;&#xff09;; 函数原型&#xff1a; int system(char *command); 作用&#xff1a;发出一个DOS命令。 特点&#xff1a;该函数是同步的&#xff0c;不灵活。只是能够改为system("start XXX"); 2.WinExec&#xff08;&#…

UE4 HTC VIVE - 番外篇 - 局域网联机三

环境设置和检测1.打开引擎下的在线调试引擎在线模式开启2.打开项目的在线调试项目在线模式开启3.检查机器网段我们需要对每台要加入局域网的机器都进行网段检测&#xff0c;这是能客户端能找到主机广播的前提&#xff0c;否则无法搜到主机Windows键R键输入CMD输入ipconfig检查I…

java异常判断_Java异常退出条件的判断示例代码

无论是功能性代码还是算法性代码&#xff0c;程序都是一系列流程的合集既然是流程就分为&#xff1a;一般流程和异常流程&#xff1b;一般流程保证了基本功能&#xff1b;异常流程则是对程序稳定性的保证&#xff0c;不能因为一些非法输入&#xff0c;项目就挂了&#xff1b;注…