Cesium材质特效

文章目录

  • 0.引言
  • 1.视频材质
  • 2.分辨率尺度
  • 3.云
  • 4.雾
  • 5.动态水面
  • 6.雷达扫描
  • 7.流动线
  • 8.电子围栏
  • 9.粒子烟花
  • 10.粒子火焰
  • 11.粒子天气

0.引言

现有的gis开发方向较流行的是webgis开发,其中Cesium是一款开源的WebGIS库,主要用于实时地球和空间数据的可视化和分析。它提供了丰富的地图显示和数据可视化功能,并能实现三维可视化开发。本文将使用一些特殊的材质,如视频材质、自定义材质和Cesium内置的一些特殊效果类、粒子系统等实现一些特效场景的模拟,包括云、雾、动态水面、雷达扫描、流动线、电子围栏、粒子烟花、粒子火焰及粒子天气等。

1.视频材质

对于通过Entity方式和Primitive方式创建的几何实体,下面介绍如何给几何实体贴上一个特殊的材质,即视频材质。
视频资源网址: https://cesium.com/public/SandcastleSampleData/big-buck-bunny_trailer.mp4
(1)实现代码
  在这里插入图片描述

6_1_视频材质.html
<!DOCTYPE html>
<html lang="en">  
  
<head>  
    <meta charset="UTF-8">  
    <meta http-equiv="X-UA-Compatible" content="IE=edge">  
    <meta name="viewport" content="width=device-width, initial-scale=1.0">  
    <title>视频</title>  
    <script src="./Build/Cesium/Cesium.js"></script>  
    <link rel="stylesheet" href="./Build/Cesium/Widgets/widgets.css">  
    <style>  
        html,  
        body,  
        #cesiumContainer {  
            width: 100%;  
            height: 100%;  
            margin: 0;  
            padding: 0;  
        }  
  
        .toolbar {  
            position: absolute;  
            top: 10px;  
            left: 20px;  
            background-color: rgb(0, 0, 0, 0);  
        }  
    </style>  
</head>  
  
<body>  
    <div id="cesiumContainer">  
    </div>  
  
    <div class="toolbar">  
        <select id="dropdown" onchange="change()">  
            <option value="edit1">视频材质</option>  
            <option value="edit2">视频重复</option>  
        </select>  
    </div>  
  
    <video id="myVideo" muted="true" autoplay="true" loop="true" style="display: none;">  
        <source src="./vedio/big-buck-bunny_trailer.mp4" type="video/mp4">  
    </video>  
  
    <script>  
        Cesium.Ion.defaultAccessToken = '你的token';  
        var viewer = new Cesium.Viewer("cesiumContainer", {  
            timeline: false,  
            animation: false,  
            fullscreenButton: false,  
        });  
  
        //viewer.scene.globe.depthTestAgainstTerrain = false;  
  
        const videoElement = document.getElementById("myVideo");  
        //将视频元素与模拟时钟同步  
        let synchronizer = new Cesium.VideoSynchronizer({  
            clock: viewer.clock,  
            element: videoElement  
        });  
        viewer.clock.shouldAnimate = true;  
  
        var sphere = viewer.entities.add({  
            position: Cesium.Cartesian3.fromDegrees(104, 39, 2200),  
            ellipsoid: {  
                radii: new Cesium.Cartesian3(1000, 1000, 1000),  
                material: videoElement,  
            },  
        });  
        //相机视角锁定sphere  
        viewer.trackedEntity = sphere;  
  
        //改变视频重复个数  
        var isRepeat = false;  
        sphere.ellipsoid.material.repeat = new Cesium.CallbackProperty(  
            function (result) {  
                if (isRepeat) {  
                    result.x = 8;  
                    result.y = 8;  
                } else {  
                    result.x = 1;  
                    result.y = 1;  
                }  
                return result;  
            },  
            false  
        );  
  
        var dropdown = document.getElementById('dropdown');  
        function change() {  
            switch (dropdown.value) {  
                case 'edit1':  
                    isRepeat = false;  
                    break;  
                case 'edit2':  
                    isRepeat = true;  
                    break;  
                default:  
                    break;  
            }  
        }  
    </script>  
</body>  
  
</html>

(2)结果显示
  在这里插入图片描述
   在这里插入图片描述

2.分辨率尺度

在Cesium中,可以通过viewer.resolutionScale获取或者设置渲染分辨率的缩放比例。当该属性值小于1.0时,可以改善性能不佳的设备的显示效果,而当该属性值大于1.0时,将以更快的速度呈现分辨率,并缩小比例,从而提高视觉保真度。例如,如果窗口的尺寸为640像素×480像素,则将viewer.resolutionScale的值设置为0.5,会导致场景以320像素×240像素渲染,之后设置为2.0,会导致场景以1280像素×960像素渲染。
(1)实现代码
  在这里插入图片描述

6_2_分辨率尺度.html
<!DOCTYPE html>
<html lang="en">  
  
<head>  
    <meta charset="UTF-8">  
    <meta http-equiv="X-UA-Compatible" content="IE=edge">  
    <meta name="viewport" content="width=device-width, initial-scale=1.0">  
    <title>材质特效篇_分辨率尺度</title>  
    <script src="./Build/Cesium/Cesium.js"></script>  
    <link rel="stylesheet" href="./Build/Cesium//Widgets/widgets.css">  
    <style>  
        html,  
        body,  
        #cesiumContainer {  
            width: 100%;  
            height: 100%;  
            margin: 0;  
            padding: 0;  
            overflow: hidden;  
        }  
        .toolbar {  
            position: absolute;  
            top: 10px;  
            left: 20px;  
            background-color: rgba(0, 0, 0, 0.6);  
        }  
  
    </style>  
  
</head>  
  
<body>  
    <div id="cesiumContainer">  
    </div>  
    <div class="toolbar">  
        <label style="color: white;">分辨率尺度</label> <br />  
        <input type="range" max="2" step="0.1" oninput="change()" id="R" value="1">  
        <input type="text" style="width:70px; " id="resolutionValue" value="1" onchange="change2()">  
  
    </div>  
    <script>  
        Cesium.Ion.defaultAccessToken = '你的token';  
        var viewer = new Cesium.Viewer("cesiumContainer", {  
            animation: false, //是否显示动画工具  
            timeline: false,  //是否显示时间轴工具  
            fullscreenButton: false,  //是否显示全屏按钮工具  
        });  
    var tileset = viewer.scene.primitives.add(  
        new Cesium.Cesium3DTileset({  
            url: './倾斜摄影/大雁塔3DTiles/tileset.json',  
        }));  
  
    viewer.zoomTo(tileset);  
    function change() {  
        //拿到滑动条当前值  
        var resolutionScale = Number(R.value);  
        //将值约束在0.1和2.0之间  
        resolutionScale = Cesium.Math.clamp(resolutionScale, 0.1, 2.0);  
        //文本框显示当前值  
        resolutionValue.value = resolutionScale;  
        //修改分辨率尺度  
        viewer.resolutionScale = resolutionScale;  
    }  
    function change2() {  
        var resolutionScale = Number(resolutionValue.value);  
        //将值约束在0.1和2.0之间  
        resolutionScale = Cesium.Math.clamp(resolutionScale, 0.1, 2.0);  
        R.value = resolutionScale;  
        change();  
    }  
    </script>  
  
</body>  
  
</html>

(2)结果显示
调整前:
  在这里插入图片描述
调整后:
  在这里插入图片描述

3.云

在模拟实际场景时,可以通过CloudCollection类在场景中渲染云,同时支持手动修改云的大小、亮度等来模拟积云。基本思路为先使用CloudCollection类创建一个云集合,然后在云集合中添加定义的不同样式的云。
(1)实现代码
  在这里插入图片描述

6_3_云.html
<!DOCTYPE html>
<html lang="en">  
  
<head>  
  <meta charset="utf-8" />  
  <meta http-equiv="X-UA-Compatible" content="IE=edge" />  
  <meta name="viewport"  
    content="width=device-width, initial-scale=1, maximum-scale=1, minimum-scale=1, user-scalable=no" />  
  <meta name="description" content="Fog post process">  
  <meta name="cesium-sandcastle-labels" content="Showcases, Post Processing">  
  <title>材质特效篇_云</title>  
  <link rel="stylesheet" href="./Build/Cesium/Widgets/widgets.css">  
  <script type="text/javascript" src="./Build/Cesium/Cesium.js"></script>  
  
