首页 > 其他分享 >Cesium结合GIS天地图 加载倾斜摄影3dtile + vue3

Cesium结合GIS天地图 加载倾斜摄影3dtile + vue3

时间:2023-05-23 15:44:34浏览次数:63  
标签:false 3dtile viewer vue3 Cesium GIS 加载 true const

实现思路

将倾斜摄影OSGB数据转换为3dtile(转换方式很多,可以利用第三方工具cesiumlab)
利用Cesium加载GIS地图,我这里使用的是天地图,可以加载其他地图都行
加载3dtile数据到地图中展示

安装插件

npm install cesium

加载地图

以加载天地图为例,需要先到天地图官网去申请开发者,获取一个token,以下代码直接粘贴就行,注释中已说明

//初始化地图
let viewer;

function initMap() {
  const tianDiTuToken = "天地图token";
  const mapOption = {
    url: `http://t0.tianditu.com/img_w/wmts?service=wmts&request=GetTile&version=1.0.0&LAYER=img&tileMatrixSet=w&TileMatrix={TileMatrix}&TileRow={TileRow}&TileCol={TileCol}&style=default&format=tiles&tk=${tianDiTuToken}`,
    layer: "tdtBasicLayer",
    style: "default",
    format: "image/jpeg",
    tileMatrixSetID: "GoogleMapsCompatible",
    maximumLevel: 18,
  };
  const imgProvider = new WebMapTileServiceImageryProvider(mapOption);

  const viewerOption = {
    animation: false, //是否创建动画小器件,左下角仪表
    baseLayerPicker: false, //是否显示图层选择器
    fullscreenButton: false, //是否显示全屏按钮
    geocoder: false, //是否显示geocoder小器件,右上角查询按钮
    homeButton: false, //是否显示Home按钮
    infoBox: false, //是否显示信息框
    sceneModePicker: false, //是否显示3D/2D选择器
    scene3DOnly: false, //如果设置为true,则所有几何图形以3D模式绘制以节约GPU资源
    selectionIndicator: false, //是否显示选取指示器组件
    timeline: false, //是否显示时间轴
    navigationHelpButton: false, //是否显示右上角的帮助按钮
    baselLayerPicker: false, // 将图层选择的控件关掉,才能添加其他影像数据
    shadows: true, //是否显示背影
    shouldAnimate: true,
    imageryProvider: imgProvider,
  };

  viewer = new Viewer("map", viewerOption);
  // 添加中文注记图层
  viewer.imageryLayers.addImageryProvider(
    new WebMapTileServiceImageryProvider({
      url: `http://t0.tianditu.com/cia_w/wmts?service=wmts&request=GetTile&version=1.0.0&LAYER=cia&tileMatrixSet=w&TileMatrix={TileMatrix}&TileRow={TileRow}&TileCol={TileCol}&style=default.jpg&tk=${tianDiTuToken}`,
      layer: "tdtAnnoLayer",
      style: "default",
      format: "image/jpeg",
      tileMatrixSetID: "GoogleMapsCompatible",
      show: false,
    })
  );
  //优化项--关闭相关特效
  viewer.scene.debugShowFramesPerSecond = true; //显示fps
  viewer.scene.moon.show = false; //月亮
  viewer.scene.fog.enabled = false; //雾
  viewer.scene.sun.show = false; //太阳
  viewer.scene.skyBox.show = false; //天空盒
  viewer.resolutionScale = 1.0; //画面细度,默认值为1.0
  //去除版权信息
  viewer._cesiumWidget._creditContainer.style.display = "none";
}

加载倾斜摄影3dtitle

3Dtitle文件我这里是以nginx代理的方式访问的,也可以直接放到项目文件夹中引用

