【CesiumJS-5】绘制动态路线实现飞行航线、汽车轨迹、路径漫游等

实现效果

前言

Cesium中,动态路线绘制的核心是借助CZML格式,CZML是一种用来描述动态场景的JSON数组,可以用来描述点、线、多边形、体、模型及其他图元,同时定义它们是怎样随时间变化的

CZML主要做三件事:

1.添加模型信息

2.添加路径信息

3.计算速度,修改时间

CZML数据格式

[
  {
    "id": "document",
    "version": "1.0",
    "clock": {
      "interval": "2024-03-13T03:00:00Z/2024-03-13T03:12:50Z",
      "currentTime": "2024-03-13T03:00:00Z",
      "multiplier": 5
    }
  },
  {
    "id": "plane",
    "availability": "2024-03-13T03:00:00Z/2024-03-13T03:12:50Z",
    "viewFrom": {
      cartesian: [-2080, -1715, 779]
    },
    "model": {
      "gltf": "/model/plane.glb",
      "scale": 2
    },
    "orientation": {
      "velocityReference": "#position"
    },
    "path": {
      "material": {
        "solidColor": {
          "color": {
            "interval": "2024-03-13T03:00:00Z/2024-03-13T03:12:50Z",
            "rgba": [0, 0, 205, 255]
          }
        }
      },
      "width": [
        {
          "interval": "2024-03-13T03:00:00Z/2024-03-13T03:12:50Z",
          "number": 2
        }
      ],
      "show": [
        {
          "interval": "2024-03-13T03:00:00Z/2024-03-13T03:12:50Z",
          "boolean": true
        }
      ]
    },
    "position": {
      "interpolationAlgorithm": "LAGRANGE",
      "interpolationDegree": 1,
      "epoch": "2024-03-13T03:00:00Z",
      "cartographicDegrees": []
    }
  }
]

主要有两个对象组成:第一个对象声明czml,第二个对象生成模型。

1. 声明czml
  {
    "id": "document",
    "version": "1.0",
    "clock": {
      "interval": "2024-03-13T03:00:00Z/2024-03-13T03:12:50Z",
      "currentTime": "2024-03-13T03:00:00Z",
      "multiplier": 5
    }
  }
2. 设置相机位置
    "viewFrom": {
      cartesian: [-2080, -1715, 779]
    },
3. 生成场景模型,设置时间范围
  {
    "id": "plane",
    "availability": "2024-03-13T03:00:00Z/2024-03-13T03:12:50Z",
    "model": {
      "gltf": "/model/plane.glb",
      "scale": 2
    }
  }
4. 设置模型运动方向
    "orientation": {
      "velocityReference": "#position"
    },
5. 生成动态路径
    "path": {
      "material": {
        "solidColor": {
          "color": {
            "interval": "2024-03-13T03:00:00Z/2024-03-13T03:12:50Z",
            "rgba": [
              0,
              0,
              205,
              255
            ]
          }
        }
      },
      "width": [
        {
          "interval": "2024-03-13T03:00:00Z/2024-03-13T03:12:50Z",
          "number": 2
        }
      ],
      "show": [
        {
          "interval": "2024-03-13T03:00:00Z/2024-03-13T03:12:50Z",
          "boolean": true
        }
      ]
    },
6. 设置运动路线

cartographicDegrees设置经纬度:4个一组分别是 时间,经度,纬度,高度

    "position": {
      "interpolationAlgorithm": "LAGRANGE",
      "interpolationDegree": 1,
      "epoch": "2024-03-13T03:00:00Z",
      "cartographicDegrees": [0, 120.13125463180745, 30.316189813141357, 20]
    }

功能实现

本人在实现功能中没用json格式,而是直接用的数组对象变量,上述中仅做学习记录。

注意: czml只能识别ISO8601日期,需要使用Cesium内置方法转换时间;

1. 将当前时间作为开始时间并将js日期转为儒略日格式
const start = Cesium.JulianDate.fromDate(new Date());
2. 创建czml数组