</head>  
<style>  
  html,  
  body,  
  #cesiumContainer {  
    width: 100%;  
    height: 100%;  
    margin: 0;  
    padding: 0;  
    overflow: hidden;  
  }  
  
  .toolbar {  
    position: absolute;  
    top: 10px;  
    left: 20px;  
    color: white;  
    background-color: rgba(0, 0, 0, 0.6);  
  }  
</style>  
  
<body>  
  <div id="cesiumContainer"></div>  
  <div class="toolbar">  
    <label>X轴尺寸</label> <br />  
    <input type="range" min="5" max="50" step="1" oninput="changeScale()" id="ScaleX" value="25">  
    <input type="text" style="width:70px; " id="ScaleXValue" value="25" onchange="changeScaleX()"> <br>  
    <label>Y轴尺寸</label> <br />  
    <input type="range" min="5" max="50" step="1" oninput="changeScale()" id="ScaleY" value="12">  
    <input type="text" style="width:70px; " id="ScaleYValue" value="12" onchange="changeScaleY()"> <br>  
    <label>亮度</label> <br />  
    <input type="range" min="0" max="1" step="0.01" oninput="changeBrightness()" id="Brightness" value="1">  
    <input type="text" style="width:70px; " id="BrightnessValue" value="1" onchange="changeBrightnessValue()"> <br>  
  </div>  
  
  <script>  
    Cesium.Ion.defaultAccessToken = '你的token';  
    var viewer = new Cesium.Viewer("cesiumContainer", {  
      animation: false, //是否显示动画工具  
      timeline: false,  //是否显示时间轴工具  
      fullscreenButton: false,  //是否显示全屏按钮工具  
    });  
  
    //创建并添加云集合  
    var clouds = viewer.scene.primitives.add(  
      new Cesium.CloudCollection({  
        noiseDetail: 16.0,  
      })  
    );  
  
    //添加云  
    var cloud = clouds.add({  
      position: Cesium.Cartesian3.fromDegrees(114.39264, 30.52252, 100),  
      scale: new Cesium.Cartesian2(25, 12),  
      slice: 0.36,  
      brightness: 1,  
    })  
  
    //设置相机位置及方向  
    viewer.camera.lookAt(  
      Cesium.Cartesian3.fromDegrees(114.39264, 30.52252, 100),  
      new Cesium.Cartesian3(30, 30, -10)  
    );  
  
    var ScaleX = document.getElementById('ScaleX'); //X轴尺寸  
    var ScaleXValue = document.getElementById('ScaleXValue'); //ScaleX滑动条值  
    var ScaleY = document.getElementById('ScaleY'); //Y轴尺寸  
    var ScaleYValue = document.getElementById('ScaleYValue'); //ScaleY滑动条值  
    var Brightness = document.getElementById('Brightness'); //亮度  
    var BrightnessValue = document.getElementById('BrightnessValue'); //亮度滑动条值  
  
    //Scale滑动条  
    function changeScale() {  
      //拿到scaleX滑动条当前值  
      var sX = Number(ScaleX.value);  
      //文本框显示当前值  
      ScaleXValue.value = sX;  
  
      //拿到scaleY滑动条当前值  
      var sY = Number(ScaleY.value);  
      //x轴旋转文本框显示当前值  
      ScaleYValue.value = sY;  
  
      //修改云的比例  
      cloud.scale = new Cesium.Cartesian2(sX, sY);  
    }  
    //ScaleX文本框  
    function changeScaleX() {  
      //拿到scaleX文本框的值并赋值给滑动条  
      ScaleX.value = Number(ScaleXValue.value);  
  
      changeScale();  
    }  
    //ScaleY文本框  
    function changeScaleY() {  
      //拿到scaleY文本框的值并赋值给滑动条  
      ScaleY.value = Number(ScaleYValue.value);  
      changeScale();  
    }  
  
    //Brightness滑动条  
    function changeBrightness() {  
      //拿到Brightness滑动条滑动条当前值  
      var brightness = Number(Brightness.value);  
      //文本框显示当前值  
      BrightnessValue.value = brightness;  
  
      //修改云的亮度  
      cloud.brightness = brightness;  
    }  
    //Brightness文本框  
    function changeBrightnessValue() {  
      //拿到文本框的值并赋值给滑动条  
      Brightness.value = Number(BrightnessValue.value);  
      changeBrightness();  
    }  
  
  </script>  
</body>  
  
</html>

(2)结果显示
  在这里插入图片描述

4.雾

Cesium在1.46版本之后新增了场景后处理功能。所谓场景后处理,我们可以将其理解为一个不断叠加的过程。例如,我们拍了一张照片,拍完之后觉得该照片亮度不够,于是我们在该照片的基础上进行了亮度的调整,得到了一张新照片,然后觉得新照片不够好看,又在新照片的基础上添加了滤镜,此后我们可能还会进行多次处理,直到最后得到的照片满足我们的要求为止,这个过程就类似于场景后处理,即我们在绘制场景时可能会不断地对场景进行一些处理,将最终符合我们要求的处理结果绘制到屏幕上。下面通过Cesium的场景后处理功能来实现雾的效果。
(1)实现代码
  在这里插入图片描述

6_4_雾效果.html
<!DOCTYPE html>
<html lang="en">  
  
<head>  
  <meta charset="utf-8" />  
  <meta http-equiv="X-UA-Compatible" content="IE=edge" />  
  <meta name="viewport"  
    content="width=device-width, initial-scale=1, maximum-scale=1, minimum-scale=1, user-scalable=no" />  
  <meta name="description" content="Fog post process">  
  <meta name="cesium-sandcastle-labels" content="Showcases, Post Processing">  
  <title>材质特效篇_雾效果</title>  
  <link rel="stylesheet" href="./Build/Cesium/Widgets/widgets.css">  
  <script type="text/javascript" src="./Build/Cesium/Cesium.js"></script>  
  
</head>  
<style>  
  html,  
  body,  
  #cesiumContainer {  
    width: 100%;  
    height: 100%;  
    margin: 0;  
    padding: 0;  
    overflow: hidden;  
  }  
</style>  
  
<body>  
  <div id="cesiumContainer"></div>  
  <script>  
    Cesium.Ion.defaultAccessToken = '你的token';  
    var viewer = new Cesium.Viewer("cesiumContainer", {  
      animation: false, //是否显示动画工具  
      timeline: false,  //是否显示时间轴工具  
      fullscreenButton: false,  //是否显示全屏按钮工具  
    });  
  
    var tileset = viewer.scene.primitives.add(  
      new Cesium.Cesium3DTileset({  
        url: './倾斜摄影/大雁塔3DTiles/tileset.json',  
      }));  
    viewer.zoomTo(tileset);  
  
    var fragmentShaderSource =  
      `//计算每个渲染顶点和视点(相机)的距离  
      float getDistance(sampler2D depthTexture, vec2 texCoords)  
      {  
          float depth = czm_unpackDepth(texture2D(depthTexture, texCoords));  
          if (depth == 0.0) {  
              return czm_infinity;  
          }  
          vec4 eyeCoordinate = czm_windowToEyeCoordinates(gl_FragCoord.xy, depth);  
          return -eyeCoordinate.z / eyeCoordinate.w;  
      }  
      //按距离进行插值  
      float interpolateByDistance(vec4 nearFarScalar, float distance)  
      {  
          float startDistance = nearFarScalar.x;  
          float startValue = nearFarScalar.y;  
          float endDistance = nearFarScalar.z;  
          float endValue = nearFarScalar.w;  
          float t = clamp((distance - startDistance) / (endDistance - startDistance), 0.0, 1.0);  
          return mix(startValue, endValue, t);  
      }  
      //计算透明度  
      vec4 alphaBlend(vec4 sourceColor, vec4 destinationColor)  
      {  
          return sourceColor * vec4(sourceColor.aaa, 1.0) + destinationColor * (1.0 - sourceColor.a);  
      }  
  
      uniform sampler2D colorTexture; //颜色纹理 内置变量  
      uniform sampler2D depthTexture; //深度纹理  内置变量  
      varying vec2 v_textureCoordinates;  //屏幕采样点坐标 内置变量  
  
      uniform vec4 fogByDistance; //自定义属性 外部变量  
      uniform vec4 fogColor;  //自定义属性 外部变量  
      void main(void)  
      {  
          float distance = getDistance(depthTexture, v_textureCoordinates);  
          vec4 sceneColor = texture2D(colorTexture, v_textureCoordinates);  
          float blendAmount = interpolateByDistance(fogByDistance, distance);  
          vec4 finalFogColor = vec4(fogColor.rgb, fogColor.a * blendAmount);  
          gl_FragColor = alphaBlend(finalFogColor, sceneColor);  
      }`;  
  
    var postProcessStage = new Cesium.PostProcessStage({  
      //片源着色器  
      fragmentShader: fragmentShaderSource,  
      uniforms: {  
        fogByDistance: new Cesium.Cartesian4(0, 0, 600, 1.0), //距离  
        fogColor: Cesium.Color.WHITE, //颜色  
      },  
    })  
    viewer.scene.postProcessStages.add(postProcessStage);  
  
  </script>  
