index.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545
  1. <template>
  2. <div class="gis-layout">
  3. <EasyMapComponent
  4. class="map"
  5. @easyMapLoad="mapLoad"
  6. />
  7. <div class="gis-menu">
  8. <img class="gis-menu-bottom" src="@/assets/images/gis-layout/gis-layout-menu_bottom.png" alt=""/>
  9. <template v-for="item in menuCpt">
  10. <div class="gis-menu-item __hover" :class="{active: $route.name === item.name, max: String(item.meta.title).length >5}" @click="$router.push({name: item.name})">
  11. {{item.meta.title}}
  12. </div>
  13. </template>
  14. </div>
  15. <div class="gis-tools" v-if="map">
  16. <div class="gt-search __box-shadow">
  17. <el-autocomplete
  18. ref="ref_search"
  19. v-model="searchInput"
  20. :fetch-suggestions="searchHandleMapSearch"
  21. popper-class="__gis-search_popper"
  22. clearable
  23. :debounce="800"
  24. placeholder="请输入关键字进行搜索"
  25. @select="searchToMapLocation"
  26. >
  27. <template #default="{ item }">
  28. <div class="search-item __hover">
  29. <img :src="item.icon" alt=""/>
  30. <div class="name">{{item.name}}</div>
  31. </div>
  32. </template>
  33. </el-autocomplete>
  34. <div class="search-icon __hover" @click="ref_search.focus()">
  35. <SvgIcon name="search" color="#ffffff"/>
  36. </div>
  37. </div>
  38. <div class="gt-tools __box-shadow">
  39. <template v-for="(item, index) in ToolsMapper">
  40. <div class="tools-line" v-if="index > 0"/>
  41. <div class="tools-item __hover" :class="{active: toolsType === item.value, disabled: item.disabled || (item.value === 'analysis' && toolsType === item.value)}" @click="(item.disabled || (item.value === 'analysis' && toolsType === item.value)) ? undefined : toolsHandleClick(item)">
  42. {{item.label}}
  43. </div>
  44. </template>
  45. </div>
  46. <div class="gt-tools-component" v-if="toolsType && toolsType !== 'clear' && toolsCom">
  47. <Component :is="toolsCom" :map="map" :mapFunc="mapFunc" v-model:transfer="toolsParams[toolsType]" @cancel="toolsType = ''"/>
  48. </div>
  49. </div>
  50. <div class="gis-content">
  51. <RouterViewCom/>
  52. </div>
  53. <VideoPlayKedaCom v-if="$store.state.gis.videoParams.show" v-model:layout="videoLayout" :form="$store.state.gis.videoParams" @close="$store.dispatch('gis/LOAD_VIDEO_PARAMS', {show: false})"/>
  54. <!-- <div class="mockButtons">-->
  55. <!-- <el-button @click="mockSB1">模拟设备1</el-button>-->
  56. <!-- <el-button @click="mockSB2">模拟设备2</el-button><br/>-->
  57. <!-- <el-button @click="mockJQ1">模拟景区1</el-button><br/>-->
  58. <!-- <el-button @click="mockLG1">模拟旅馆1</el-button><br/>-->
  59. <!-- <el-button @click="mockCZW1">模拟出租屋1</el-button><br/>-->
  60. <!-- <el-button @click="mockHCZ1">模拟火车站1</el-button>-->
  61. <!-- </div>-->
  62. </div>
  63. <GisDefaultDom ref="ref_gisDefault"/>
  64. <GisQyDom ref="ref_gisQy"/>
  65. </template>
  66. <script lang="ts">
  67. import {
  68. defineComponent,
  69. computed,
  70. onMounted,
  71. ref,
  72. reactive,
  73. watch,
  74. getCurrentInstance,
  75. ComponentInternalInstance,
  76. toRefs,
  77. nextTick,
  78. markRaw
  79. } from 'vue'
  80. import {useStore} from 'vuex'
  81. import {useRouter, useRoute} from 'vue-router'
  82. import {ElMessage, ElMessageBox} from "element-plus";
  83. import RouterViewCom from '@/layout/router-view.vue'
  84. import ElementCom from './tools/element.vue'
  85. import SelectCom from './tools/select.vue'
  86. import AnalysisCom from './tools/analysis.vue'
  87. import ToolCom from './tools/tool.vue'
  88. import PositionCom from './tools/position.vue'
  89. import BaseCom from './tools/base.vue'
  90. import ExampleCom from './tools/example.vue'
  91. import VideoPlayKedaCom from "@/views/gis/business/common/VideoPlayKeda.vue";
  92. import {clearLocationDom} from '@/components/easyMap/func/location'
  93. import {clearMeasureDom} from '@/components/easyMap/func/measure'
  94. import GisDefaultDom from '../map-info/overlay/default/index.vue'
  95. import GisQyDom from '../map-info/overlay/qy/index.vue'
  96. import axios from "axios";
  97. import store from "@/store";
  98. export default defineComponent({
  99. name: '',
  100. components: {
  101. RouterViewCom,
  102. ElementCom,
  103. SelectCom,
  104. AnalysisCom,
  105. ToolCom,
  106. PositionCom,
  107. BaseCom,
  108. ExampleCom,
  109. VideoPlayKedaCom,
  110. GisDefaultDom,
  111. GisQyDom,
  112. },
  113. props: {},
  114. setup(props, {emit}) {
  115. const store = useStore();
  116. const router = useRouter();
  117. const route = useRoute();
  118. const that = (getCurrentInstance() as ComponentInternalInstance).appContext.config.globalProperties
  119. const state = reactive({
  120. map: <any>null,
  121. mapFunc: <any>null,
  122. searchInput: '',
  123. toolsType: '',
  124. toolsCom: null,
  125. toolsParams: {
  126. element: null,
  127. position: null,
  128. select: null,
  129. analysis: null
  130. },
  131. videoLayout: {
  132. width: 640,
  133. height: 366,
  134. left: 1200,
  135. top: 460
  136. },
  137. })
  138. const ToolsMapper = [
  139. {label: '图层', value: 'element', com: ElementCom, disabled: false},
  140. {label: '框选', value: 'select', com: SelectCom, disabled: false},
  141. {label: '周边分析', value: 'analysis', com: AnalysisCom, disabled: false},
  142. {label: '工具', value: 'tool', com: ToolCom},
  143. {label: '定位', value: 'position', com: PositionCom},
  144. {label: '清空标绘', value: 'clear', com: undefined},
  145. {label: '底图', value: 'base', com: BaseCom},
  146. {label: '图例', value: 'example', com: ExampleCom},
  147. ]
  148. const ref_search = ref()
  149. const ref_gisDefault = ref()
  150. const ref_gisQy = ref()
  151. const mapLoad = (map, func) => {
  152. state.map = map
  153. store.dispatch('gis/LOAD_GIS_MAP', {
  154. map: state.map,
  155. defaultDom: ref_gisDefault.value,
  156. qyDom: ref_gisQy.value,
  157. })
  158. state.mapFunc = func
  159. store.dispatch('gis/LOAD_GIS_ELEMENT')
  160. }
  161. const menuCpt = computed(() => {
  162. return router.options.routes.filter(v => v.name === store.state.gis.menuRootName)[0].children?.filter(v => !v.meta.noMenu)
  163. })
  164. const searchHandleMapSearch = (queryString: string, cb: (arg: any) => void) => {
  165. if (queryString.trim()) {
  166. const arr: any = []
  167. const typeName: any = []
  168. store.state.dictionary.elementTypeList.forEach(v => {
  169. if (['sb', 'qy'].includes(v.dictType)) {
  170. typeName.push(v.geoType)
  171. }
  172. })
  173. axios({
  174. url: store.state.gis.element.layer.getSource().getUrls()[0],
  175. method: 'get',
  176. params: {
  177. service: 'WFS',
  178. version: '1.0.0',
  179. request: 'GetFeature',
  180. typename: typeName.join(','),
  181. srsName: 'EPSG:4326',
  182. outputFormat: 'application/json',
  183. CQL_FILTER: `name like '%${queryString.trim()}%'`
  184. }
  185. }).then(res => {
  186. res.data.features?.forEach(v => {
  187. const obj = {
  188. info: v.properties,
  189. id: v.properties.dataId,
  190. name: v.properties.name,
  191. typeValue: v.properties.typeValue,
  192. wkt: `POINT(${v.geometry.coordinates.join(' ')})`,
  193. icon: store.getters['dictionary/elementTypeMapObj'].get(v.properties.typeValue).icon,
  194. featureType: '',
  195. }
  196. if (['lgszyjkscsb', 'jgzzmgs', 'lgsjkyfl'].includes(v.properties.typeValue)) {
  197. obj.featureType = 'qy'
  198. } else if (['gal', 'shl', 'myl'].includes(v.properties.typeValue)) {
  199. obj.featureType = 'sb'
  200. }
  201. arr.push(obj)
  202. })
  203. cb(arr)
  204. }).catch(() => {
  205. cb(arr)
  206. })
  207. } else {
  208. cb([])
  209. }
  210. }
  211. const searchToMapLocation = (val) => {
  212. state.searchInput = val.name
  213. switch (val.featureType) {
  214. case 'qy': {
  215. store.dispatch('gis/LOAD_GIS_PARAMS_QY_RESET')
  216. store.dispatch('gis/LOAD_GIS_PARAMS_QY', {
  217. wkt: val.wkt,
  218. id: val.id,
  219. info: val.info
  220. })
  221. } break
  222. case 'sb': {
  223. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_RESET')
  224. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_SB', {
  225. wkt: val.wkt,
  226. id: val.id,
  227. info: val.info
  228. })
  229. } break
  230. }
  231. }
  232. const toolsHandleClick = (item) => {
  233. if (item.value === 'clear') {
  234. state.toolsCom = null
  235. state.toolsType = ''
  236. // 工具-标绘'layerName', 'toolDrawViewsLayer'
  237. state.map.getLayers().getArray().filter(v => v.get('layerName') === 'toolDrawViewsLayer')?.[0]?.getSource()?.clear()
  238. state.map.getOverlayById('toolOverlay')?.setPosition(undefined)
  239. // 工具-测量'layerName', 'measureLayer'
  240. state.map.getLayers().getArray().filter(v => v.get('layerName') === 'measureLayer')?.[0]?.getSource()?.clear()
  241. clearMeasureDom(state.map)
  242. // 定位'layerName', 'positionLayer'
  243. state.map.getLayers().getArray().filter(v => v.get('layerName') === 'positionLayer')?.[0]?.getSource()?.clear()
  244. clearLocationDom()
  245. // 框选清空
  246. state.toolsParams.select?.clearFunc()
  247. state.toolsParams.select = null
  248. // 周边分析清空
  249. state.toolsParams.analysis = null
  250. } else {
  251. const _type = JSON.parse(JSON.stringify(state.toolsType))
  252. state.toolsType = ''
  253. setTimeout(() => {
  254. state.toolsType = (_type === item.value ? '' : item.value)
  255. state.toolsCom = markRaw(item.com)
  256. }, 100)
  257. }
  258. }
  259. const mockSB1 = () => {
  260. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_SB', {
  261. type: 'sb',
  262. wkt: 'POINT(110.73667031084902 19.011386491828695)',
  263. id: 1,
  264. info: {
  265. "pn": "2526",
  266. "groupid": "89800003866",
  267. "monitorid": "18678",
  268. "name": "测试通道2526",
  269. "online": "0",
  270. "savestatus": "1",
  271. "ptztype": "3",
  272. "deviceid": "46012003001310000011 aaa",
  273. "areacode": "4.60E+19",
  274. "longitude": "110.455893",
  275. "latitude": "19.89477",
  276. "faultedflag": "0",
  277. "status": "OFF",
  278. "isfront": "363658824",
  279. "type": "公安类",
  280. "civilcode": "46012003"
  281. }
  282. })
  283. }
  284. const mockSB2 = () => {
  285. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_SB', {
  286. type: 'sb',
  287. wkt: 'POINT(110.95227700030215 19.027865984016195)',
  288. id: 1,
  289. info: {
  290. "pn": "2526",
  291. "groupid": "89800003866",
  292. "monitorid": "18678",
  293. "name": "2526测试通道2526测试通道2526测试通道2526测试通道2526测试通道",
  294. "online": "0",
  295. "savestatus": "1",
  296. "ptztype": "3",
  297. "deviceid": "46012003001310000011",
  298. "areacode": "4.60E+19",
  299. "longitude": "110.455893",
  300. "latitude": "19.89477",
  301. "faultedflag": "0",
  302. "status": "OFF",
  303. "isfront": "363658824",
  304. "type": "公安类",
  305. "civilcode": "46012003"
  306. }
  307. })
  308. }
  309. const mockJQ1 = () => {
  310. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_JQ', {
  311. type: 'jq',
  312. wkt: 'POINT(110.74277601017481 18.85289817214939)',
  313. id: 1,
  314. info: {
  315. "name": "海南省博物馆",
  316. "address": "在海南省海口市琼山区国兴大道68号线",
  317. "level": "4A",
  318. }
  319. })
  320. }
  321. const mockLG1 = () => {
  322. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_LG', {
  323. type: 'lg',
  324. wkt: 'POINT(110.74277601017481 18.85289817214939)',
  325. id: 1,
  326. info: {
  327. "name": "佳捷连锁酒店(解放西路)",
  328. "address": "海口市龙华区解放西路9号金棕榈商业广场F3",
  329. }
  330. })
  331. }
  332. const mockCZW1 = () => {
  333. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_CZW', {
  334. type: 'czw',
  335. wkt: 'POINT(110.74277601017481 18.85289817214939)',
  336. id: 1,
  337. info: {
  338. "name": "大英出租屋",
  339. "address": "海口市龙华区解放西路9号金棕榈商业广场F3",
  340. }
  341. })
  342. }
  343. const mockHCZ1 = () => {
  344. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_HCZ', {
  345. type: 'hcz',
  346. wkt: 'POINT(110.74277601017481 18.85289817214939)',
  347. id: 1,
  348. info: {
  349. "name": "海口东站",
  350. "address": "海口市龙华区解放西路9号金棕榈商业广场F3",
  351. }
  352. })
  353. }
  354. onMounted(() => {
  355. if (window.cusConfig?.internetAuth) {
  356. if (!sessionStorage.getItem('auth')) {
  357. if(navigator.userAgent.indexOf("Firefox") != -1 || navigator.userAgent.indexOf("Chrome") != -1){
  358. window.location.href = "about:blank";
  359. window.close();
  360. }else{
  361. window.opener = null;
  362. window.open("", "_self");
  363. window.close();
  364. }
  365. }
  366. }
  367. })
  368. return {
  369. ...toRefs(state),
  370. mapLoad,
  371. ToolsMapper,
  372. ref_search,
  373. menuCpt,
  374. searchHandleMapSearch,
  375. searchToMapLocation,
  376. toolsHandleClick,
  377. ref_gisDefault,
  378. ref_gisQy,
  379. mockSB1,
  380. mockSB2,
  381. mockJQ1,
  382. mockLG1,
  383. mockCZW1,
  384. mockHCZ1,
  385. }
  386. },
  387. })
  388. </script>
  389. <style scoped lang="scss">
  390. .gis-layout {
  391. width: 100%;
  392. height: 100vh;
  393. position: absolute;
  394. top: 0;
  395. left: 0;
  396. z-index: 2;
  397. display: flex;
  398. align-items: center;
  399. justify-content: center;
  400. .map {
  401. z-index: 1;
  402. }
  403. .gis-menu {
  404. $diff: 14px;
  405. $menuBottom: 16px;
  406. position: absolute;
  407. z-index: 2;
  408. bottom: $menuBottom;
  409. display: flex;
  410. justify-content: center;
  411. align-items: center;
  412. &:before, &:after {
  413. content: '';
  414. background-image: url("@/assets/images/gis-layout/gis-layout-menu_side.png");
  415. width: 246px;
  416. height: 39px;
  417. position: absolute;
  418. bottom: -$menuBottom;
  419. }
  420. &:before {
  421. left: calc((#{$diff} + 246px) * -1);
  422. }
  423. &:after {
  424. right: calc((#{$diff} + 246px) * -1);
  425. transform: rotateY(180deg);
  426. }
  427. .gis-menu-bottom {
  428. position: absolute;
  429. height: 9px;
  430. width: calc(100% + 10px + #{$diff} * 2);
  431. bottom: -$menuBottom;
  432. }
  433. .gis-menu-item {
  434. width: 95px;
  435. height: 36px;
  436. background-image: url("@/assets/images/gis-layout/gis-layout-menu_item-min.png");
  437. background-size: 100% 100%;
  438. background-repeat: no-repeat;
  439. display: flex;
  440. align-items: center;
  441. justify-content: center;
  442. font-size: 18px;
  443. font-family: YouSheBiaoTiHei;
  444. font-weight: 400;
  445. color: #60AEFF;
  446. &:not(&:last-child) {
  447. margin-right: 4px;
  448. }
  449. &.active {
  450. color: #FFFFFF;
  451. background-image: url("@/assets/images/gis-layout/gis-layout-menu_item-min_active.png");
  452. }
  453. &.max {
  454. width: 125px;
  455. background-image: url("@/assets/images/gis-layout/gis-layout-menu_item-max.png");
  456. &.active {
  457. background-image: url("@/assets/images/gis-layout/gis-layout-menu_item-max_active.png");
  458. }
  459. }
  460. }
  461. }
  462. .gis-tools {
  463. position: absolute;
  464. z-index: 3;
  465. top: 10px;
  466. right: 10px;
  467. width: 404px;
  468. >div {
  469. box-sizing: border-box;
  470. }
  471. .gt-search {
  472. height: 38px;
  473. width: 100%;
  474. display: flex;
  475. align-items: center;
  476. .search-icon {
  477. width: 54px;
  478. height: 100%;
  479. background-color: #3e8ef7;
  480. display: flex;
  481. align-items: center;
  482. justify-content: center;
  483. }
  484. :deep(.el-autocomplete) {
  485. flex: 1;
  486. height: 100%;
  487. .el-input {
  488. width: 100%;
  489. height: 100%;
  490. .el-input__wrapper {
  491. border-radius: 0;
  492. }
  493. }
  494. }
  495. }
  496. .gt-tools {
  497. width: 100%;
  498. height: 38px;
  499. margin-top: 2px;
  500. background-color: #FFFFFF;
  501. border: 1px solid #C7CFDE;
  502. display: flex;
  503. align-items: center;
  504. justify-content: space-around;
  505. padding: 0 8px;
  506. .tools-item {
  507. font-size: 14px;
  508. font-family: Microsoft YaHei;
  509. font-weight: 400;
  510. color: #AEAEAE;
  511. &.active {
  512. color: #1174DB;
  513. font-weight: bold;
  514. }
  515. &.disabled {
  516. opacity: 0.5;
  517. cursor: not-allowed;
  518. }
  519. }
  520. .tools-line {
  521. width: 1px;
  522. height: 16px;
  523. background: linear-gradient(0deg, rgba(174,174,174,0) 0%, rgba(213,213,213,0.99) 50%, rgba(174,174,174,0) 100%);
  524. }
  525. }
  526. .gt-tools-component {
  527. width: 100%;
  528. margin-top: 2px;
  529. }
  530. }
  531. .gis-content {
  532. z-index: 4;
  533. }
  534. }
  535. .mockButtons {
  536. position: absolute;
  537. top: 0;
  538. left: 500px;
  539. }
  540. </style>