使用Cesium.JulianDate.toIso8601()方法将时间转为ISO8601日期格式

const czml = [
  {
    id: "document",
    version: "1.0",
    clock: {
      interval: `${Cesium.JulianDate.toIso8601(start, 0)}/${Cesium.JulianDate.toIso8601(Cesium.JulianDate.addSeconds(start, 450, new Cesium.JulianDate()), 0)}`,
      currentTime: `${Cesium.JulianDate.toIso8601(start, 0)}`,
      multiplier: 10,
    }
  },
  {
    id: "plane",
    availability: `${Cesium.JulianDate.toIso8601(start, 0)}/${Cesium.JulianDate.toIso8601(Cesium.JulianDate.addSeconds(start, 450, new Cesium.JulianDate()), 0)}`,
    model: {
      gltf: "/model/plane.glb",
      scale: 2
    },
    // viewFrom: {
    //   cartesian: [-2080, -1715, 779]
    // },
    orientation: {
      velocityReference: "#position"
    },
    path: {
      material: {
        solidColor: {
          color: {
            interval: `${Cesium.JulianDate.toIso8601(start, 0)}/${Cesium.JulianDate.toIso8601(Cesium.JulianDate.addSeconds(start, 450, new Cesium.JulianDate()), 0)}`,
            rgba: [0, 0, 205, 255]
          }
        }
      },
      width: [
        {
          interval: `${Cesium.JulianDate.toIso8601(start, 0)}/${Cesium.JulianDate.toIso8601(Cesium.JulianDate.addSeconds(start, 450, new Cesium.JulianDate()), 0)}`,
          number: 2
        }
      ],
      show: [
        {
          interval: `${Cesium.JulianDate.toIso8601(start, 0)}/${Cesium.JulianDate.toIso8601(Cesium.JulianDate.addSeconds(start, 450, new Cesium.JulianDate()), 0)}`,
          boolean: true
        }
      ]
    },
    position: {
      interpolationAlgorithm: "LAGRANGE",
      interpolationDegree: 1,
      epoch: `${Cesium.JulianDate.toIso8601(start, 0)}`,
      cartographicDegrees: [
        0, 120.13125463180745, 30.316189813141357, 5,
        10, 120.13111416662538, 30.315539106519307, 5,
        20, 120.1309786622693, 30.31489588182787, 5,
        30, 120.13084876734133, 30.314235695359727, 5,
        40, 120.13069473575192, 30.313540629197277, 5,
        50, 120.13054390032266, 30.312912654513724, 5,
        60, 120.13044947371688, 30.312385482730374, 5,
        70, 120.13030971147948, 30.311910736974383, 5,
        80, 120.13009317468772, 30.311390245082837, 5,
        90, 120.12978893985873, 30.310733495046055, 5,
        100, 120.12959024998796, 30.310326531696383, 10,
        110, 120.12938465577722, 30.309954975972268, 30,
        120, 120.12903852757415, 30.309336955328902, 60,
        130, 120.12871206396551, 30.308875801472478, 70,
        140, 120.12838069578686, 30.308490868344943, 70,
        150, 120.12810457466928, 30.308113523536672, 70,
        160, 120.12779772791689, 30.307774086249907, 80,
        170, 120.12737964279644, 30.307483823729378, 80,
        180, 120.1269732925855, 30.307160879171903, 80,
        190, 120.12647213239983, 30.30666310390423, 80,
        200, 120.12546262988464, 30.307181930544864, 80,
        210, 120.1245785033241, 30.307683795316315, 80,
        220, 120.12375576753382, 30.308035432883194, 100,
        230, 120.12308850447974, 30.308497699718018, 100,
        240, 120.12253685192415, 30.308860660895174, 100,
        250, 120.12209783635902, 30.309214249085528, 100,
        260, 120.12124539807733, 30.309730900111948, 120,
        270, 120.12062513785223, 30.30968143768237, 125,
        280, 120.11996146408914, 30.30903962813221, 130,
        290, 120.12080592297113, 30.308199027265523, 135,
        300, 120.1206419712799, 30.307743185311676, 140,
        310, 120.11991139765998, 30.306939478614392, 200,
        320, 120.11894945108537, 30.307236007284757, 220,
        330, 120.11843736124806, 30.307682602103153, 240,
        340, 120.11801295026378, 30.308682424137178, 260,
        350, 120.117608250043, 30.30896814314354, 280,
        360, 120.11756912872843, 30.309470200064347, 300,
        370, 120.11718042273918, 30.310442342271475, 300,
        380, 120.11602779574089, 30.3098989524904, 310,
        390, 120.11550616761767, 30.31093863353403, 310,
        400, 120.11644406776917, 30.31177509905328, 310,
        410, 120.11566744726449, 30.312255943277282, 320,
        420, 120.11627775321915, 30.312631949114188, 320,
        430, 120.11695495518998, 30.313251980473588, 330,
        440, 120.11755845523233, 30.313636377593916, 330,
        450, 120.11843011762987, 30.31389033163698, 340
      ]
    }
  }
]
3. 加载CZML数据并添加到场景中