</body>  
  
</html>

(2)结果显示
  在这里插入图片描述

5.动态水面

模拟水面效果也是Cesium场景中常见的功能,例如,有的项目可能通过绘制实体面,并设置材质为淡蓝色来模拟水面。但是,在实际生活中,水面往往不是静止的而是动态的,下面通过修改水面的材质来实现动态水面的效果。动态水面的具体实现思路为:先准备一张水面纹理图片,然后通过Primitive方式创建一个矩形实体,使用EllipsoidSurfaceAppearance定义一个水面材质,并给矩形实体设置该材质,即可实现简单的动态水面效果。
(1)实现代码
  在这里插入图片描述

6_5_动态水面.html
<!DOCTYPE html>
<html lang="en">  
  
<head>  
    <meta charset="UTF-8">  
    <meta http-equiv="X-UA-Compatible" content="IE=edge">  
    <meta name="viewport" content="width=device-width, initial-scale=1.0">  
    <title>材质特效篇_动态水面</title>  
    <script src="./Build/Cesium/Cesium.js"></script>  
    <link rel="stylesheet" href="./Build/Cesium/Widgets/widgets.css">  
    <style>  
        html,  
        body,  
        #cesiumContainer {  
            width: 100%;  
            height: 100%;  
            margin: 0;  
            padding: 0;  
            overflow: hidden;  
        }  
  
    </style>  
  
</head>  
  
<body>  
    <div id="cesiumContainer">  
    </div>  
    <script>  
        Cesium.Ion.defaultAccessToken = '你的token';  
        var viewer = new Cesium.Viewer("cesiumContainer", {  
            animation: false, //是否显示动画工具  
            timeline: false,  //是否显示时间轴工具  
            fullscreenButton: false,  //是否显示全屏按钮工具  
            terrainProvider: Cesium.createWorldTerrain()  
        });  
        viewer.scene.globe.depthTestAgainstTerrain = true;//开启深度检测后 会有高程遮挡效果  
    var rectangle = new Cesium.GeometryInstance({  
        geometry: new Cesium.RectangleGeometry({  
            rectangle: Cesium.Rectangle.fromDegrees(95.0, 39.0, 100.0, 42.0),  
            height: 3500.0  
        })  
    });  
    //定义外观  
    var rectangleAppearance = new Cesium.EllipsoidSurfaceAppearance({  
        aboveGround: true,  
        material: new Cesium.Material({  
            fabric:  
            {  
                type: 'Water',        //材质类型  
                uniforms: {  
                    //baseWaterColor: new Cesium.Color.fromBytes(24, 173, 247, 100),//基础颜色  
                    normalMap: './RasterImage/图片/动态水面.jpg',        //法线纹理贴图  
                    frequency: 100.0,        //波的数量  
                    animationSpeed: 0.01,        //水波震动速度  
                    amplitude: 10.0                //振幅大小  
                },  
            }  
        }),  
        //重写shader,修改水面的透明度  
        fragmentShaderSource: 'varying vec3 v_positionMC;\n' +  
            'varying vec3 v_positionEC;\n' +  
            'varying vec2 v_st;\n' +  
            'void main()\n' +  
            '{\n' +  
            'czm_materialInput materialInput;\n' +  
            'vec3 normalEC = normalize(czm_normal3D * czm_geodeticSurfaceNormal(v_positionMC, vec3(0.0), vec3(1.0)));\n' +  
            '#ifdef FACE_FORWARD\n' +  
            'normalEC = faceforward(normalEC, vec3(0.0, 0.0, 1.0), -normalEC);\n' +  
            '#endif\n' +  
            'materialInput.s = v_st.s;\n' +  
            'materialInput.st = v_st;\n' +  
            'materialInput.str = vec3(v_st, 0.0);\n' +  
            'materialInput.normalEC = normalEC;\n' +  
            'materialInput.tangentToEyeMatrix = czm_eastNorthUpToEyeCoordinates(v_positionMC, materialInput.normalEC);\n' +  
            'vec3 positionToEyeEC = -v_positionEC;\n' +  
            'materialInput.positionToEyeEC = positionToEyeEC;\n' +  
            'czm_material material = czm_getMaterial(materialInput);\n' +  
            '#ifdef FLAT\n' +  
            'gl_FragColor = vec4(material.diffuse + material.emission, material.alpha);\n' +  
            '#else\n' +  
            'gl_FragColor = czm_phong(normalize(positionToEyeEC), material, czm_lightDirectionEC);\n' +  
            'gl_FragColor.a=0.55;\n' +  
            '#endif\n' +  
            '}\n'  
    });  
    var addRectangleGeometry = new Cesium.Primitive({  
        geometryInstances: rectangle,  
        appearance: rectangleAppearance  
    })  
    viewer.scene.primitives.add(addRectangleGeometry);  
    viewer.camera.flyTo({  
        destination: Cesium.Cartesian3.fromDegrees(108, 42, 6000000),  
    })  
    </script>  
  
</body>  
  
</html>

(2)结果显示
  在这里插入图片描述
   在这里插入图片描述

6.雷达扫描

使用飞机或无人机沿着飞行路线进行雷达扫描的效果在实际应用中是很常见的。在Cesium中实现雷达扫描效果的方法有很多,可以通过对Entity实体贴纹理并对材质进行不断的旋转来实现,或者通过着色器重写Entity实体的材质shader来实现。比较而言,前者对于新手来说更容易实现,下面通过第一种方法来模拟雷达扫描效果。
(1)实现代码
  在这里插入图片描述

6_6_雷达扫描.html
<!DOCTYPE html>
<html lang="en">  
  
<head>  
    <meta charset="UTF-8">  
    <meta http-equiv="X-UA-Compatible" content="IE=edge">  
    <meta name="viewport" content="width=device-width, initial-scale=1.0">  
    <title>材质特效篇_雷达扫描</title>  
    <script src="./Build/Cesium/Cesium.js"></script>  
    <link rel="stylesheet" href="./Build/Cesium//Widgets/widgets.css">  
    <style>  
        html,  
        body,  
        #cesiumContainer {  
            width: 100%;  
            height: 100%;  
            margin: 0;  
            padding: 0;  
            overflow: hidden;  
        }  
  
    </style>  
  
</head>  
  
<body>  
    <div id="cesiumContainer">  
    </div>  
    <script>  
        Cesium.Ion.defaultAccessToken = '你的token';  
        var viewer = new Cesium.Viewer("cesiumContainer", {  
            animation: false, //是否显示动画工具  
            timeline: false,  //是否显示时间轴工具  
            fullscreenButton: false,  //是否显示全屏按钮工具  
        });  
    var rotation = 0; //纹理旋转角度  
    var amount = 4;        //旋转变化量  
    var rader = {  
        position: Cesium.Cartesian3.fromDegrees(114.40372, 30.52252),  
        ellipse: {  
            semiMajorAxis: 300.0,  
            semiMinorAxis: 300.0,  
            //指定材质  
            material: new Cesium.ImageMaterialProperty({  
                image: './RasterImage/图片/color.png',  
                color: new Cesium.Color(1.0, 0.0, 0.0, 0.7),  
            }),  
    // 不设置高度则无法渲染外框线  
    height: 0.0,  
    //heightReference: Cesium.HeightReference.CLAMP_TO_GROUND,  
    //外边框  
    outline: true,  
    outlineWidth: 2,  
    outlineColor: new Cesium.Color(1.0, 1.0, 0.0, 1.0),  

    //纹理旋转角度通过CallbackProperty回调  
    stRotation: new Cesium.CallbackProperty(function () {  
        rotation += amount;  
        if (rotation >= 360 || rotation <= -360) {  
            rotation = 0;  
        }  
        //度数转弧度  
        return Cesium.Math.toRadians(rotation);  
    }, false)  
    }  
    }  
    //将rader添加进entity集合  
    viewer.entities.add(rader)  
    var point = viewer.entities.add({  
        position: Cesium.Cartesian3.fromDegrees(114.40372, 30.52252),  
        point: {  
            pixelSize: 10,  
            color: Cesium.Color.RED,  
            heightReference: Cesium.HeightReference.CLAMP_TO_GROUND  
        }  
    });  
    viewer.camera.setView({  
        destination: Cesium.Cartesian3.fromDegrees(114.40372, 30.52252, 2000)  
    });  
    </script>  
  
