在基础篇中我们讲过空间数据可视化之Entity实体类,今天我们介绍另外一个比较接近渲染引擎底层的类Primitive,虽然两者都可用于绘制同样的几何图形,但考虑到性能问题,我们更推荐您使用Primitive类实现。在使用Primitive API之前,您最好具备WebGL基础知识,如果对WebGL不是太了解,建议先学习《WebGL编程指南》这本书。

Primitive介绍

1. Primitive组成
Primitive由两部分组成:几何形状(Geometry)和外观(Appearance)。几何形状定义了Primitive的结构,例如三角形、多边形、折线、点、标签等;外观则定义了Primitive的着色或渲染(Shading),包括GLSL(OpenGL着色语言,OpenGL Shading Language)顶点着色器和片元着色器( vertex and fragment shaders),以及渲染状态(render state)。

2. Primitive优劣势
相对于Entity,使用Primitive具有以下优势:
(1)性能:绘制大量Primitive时,可以将其合并为单个Geometry以减轻CPU负担、更好的使用GPU。合并Primitive由web worker线程执行,以保持UI响应性;
(2)灵活性:Geometry与Appearance 解耦,两者可以分别进行修改;
(3)低级别访问:易于编写GLSL顶点、片段着色器、使用自定义的渲染状态 。

同时,也具有以下劣势:
(1)需要编写更多的代码,并且对图形编程有更深刻的理解,尤其是OpenGL知识;
(2)需要对组合几何形状对于静态数据有效,而对于动态数据则不一定有效。

3.几何图形绘制方式
以下是通过Entity和Primitive两种方式绘制矩形图形的方法:

// Entity方式
viewer.entities.add({rectangle : {coordinates : Cesium.Rectangle.fromDegrees(-100.0, 20.0, -90.0, 30.0),material : new Cesium.StripeMaterialProperty({evenColor: Cesium.Color.WHITE,oddColor: Cesium.Color.BLUE,repeat: 5})}
});// Primitive方式
var instance = new Cesium.GeometryInstance({geometry : new Cesium.RectangleGeometry({rectangle : Cesium.Rectangle.fromDegrees(-100.0, 20.0, -90.0, 30.0),vertexFormat : Cesium.EllipsoidSurfaceAppearance.VERTEX_FORMAT})
});scene.primitives.add(new Cesium.Primitive({geometryInstances : instance,appearance : new Cesium.EllipsoidSurfaceAppearance({material : Cesium.Material.fromType('Stripe')})
}));

几何Geometry

1.支持的几何类型

从基础篇的Entity篇幅我们知道,Entity支持的图形类型是以Graphics结尾的,一共有17种类型。而Primitive支持的几何类型则是以Geometry结尾的,和Entity除了结尾命名不一样之外,Cesium中还提供了独有的点形状PointPrimitive和一些形状的集合,包括PointPrimitiveCollection、BillboardCollection、LabelCollection、PolylineCollection。支持的形状如下图所示:

添加简单的点图元集合方法如下:

 // Create a pointPrimitive collection with two pointsvar points = scene.primitives.add(new Cesium.PointPrimitiveCollection({modelMatrix: Cesium.Matrix4.IDENTITY,debugShowBoundingVolume: false,// OPAQUE 完全不透明;TRANSLUCENT 完全透明;OPAQUE_AND_TRANSLUCENT 不透明和半透明blendOption: Cesium.BlendOption.OPAQUE_AND_TRANSLUCENT,}));// add PointPrimitivepoints.add({position: Cesium.Cartesian3.fromDegrees(-75.59777, 40.53883, 1000.0),color: Cesium.Color.YELLOW,});points.add({position: Cesium.Cartesian3.fromDegrees(-74.59777, 40.53883, 1000.0),color: Cesium.Color.CYAN,});