dataSources:是一个异步Promise,回调参数为我们传入的czml数据;

let dataSource = viewer.dataSources.add(Cesium.CzmlDataSource.load(czml));
4. 相机跟随模型
  dataSource.then(part => {
    // 根据id获取czml中的模型
    let e = part.entities.getById("plane");
    // 相机跟随模型
    viewer.trackedEntity = e;
  });
5. 视角跟随模型
  dataSource.then(part => {
    // 根据id获取czml中的模型
    let e = part.entities.getById("plane");
    // 相机跟随模型
    viewer.trackedEntity = e;

    //++++++++++++++
    // 视角跟随模型
    let prePoint = null; // 前一个点
    viewer.scene.postRender.addEventListener(() => {
      if (e && viewer.clock.shouldAnimate) {
        // 获取当前时间的位置
        let curPoint = e.position.getValue(viewer.clock.currentTime);
        if (prePoint) {
          // 计算heading-代表 Z 轴旋转
          let heading = getHeading(prePoint, curPoint);
          // 计算pitch-代表 Y 轴朝向
          let pitch = Cesium.Math.toRadians(-30.0);
          let range = 100;
          viewer.camera.lookAt(curPoint, new Cesium.HeadingPitchRange(heading, pitch, range));
        }
        // 当前点在下一次渲染时为前一个点
        prePoint = Cesium.Cartesian3.clone(curPoint)
      }
    }) 
  });


function getHeading(pointA, pointB) {
  //建立以点A为原点,X轴为east,Y轴为north,Z轴朝上的坐标系
  const transform = Cesium.Transforms.eastNorthUpToFixedFrame(pointA);
  //向量AB
  const positionvector = Cesium.Cartesian3.subtract(pointB, pointA, new Cesium.Cartesian3());
  //因transform是将A为原点的eastNorthUp坐标系中的点转换到世界坐标系的矩阵
  //AB为世界坐标中的向量
  //因此将AB向量转换为A原点坐标系中的向量,需乘以transform的逆矩阵。
  const vector = Cesium.Matrix4.multiplyByPointAsVector(Cesium.Matrix4.inverse(transform, new Cesium.Matrix4()), positionvector, new Cesium.Cartesian3());
  //归一化
  const direction = Cesium.Cartesian3.normalize(vector, new Cesium.Cartesian3());
  //heading
  const heading = Math.atan2(direction.y, direction.x) - Cesium.Math.PI_OVER_TWO;
  return Cesium.Math.TWO_PI - Cesium.Math.zeroToTwoPi(heading);
}

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

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

相关文章

热流道融合3D打印技术正在成为模具制造新利器

