开启掘金成长之旅!这是我参与「掘金日新计划 · 2 月更文挑战」的第 18 天
1 Geode
osg::Geode是OSG中的叶节点,它用于保存几何信息以便渲染。同时,作为叶节点,它就不会再包含子节点。在应用程序中,所有相关的几何体的渲染都必须与Geode节点相关联。在osg::Geode类中,也提供了addDrawable()函数来关联应用程序中需要渲染的几何体信息。
2 Billboard 节点
Billboard的继承关系图如图所示。
从图可以看出,Billboard 节点继承自 Geode节点,因此它也是一个叶节点,不可再包含其他的子节点,只能像叶节点那样通过添加Drawable来绘制信息。Billboard 有下面3种模式:
enum Mode
{
POINT_ROT_EYE,//绕视点
POINT_ROT_WORLD,//绕世界坐标
AXIAL_ROT//绕轴
}
Billboard,从英文名可以看出是一种布告板技术,这是一种非常实用的技术,很多特殊效果都利用它来实现。布告板实现的原理是:将图形绘制在朝向视点的多边形表面上,根据视点的观察方向来确定多边形的方向,随着观察角度的变化,多边形的方向也随之变换。
osg::Billboard 布告板与 Alpha纹理和动画融合技术相结合可以实现很多没有实心表面的现象,如后面会讲到的粒子系统中的烟、火、雾、爆炸、雨、雪及云朵等。
在OSG中使用了几种常见的布告板技术,主要包括面向世界的布告板和轴向布告板。其中,面向世界的布告板包括面向视平面的布告板和面向视点的布告板技术。
下图表示了这两种技术的区别。
在轴向布告板技术中,经过纹理贴图的多边形通常不直接朝向视点,而是允许多边形围绕某个固定世界空间轴旋转,但调整多边形使其在此范围内尽可能朝向视点。
这种布告板技术通常用来在场景中显示树木,它不是用实心表面来显示树木,而是使用一个四边形表面纹理显示树木。
此时,一个树木布告板以树干为轴心作为世界的向上向量,保持世界的向上向量固定,视点观察方向作为树木旋转的第二个调整向量,一旦视点旋转矩阵生成,就可以对树木位置进行旋转变换。
但轴向布告板技术存在一个问题:当观察者处于空中飞行模式下,从树木顶部飞过并且垂直向下看时,看到的树木就像一个切面一样,非常难看。似乎对于这样的问题,前面的功夫都白费了。可以使用细节层次LOD模型或者替代模型弥补效果,或者可以使用一个球形树木表面来改善。
3 布告板示例
通过上面的解释,读者应该已经明白了什么是Billboard技术。现在就通过一个使用Billboard的例子来讲解如何在程序中使用它。
在下面的程序中会使用到如何绘制几何体、PositionAttitudeTransform 节点以及纹理贴图,在后面的章节都会讲到这些,读者主要的任务是学习如何使用Billboard技术。
代码
#include <osgViewer/Viewer>
#include <osg/Node>
#include <osg/Geode>
#include <osg/Group>
#include <osg/PositionAttitudeTransform>
#include <osgDB/ReadFile>
#include <osgDB/WriteFile>
#include <osgUtil/Optimizer>
#include "utility.h"
osg::ref_ptr<osg::Node> createBillboardTree(osg::ref_ptr<osg::Image> image)
{
// 创建四边形
osg::ref_ptr<osg::Geometry> geometry = new osg::Geometry();
// 设置顶点
osg::ref_ptr<osg::Vec3Array> v = new osg::Vec3Array;
v->push_back(osg::Vec3(-0.5f, 0.0f, -0.5f));
v->push_back(osg::Vec3(0.5f, 0.0f, -0.5f));
v->push_back(osg::Vec3(0.5f, 0.0f, 0.5f));
v->push_back(osg::Vec3(-0.5f, 0.0f, 0.5f));
geometry->setVertexArray(v.get());
// 设置法线
osg::ref_ptr<osg::Vec3Array> normal = new osg::Vec3Array;
normal->push_back(osg::Vec3(1.0f, 0.0f, 0.0f) ^ osg::Vec3(0.0f, 0.0f, 1.0f));
geometry->setNormalArray(normal.get());
geometry->setNormalBinding(osg::Geometry::BIND_OVERALL);
// 设置纹理坐标
osg::ref_ptr<osg::Vec2Array> vt = new osg::Vec2Array;
vt->push_back(osg::Vec2(0.0f, 0.0f));
vt->push_back(osg::Vec2(1.0f, 0.0f));
vt->push_back(osg::Vec2(1.0f, 1.0f));
vt->push_back(osg::Vec2(0.0f, 1.0f));
geometry->setTexCoordArray(0, vt.get());
// 绘制四边形
geometry->addPrimitiveSet(new osg::DrawArrays(osg::PrimitiveSet::QUADS, 0, 4));
if (image.get())
{
// 状态属性对象
osg::ref_ptr<osg::StateSet> stateset = new osg::StateSet;
// 创建一个Texture2D属性对象
osg::ref_ptr<osg::Texture2D> texture = new osg::Texture2D;
// 关联image
texture->setImage(image.get());
// 关联Texture2D纹理对象,第三个参数默认是ON
stateset->setTextureAttributeAndModes(0, texture, osg::StateAttribute::ON);
// 启用混合
stateset->setMode(GL_BLEND, osg::StateAttribute::ON);
// 关闭光照,如果开启光照,纹理很暗
stateset->setMode(GL_LIGHTING, osg::StateAttribute::OFF);
geometry->setStateSet(stateset.get());
}
// 创建Billboard对象1
osg::ref_ptr<osg::Billboard> billboard1 = new osg::Billboard;
// 设置旋转模式为绕视点
billboard1->setMode(osg::Billboard::POINT_ROT_EYE);
// 添加Drawable,并设置其位置,默认位置为osg::Vec3(0.0f, 0.0f, 0.0f)
billboard1->addDrawable(geometry.get(), osg::Vec3(3.0f, 0.0f, 0.0f));
// 创建Billboard对象2
osg::ref_ptr<osg::Billboard> billboard2 = new osg::Billboard;
// 设置旋转模式为绕轴转,因此还要设置旋转轴
billboard2->setMode(osg::Billboard::AXIAL_ROT);
// 设置旋转轴
billboard2->setAxis(osg::Vec3(0.0f, 0.0f, 1.0f));
billboard2->addDrawable(geometry.get(), osg::Vec3(6.0f, 0.0f, 0.0f));
osg::ref_ptr<osg::Group> billboard = new osg::Group;
billboard->addChild(billboard1.get());
billboard->addChild(billboard2.get());
return billboard.get();
}
int main()
{
osg::ref_ptr<osgViewer::Viewer> viewer = new osgViewer::Viewer();
osg::ref_ptr<osg::Group> root = new osg::Group();
// 读取图像
osg::ref_ptr<osg::Image> image = osgDB::readImageFile(GetCurrentPath() + "\\Data\\fengjing1.jpeg");
// 缩放达到合适大小
osg::ref_ptr<osg::PositionAttitudeTransform> pat = new osg::PositionAttitudeTransform();
pat->setScale(osg::Vec3(5.0f, 5.0f, 5.0f));
pat->addChild(createBillboardTree(image.get()));
//osg::ref_ptr<osg::Group> pat = new osg::Group;
//pat->addChild(createBillboardTree(image.get()));
// 读取cow模型,进行对比
osg::ref_ptr<osg::Node> node = osgDB::readNodeFile(GetCurrentPath() + "\\Data\\cow.osg");
root->addChild(node);
root->addChild(pat.get());
osgUtil::Optimizer optimizer;
optimizer.optimize(root.get());
setWindowSize(viewer.get(), 600, 400, "Billboard");
viewer->setSceneData(root.get());
viewer->realize();
viewer->run();
return 0;
}
效果图
动画效果