2.贴地或贴模型特性
跟Entity类似,Primitive也支持贴地或贴模型的特性,但不一样的是,Primitive是通过classificationType属性控制的。其中GroundPolylineGeometry、GroundPolylinePrimitive结合实现贴地线;GroundPrimitive实现贴地几何形状,包括CircleGeometry、CorridorGeometry、EllipseGeometry、PolygonGeometry、RectangleGeometry;ClassificationPrimitive可实现贴地或贴模型,包括BoxGeometry、CylinderGeometry、EllipsoidGeometry、PolylineVolumeGeometry、SphereGeometry几何形状。下面为一简单的贴模型示例:

 scene.primitives.add(new Cesium.ClassificationPrimitive({geometryInstances: new Cesium.GeometryInstance({geometry: Cesium.BoxGeometry.fromDimensions({vertexFormat: Cesium.PerInstanceColorAppearance.VERTEX_FORMAT,dimensions: new Cesium.Cartesian3(8.0, 5.0, 8.0),}),modelMatrix: modelMatrix,attributes: {color: Cesium.ColorGeometryInstanceAttribute.fromColor(new Cesium.Color(1.0, 0.0, 0.0, 0.5)),show: new Cesium.ShowGeometryInstanceAttribute(true),},id: "volume",}),classificationType: Cesium.ClassificationType.CESIUM_3D_TILE,}));

3.组合几何
当我们使用一个图元绘制多个静态几何图形时,我们就会看到性能的优势。组合多个GeometryInstances 为一个Primitive可以极大的提高性能,以下示例绘制了2592个颜色各异的矩形,并覆盖整个地球。

var viewer = new Cesium.Viewer('cesiumContainer');
var scene = viewer.scene;var instances = [];for (var lon = -180.0; lon < 180.0; lon += 5.0) {for (var lat = -85.0; lat < 85.0; lat += 5.0) {instances.push(new Cesium.GeometryInstance({geometry : new Cesium.RectangleGeometry({rectangle : Cesium.Rectangle.fromDegrees(lon, lat, lon + 5.0, lat + 5.0),vertexFormat: Cesium.PerInstanceColorAppearance.VERTEX_FORMAT}),attributes : {color : Cesium.ColorGeometryInstanceAttribute.fromColor(Cesium.Color.fromRandom({alpha : 0.5}))}}));}
}scene.primitives.add(new Cesium.Primitive({geometryInstances : instances,appearance : new Cesium.PerInstanceColorAppearance()
}));

4.实例化几何
实例化可用于在场景的不同部分定位、缩放和旋转相同的几何体。多个实例可以引用相同的Geometry,并且每个实例可以具有不同的modelMatrix。这允许我们只需计算一次几何图形,并多次重复使用它。

下面的示例创建一个EllipsoidGeometry和两个实例。每个实例都引用相同的椭球几何体,但使用不同的modelMatrix放置它, 从而导致一个椭球位于另一个之上。

var viewer = new Cesium.Viewer('cesiumContainer');
var scene = viewer.scene;
var ellipsoidGeometry = new Cesium.EllipsoidGeometry({vertexFormat : Cesium.PerInstanceColorAppearance.VERTEX_FORMAT,radii : new Cesium.Cartesian3(300000.0, 200000.0, 150000.0)
});var cyanEllipsoidInstance = new Cesium.GeometryInstance({geometry : ellipsoidGeometry,modelMatrix : Cesium.Matrix4.multiplyByTranslation(Cesium.Transforms.eastNorthUpToFixedFrame(Cesium.Cartesian3.fromDegrees(-100.0, 40.0)),new Cesium.Cartesian3(0.0, 0.0, 150000.0),new Cesium.Matrix4()),attributes : {color : Cesium.ColorGeometryInstanceAttribute.fromColor(Cesium.Color.CYAN)}
});var orangeEllipsoidInstance = new Cesium.GeometryInstance({geometry : ellipsoidGeometry,modelMatrix : Cesium.Matrix4.multiplyByTranslation(Cesium.Transforms.eastNorthUpToFixedFrame(Cesium.Cartesian3.fromDegrees(-100.0, 40.0)),new Cesium.Cartesian3(0.0, 0.0, 450000.0),new Cesium.Matrix4()),attributes : {color : Cesium.ColorGeometryInstanceAttribute.fromColor(Cesium.Color.ORANGE)}
});scene.primitives.add(new Cesium.Primitive({geometryInstances : [cyanEllipsoidInstance, orangeEllipsoidInstance],appearance : new Cesium.PerInstanceColorAppearance({translucent : false,closed : true})
}));

5.更新每个示例的属性
在将几何图形添加到Primitive中以后,仍然可以修改几何图形实例的某些属性:
(1)颜色:如果Primitive设置了PerInstanceColorAppearance外观,则可以修改ColorGeometryInstanceAttribute类型的颜色
(2)可见性:任何实例可以修改可见性