</body>  
  
</html>

(2)结果显示
  在这里插入图片描述
   在这里插入图片描述

7.流动线

Cesium中有许多封装好的内置纹理,如条纹、颜色、虚线、棋盘、水面等,但是这些内置纹理大多是静态的,并不能满足我们在实际开发中的需求,这时就需要我们通过自定义材质来达到特定的纹理效果。
自定义材质可以通过现有的内置材质派生,也可以使用Fabric和GLSL来自定义。但是在实际开发中,为了减少代码冗余,我们通常将常用的自定义材质封装成一个个Material材质类以便复用,下面将介绍如何封装一个自定义流动线材质类。
(1)实现代码
  在这里插入图片描述

6_7_流动线.html
<!DOCTYPE html>
<html lang="en">  
  
<head>  
  <meta charset="UTF-8">  
  <meta http-equiv="X-UA-Compatible" content="IE=edge">  
  <meta name="viewport" content="width=device-width, initial-scale=1.0">  
  <title>材质特效篇_流动线</title>  
  <script src="./Build/Cesium/Cesium.js"></script>  
  <link rel="stylesheet" href="./Build/Cesium/Widgets/widgets.css">  
  <script src="./Build/js/jquery.min.js"></script>  
  <style>  
    html,  
    body,  
    #cesiumContainer {  
      width: 100%;  
      height: 100%;  
      margin: 0;  
      padding: 0;  
      overflow: hidden;  
    }  
  </style>  
</head>  
  
<body>  
  <div id="cesiumContainer">  
  </div>  
  
  <script>  
    Cesium.Ion.defaultAccessToken =  
      '你的token';  
    var viewer = new Cesium.Viewer("cesiumContainer", {  
      animation: true, //是否显示动画工具  
      timeline: true, //是否显示时间轴工具  
      fullscreenButton: false, //是否显示全屏按钮工具  
    });  
    /* console.log('selectionIndicator',viewer.selectionIndicator);  
    $(".cesium-viewer-selectionIndicatorContainer").css('display','none'); */  
  
    viewer.scene.fxaa = false  
    viewer.scene.postProcessStages.fxaa.enabled = false;  
  
    var supportsImageRenderingPixelated = viewer.cesiumWidget._supportsImageRenderingPixelated;  
    if (supportsImageRenderingPixelated) {  
      var vtxf_dpr = window.devicePixelRatio;  
      while (vtxf_dpr >= 2.0) {  
        vtxf_dpr /= 2.0;  
      }  
      viewer.resolutionScale = vtxf_dpr;  
    }  
  
    //创建构造函数  
    function PolylineTrailLinkMaterialProperty(color, duration) {  
      this._definitionChanged = new Cesium.Event();  
      this._color = undefined;  
      this._colorSubscription = undefined;  
      this.color = color;  
      this.duration = duration;  
      this._time = (new Date()).getTime();  
    }  
  
    //Object.defineProperties() 方法直接在一个对象上定义新的属性或修改现有属性,并返回该对象。  
    //Object.defineProperties(obj, props)  
    //obj:在其上定义或修改属性的对象   props:要定义其可枚举属性或修改的属性描述符的对象。  
    Object.defineProperties(PolylineTrailLinkMaterialProperty.prototype, {  
      isConstant: {  
        get: function () {  
          return false;  
        }  
      },  
      definitionChanged: {  
        get: function () {  
          return this._definitionChanged;  
        }  
      },  
      color: Cesium.createPropertyDescriptor('color')  
    });  
  
    PolylineTrailLinkMaterialProperty.prototype.getType = function (time) {  
      return 'PolylineTrailLink';  
    }  
    PolylineTrailLinkMaterialProperty.prototype.getValue = function (time, result) {  
      if (!Cesium.defined(result)) {  
        result = {};  
      }  
      result.color = Cesium.Property.getValueOrClonedDefault(this._color, time, Cesium.Color.WHITE, result.color);  
      result.image = Cesium.Material.PolylineTrailLinkImage;  
      result.time = (((new Date()).getTime() - this._time) % this.duration) / this.duration;  
      return result;  
    }  
    PolylineTrailLinkMaterialProperty.prototype.equals = function (other) {  
      return this === other || (other instanceof PolylineTrailLinkMaterialProperty &amp;&amp; Property.equals(this._color,  
        other._color))  
    };  
  
    Cesium.PolylineTrailLinkMaterialProperty = PolylineTrailLinkMaterialProperty;  
    //纹理类型  
    Cesium.Material.PolylineTrailLinkType = 'PolylineTrailLink';  
    //纹理图片  
    Cesium.Material.PolylineTrailLinkImage = "./RasterImage/图片/color.png";  
    //纹理资源  
    Cesium.Material.PolylineTrailLinkSource =  
      "czm_material czm_getMaterial(czm_materialInput materialInput)\n\  
            {\n\  
                float time = czm_frameNumber/100.0;\n\  
                czm_material material = czm_getDefaultMaterial(materialInput);\n\  
                vec2 st = materialInput.st;\n\  
                vec4 colorImage = texture2D(image, vec2(fract(3.0*st.s - time), st.s));\n\  
                material.alpha = colorImage.a * color.a;\n\  
                material.diffuse = (colorImage.rgb+color.rgb)/2.0;\n\  
                return material;\n\  
            }";  
    //time越小,速度越慢  
    //colorImage控制纹理  
    //fract中 3.0是纹理个数  -time是逆时针 +time是顺时针  
    //alpha 透明度  
    //diffuse 颜色  
  
    /* "czm_material czm_getMaterial(czm_materialInput materialInput)\n\  
    {\n\  
        czm_material material = czm_getDefaultMaterial(materialInput);\n\  
        vec2 st = materialInput.st;\n\  
        vec4 colorImage = texture2D(image, vec2(fract(st.s - time), st.t));\n\  
        material.alpha = colorImage.a * color.a;\n\  
        material.diffuse = (colorImage.rgb+color.rgb)/2.0;\n\  
        return material;\n\  
    }" */  
  
    //添加自定义材质  
    Cesium.Material._materialCache.addMaterial(Cesium.Material.PolylineTrailLinkType, {  
      fabric: {  
        //纹理类型  
        type: Cesium.Material.PolylineTrailLinkType,  
        //传递给着色器的外部属性  
        uniforms: {  
          color: new Cesium.Color(0.0, 0.0, 0.0, 1),  
          image: Cesium.Material.PolylineTrailLinkImage,  
          time: 0  
        },  
        //纹理资源  
        source: Cesium.Material.PolylineTrailLinkSource  
      },  
      //是否透明  
      translucent: function (material) {  
        return true;  
      }  
    })  
  
    var line = viewer.entities.add({  
      name: 'PolylineTrailLink',  
      polyline: {  
        positions: Cesium.Cartesian3.fromDegreesArray([  
          118.286419, 31.864436,  
          119.386419, 31.864436,  
          119.386419, 32.864436,  
          118.686419, 32.864436,  
        ]),  
        width: 10,  
        //设置材质为自定义材质  
        material: new Cesium.PolylineTrailLinkMaterialProperty(  
          Cesium.Color.fromBytes(255, 0, 0).withAlpha(0.8),  
          /* 1000 */  
        ),  
      }  
    });  
    viewer.flyTo(line)  
  </script>  
</body>  
  
</html>

(2)结果显示
  在这里插入图片描述
   在这里插入图片描述

8.电子围栏

下面封装一个自定义电子围栏材质类,能够对Entity墙体贴动态材质,实现电子围栏效果。封装自定义电子围栏材质类的流程和封装自定义流动线材质类的流程一样。
(1)实现代码
  在这里插入图片描述

6_8_电子围栏.html
<!DOCTYPE html>
<html lang="en">  
  
<head>  
    <meta charset="UTF-8">  
    <meta http-equiv="X-UA-Compatible" content="IE=edge">  
    <meta name="viewport" content="width=device-width, initial-scale=1.0">  
    <title>材质特效篇_电子围栏</title>  
    <script src="./Build/Cesium/Cesium.js"></script>  
    <link rel="stylesheet" href="./Build/Cesium/Widgets/widgets.css">  
    <style>  
        html,  
        body,  
        #cesiumContainer {  
            width: 100%;  
            height: 100%;  
            margin: 0;  
            padding: 0;  
            overflow: hidden;  
        }  
    </style>  
