arcgis.vue 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503
  1. <template>
  2. <div class="arcgis" :class="{'pointer': pointer}">
  3. <div id="viewDiv"></div>
  4. <teleport to=".esri-ui" v-if="teleport">
  5. <div id="marker">
  6. <div v-for="(marker, index) of markers" :key="index" v-show="marker.show"
  7. :id="'arcgic-marker-' + marker.slot" class="arcgis-marker" :style="'left: '+marker.x+'px; top: '+marker.y+'px; transform: scale('+marker.scale+')'">
  8. <slot :name="marker.slot"></slot>
  9. </div>
  10. </div>
  11. </teleport>
  12. </div>
  13. </template>
  14. <script>
  15. import Map from "@arcgis/core/Map";
  16. import Basemap from "@arcgis/core/Basemap";
  17. import Graphic from "@arcgis/core/Graphic";
  18. import SpatialReference from "@arcgis/core/geometry/SpatialReference";
  19. import MapView from "@arcgis/core/views/MapView";
  20. import SceneView from "@arcgis/core/views/SceneView";
  21. import MapImageLayer from "@arcgis/core/layers/MapImageLayer";
  22. import GraphicsLayer from "@arcgis/core/layers/GraphicsLayer";
  23. import SimpleMarkerSymbol from "@arcgis/core/symbols/SimpleMarkerSymbol";
  24. import "@arcgis/core/assets/esri/themes/light/main.css";
  25. const mapUrl = "http://10.155.32.4:9080/arcgis/rest/services/nxmap/MapServer"; // ArcMap地址
  26. // const mapUrl = "http://10.155.32.4:9080/arcgis/rest/services/NX_MAP_15/MapServer"; // ArcMap地址
  27. let basemap = null;
  28. let map = null;
  29. let view = null;
  30. let point_graphicsLayer = null;
  31. let line_graphicsLayer = null;
  32. let text_graphicsLayer = null;
  33. let img_graphicsLayer = null;
  34. let html_graphicsLayer = null;
  35. let point_graphics = [];
  36. let line_graphics = [];
  37. let text_graphics = [];
  38. let img_graphics = [];
  39. let html_graphics = [];
  40. export default {
  41. // 名称
  42. name: "arcgis",
  43. // 使用组件
  44. components: {},
  45. // 传入参数
  46. props: {
  47. data: {
  48. type: Object,
  49. default: () => {
  50. return {
  51. mode: "2D", // 模式 2D 3D
  52. title: "宁夏地图", // 标题
  53. center: [106.230909, 38.487193], // 初始中心点
  54. height: 654, // 3D地图初始相机高度
  55. tilt: 65, // 俯视角
  56. scale: 128000, // 缩放
  57. rotation: 0, // 正北顺时针旋转角度
  58. ox: 0, // 标记整体偏移
  59. oy: 0, // 标记整体偏移
  60. }
  61. },
  62. }
  63. },
  64. // 自定义事件
  65. emits: {
  66. when: null, // 地图加载完成
  67. clickMap: null, // 点击地图(可点击元素)
  68. },
  69. // 数据
  70. data() {
  71. return {
  72. fanStateImgMapping: [{
  73. name: "待机",
  74. code: "0",
  75. img: require("@assets/map/fan/green.png")
  76. },
  77. {
  78. name: "运行",
  79. code: "1",
  80. img: require("@assets/map/fan/blue.png")
  81. },
  82. {
  83. name: "故障",
  84. code: "2",
  85. img: require("@assets/map/fan/red.png")
  86. },
  87. {
  88. name: "离线",
  89. code: "3",
  90. img: require("@assets/map/fan/black.png")
  91. },
  92. {
  93. name: "检修",
  94. code: "4",
  95. img: require("@assets/map/fan/orange.png")
  96. },
  97. {
  98. name: "限电",
  99. code: "5",
  100. img: require("@assets/map/fan/purple.png")
  101. },
  102. {
  103. name: "限停",
  104. code: "6",
  105. img: require("@assets/map/fan/white.png")
  106. },
  107. ],
  108. syzImg: require("@assets/png/booster-station.png"),
  109. markers: [],
  110. pointer: false,
  111. teleport: false, // 是否进行传送
  112. }
  113. },
  114. // 函数
  115. methods: {
  116. // 世界转屏幕
  117. worldToScreen: function (point) {
  118. // var world = new Point(point.x, point.y, SpatialReference.WGS84);
  119. // var screenPoint = map.toScreen(point);
  120. },
  121. // 屏幕转世界
  122. screenToWorld: function (point) {},
  123. // 偏移转经纬度
  124. oToLL: function (x, y) {
  125. return {
  126. x: 0.00000899 * x,
  127. y: 0.00001141 * y
  128. }
  129. },
  130. // 初始化地图
  131. initMap: function() {
  132. basemap = new Basemap({
  133. // 底图
  134. baseLayers: [
  135. new MapImageLayer({
  136. url: mapUrl,
  137. title: this.data.title,
  138. spatialReference: SpatialReference.WGS84,
  139. }),
  140. ],
  141. title: this.data.title,
  142. spatialReference: SpatialReference.WGS84,
  143. });
  144. map = new Map({
  145. // 地图容器
  146. basemap: basemap,
  147. });
  148. if (this.data.mode == "3D") {
  149. view = new SceneView({
  150. // 3D显示图层
  151. map: map,
  152. center: this.data.center,
  153. camera: {
  154. position: {
  155. x: this.data.center[0], // lon
  156. y: this.data.center[1], // lat
  157. z: this.data.height, // elevation in meters
  158. },
  159. tilt: this.data.tilt,
  160. },
  161. container: this.$el.querySelector("#viewDiv"),
  162. });
  163. } else {
  164. view = new MapView({
  165. // 2D显示图层
  166. map: map,
  167. center: this.data.center,
  168. container: this.$el.querySelector("#viewDiv"),
  169. rotation: this.data.rotation,
  170. });
  171. }
  172. view.scale = this.data.scale;
  173. //去掉版权
  174. view.ui._removeComponents(["attribution"]);
  175. view.when(() => {
  176. this.when();
  177. this.$emit('when');
  178. });
  179. },
  180. when: function() {
  181. // 当地图加载完成
  182. this.teleport = true; // 传送
  183. point_graphicsLayer = new GraphicsLayer({
  184. graphics: []
  185. });
  186. line_graphicsLayer = new GraphicsLayer({
  187. graphics: []
  188. });
  189. text_graphicsLayer = new GraphicsLayer({
  190. graphics: []
  191. });
  192. img_graphicsLayer = new GraphicsLayer({
  193. graphics: []
  194. });
  195. html_graphicsLayer = new GraphicsLayer({
  196. graphics: []
  197. });
  198. map.add(line_graphicsLayer);
  199. map.add(point_graphicsLayer);
  200. map.add(text_graphicsLayer);
  201. map.add(img_graphicsLayer);
  202. map.add(html_graphicsLayer);
  203. view.on("click", (event) => { // 监听鼠标点击 判断是否可以点击
  204. view.hitTest(event).then((response) => {
  205. let result = null;
  206. if (response.results.length) {
  207. const canClick = ["picture-marker", "text"];
  208. response.results.forEach(item => {
  209. if (canClick.indexOf(item.graphic.symbol.type) >= 0) {
  210. result = item.graphic.attributes;
  211. }
  212. })
  213. }
  214. if (result != null) {
  215. this.$emit('clickMap', result);
  216. }
  217. })
  218. });
  219. view.on("pointer-move", (event) => { // 监听鼠标移动 判断是否把鼠标变成小手
  220. view.hitTest(event).then((response) => {
  221. let result = null;
  222. if (response.results.length) {
  223. const canClick = ["picture-marker", "text"];
  224. response.results.forEach(item => {
  225. if (canClick.indexOf(item.graphic.symbol.type) >= 0) {
  226. result = item.graphic.attributes;
  227. }
  228. })
  229. }
  230. if (result) {
  231. this.pointer= true;
  232. } else {
  233. this.pointer= false;
  234. }
  235. })
  236. })
  237. view.watch("extent", (evt) => {
  238. this.markers.forEach((marker, index) => {
  239. let graphics = html_graphics[index];
  240. let screenPoint = view.toScreen(graphics.geometry);
  241. marker.x = screenPoint.x - marker.ox;
  242. marker.y = screenPoint.y - marker.oy;
  243. if (marker.canScale) {
  244. marker.scale = 0.0272 / evt.height;
  245. // console.log(evt.height)
  246. if (evt.height < 0.295) {
  247. if (!marker.show) {
  248. marker.show = true;
  249. graphics.symbol = new SimpleMarkerSymbol({
  250. color: "transparent",
  251. outline: {
  252. color: [255, 255, 255],
  253. width: 0
  254. }
  255. });
  256. }
  257. } else {
  258. if (marker.show) {
  259. marker.show = false;
  260. graphics.symbol = new SimpleMarkerSymbol({
  261. color: [5,187,76,0.75],
  262. outline: {
  263. color: [5,187,76],
  264. width: 2
  265. }
  266. });
  267. }
  268. }
  269. }
  270. });
  271. });
  272. },
  273. // 通过name或code获取img
  274. getFanImg: function(nameOrState) {
  275. const item = this.fanStateImgMapping.find(t => t.name == nameOrState || t.code ==
  276. nameOrState);
  277. if (item) {
  278. return item.img;
  279. } else {
  280. console.error(nameOrState, "对应的图片不存在");
  281. return "";
  282. }
  283. },
  284. // 添加html
  285. addHtmlPoint: function (point, slot, ox=0, oy=0, scale=true) {
  286. point[0] += this.oToLL(this.data.ox, this.data.oy).x;
  287. point[1] += this.oToLL(this.data.ox, this.data.oy).y;
  288. let item = {
  289. geometry: {
  290. type: "point",
  291. spatialReference: SpatialReference.WGS84,
  292. x: point[0],
  293. y: point[1],
  294. },
  295. symbol: {
  296. type: "simple-marker",
  297. color: "transparent",
  298. outline: {
  299. color: [255, 255, 255],
  300. width: 0
  301. }
  302. },
  303. // symbol: null,
  304. attributes: {},
  305. popupTemplate: null
  306. };
  307. let graphic = new Graphic(item);
  308. html_graphics.push(graphic);
  309. html_graphicsLayer.add(graphic);
  310. let screenPoint = view.toScreen(graphic.geometry);
  311. let marker = {
  312. slot: slot,
  313. ox: ox,
  314. oy: oy,
  315. x: screenPoint.x - ox,
  316. y: screenPoint.y - oy,
  317. scale: 1,
  318. canScale: scale,
  319. show: true,
  320. };
  321. this.markers.push(marker);
  322. return marker;
  323. },
  324. // 添加图片点
  325. addImagePoint: function (point, src, width, height) {
  326. let item = {
  327. geometry: {
  328. type: "point",
  329. spatialReference: SpatialReference.WGS84,
  330. x: point[0],
  331. y: point[1]
  332. },
  333. symbol: {
  334. type: "picture-marker",
  335. url: this.tempImg,
  336. width: width,
  337. height: height,
  338. xoffset: 0,
  339. yoffset: 0
  340. },
  341. attributes: {},
  342. popupTemplate: null
  343. };
  344. console.log(item)
  345. let graphic = new Graphic(item);
  346. img_graphics.push(graphic);
  347. img_graphicsLayer.add(graphic);
  348. },
  349. // 移除
  350. remove: function (type, obj) {
  351. if (type == "html") {
  352. let markerIndex = this.markers.findIndex(t => t.slot == obj.slot);
  353. // let marker = this.markers[markerIndex];
  354. this.markers.splice(markerIndex, 1);
  355. let graphic = html_graphics[markerIndex];
  356. html_graphics.splice(markerIndex, 1);
  357. html_graphicsLayer.remove(graphic);
  358. } else {
  359. console.info("arcgis: No type:", type)
  360. }
  361. },
  362. // 添加图片点
  363. addImgPoint: function(jsonItem) {
  364. let item = JSON.parse(JSON.stringify(jsonItem));
  365. item.geometry.type = "point";
  366. item.geometry.spatialReference = SpatialReference.WGS84;
  367. item.geometry.x += this.oToLL(this.data.ox, this.data.oy).x;
  368. item.geometry.y += this.oToLL(this.data.ox, this.data.oy).y;
  369. item.symbol = {
  370. type: "picture-marker",
  371. url: item.attributes.type == "升压站" ? this.syzImg : this.getFanImg(item.attributes.state),
  372. width: item.attributes.type == "升压站" ? "96px" : "48px",
  373. height: item.attributes.type == "升压站" ? "96px" : "48px",
  374. xoffset: 0,
  375. yoffset: 18
  376. };
  377. let graphic = new Graphic(item);
  378. point_graphics.push(graphic);
  379. point_graphicsLayer.add(graphic);
  380. this.addImgText(jsonItem);
  381. },
  382. // 添加图片文字
  383. addImgText: function(jsonItem) {
  384. let item = JSON.parse(JSON.stringify(jsonItem));
  385. item.geometry.type = "point";
  386. item.geometry.spatialReference = SpatialReference.WGS84;
  387. item.geometry.x += this.oToLL(this.data.ox, this.data.oy).x;
  388. item.geometry.y += this.oToLL(this.data.ox, this.data.oy).y;
  389. item.symbol = {
  390. type: "text",
  391. color: "white",
  392. haloColor: "black",
  393. haloSize: "1px",
  394. text: item.attributes.code,
  395. xoffset: 0,
  396. yoffset: -10,
  397. font: {
  398. size: 12,
  399. // family: "Josefin Slab",/
  400. weight: "bold"
  401. }
  402. };
  403. let graphic = new Graphic(item);
  404. text_graphics.push(graphic);
  405. text_graphicsLayer.add(graphic);
  406. },
  407. // 添加线段
  408. addLine: function(jsonItem) {
  409. let item = JSON.parse(JSON.stringify(jsonItem));
  410. item.geometry.type = "polyline";
  411. item.geometry.spatialReference = SpatialReference.WGS84;
  412. item.geometry.paths.forEach(item => {
  413. item[0] += this.oToLL(this.data.ox, this.data.oy).x;
  414. item[1] += this.oToLL(this.data.ox, this.data.oy).y;
  415. });
  416. item.symbol = {
  417. type: "simple-line",
  418. color: "#05bb4c",
  419. width: item.attributes.width,
  420. }
  421. let graphic = new Graphic(item);
  422. line_graphics.push(graphic);
  423. line_graphicsLayer.add(graphic);
  424. },
  425. addFanByJson: function(jsonObj, lineJsonObj) {
  426. // jsonObj.forEach(item => {
  427. // this.addImgPoint(item);
  428. // });
  429. lineJsonObj.forEach(item => {
  430. this.addLine(item);
  431. });
  432. },
  433. goto: function(point) {
  434. view.goTo({
  435. center: point
  436. }).catch(function(error) {
  437. if (error.name != "AbortError") {
  438. console.error(error);
  439. }
  440. });
  441. }
  442. },
  443. // 生命周期钩子
  444. beforeCreate() {
  445. // 创建前
  446. },
  447. created() {
  448. // 创建后
  449. },
  450. beforeMount() {
  451. // 渲染前
  452. },
  453. mounted() {
  454. // 渲染后
  455. this.initMap();
  456. },
  457. beforeUpdate() {
  458. // 数据更新前
  459. },
  460. updated() {
  461. // 数据更新后
  462. },
  463. }
  464. </script>
  465. <style lang="less">
  466. .arcgis {
  467. width: 100%;
  468. height: 100%;
  469. position: relative;
  470. overflow: hidden;
  471. &.pointer {
  472. cursor: pointer;
  473. }
  474. #viewDiv {
  475. position: absolute;
  476. z-index: 1;
  477. width: 100%;
  478. height: 100%;
  479. top: 0;
  480. left: 0;
  481. }
  482. #marker {
  483. position: absolute;
  484. z-index: 2;
  485. width: 0;
  486. height: 0;
  487. top: 0;
  488. left: 0;
  489. }
  490. .arcgis-marker {
  491. position: absolute;
  492. z-index: 2;
  493. }
  494. }
  495. </style>