Map.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542
  1. <template>
  2. <div id="skysceneryContainer">
  3. <Popup />
  4. </div>
  5. </template>
  6. <script>
  7. import Popup from "@/components/Popup.vue";
  8. export default {
  9. name: "Map",
  10. components: {
  11. Popup,
  12. },
  13. data() {
  14. return {
  15. // viewer3DTiles实例《systemConfig.data3DTiles keys,tile》
  16. viewer3DTiles: {},
  17. // 3DTiles options集合《systemConfig.data3DTiles keys,options》
  18. keysByOptions: {},
  19. };
  20. },
  21. mounted() {
  22. window.SkySceneryConfig = {};
  23. for (let tileName in systemConfig.data3DTiles) {
  24. for (let key in systemConfig.data3DTiles[tileName]) {
  25. let options = systemConfig.data3DTiles[tileName][key];
  26. this.keysByOptions[tileName + key] = options;
  27. }
  28. }
  29. this.getToken();
  30. },
  31. methods: {
  32. getToken() {
  33. let that = this;
  34. let loginInfo = new FormData();
  35. for (const key in userLoginConfig) {
  36. if (Object.prototype.hasOwnProperty.call(userLoginConfig, key)) {
  37. loginInfo.append(key, userLoginConfig[key]);
  38. }
  39. }
  40. fetch(oauthUrl + "/oauth/api/user/login", {
  41. method: "POST",
  42. body: loginInfo,
  43. })
  44. .then((resp) => resp.json())
  45. .then((data) => {
  46. SkySceneryConfig = {
  47. authUrl: oauthUrl,
  48. token: data.message,
  49. };
  50. that.pushAllScripts();
  51. });
  52. },
  53. pushAllScripts() {
  54. let that = this;
  55. that.addScripts(scriptObj.main).then(function () {
  56. let arr = scriptObj.plugins.map(function (src) {
  57. return that.addScripts(src);
  58. });
  59. Promise.all(arr).then(function () {
  60. setTimeout(() => {
  61. if (SkyScenery) {
  62. that.creatMap();
  63. } else {
  64. that.pushAllScripts();
  65. }
  66. });
  67. });
  68. });
  69. },
  70. addScripts(src) {
  71. return new Promise((resolve, reject) => {
  72. // 创建一个新的script标签
  73. var script = document.createElement("script");
  74. // 设置script标签的src属性为要引入的JavaScript文件的URL
  75. script.src = src;
  76. // 将script标签添加到页面的head部分或者其他合适的位置
  77. document.head.appendChild(script);
  78. if (script.readyState) {
  79. // IE
  80. script.onreadystatechange = function () {
  81. if (script.readyState === "loaded" || script.readyState === "complete") {
  82. script.onreadystatechange = null;
  83. resolve();
  84. }
  85. };
  86. } else {
  87. // 其他浏览器
  88. script.onload = function () {
  89. resolve();
  90. };
  91. }
  92. });
  93. },
  94. creatMap() {
  95. try {
  96. let that = this;
  97. window.viewer = new SkyScenery.Viewer("skysceneryContainer", {
  98. animation: false, //是否创建动画小器件,左下角仪表
  99. baseLayerPicker: false, //是否显示图层选择器
  100. imageryProvider: new SkyScenery.SingleTileImageryProvider({
  101. url: (function createColorCanvas(color) {
  102. var width = 1,
  103. height = 1;
  104. var canvas = document.createElement("canvas");
  105. canvas.width = width;
  106. canvas.height = height;
  107. var ctx = canvas.getContext("2d");
  108. ctx.fillStyle = color;
  109. ctx.fillRect(0, 0, width, height);
  110. return canvas.toDataURL();
  111. })("#ffffff00"),
  112. tileWidth: document.createElement("canvas").width,
  113. tileHeight: document.createElement("canvas").height,
  114. rectangle: SkyScenery.Rectangle.fromDegrees(-180.0, -90.0, 180.0, 90.0),
  115. }),
  116. fullscreenButton: false, //是否显示全屏按钮
  117. geocoder: false, //是否显示geocoder小器件,右上角查询按钮
  118. homeButton: false, //是否显示Home按钮
  119. infoBox: false, //是否显示信息框
  120. sceneModePicker: false, //是否显示3D/2D选择器
  121. selectionIndicator: false, //是否显示选取指示器组件
  122. timeline: false, //是否显示时间轴
  123. navigationHelpButton: false, //是否显示右上角的帮助按钮
  124. scene3DOnly: true, //如果设置为true,则所有几何图形以3D模式绘制以节约GPU资源
  125. infoBox: false, //是否显示点击要素之后显示的信息
  126. shouldAnimate: false, //是否自动播放
  127. // contextOptions: {
  128. // requestWebgl1: true
  129. // }
  130. });
  131. setTimeout(() => {
  132. // 根据配置文件中的配置信息加载底图
  133. systemConfig.imageryProviders.forEach((url) => {
  134. that.loadTheWorldMapImage(url);
  135. });
  136. // 设置地表透明度为 0.5(值在 0 到 1 之间,0 为完全透明,1 为不透明)
  137. // viewer.scene.globe.translucency.enabled = true;
  138. // viewer.scene.globe.baseColor = new SkyScenery.Color(1.0, 1.0, 1.0, 0.1);
  139. viewer.camera.setView({
  140. destination: SkyScenery.Cartesian3.fromDegrees(
  141. 121.28277083019914,
  142. 31.065009352291785,
  143. 12790.087231596899
  144. ), // 设置位置
  145. orientation: {
  146. heading: SkyScenery.Math.toRadians(356.03124445628373), // 方向
  147. pitch: SkyScenery.Math.toRadians(-45.31423437919367), // 倾斜角度
  148. roll: SkyScenery.Math.toRadians(0.0047802614997811636)
  149. }
  150. });
  151. this.$store.commit("createdMap", true);
  152. // 绑定点击事件
  153. let handler = new SkyScenery.ScreenSpaceEventHandler(viewer.canvas);
  154. handler.setInputAction(function (movement) {
  155. var pick = viewer.scene.pick(movement.position); // 拾取鼠标所在的entity
  156. if (SkyScenery.defined(pick)) {
  157. let entity = pick.id;
  158. if (!entity) return;
  159. if (entity.type && (entity.type == "layers" || entity.type == "edge")) {
  160. let cartesian = viewer.camera.pickEllipsoid(
  161. movement.position,
  162. viewer.scene.globe.ellipsoid
  163. );
  164. // 空间坐标转世界坐标(弧度)
  165. if (!cartesian) return;
  166. let cartographic = SkyScenery.Cartographic.fromCartesian(cartesian);
  167. if (!cartographic) return;
  168. // 弧度转为角度(经纬度)
  169. let lon = SkyScenery.Math.toDegrees(cartographic.longitude); // 经度值
  170. let lat = SkyScenery.Math.toDegrees(cartographic.latitude); // 纬度值
  171. let showEntity = SkyScenery.Cartesian3.fromDegrees(lon, lat);
  172. showEntity.data = entity.properties.getValue();
  173. that.$store.state.showEntity = showEntity;
  174. } else {
  175. }
  176. } else {
  177. that.$store.state.showEntity = null;
  178. }
  179. }, SkyScenery.ScreenSpaceEventType.LEFT_CLICK);
  180. return;
  181. }, 0);
  182. } catch (error) {
  183. console.log("createMap error:", error);
  184. window.location.reload();
  185. }
  186. },
  187. /**
  188. * 加载天地图底图
  189. * @author LiuMengxiang
  190. * 加载影像底图和影像注记
  191. */
  192. async loadTheWorldMapImage(url) {
  193. // if (url.indexOf("//t0.") != -1) {
  194. // url = url.replace("//t0.", "//t{s}.");
  195. // }
  196. // viewer.imageryLayers.addImageryProvider(
  197. // new SkyScenery.WebMapTileServiceImageryProvider({
  198. // url:
  199. // url +
  200. // "?service=wmts&request=GetTile&version=1.0.0&LAYER=img&tileMatrixSet=w&TileMatrix={TileMatrix}&TileRow={TileRow}&TileCol={TileCol}&style=default&format=tiles&tk=" +
  201. // systemConfig.tdt_tk,
  202. // layer: "img",
  203. // style: "default",
  204. // tileMatrixSetID: "w",
  205. // format: "image/jpeg",
  206. // subdomains: ["0", "1", "2", "3", "4", "5", "6", "7"],
  207. // show: true,
  208. // minimumLevel: 1,
  209. // maximumLevel: 18
  210. // })
  211. // );
  212. // let layer = await SkyScenery.ArcGisMapServerImageryProvider.fromUrl(
  213. // "https://services.arcgisonline.com/arcgis/rest/services/World_Imagery/MapServer"
  214. // );
  215. let layer = new SkyScenery.ArcGisMapServerImageryProvider({
  216. url:
  217. "https://services.arcgisonline.com/arcgis/rest/services/World_Imagery/MapServer",
  218. });
  219. viewer.imageryLayers.addImageryProvider(layer);
  220. },
  221. // 加载geojson数据
  222. addGeoJson(url, options) {
  223. // options = {
  224. // point: {
  225. // imgUrl: ""
  226. // },
  227. // polyline: {
  228. // color: "#ffffff",
  229. // width: 3,
  230. // alpha: 0.7
  231. // },
  232. // polygon: {
  233. // outerColor: "#ffffff",
  234. // outerWidth: 3,
  235. // innerColor: "#ffffff",
  236. // alpha: 0.7
  237. // }
  238. // };
  239. SkyScenery.GeoJsonDataSource.load(url).then(function (dataSource) {
  240. viewer.dataSources.add(dataSource);
  241. var entities = dataSource.entities.values;
  242. for (var i = 0; i < entities.length; i++) {
  243. var entity = entities[i];
  244. if (entity.billboard) {
  245. entity.billboard = undefined;
  246. entity.billboard = new SkyScenery.BillboardGraphics({
  247. image: options.point.imgUrl,
  248. width: 50,
  249. height: 50,
  250. pixelOffset: new SkyScenery.Cartesian2(0, -25),
  251. heightReference: SkyScenery.HeightReference.CLAMP_TO_GROUND,
  252. });
  253. }
  254. if (entity.polyline) {
  255. entity.polyline.width = options.polyline.width;
  256. entity.polyline.material = SkyScenery.Color.fromCssColorString(
  257. options.polyline.color
  258. ).withAlpha(options.polyline.alpha); // 颜色
  259. }
  260. if (entity.polygon) {
  261. entity.polygon.height = 0.2;
  262. entity.polygon.outline = true; // 边框是否显示
  263. entity.polygon.outlineColor = SkyScenery.Color.fromCssColorString(
  264. options.polygon.outerColor
  265. ); // 边框颜色
  266. entity.polygon.outlineWidth = options.polygon.outerWidth; // 边框宽度
  267. entity.polygon.material = SkyScenery.Color.fromCssColorString(
  268. options.polygon.innerColor
  269. ).withAlpha(options.polygon.alpha); // 填充色
  270. }
  271. }
  272. });
  273. },
  274. /**
  275. * 加载动态围墙
  276. * options = {
  277. * maxH: 100,
  278. * color: "#00c4ff80",
  279. * duration: 3000,
  280. * }
  281. */
  282. addDynamicWall(positions, options) {
  283. console.log("addDynamicWall", positions, options);
  284. return viewer.entities.add({
  285. name: "立体墙效果",
  286. wall: {
  287. positions: positions.map(function (item) {
  288. return SkyScenery.Cartesian3.fromDegrees(item[0], item[1]);
  289. }),
  290. // 设置高度
  291. maximumHeights: new Array(positions.length).fill(options.maxH || 100),
  292. minimumHeights: new Array(positions.length).fill(0),
  293. material: new SkyScenery.DynamicWallMaterialProperty({
  294. color: SkyScenery.Color.fromCssColorString(options.color || "#00c4ff80"), // "#"
  295. trailImage: "/static/image/color.png",
  296. duration: options.duration || 3000, // 3000
  297. }),
  298. },
  299. });
  300. },
  301. /**
  302. * 加载3Dtiles
  303. * @author LiuMengxiang
  304. * @param {*} url 3Dtiles的systemConfig.data3DTiles keys
  305. * @param {*} flyto 是否自动飞行到3Dtiles
  306. */
  307. async add3DTiles(keys, flyto) {
  308. let that = this;
  309. if (this.viewer3DTiles[keys]) {
  310. this.viewer3DTiles[keys].show = true;
  311. if (flyto) {
  312. viewer.zoomTo(this.viewer3DTiles[keys]);
  313. }
  314. } else {
  315. let option = {
  316. url: this.keysByOptions[keys].url + "/tileset.json",
  317. skipLevelOfDetail: true, //开启跳级加载
  318. maximumMemoryUsage: 1024, //最大内存占用 推荐显存的一般
  319. preferLeaves: true,
  320. maximumScreenSpaceError: 16,
  321. maximumNumberOfLoadedTiles: 2000,
  322. };
  323. // Common settings for the dynamicScreenSpaceError optimization
  324. // const tile = await SkyScenery.Cesium3DTileset.fromUrl(
  325. // this.keysByOptions[keys].url + "/tileset.json",
  326. // option
  327. // );
  328. var tile = new SkyScenery.Cesium3DTileset(option);
  329. // 可以通过viewer.scene来访问场景对象进行进一步设置
  330. viewer.scene.postProcessStages.fxaa.enabled = true; // 启用FXAA(快速近似抗锯齿)
  331. viewer.scene.primitives.add(tile);
  332. // // 设置相机视角
  333. // if (flyto) {
  334. // viewer.zoomTo(tile);
  335. // }
  336. tile.readyPromise.then(function () {
  337. // 设置相机视角
  338. if (flyto) {
  339. viewer.zoomTo(tile);
  340. }
  341. });
  342. this.viewer3DTiles[keys] = tile;
  343. setTimeout(() => {
  344. this.update3DTilesStyleByKeys(
  345. this.viewer3DTiles[keys],
  346. Object.assign({}, this.keysByOptions[keys])
  347. );
  348. });
  349. }
  350. },
  351. /**
  352. * 修改3DTiles样式
  353. */
  354. update3DTilesStyleByKeys(tileset, options) {
  355. try {
  356. tileset.readyPromise.then(function () {
  357. // 设置样式
  358. var cartographic = SkyScenery.Cartographic.fromCartesian(
  359. tileset.boundingSphere.center
  360. );
  361. var lon = SkyScenery.Math.toDegrees(cartographic.longitude);
  362. var lat = SkyScenery.Math.toDegrees(cartographic.latitude);
  363. var height = cartographic.height;
  364. var surface = SkyScenery.Cartesian3.fromDegrees(lon, lat, height);
  365. var offset = SkyScenery.Cartesian3.fromDegrees(
  366. lon + options.lon,
  367. lat + options.lat,
  368. height + options.height
  369. );
  370. var translation = SkyScenery.Cartesian3.subtract(
  371. offset,
  372. surface,
  373. new SkyScenery.Cartesian3()
  374. );
  375. tileset.modelMatrix = SkyScenery.Matrix4.fromTranslation(translation);
  376. var m = tileset._root.transform;
  377. if (options.rx != 0) {
  378. const mx = SkyScenery.Matrix3.fromRotationX(
  379. SkyScenery.Math.toRadians(options.rx)
  380. );
  381. const rotate = SkyScenery.Matrix4.fromRotationTranslation(mx);
  382. SkyScenery.Matrix4.multiply(m, rotate, m);
  383. }
  384. if (options.ry != 0) {
  385. const my = SkyScenery.Matrix3.fromRotationY(
  386. SkyScenery.Math.toRadians(options.ry)
  387. );
  388. const rotate = SkyScenery.Matrix4.fromRotationTranslation(my);
  389. SkyScenery.Matrix4.multiply(m, rotate, m);
  390. }
  391. if (options.rz != 0) {
  392. const mz = SkyScenery.Matrix3.fromRotationZ(
  393. SkyScenery.Math.toRadians(options.rz)
  394. );
  395. const rotate = SkyScenery.Matrix4.fromRotationTranslation(mz);
  396. SkyScenery.Matrix4.multiply(m, rotate, m);
  397. }
  398. if (options.scale != 1) {
  399. const _scale = SkyScenery.Matrix4.fromUniformScale(options.scale);
  400. SkyScenery.Matrix4.multiply(m, _scale, m);
  401. }
  402. tileset._root.transform = m;
  403. });
  404. } catch (error) {
  405. console.error(error);
  406. }
  407. },
  408. /**
  409. * 修改3DTiles样式(测试)
  410. */
  411. update3DTilesStyle(keys, options) {
  412. if (this.viewer3DTiles[keys]) {
  413. let tileset = this.viewer3DTiles[keys];
  414. this.update3DTilesStyleByKeys(tileset, options);
  415. } else {
  416. console.log("update3DTilesStyle未找到3DTiles模型实例!keys:", keys);
  417. }
  418. },
  419. adjust3DTilesPosition(tileset, options) {
  420. var cartographic = Cesium.Cartographic.fromCartesian(tileset.boundingSphere.center);
  421. var lon = Cesium.Math.toDegrees(cartographic.longitude);
  422. var lat = Cesium.Math.toDegrees(cartographic.latitude);
  423. var height = cartographic.height;
  424. var surface = Cesium.Cartesian3.fromDegrees(lon, lat, height);
  425. var offset = Cesium.Cartesian3.fromDegrees(
  426. lon + options.lon,
  427. lat + options.lat,
  428. height + options.height
  429. );
  430. var translation = Cesium.Cartesian3.subtract(
  431. offset,
  432. surface,
  433. new Cesium.Cartesian3()
  434. );
  435. tileset.modelMatrix = Cesium.Matrix4.fromTranslation(translation);
  436. // var m = Cesium.Transforms.eastNorthUpToFixedFrame(surface);
  437. var m = tileset._root.transform;
  438. if (options.rx != 0) {
  439. const mx = Cesium.Matrix3.fromRotationX(Cesium.Math.toRadians(options.rx));
  440. const rotate = Cesium.Matrix4.fromRotationTranslation(mx);
  441. Cesium.Matrix4.multiply(m, rotate, m);
  442. }
  443. if (options.ry != 0) {
  444. const my = Cesium.Matrix3.fromRotationY(Cesium.Math.toRadians(options.ry));
  445. const rotate = Cesium.Matrix4.fromRotationTranslation(my);
  446. Cesium.Matrix4.multiply(m, rotate, m);
  447. }
  448. if (options.rz != 0) {
  449. const mz = Cesium.Matrix3.fromRotationZ(Cesium.Math.toRadians(options.rz));
  450. const rotate = Cesium.Matrix4.fromRotationTranslation(mz);
  451. Cesium.Matrix4.multiply(m, rotate, m);
  452. }
  453. if (options.scale != 1) {
  454. const _scale = Cesium.Matrix4.fromUniformScale(options.scale);
  455. Cesium.Matrix4.multiply(m, _scale, m);
  456. }
  457. tileset._root.transform = m;
  458. },
  459. /**
  460. * 删除3DTiles
  461. * @author LiuMengxiang
  462. * @param tiles 3DTiles实例
  463. */
  464. remove3Dtiles(keys) {
  465. if (this.viewer3DTiles[keys]) {
  466. viewer.scene.primitives.remove(this.viewer3DTiles[keys]);
  467. }
  468. },
  469. /**
  470. * 隐藏3DTiles(测试)
  471. * @author LiuMengxiang
  472. * @param tiles 3DTiles实例
  473. */
  474. hide3DTiles(keys) {
  475. if (this.viewer3DTiles[keys]) {
  476. this.viewer3DTiles[keys].show = false;
  477. }
  478. },
  479. /**
  480. * 根据首页下拉框值判断3DTiles加载状态
  481. * @author LiuMengxiang
  482. * @param cascaderValue 首页
  483. */
  484. HomeHandleChangeCascader(cascaderValue) {
  485. if (this.$store.state.initMap) {
  486. // 暂存首页下拉框选中值
  487. let homeVs = [];
  488. cascaderValue.forEach((item) => {
  489. homeVs.push(item[0] + item[1]);
  490. });
  491. // 遍历所有的3Dtiles
  492. for (let keys in this.keysByOptions) {
  493. // 如果选中的话
  494. if (homeVs.length > 0 && homeVs.indexOf(keys) != -1) {
  495. this.add3DTiles(keys, true);
  496. } else {
  497. this.hide3DTiles(keys);
  498. }
  499. }
  500. } else {
  501. setTimeout(() => {
  502. this.HomeHandleChangeCascader(cascaderValue);
  503. }, 1000);
  504. }
  505. },
  506. },
  507. };
  508. </script>
  509. <style lang="less" scoped>
  510. #skysceneryContainer {
  511. width: 100%;
  512. height: 100%;
  513. }
  514. </style>