</head>  
  
<body>  
    <div id="cesiumContainer">  
    </div>  
  
    <script>  
        Cesium.Ion.defaultAccessToken = '你的token';  
        var viewer = new Cesium.Viewer("cesiumContainer", {  
            animation: false, //是否显示动画工具  
            timeline: false,  //是否显示时间轴工具  
            fullscreenButton: false,  //是否显示全屏按钮工具  
        });  
  
        function DynamicWallMaterialProperty(color, duration) {  
            this._definitionChanged = new Cesium.Event();  
            this._color = undefined;  
            this._colorSubscription = undefined;  
            this.color = color;  
            this.duration = duration;  
            this._time = (new Date()).getTime();  
        }  
  
        Object.defineProperties(DynamicWallMaterialProperty.prototype, {  
            isConstant: {  
                get: function () {  
                    return false;  
                }  
            },  
            definitionChanged: {  
                get: function () {  
                    return this._definitionChanged;  
                }  
            },  
            color: Cesium.createPropertyDescriptor('color')  
        });  
  
        DynamicWallMaterialProperty.prototype.getType = function (time) {  
            return 'DynamicWall';  
        }  
        DynamicWallMaterialProperty.prototype.getValue = function (time, result) {  
            if (!Cesium.defined(result)) {  
                result = {};  
            }  
            result.color = Cesium.Property.getValueOrClonedDefault(this._color, time, Cesium.Color.WHITE, result.color);  
            result.image = Cesium.Material.DynamicWallImage;  
            result.time = (((new Date()).getTime() - this._time) % this.duration) / this.duration;  
            return result;  
        }  
        DynamicWallMaterialProperty.prototype.equals = function (other) {  
            return this === other || (other instanceof DynamicWallMaterialProperty &amp;&amp; Property.equals(this._color, other._color))  
        };  
  
        Cesium.DynamicWallMaterialProperty = DynamicWallMaterialProperty;  
        Cesium.Material.DynamicWallType = 'DynamicWall';  
        Cesium.Material.DynamicWallImage = "./RasterImage/图片/color.png";//图片  
        Cesium.Material.DynamicWallSource =  
            `czm_material czm_getMaterial(czm_materialInput materialInput)  
            {  
                float time = czm_frameNumber/100.0;  
                czm_material material = czm_getDefaultMaterial(materialInput);  
                vec2 st = materialInput.st;  
                vec4 colorImage = texture2D(image, vec2(fract(1.0*st.t - time), st.t));  
                material.alpha = colorImage.a * color.a;  
                material.diffuse = (colorImage.rgb+color.rgb)/2.0;  
                return material;  
            }`      //由上到下  
  
        //添加自定义材质  
        Cesium.Material._materialCache.addMaterial(Cesium.Material.DynamicWallType, {  
            fabric: {  
                //纹理类型  
                type: Cesium.Material.DynamicWallType,  
                //传递给着色器的外部属性  
                uniforms: {  
                    color: new Cesium.Color(0.0, 0.0, 0.0, 1),  
                    image: Cesium.Material.DynamicWallImage,  
                    time: 0  
                },  
                //纹理资源  
                source: Cesium.Material.DynamicWallSource  
            },  
            //是否透明  
            translucent: function (material) {  
                return true;  
            }  
        })  
  
        var dynamicWall = viewer.entities.add({  
            wall: {  
                positions: Cesium.Cartesian3.fromDegreesArrayHeights([  
                    118.286419, 31.864436, 20000.0,  
                    119.386419, 31.864436, 20000.0,  
                    119.386419, 32.864436, 20000.0,  
                    118.286419, 32.864436, 20000.0,  
                    118.286419, 31.864436, 20000.0,  
                ]),  
                material: new Cesium.DynamicWallMaterialProperty(Cesium.Color.fromBytes(255, 200, 10).withAlpha(0.8), 3000),  
            }  
        })  
        viewer.flyTo(dynamicWall)  
  
    </script>  
</body>  
  
</html>

(2)结果显示
  在这里插入图片描述
   在这里插入图片描述

9.粒子烟花

粒子系统表示三维计算机图形学中用于模拟一些特定模糊现象的技术,而这些现象用其他传统的渲染技术难以实现其真实感的物理运动规律。经常使用粒子系统模拟的现象有烟花、火焰、雨水及雪花等。简而言之,粒子系统就是一种用于模拟真实现象的图形技术,是由一个个的小图像集合而成的,从远处看会形成一个“复杂”的场景来模拟一些现象。
Cesium粒子系统不仅是多个小图像的直接集合,而且允许控制单个粒子的寿命、速度、位置等属性,也正是由于粒子的各种属性可以控制,才能够模拟各种复杂的场景。粒子系统效果在电影和电子游戏中应用广泛。下面使用粒子系统模拟烟花爆炸效果。
(1)实现代码
  在这里插入图片描述

6_9_粒子烟花.html
<!DOCTYPE html>
<html lang="en">  
  
<head>  
  <meta charset="utf-8" />  
  <meta http-equiv="X-UA-Compatible" content="IE=edge" />  
  <meta name="description" content="Particle system fireworks.">  
  <meta name="cesium-sandcastle-labels" content="Beginner, Showcases">  
  <title>材质特效篇_粒子烟花</title>  
  <link rel="stylesheet" href="./Build/Cesium/Widgets/widgets.css">  
  <script src="./Build/Cesium/Cesium.js"></script>  
  
</head>  
<style>  
  html,  
  body,  
  #cesiumContainer {  
    width: 100%;  
    height: 100%;  
    margin: 0;  
    padding: 0;  
    overflow: hidden;  
  }  
</style>  
  