//加载倾斜摄影图像
function init3Dtiles() {
  const tileSet = new Cesium3DTileset({
    url: "http://127.0.0.1/data/tileset.json",
    maximumMemoryUsage: 100, //不可设置太高,目标机子空闲内存值以内,防止浏览器过于卡
    maximumScreenSpaceError: 32, //用于驱动细节细化级别的最大屏幕空间错误;较高的值可提供更好的性能,但视觉质量较低。
    maximumNumberOfLoadedTiles: 1000, //最大加载瓦片个数
    shadows: false, //是否显示阴影
    skipLevelOfDetail: true, // 确定是否应在遍历期间应用详细级别跳过(默认false)
    baseScreenSpaceError: 1024, //When skipLevelOfDetailis true,在跳过详细级别之前必须达到的屏幕空间错误(默认1024)
    skipScreenSpaceErrorFactor: 16, // 定义要跳过的最小屏幕空间错误的乘数。与 一起使用skipLevels来确定要加载哪些图块(默认16)
    skipLevels: 1, //skipLevelOfDetail是true 一个常量,定义了加载图块时要跳过的最小级别数。为 0 时,不跳过任何级别。与 一起使用skipScreenSpaceErrorFactor来确定要加载哪些图块。(默认1)
    immediatelyLoadDesiredLevelOfDetail: false, //当skipLevelOfDetail是时true,只会下载满足最大屏幕空间错误的图块。忽略跳过因素,只加载所需的图块(默认false)
    loadSiblings: false, // 如果为true则不会在已加载完概况房屋后,自动从中心开始超清化房屋 --- 何时确定在遍历期间skipLevelOfDetail是否true始终下载可见瓦片的兄弟姐妹(默认false)
    cullWithChildrenBounds: true, //是否使用子边界体积的并集来剔除瓦片(默认true)
    dynamicScreenSpaceError: true, //减少距离相机较远的图块的屏幕空间错误(默认false)
    dynamicScreenSpaceErrorDensity: 0.00278, //数值加大,能让周边加载变快 --- 用于调整动态屏幕空间误差的密度,类似于雾密度(默认0.00278)
    dynamicScreenSpaceErrorFactor: 4.0, // 用于增加计算的动态屏幕空间误差的因素(默认4.0)
    dynamicScreenSpaceErrorHeightFalloff: 0.25, //密度开始下降的瓦片集高度的比率(默认0.25)
  });

  viewer.scene.primitives.add(tileSet);
  viewer.zoomTo(tileSet);
}

加载后发现倾斜图像与地图存在一定位置和高度的偏差,这里我们需要再调整一下图像的位置以及高度,调整代码如下

//更新倾斜摄影位置
function update3dtilesMaxtrix(tileSet) {
  //调整参数
  let params = {
    tx: 113.06265738392063, //模型中心X轴坐标(经度,单位:十进制度)
    ty: 22.646803971034342, //模型中心Y轴坐标(纬度,单位:十进制度)
    tz: 40, //模型中心Z轴坐标(高程,单位:米)
    rx: 0, //X轴(经度)方向旋转角度(单位:度)
    ry: 0, //Y轴(纬度)方向旋转角度(单位:度)
    rz: 2, //Z轴(高程)方向旋转角度(单位:度)
    scale: 1.3, //缩放比例
  };
  //旋转
  const mx = Cesium.Matrix3.fromRotationX(Cesium.Math.toRadians(params.rx));
  const my = Cesium.Matrix3.fromRotationY(Cesium.Math.toRadians(params.ry));
  const mz = Cesium.Matrix3.fromRotationZ(Cesium.Math.toRadians(params.rz));
  const rotationX = Cesium.Matrix4.fromRotationTranslation(mx);
  const rotationY = Cesium.Matrix4.fromRotationTranslation(my);
  const rotationZ = Cesium.Matrix4.fromRotationTranslation(mz);
  //平移
  const position = Cesium.Cartesian3.fromDegrees(
    params.tx,
    params.ty,
    params.tz
  );
  const m = Cesium.Transforms.eastNorthUpToFixedFrame(position);
  //旋转、平移矩阵相乘
  Cesium.Matrix4.multiply(m, rotationX, m);
  Cesium.Matrix4.multiply(m, rotationY, m);
  Cesium.Matrix4.multiply(m, rotationZ, m);
  //比例缩放
  const scale = Cesium.Matrix4.fromUniformScale(params.scale);
  Cesium.Matrix4.multiply(m, scale, m);
  console.log("矩阵m:", m);
  //赋值给tileset
  tileSet._root.transform = m;
}

在加载倾斜摄影成功后调用以上方法:

