leafletMap.vue 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603
  1. <template>
  2. <div class="mapCom">
  3. <div id="map"></div>
  4. </div>
  5. </template>
  6. <script>
  7. // import httpRequest from "@/utils/request.js";
  8. import httpRequest from "@/tools/request.js";
  9. import {
  10. CanvasLabel
  11. } from "@panzhiyue/leaflet-canvaslabel";
  12. import "./leaflet.canvas-markers.js";
  13. export default {
  14. props: {
  15. ids: {
  16. type: Array,
  17. default: () => {
  18. return [];
  19. },
  20. },
  21. windList: {
  22. type: Array,
  23. default: () => {
  24. return [];
  25. },
  26. },
  27. },
  28. data() {
  29. return {
  30. map: null,
  31. DefaultIcon1: null,
  32. layerGroup: [],
  33. layers: [],
  34. rightObj: {},
  35. areaLayer: null,
  36. tilsUrl: "./static/kMapTiles/{z}/{x}/{y}.png",
  37. ciLayer: null,
  38. viewCenterMap: {
  39. GJNY_SXGS_ZZ_FDC_STA: {
  40. lng: 107.86058,
  41. lat: 37.37861
  42. },
  43. GJNY_SXGS_JSL_FDC_STA: {
  44. lng: 108.51853,
  45. lat: 37.45717
  46. },
  47. GJNY_SXGS_XZ_FDC_STA: {
  48. lng: 107.9109,
  49. lat: 37.46585
  50. },
  51. GJNY_SXGS_FSG_FDC_STA: {
  52. lng: 107.85678,
  53. lat: 37.41682
  54. },
  55. GJNY_SXGS_CSL_FDC_STA: {
  56. lng: 108.2031,
  57. lat: 37.2811
  58. },
  59. GJNY_SXGS_LJS_FDC_STA: {
  60. lng: 108.39083,
  61. lat: 37.485
  62. },
  63. },
  64. };
  65. },
  66. watch: {
  67. ids(val) {
  68. this.funStationPos(val);
  69. // this.funStationPosLabel(val)
  70. },
  71. },
  72. mounted() {
  73. this.initMap();
  74. },
  75. unmounted() {
  76. this.map.remove();
  77. this.map = null;
  78. },
  79. methods: {
  80. initMap() {
  81. // 矢量文本标签渲染器
  82. let canvasLabel = new L.CanvasLabel({
  83. collisionFlg: true,
  84. scale: 2,
  85. });
  86. this.map = L.map("map", {
  87. renderer: canvasLabel,
  88. // center: [40.02404009136253, 116.50641060224784], // 地图中心--北京
  89. // center: [38.44673272215545, 106.27624511718751], // 地图中心--银川
  90. // center: [108.953939, 34.266611], // 地图中心--陕西
  91. // center: [109.470962, 34.520632], // 地图中心--渭南
  92. zoom: 16, //缩放比列
  93. zoomControl: false, //禁用 + - 按钮
  94. doubleClickZoom: true, // 禁用双击放大
  95. attributionControl: false, // 移除右下角leaflet标识
  96. preferCanvas: true,
  97. contextmenu: true,
  98. contextmenuWidth: 140,
  99. // contextmenuItems: [{
  100. // text: "功率曲线拟合分析",
  101. // callback: this.powerLine,
  102. // },
  103. // {
  104. // text: "对风偏差分析",
  105. // callback: this.windAny,
  106. // },
  107. // {
  108. // text: "曲线偏差率分析",
  109. // callback: this.qxAny,
  110. // },
  111. // {
  112. // text: "温度与功率分析",
  113. // callback: this.wdyglAny,
  114. // },
  115. // {
  116. // text: "损失电量分析",
  117. // callback: this.ssdlAny,
  118. // },
  119. // {
  120. // text: "桨距角分析",
  121. // callback: this.jjjAny,
  122. // },
  123. // ],
  124. });
  125. // let name = L.tileLayer(
  126. // "http://webrd01.is.autonavi.com/appmaptile?lang=zh_cn&size=1&scale=1&style=7&x={x}&y={y}&z={z}"
  127. // ).addTo(this.map);
  128. let name = L.tileLayer(this.tilsUrl, {
  129. minZoom: 1,
  130. maxZoom: 16,
  131. }).addTo(this.map);
  132. this.setAreaLayer("db", true);
  133. this.setAreaLayer("jb", true);
  134. },
  135. async funStationPos(ids) {
  136. let res = null;
  137. this.rightObj = {};
  138. if (ids.length > 0) {
  139. res = await httpRequest.get("/base/windturbine", {
  140. params: {
  141. wpids: this.ids.join(","),
  142. },
  143. });
  144. } else {
  145. // res = await httpRequest.get("/base/station", {
  146. // params: {},
  147. // });
  148. tabStr = "station"
  149. let datas = await httpRequest.get("/benchmarking/wpByCplist?companyids=NX_FGS&type=-1");
  150. res = datas.data
  151. }
  152. if (res.code === 200) {
  153. if (res.data && res.data.data && res.data.data.length) {
  154. // 清除现有的 layers
  155. this.layers.forEach((layer) => {
  156. this.map.removeLayer(layer);
  157. });
  158. this.layers = [];
  159. const viewCenter =
  160. this.viewCenterMap[res.data.data[0].windpowerstationId] || null;
  161. if (!this.ciLayer) {
  162. this.ciLayer = L.canvasIconLayer({}).addTo(this.map);
  163. } else {
  164. this.ciLayer.clearLayers();
  165. }
  166. // let iconUrl = require(`@/assets/images/indexCom/run_Icon.gif`);
  167. let iconUrl = require(`@/assets/images/indexCom/fengji.png`);
  168. // const img = new Image();
  169. // img.src = require("@/assets/images/indexCom/run_Icon.gif");
  170. for (let i = 0; i < res.data.data.length; i++) {
  171. let item = res.data.data[i];
  172. let marker = L.marker(
  173. [Number(item.latitude), Number(item.longitude)], {
  174. icon: L.divIcon({
  175. className: "iconSty",
  176. iconUrl: iconUrl,
  177. iconSize: [30, 39],
  178. iconAnchor: [15, 12.5],
  179. }
  180. // {
  181. // className: "currentMapIcon",
  182. // html: `<img src="${img.src}" style="width: 30px; height: 39px;">`,
  183. // iconSize: [30, 39],
  184. // iconAnchor: [15, 12.5],
  185. // }
  186. ),
  187. data: item,
  188. }
  189. ).bindTooltip(
  190. `
  191. <div class="tip-box-top">
  192. <div class="item">${item.aname}</div>
  193. <div class="item">经度:${item.latitude}°</div>
  194. <div class="item">纬度:${item.longitude}°</div>
  195. <div class="item">海拔高度:${item.altitude}m</div>
  196. </div>`
  197. );
  198. let latlng = L.latLng(
  199. Number(item.latitude),
  200. Number(item.longitude)
  201. );
  202. let c1 = L.circleMarker(latlng, {
  203. radius: 5,
  204. color: "transparent",
  205. labelStyle: {
  206. text: item.aname,
  207. scale: 1.2,
  208. rotation: 0,
  209. offsetY: 35,
  210. fillStyle: "#fff",
  211. zIndex: i,
  212. },
  213. data: item,
  214. }).addTo(this.map);
  215. marker.on("tooltipopen", (e) => {
  216. this.rightObj = item;
  217. });
  218. marker.addTo(this.map);
  219. this.layers.push(c1);
  220. this.layers.push(marker);
  221. this.ciLayer.addLayer(marker);
  222. }
  223. let center = this.map.getCenter();
  224. this.map.panTo([center.lat, center.lng], {
  225. animate: true
  226. });
  227. if (viewCenter) {
  228. this.map.setView(viewCenter, 13);
  229. } else if (this.layers.length > 0) {
  230. this.map.setView(this.layers[0].getLatLng(), 13);
  231. }
  232. }
  233. }
  234. },
  235. async funStationPosLabel(ids) {
  236. if (this.layers.length > 0) {
  237. for (var i = 0; i < this.layers.length; i++) {
  238. this.map.removeLayer(this.layers[i]);
  239. }
  240. this.layers = [];
  241. }
  242. let res = null;
  243. if (ids.length > 0) {
  244. res = await httpRequest.get("/base/location", {
  245. params: {
  246. ids: this.ids.join(","),
  247. },
  248. });
  249. } else {
  250. res = await httpRequest.get("/base/station", {
  251. params: {},
  252. });
  253. }
  254. if (res.code === 200) {
  255. if (res.data && res.data.length) {
  256. this.layers = [];
  257. const viewCenter =
  258. this.viewCenterMap[res.data[0].windpowerstationId] || null;
  259. for (let i = 0; i < res.data.length; i++) {
  260. let item = res.data[i];
  261. let latlng = L.latLng(
  262. Number(item.latitude),
  263. Number(item.longitude)
  264. );
  265. let c = L.circleMarker(latlng, {
  266. radius: 5,
  267. color: "#12e799",
  268. labelStyle: {
  269. text: item.aname,
  270. scale: 1,
  271. rotation: 0,
  272. offsetY: 15,
  273. fillStyle: "#000",
  274. zIndex: i,
  275. },
  276. data: item,
  277. })
  278. .bindTooltip(
  279. `
  280. <div class="tip-box-top">
  281. <div class="item">${item.aname}</div>
  282. <div class="item">经度:${item.latitude}°</div>
  283. <div class="item">纬度:${item.longitude}°</div>
  284. <div class="item">海拔高度:${item.altitude}m</div>
  285. </div>`
  286. )
  287. .addTo(this.map);
  288. this.layers.push(c);
  289. let that = this;
  290. c.on("mouseover", function onmouseover(e) {
  291. that.rightObj = item;
  292. });
  293. }
  294. let center = this.map.getCenter();
  295. this.map.panTo([center.lat, center.lng], {
  296. animate: true,
  297. });
  298. this.map.setView(viewCenter || this.layers[0].getLatLng(), 13);
  299. }
  300. }
  301. },
  302. setAreaLayer(jsonName, isBounds) {
  303. const wfAllGeoJson = require(`@/assets/${jsonName}.json`);
  304. this.areaLayer = L.geoJSON(wfAllGeoJson, {
  305. style: (feature) => {
  306. return {
  307. fillOpacity: 0.1,
  308. fillColor: "rgb(27, 242, 245)",
  309. weight: 2,
  310. color: "rgb(27, 242, 245)",
  311. };
  312. },
  313. });
  314. this.map.addLayer(this.areaLayer);
  315. if (isBounds) {
  316. this.map.fitBounds(this.areaLayer.getBounds());
  317. }
  318. },
  319. powerLine(e) {
  320. if (!this.rightObj.latitude && !this.rightObj.longitude) {
  321. this.$message({
  322. message: "该坐标系下暂无功率曲线拟合",
  323. type: "error",
  324. });
  325. } else {
  326. this.layers.forEach((item, index) => {
  327. if (item.options.data) {
  328. if (item.options.data.name.indexOf("风电场") === -1) {
  329. if (
  330. item.options.data.latitude === this.rightObj.latitude &&
  331. item.options.data.longitude === this.rightObj.longitude
  332. ) {
  333. this.$emit("rightClick", {
  334. menuIndex: 0,
  335. current: this.rightObj,
  336. });
  337. }
  338. } else {
  339. if (item.options.data.name === this.rightObj.name) {
  340. this.$message({
  341. message: "风场暂无功率曲线拟合功能",
  342. type: "error",
  343. });
  344. }
  345. }
  346. }
  347. });
  348. }
  349. },
  350. windAny(e) {
  351. if (!this.rightObj.latitude && !this.rightObj.longitude) {
  352. this.$message({
  353. message: "该坐标系下暂无对风偏差分析",
  354. type: "error",
  355. });
  356. } else {
  357. this.layers.forEach((item) => {
  358. if (item.options.data) {
  359. if (item.options.data.name.indexOf("风电场") === -1) {
  360. if (
  361. item.options.data.latitude === this.rightObj.latitude &&
  362. item.options.data.longitude === this.rightObj.longitude
  363. ) {
  364. this.$emit("rightClick", {
  365. menuIndex: 1,
  366. current: this.rightObj,
  367. });
  368. }
  369. } else {
  370. if (item.options.data.name === this.rightObj.name) {
  371. this.$message({
  372. message: "风场暂无对风偏差分析功能",
  373. type: "error",
  374. });
  375. }
  376. }
  377. }
  378. });
  379. }
  380. },
  381. qxAny(e) {
  382. if (!this.rightObj.latitude && !this.rightObj.longitude) {
  383. this.$message({
  384. message: "该坐标系下暂无曲线偏差分析",
  385. type: "error",
  386. });
  387. } else {
  388. this.layers.forEach((item) => {
  389. if (item.options.data) {
  390. if (item.options.data.name.indexOf("风电场") === -1) {
  391. if (
  392. item.options.data.latitude === this.rightObj.latitude &&
  393. item.options.data.longitude === this.rightObj.longitude
  394. ) {
  395. this.$emit("rightClick", {
  396. menuIndex: 2,
  397. current: this.rightObj,
  398. });
  399. }
  400. } else {
  401. if (item.options.data.name === this.rightObj.name) {
  402. this.$message({
  403. message: "风场暂无曲线偏差分析功能",
  404. type: "error",
  405. });
  406. }
  407. }
  408. }
  409. });
  410. }
  411. },
  412. wdyglAny(e) {
  413. if (!this.rightObj.latitude && !this.rightObj.longitude) {
  414. this.$message({
  415. message: "该坐标系下暂无温度与功率分析",
  416. type: "error",
  417. });
  418. } else {
  419. this.layers.forEach((item) => {
  420. if (item.options.data) {
  421. if (item.options.data.name.indexOf("风电场") === -1) {
  422. if (
  423. item.options.data.latitude === this.rightObj.latitude &&
  424. item.options.data.longitude === this.rightObj.longitude
  425. ) {
  426. this.$emit("rightClick", {
  427. menuIndex: 3,
  428. current: this.rightObj,
  429. });
  430. }
  431. } else {
  432. if (item.options.data.name === this.rightObj.name) {
  433. this.$message({
  434. message: "风场暂无温度与功率分析功能",
  435. type: "error",
  436. });
  437. }
  438. }
  439. }
  440. });
  441. }
  442. },
  443. ssdlAny(e) {
  444. if (!this.rightObj.latitude && !this.rightObj.longitude) {
  445. this.$message({
  446. message: "该坐标系下暂无损失电量分析",
  447. type: "error",
  448. });
  449. } else {
  450. this.layers.forEach((item) => {
  451. if (item.options.data) {
  452. if (item.options.data.name.indexOf("风电场") === -1) {
  453. if (
  454. item.options.data.latitude === this.rightObj.latitude &&
  455. item.options.data.longitude === this.rightObj.longitude
  456. ) {
  457. this.$emit("rightClick", {
  458. menuIndex: 4,
  459. current: this.rightObj,
  460. });
  461. }
  462. } else {
  463. if (item.options.data.name === this.rightObj.name) {
  464. this.$message({
  465. message: "风场暂无损失电量分析功能",
  466. type: "error",
  467. });
  468. }
  469. }
  470. }
  471. });
  472. }
  473. },
  474. jjjAny(e) {
  475. if (!this.rightObj.latitude && !this.rightObj.longitude) {
  476. this.$message({
  477. message: "该坐标系下暂无桨距角分析",
  478. type: "error",
  479. });
  480. } else {
  481. this.layers.forEach((item) => {
  482. if (item.options.data) {
  483. if (item.options.data.name.indexOf("风电场") === -1) {
  484. if (
  485. item.options.data.latitude === this.rightObj.latitude &&
  486. item.options.data.longitude === this.rightObj.longitude
  487. ) {
  488. this.$emit("rightClick", {
  489. menuIndex: 5,
  490. current: this.rightObj,
  491. });
  492. }
  493. } else {
  494. if (item.options.data.name === this.rightObj.name) {
  495. this.$message({
  496. message: "风场暂无桨距角分析功能",
  497. type: "error",
  498. });
  499. }
  500. }
  501. }
  502. });
  503. }
  504. },
  505. },
  506. };
  507. </script>
  508. <style scoped lang="less">
  509. .mapCom {
  510. height: 100%;
  511. .iconLabel {
  512. width: 80px !important;
  513. }
  514. .iconSty {
  515. .iconStyClass {
  516. width: 50px;
  517. height: 100px;
  518. position: relative;
  519. top: 40px;
  520. }
  521. }
  522. }
  523. #map {
  524. width: 100%;
  525. height: 100%;
  526. }
  527. .lmap-image {
  528. width: 32px;
  529. height: 32px;
  530. }
  531. .lmap-span {
  532. display: inline-block;
  533. margin-left: 5px;
  534. padding: 5px;
  535. font-weight: bold;
  536. line-height: 20px;
  537. font-size: 14px;
  538. color: #fff;
  539. white-space: nowrap;
  540. }
  541. .lmap-text {
  542. display: inline-block;
  543. margin-left: 5px;
  544. padding: 5px;
  545. font-weight: bold;
  546. line-height: 20px;
  547. font-size: 16px;
  548. color: #fff;
  549. width: 500px;
  550. white-space: nowrap;
  551. position: absolute;
  552. text-align: center;
  553. top: 25px;
  554. left: -250px;
  555. }
  556. </style>
  557. <style lang="less">
  558. .mapCom {
  559. .currentMapIcon {
  560. img {
  561. position: absolute;
  562. z-index: 1000;
  563. }
  564. }
  565. }
  566. </style>