index.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534
  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. },
  129. videoLayout: {
  130. width: 640,
  131. height: 366,
  132. left: 1200,
  133. top: 460
  134. },
  135. })
  136. const ToolsMapper = [
  137. {label: '图层', value: 'element', com: ElementCom, disabled: false},
  138. {label: '框选', value: 'select', com: SelectCom, disabled: false},
  139. {label: '周边分析', value: 'analysis', com: AnalysisCom, disabled: false},
  140. {label: '工具', value: 'tool', com: ToolCom},
  141. {label: '定位', value: 'position', com: PositionCom},
  142. {label: '清空标绘', value: 'clear', com: undefined},
  143. {label: '底图', value: 'base', com: BaseCom},
  144. {label: '图例', value: 'example', com: ExampleCom},
  145. ]
  146. const ref_search = ref()
  147. const ref_gisDefault = ref()
  148. const ref_gisQy = ref()
  149. const mapLoad = (map, func) => {
  150. state.map = map
  151. store.dispatch('gis/LOAD_GIS_MAP', {
  152. map: state.map,
  153. defaultDom: ref_gisDefault.value,
  154. qyDom: ref_gisQy.value,
  155. })
  156. state.mapFunc = func
  157. store.dispatch('gis/LOAD_GIS_ELEMENT')
  158. }
  159. const menuCpt = computed(() => {
  160. return router.options.routes.filter(v => v.name === store.state.gis.menuRootName)[0].children?.filter(v => !v.meta.noMenu)
  161. })
  162. const searchHandleMapSearch = (queryString: string, cb: (arg: any) => void) => {
  163. if (queryString.trim()) {
  164. const arr: any = []
  165. const typeName: any = []
  166. store.state.dictionary.elementTypeList.forEach(v => {
  167. if (['sb', 'qy'].includes(v.dictType)) {
  168. typeName.push(v.geoType)
  169. }
  170. })
  171. axios({
  172. url: store.state.gis.element.layer.getSource().getUrls()[0],
  173. method: 'get',
  174. params: {
  175. service: 'WFS',
  176. version: '1.0.0',
  177. request: 'GetFeature',
  178. typename: typeName.join(','),
  179. srsName: 'EPSG:4326',
  180. outputFormat: 'application/json',
  181. CQL_FILTER: `name like '%${queryString.trim()}%'`
  182. }
  183. }).then(res => {
  184. res.data.features?.forEach(v => {
  185. const obj = {
  186. info: v.properties,
  187. id: v.properties.dataId,
  188. name: v.properties.name,
  189. typeValue: v.properties.typeValue,
  190. wkt: `POINT(${v.geometry.coordinates.join(' ')})`,
  191. icon: store.getters['dictionary/elementTypeMapObj'].get(v.properties.typeValue).icon,
  192. featureType: '',
  193. }
  194. if (['lgszyjkscsb', 'jgzzmgs', 'lgsjkyfl'].includes(v.properties.typeValue)) {
  195. obj.featureType = 'qy'
  196. } else if (['gal', 'shl', 'myl'].includes(v.properties.typeValue)) {
  197. obj.featureType = 'sb'
  198. }
  199. arr.push(obj)
  200. })
  201. cb(arr)
  202. }).catch(() => {
  203. cb(arr)
  204. })
  205. } else {
  206. cb([])
  207. }
  208. }
  209. const searchToMapLocation = (val) => {
  210. state.searchInput = val.name
  211. switch (val.featureType) {
  212. case 'qy': {
  213. console.log(1)
  214. store.dispatch('gis/LOAD_GIS_PARAMS_QY_RESET')
  215. store.dispatch('gis/LOAD_GIS_PARAMS_QY', {
  216. wkt: val.wkt,
  217. id: val.id,
  218. info: val.info
  219. })
  220. } break
  221. case 'sb': {
  222. console.log(2)
  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. // 工具-标绘'layerName', 'toolDrawViewsLayer'
  235. state.map.getLayers().getArray().filter(v => v.get('layerName') === 'toolDrawViewsLayer')?.[0]?.getSource()?.clear()
  236. state.map.getOverlayById('toolOverlay')?.setPosition(undefined)
  237. // 工具-测量'layerName', 'measureLayer'
  238. state.map.getLayers().getArray().filter(v => v.get('layerName') === 'measureLayer')?.[0]?.getSource()?.clear()
  239. clearMeasureDom(state.map)
  240. // 定位'layerName', 'positionLayer'
  241. state.map.getLayers().getArray().filter(v => v.get('layerName') === 'positionLayer')?.[0]?.getSource()?.clear()
  242. clearLocationDom()
  243. } else {
  244. state.toolsCom = markRaw(item.com)
  245. state.toolsType = (state.toolsType === item.value ? '' : item.value)
  246. }
  247. }
  248. const mockSB1 = () => {
  249. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_SB', {
  250. type: 'sb',
  251. wkt: 'POINT(110.73667031084902 19.011386491828695)',
  252. id: 1,
  253. info: {
  254. "pn": "2526",
  255. "groupid": "89800003866",
  256. "monitorid": "18678",
  257. "name": "测试通道2526",
  258. "online": "0",
  259. "savestatus": "1",
  260. "ptztype": "3",
  261. "deviceid": "46012003001310000011 aaa",
  262. "areacode": "4.60E+19",
  263. "longitude": "110.455893",
  264. "latitude": "19.89477",
  265. "faultedflag": "0",
  266. "status": "OFF",
  267. "isfront": "363658824",
  268. "type": "公安类",
  269. "civilcode": "46012003"
  270. }
  271. })
  272. }
  273. const mockSB2 = () => {
  274. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_SB', {
  275. type: 'sb',
  276. wkt: 'POINT(110.95227700030215 19.027865984016195)',
  277. id: 1,
  278. info: {
  279. "pn": "2526",
  280. "groupid": "89800003866",
  281. "monitorid": "18678",
  282. "name": "2526测试通道2526测试通道2526测试通道2526测试通道2526测试通道",
  283. "online": "0",
  284. "savestatus": "1",
  285. "ptztype": "3",
  286. "deviceid": "46012003001310000011",
  287. "areacode": "4.60E+19",
  288. "longitude": "110.455893",
  289. "latitude": "19.89477",
  290. "faultedflag": "0",
  291. "status": "OFF",
  292. "isfront": "363658824",
  293. "type": "公安类",
  294. "civilcode": "46012003"
  295. }
  296. })
  297. }
  298. const mockJQ1 = () => {
  299. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_JQ', {
  300. type: 'jq',
  301. wkt: 'POINT(110.74277601017481 18.85289817214939)',
  302. id: 1,
  303. info: {
  304. "name": "海南省博物馆",
  305. "address": "在海南省海口市琼山区国兴大道68号线",
  306. "level": "4A",
  307. }
  308. })
  309. }
  310. const mockLG1 = () => {
  311. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_LG', {
  312. type: 'lg',
  313. wkt: 'POINT(110.74277601017481 18.85289817214939)',
  314. id: 1,
  315. info: {
  316. "name": "佳捷连锁酒店(解放西路)",
  317. "address": "海口市龙华区解放西路9号金棕榈商业广场F3",
  318. }
  319. })
  320. }
  321. const mockCZW1 = () => {
  322. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_CZW', {
  323. type: 'czw',
  324. wkt: 'POINT(110.74277601017481 18.85289817214939)',
  325. id: 1,
  326. info: {
  327. "name": "大英出租屋",
  328. "address": "海口市龙华区解放西路9号金棕榈商业广场F3",
  329. }
  330. })
  331. }
  332. const mockHCZ1 = () => {
  333. store.dispatch('gis/LOAD_GIS_PARAMS_DEFAULT_HCZ', {
  334. type: 'hcz',
  335. wkt: 'POINT(110.74277601017481 18.85289817214939)',
  336. id: 1,
  337. info: {
  338. "name": "海口东站",
  339. "address": "海口市龙华区解放西路9号金棕榈商业广场F3",
  340. }
  341. })
  342. }
  343. onMounted(() => {
  344. if (window.cusConfig?.internetAuth) {
  345. if (!sessionStorage.getItem('auth')) {
  346. if(navigator.userAgent.indexOf("Firefox") != -1 || navigator.userAgent.indexOf("Chrome") != -1){
  347. window.location.href = "about:blank";
  348. window.close();
  349. }else{
  350. window.opener = null;
  351. window.open("", "_self");
  352. window.close();
  353. }
  354. }
  355. }
  356. })
  357. return {
  358. ...toRefs(state),
  359. mapLoad,
  360. ToolsMapper,
  361. ref_search,
  362. menuCpt,
  363. searchHandleMapSearch,
  364. searchToMapLocation,
  365. toolsHandleClick,
  366. ref_gisDefault,
  367. ref_gisQy,
  368. mockSB1,
  369. mockSB2,
  370. mockJQ1,
  371. mockLG1,
  372. mockCZW1,
  373. mockHCZ1,
  374. }
  375. },
  376. })
  377. </script>
  378. <style scoped lang="scss">
  379. .gis-layout {
  380. width: 100%;
  381. height: 100vh;
  382. position: absolute;
  383. top: 0;
  384. left: 0;
  385. z-index: 2;
  386. display: flex;
  387. align-items: center;
  388. justify-content: center;
  389. .map {
  390. z-index: 1;
  391. }
  392. .gis-menu {
  393. $diff: 14px;
  394. $menuBottom: 16px;
  395. position: absolute;
  396. z-index: 2;
  397. bottom: $menuBottom;
  398. display: flex;
  399. justify-content: center;
  400. align-items: center;
  401. &:before, &:after {
  402. content: '';
  403. background-image: url("@/assets/images/gis-layout/gis-layout-menu_side.png");
  404. width: 246px;
  405. height: 39px;
  406. position: absolute;
  407. bottom: -$menuBottom;
  408. }
  409. &:before {
  410. left: calc((#{$diff} + 246px) * -1);
  411. }
  412. &:after {
  413. right: calc((#{$diff} + 246px) * -1);
  414. transform: rotateY(180deg);
  415. }
  416. .gis-menu-bottom {
  417. position: absolute;
  418. height: 9px;
  419. width: calc(100% + 10px + #{$diff} * 2);
  420. bottom: -$menuBottom;
  421. }
  422. .gis-menu-item {
  423. width: 95px;
  424. height: 36px;
  425. background-image: url("@/assets/images/gis-layout/gis-layout-menu_item-min.png");
  426. background-size: 100% 100%;
  427. background-repeat: no-repeat;
  428. display: flex;
  429. align-items: center;
  430. justify-content: center;
  431. font-size: 18px;
  432. font-family: YouSheBiaoTiHei;
  433. font-weight: 400;
  434. color: #60AEFF;
  435. &:not(&:last-child) {
  436. margin-right: 4px;
  437. }
  438. &.active {
  439. color: #FFFFFF;
  440. background-image: url("@/assets/images/gis-layout/gis-layout-menu_item-min_active.png");
  441. }
  442. &.max {
  443. width: 125px;
  444. background-image: url("@/assets/images/gis-layout/gis-layout-menu_item-max.png");
  445. &.active {
  446. background-image: url("@/assets/images/gis-layout/gis-layout-menu_item-max_active.png");
  447. }
  448. }
  449. }
  450. }
  451. .gis-tools {
  452. position: absolute;
  453. z-index: 3;
  454. top: 10px;
  455. right: 10px;
  456. width: 404px;
  457. >div {
  458. box-sizing: border-box;
  459. }
  460. .gt-search {
  461. height: 38px;
  462. width: 100%;
  463. display: flex;
  464. align-items: center;
  465. .search-icon {
  466. width: 54px;
  467. height: 100%;
  468. background-color: #3e8ef7;
  469. display: flex;
  470. align-items: center;
  471. justify-content: center;
  472. }
  473. :deep(.el-autocomplete) {
  474. flex: 1;
  475. height: 100%;
  476. .el-input {
  477. width: 100%;
  478. height: 100%;
  479. .el-input__wrapper {
  480. border-radius: 0;
  481. }
  482. }
  483. }
  484. }
  485. .gt-tools {
  486. width: 100%;
  487. height: 38px;
  488. margin-top: 2px;
  489. background-color: #FFFFFF;
  490. border: 1px solid #C7CFDE;
  491. display: flex;
  492. align-items: center;
  493. justify-content: space-around;
  494. padding: 0 8px;
  495. .tools-item {
  496. font-size: 14px;
  497. font-family: Microsoft YaHei;
  498. font-weight: 400;
  499. color: #AEAEAE;
  500. &.active {
  501. color: #1174DB;
  502. font-weight: bold;
  503. }
  504. &.disabled {
  505. opacity: 0.5;
  506. cursor: not-allowed;
  507. }
  508. }
  509. .tools-line {
  510. width: 1px;
  511. height: 16px;
  512. background: linear-gradient(0deg, rgba(174,174,174,0) 0%, rgba(213,213,213,0.99) 50%, rgba(174,174,174,0) 100%);
  513. }
  514. }
  515. .gt-tools-component {
  516. width: 100%;
  517. margin-top: 2px;
  518. }
  519. }
  520. .gis-content {
  521. z-index: 4;
  522. }
  523. }
  524. .mockButtons {
  525. position: absolute;
  526. top: 0;
  527. left: 500px;
  528. }
  529. </style>