在模具领域中,3D打印技术与热流道技术联手,能迸发出更耀眼的光芒。两种技术虽然各有特点,但两者结合将形成互补作用,从而实现11>2”的跨越式提升。 将增材制造的灵活思维融入传统模具设计时,不仅能够突破传…

ubuntu下docker安装

目录 官网链接 安装步骤 docker使用方法 拉取镜像 创建镜像 运行镜像 查看运行结果 保存镜像文件 传输到windows下 官网链接 Install Docker Engine on Ubuntu | Docker Docs 安装步骤 1.运行以下命令卸载所有冲突的包: for pkg in docker.io docker-d…

jeecg 启动 微服务 更改配置本地host地址

127.0.0.1 jeecg-boot-redis 127.0.0.1 jeecg-boot-mysql 127.0.0.1 jeecg-boot-nacos 127.0.0.1 jeecg-boot-gateway 127.0.0.1 jeecg-boot-system 127.0.0.1 jeecg-boot-sentinel 127.0.0.1 jeecg-boot-xxljob 127.0.0.1 jeecg-boot-rabbitmq1. windows系统下,在开…

JAVA实战开源项目:车险自助理赔系统(Vue+SpringBoot)

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 数据中心模块2.2 角色管理模块2.3 车辆档案模块2.4 车辆理赔模块2.5 理赔照片模块 三、系统设计3.1 用例设计3.2 数据库设计3.2.1 角色表3.2.2 车辆表3.2.3 理赔表3.2.4 理赔照片表 四、系统展示五、核心代码5.1 查询车…

网站搭建教程

网站搭建教程 一.领取一个免费域名和SSL证书,和CDN 1.打开网站链接:https://www.rainyun.com/ycpcp_ 首先创建一个CDN,这里以我加速域名“cdntest.biliwind.com 1”为例 这里就要填写 cdntest.biliwind.com 1 ,而不是 https:/…

Linux学习笔记:什么是文件描述符

什么是文件描述符 C语言的文件接口文件的系统调用什么是文件描述符,文件描述符为什么是int类型?为什么新打开的文件的文件描述符不是从0开始? 文件描述符 fd (file descriptor) C语言的文件接口 当时学习C语言的时候,学习了文件接口 具体可以查看之前的文章: 链接:C语言的文…

鸿蒙Harmony应用开发—ArkTS声明式开发(基础手势:Select)

提供下拉选择菜单&#xff0c;可以让用户在多个选项之间选择。 说明&#xff1a; 该组件从API Version 8开始支持。后续版本如有新增内容&#xff0c;则采用上角标单独标记该内容的起始版本。 子组件 无 接口 Select(options: Array<SelectOption>) 参数&#xff1a;…

Autosar教程-Mcal教程-Fls配置教程

3.11.1 FLS基础知识 flash操作中有两个术语:block和page。block是flash最小的擦除单位,page则是flash写入的最小单位。以我们使用的F1KM-S4(R7F7016533)来说,它的是64 bytes, page是4bytes。这也就意味着,如果要擦除的话,最小要擦除64 bytes,但是写入可以按4字节的大小写入…

SketchUp Pro 2023 for Mac/Win:重塑设计,引领未来

在数字化浪潮席卷全球的今天&#xff0c;设计行业也迎来了前所未有的变革。SketchUp Pro 2023&#xff0c;这款专为设计师打造的草图大师软件&#xff0c;正以其强大的功能和卓越的性能&#xff0c;引领着设计界的新潮流。 SketchUp Pro 2023不仅继承了前代产品的优秀基因&…

【Java设计模式】二十三、解释器模式

文章目录 1、解释器模式2、案例 1、解释器模式 计算一个表达式的值&#xff0c;比如12-34-7&#xff0c;单纯的定义方法或函数很难适配所有&#xff0c;因为数值和运算符可以有无数种组合。 //用于n个整数相加 public static int add(Integer ... arr) {int sum 0;for (Inte…

网络学习:BGP路径属性分类