var viewer = new Cesium.Viewer('cesiumContainer');
var scene = viewer.scene;
var circleInstance = new Cesium.GeometryInstance({geometry : new Cesium.CircleGeometry({center : Cesium.Cartesian3.fromDegrees(-95.0, 43.0),radius : 250000.0,vertexFormat : Cesium.PerInstanceColorAppearance.VERTEX_FORMAT}),attributes : {color : Cesium.ColorGeometryInstanceAttribute.fromColor(new Cesium.Color(1.0, 0.0, 0.0, 0.5))},id: 'circle'
});
var primitive = new Cesium.Primitive({geometryInstances : circleInstance,appearance : new Cesium.PerInstanceColorAppearance({translucent : false,closed : true})
});
scene.primitives.add(primitive);setInterval(function() {var attributes = primitive.getGeometryInstanceAttributes('circle');attributes.color = Cesium.ColorGeometryInstanceAttribute.toValue(Cesium.Color.fromRandom({alpha : 1.0}));
},2000);

外观Apperance

Primitive由两个重要部分组成:几何图形实例、外观。一个Primitive可以有多个几何实例,但只能有一个外观。几何图形定义了结构,外观定义了每个像素如何被着色,外观可能直接使用材质(Material)。一个Primitive结构组成如下图所示:

同时,Cesium定义了以下外观:

外观定义了需要在GPU上执行的完整的GLSL顶点、片段着色器,通常不需要修改这一部分,除非需要定义自己的外观。外观还定义了完整的渲染状态,用于在绘制Primitive时控制GPU的状态,我们可以直接或者通过高层API来定义渲染状态,如“闭合(closed)”和“半透明(translucent)”,外观将转换为渲染状态。如右图所示:

//下面的外观可用于定义一个不可进入的不透明的盒子var appearance = new Cesium.PerInstanceColorAppearance({translucent: false,closed: true,});//下面的代码效果同上var anotherAppearance = new Cesium.PerInstanceColorAppearance({renderState: {depthTest: {enabled: true,},cull: {enabled: true,face: Cesium.CullFace.BACK,},},});

创建外观后,不能更改其renderState属性,但可以更改其material。我们还可以更改primitive的appearnace属性。。
大部分外观具有flat、faceForward属性,可以间接的控制GLSL着色器:
(1)flat:扁平化着色,不考虑光线的作用
(2)faceForward:布尔值,控制光照效果

着色器shader
shader即着色器,分为顶点着色器(Vertex Shader)、片元着色器(Fragment Shader)、几何着色器(Geometry shader)、计算着色器(Compute shader)、细分曲面着色器(Tessellation or hull shader),其中可编程的是顶点着色器和片元着色器。示意图如下:

在屏幕上绘制或显示一些物体时,这些物体的显示形式是图元(Primitive)或者网格(Mesh),比如一个贴在网格上的纹理角色。

几何和外观兼容性

并非所有外观都适用于所有几何图形。例如,EllipsoidSurfaceAppearance外观不适用于WallGeometry几何图形,因为墙不在球体的表面上。要使外观与几何图形兼容,它们必须具有匹配的顶点格式,这意味着几何图形必须具有外观所期待的输入数据。创建几何图形时可以提供vertexFormat。

几何图形的vertexFormat确定它是否可以与其他几何图形组合。两个几何图形不必是相同的类型,但它们需要匹配的顶点格式。为方便起见,外观要么具有vertexFormat属性,要么具有可作为几何体选项传入的VERTEX_FORMAT静态常量。

var geometry = new Ceisum.RectangleGeometry({vertexFormat : Ceisum.EllipsoidSurfaceAppearance.VERTEX_FORMAT// ...
});var geometry2 = new Ceisum.RectangleGeometry({vertexFormat : Ceisum.PerInstanceColorAppearance.VERTEX_FORMAT// ...
});var appearance = new Ceisum.MaterialAppearance(/* ... */);
var geometry3 = new Ceisum.RectangleGeometry({vertexFormat : appearance.vertexFormat// ...
});

微信扫描下方二维码,获取更多资讯。

微信公众号二维码

