Cesium 简单航飞线路规划,模拟飞行扫描效果

目录

效果

源码

使用注意

追加说明

参考文档


效果

 动图:

源码

vue2+elementUI+cesium1.97,功能源码已整理到单个文件中。单独运行需要引入cesium、elementUI和turf.js,原理看注释基本都能看懂。上源码:

<template>
  <!-- 航线规划 -->
  <div class="uvaRoutePlanBox" :class="{ hide: !isShow, show: isShow }">
    <div id="cesiumContainerBox" />
    <div
      id="toolTip"
      style="display: none;pointer-events: none;position: fixed;background: rgba(0,0,0,0.5);z-index: 1000;opacity: 0.8;border-radius: 4px;padding: 4px 8px;white-space: nowrap;font-family:黑体;color:white;font-weight: bolder;font-size: 14px;"
    />
    <div class="body">
      <div class="form-item">
        <span class="form-label">
          航飞区域
        </span>
        <div class="form-connect">
          <el-button type="primary" style="width:130px;" @click="drawPoly">
            绘制
          </el-button>
        </div>
      </div>
      <div class="form-item">
        <span class="form-label">航飞间距</span>
        <div class="form-connect">
          <el-input-number
            v-model="hfDistance"
            controls-position="right"
            :min="10"
            :step="10"
            @change="distanceHandleChange"
          />
        </div>
      </div>
      <div class="form-item">
        <div class="form-connect">
          <el-button type="primary" @click="beginCalc">
            开始计算
          </el-button>
          <el-button type="success" @click="moniFly">
            模拟飞行
          </el-button>
          <el-button type="danger" @click="cleanEntity">清除</el-button>
        </div>
      </div>
    </div>
  </div>
</template>

<script>
import * as Cesium from '@assets/Cesium'
import * as turf from '@turf/turf'


