赞
踩
推荐:将 NSDT场景编辑器 加入你的3D开发工具链
网格对象是由三角形和顶点组成的形状
在 CHAI3D 中,多边形网格是定义多面体对象形状的顶点和三角形的集合。
顶点是一个位置以及其他信息,例如颜色、法线矢量和纹理坐标。由直线连接的两个顶点成为一条边。三个顶点通过三条边相互连接,定义一个三角形,这是欧几里得空间中最简单的多边形。
网格对象可以平移、旋转或调整大小,并具有分配给它们的材质和纹理属性。 以下代码演示如何创建由 4 个顶点和 2 个三角形组成的带纹理的正方形。
using namespace chai3d;
// create a mesh
object = new cMesh();
// create a texture map
object->m_texture = cTexture2d::create();
// load a texture file
object->m_texture->loadFromFile("myTexture.jpg");
// enable texture mapping
object1->setUseTexture(true);
// set material to white
object1->m_material->setWhite();
// create vertices
int vertex0 = object0->newVertex();
int vertex1 = object0->newVertex();
int vertex2 = object0->newVertex();
int vertex3 = object0->newVertex();
// set position, surface normal, and texture coordinate for vertex 0
object0->m_vertices->setLocalPos(vertex0,-0.1,-0.1, 0.0);
object0->m_vertices->setNormal(vertex0, 0.0, 0.0, 1.0);
object0->m_vertices->setTexCoord(vertex0, 0.0, 0.0);
// set position, surface normal, and texture coordinate for vertex 1
object0->m_vertices->setLocalPos(vertex1, 0.1,-0.1, 0.0);
object0->m_vertices->setNormal(vertex1, 0.0, 0.0, 1.0);
object0->m_vertices->setTexCoord(vertex1, 1.0, 0.0);
// set position, surface normal, and texture coordinate for vertex 2
object0->m_vertices->setLocalPos(vertex2, 0.1, 0.1, 0.0);
object0->m_vertices->setNormal(vertex2, 0.0, 0.0, 1.0);
object0->m_vertices->setTexCoord(vertex2, 1.0, 1.0);
// set position, surface normal, and texture coordinate for vertex 3
object0->m_vertices->setLocalPos(vertex3,-0.1, 0.1, 0.0);
object0->m_vertices->setNormal(vertex3, 0.0, 0.0, 1.0);
object0->m_vertices->setTexCoord(vertex3, 0.0, 1.0);
// create two triangles by assigning their vertex IDs
object0->m_triangles->newTriangle(vertex0, vertex1, vertex2);
object0->m_triangles->newTriangle(vertex0, vertex2, vertex3);
材料属性定义表面如何反射光线。他们通过定义颜色值来实现这一点,其方式与灯光大致相同。材质具有漫射(直射)光、环境(散射)光和镜面反射(反射)光的颜色值,但这些值定义了光的分量如何从材质表面反射。更多信息可以在有关材料的部分找到。
材质属性可以启用或禁用,并且需要在世界中至少使用一个光源才能使对象可见。
using namespace chai3d;
// create a mesh
cMesh* mesh = new cMesh();
// add mesh to world
world->addChild(mesh);
// build mesh using a cylinder primitive
cCreatePipe(mesh,
0.15,
0.05,
0.06,
32,
1,
cVector3d(-0.05,-0.20, 0.0),
cMatrix3d(cDegToRad(0), cDegToRad(0), cDegToRad(170), C_EULER_ORDER_XYZ)
);
// enable material property
cylinder->setUseMaterial(true);
// (1) set material by simply assigning a color name
mesh->m_material->setBlueCornflower();
// (2) or set material by assigning a color values for each component
mesh->m_material->m_ambient.set(0.2, 0.1, 0.1);
mesh->m_material->m_diffuse.set(0.6, 0.3, 0.3);
mesh->m_material->m_specular.set(1.0, 1.0, 1.0);
mesh->m_material->setShininess(10);
可以为网格的每个顶点指定不同的颜色值。如果三角形的三个顶点共享不同的颜色,则三角形上任何点的颜色都是通过三种顶点颜色的插值来计算的。
三角形和顶点颜色
如果启用了材料属性,顶点颜色将与材料颜色组合在一起。通常将材质设置为白色并为不同的顶点分配颜色以产生所需的光晕。如果禁用材质属性,则忽略所有光源,并使用纯顶点颜色来渲染对象。
using namespace chai3d;
// create a mesh
cMesh* mesh = new cMesh();
// add mesh to world
world->addChild(mesh);
// create three new vertices
int vertex0 = mesh->newVertex();
int vertex1 = mesh->newVertex();
int vertex2 = mesh->newVertex();
// set position of each vertex
mesh->m_vertices->setLocalPos(vertex0, 0.0, 0.0, 0.0);
mesh->m_vertices->setLocalPos(vertex1, 0.0, 1.0, 0.0);
mesh->m_vertices->setLocalPos(vertex2, 1.0, 1.0, 0.0);
// assign color value to each vertex
cColorf color(1.0, 0.2, 0.2);
mesh->m_vertices->setColor(vertex0, color);
mesh->m_vertices->setColor(vertex1, color);
mesh->m_vertices->setColor(vertex2, color);
// create new triangle from vertices
mesh->newTriangle(vertex0, vertex1, vertex2);
// enable vertex colors
mesh->setUseVertexColors(true);
// compute surface normals
mesh->computeAllNormals();
纹理映射
纹理贴图将应用(映射)到形状或多边形的表面。此过程类似于将图案纸应用于纯白框。多边形中的每个顶点都通过显式分配或过程定义分配一个纹理坐标(在 2d 情况下也称为 UV 坐标)。然后,在多边形的表面插值图像采样位置,以生成视觉结果,该结果似乎比使用有限数量的多边形可以实现的更丰富。
要了解如何在 CHAI3D 中创建纹理,请参阅本页顶部的第一个示例。
三角形可以在实心或线模式下渲染。
(左)固体模式。(右)线框模式。
using namespace chai3d;
// enable wireframe rendering mode
mesh->setWireMode(true);
// enable solid rendering mode
mesh->setWireMode(false);
要同时渲染边和三角形,CHAI3D 提供了显示三角形的选项,并覆盖根据相邻三角形之间所需的最小角度选择的边子集。
(左)实心和边缘。(右)仅边缘。
using namespace chai3d;
// compute all edges for which adjacent triangles have more than 40 degrees angle
mesh->computeAllEdges(40);
// set line width of edges and color
cColorf color;
color.setBlack();
mesh->setEdgeProperties(1, color);
// (1) enable surfaces and edges
mesh->setShowTriangles(true);
mesh->setShowEdges(true);
// (2) enable edges only
mesh->setShowTriangles(false);
mesh->setShowEdges(true);
背面剔除确定网格对象的三角形是否可见。这是图形管道中的一个步骤,用于测试三角形中的顶点在投影到屏幕上时是否按顺时针顺序显示。
using namespace chai3d;
// enable culling
object->setUseCulling(true);
如果启用了面部剔除,但投影在屏幕上的多边形具有逆时针缠绕,则它已旋转为背离相机,不会被绘制。
该过程通过减少程序绘制的多边形数量,使渲染对象更快、更高效。例如,在城市街道场景中,通常不需要在建筑物背对摄影机的侧面绘制多边形;它们被面向相机的侧面完全遮挡。
通常,如果背面剔除仅包含封闭和不透明的几何体,则可以假定在渲染场景中不会产生可见的伪影。在包含透明多边形的场景中,通过 Alpha 合成过程,后向多边形可能会变得可见。
渲染相机的半透明镜头
透明的物理材质显示其后面的对象未被遮挡,并且不会从其表面反射光线。透明玻璃是一种近乎透明的材料。虽然玻璃允许大多数光线不受遮挡地通过,但实际上它也反射了一些光。完全透明的材料是完全看不见的。
半透明物理材质显示其后面的对象,但这些对象被半透明材质遮挡。此外,半透明材质会反射照射到它的一些光线,使材料可见。半透明材料的物理示例包括透明布料、薄塑料和烟熏玻璃。
透明和半透明通常是同义词。既不透明也不半透明的材料是不透明的。
混合是OpenGL的机制,用于将帧缓冲中已有的颜色与传入基元的颜色组合在一起。然后,这种组合的结果将存储回帧缓冲器中。混合经常用于模拟半透明的物理材质。一个例子是渲染汽车的烟熏玻璃挡风玻璃。驾驶员和内饰仍然可见,但它们被烟熏玻璃的深色所掩盖。
渲染网格的顶点法线
在 CHAI3D 中启用照明时,法线矢量用于确定在指定顶点或表面上接收的光量。如果尚未定义曲面法线,则可以通过取该三角形两条边的向量叉积来计算每个三角形的法线。 出于调试目的,可以使用以下调用显示曲面法线:
using namespace chai3d;
// compute all surface normals
mesh->computeAllNormals();
// set normal properties for display
cColorf color;
color.setOrangeTomato();
mesh->setNormalsProperties(0.01, color);
// display surface normals
mesh->setShowNormals(true);
由于网格对象每个对象仅包含一个材质和纹理属性,因此创建使用材质和纹理集合的网格需要使用称为 cMultiMesh 的不同类,该类将网格列表连接在一起。通过在单独的网格对象中根据三角形的材质和纹理属性来组织三角形,可以有效地构建和渲染具有丰富属性集的复杂对象。
using namespace chai3d;
// create a multi mesh object
cMultiMesh* multiMesh = new cMultiMesh();
// add multi mesh object to world
world->addChild(multiMesh);
// create a first mesh
cMesh* mesh1 = multiMesh->newMesh();
// create a second mesh
cMesh* mesh2 = multiMesh->newMesh();
CHAI3D 支持 3DS、OBJ 和 SLT 文件格式来导入或导出模型。如果您希望导入不同格式的文件,可以使用Blender或Visual Enterprise Author(以前称为Deep Exploration)等应用程序来编辑文件并将其转换为所需的格式。
using namespace chai3d;
// load 3DS model file
multiMesh->loadFromFile("myModel.3ds");
// load OBJ model file
multiMesh->loadFromFile("myModel.obj");
// load STL model file
multiMesh->loadFromFile("myModel.stl");
// save model to 3DS file
multiMesh->saveToFile("myModel.3ds");
// save model to OBJ file
multiMesh->saveToFile("myModel.obj");
// save model to STL file
multiMesh->saveToFile("myModel.stl");
碰撞检测器用于计算射线与组成网格(或多网格)的任何三角形之间的交点。为了使此操作快速进行,CHAI3D 使用边界体积层次结构或 BVH。BVH 是一组几何对象上的树结构。所有几何对象都包裹在构成树的叶节点的边界体积中。然后将这些节点分组为小集合,并包含在较大的边界体积中。反过来,这些也以递归方式分组并封闭在其他较大的边界体积中,最终导致树结构在树的顶部具有单个边界体积。边界体积层次结构用于有效地支持对几何对象集的多项操作,例如在触觉交互计算、光线追踪或鼠标选择的碰撞检测中。
轴对齐边界体积层次结构
在 CHAI3D 中,必须在定义三角形后生成边界体积层次结构。如果修改了三角形或移动了顶点,则必须再次计算层次结构。更新碰撞树时,正确包含世界上使用的最大触觉点的半径非常重要。此信息用于创建足够大的包络,以检测与半径小于或等于定义阈值的接触点的任何交互。
using namespace chai3d;
// create axis aligned bounding volume hierarchy
multiMesh->createAABBCollisionDetector(toolRadius);
网格对象的触觉渲染
为了触觉渲染网格对象,CHAI3D使用由Ruspini和Khatib开发的虚拟“手指代理”算法,类似于Zilles和Salisbury提出的“上帝对象”。虚拟代理是替代虚拟环境中物理手指或探测器的代表性对象。
Ruspini和Khatib的手指代理算法
上图说明了虚拟代理的运动,因为触觉设备的位置发生了变化。代理的运动试图始终朝着目标移动。当畅通无阻时,代理直接向目标移动。当代理遇到障碍物时,无法直接移动,但代理仍可以通过沿一个或多个约束曲面移动来缩短到目标的距离。选择运动以局部最小化到目标的距离。当代理无法减少其与目标的距离时,它将停止在本地最小配置处。力是通过对触觉设备和代理位置之间的虚拟弹簧进行建模来计算的。弹簧的刚度在材料属性中定义。
using namespace chai3d;
// create collision detector
mesh->createAABBCollisionDetector(toolRadius);
// set haptic properties
mesh->m_material->setStiffness(1000);
mesh->m_material->setStaticFriction(0.3);
mesh->m_material->setDynamicFriction(0.4);
还可以通过材料属性等级激活和调整其他触觉效果。触觉效果使用势场算法,任何通用形状对象也支持这些算法。请注意,网格对象目前不支持磁性效果。
using namespace chai3d;
// create a haptic stick-slip effect
mesh->createEffectStickSlip();
// set haptic properties
mesh->m_material->setStickSlipForceMax(5.0);
mesh->m_material->setStickSlipStiffness(1000);
3D建模学习工作室翻译整理,转载请标明出处!
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。