analysis.vue 19 KB

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