Cesium开发高级篇 | 01空间数据可视化之Primitive相关推荐

  1. Cesium开发基础篇 | 04空间数据可视化之Entity

    前面介绍了Cesium如何加载影像数据.地形数据.以及矢量数据,但是作为一个完整的三维系统,仅仅包括这些数据还是远远不够的.当然,还需要一些其他数据,比如空间可视化数据.三维数据数据等,今天我们先从空 ...

  2. Cesium开发高级篇 | 04粒子系统

    在三维GIS系统中,我们经常会看到一些常见的特效,比如雨雪雾天气.烟火.爆炸.喷泉等,这些都可以通过粒子系统实现,今天我就来带领大家学习一下Cesium中的粒子系统. 1.什么是粒子系统 粒子系统是一 ...

  3. Cesium开发高级篇 | 05场景后期处理

    为实现三维模型的更炫.更酷.更美观,Cesium在1.46的版本中新增了场景的后期处理(Post Processing)功能,包括模型描边.黑白图.明亮度调整.夜市效果.环境光遮蔽,也包括雷达扫描.原 ...

  4. Cesium开发工具篇 | 07回顾总结

    到目前为止,Cesium开发课程的篇幅已经学完四分之三了,大家如果有什么问题可以私信我或者扫描以下二维码进群沟通,若群二维码失效,可扫描最下方的公众号二维码,加我为好友并备注加群.有时会比较忙来不及回 ...

  5. Nginx 模块开发高级篇

    Nginx 模块开发高级篇 变量 综述 在Nginx中同一个请求需要在模块之间数据的传递或者说在配置文件里面使用模块动态的数据一般来说都是使用变量,比如在HTTP模块中导出了host/remote_a ...

  6. [洪流学堂]Hololens开发高级篇5:空间映射(Spatial mapping)

    本教程基于Unity2017.2及Visual Studio 2017 本教程编写时间:2017年12月16日 本文内容提要 空间映射让holograms了解周围环境,将真实世界和虚拟世界更好地结合在 ...

  7. [洪流学堂]Hololens开发高级篇3:语音(Voice)

    本教程基于Unity2017.2及Visual Studio 2017 本教程编写时间:2017年12月8日 本文内容提要 设计语音命令并针对Hololens语音引擎优化 让用户知道可以用什么语音命令 ...

  8. [洪流学堂]Hololens开发高级篇2:手势(Gesture)

    本教程基于Unity2017.2及Visual Studio 2017 本教程编写时间:2017年12月7日 本文内容提要 当跟踪到用户的手时提供反馈 使用导航手势旋转hologram 当用户的手要离 ...

  9. Cesium开发基础篇 | 02加载地形数据

    Cesium中的地形图层类 前面我们主要学习了cesium内置接口如何操作影像数据,但是在一些应用场景中我们需要操作地形数据,例如模拟逼真的三维场景.与高程相关的一些空间分析和计算等.Cesium提供 ...

最新文章

  1. iMeta | 中科院李小方等膳食甘草促进小鼠镉解毒并调节肠道菌群代谢
  2. java client类_Jmeter中自定义JavaSamplerClient类的编写
  3. 第三篇 用户和权限
  4. git管理和自动部署项目
  5. 十五道java开发常遇到的计算机网络协议高频面试题
  6. 理解Go 1.5 vendor
  7. 3.11 框架和样式表
  8. Flask详解(下篇)
  9. pyspark注意事项
  10. python多图合并成一张图_将现有的图形合并成一个图形PYTHON的pdf格式
  11. 彻底理解文本主题模型LDA(极致原理讲解+实战)
  12. C#验证类 可验证:邮箱,电话,手机,数字,英文,日期,身份证,邮编,网址,IP
  13. pcs7更改项目计算机名时出错,pcs7的C/S模式,从AS下装到OS或客户机项目时出错-工业支持中心-西门子中国...
  14. python并行计算
  15. 啥是“quoting reference XXX” 递交申请材料时
  16. 到底值不值?TTS语音模块——友军计划(1)
  17. MATLAB二维数组扩充为三维数组-repmat函数
  18. Android开发:教你开发TVBox应用时使用遥控器组合快捷键
  19. Solidworks进阶之路
  20. 浙大1018 Be carefull

热门文章

  1. java 修改word_Java 修改/替换Word批注
  2. 2021.10.18股票小计
  3. 实验1 数据库、表的创建
  4. Elasticsearch 字段膨胀不要怕,Flattened 类型解千愁!
  5. C#和.Ne学习第二天
  6. python程序的文件拓展名主要是_Python程序文件扩展名主要有________和________两种,其中后者常用于GUI程序...
  7. 信号与系统仿真实验——实验一 连续系统时域分析的MATLAB实现
  8. J - 马老师的餐厅
  9. iTunes Connect相关
  10. 自定义view留声机样式--音乐播放器