cesium vue 气泡窗

490 阅读1分钟

效果:

源码

Popup 代码

<!-- 悬浮在模型头顶的漂浮框 -->
<template>
  <div ref="popupTitle" class="earth-popup-imgbg-blue-simple" style="position:absolute;">
    <div class="title">{{ $title }}</div>
  </div>
</template>

<script>
import Vue from 'vue'
import { getUId } from '../../utils/uuid'

export default {

  props: {
    $title: {
      type: String,
      default: ''
    },
    $position: {}
  },

  mounted() {
    this.init()
  },

  data() {
    return {
      $popup: undefined,
      $cartesian: undefined,
      $lastScreenPosition: undefined
    }
  },

  methods: {
    init() {
      this.$popup = this.$refs.popupTitle
      this.$cartesian = Cesium.Cartographic.toCartesian(this.$position)
      this.$wgs84 = Cesium.Cartesian3.fromRadians(this.$position.longitude, this.$position.latitude, this.$position.height)
      this.$popup.id = getUId()
      viewer.scene.postRender.addEventListener(() => {
        this.postRenderPopup(this.$position)
      })

    },
    // 视角变化时,修改popup样式
    postRenderPopup(position) {
      if (!viewer || !position) {
        return
      }


      const cameraPosition = viewer.camera.position

      //距离500时,隐藏标签
      if ((Cesium.Cartesian3.distance(cameraPosition, this.$cartesian) > 5000)) {
        // console.log('hidden')
        if (this.$popup.style.visibility !== 'hidden') {
          this.$popup.style.visibility = 'hidden'
        }
      } else {
        let windowPosition = Cesium.SceneTransforms.wgs84ToWindowCoordinates(viewer.scene, this.$wgs84)

        if (windowPosition && this.$popup) {

          if (this.$lastScreenPosition) {
            if (this.$lastScreenPosition.x === windowPosition.x && this.$lastScreenPosition.y === windowPosition.y) {
              return
            }
          }

          this.$lastScreenPosition = windowPosition
          const clientHeight = this.$popup.clientHeight

          this.$popup.style.display = 'block'
          this.$popup.style.left = windowPosition.x + 'px'
          this.$popup.style.top = windowPosition.y - clientHeight + 'px'

          if (this.$popup.style.visibility !== 'visible') {
            this.$popup.style.visibility = 'visible'
          }


        }


      }
    },
    // 改造只留一个事件监听
    postRenderPopups(positions = []) {

    },

    close() {
      this.$emit('update:visible', false)
    }
  }
}

</script>

<style lang="scss">
.earth-popup-imgbg-blue-simple {
  position: absolute;
  background-size: 100%;
  background-image: url();
  background-repeat: no-repeat;
  width: 200px;
  height: 157px;
  z-index: 100;
  pointer-events: none;

  .title {
    pointer-events: none;
    position: inherit;
    top: 20px;
    left: 70px;
    font-size: 14px;
    text-align: left;
    color: rgba(255, 255, 255, 1);
    text-shadow: 0px 0px 9px #000000bf;
  }
}
</style>

CreatePopup

import Vue from 'vue'

export function createPopup(cmp, title, position) {
  const CmpConstructor = Vue.extend(cmp)
  const instance = new CmpConstructor({
    propsData: {
      $title: title,
      $position: position
    }
  })
  instance.$mount()
  
  return instance.$el
  
}

使用

<template>
  <div id="app"></div>
  <div id="cesiumContainer"></div>
</template>

<script>
 import popupTitle from './tools/popup-title/index.vue'
import { createPopup } from './tools/popup-title/createPopup'
  
  export default{
    ......
    methods:{
      popup(Cartographic,name){
          const popup = createPopup(popupTitle, name, Cartographic)
          document.getElementById('app').appendChild(popup)
      }
    }
  }
</script>

踩坑

之前因为没加这句判断代码,当页面超过两个气泡窗时候,会出现卡死的情况,也没有报错信息,看起来像是postRender事件监听的问题。

        let windowPosition = Cesium.SceneTransforms.wgs84ToWindowCoordinates(viewer.scene, this.$wgs84)

        if (windowPosition && this.$popup) {
        }

加上判断屏幕坐标是否存在的这句代码后,问题解决。查看cesium源码时,发现这样一句话:

所以问题就是因为屏幕坐标有时没有获取到,导致的屏幕卡死问题。