目录 前言&#xff1a; 路径属性分类 公认必遵 公认任意 可选过渡 可选非过渡 前言&#xff1a; 在默认情况下&#xff0c;到达同一目的地&#xff0c;BGP只走单条路径&#xff0c;并不会在多条路径之间执行负载均衡。对于IGP路由协议&#xff0c;当有多条路径可以到达同…

腐烂的橘子 力扣bfs

BFS用来搜索最短路径的解法是比较合适的。 比如求最少步数的解&#xff0c;最少交换次数的解&#xff0c;最快走出迷宫等等&#xff0c;因为bfs搜索过程中遇到的第一个解一定是离最初位置最近的&#xff0c;所以遇到第一个解&#xff0c;一定就是最优解&#xff0c;此时搜索算…

表单修饰符和事件修饰符

表单修饰符和事件修饰符 表单修饰符 v-model.lazy v-model.lazy 失去焦点后再收集数据 <div id"app"><textarea name"" id"" cols"30" rows"10" v-model.lazy"a"></textarea>{{a}}<textar…

水果音乐编曲和制作软件 FL Studio 中文汉化破解版下载 FL Studio 中文设置教程

FL Studio 21 也就是 Image-Line 出品的一款功能强大的编曲软件&#xff0c;全名 Fruity Loops Studio 简称“FL Studio”。 FL Studio 21.2汉化破解版是功能强大的音乐制作解决方案&#xff0c;使用旨在为用户提供一个友好完整的音乐创建环境&#xff0c;让您能够轻松创建、管…

【ansible】ansible的介绍和安装

前言运维自动化 云计算核心职能 搭建平台架构 日常运营保障 性能效率优化 相关工具 代码管理&#xff08;SCM&#xff09;&#xff1a;GitHub、GitLab、BitBucket、SubVersion 构建工具&#xff1a;maven、Ant、Gradle 自动部署&#xff1a;Capistrano、CodeDeploy 持续…

JVM-5

1.选择垃圾收集器 如果你的堆大小不是很大&#xff08;比如 100MB &#xff09;&#xff0c;选择串行收集器一般是效率最高的。 参数&#xff1a; -XX:UseSerialGC 。如果你的应用运行在单核的机器上&#xff0c;或者你的虚拟机核数只有单核&#xff0c;选择串行收集器依然是合…

P6技巧:ORACLE Primavera P6 反馈项目Reflections的使用

前言 反馈是一个有趣的概念&#xff0c;就目前的了解而言&#xff0c;他是 Primavera P6 所独有的。 你可以将反馈视为项目的特殊假设副本。 然而&#xff0c;与直接拷贝副本不同的是&#xff0c;反馈保留了返回源项目的链接。 这意味着如果反馈发生更改&#xff0c;你可以将部…

Javascript 线性搜索算法

线性搜索被定义为一种顺序搜索算法&#xff0c;从一端开始&#xff0c;遍历列表中的每个元素&#xff0c;直到找到所需的元素&#xff0c;否则搜索将继续&#xff0c;直到数据集的末尾。 线性搜索算法 线性搜索算法如何工作&#xff1f; 在线性搜索算法中&#xff1a; …

C++语法、Linux命令查询网站

文章目录 1.cplusplus2.cppreference3.Linux命令查询网站 1.cplusplus 网址&#xff1a;https://legacy.cplusplus.com/ 2.cppreference 1.cppreference中文网站&#xff1a;https://zh.cppreference.com/w/首页 2.cppreference英文原站&#xff1a;https://en.cppreference…

openssl3.2 - note - Decoders and Encoders with OpenSSL

文章目录 openssl3.2 - note - Decoders and Encoders with OpenSSL概述笔记编码器/解码器的调用链OSSL_STORE 编码器/解码器的名称和属性OSSL_FUNC_decoder_freectx_fnOSSL_FUNC_encoder_encode_fn官方文档END openssl3.2 - note - Decoders and Encoders with OpenSSL 概述 …