//加载倾斜摄影图像
function init3Dtiles() {
  const tileSet = new Cesium3DTileset({
    url: "http://127.0.0.1/data/tileset.json",
    //...代码省略,同上
  });

  viewer.scene.primitives.add(tileSet);
  viewer.zoomTo(tileSet);

  //调用方法
  //加载后调整倾斜摄影的位置、角度等参数
  tileSet.readyPromise.then((tileset) => {
    update3dtilesMaxtrix(tileset);
  });
}

点击查看完整代码
<template>
  <div
    id="map"
    style="
      z-index: 100;
      position: absolute;
      top: 0;
      bottom: 0;
      right: 0;
      left: 0;
    "
  ></div>
</template>

<script>
  import * as Cesium from "cesium";
  import {
    Cesium3DTileset,
    Viewer,
    WebMapTileServiceImageryProvider,
  } from "cesium";
  import { onMounted } from "vue";

  export default {
    name: "MapComponent",
    setup() {
      //初始化地图
      let viewer;

      function initMap() {
        const tianDiTuToken = "天地图token";
        const mapOption = {
          url: `http://t0.tianditu.com/img_w/wmts?service=wmts&request=GetTile&version=1.0.0&LAYER=img&tileMatrixSet=w&TileMatrix={TileMatrix}&TileRow={TileRow}&TileCol={TileCol}&style=default&format=tiles&tk=${tianDiTuToken}`,
          layer: "tdtBasicLayer",
          style: "default",
          format: "image/jpeg",
          tileMatrixSetID: "GoogleMapsCompatible",
          maximumLevel: 18,
        };
        const imgProvider = new WebMapTileServiceImageryProvider(mapOption);

        const viewerOption = {
          animation: false, //是否创建动画小器件,左下角仪表
          baseLayerPicker: false, //是否显示图层选择器
          fullscreenButton: false, //是否显示全屏按钮
          geocoder: false, //是否显示geocoder小器件,右上角查询按钮
          homeButton: false, //是否显示Home按钮
          infoBox: false, //是否显示信息框
          sceneModePicker: false, //是否显示3D/2D选择器
          scene3DOnly: false, //如果设置为true,则所有几何图形以3D模式绘制以节约GPU资源
          selectionIndicator: false, //是否显示选取指示器组件
          timeline: false, //是否显示时间轴
          navigationHelpButton: false, //是否显示右上角的帮助按钮
          baselLayerPicker: false, // 将图层选择的控件关掉,才能添加其他影像数据
          shadows: true, //是否显示背影
          shouldAnimate: true,
          imageryProvider: imgProvider,
        };

        viewer = new Viewer("map", viewerOption);
        // 添加中文注记图层
        viewer.imageryLayers.addImageryProvider(
          new WebMapTileServiceImageryProvider({
            url: `http://t0.tianditu.com/cia_w/wmts?service=wmts&request=GetTile&version=1.0.0&LAYER=cia&tileMatrixSet=w&TileMatrix={TileMatrix}&TileRow={TileRow}&TileCol={TileCol}&style=default.jpg&tk=${tianDiTuToken}`,
            layer: "tdtAnnoLayer",
            style: "default",
            format: "image/jpeg",
            tileMatrixSetID: "GoogleMapsCompatible",
            show: false,
          })
        );
        //优化项--关闭相关特效
        viewer.scene.debugShowFramesPerSecond = true; //显示fps
        viewer.scene.moon.show = false; //月亮
        viewer.scene.fog.enabled = false; //雾
        viewer.scene.sun.show = false; //太阳
        viewer.scene.skyBox.show = false; //天空盒
        viewer.resolutionScale = 1.0; //画面细度,默认值为1.0
        //去除版权信息
        viewer._cesiumWidget._creditContainer.style.display = "none";
      }

      //加载倾斜摄影图像
      function init3Dtiles() {
        const tileSet = new Cesium3DTileset({
          url: "http://127.0.0.1/data/tileset.json",
          maximumMemoryUsage: 100, //不可设置太高,目标机子空闲内存值以内,防止浏览器过于卡
          maximumScreenSpaceError: 32, //用于驱动细节细化级别的最大屏幕空间错误;较高的值可提供更好的性能,但视觉质量较低。
          maximumNumberOfLoadedTiles: 1000, //最大加载瓦片个数
          shadows: false, //是否显示阴影
          skipLevelOfDetail: true, // 确定是否应在遍历期间应用详细级别跳过(默认false)
          baseScreenSpaceError: 1024, //When skipLevelOfDetailis true,在跳过详细级别之前必须达到的屏幕空间错误(默认1024)
          skipScreenSpaceErrorFactor: 16, // 定义要跳过的最小屏幕空间错误的乘数。与 一起使用skipLevels来确定要加载哪些图块(默认16)
          skipLevels: 1, //skipLevelOfDetail是true 一个常量,定义了加载图块时要跳过的最小级别数。为 0 时,不跳过任何级别。与 一起使用skipScreenSpaceErrorFactor来确定要加载哪些图块。(默认1)
          immediatelyLoadDesiredLevelOfDetail: false, //当skipLevelOfDetail是时true,只会下载满足最大屏幕空间错误的图块。忽略跳过因素,只加载所需的图块(默认false)
          loadSiblings: false, // 如果为true则不会在已加载完概况房屋后,自动从中心开始超清化房屋 --- 何时确定在遍历期间skipLevelOfDetail是否true始终下载可见瓦片的兄弟姐妹(默认false)
          cullWithChildrenBounds: true, //是否使用子边界体积的并集来剔除瓦片(默认true)
          dynamicScreenSpaceError: true, //减少距离相机较远的图块的屏幕空间错误(默认false)
          dynamicScreenSpaceErrorDensity: 0.00278, //数值加大,能让周边加载变快 --- 用于调整动态屏幕空间误差的密度,类似于雾密度(默认0.00278)
          dynamicScreenSpaceErrorFactor: 4.0, // 用于增加计算的动态屏幕空间误差的因素(默认4.0)
          dynamicScreenSpaceErrorHeightFalloff: 0.25, //密度开始下降的瓦片集高度的比率(默认0.25)
        });

        viewer.scene.primitives.add(tileSet);
        viewer.zoomTo(tileSet);

        //加载后调整倾斜摄影的位置、角度等参数
        tileSet.readyPromise.then((tileset) => {
          update3dtilesMaxtrix(tileset);
        });
      }

      //更新倾斜摄影位置
      function update3dtilesMaxtrix(tileSet) {
        //调整参数
        let params = {
          tx: 113.06265738392063, //模型中心X轴坐标(经度,单位:十进制度)
          ty: 22.646803971034342, //模型中心Y轴坐标(纬度,单位:十进制度)
          tz: 40, //模型中心Z轴坐标(高程,单位:米)
          rx: 0, //X轴(经度)方向旋转角度(单位:度)
          ry: 0, //Y轴(纬度)方向旋转角度(单位:度)
          rz: 2, //Z轴(高程)方向旋转角度(单位:度)
          scale: 1.3, //缩放比例
        };
        //旋转
        const mx = Cesium.Matrix3.fromRotationX(
          Cesium.Math.toRadians(params.rx)
        );
        const my = Cesium.Matrix3.fromRotationY(
          Cesium.Math.toRadians(params.ry)
        );
        const mz = Cesium.Matrix3.fromRotationZ(
          Cesium.Math.toRadians(params.rz)
        );
        const rotationX = Cesium.Matrix4.fromRotationTranslation(mx);
        const rotationY = Cesium.Matrix4.fromRotationTranslation(my);
        const rotationZ = Cesium.Matrix4.fromRotationTranslation(mz);
        //平移
        const position = Cesium.Cartesian3.fromDegrees(
          params.tx,
          params.ty,
          params.tz
        );
        const m = Cesium.Transforms.eastNorthUpToFixedFrame(position);
        //旋转、平移矩阵相乘
        Cesium.Matrix4.multiply(m, rotationX, m);
        Cesium.Matrix4.multiply(m, rotationY, m);
        Cesium.Matrix4.multiply(m, rotationZ, m);
        //比例缩放
        const scale = Cesium.Matrix4.fromUniformScale(params.scale);
        Cesium.Matrix4.multiply(m, scale, m);
        console.log("矩阵m:", m);
        //赋值给tileset
        tileSet._root.transform = m;
      }

      onMounted(() => {
        initMap();
        init3Dtiles();
      });
    },
  };