<body>  
  
  <div id="cesiumContainer"></div>  
  <script>  
    Cesium.Ion.defaultAccessToken = '你的token';  
    var viewer = new Cesium.Viewer("cesiumContainer", {  
      animation: false, //是否显示动画工具  
      timeline: false,  //是否显示时间轴工具  
      fullscreenButton: false,  //是否显示全屏按钮工具  
      shouldAnimate: true,  //必须开启,自动播放动画  
    });  
  
    /* Cesium.Math.setRandomNumberSeed(315); */  
  
    //东北天到指定原点变换矩阵,将粒子系统从模型坐标转换为世界坐标  
    const modelMatrix = Cesium.Transforms.eastNorthUpToFixedFrame(  
      Cesium.Cartesian3.fromDegrees(114.39664, 30.52052)  
    );  
    //粒子发射器高度  
    const emitterInitialLocation = new Cesium.Cartesian3(0.0, 0.0, 100.0);  
  
    //粒子贴图  
    var particleCanvas;  
    //绘制图形  
    function getImage() {  
      if (!Cesium.defined(particleCanvas)) {  
        particleCanvas = document.createElement("canvas");  
        particleCanvas.width = 20;  
        particleCanvas.height = 20;  
        const context2D = particleCanvas.getContext("2d");  
        context2D.beginPath();  
        //圆心x 圆心y 半径 起始角度 终止角度 逆时针  
        context2D.arc(10, 10, 8, 0, Cesium.Math.TWO_PI, true);  
        context2D.closePath();  
        context2D.fillStyle = "rgba(255, 255, 255, 1)";  
        context2D.fill();  
      }  
      return particleCanvas;  
    }  
  
    /* var radar = viewer.entities.add({  
    rectangle: {  
        coordinates: Cesium.Rectangle.fromDegrees(114.40072, 30.51952, 114.40572, 30.52452),  
        material: new Cesium.ImageMaterialProperty({  
            //image: new Cesium.CallbackProperty(drawCanvas, false),  
            image:getImage(),  
            //transparent: true  
        }),  
    }  
    }); */  
    /* const minimumExplosionSize = 30.0; //最小爆炸尺寸  
    const maximumExplosionSize = 100.0;  //最大爆炸尺寸 */  
    var particlePixelSize = new Cesium.Cartesian2(7.0, 7.0);  //粒子大小  
    var burstNum = 400.0;  //爆炸粒子个数  
    var lifetime = 10.0;    //粒子系统发射粒子的时间  
    var numberOfFireworks = 20.0;  //烟花个数  
  
    //创建烟花函数  
    function createFirework(offset, color, bursts) {  
      var position = Cesium.Cartesian3.add(  
        emitterInitialLocation,  
        offset,  
        new Cesium.Cartesian3()  
      );  
      //从发射位置创建表示转换的Matrix4  
      var emitterModelMatrix = Cesium.Matrix4.fromTranslation(position);  
  
      //随机设置烟花的生命周期  
      /* const size = Cesium.Math.randomBetween(  
        minimumExplosionSize,  
        maximumExplosionSize  
      );  
      const normalSize =  
        (size - minimumExplosionSize) /  
        (maximumExplosionSize - minimumExplosionSize);  
      const minLife = 0.3;  
      const maxLife = 1.0;  
      const life = normalSize * (maxLife - minLife) + minLife; */  
  
      viewer.scene.primitives.add(  
        new Cesium.ParticleSystem({  
          image: getImage(),  //粒子贴图  
          startColor: color,  //粒子在其生命初期的颜色  
          endColor: color.withAlpha(0.0),//粒子在其生命结束的颜色  
          //particleLife: life, //粒子生命周期  
          particleLife: 1, //粒子生命周期  
          speed: 100.0, //粒子扩散速度  
          imageSize: particlePixelSize, //粒子像素大小  
          emissionRate: 0,  //每秒要发射的粒子数  
          emitter: new Cesium.SphereEmitter(0.1), //系统粒子发射器  
          bursts: bursts, //粒子爆炸,ParticleBurst 的数组  
          lifetime: lifetime, //粒子系统发射粒子的时间  
          //updateCallback: force,  //每帧都要调用一次回调函数以更新粒子  
          modelMatrix: modelMatrix, //将粒子系统从模型转换为世界坐标的4x4转换矩阵。  
          emitterModelMatrix: emitterModelMatrix,//在粒子系统局部坐标系内转换粒子系统发射器的4x4转换矩阵  
          loop: true //粒子循环爆发  
        })  
      );  
    }  
  
    //粒子发射器偏移量范围  
    var xMin = -100.0;  
    var xMax = 100.0;  
    var yMin = -80.0;  
    var yMax = 100.0;  
    var zMin = -50.0;  
    var zMax = 50.0;  
  
    //设置随机颜色选项数组  
    var colorOptions = [  
      {  
        minimumRed: 0.75,  
        green: 0.0,  
        minimumBlue: 0.8,  
        alpha: 1.0,  
      },  
      {  
        red: 0.0,  
        minimumGreen: 0.75,  
        minimumBlue: 0.8,  
        alpha: 1.0,  
      },  
      {  
        red: 0.0,  
        green: 0.0,  
        minimumBlue: 0.8,  
        alpha: 1.0,  
      },  
      {  
        minimumRed: 0.75,  
        minimumGreen: 0.75,  
        blue: 0.0,  
        alpha: 1.0,  
      },  
    ];  
  
    //创建烟花  
    for (let i = 0; i < numberOfFireworks; ++i) {  
      var x = Cesium.Math.randomBetween(xMin, xMax);  
      var y = Cesium.Math.randomBetween(yMin, yMax);  
      var z = Cesium.Math.randomBetween(zMin, zMax);  
      var offset = new Cesium.Cartesian3(x, y, z);  
      //使用提供的选项创建随机颜色  
      var color = Cesium.Color.fromRandom(  
        colorOptions[i % colorOptions.length]  
      );  
      //粒子爆炸,ParticleBurst 的数组,在周期时间发射粒子爆发  
      var bursts = [];  
      for (let j = 0; j < 3; ++j) {  
        bursts.push(  
          new Cesium.ParticleBurst({  
            time: Cesium.Math.nextRandomNumber() * lifetime, //粒子系统生命周期开始后以秒为单位的时间,将发生爆发  
            minimum: burstNum, //爆发中发射的最小粒子数。  
            maximum: burstNum, //爆发中发射的最大粒子数。  
          })  
        );  
      }  
      //传参,创建烟花  
      createFirework(offset, color, bursts);  
    }  
  
    viewer.scene.camera.setView({  
      destination:  
        Cesium.Cartesian3.fromDegrees(114.39664, 30.52052, 2000)  
    })  
  
  </script>  
</body>  
  
</html>

(2)结果显示
  在这里插入图片描述
   在这里插入图片描述

10.粒子火焰

下面使用Cesium粒子系统模拟火焰燃烧效果。
(1)实现代码
  在这里插入图片描述

6_10_粒子火焰.html
<!DOCTYPE html>
<html lang="en">  
  
<head>  
  <meta charset="utf-8" />  
  <meta http-equiv="X-UA-Compatible" content="IE=edge" />  
  <meta name="description" content="Particle system fireworks.">  
  <meta name="cesium-sandcastle-labels" content="Beginner, Showcases">  
  <title>材质特效篇_粒子火焰</title>  
  <link rel="stylesheet" href="./Build/Cesium/Widgets/widgets.css">  
  <script src="./Build/Cesium/Cesium.js"></script>  
  
</head>  
<style>  
  html,  
  body,  
  #cesiumContainer {  
    width: 100%;  
    height: 100%;  
    margin: 0;  
    padding: 0;  
    overflow: hidden;  
  }  
</style>  
  
<body>  
  
  <div id="cesiumContainer"></div>  
  
  <script>  
    Cesium.Ion.defaultAccessToken = '你的token';  
    var viewer = new Cesium.Viewer("cesiumContainer", {  
      animation: false, //是否显示动画工具  
      timeline: false,  //是否显示时间轴工具  
      fullscreenButton: false,  //是否显示全屏按钮工具  
      shouldAnimate: true,  //必须开启 自动播放动画  
    });  
  
    // 加载飞机模型  
    var entity = viewer.entities.add({  
      model: {  
        uri: './3D格式数据/glb/Cesium_Air.glb',  
        minimumPixelSize: 64  
      },  
      position: Cesium.Cartesian3.fromDegrees(114.39264, 30.52252, 100)  
    });  
    //视角追踪模型  
    viewer.trackedEntity = entity;  
  
    //计算把粒子系统从模型坐标系转到世界坐标系指定原点的矩阵  
    const modelMatrix = Cesium.Transforms.eastNorthUpToFixedFrame(  
      Cesium.Cartesian3.fromDegrees(114.39264, 30.52252, 100)  
    );  
  
    //计算模型坐标系的平移矩阵  
    function computeEmitterModelMatrix() {  
      //定义粒子发射器的方向、俯仰角以及翻滚角  
      var hpr = Cesium.HeadingPitchRoll.fromDegrees(0.0, 0.0, 0.0, new Cesium.HeadingPitchRoll());  
      //定义一个由平移,旋转和缩放定义的仿射变换  
      var trs = new Cesium.TranslationRotationScale();  
      //火焰位置  
      //平移  
      trs.translation = Cesium.Cartesian3.fromElements(2.5, 4.0, 1.0, new Cesium.Cartesian3());  
      //旋转  
      trs.rotation = Cesium.Quaternion.fromHeadingPitchRoll(hpr, new Cesium.Quaternion());  
      return Cesium.Matrix4.fromTranslationRotationScale(trs, new Cesium.Matrix4());  
    }  
  
    var particleSystem = new Cesium.ParticleSystem({  
      image: './RasterImage/图片/fire.png',  
      startScale: 1.0,  //开始比例  
      endScale: 4.0,  //结束比例  
      particleLife: 1.0,   //粒子生命周期  
      speed: 5.0, //粒子速度  
      imageSize: new Cesium.Cartesian2(20, 20),   //粒子图形尺寸  
      emissionRate: 5.0,  //每秒发射粒子个数  
      lifetime: 16.0,   //粒子系统发射粒子的时间  
      modelMatrix: modelMatrix,  //将粒子系统从模型转换为世界坐标的4x4转换矩阵  
      emitterModelMatrix: computeEmitterModelMatrix() //在粒子系统局部坐标系内转换粒子系统发射器的4x4转换矩阵  
    })  
    viewer.scene.primitives.add(particleSystem);  
  
  </script>  
</body>  
  
</html>

(2)结果显示
  在这里插入图片描述
   在这里插入图片描述

11.粒子天气

常见的粒子特效还有雨、雪等粒子天气特效,下面使用Cesium粒子系统模拟天气特效,包括下雨天与下雪天两种情况。
(1)实现代码
  在这里插入图片描述

6_11_粒子天气.html
<!DOCTYPE html>
<html lang="en">  
  
