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

news2024/11/16 8:58:57

实现效果

前言

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);
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Hadoop大数据应用:Yarn 节点实现扩容与缩容

目录 一、实验 1.环境 2.Yarn 节点扩容 3.Yarn 节点缩容 二、问题 1.yarn启动服务报错 一、实验 1.环境 (1)主机 表1 主机 主机架构软件版本IP备注hadoop NameNode (已部署) SecondaryNameNode (已部署&…

【vue baidu-map】marker鼠标悬浮点击事件失效

要实现的效果:鼠标悬浮或者点击标注点会出现弹窗 验证过鼠标点击悬浮代码没问题,最后发现是控件样式影响的 ::v-deep #bjmap .BMap_noprint {inset: 10px 90% auto auto !important; } 只要增加上述样式,鼠标悬浮事件就会失效

QT使用dumpcpp为COM生成h及cpp的方式,COM是C#的dll注册的

目录 1.C#的dll注册为COM,采用bat的方式 2.通过qt的dumpcpp来生成h及cpp文件 3.h文件和cpp文件处理。 台达数控系统的C#的dll dumpcpp用的tlb文件 dumpcpp生成的原生h文件 dumpcpp生成的原生cpp dump生成后的的原生cpp文件修改后的cpp文资源 dump生成后的的…

【探索Linux】—— 强大的命令行工具 P.27(网络编程套接字 —— UDP协议介绍 | TCP协议介绍 | UDP 和 TCP 的异同)

阅读导航 引言一、UDP协议1. UDP简介2. UDP的特点3. UDP的使用场景4. UDP的局限性 二、TCP协议1. TCP简介2. TCP的特点3. TCP的应用场景 三、UDP 和 TCP 的异同温馨提示 引言 在上一篇文章中,我们深入探讨了Linux网络编程的基石——套接字(Socket&#…

pytorch loss及其梯度

目录 1.loss的种类1.1 MSE1.2 MSE推导1.3 autograd.grad1.4 loss.backward 2. Softmax2.1 softmax推导 1.loss的种类 常见两种一种是均方差,一种是交叉熵 1.1 MSE 1.2 MSE推导 1.3 autograd.grad 1.4 loss.backward 注意:autograd.grad直接返回梯度&a…

守护健康,从营养开始 —— 帕金森患者的饮食秘籍

亲爱的读者朋友们,您是否知道,在对抗帕金森病的道路上,正确的饮食和营养补充可以成为我们的有力盟友?今天,就让我们一起探索那些能够帮助帕金森患者改善症状、提高生活质量的营养素,开启健康生活的新篇章。…

【保姆级】GPT的Oops问题快速解决方案

GPT的"Oops"问题通常指的是GPT在处理请求时突然遇到错误或无法提供预期输出的情况。要快速解决这个问题,可以尝试以下分步策略: 确认问题范围: 首先,确认问题是偶发的还是持续存在的。如果是偶发的,可能是临…

深度学习_GoogLeNet_4

目标 知道GoogLeNet网络结构的特点能够利用GoogLeNet完成图像分类 一、开发背景 GoogLeNet在2014年由Google团队提出, 斩获当年ImageNet(ILSVRC14)竞赛中Classification Task (分类任务) 第一名,VGG获得了第二名,为了向“LeNet”致敬&#x…

不同抓手的码垛机:适配多元应用场景的灵活之选

在现代工业生产中,码垛机作为一种高效、自动化的物料搬运设备,已经广泛应用于各个行业。而抓手作为码垛机的关键部件,其种类的多样性和适用场景的广泛性,使得不同抓手的码垛机能够满足各种复杂、多变的生产需求。 首先&#xff0c…

蓝桥杯单片机快速开发笔记——LED、蜂鸣器和继电器

一、原理分析 二、简单示例 总结:HC138令Y5C等于1后,通过控制P0^4、P0^6置1打开、置0关闭,便可以控制继电器和蜂鸣器,具体看上述的原理分析,LED同理通过给P0置0便可以控制LED点亮,利用本专栏上一节知识即可简单控制 …

vue 引用百度地图

address.vue <template><div><!-- 地图 --><el-drawer:visible.sync"type1"direction"rtl"size"50%"append-to-bodyclass"map-drawer":before-close"beforeClose"><div style"width: 100%…

Linux学习笔记(一)Linux基本指令

文章目录 前言目录常见命令1. pwd 打印当前所在路径2. cd 改变路径、切换路径3. 家目录 回到顶级目录4. 当前路径和上一路径5. 上一次路径6. 绝对路径和相对路径7. ls 列出目录内容8. mkdir 创建目录9. rmdir 删除目录10. touch 创建文件11. mv 修改文件目录、移动路径12. cp 复…

Vue3基础笔记(1)模版语法 属性绑定 渲染

Vue全称Vue.js是一种渐进式的JavaScript框架&#xff0c;采用自底向上增量开发的设计&#xff0c;核心库只关注视图层。性能丰富&#xff0c;完全有能力驱动采用单文件组件和Vue生态系统支持的库开发的复杂单页应用&#xff0c;适用于场景丰富的web前端框架。灵活性和可逐步集成…

一周学会Django5 Python Web开发-Jinja3模版引擎-模板语法

锋哥原创的Python Web开发 Django5视频教程&#xff1a; 2024版 Django5 Python web开发 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili2024版 Django5 Python web开发 视频教程(无废话版) 玩命更新中~共计37条视频&#xff0c;包括&#xff1a;2024版 Django5 Python we…

Linux系统架构----Tomcat 部署

一.Tomcat概述 Tomcat服务器是一个免费的开放式源代码的web应用服务器&#xff0c;属于轻量级应用级服务器&#xff0c;在中小型系统和并发访问用户不是很多的场合下被普遍使用&#xff0c;是开发和调试JSP程序的首首选。 一般来说&#xff0c;tomcat虽然和Apache或者Nginx这些…

ThingsBoard 开源物联网平台

文章目录 1.ThingsBoard 介绍2.ThingsBoard 架构2.1.单体架构2.2.微服务架构 3.物联网网关4.边缘计算 ThingsBoard # ThingsBoardhttps://iothub.org.cn/docs/iot/ https://iothub.org.cn/docs/iot/thingsboard-ce/1.ThingsBoard 介绍 ThingsBoard 是一个开源物联网平台&…

MySQL 数据库 下载地址 国内阿里云站点

mysql安装包下载_开源镜像站-阿里云 以 MySQL 5.7 为例 mysql-MySQL-5.7安装包下载_开源镜像站-阿里云

C++ 拷贝构造函数和运算符重载

目录 一. 拷贝构造函数 1. 引入 2. 拷贝构造的概念 3. 浅拷贝 4. 深拷贝 二. C运算符重载 1. 概念 2. 注意事项 3.举例 一. 拷贝构造函数 1. 引入 我们在创建对象时&#xff0c;能不能创建一个与原先对象一模一样的新对象呢&#xff1f;为了解决这个问题&#x…

Qt/QML编程之路:基于QWidget编程及各种2D/3D/PIC绘制的示例(45)

关于使用GWidget,这里有一个示例,看了之后很多图形绘制,控件使用,及最基本的QWidget编程都比较清楚了。ui的绘制: 运行后的界面如 工程中有非常丰富的关于各种图形的绘制,比如上图中circle,还有image。有下面一段readme的说明: # EasyQPainter Various operation pra…

java数据结构与算法刷题-----LeetCode47. 全排列 II

java数据结构与算法刷题目录&#xff08;剑指Offer、LeetCode、ACM&#xff09;-----主目录-----持续更新(进不去说明我没写完)&#xff1a;https://blog.csdn.net/grd_java/article/details/123063846 文章目录 1. 暴力回溯2. 分区法回溯 此题为46题的衍生题&#xff0c;在46题…