</script>

<style scoped>
  @import url("../../node_modules/cesium/Source/Widgets/widgets.css");

  html,
  body,
  #map {
    height: 100%;
    width: 100%;
    margin: 0;
    padding: 0;
    overflow: hidden;
  }
</style>

转载自Cesium结合GIS天地图 加载倾斜摄影3dtile + vue3

标签:false,3dtile,viewer,vue3,Cesium,GIS,加载,true,const
From: https://www.cnblogs.com/echohye/p/17425434.html

相关文章

  • Vue2 到 Vue3 升级插件gogocode-plugin-vue
    配合gogocode-cli使用开始迁移​Vue3的到来为我们带来了许多惊喜的变化,但是由于Vue3对于Vue2在Api层面存在诸多兼容问题,并不能做到平滑升级。所以我们根据v3迁移指南利用gogocode这个代码转换利器,利用它操作AST,开发了一套Vue2升级工具。利用这套工具能够快速地把你的Vue2代码升......
  • vue3遇到的问题集合
    1.vue3打包设置 2.vue3路由配置history 3.vue3屏蔽eslint ......
  • vue3 左侧菜单栏默认展开关闭
    <el-menuclass="mainMenu"routerbackground-color="transparent"active-text-color="#fff":unique-opened="true":collapse-transition="false"......
  • Vue3+Vite部署到Netlify
    前言本项目的前端是使用Vue3技术。因为用到了Vite做代理解决跨域问题,所以部署到Netlify需要多做一些步骤。使用Vite代理后,如果前端是在本地部署,需要添加VITE_PROXY环境变量。例如在.env.development文件中添加VITE_PROXY=[["/api/v1","https://xx.naihe.me/api/v1"],["/ws",......
  • 如何对k8s中docker registry secret中保存的docker的用户名、密码进行解密?
    在k8s的dockerregistrysecret中,保存的是docker的用户名、密码的信息,并且是使用加密的方式进行保存的 [root@nccztsjb-node-11~]#kubectlgetsecret-nyonbipc87e2267-1001-4c70-bb2a-ab41f3b81aa3-oyamlapiVersion:v1data:.dockerconfigjson:eyJhdXRocyI6ey......
  • vue3 +leaflet + 天地图
    vue3使用leafletnpminstallleaflet-D如果使用了tsnpmi--save-dev@types/leaflet//使用了ts需要下载声明类型//更具需要获取不通过类型的urlfunctiongetUrl(type:string){interfaceMyObject{[key:string]:Array<string>;}letobj:MyObject=......
  • vue3+vant3封装省市区组件
    因为后台返回的数据格式和vant不一致,所以自己遍历循环得到自己想到的格式。省市区的接口并不是一个以children格式,一次性全部获取,而是选择市的时候需要得到省的id,一层一层请求接口才能得到省市区。 父组件:1<selectArea2@addressOnFinish="onChangeArea"3......
  • 数字化汇报方案:BIM、CAD和GIS的结合是关键!
    身为设计方的你有没有这样的经历:1.一个非常优秀的方案未能被甲方采纳,反而甲方选择了一个不如自己的方案,造成了很大的遗憾;2.在讲述自己的设计方案的时候,经常越说越散,甚至到了最后自己都不知道说了什么、没说什么...出师未捷身先死,方案都没说完就被人在心里给枪毙了!3.叨叨了半......
  • Vue3迎来升级,全面助力企业数字化转型
    近年来,随着“互联网+”的推进与应用普及,数字化转型已经成为企业发展的必备选项。低代码开发平台的迅速普及和广泛应用,也为企业数字化的转型提供了支撑技术。JNPF快速开发平台深度集成java+.net 6 双技术引擎,具备易维护、便部署、高集成、高效率等多方面特性,面向企业项目提供开......
  • 看模型、做技术交底、做项目汇报,这款BIM+CAD+GIS工具免费了
    现场汇报效果不好,导致丢了一个项目!项目汇报平淡无奇,方案屡次被毙!面对专家质疑,回答苍白无力!估计大家都有过这种经历和感受。详细分析一下,基本上有以下几个方面的原因:➤ 汇报的时候,找不到合适的软件做三维演示?Revit、SketchUp等建模软件不适合演示汇报,而且还缺少周边环境信息。......