index.vue 37 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145
  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">
  16. <div class="gt-search __box-shadow">
  17. <el-autocomplete
  18. ref="ref_search"
  19. v-model="searchInput"
  20. :fetch-suggestions="searchHandleMapSearch"
  21. :popper-append-to-body="false"
  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"/>
  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}" @click="item.disabled ? 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="showVideo" v-model:layout="videoLayout" :form="qyParams.sbInfo" @close="showVideo = false"/>
  54. </div>
  55. <div ref="ref_qyDom" class="qy-info" :class="`qy--info-${qyParams.qyInfo.tab}`">
  56. <div class="qy-bg-icon qy-bg-icon-1"/>
  57. <div class="qy-bg-icon qy-bg-icon-2"/>
  58. <div class="qy-bg-icon qy-bg-icon-3"/>
  59. <div class="qy-bg-icon qy-bg-icon-4"/>
  60. <div class="qy-main">
  61. <div class="qy-main-head">
  62. <div class="qy-main-head-tips">【企业】</div>
  63. <div class="qy-main-head-name">{{ qyParams.qyInfo.name }}</div>
  64. <SvgIcon class="__hover" name="close_4" size="14" color="#8FFFFF" @click="onCloseQy"/>
  65. </div>
  66. <div class="qy-main-tab">
  67. <template v-for="item in [
  68. {label: '基本信息', value: '1', disabled: false},
  69. {label: '税收信息', value: '2', disabled: true},
  70. {label: '运输车辆', value: '3', disabled: true},
  71. {label: '能耗消息', value: '4', disabled: true},
  72. {label: '周边设备', value: '5', disabled: false},
  73. ]">
  74. <div class="qy-main-tab-item __hover" :class="{active: item.value === qyParams.qyInfo.tab, disabled: item.disabled}" @click="item.disabled ? undefined : qyParams.qyInfo.tab = item.value">{{item.label}}</div>
  75. </template>
  76. </div>
  77. <div v-if="qyParams.qyInfo.tab === '1'" class="qy-main-content-1">
  78. <div>企业名称:{{qyParams.qyInfo[qyParams.qyInfo.tab].name}}</div>
  79. <div>企业法人:{{qyParams.qyInfo[qyParams.qyInfo.tab].people}}</div>
  80. <div>统一社会信用代码:{{qyParams.qyInfo[qyParams.qyInfo.tab].number}}</div>
  81. <div>经营地址:{{qyParams.qyInfo[qyParams.qyInfo.tab].address}}</div>
  82. </div>
  83. <div v-else-if="qyParams.qyInfo.tab === '5'" class="qy-main-content-5">
  84. <div class="qy-main-content-5-radius">
  85. 周边范围:
  86. <div class="radius-min __hover" @click="qyParams.qyInfo[qyParams.qyInfo.tab].radius > 1 ? (qyParams.qyInfo[qyParams.qyInfo.tab].radius--) : undefined">-</div>
  87. <CusFormColumn
  88. link="number"
  89. label=""
  90. :min="1"
  91. :max="60"
  92. :clearable="false"
  93. v-model:param="qyParams.qyInfo[qyParams.qyInfo.tab].radius"
  94. @blur="handleRangeBlur"/>
  95. <div class="radius-max __hover" @click="qyParams.qyInfo[qyParams.qyInfo.tab].radius < 60 ? (qyParams.qyInfo[qyParams.qyInfo.tab].radius++) : undefined">+</div>
  96. km
  97. <div class="submit __hover" @click="onRadiusSubmit">确定</div>
  98. <div class="reset __hover" @click="onRadiusReset">重置</div>
  99. </div>
  100. <div class="qy-main-content-5-table" :style="`height: ${25 + (qyParams.qyInfo[qyParams.qyInfo.tab].tableData.length > 5 ? 5 * 25 : qyParams.qyInfo[qyParams.qyInfo.tab].tableData.length * 25)}px;`" v-loading="qyParams.qyInfo[qyParams.qyInfo.tab].loading" element-loading-background="rgba(0, 0, 0, 0.3)">
  101. <el-auto-resizer>
  102. <template #default="{ height, width }">
  103. <V2Table
  104. :width="width"
  105. :height="height"
  106. :data="qyParams.qyInfo[qyParams.qyInfo.tab].tableData"
  107. :center="qyParams.qyInfo['5'].center"
  108. />
  109. <!-- <el-table-v2-->
  110. <!-- class="__gis-overlay_table-v2"-->
  111. <!-- :columns="qyParams.qyInfo[qyParams.qyInfo.tab].tableHead"-->
  112. <!-- :data="qyParams.qyInfo[qyParams.qyInfo.tab].tableData"-->
  113. <!-- :width="width"-->
  114. <!-- :height="height"-->
  115. <!-- fixed-->
  116. <!-- :row-height="25"-->
  117. <!-- :header-height="25"-->
  118. <!-- >-->
  119. <!-- <template #empty></template>-->
  120. <!-- </el-table-v2>-->
  121. </template>
  122. </el-auto-resizer>
  123. </div>
  124. </div>
  125. </div>
  126. </div>
  127. <div ref="ref_sbDom" class="sb-info">
  128. <div class="sb-info-head">
  129. <SvgIcon class="__hover" name="tips" size="14" color="#8FFFFF"/>设备
  130. </div>
  131. <div class="sb-info-close __hover" @click="onCloseSb">
  132. <img src="@/components/easyMap/images/close.png" alt=""/>
  133. </div>
  134. <div class="sb-main">
  135. <div class="sb-item">
  136. <div class="sb-item-label">名称:</div>
  137. <div class="sb-item-value">{{qyParams.sbInfo?.name}}</div>
  138. </div>
  139. <div class="sb-item">
  140. <div class="sb-item-label">状态:</div>
  141. <div class="sb-item-value">{{qyParams.sbInfo?.online === '0' ? '在线' : '离线'}}</div>
  142. </div>
  143. <div class="play-button __hover" @click="showVideo = true">视频调阅</div>
  144. </div>
  145. </div>
  146. </template>
  147. <script lang="ts">
  148. import {
  149. defineComponent,
  150. computed,
  151. onMounted,
  152. ref,
  153. reactive,
  154. watch,
  155. getCurrentInstance,
  156. ComponentInternalInstance,
  157. toRefs,
  158. nextTick,
  159. markRaw
  160. } from 'vue'
  161. import {useStore} from 'vuex'
  162. import {useRouter, useRoute} from 'vue-router'
  163. import {ElMessage, ElMessageBox} from "element-plus";
  164. import RouterViewCom from '@/layout/router-view.vue'
  165. import ElementCom from './tools/element.vue'
  166. import SelectCom from './tools/select.vue'
  167. import AnalysisCom from './tools/analysis.vue'
  168. import ToolCom from './tools/tool.vue'
  169. import PositionCom from './tools/position.vue'
  170. import BaseCom from './tools/base.vue'
  171. import ExampleCom from './tools/example.vue'
  172. import * as source from "ol/source";
  173. import * as layer from "ol/layer";
  174. import * as style from "ol/style";
  175. import * as format from "ol/format";
  176. import QyStyle from '../map-info/style/qy'
  177. import SbStyle from '../map-info/style/sb'
  178. import CommonStyle from '../map-info/style/common'
  179. import * as ol from "ol";
  180. import { v4 } from "uuid";
  181. import * as proj from "ol/proj";
  182. import * as turf from '@turf/turf'
  183. import * as geom from 'ol/geom';
  184. import VideoPlayKedaCom from "@/views/gis/business/common/VideoPlayKeda.vue";
  185. import V2Table from "./v2-table.vue";
  186. import {deviceQuery, enterpriseQuery} from "@/api/modules/enterprise";
  187. import {clearLocationDom} from '@/components/easyMap/func/location'
  188. import {clearMeasureDom} from '@/components/easyMap/func/measure'
  189. export default defineComponent({
  190. name: '',
  191. components: {
  192. RouterViewCom,
  193. ElementCom,
  194. SelectCom,
  195. AnalysisCom,
  196. ToolCom,
  197. PositionCom,
  198. BaseCom,
  199. ExampleCom,
  200. VideoPlayKedaCom,
  201. V2Table
  202. },
  203. props: {},
  204. setup(props, {emit}) {
  205. const store = useStore();
  206. const router = useRouter();
  207. const route = useRoute();
  208. const that = (getCurrentInstance() as ComponentInternalInstance).appContext.config.globalProperties
  209. const state = reactive({
  210. map: <any>null,
  211. mapFunc: <any>null,
  212. searchInput: '',
  213. toolsType: '',
  214. toolsCom: null,
  215. toolsParams: {
  216. element: null,
  217. position: null
  218. },
  219. qyParams: {
  220. layer: <any>null,
  221. source: <any>null,
  222. tempFeature: <any>null,
  223. overlay: <any>null,
  224. qyInfo: <any>{},
  225. analysisLayer: <any>null,
  226. analysisSource: <any>null,
  227. analysisCircle: <any>null,
  228. tempSbFeature: <any>null,
  229. sbOverlay: <any>null,
  230. sbInfo: <any>{
  231. name: '505县道新安村路口1-枪机-0110580',
  232. code: '46044123124125125',
  233. },
  234. },
  235. videoLayout: {
  236. width: 640,
  237. height: 366,
  238. left: 1200,
  239. top: 460
  240. },
  241. showVideo: false
  242. })
  243. const ToolsMapper = [
  244. {label: '图层', value: 'element', com: ElementCom, disabled: true},
  245. {label: '框选', value: 'select', com: SelectCom, disabled: true},
  246. {label: '周边分析', value: 'analysis', com: AnalysisCom, disabled: true},
  247. {label: '清空图层', value: 'clear', com: undefined},
  248. {label: '工具', value: 'tool', com: ToolCom},
  249. {label: '定位', value: 'position', com: PositionCom},
  250. {label: '底图', value: 'base', com: BaseCom},
  251. {label: '图例', value: 'example', com: ExampleCom},
  252. ]
  253. const ref_search = ref()
  254. const mapLoad = (map, func) => {
  255. state.map = map
  256. store.dispatch('gis/LOAD_GIS_MAP', state.map)
  257. state.mapFunc = func
  258. state.map.on('singleclick', e => {
  259. let flag = false
  260. map.forEachFeatureAtPixel(e.pixel, (f) => {
  261. if (!flag && f.get('featureType')) {
  262. flag = true
  263. if (f.get('featureType') === 'qy') { // 企业
  264. // 恢复上一个要素的样式
  265. if (f.getId() !== state.qyParams.tempFeature?.getId()) {
  266. state.qyParams.tempFeature?.get('resetStyle')?.()
  267. state.qyParams.analysisSource?.clear()
  268. // 新的要素
  269. if (f.get('isAnalysis')) {
  270. f.setStyle(f.get('analysisActiveStyle'))
  271. } else {
  272. f.setStyle(f.get('activeStyle'))
  273. }
  274. state.qyParams.qyInfo = JSON.parse(JSON.stringify(f.get('info')))
  275. store.dispatch('gis/LOAD_ACTIVE_QY_ID', f.getId())
  276. // 备份新的要素
  277. state.qyParams.tempFeature = f
  278. }
  279. state.qyParams.overlay.setPosition(f.getGeometry().getCoordinates())
  280. } else if (f.get('featureType') === 'sb') { // 设备
  281. // 恢复上一个要素的样式
  282. if (f.getId() !== state.qyParams.tempSbFeature?.getId()) {
  283. state.qyParams.tempSbFeature?.get('resetStyle')?.()
  284. // 新的要素
  285. if (f.get('isAnalysis')) {
  286. f.setStyle(f.get('analysisActiveStyle'))
  287. } else {
  288. f.setStyle(f.get('activeStyle'))
  289. }
  290. state.qyParams.sbInfo = JSON.parse(JSON.stringify(f.get('info')))
  291. state.qyParams.sbOverlay.setPosition(f.getGeometry().getCoordinates())
  292. // 备份新的要素
  293. state.qyParams.tempSbFeature = f
  294. }
  295. state.qyParams.overlay.setPosition(undefined)
  296. }
  297. }
  298. }, {
  299. hitTolerance: 0,
  300. });
  301. })
  302. initQYLayer()
  303. }
  304. const menuCpt = computed(() => {
  305. return router.options.routes.filter(v => v.name === store.state.gis.menuRootName)[0].children?.filter(v => !v.meta.noMenu)
  306. })
  307. const searchHandleMapSearch = (queryString: string, cb: (arg: any) => void) => {
  308. if (queryString.trim()) {
  309. const arr = [
  310. {name: 123},
  311. {name: 123},
  312. {name: 123},
  313. {name: 123},
  314. {name: 123},
  315. {name: 123},
  316. ]
  317. cb(arr)
  318. } else {
  319. cb([])
  320. }
  321. }
  322. const searchToMapLocation = (val) => {
  323. state.searchInput = val.name
  324. // if (val.source === 'element') {
  325. // state.elementFilter.forEach(l => {
  326. // l.list.forEach(v => {
  327. // if (v.value === val.elementType) {
  328. // v.active = true
  329. // }
  330. // })
  331. // })
  332. // }
  333. // positionSwitchGeom(val.geomType, val.coordinates, true, val.source)
  334. }
  335. const toolsHandleClick = (item) => {
  336. if (item.value === 'clear') {
  337. // 工具-标绘'layerName', 'toolDrawViewsLayer'
  338. state.map.getLayers().getArray().filter(v => v.get('layerName') === 'toolDrawViewsLayer')?.[0]?.getSource()?.clear()
  339. // 工具-测量'layerName', 'measureLayer'
  340. state.map.getLayers().getArray().filter(v => v.get('layerName') === 'measureLayer')?.[0]?.getSource()?.clear()
  341. clearMeasureDom(state.map)
  342. // 定位'layerName', 'positionLayer'
  343. state.map.getLayers().getArray().filter(v => v.get('layerName') === 'positionLayer')?.[0]?.getSource()?.clear()
  344. clearLocationDom()
  345. } else {
  346. state.toolsCom = markRaw(item.com)
  347. state.toolsType = (state.toolsType === item.value ? '' : item.value)
  348. }
  349. }
  350. const ref_qyDom = ref()
  351. const ref_sbDom = ref()
  352. const initQYLayer = () => {
  353. enterpriseQuery({
  354. "pageNumber":1,
  355. "pageSize":200,
  356. // "entName":null,
  357. // "entType":"加工增值"
  358. }).then((res: any) => {
  359. if (res.resp_code === 0 && res.datas?.length > 0) {
  360. const tempStatistic = {
  361. total: res.datas.length,
  362. lgszyjkscsb: 0,
  363. jgzzmgs: 0,
  364. lgsjkyfl: 0
  365. }
  366. const features: any = []
  367. res.datas.forEach(v => {
  368. try {
  369. const feat: any = new format.WKT().readFeature(`POINT(${v.longitude} ${v.latitude})`)
  370. let type = ''
  371. if (v.qykx === '零关税自用进口生产设备') {
  372. tempStatistic.lgszyjkscsb += 1
  373. type = 'lgszyjkscsb'
  374. } else if (v.qykx === '加工增值免关税') {
  375. tempStatistic.jgzzmgs += 1
  376. type = 'jgzzmgs'
  377. } else if (v.qykx === '零关税进口原辅料') {
  378. tempStatistic.lgsjkyfl += 1
  379. type = 'lgsjkyfl'
  380. }
  381. feat.set('defaultStyle', QyStyle.qyStyle(type))
  382. feat.set('activeStyle', [...CommonStyle.activeStyle(), ...QyStyle.qyStyle(type)])
  383. feat.set('analysisStyle', [...CommonStyle.analysisStyle(), ...QyStyle.qyStyle(type)])
  384. feat.set('analysisActiveStyle', [...CommonStyle.activeStyle(), ...CommonStyle.analysisStyle(), ...QyStyle.qyStyle(type)])
  385. feat.set('isAnalysis', false)
  386. feat.set('resetStyle', () => {
  387. if (feat.get('isAnalysis')) {
  388. feat.setStyle(feat.get('analysisStyle'))
  389. } else {
  390. feat.setStyle(feat.get('defaultStyle'))
  391. }
  392. })
  393. const obj = {
  394. coordinates: feat.getGeometry().getCoordinates(),
  395. tab: '1',
  396. name: v.qymc,
  397. 1: {
  398. name: v.qymc,
  399. people: v.lerep,
  400. number: v.uniscid,
  401. address: v.dom
  402. },
  403. 2: {},
  404. 3: {},
  405. 4: {},
  406. 5: {
  407. radius: 10,
  408. center: feat.getGeometry().getCoordinates(),
  409. tableData: [],
  410. loading: false
  411. },
  412. }
  413. feat.set('info', obj)
  414. feat.set('featureType', 'qy')
  415. feat.setStyle(feat.get('defaultStyle'))
  416. feat.setId(v.id)
  417. feat.set('mockClick', () => {
  418. state.qyParams.tempFeature?.get('resetStyle')?.()
  419. state.qyParams.analysisSource?.clear()
  420. // 新的要素
  421. if (feat.get('isAnalysis')) {
  422. feat.setStyle(feat.get('analysisActiveStyle'))
  423. } else {
  424. feat.setStyle(feat.get('activeStyle'))
  425. }
  426. state.qyParams.qyInfo = JSON.parse(JSON.stringify(feat.get('info')))
  427. // 备份新的要素
  428. state.qyParams.tempFeature = feat
  429. state.qyParams.overlay.setPosition(feat.getGeometry().getCoordinates())
  430. store.dispatch('gis/LOAD_ACTIVE_QY_ID', v.id)
  431. })
  432. features.push(feat)
  433. } catch (e) {
  434. console.error('异常企业:', v)
  435. }
  436. })
  437. store.dispatch('gis/LOAD_tempStatistic', tempStatistic)
  438. state.qyParams.source = new source.Vector({
  439. features: features,
  440. wrapX: false
  441. })
  442. state.qyParams.layer = new layer.VectorImage({
  443. source: state.qyParams.source,
  444. zIndex: 10,
  445. layerName: 'qy'
  446. })
  447. state.map.addLayer(state.qyParams.layer)
  448. // 详情
  449. state.qyParams.overlay = new ol.Overlay({
  450. id: v4(),
  451. element: ref_qyDom.value,
  452. autoPan: false,
  453. offset: [0, -60],
  454. positioning: 'bottom-center',
  455. stopEvent: true,
  456. autoPanAnimation: {
  457. duration: 250
  458. }
  459. })
  460. state.map.addOverlay(state.qyParams.overlay)
  461. // 详情
  462. state.qyParams.sbOverlay = new ol.Overlay({
  463. id: v4(),
  464. element: ref_sbDom.value,
  465. autoPan: false,
  466. offset: [0, -60],
  467. positioning: 'bottom-center',
  468. stopEvent: true,
  469. autoPanAnimation: {
  470. duration: 250
  471. }
  472. })
  473. state.map.addOverlay(state.qyParams.sbOverlay)
  474. }
  475. })
  476. }
  477. const onCloseQy = () => {
  478. state.qyParams.overlay.setPosition(undefined)
  479. state.qyParams.tempFeature.get('resetStyle')()
  480. state.qyParams.tempFeature = null
  481. state.qyParams.qyInfo = {}
  482. state.qyParams.analysisSource?.clear()
  483. store.dispatch('gis/LOAD_ACTIVE_QY_ID', null)
  484. }
  485. const onCloseSb = () => {
  486. state.qyParams.sbOverlay.setPosition(undefined)
  487. state.qyParams.tempSbFeature.get('resetStyle')()
  488. state.qyParams.tempSbFeature = null
  489. }
  490. const handleRangeBlur = () => {
  491. if (!state.qyParams.qyInfo['5'].radius) {
  492. state.qyParams.qyInfo['5'].radius = 10
  493. }
  494. setCircle()
  495. }
  496. const setCircle = () => {
  497. const transformProjection = (arr, EPSG, EPSG2) => {
  498. try {
  499. if (EPSG2 && EPSG) {
  500. if (arr && arr.length === 4) {
  501. return proj.transformExtent(arr, EPSG, EPSG2);
  502. } else {
  503. return proj.transform(arr, EPSG, EPSG2);
  504. }
  505. }
  506. return undefined;
  507. } catch (e) {
  508. console.error(e);
  509. }
  510. }
  511. state.qyParams.analysisCircle.getGeometry().setRadius(transformProjection([state.qyParams.qyInfo['5'].radius * 1000, 0], 'EPSG:3857', 'EPSG:4326')[0] - transformProjection([0, 0], 'EPSG:3857', 'EPSG:4326')[0],'XY')
  512. }
  513. const onRadiusSubmit = () => {
  514. if (!state.qyParams.analysisLayer) {
  515. state.qyParams.analysisSource = new source.Vector()
  516. state.qyParams.analysisLayer = new layer.Vector({
  517. zIndex: 9,
  518. source: state.qyParams.analysisSource,
  519. style: [
  520. new style.Style({
  521. stroke: new style.Stroke({
  522. color: '#2860F1',
  523. width: 2,
  524. lineDash: [10, 10]
  525. }),
  526. fill: new style.Fill({
  527. color: 'rgba(20, 129, 241, 0.3)',
  528. }),
  529. })
  530. ]
  531. });
  532. state.qyParams.analysisCircle = new ol.Feature({
  533. geometry: new geom.Circle(state.qyParams.qyInfo.coordinates, 0),
  534. })
  535. state.qyParams.analysisSource.addFeature(state.qyParams.analysisCircle)
  536. setCircle()
  537. state.map.addLayer(state.qyParams.analysisLayer)
  538. }
  539. state.qyParams.analysisSource.clear()
  540. state.qyParams.analysisSource.addFeature(state.qyParams.analysisCircle)
  541. state.qyParams.qyInfo['5'].tableData = []
  542. state.qyParams.qyInfo['5'].loading = true
  543. that.$api.deviceQuery({
  544. lon: state.qyParams.qyInfo.coordinates[0],
  545. lat: state.qyParams.qyInfo.coordinates[1],
  546. radius: state.qyParams.qyInfo['5'].radius
  547. }).then((res: any) => {
  548. // console.log(res)
  549. // for (let i = 0; i < 500; i++) {
  550. // state.qyParams.qyInfo['5'].tableData.push({
  551. // name: '505县道新安村路口1-枪机-0110580_' + i,
  552. // code: '46044123124125125',
  553. // status: i % 3 === 0 ? '1' : '0',
  554. // typeName: '公安类',
  555. // type: i % 3 === 0 ? 'galsb' : (i % 3 === 1 ? 'shlsb' : 'mylsb'),
  556. // wkt: `POINT(${that.$util.randomNum(108.738329, 110.912130, 6)} ${that.$util.randomNum(18.154784, 20, 6)})`
  557. // })
  558. // }
  559. if (res.resp_code === 0 && res.datas?.length > 0) {
  560. console.log(res.datas?.length)
  561. const features: any = []
  562. res.datas?.forEach(v => {
  563. try {
  564. const feat: any = new format.WKT().readFeature(`POINT(${v.longitude} ${v.latitude})`)
  565. let type = ''
  566. if (v.type = '公安类') {
  567. type = 'galsb'
  568. } else if (v.type = '社会类') {
  569. type = 'shlsb'
  570. } else if (v.type = '民用类') {
  571. type = 'mylsb'
  572. }
  573. feat.set('defaultStyle', SbStyle.sbStyle(type))
  574. feat.set('activeStyle', [...CommonStyle.activeStyle(), ...SbStyle.sbStyle(type)])
  575. feat.set('analysisStyle', [...CommonStyle.analysisStyle(), ...SbStyle.sbStyle(type)])
  576. feat.set('analysisActiveStyle', [...CommonStyle.activeStyle(), ...CommonStyle.analysisStyle(), ...SbStyle.sbStyle(type)])
  577. feat.set('isAnalysis', true)
  578. feat.set('resetStyle', () => {
  579. if (feat.get('isAnalysis')) {
  580. feat.setStyle(feat.get('analysisStyle'))
  581. } else {
  582. feat.setStyle(feat.get('defaultStyle'))
  583. }
  584. })
  585. feat.setStyle(feat.get('analysisStyle'))
  586. feat.set('featureType', 'sb')
  587. feat.set('info', v)
  588. feat.setId(v.deviceid)
  589. features.push(feat)
  590. state.qyParams.qyInfo['5'].tableData.push(v)
  591. } catch (e) {
  592. console.error('异常设备', v)
  593. }
  594. })
  595. state.qyParams.analysisSource.addFeatures(features)
  596. }
  597. state.qyParams.qyInfo['5'].loading = false
  598. }).catch(() => {
  599. state.qyParams.qyInfo['5'].loading = true
  600. })
  601. }
  602. const onRadiusReset = () => {
  603. state.qyParams.qyInfo['5'].radius = 10
  604. onRadiusSubmit()
  605. }
  606. onMounted(() => {
  607. if (!sessionStorage.getItem('auth')) {
  608. if(navigator.userAgent.indexOf("Firefox") != -1 || navigator.userAgent.indexOf("Chrome") != -1){
  609. window.location.href = "about:blank";
  610. window.close();
  611. }else{
  612. window.opener = null;
  613. window.open("", "_self");
  614. window.close();
  615. }
  616. }
  617. })
  618. return {
  619. ...toRefs(state),
  620. mapLoad,
  621. ToolsMapper,
  622. ref_search,
  623. menuCpt,
  624. searchHandleMapSearch,
  625. searchToMapLocation,
  626. toolsHandleClick,
  627. ref_qyDom,
  628. onCloseQy,
  629. onCloseSb,
  630. handleRangeBlur,
  631. setCircle,
  632. onRadiusSubmit,
  633. onRadiusReset,
  634. ref_sbDom
  635. }
  636. },
  637. })
  638. </script>
  639. <style scoped lang="scss">
  640. .gis-layout {
  641. width: 100%;
  642. height: 100vh;
  643. position: absolute;
  644. top: 0;
  645. left: 0;
  646. z-index: 2;
  647. display: flex;
  648. align-items: center;
  649. justify-content: center;
  650. .map {
  651. z-index: 1;
  652. }
  653. .gis-menu {
  654. $diff: 14px;
  655. $menuBottom: 16px;
  656. position: absolute;
  657. z-index: 2;
  658. bottom: $menuBottom;
  659. display: flex;
  660. justify-content: center;
  661. align-items: center;
  662. &:before, &:after {
  663. content: '';
  664. background-image: url("@/assets/images/gis-layout/gis-layout-menu_side.png");
  665. width: 246px;
  666. height: 39px;
  667. position: absolute;
  668. bottom: -$menuBottom;
  669. }
  670. &:before {
  671. left: calc((#{$diff} + 246px) * -1);
  672. }
  673. &:after {
  674. right: calc((#{$diff} + 246px) * -1);
  675. transform: rotateY(180deg);
  676. }
  677. .gis-menu-bottom {
  678. position: absolute;
  679. height: 9px;
  680. width: calc(100% + 10px + #{$diff} * 2);
  681. bottom: -$menuBottom;
  682. }
  683. .gis-menu-item {
  684. width: 95px;
  685. height: 36px;
  686. background-image: url("@/assets/images/gis-layout/gis-layout-menu_item-min.png");
  687. background-size: 100% 100%;
  688. background-repeat: no-repeat;
  689. display: flex;
  690. align-items: center;
  691. justify-content: center;
  692. font-size: 18px;
  693. font-family: YouSheBiaoTiHei;
  694. font-weight: 400;
  695. color: #60AEFF;
  696. &:not(&:last-child) {
  697. margin-right: 4px;
  698. }
  699. &.active {
  700. color: #FFFFFF;
  701. background-image: url("@/assets/images/gis-layout/gis-layout-menu_item-min_active.png");
  702. }
  703. &.max {
  704. width: 125px;
  705. background-image: url("@/assets/images/gis-layout/gis-layout-menu_item-max.png");
  706. &.active {
  707. background-image: url("@/assets/images/gis-layout/gis-layout-menu_item-max_active.png");
  708. }
  709. }
  710. }
  711. }
  712. .gis-tools {
  713. position: absolute;
  714. z-index: 3;
  715. top: 10px;
  716. right: 10px;
  717. width: 404px;
  718. >div {
  719. box-sizing: border-box;
  720. }
  721. .gt-search {
  722. height: 38px;
  723. width: 100%;
  724. display: flex;
  725. align-items: center;
  726. .search-icon {
  727. width: 54px;
  728. height: 100%;
  729. background-color: #3e8ef7;
  730. display: flex;
  731. align-items: center;
  732. justify-content: center;
  733. }
  734. :deep(.el-autocomplete) {
  735. flex: 1;
  736. height: 100%;
  737. .el-input {
  738. width: 100%;
  739. height: 100%;
  740. .el-input__wrapper {
  741. border-radius: 0;
  742. }
  743. }
  744. }
  745. :deep(.el-autocomplete__popper) {
  746. opacity: 0.9;
  747. background-image: linear-gradient(45deg,transparent 10px,#101D69 10px), linear-gradient(-135deg,transparent 10px,#101D69 10px);
  748. border: none;
  749. .el-popper__arrow::before {
  750. background: #101D69;
  751. border-color: #101D69;
  752. }
  753. .el-autocomplete-suggestion {
  754. .el-scrollbar {
  755. .el-autocomplete-suggestion__wrap {
  756. .el-scrollbar__view {
  757. >li {
  758. &:hover {
  759. background-color: rgba(59,84,172,0.5);
  760. }
  761. .search-item {
  762. display: flex;
  763. align-items: center;
  764. justify-content: flex-start;
  765. >img {
  766. margin-right: 6px;
  767. width: 16px;
  768. height: 16px;
  769. }
  770. .name {
  771. color: #ffffff;
  772. }
  773. }
  774. }
  775. }
  776. }
  777. }
  778. }
  779. }
  780. }
  781. .gt-tools {
  782. width: 100%;
  783. height: 38px;
  784. margin-top: 2px;
  785. background-color: #FFFFFF;
  786. border: 1px solid #C7CFDE;
  787. display: flex;
  788. align-items: center;
  789. justify-content: space-around;
  790. padding: 0 8px;
  791. .tools-item {
  792. font-size: 14px;
  793. font-family: Microsoft YaHei;
  794. font-weight: 400;
  795. color: #AEAEAE;
  796. &.active {
  797. color: #1174DB;
  798. font-weight: bold;
  799. }
  800. &.disabled {
  801. opacity: 0.5;
  802. cursor: not-allowed;
  803. }
  804. }
  805. .tools-line {
  806. width: 1px;
  807. height: 16px;
  808. background: linear-gradient(0deg, rgba(174,174,174,0) 0%, rgba(213,213,213,0.99) 50%, rgba(174,174,174,0) 100%);
  809. }
  810. }
  811. .gt-tools-component {
  812. width: 100%;
  813. margin-top: 2px;
  814. }
  815. }
  816. .gis-content {
  817. z-index: 4;
  818. }
  819. .qy-info {
  820. $bgColor: #2860CE;
  821. $footH: 10px;
  822. min-width: 315px;
  823. min-height: 168px - $footH;
  824. background-color: $bgColor;
  825. position: relative;
  826. display: flex;
  827. justify-content: center;
  828. &:after {
  829. content: '';
  830. position: absolute;
  831. bottom: -$footH;
  832. border-top: $footH solid $bgColor;
  833. border-left: $footH solid transparent;
  834. border-right: $footH solid transparent;
  835. }
  836. .qy-bg-icon {
  837. width: 16px;
  838. height: 16px;
  839. background-image: url("@/views/gis/map-info/qy-dom-icon1.png");
  840. background-repeat: no-repeat;
  841. background-size: 100% 100%;
  842. position: absolute;
  843. z-index: 1;
  844. &.qy-bg-icon-1 {
  845. left: 0;
  846. top: 0;
  847. }
  848. &.qy-bg-icon-2 {
  849. right: 0;
  850. top: 0;
  851. transform: rotate(90deg);
  852. }
  853. &.qy-bg-icon-3 {
  854. right: 0;
  855. bottom: 0;
  856. transform: rotate(180deg);
  857. }
  858. &.qy-bg-icon-4 {
  859. left: 0;
  860. bottom: 0;
  861. transform: rotate(270deg);
  862. }
  863. }
  864. .qy-main {
  865. width: 100%;
  866. height: auto;
  867. z-index: 2;
  868. padding: 12px;
  869. .qy-main-head {
  870. display: flex;
  871. align-items: center;
  872. position: relative;
  873. &:after {
  874. content: '';
  875. position: absolute;
  876. width: 100%;
  877. height: 4px;
  878. bottom: -9px;
  879. background-image: url("@/views/gis/map-info/qy-info-icon2.png");
  880. background-repeat: no-repeat;
  881. }
  882. .qy-main-head-tips {
  883. font-size: 10px;
  884. font-family: Microsoft YaHei;
  885. font-weight: bold;
  886. color: #8FFFFF;
  887. }
  888. .qy-main-head-name {
  889. font-size: 14px;
  890. font-family: Microsoft YaHei;
  891. font-weight: bold;
  892. color: #FFFFFF;
  893. }
  894. .svg-icon {
  895. margin-left: auto;
  896. }
  897. }
  898. .qy-main-tab {
  899. margin: 20px 0 10px 0;
  900. display: flex;
  901. align-items: center;
  902. .qy-main-tab-item {
  903. font-size: 12px;
  904. font-family: Microsoft YaHei;
  905. font-weight: 400;
  906. color: #CEE6FF;
  907. height: 16px;
  908. display: flex;
  909. align-items: center;
  910. justify-content: center;
  911. padding: 0 4px;
  912. border-radius: 2px;
  913. background-color: rgba(255, 255, 255, 0.2);
  914. &.active {
  915. background-color: #1280F1;
  916. color: #61FFFF;
  917. }
  918. &.disabled {
  919. cursor: not-allowed;
  920. opacity: 0.7;
  921. }
  922. &:not(:last-child) {
  923. margin-right: 2px;
  924. }
  925. }
  926. }
  927. .qy-main-content-1 {
  928. font-size: 12px;
  929. font-family: Microsoft YaHei;
  930. font-weight: 400;
  931. color: #FFFFFF;
  932. >div {
  933. line-height: 18px;
  934. }
  935. }
  936. .qy-main-content-5 {
  937. .qy-main-content-5-radius {
  938. display: flex;
  939. align-items: center;
  940. font-size: 12px;
  941. font-family: Microsoft YaHei;
  942. font-weight: 400;
  943. color: #FFFFFF;
  944. .radius-min, .radius-max {
  945. margin: 0 3px;
  946. &:hover {
  947. color: #409EFF;
  948. }
  949. }
  950. :deep(.cus-form-column) {
  951. max-width: unset;
  952. width: 44px;
  953. flex: unset;
  954. .el-form-item {
  955. margin: 0;
  956. .el-form-item__content {
  957. height: 18px;
  958. .el-input {
  959. .el-input__wrapper {
  960. padding: 0;
  961. border-radius: 5px;
  962. background-color: transparent;
  963. .el-input__inner {
  964. color: #FFFFFF;
  965. background-color: transparent;
  966. height: 100%;
  967. text-align: center;
  968. border: 1px solid #ffffff;
  969. border-radius: 5px;
  970. &::placeholder {
  971. font-size: 12px;
  972. font-family: Microsoft YaHei;
  973. }
  974. }
  975. }
  976. }
  977. }
  978. }
  979. }
  980. .submit {
  981. display: flex;
  982. align-items: center;
  983. justify-content: center;
  984. width: 36px;
  985. height: 18px;
  986. background: #8FFFFF;
  987. border-radius: 4px;
  988. font-size: 12px;
  989. font-family: Microsoft YaHei;
  990. font-weight: 400;
  991. color: #1174DB;
  992. margin-left: 10px;
  993. }
  994. .reset {
  995. display: flex;
  996. align-items: center;
  997. justify-content: center;
  998. width: 36px;
  999. height: 18px;
  1000. border: 1px solid #8FFFFF;
  1001. border-radius: 4px;
  1002. font-size: 12px;
  1003. font-family: Microsoft YaHei;
  1004. font-weight: 400;
  1005. color: #8FFFFF;
  1006. margin-left: 8px;
  1007. }
  1008. }
  1009. .qy-main-content-5-table {
  1010. margin-top: 10px;
  1011. width: 440px;
  1012. :deep(.__gis-overlay_table-v2) {
  1013. .el-table-v2__header-cell{
  1014. justify-content: center;
  1015. &:nth-child(2) {
  1016. color: #00C6FC;
  1017. }
  1018. }
  1019. .el-table-v2__header-cell, .el-table-v2__row-cell {
  1020. width: 60px !important;
  1021. $w1: 60px;
  1022. $w3: 60px;
  1023. $w4: 60px;
  1024. $w5: 60px;
  1025. &:nth-child(1) {
  1026. width: $w1 !important;
  1027. }
  1028. &:nth-child(2) {
  1029. width: calc(100% - #{$w1} - #{$w3} - #{$w4} - #{$w5} - 6px) !important;
  1030. }
  1031. &:nth-child(3) {
  1032. width: $w3 !important;
  1033. }
  1034. &:nth-child(4) {
  1035. width: $w4 !important;
  1036. }
  1037. &:nth-child(5) {
  1038. width: $w5 !important;
  1039. }
  1040. }
  1041. }
  1042. }
  1043. }
  1044. }
  1045. }
  1046. .sb-info {
  1047. $footH: 10px;
  1048. width: 220px;
  1049. background: linear-gradient(180deg, #3874C9 0%, #0043C4 100%);
  1050. border-radius: 0px 4px 4px 4px;
  1051. position: relative;
  1052. display: flex;
  1053. justify-content: center;
  1054. &:after {
  1055. content: '';
  1056. position: absolute;
  1057. bottom: -$footH;
  1058. border-top: $footH solid #0043C4;
  1059. border-left: $footH solid transparent;
  1060. border-right: $footH solid transparent;
  1061. }
  1062. .sb-info-head {
  1063. min-width: 68px;
  1064. height: 18px;
  1065. position: absolute;
  1066. top: -18px;
  1067. left: 0;
  1068. font-size: 12px;
  1069. font-family: PingFang SC;
  1070. font-weight: 500;
  1071. color: #FFFFFF;
  1072. display: flex;
  1073. align-items: center;
  1074. line-height: 8px;
  1075. &:before {
  1076. z-index: -1;
  1077. content: '';
  1078. position: absolute;
  1079. width: 100%;
  1080. height: 100%;
  1081. background: linear-gradient(180deg, #3874C9 0%, #0043C4 100%);
  1082. border-radius: 2px 2px 0 0;/* 设置圆角 */
  1083. transform: perspective(20px)rotateX(4deg);
  1084. /* 镜头距离元素表面的位置为8px,x轴为1.1倍y轴为1.3倍,绕x轴旋转5度 */
  1085. transform-origin: bottom left;
  1086. /* bottom left = left bottom = 0 100% 中心点偏移量*/
  1087. }
  1088. .svg-icon {
  1089. margin: 0 4px 0 6px;
  1090. }
  1091. }
  1092. .sb-info-close {
  1093. position: absolute;
  1094. right: 0;
  1095. top: -16px;
  1096. }
  1097. .sb-main {
  1098. width: 100%;
  1099. height: auto;
  1100. padding: 10px;
  1101. .sb-item {
  1102. display: flex;
  1103. .sb-item-label {
  1104. width: 42px;
  1105. font-size: 14px;
  1106. font-family: PingFang SC;
  1107. font-weight: 600;
  1108. color: #08FFFF;
  1109. line-height: 20px;
  1110. }
  1111. .sb-item-value {
  1112. flex: 1;
  1113. font-size: 14px;
  1114. font-family: PingFang SC;
  1115. font-weight: 400;
  1116. color: #FFFFFF;
  1117. line-height: 20px;
  1118. }
  1119. }
  1120. .play-button {
  1121. width: 76px;
  1122. height: 24px;
  1123. background: #1280F1;
  1124. border-radius: 2px;
  1125. border: 1px solid #4BA0FF;
  1126. display: flex;
  1127. align-items: center;
  1128. justify-content: center;
  1129. font-size: 14px;
  1130. font-family: PingFang SC;
  1131. font-weight: 400;
  1132. color: #FFFFFF;
  1133. margin-left: calc((100% - 76px) / 2);
  1134. margin-top: 10px;
  1135. }
  1136. }
  1137. }
  1138. }
  1139. </style>