<head>  
  <meta charset="utf-8" />  
  <meta http-equiv="X-UA-Compatible" content="IE=edge" />  
  <meta name="description" content="Particle system fireworks.">  
  <meta name="cesium-sandcastle-labels" content="Beginner, Showcases">  
  <title>材质特效篇_粒子天气</title>  
  <link rel="stylesheet" href="./Build/Cesium/Widgets/widgets.css">  
  <script src="./Build/Cesium/Cesium.js"></script>  
  
</head>  
<style>  
  html,  
  body,  
  #cesiumContainer {  
    width: 100%;  
    height: 100%;  
    margin: 0;  
    padding: 0;  
    overflow: hidden;  
  }  
  
  .toolbar {  
    position: absolute;  
    top: 10px;  
    left: 20px;  
    background-color: rgb(0, 0, 0, 0);  
  }  
</style>  
  
<body>  
  
  <div id="cesiumContainer"></div>  
  <div class="toolbar">  
    <select id="dropdown" onchange="change()">  
      <option value="snow"></option>  
      <option value="rain"></option>  
      <option value="null">null</option>  
    </select>  
  </div>  
  
  <script>  
    Cesium.Ion.defaultAccessToken = '你的token';  
    var viewer = new Cesium.Viewer("cesiumContainer", {  
      animation: false, //是否显示动画工具  
      timeline: false,  //是否显示时间轴工具  
      fullscreenButton: false,  //是否显示全屏按钮工具  
      shouldAnimate: true,  //必须开启  
      terrainProvider: Cesium.createWorldTerrain(),  
    });  
  
    //粒子特效位置  
    var position = new Cesium.Cartesian3.fromDegrees(114.39664, 30.52052, 2000);  
    var modelMatrix = new Cesium.Matrix4.fromTranslation(position)  
  
    //模拟下雪天粒子特效常量定义  
    const snowRadius = 100000.0; //下雪的范围半径  
    const minimumSnowImageSize = new Cesium.Cartesian2(10, 10); //雪花最小尺寸  
    const maximumSnowImageSize = new Cesium.Cartesian2(20, 20); //雪花最大尺寸  
  
    //创建Cartesian3对象,用于在回调函数中实时更新粒子位置  
    var snowGravityScratch = new Cesium.Cartesian3();  
    //粒子更新回调函数  
    function snowUpdate (particle) {  
      //计算提供的笛卡尔坐标系的标准化形式  
      Cesium.Cartesian3.normalize(  
        particle.position,  //要标准化的笛卡尔坐标  
        snowGravityScratch  //结果存储对象  
      );  
      //将提供的笛卡尔分量乘以标准的标量  
      Cesium.Cartesian3.multiplyByScalar(  
        snowGravityScratch, //要缩放的笛卡尔坐标  
        //要与之相乘的标量,负值代表粒子位置下降即粒子从上往下落  
        Cesium.Math.randomBetween(-30.0, -300.0),  
        snowGravityScratch  //结果存储对象  
      );  
      //粒子位置根据snowGravityScratch变化  
      Cesium.Cartesian3.add(  
        particle.position,  
        snowGravityScratch,  
        particle.position  
      );  
    };  
  
    // 雨  
    const rainRadius = 100000.0; //下雨的范围半径  
    const rainImageSize = new Cesium.Cartesian2(20, 35); //15,30分别代表宽高  
  
    var rainGravityScratch = new Cesium.Cartesian3();  
    //粒子更新回调函数  
    function rainUpdate (particle) {  
      //计算提供的笛卡尔坐标系的标准化形式  
      Cesium.Cartesian3.normalize(  
        particle.position,  //要标准化的笛卡尔坐标  
        rainGravityScratch  //结果存储对象  
      );  
      //将提供的笛卡尔分量乘以标准的标量  
      Cesium.Cartesian3.multiplyByScalar(  
        rainGravityScratch,  //要缩放的笛卡尔坐标  
        -1000.0,             //要与之相乘的标量,雨比雪下落速度快的多 所以这个值负的多点  
        rainGravityScratch   //结果存储对象  
      );  
      //粒子位置根据rainGravityScratch变化  
      Cesium.Cartesian3.add(  
        particle.position,  
        rainGravityScratch,  
        particle.position  
      );  
    };  
  
    //粒子系统-雪配置项  
    var snowOption = {  
      modelMatrix: modelMatrix, //将粒子系统从模型转换为世界坐标的4x4转换矩阵。  
      lifetime: 15.0, //粒子系统发射粒子的时间(以秒为单位)  
      emitter: new Cesium.SphereEmitter(snowRadius),  //该系统的粒子发射器  
      startScale: 0.5,  //在粒子寿命开始时应用于粒子图像的初始比例  
      endScale: 1.0,  //在粒子寿命结束时应用于粒子图像的最终比例。  
      image: "./RasterImage/图片/snowflake_particle.png", //粒子贴图  
      emissionRate: 7000.0, //每秒要发射的粒子数  
      startColor: Cesium.Color.WHITE.withAlpha(0.0),  //粒子在其生命初期的颜色。  
      endColor: Cesium.Color.WHITE.withAlpha(1.0),  //粒子寿命结束时的颜色。  
      minimumImageSize: minimumSnowImageSize, //设置宽度的最小范围,以高度为单位,在该范围上可以随机缩放粒子图像的尺寸(以像素为单位)  
      maximumImageSize: maximumSnowImageSize, //设置最大宽度边界,以高度为单位,在该边界以下可以随机缩放粒子图像的尺寸(以像素为单位)  
      updateCallback: snowUpdate, //每帧都要调用一次回调函数以更新粒子  
    }  
  
    //粒子系统-雨配置项  
    var rainOption = {  
      modelMatrix: modelMatrix,//将粒子系统从模型转换为世界坐标的4x4转换矩阵。  
      lifetime: 15.0,//粒子系统发射粒子的时间(以秒为单位)  
      emitter: new Cesium.SphereEmitter(rainRadius),//该系统的粒子发射器  
      startScale: 1.0,//在粒子寿命开始时应用于粒子图像的初始比例  
      endScale: 0.0,//在粒子寿命结束时应用于粒子图像的最终比例。  
      image: "./RasterImage/图片/circular_particle.png",//粒子贴图  
      emissionRate: 9000.0,//每秒要发射的粒子数  
      startColor: new Cesium.Color(1, 1, 1, 0.0),//粒子在其生命初期的颜色。  
      endColor: new Cesium.Color(1.0, 1.0, 1.0, 0.98),//粒子寿命结束时的颜色。  
      imageSize: rainImageSize,//粒子贴图尺寸  
      updateCallback: rainUpdate,//每帧都要调用一次回调函数以更新粒子  
    }  
  
    //默认下雪天  
    viewer.scene.primitives.add(new Cesium.ParticleSystem(snowOption));  
  
    //下拉框回调函数  
    var dropdown = document.getElementById('dropdown');  
    function change() {  
      switch (dropdown.value) {  
        case 'snow':  
          viewer.scene.primitives.removeAll();  
          viewer.scene.primitives.add(new Cesium.ParticleSystem(snowOption));  
          break;  
        case 'rain':  
          viewer.scene.primitives.removeAll();  
          viewer.scene.primitives.add(new Cesium.ParticleSystem(rainOption));  
          break;  
        case 'null':  
          viewer.scene.primitives.removeAll();  
          break;  
        default:  
          break;  
      }  
    }  
  
    //设置相机视角  
    /* viewer.scene.camera.setView({  
      destination:  
        Cesium.Cartesian3.fromDegrees(114.39664, 30.40052, 10000),  
      orientation: {  
        heading: 4.731089976107251,  
        pitch: -0.32003481981370063,  
      },  
    }) */  
  
    //设置相机初始位置  
    viewer.scene.camera.setView({  
      destination: new Cesium.Cartesian3(-2318006.190591779, 5016113.738321363,3239729.8052793955),  
      orientation: {  
        heading: 5.0433812878480655,  
        pitch: -0.25943108890985744,  
        roll: 0.000002292722656171975  
      },  
      duration: 0.0  
    });  
  
  </script>  
</body>  
  
</html>

(2)结果显示
  在这里插入图片描述
  在这里插入图片描述


cesium文章涉及数据

参考资料:
[1] 郭明强. 《WebGIS之Cesium三维软件开发》; 2023-04-01 [accessed 2024-01-27].
[2] WaqarLeaver. Cesium开源water材质和粒子效果示例代码研究; 2021-05-30 [accessed 2024-01-27].
[3] GIS兵墩墩. C2——cesium流动特效; 2020-11-04 [accessed 2024-01-27].
[4] 那那那那那么长的哲尘. Cesium实现流动线/动态纹理; 2024-01-11 [accessed 2024-01-27].

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

