analysis.vue 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569
  1. <template>
  2. <div class="analysis-com">
  3. <div class="draw-edit" v-if="cusTransfer.draw.center">
  4. <div class="draw-edit-content">
  5. 周边范围:
  6. <div class="radius-min __hover" @click="cusTransfer.draw.radius > 1 ? (cusTransfer.draw.radius--, handleRangeInput(cusTransfer.draw.radius)) : undefined">-</div>
  7. <CusFormColumn
  8. link="number"
  9. label=""
  10. :clearable="false"
  11. v-model:param="cusTransfer.draw.radius"
  12. @input="handleRangeInput"/>
  13. <div class="radius-max __hover" @click="cusTransfer.draw.radius < 60 ? (cusTransfer.draw.radius++, handleRangeInput(cusTransfer.draw.radius)) : undefined">+</div>
  14. <div class="tips" v-if="cusTransfer.draw.radiusTips">请输入1~60的正整数!</div>
  15. km
  16. </div>
  17. <div class="__cus-buttons-3">
  18. <div class="__cus-button-submit __hover" :style="`cursor: ${cusTransfer.draw.radiusTips ? 'not-allowed' : ''};`" @click="initData">确定</div>
  19. <div class="__cus-button-cancel __hover" @click="onCancel">取消</div>
  20. </div>
  21. </div>
  22. <div class="content __box-shadow" v-if="cusTransfer.result.isInit" v-loading="$store.state.gis.analysis.loading">
  23. <div class="head-tab">
  24. <div class="head-tab-item __hover" :class="{active: cusTransfer.switchType === 'power'}" @click="cusTransfer.switchType = 'power'">
  25. <SvgIcon name="search"/>处置力量({{ $store.state.gis.analysis.power.length }}人)
  26. </div>
  27. <div class="head-tab-item __hover" :class="{active: cusTransfer.switchType === 'device'}" @click="cusTransfer.switchType = 'device'">
  28. <SvgIcon name="search"/>设备({{ $store.state.gis.analysis.device.length }}台)
  29. </div>
  30. <div class="head-tab-close __hover" @click="$emit('cancel')">
  31. <SvgIcon name="close_2" color="#0069FF" size="14"/>
  32. </div>
  33. </div>
  34. <div class="result">
  35. <template v-if="cusTransfer.switchType === 'power'">
  36. <div class="form">
  37. <CusFormColumn
  38. labelWidth="50"
  39. :span="24"
  40. label="搜索:"
  41. v-model:param="cusTransfer.result.power.tempForm.text"
  42. />
  43. <div class="__cus-buttons-2">
  44. <div class="__cus-button-submit __hover" @click="onSearchPower">搜索</div>
  45. <div class="__cus-button-cancel __hover" @click="onResetPower">重置</div>
  46. </div>
  47. </div>
  48. <div class="table">
  49. <CusTable
  50. :tableData="powerTableDataCpt"
  51. :tableHead="cusTransfer.result.power.head"
  52. :total="powerTableFilterCpt.length"
  53. :page="cusTransfer.result.power.pageNum"
  54. :pageSize="cusTransfer.result.power.pageSize"
  55. @handlePage="handlePagePower"
  56. >
  57. </CusTable>
  58. </div>
  59. </template>
  60. <template v-else>
  61. <div class="form">
  62. <CusFormColumn
  63. labelWidth="50"
  64. :span="24"
  65. label="搜索:"
  66. v-model:param="cusTransfer.result.device.tempForm.text"
  67. />
  68. <div class="__cus-buttons-2">
  69. <div class="__cus-button-submit __hover" @click="onSearchDevice">搜索</div>
  70. <div class="__cus-button-cancel __hover" @click="onResetDevice">重置</div>
  71. </div>
  72. </div>
  73. <div class="table">
  74. <CusTable
  75. :tableData="deviceTableDataCpt"
  76. :tableHead="cusTransfer.result.device.head"
  77. :total="deviceTableFilterCpt.length"
  78. :page="cusTransfer.result.device.pageNum"
  79. :pageSize="cusTransfer.result.device.pageSize"
  80. @handlePage="handlePageDevice"
  81. :row-class-name="deviceTableRowClassName"
  82. @row-click="deviceHandleRowClick"
  83. >
  84. </CusTable>
  85. </div>
  86. </template>
  87. </div>
  88. </div>
  89. </div>
  90. </template>
  91. <script lang="ts">
  92. import {
  93. defineComponent,
  94. computed,
  95. onMounted,
  96. ref,
  97. reactive,
  98. watch,
  99. getCurrentInstance,
  100. ComponentInternalInstance,
  101. toRefs,
  102. nextTick, onUnmounted
  103. } from 'vue'
  104. import {useStore} from 'vuex'
  105. import {useRouter, useRoute} from 'vue-router'
  106. import {ElMessage, ElMessageBox} from "element-plus";
  107. import * as interaction from "ol/interaction";
  108. import * as geom from 'ol/geom';
  109. import * as style from "ol/style";
  110. import * as ol from "ol";
  111. import SelectChartCom from "@/views/gis/layout/tools/select-chart.vue";
  112. import * as turf from "@turf/turf";
  113. export default defineComponent({
  114. name: '',
  115. components: {SelectChartCom},
  116. props: {
  117. map: {
  118. required: true,
  119. default: <any>{}
  120. },
  121. mapFunc: {
  122. required: true
  123. },
  124. transfer: {}
  125. },
  126. setup(props, {emit}) {
  127. const store = useStore();
  128. const router = useRouter();
  129. const route = useRoute();
  130. const that = (getCurrentInstance() as ComponentInternalInstance).appContext.config.globalProperties
  131. const state = reactive({
  132. transfer: <any>props.transfer,
  133. cusTransfer: <any>{
  134. draw: {
  135. center: null,
  136. radius: 5,
  137. wkt: '',
  138. radiusTips: false
  139. },
  140. analysisDraw: <any>null,
  141. switchType: 'power',
  142. result: {
  143. isInit: false,
  144. power: {
  145. head: [
  146. {value: "name", label: "姓名", show: true,},
  147. {value: "phone", label: "联系电话", show: true, width: 120},
  148. {value: "area", label: "管辖区域", show: true,},
  149. {value: "distance", label: "距离", show: true},
  150. ],
  151. data: <any>[],
  152. pageNum: 1,
  153. pageSize: 20,
  154. form: {},
  155. tempForm: {
  156. text: ''
  157. }
  158. },
  159. device: {
  160. head: [
  161. {value: "name", label: "名称", show: true,},
  162. {value: "status", label: "状态", show: true, width: 60},
  163. {value: "type", label: "类型", show: true, width: 90},
  164. {value: "distance", label: "距离", show: true, width: 90},
  165. ],
  166. data: <any>[],
  167. pageNum: 1,
  168. pageSize: 20,
  169. form: {},
  170. tempForm: {
  171. text: ''
  172. }
  173. }
  174. },
  175. analysisDrawHelpTooltipElement: null,
  176. },
  177. })
  178. const handleRangeInput = (v) => {
  179. const nV = Number(v)
  180. if (!isNaN(Number(nV))) {
  181. if (nV < 1 || nV > 60) {
  182. state.cusTransfer.draw.radiusTips = true
  183. } else {
  184. state.cusTransfer.draw.radiusTips = false
  185. state.cusTransfer.draw.radius = nV
  186. setCircle()
  187. }
  188. }
  189. }
  190. const initLayer = () => {
  191. store.state.gis.analysis.layer.setVisible(true)
  192. if (store.state.gis.analysis.wkt) {
  193. if (state.cusTransfer.analysisDraw) {
  194. props.map.addInteraction(state.cusTransfer.analysisDraw);
  195. }
  196. } else {
  197. let sketch;
  198. let helpTooltip;
  199. const createHelpTooltip = () => {
  200. const id = 'analysisDrawHelpTooltipElementId'
  201. if (state.cusTransfer.analysisDrawHelpTooltipElement) {
  202. props.map.removeOverlay(props.map.getOverlayById(id))
  203. state.cusTransfer.analysisDrawHelpTooltipElement.parentNode?.removeChild(state.cusTransfer.analysisDrawHelpTooltipElement);
  204. }
  205. state.cusTransfer.analysisDrawHelpTooltipElement = document.createElement('div');
  206. state.cusTransfer.analysisDrawHelpTooltipElement.className = 'tooltip hidden';
  207. helpTooltip = new ol.Overlay({
  208. id,
  209. element: state.cusTransfer.analysisDrawHelpTooltipElement,
  210. offset: [15, 0],
  211. positioning: 'center-left'
  212. });
  213. props.map.addOverlay(helpTooltip);
  214. }
  215. createHelpTooltip(); //创建帮助提示框
  216. // 标绘
  217. state.cusTransfer.analysisDraw = new interaction.Draw({
  218. source: store.state.gis.analysis.source,//测量绘制层数据源
  219. type: 'Circle', //几何图形类型
  220. style: new style.Style({
  221. stroke: new style.Stroke({
  222. color: '#2860F1',
  223. width: 2,
  224. lineDash: [10, 10]
  225. }),
  226. fill: new style.Fill({
  227. color: 'rgba(20, 129, 241, 0.1)',
  228. }),
  229. })
  230. });
  231. // @ts-ignore
  232. props.map.addInteraction(state.cusTransfer.analysisDraw);
  233. const drawstartHandle = (evt) => {
  234. sketch = evt.feature; //绘制的要素
  235. }
  236. state.cusTransfer.analysisDraw.on('drawstart', drawstartHandle);
  237. const drawendHandle = (evt) => {
  238. // 标绘的时候不需要最终结果dom
  239. props.map.removeOverlay(helpTooltip)
  240. sketch = null; //置空当前绘制的要素对象
  241. state.cusTransfer.analysisDrawHelpTooltipElement.parentNode.removeChild(state.cusTransfer.analysisDrawHelpTooltipElement);
  242. state.cusTransfer.analysisDrawHelpTooltipElement = null; //置空测量工具提示框对象
  243. state.cusTransfer.analysisDraw.un('drawstart', drawstartHandle);
  244. state.cusTransfer.analysisDraw.un('drawend', drawendHandle);
  245. props.map.removeInteraction(state.cusTransfer.analysisDraw);
  246. props.map.un('pointermove', pointerMoveHandler)
  247. state.cusTransfer.analysisDraw = null
  248. store.state.gis.analysis.feature = evt.feature
  249. store.state.gis.analysis.feature.set('featureType', 'analysisCircle')
  250. getCircleParams(evt.feature.getGeometry())
  251. }
  252. state.cusTransfer.analysisDraw.on('drawend', drawendHandle);
  253. const getCircleParams = (circle) => {
  254. const sourceProj = props.map.getView().getProjection();
  255. let radius = Math.round(circle.getRadius() * sourceProj.getMetersPerUnit() / 1000)
  256. if (radius > 60) {
  257. radius = 60
  258. }
  259. state.cusTransfer.draw.center = circle.getCenter()
  260. state.cusTransfer.draw.radius = radius
  261. setCircle()
  262. }
  263. const pointerMoveHandler = (evt) => {
  264. if (evt.dragging) {
  265. return;
  266. }
  267. let helpMsg = '单击开始标绘';//当前默认提示信息
  268. //判断绘制几何类型设置相应的帮助提示信息
  269. if (sketch) {
  270. helpMsg = '双击结束标绘';
  271. }
  272. if (state.cusTransfer.analysisDrawHelpTooltipElement) {
  273. state.cusTransfer.analysisDrawHelpTooltipElement.innerHTML = helpMsg; //将提示信息设置到对话框中显示
  274. helpTooltip.setPosition(evt.coordinate);//设置帮助提示框的位置
  275. state.cusTransfer.analysisDrawHelpTooltipElement.classList.remove('hidden');//移除帮助提示框的隐藏样式进行显示
  276. }
  277. };
  278. props.map.on('pointermove', pointerMoveHandler); //地图容器绑定鼠标移动事件,动态显示帮助提示框内容
  279. }
  280. }
  281. const setCircle = () => {
  282. const circle = turf.circle(state.cusTransfer.draw.center, state.cusTransfer.draw.radius, {steps: 32, units: 'kilometers'})
  283. store.state.gis.analysis.feature.setGeometry(new geom.Polygon(circle.geometry.coordinates))
  284. state.cusTransfer.draw.wkt = that.$easyMap.formatPosition.cpnTwpn(store.state.gis.analysis.feature.getGeometry().getCoordinates())
  285. }
  286. const initData = () => {
  287. if (!state.cusTransfer.draw.radiusTips) {
  288. state.cusTransfer.result.isInit = true
  289. state.cusTransfer.result.power.pageNum = 1
  290. state.cusTransfer.result.device.pageNum = 1
  291. store.dispatch('gis/LOAD_GIS_ANALYSIS', {
  292. center: state.cusTransfer.draw.center,
  293. wkt: state.cusTransfer.draw.wkt
  294. })
  295. that.$easyMap.getShapeView(props.map, that.$easyMap.formatPosition.wpnTcpn(state.cusTransfer.draw.wkt)[0])
  296. }
  297. }
  298. const powerTableFilterCpt = computed(() => {
  299. return store.state.gis.analysis.power.filter(v => v.name.includes(state.cusTransfer.result.power.form.text))
  300. })
  301. const powerTableDataCpt = computed(() => {
  302. return powerTableFilterCpt.value.slice((state.cusTransfer.result.power.pageNum - 1) * state.cusTransfer.result.power.pageSize, state.cusTransfer.result.power.pageNum * state.cusTransfer.result.power.pageSize)
  303. })
  304. const handlePagePower = ({page, pageSize}: any) => {
  305. state.cusTransfer.result.power.pageNum = page
  306. state.cusTransfer.result.power.pageSize = pageSize
  307. }
  308. const onSearchPower = () => {
  309. state.cusTransfer.result.power.pageNum = 1
  310. state.cusTransfer.result.power.form = JSON.parse(JSON.stringify(state.cusTransfer.result.power.tempForm))
  311. }
  312. const onResetPower = () => {
  313. state.cusTransfer.result.power.tempForm = {
  314. text: '',
  315. }
  316. onSearchPower()
  317. }
  318. const deviceTableFilterCpt = computed(() => {
  319. return store.state.gis.analysis.device.filter(v => v.name.includes(state.cusTransfer.result.device.form.text))
  320. })
  321. const deviceTableDataCpt = computed(() => {
  322. return deviceTableFilterCpt.value.slice((state.cusTransfer.result.device.pageNum - 1) * state.cusTransfer.result.device.pageSize, state.cusTransfer.result.device.pageNum * state.cusTransfer.result.device.pageSize)
  323. })
  324. const handlePageDevice = ({page, pageSize}: any) => {
  325. state.cusTransfer.result.device.pageNum = page
  326. state.cusTransfer.result.device.pageSize = pageSize
  327. }
  328. const onSearchDevice = () => {
  329. state.cusTransfer.result.device.pageNum = 1
  330. state.cusTransfer.result.device.form = JSON.parse(JSON.stringify(state.cusTransfer.result.device.tempForm))
  331. }
  332. const onResetDevice = () => {
  333. state.cusTransfer.result.device.tempForm = {
  334. text: '',
  335. }
  336. onSearchDevice()
  337. }
  338. const onCancel = () => {
  339. emit('cancel')
  340. }
  341. const deviceTableRowClassName = ({row}) => {
  342. if (row.id === store.state.gis.gisParams.default.feature?.getId()) {
  343. return 'row-active'
  344. }
  345. return ''
  346. }
  347. const deviceHandleRowClick = (row) => {
  348. if (row.id === store.state.gis.gisParams.default.feature?.getId()) {
  349. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_RESET')
  350. } else {
  351. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_SB', {
  352. wkt: row.wkt,
  353. id: row.id,
  354. info: row.info
  355. })
  356. }
  357. }
  358. watch(() => state.cusTransfer, () => {
  359. emit('update:transfer', state.cusTransfer)
  360. }, { deep: true })
  361. onMounted(() => {
  362. if (store.state.gis.gisParams.qy.analysisWKT) {
  363. emit('cancel')
  364. ElMessage.warning('请先关闭企业周边设备!')
  365. } else {
  366. if (props.transfer) {
  367. state.cusTransfer = props.transfer
  368. } else {
  369. emit('update:transfer', state.cusTransfer)
  370. }
  371. state.cusTransfer.result.power.form = JSON.parse(JSON.stringify(state.cusTransfer.result.power.tempForm))
  372. state.cusTransfer.result.device.form = JSON.parse(JSON.stringify(state.cusTransfer.result.device.tempForm))
  373. initLayer()
  374. }
  375. })
  376. onUnmounted(() => {
  377. state.cusTransfer.result.isInit = false
  378. store.state.gis.analysis.layer.setVisible(false)
  379. if (state.cusTransfer.analysisDraw) {
  380. props.map.removeInteraction(state.cusTransfer.analysisDraw);
  381. }
  382. state.cusTransfer.analysisDrawHelpTooltipElement?.parentNode.removeChild(state.cusTransfer.analysisDrawHelpTooltipElement);
  383. store.dispatch('gis/LOAD_GIS_ANALYSIS_RESET')
  384. })
  385. return {
  386. ...toRefs(state),
  387. handleRangeInput,
  388. initData,
  389. setCircle,
  390. powerTableFilterCpt,
  391. powerTableDataCpt,
  392. handlePagePower,
  393. onSearchPower,
  394. onResetPower,
  395. deviceTableFilterCpt,
  396. deviceTableDataCpt,
  397. handlePageDevice,
  398. onSearchDevice,
  399. onResetDevice,
  400. onCancel,
  401. deviceTableRowClassName,
  402. deviceHandleRowClick,
  403. }
  404. },
  405. })
  406. </script>
  407. <style scoped lang="scss">
  408. .draw-edit {
  409. width: 160px;
  410. height: 66px;
  411. background: #FFFFFF;
  412. opacity: 0.8;
  413. border-radius: 5px;
  414. position: fixed;
  415. bottom: 68px;
  416. left: calc((100% - 160px) / 2);
  417. display: flex;
  418. flex-direction: column;
  419. justify-content: space-between;
  420. padding: 9px 0;
  421. .draw-edit-content {
  422. display: flex;
  423. align-items: center;
  424. justify-content: center;
  425. font-size: 12px;
  426. font-family: Microsoft YaHei;
  427. font-weight: 400;
  428. color: #808080;
  429. .radius-min, .radius-max {
  430. margin: 0 3px;
  431. &:hover {
  432. color: #409EFF;
  433. }
  434. }
  435. :deep(.cus-form-column) {
  436. max-width: unset;
  437. width: 44px;
  438. flex: unset;
  439. .el-form-item {
  440. margin: 0;
  441. .el-form-item__content {
  442. height: 20px;
  443. .el-input {
  444. .el-input__wrapper {
  445. padding: 0;
  446. border-radius: 5px;
  447. .el-input__inner {
  448. height: 100%;
  449. text-align: center;
  450. border: 1px solid #808080;
  451. border-radius: 5px;
  452. &::placeholder {
  453. font-size: 12px;
  454. font-family: Microsoft YaHei;
  455. }
  456. }
  457. }
  458. }
  459. }
  460. }
  461. }
  462. .tips {
  463. $footH: 6px;
  464. $bgColor: rgba(255, 255, 255, 0.8);
  465. position: absolute;
  466. top: -28px;
  467. right: -10px;
  468. font-size: 12px;
  469. font-family: Microsoft YaHei;
  470. font-weight: 400;
  471. color: #E60012;
  472. background: $bgColor;
  473. border-radius: 4px;
  474. padding: 6px 8px;
  475. display: flex;
  476. align-items: center;
  477. justify-content: center;
  478. &:after {
  479. content: '';
  480. position: absolute;
  481. bottom: -$footH;
  482. border-top: $footH solid $bgColor;
  483. border-left: $footH solid transparent;
  484. border-right: $footH solid transparent;
  485. }
  486. }
  487. }
  488. }
  489. .content {
  490. position: fixed;
  491. width: 404px;
  492. height: calc(100% - 100px);
  493. background-color: #FFFFFF;
  494. box-sizing: border-box;
  495. display: flex;
  496. flex-direction: column;
  497. .head-tab {
  498. height: 40px;
  499. width: 100%;
  500. border-bottom: 1px solid #EEEEEE;
  501. display: flex;
  502. align-items: center;
  503. box-sizing: border-box;
  504. padding-left: 12px;
  505. padding-right: 22px;
  506. .head-tab-item {
  507. height: 100%;
  508. display: flex;
  509. align-items: center;
  510. justify-content: center;
  511. position: relative;
  512. font-size: 14px;
  513. font-family: Microsoft YaHei;
  514. font-weight: 400;
  515. color: #757575;
  516. padding: 0 4px;
  517. min-width: 60px;
  518. margin-right: 10px;
  519. &:last-child {
  520. margin-right: 0;
  521. }
  522. &.active {
  523. color: #0069FF;
  524. &:after {
  525. content: '';
  526. position: absolute;
  527. width: 100%;
  528. height: 3px;
  529. bottom: -1px;
  530. background-color: #0062E9;
  531. }
  532. }
  533. }
  534. .head-tab-close {
  535. margin-left: auto;
  536. }
  537. }
  538. .result {
  539. flex: 1;
  540. display: flex;
  541. flex-direction: column;
  542. padding: 0 10px 10px 10px;
  543. .form {
  544. display: flex;
  545. align-items: center;
  546. margin: 10px 0;
  547. :deep(.el-form-item) {
  548. margin-bottom: 0px;
  549. }
  550. .cus-form-column {
  551. flex: 1;
  552. margin-right: 8px;
  553. }
  554. }
  555. .table {
  556. flex: 1;
  557. :deep(.row-active) {
  558. >td {
  559. background: rgba(16,140,243,0.1);
  560. }
  561. }
  562. }
  563. }
  564. }
  565. </style>