var polyArr = [] // 面数据
var jdArrs = [] // 交点集合
export default {
  name: 'UvaRoutePlan',
  components: {},
  props: [],
  data() {
    return { isShow: true, hfDistance: 100, isFly: false }
  },
  watch: {},
  created() {
    this.$nextTick(function() {
      this.startInit()
    })
  },
  mounted() {},
  methods: {
    startInit() {
      const key =
        '你的token'
      Cesium.Ion.defaultAccessToken = key

      var viewer = new Cesium.Viewer('cesiumContainerBox', {
        imageryProvider: new Cesium.ArcGisMapServerImageryProvider({
          url:
            'https://services.arcgisonline.com/ArcGIS/rest/services/World_Imagery/MapServer'
        }),
        geocoder: true,
        homeButton: true,
        sceneModePicker: true,
        baseLayerPicker: true,
        navigationHelpButton: true,
        shouldAnimate: true,
        animation: true,
        timeline: true,
        fullscreenButton: true,
        vrButton: true,
        // 关闭点选出现的提示框
        selectionIndicator: false,
        infoBox: false
      })
      viewer._cesiumWidget._creditContainer.style.display = 'none' // 隐藏版权
      viewer.camera.flyTo({
        destination: Cesium.Cartesian3.fromDegrees(
          113.6440552299206,
          34.78411814959118,
          2000
        ),
        orientation: {
          heading: Cesium.Math.toRadians(0.0), // 左右方向
          pitch: Cesium.Math.toRadians(-90.0), // 上下方向
          roll: Cesium.Math.toRadians(0) // 镜头(屏幕)到定位目标点(实体)的距离
        },
        duration: 3 // 执行定位动画的时间
      })
      window.viewer = viewer
    },
    endClose() {
      this.cleanEntity()
    },
    // 画航飞区域
    drawPoly() {
      var viewer = window.viewer
      this.cleanEntity()
      var handler = new Cesium.ScreenSpaceEventHandler(viewer.scene.canvas)
      // 鼠标事件
      handler = new Cesium.ScreenSpaceEventHandler(
        viewer.scene._imageryLayerCollection
      )
      polyArr = []
      var positions = []
      var tempPoints = []
      var polygon = null
      var tooltip = document.getElementById('toolTip')
      var cartesian = null
      // 鼠标移动事件
      handler.setInputAction(function(movement) {
        tooltip.style.left = movement.endPosition.x + 3 + 'px'
        tooltip.style.top = movement.endPosition.y - 25 + 'px'
        tooltip.innerHTML = '<p>单击开始,右击结束</p>'
        const ray = viewer.camera.getPickRay(movement.endPosition)
        cartesian = viewer.scene.globe.pick(ray, viewer.scene)
        if (positions.length >= 2) {
          if (!Cesium.defined(polygon)) {
            polygon = new PolygonPrimitive(positions)
          } else {
            positions.pop()
            positions.push(cartesian)
          }
        }
      }, Cesium.ScreenSpaceEventType.MOUSE_MOVE)
      // 左击鼠标事件
      handler.setInputAction(function(movement) {
        tooltip.style.display = 'block'
        tooltip.style.left = movement.position.x + 3 + 'px'
        tooltip.style.top = movement.position.y - 25 + 'px'
        tooltip.innerHTML = '<p>单击开始,右击结束</p>'
        const ray = viewer.camera.getPickRay(movement.position)
        cartesian = viewer.scene.globe.pick(ray, viewer.scene)
        if (positions.length === 0) {
          positions.push(cartesian.clone())
        }

        // positions.pop();
        positions.push(cartesian)
        // 在三维场景中添加点
        var cartographic = Cesium.Cartographic.fromCartesian(
          positions[positions.length - 1]
        )
        var longitudeString = Cesium.Math.toDegrees(cartographic.longitude)
        var latitudeString = Cesium.Math.toDegrees(cartographic.latitude)
        var heightString = cartographic.height
        tempPoints.push({
          lon: longitudeString,
          lat: latitudeString,
          hei: heightString
        })
      }, Cesium.ScreenSpaceEventType.LEFT_CLICK)
      // 右击鼠标事件
      handler.setInputAction(function() {
        handler.destroy()
        tooltip.style.display = 'none'
        positions.pop()
        polyArr = positions
      }, Cesium.ScreenSpaceEventType.RIGHT_CLICK)
      var PolygonPrimitive = (function() {
        function _(positions) {
          this.options = {
            name: 'uav-poly',
            polygon: {
              hierarchy: [],
              // perPositionHeight : true,
              material: Cesium.Color.fromCssColorString('#fafafa').withAlpha(
                0.5
              )
            }
          }

          this.hierarchy = { positions }
          this._init()
        }

        _.prototype._init = function() {
          var _self = this
          var _update = function() {
            return _self.hierarchy
          }
          // 实时更新polygon.hierarchy
          this.options.polygon.hierarchy = new Cesium.CallbackProperty(
            _update,
            false
          )
          viewer.entities.add(this.options)
        }

        return _
      })()
    },
    // 开始计算
    beginCalc() {
      if (polyArr.length === 0) {
        this.$message.warning('未检测到航飞区域')
        return
      }
      var viewer = window.viewer
      // 清除之前计算结果entity
      this.cesium.removeEntityLikeName(viewer, 'uav-tmp')
      var geodesic = Cesium.BoundingRectangle.fromPoints(polyArr) // 外接矩形获取
      var rectangle = Cesium.Rectangle.fromCartesianArray(polyArr) // 外接矩形获取
      // 把面拆分成线并转换为经纬度格式存储
      var polyLines = []
      polyArr.map((res, index) => {
        var tmp = []
        var ellipsoid = viewer.scene.globe.ellipsoid
        var cartesian3 = new Cesium.Cartesian3(res.x, res.y, res.z)
        var cartographic = ellipsoid.cartesianToCartographic(cartesian3)
        var cartesian31 = null
        var cartographic1 = null
        if (index === polyArr.length - 1) {
          // 最后一个点连接顶点
          cartesian31 = new Cesium.Cartesian3(
            polyArr[0].x,
            polyArr[0].y,
            polyArr[0].z
          )
          cartographic1 = ellipsoid.cartesianToCartographic(cartesian31)

          tmp.push([
            [
              Cesium.Math.toDegrees(cartographic.longitude),
              Cesium.Math.toDegrees(cartographic.latitude)
            ],
            [
              Cesium.Math.toDegrees(cartographic1.longitude),
              Cesium.Math.toDegrees(cartographic1.latitude)
            ]
          ])
        } else {
          cartesian31 = new Cesium.Cartesian3(
            polyArr[index + 1].x,
            polyArr[index + 1].y,
            polyArr[index + 1].z
          )
          cartographic1 = ellipsoid.cartesianToCartographic(cartesian31)

          tmp.push([
            [
              Cesium.Math.toDegrees(cartographic.longitude),
              Cesium.Math.toDegrees(cartographic.latitude)
            ],
            [
              Cesium.Math.toDegrees(cartographic1.longitude),
              Cesium.Math.toDegrees(cartographic1.latitude)
            ]
          ])
        }
        polyLines.push(tmp)
      })
      // 等高分割矩形
      var bool = geodesic.width < geodesic.height
      var len = Math.floor(geodesic.height / (this.hfDistance / 2)) // 高度分割数量
      var step = rectangle.height / len // 步长
      if (bool) {
        len = Math.floor(geodesic.width / (this.hfDistance / 2)) // 宽度分割数量
        step = rectangle.width / len // 步长
      }
      jdArrs = [] // 交点集合
      //   console.log(rectangle, len, step)
      for (var i = 0; i < len; i++) {
        var tmp = null
        if (bool) {
          tmp = new Cesium.Rectangle(
            rectangle.east - step * (i + 1),
            rectangle.south,
            rectangle.east - step * i,
            rectangle.north
          )
        } else {
          tmp = new Cesium.Rectangle(
            rectangle.west,
            rectangle.north - step * (i + 1),
            rectangle.east,
            rectangle.north - step * i
          )
        }
        // 弧度转换为经纬度
        var tmpLonLat = this.rectangle2LonLat(tmp)
        // 计算交点
        var tmpJdarr = []
        polyLines.map(res => {
          var mb = null
          if (bool) {
            mb = turf.lineString([
              [tmpLonLat[1][0], tmpLonLat[1][1]],
              [tmpLonLat[3][0], tmpLonLat[3][1]]
            ])
          } else {
            mb = turf.lineString([
              [tmpLonLat[1][0], tmpLonLat[1][1]],
              [tmpLonLat[0][0], tmpLonLat[0][1]]
            ])
          }

          var intersects = turf.lineIntersect(turf.lineString(res[0]), mb)
          if (intersects.features.length > 0) {
            var tmplatlon = intersects.features[0].geometry.coordinates
            tmpJdarr.push(tmplatlon)
          }
        })
        // 就近往返
        if (i > 0) {
          var distance1 = turf.distance(
            turf.point(tmpJdarr[0]),
            turf.point(jdArrs[jdArrs.length - 1]),
            { units: 'kilometers' }
          )
          var distance2 = turf.distance(
            turf.point(tmpJdarr[1]),
            turf.point(jdArrs[jdArrs.length - 1]),
            { units: 'kilometers' }
          )
          //   console.log(i, distance1, distance2)
          if (distance1 > distance2) {
            tmpJdarr = tmpJdarr.reverse()
          }
        }
        // 存储交点
        tmpJdarr.map(res => {
          jdArrs.push(res)
        })
      }
      if (jdArrs[0][0] === jdArrs[1][0] && jdArrs[0][1] === jdArrs[1][1]) {
        jdArrs.shift()
      }

      // 线数据
      var linesArrs = []
      // 标字
      jdArrs.map((res, index) => {
        viewer.entities.add({
          name: 'uav-tmp-point',
          position: Cesium.Cartesian3.fromDegrees(res[0], res[1]),
          label: {
            text: index + '',
            font: '14pt SongTi',
            eyeOffset: new Cesium.Cartesian3(0, 0, -100)
          }
        })
        // 存储线数据
        linesArrs.push(res[0])
        linesArrs.push(res[1])
      })
      // 画方向线
      viewer.entities.add({
        name: 'uav-tmp-line',
        // corridor polyline
        polyline: {
          positions: Cesium.Cartesian3.fromDegreesArray(linesArrs),
          material: new Cesium.PolylineArrowMaterialProperty(Cesium.Color.RED),
          followSurface: true,
          width: 10
        }
      })
    },
    // 弧度转换为经纬度
    rectangle2LonLat(coor) {
      const northwest = Cesium.Rectangle.northwest(coor)
      const southwest = Cesium.Rectangle.southwest(coor)
      const northeast = Cesium.Rectangle.northeast(coor)
      const southeast = Cesium.Rectangle.southeast(coor)

      const leftTop = [
        Cesium.Math.toDegrees(northwest.longitude),
        Cesium.Math.toDegrees(northwest.latitude)
      ]
      const leftBottom = [
        Cesium.Math.toDegrees(southwest.longitude),
        Cesium.Math.toDegrees(southwest.latitude)
      ]

      const rightTop = [
        Cesium.Math.toDegrees(northeast.longitude),
        Cesium.Math.toDegrees(northeast.latitude)
      ]

      const rightBottom = [
        Cesium.Math.toDegrees(southeast.longitude),
        Cesium.Math.toDegrees(southeast.latitude)
      ]

      return [leftTop, rightTop, leftBottom, rightBottom]
    },
    // 清理entity
    cleanEntity() {
      var viewer = window.viewer
      this.cesium.removeEntityLikeName(viewer, 'uav-')
      this.cesium.cleanEntityCollection(viewer, 'uva')
      polyArr = []

      viewer.trackedEntity = undefined
      this.isFly = false
    },
    // 更改航飞间距触发事件
    distanceHandleChange(val) {
      this.beginCalc()
      // 重置飞行
      if (this.isFly) {
        this.moniFly()
      }
    },
    // 模拟飞行
    moniFly() {
      var viewer = window.viewer
      var that = this
      that.isFly = true
      if (polyArr.length === 0) {
        this.$message.warning('未检测到航飞区域')
        return
      }
      // 清除上一个动画
      this.cesium.cleanEntityCollection(viewer, 'uva')
      this.cesium.removeEntityLikeName(viewer, 'uav-tmp-fly')
      viewer.trackedEntity = undefined
      // 加载新动画
      const czml = [
        {
          id: 'document',
          name: 'uva',
          version: '1.0',
          clock: {
            interval: '2022-08-04T10:00:00Z/2022-08-04T15:00:00Z',
            currentTime: '2022-08-04T10:00:00Z',
            range: 'LOOP_STOP',
            multiplier: 10
          }
        },
        {
          id: 'path',
          name: 'uva-tmp-fly',
          description: '<p> 飞行器</p>',
          availability: '2022-08-04T10:00:00Z/2022-08-04T15:00:00Z',
          path: {
            material: {
              polylineOutline: {
                color: {
                  rgba: [255, 215, 0, 255]
                },
                outlineColor: {
                  rgba: [192, 192, 192, 255]
                },
                outlineWidth: 5
              }
            },
            width: 8,
            leadTime: 10,
            // trailTime: 1000,
            resolution: 5
          },
          billboard: {
            image:
'',
            scale: 1.5,
            eyeOffset: {
              cartesian: [0.0, 0.0, -10.0]
            }
          },
          position: {
            epoch: '2022-08-04T10:00:00Z',
            cartographicDegrees: []
          }
        }
      ]
      var tmp = []
      var timesArr = []
      var timeTmp = 0
      var height = 500 // 飞行高度
      var v = 20 // 飞行速度
      var yc = 2 // 重复飞行延迟时间 秒
      // 手动插值
      timesArr.push(0)
      tmp.push(0)
      tmp.push(jdArrs[0][0])
      tmp.push(jdArrs[0][1])
      tmp.push(height + Math.random() * 5 + 5)
      for (var i = 0; i < jdArrs.length; i++) {
        var times = 0
        if (i < jdArrs.length - 1) {
          var from = turf.point(jdArrs[i])
          var to = turf.point(jdArrs[i + 1])
          var options = { units: 'kilometers' }
          var distance = turf.distance(from, to, options)
          times = Math.round((distance * 1000) / v)
          timeTmp += times
          timesArr.push(timeTmp)
          tmp.push(timeTmp)
          tmp.push(jdArrs[i + 1][0])
          tmp.push(jdArrs[i + 1][1])
          tmp.push(height + Math.random() * 5 + 5)
        }
      }
      // 动态配置CZML
      // 动画结束时间
      var tmpsss = new Date(
        
          new Date(czml[0].clock.currentTime).getTime() +
            (timesArr[timesArr.length - 1] + yc) * 1000
     
      ).toISOString()

      var str = czml[0].clock.currentTime + '/' + tmpsss
      czml[0].clock.interval = str
      czml[1].availability = str
      czml[1].path.trailTime = timesArr[2]
      czml[1].position.cartographicDegrees = tmp

      // 加载CZML
      var dataSource = viewer.dataSources.add(Cesium.CzmlDataSource.load(czml))
      // 加载同步扫描椎体
      dataSource
        .then(function(dataSource) {
          var entity = dataSource.entities.getById('path')
          entity.viewFrom = new Cesium.Cartesian3(0.0, -1000.0, 1500.0)
          viewer.trackedEntity = entity
          var cylinderEntitys = that.addFrustum({
            length: 510.0,
            topRadius: 0.0,
            bottomRadius: that.hfDistance / 2,
            color: Cesium.Color.GREEN.withAlpha(0.5)
          })
          var property = new Cesium.SampledPositionProperty()
          for (var ind = 0; ind < timesArr.length; ind++) {
            var time = Cesium.JulianDate.addSeconds(
              viewer.clock.currentTime,
              timesArr[ind],
              new Cesium.JulianDate()
            )
            var position = entity.position.getValue(time)
            if (position) {
              var cartographic = Cesium.Ellipsoid.WGS84.cartesianToCartographic(
                position
              )
              var lat = Cesium.Math.toDegrees(cartographic.latitude)
              var lng = Cesium.Math.toDegrees(cartographic.longitude)
              var hei = cartographic.height / 1.9
              property.addSample(
                time,
                Cesium.Cartesian3.fromDegrees(lng, lat, hei)
              )
            }
          }
          cylinderEntitys.position = property
        })
        .catch(function(error) {
          window.alert(error)
        })
    },
    // 创建视锥体
    addFrustum(option) {
      var viewer = window.viewer
      return viewer.entities.add({
        name: 'uav-tmp-fly-wxsimple',
        position: Cesium.Cartesian3.fromDegrees(114.0, 36.0, 200000.0),
        cylinder: {
          slices: option.slices,
          length: option.length,
          topRadius: option.topRadius,
          bottomRadius: option.bottomRadius,
          material: option.color
        }
      })
    }
  }
}
</script>