函数式接口当参数使用

如果函数式接口作为一个方法的参数&#xff0c;就以为着要方法调用方自己实现业务逻辑&#xff0c;常见的使用场景是一个业务整体逻辑是不相上下的&#xff0c;但是在某一个步骤有不同的逻辑&#xff0c;例如数据处理有不同的策略。上代码 package com.dj.lambda;import java.…

加密域可逆信息隐藏算法分类及评价指标

一、加密域可逆信息隐藏算法框架分类 加密图像可逆信息隐藏(RDHEI)是将图像加密和信息隐藏结合使用的一种技术。图像拥有者先对原始图像使用加密密钥keyc进行加密&#xff0c;信息隐藏者根据隐藏密钥keyd将秘密信息嵌入到密文图像中去。在接收端&#xff0c;接收者根据密钥key…

【Docker】快速入门手册

目录 1.概述 1.1.安装 1.2.阿里云镜像加速 1.3.运行原理 2.常用操作 2.1.帮助命令 2.2.镜像操作 2.3.容器操作 2.3.1创建、启动 2.3.2.退出、停止 2.3.3.进入交互式界面 2.3.4.守护式容器交互 2.3.5.查看 2.3.6.删除 2.3.7.拷贝 3.容器数据卷 3.1.概述 3.2.使…

linux03 用户权限

01.三种权限 02.UGO&#xff08;root账号&#xff09; 查看权限 不在root文件中写&#xff0c;是因为其他用户不能进来 举个例子 ll是ls -l 第一部分&#xff1a;权限&#xff08;11个字节&#xff09; 第一个&#xff1a;d/- d表示文件夹 - 表示一般文件 二到四&#xff1a…

R语言学习case6:ggplot基础画图(Scatter散点图)

step1: 导入ggplot2库文件 library(ggplot2)step2&#xff1a;带入自带的iris数据集 iris <- datasets::irisstep3&#xff1a;查看数据信息 dim(iris)维度为 [150,5] head(iris)查看数据前6行的信息 step4&#xff1a;利用ggplot工具包绘图 plot1 <- ggplot(iris…

人工智能的圣杯:关于可解释AI(XAI)的一切

​​​​​​​ 在过去十年间&#xff0c;无数个人工智能解决方案在各大企业得到部署。 智能受众评测系统、智能财务合规系统、智能人员招聘系统&#xff0c;不一而足。 这期间&#xff0c;在企业客户却也始终存在一种怀疑态度&#xff1a;AI系统做出的产品部署是否真的值得…

初识人工智能,一文读懂机器学习之逻辑回归知识文集(6)

&#x1f3c6;作者简介&#xff0c;普修罗双战士&#xff0c;一直追求不断学习和成长&#xff0c;在技术的道路上持续探索和实践。 &#x1f3c6;多年互联网行业从业经验&#xff0c;历任核心研发工程师&#xff0c;项目技术负责人。 &#x1f389;欢迎 &#x1f44d;点赞✍评论…

QT之 QDebug 调试(一)

在QT中&#xff0c;进行调试&#xff0c;则需要在头文件地方加上 #include <QDebug> 加上之后&#xff0c;在编译之后则其输出的信息则在应用程序输出那里显示信息。 其QDebug 信息调试则如&#xff1a; qDebug() << " 需要插入的信息 "…

以太网交换基础VLAN原理与配置

目录 7.以太网交换基础 7.1.以太网协议 7.2.以太网帧介绍 7.3.以太网交换机 7.4.同网段数据通信全过程 8.VLAN原理与配置 8.1.VLAN的基本概念 8.2.VLAN的应用 7.以太网交换基础 7.1.以太网协议 以太网是当今现有局域网(Local Area Network,LAN)采用的最通用的通信协议…

【王道数据结构】【chapter2线性表】【P44t16】

设有一个长度为n&#xff08;n为偶数&#xff09;的不带头结点的单链表且结点值都大于0&#xff0c;设计算法求这个单链表的最大的孪生和。孪生和的定义为一个结点值与其孪生结点值的和&#xff0c;对于第i个结点&#xff08;从0开始&#xff09;&#xff0c;其孪生结点为第n-i…

【RT-DETR有效改进】EfficientFormerV2移动设备优化的视觉网络(附对比试验效果图)

前言 大家好&#xff0c;我是Snu77&#xff0c;这里是RT-DETR有效涨点专栏。 本专栏的内容为根据ultralytics版本的RT-DETR进行改进&#xff0c;内容持续更新&#xff0c;每周更新文章数量3-10篇。 专栏以ResNet18、ResNet50为基础修改版本&#xff0c;同时修改内容也支持Re…

【JaveWeb教程】(35)SpringBootWeb案例之《智能学习辅助系统》登录功能的详细实现步骤与代码示例(8)

目录 案例-登录和认证1. 登录功能1.1 需求1.2 接口文档1.3 思路分析1.4 功能开发1.5 测试 案例-登录和认证 在前面的课程中&#xff0c;我们已经实现了部门管理、员工管理的基本功能&#xff0c;但是大家会发现&#xff0c;我们并没有登录&#xff0c;就直接访问到了Tlias智能…

DS:带头双向循环链表的实现(超详细!!)

创作不易&#xff0c;友友们给个三连吧&#xff01;&#xff01;&#xff01; 博主的上篇文章介绍了链表&#xff0c;以及单链表的实现。 单链表的实现&#xff08;超详细&#xff01;&#xff01;&#xff09; 其实单链表的全称叫做不带头单向不循环链表&#xff0c;本文…

STP生成树协议实验

实验大纲 一、什么是生成树协议 二、生成树原理 1.STP工作原理 2.STP主要参数 3.STP根网桥 4.STP协议版本 三、实验 1.构建网络拓扑结构图 2.配置IP地址&#xff08;8台PC机&#xff09;&#xff1a;192.168.7.1~192.168.7.8 3.配置SW1 4.配置SW2 5.配置SW3 6.配置…

浪潮信息打造高效算力架构 为金融业数字化坚实基座

新时期&#xff0c;数据智能已经逐渐成为金融商业中的重要力量&#xff0c;构建更强大的算力系统&#xff0c;推动金融业务的高效发展&#xff0c;已经成为了金融行业的目标。对此&#xff0c;浪潮信息也为金融客户提供了崭新的解决方案。此前&#xff0c;某银行基于浪潮信息量…

第二模块 函数模块

第二模块 函数&模块 day09 文件操作相关1. 文件操作1.1 读文件1.2 写文件1.3 文件打开模式1.4 常见功能1.5 上下文管理练习题 2.csv格式文件3.ini格式文件4.XML格式文件4.1 读取文件和内容4.2 读取节点数据4.3 修改和删除节点4.4 构建文档 5.Excel格式文件5.1 读Excel5.1 写…

ESP32 SPIFFS文件系统

简介 本章涉及知识点&#xff1a;ESP32 SPIFFS文件系统、日志输出。 ESP-IDF版本&#xff1a;V5.1.2 源码 小智学长的源码&#xff1a;DesktopScreen 7 文件系统 系统配置 如果是自己构建的项目&#xff0c;如图。要在CMakeLists中配置上spiffs。 如果是直接跑官方例程则忽略系…

wireshark利用sshdump自身组件进行远程实时抓包过滤

引言 以前在不了解wireshark可以远程抓包的时间&#xff0c;经常通过tcpdump在远程linux主机将抓包文件保存下来后&#xff0c;然后拖拽入windows中再打开&#xff0c;进行分析查看。 此过程比较繁琐&#xff0c;也不够实时。比较常用的抓包动作是仅出现某特征的报文后&#…

手动导入jar包到Maven的解决方案(简单有效!)

想要导入一个jar包到项目中&#xff0c;这个jar包在Maven中没有可以尝试以下方式。 第一步 先找到你maven的本地仓库&#xff0c;我的仓库就在这里&#xff0c;你可以根据你安装的maven找到你的目录 第二步 根据坐标创建文件夹。 这个依赖modbus4j.jar&#xff0c;Maven远…

Cesium.js实现显示点位对应的自定义信息弹窗(数据面板)

博客&#xff1a;关于Cesium的常见需求整理之点位和弹窗&#xff08;点位弹窗&#xff09; 博客&#xff1a;cesium添加点、线、面、文字、图标、模型等标绘 零、相关技术选型&#xff1a; Vue2 Vuecli5 Cesium.js 天地图 一、需求说明 在使用2D地图&#xff08;天地图、高德…