<style lang="scss" scoped>
.show {
  display: block;
}
.uvaRoutePlanBox {
  width: 100vw;
  height: 100vh;
}
/* 可视域 */
#cesiumContainer {
  width: 100%;
  height: 100%;
}
.body {
  position: fixed;
  right: calc(100% - 300px);
  top: 166px;
  z-index: 999;
  width: 300px;
  text-align: center;
  background: rgba(0, 0, 0, 0.6);
  padding: 15px;
  color: #fff;
}
.form-item {
  margin: 5px;
  .form-label {
    display: inline-block;
    width: 100px;
    text-align: center;
    font-size: 16px;
    font-weight: bold;
  }
  .form-connect {
    display: inline-block;
    // width: calc(100% - 120px);
  }
}

.hide {
  display: none;
}
</style>

使用注意

1、cesium引用切换为自己的引用。

2、this.cesium.removeEntityLikeName 和 this.cesium.cleanEntityCollection报错。

/**
 * 根据名称删除DataSources
 * @param {*} name
 */
export function cleanEntityCollection(viewer, key) {
  var tmp = viewer.dataSources.getByName(key)
  if (tmp.length > 0) {
    tmp.map(res => {
      viewer.dataSources.remove(res)
    })
  }
}
/**
 * 根据name获取entity
 *
 * */
export function getEntityLikeName(viewer, name) {
  if (name) {
    var entities = viewer.entities.values
    var findEntities = []
    for (var i = 0; i < entities.length; i++) {
      var entity = entities[i]
      // console.log(entity);
      if (entity.name && entity.name.indexOf(name) !== -1) {
        findEntities.push(entity)
      }
    }
    return findEntities
  } else {
    return []
  }
}

/**
 * 根据类似name移除entity
 *
 * */
export function removeEntityLikeName(viewer, name) {
  if (name) {
    var cleanEntities = getEntityLikeName(viewer, name)
    // 清除
    cleanEntities.map(res => {
      viewer.entities.removeById(res.id)
    })
  } else {
    viewer.entities.removeAll()
  }
}

3、this.$message 报错可将 this.$message.warning删除或换成alert弹窗或其他提示方法。

追加说明

        由于很多人咨询配置环境问题,现配置一份简单的vue的框架。cesium引入采用cdn,npm i 和npm run dev 即可运行。

源码地址:

uav: Cesium 简单航飞线路规划,模拟飞行扫描效果 - Gitee.com

参考文档

数据计算参考:

GET START | Turf.js中文网 (fenxianglu.cn)

飞行效果参考:

CZML Path - Cesium Sandcastle

相关方法参考:

Viewer - Cesium Documentation

  • 7
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 13
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 13
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Zk9509

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值