location.vue 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866
  1. <template>
  2. <div>
  3. <div id="route-map">
  4. <span class="icon-base fence-showoff" :class="{buttonActive: showFence}" @click="toggleShowFence">
  5. <yd-icon name="dianziweilan" size=".4rem" custom></yd-icon>
  6. </span>
  7. <span class="refresh-button" @click="refreshPosAndRoute">
  8. <yd-icon name="refresh" size="14px"></yd-icon>
  9. </span>
  10. <div class="setting-button">
  11. <van-popover v-model="showSettings" trigger="click" placement="left">
  12. <div style="padding: 6px;">
  13. <van-checkbox v-model="isOnlyFenceDevice" @change="changeSetting('isOnlyFenceDevice',$event)" shape="square" icon-size="14">只显示选中围栏内设备</van-checkbox>
  14. <!-- <van-checkbox v-model="isClusterMarker" @change="changeSetting('isClusterMarker',$event)" shape="square" icon-size="14">聚合图标</van-checkbox> -->
  15. </div>
  16. <template #reference>
  17. <van-icon size="18" name="setting" />
  18. </template>
  19. </van-popover>
  20. </div>
  21. </div>
  22. <van-popup v-model="panelShow" round :overlay="false" position="bottom"
  23. style="box-shadow: 0 2px 4px rgba(0, 0, 0, .12), 0 0 6px rgba(0, 0, 0, .04);margin-bottom:1rem;">
  24. <div class="info-panel">
  25. <div class="panel-first-row">
  26. <div style="font-size:14px;font-weight:550;color:gray">{{selectDevice.date_time}}({{selectDevice.awayTime}}前)</div>
  27. <van-tag plain :type="selectDevice.alarm_state == 0 ? 'primary' : 'danger'">{{selectDevice.alarm_state == 0 ? '安全' : '告警'}}</van-tag>
  28. </div>
  29. <div class="panel-second-row">
  30. <div class="second-row-left">
  31. <div class="row-left-layout">
  32. <span style="font-size: 10px;font-weight: 600;">{{selectDevice.loc_mode}}</span>
  33. <span style="color:darkgray">设备型号</span>
  34. </div>
  35. <div class="row-left-layout">
  36. <span style="font-size:10px;font-weight:600">{{selectDevice.state}} </span>
  37. <span style="color:darkgray">设备状态</span>
  38. </div>
  39. </div>
  40. <div class="row-right-layout">
  41. <!-- <van-image
  42. width="1.08rem"
  43. height="0.6rem"
  44. fit="cover"
  45. src="https://rlfd.oss-cn-hangzhou.aliyuncs.com/wxt_school/student_card_product.png"
  46. @click.native="showPreview"
  47. /> -->
  48. <div class="right-number" >
  49. <p>名称:{{selectDevice.device_name}}</p>
  50. <p>出厂编号:{{selectDevice.factory_number}}</p>
  51. </div>
  52. </div>
  53. </div>
  54. <div class="panel-plain-text">
  55. {{selectDevice.address}}
  56. </div>
  57. <div class="route-search-button" style="position:relative">
  58. <router-link :to="'yys_route_search/'+ selectDevice.factory_number" class="voice-device-item">
  59. <van-button type="info" icon="search" size="small" round style="width:3rem" >历史定位</van-button>
  60. </router-link>
  61. <van-button @click.native="refreshSingle" plain type="info" icon="replay" size="mini" style="font-size:12px;position:absolute;right:0.5rem;top:0.1rem"></van-button>
  62. </div>
  63. </div>
  64. </van-popup>
  65. </div>
  66. </template>
  67. <script type="text/babel">
  68. import Vue from 'vue'
  69. import { Popup, Image as VanImage,Button,Icon,Tag, ImagePreview, Checkbox, Popover } from 'vant';
  70. Vue.use(VanImage);
  71. Vue.use(Popup);
  72. Vue.use(Button);
  73. Vue.use(Icon);
  74. Vue.use(Tag);
  75. Vue.use(Checkbox);
  76. Vue.use(Popover);
  77. export default {
  78. filters: {
  79. upperCase(val) {
  80. if (val) {
  81. return val.toUpperCase();
  82. }
  83. }
  84. },
  85. mounted(){
  86. let that = this;
  87. let map = this.initMap();
  88. map.then((result) => {
  89. if(result === false){
  90. this.$toast('地图初始化失败,请刷新重试');
  91. return;
  92. }
  93. that.getLastPositon();
  94. }).catch((err) => {
  95. this.$toast('地图数据初始化出错,请刷新重试');
  96. console.log(err);
  97. return;
  98. });
  99. // that.getPositioning();
  100. },
  101. data(){
  102. return {
  103. isClusterMarker: false, // 是否聚合
  104. isOnlyFenceDevice: false, // 是否仅显示围栏内设备
  105. showSettings: false, // 显示设置项
  106. checkedFenceObject: null, // 选中的围栏对象
  107. cluster: null, // 聚合对象
  108. userid: '',
  109. openid: '',
  110. map: null,
  111. geocoder: null,
  112. positionInfo: '',
  113. parsedAddress: '',
  114. selectDevice: '',
  115. selectImei:'',
  116. markersArr: [],
  117. fenceOverLays: [],
  118. panelShow: false, // 显示设备信息栏
  119. showFence: false, // 显示围栏
  120. normalColor: 'normalColor',
  121. alarmColor: 'alarmColor'
  122. }
  123. },
  124. created() {
  125. this.openid = localStorage.getItem('openid');
  126. this.userid = localStorage.getItem('yysUserid');
  127. this.isOnlyFenceDevice = localStorage.getItem('isOnlyFenceDevice');
  128. // this.isClusterMarker = localStorage.getItem('clusterMarker');
  129. },
  130. destroyed() {
  131. this.infoWindow && this.infoWindow.remove();
  132. this.infoWindow = null;
  133. this.markersArr && this.markersArr.forEach(item => {
  134. item.remove();
  135. })
  136. this.markersArr = null;
  137. this.fenceOverLays && this.fenceOverLays.forEach(item => {
  138. item.destroy();
  139. })
  140. this.fenceOverLays = null;
  141. this.map && this.map.clearMap() && this.map.destroy();
  142. this.map = null;
  143. // console.log(this);
  144. },
  145. methods: {
  146. /**
  147. * 更改配置
  148. */
  149. changeSetting(item, val) {
  150. if (val) {
  151. localStorage.setItem(item, val);
  152. } else {
  153. localStorage.removeItem(item);
  154. }
  155. if (item == 'isOnlyFenceDevice') {
  156. // 如果没有选中的围栏,不重新请求
  157. if (!this.checkedFenceObject) {
  158. return;
  159. }
  160. }
  161. this.getLastPositon();
  162. },
  163. /**
  164. * 刷新设备信息
  165. */
  166. refreshPosAndRoute(){
  167. this.panelShow = false;
  168. this.getLastPositon()
  169. },
  170. /**
  171. * 创建信息窗
  172. */
  173. createInfowind(str){
  174. let content = '<div style="background: white;box-shadow: 0 2px 4px rgba(0, 0, 0, .12), 0 0 6px rgba(0, 0, 0, .04);border-radius: 30px;padding:2px 8px">'+
  175. // '<div>' +
  176. // '<img style="width:0.4rem;height:0.4rem" src="@/assets/image/walking.png" alt="">' +
  177. // '</div>' +
  178. '<div class="info-window-data">' +
  179. '<span> '+ str +'</span>' +
  180. // '<span>'+ data.time +'分钟</span>' +
  181. ' </div>' +
  182. '</div>';
  183. return content;
  184. },
  185. /**
  186. * 刷新选中单个设备信息
  187. */
  188. refreshSingle(){
  189. let markerData = this.selectDevice;
  190. let that =this;
  191. this.$http.get("getLastPositionSingle?id=" + markerData.id +'&openid=' + this.openid).then(res => {
  192. if(res.data.code!=10000){
  193. that.$dialog.toast({
  194. mes: res.data.message,
  195. timeout: 3000
  196. });
  197. return
  198. }
  199. let devices = res.data.data;
  200. devices._index = markerData._index;
  201. that.markersArr[markerData._index].setExtData(devices);
  202. that.markersArr[markerData._index].setPosition(new AMap.LngLat(devices.lng, devices.lat));
  203. that.infoWindow.open(that.map, that.markersArr[markerData._index].getPosition());
  204. that.map.panTo(that.markersArr[markerData._index].getPosition());
  205. that.selectDevice = devices;
  206. // console.log(devices, markerData._index);
  207. })
  208. .catch(res => {
  209. console.log(res);
  210. });
  211. },
  212. /**
  213. * 预览图片
  214. */
  215. showPreview(){
  216. ImagePreview(['https://rlfd.oss-cn-hangzhou.aliyuncs.com/wxt_school/student_card_product.png']);
  217. },
  218. /**
  219. * 显示/隐藏信息窗
  220. */
  221. togglePanel(){
  222. this.panelShow = !this.panelShow
  223. },
  224. /**
  225. * 显示/隐藏围栏
  226. */
  227. toggleShowFence(){
  228. this.showFence = !this.showFence
  229. // console.log(this.showFence)
  230. if(this.showFence && this.fenceOverLays.length === 0){
  231. this.getAllFences();
  232. return;
  233. }
  234. if(this.showFence){
  235. this.fenceOverLays.forEach(item => {
  236. item.show();
  237. item.getExtData().textLabel.show()
  238. });
  239. this.map.setFitView(this.fenceOverLays);
  240. }else{
  241. this.fenceOverLays.forEach(item => {
  242. item.hide();
  243. item.getExtData().textLabel.hide()
  244. });
  245. }
  246. },
  247. /**
  248. * 初始化地图
  249. */
  250. initMap(){
  251. let that = this;
  252. return new Promise((resolve, reject) => {
  253. try {
  254. var map = new AMap.Map('route-map', {
  255. zoom:15,//级别
  256. center: [119.5,32.9],//中心点坐标
  257. // viewMode:'3D',//使用3D视图
  258. // mapStyle: 'amap://styles/light',
  259. showBuildingBlock: true
  260. });
  261. AMap.plugin([
  262. 'AMap.ToolBar',
  263. 'AMap.Scale',
  264. 'AMap.MapType',
  265. 'AMap.Geocoder',
  266. 'AMap.Geolocation'
  267. // 'AMap.MarkerCluster',
  268. ], function () {
  269. let geolocation = new AMap.Geolocation({
  270. enableHighAccuracy: true,
  271. // 设置定位超时时间,默认:无穷大
  272. timeout: 10000,
  273. // 定位按钮的停靠位置的偏移量,默认:Pixel(10, 20)
  274. buttonOffset: new AMap.Pixel(10, 20),
  275. // 定位成功后调整地图视野范围使定位位置及精度范围视野内可见,默认:false
  276. zoomToAccuracy: true,
  277. panToLocation: true,
  278. noIpLocate:0,
  279. // 定位按钮的排放位置, RB表示右下
  280. buttonPosition: 'RB'
  281. });
  282. map.addControl(geolocation);
  283. geolocation.getCurrentPosition()
  284. console.log(AMap.Event);
  285. AMap.Event.addListener(geolocation, 'complete', onComplete);
  286. AMap.Event.addListener(geolocation, 'error', onError);
  287. function onComplete(data) {
  288. // data是具体的定位信息
  289. console.log(data)
  290. if (data.position) {
  291. that.saveLocationInfo(data.position);
  292. } else {
  293. that.$notify({
  294. type: 'warning',
  295. message: '获取定位失败',
  296. });
  297. }
  298. }
  299. function onError(data) {
  300. // 定位出错
  301. that.$notify({
  302. type: 'warning',
  303. message: '获取定位失败',
  304. });
  305. }
  306. // 在图面添加工具条控件,工具条控件集成了缩放、平移、定位等功能按钮在内的组合控件
  307. map.addControl(new AMap.ToolBar({position: 'RT', offset: [15, 50]}));
  308. // 在图面添加比例尺控件,展示地图在当前层级和纬度下的比例尺
  309. map.addControl(new AMap.Scale({position: 'LT'}));
  310. // // 在图面添加类别切换控件,实现默认图层与卫星图、实施交通图层之间切换的控制
  311. // map.addControl(new AMap.MapType());
  312. that.geocoder = new AMap.Geocoder({})
  313. });
  314. that.infoWindow = new AMap.InfoWindow({
  315. isCustom: true, //使用自定义窗体
  316. content: '',
  317. autoMove: false,
  318. offset: new AMap.Pixel(0, -40)
  319. });
  320. that.map = map;
  321. that.map.on('click', function(){
  322. that.panelShow = false;
  323. that.infoWindow.close();
  324. });
  325. resolve(that.map);
  326. } catch (error) {
  327. console.log(error)
  328. reject(false);
  329. }
  330. });
  331. },
  332. //保存个人定位信息
  333. saveLocationInfo(data)
  334. {
  335. localStorage.setItem('position',data);
  336. },
  337. /**
  338. * 获取围栏数据
  339. */
  340. getAllFences(){
  341. let that= this;
  342. this.$http.get("/getAllFences&userid=" + this.userid).then(res => {
  343. // console.log(res.data, 'fence res')
  344. let resp = res.data;
  345. if(!resp.success){
  346. that.$dialog.toast({
  347. mes: resp.message,
  348. timeout: 3000
  349. });
  350. return
  351. }
  352. let fenceData = resp.data, fenceOverLays = [];
  353. fenceData.forEach(item => {
  354. let overLay = that.createFenceOverlay(item);
  355. if(!overLay){
  356. }else{
  357. overLay.setMap(that.map)
  358. overLay.getExtData().textLabel.setMap(that.map)
  359. // that.map.add(overLay)
  360. fenceOverLays.push(overLay);
  361. }
  362. });
  363. that.fenceOverLays = fenceOverLays;
  364. that.map.setFitView(that.fenceOverLays);
  365. })
  366. .catch(res => {
  367. console.log(res);
  368. });
  369. },
  370. /**
  371. * 画围栏
  372. */
  373. createFenceOverlay(fenceData){
  374. let that = this;
  375. let overLay = '';
  376. let type = fenceData.type ? '用户围栏' : '系统围栏';
  377. if(fenceData.shape == 'circle'){
  378. let center = fenceData.info.center, radius = fenceData.info.radius;
  379. let textLabel = new AMap.Text({
  380. text: fenceData.name + '(' + type + ')',
  381. anchor:'center', // 设置文本标记锚点
  382. draggable:false,
  383. style:{
  384. 'padding': '.2rem',
  385. 'background-color': 'white',
  386. 'min-width': '3rem',
  387. 'border-width': 0,
  388. 'box-shadow': '0 2px 6px 0 rgba(114, 124, 245, .5)',
  389. 'text-align': 'center',
  390. 'opacity': '0.8',
  391. 'font-size': '12px',
  392. 'color': 'blue'
  393. },
  394. position: [center.lng, center.lat]
  395. });
  396. textLabel.on('click', function(e) {
  397. that.map.setBounds(e.target.getBounds())
  398. })
  399. overLay = new AMap.Circle({
  400. center: new AMap.LngLat(center.lng, center.lat), // 圆心位置
  401. radius: radius, //半径
  402. strokeColor: "#F33", //线颜色
  403. strokeOpacity: 1, //线透明度
  404. strokeWeight: 3, //线粗细度
  405. fillColor: "#ee2200", //填充颜色
  406. fillOpacity: 0.35, //填充透明度
  407. extData: {name: fenceData.name, type: type, textLabel: textLabel}
  408. });
  409. }
  410. if(fenceData.shape == 'polygon'){
  411. let originPt = fenceData.info;
  412. let path = []
  413. originPt.forEach(item => {
  414. let pt= [item.lng, item.lat]
  415. path.push(pt);
  416. });
  417. let textLabel = new AMap.Text({
  418. text: fenceData.name + '(' + type + ')',
  419. anchor:'center', // 设置文本标记锚点
  420. draggable:false,
  421. style:{
  422. 'padding': '.2rem 0.5rem',
  423. 'margin-bottom': '1rem',
  424. 'border-radius': '.25rem',
  425. 'background-color': 'white',
  426. 'width': '4rem',
  427. 'border-width': 0,
  428. 'box-shadow': '0 2px 6px 0 rgba(114, 124, 245, .5)',
  429. 'text-align': 'center',
  430. 'opacity': '0.8',
  431. 'font-size': '12px',
  432. 'color': 'blue'
  433. },
  434. position: path[0]
  435. });
  436. overLay = new AMap.Polygon({
  437. path: path,
  438. strokeColor: "#FF33FF",
  439. strokeWeight: 6,
  440. strokeOpacity: 0.2,
  441. fillOpacity: 0.4,
  442. fillColor: '#1791fc',
  443. zIndex: 50,
  444. extData: {name: fenceData.name, type: type, textLabel: textLabel}
  445. })
  446. }
  447. // 添加点击事件
  448. overLay.on('click', function(e) {
  449. that.panelShow = false;
  450. that.checkedFenceObject = e.target;
  451. that.getLastPositon();
  452. });
  453. return overLay;
  454. },
  455. /**
  456. * 获去设备定位数据列表
  457. */
  458. getLastPositon(){
  459. let that = this;
  460. this.$http.get("getHydList?userid="+this.userid+'&openid='+this.openid ).then(res => {
  461. if(res.data.code!=10000){
  462. that.$toast(res.data.message || '获取设备数据失败');
  463. return;
  464. }
  465. let devices = res.data.data;
  466. if (!devices.length) {
  467. return;
  468. }
  469. // 移除设备覆盖物
  470. // that.map.clearMap()
  471. that.markersArr && that.map.remove(that.markersArr);
  472. that.infoWindow && that.infoWindow.close();
  473. let markerArr = [];
  474. devices.forEach( (item,index) => {
  475. item._index = index;
  476. let markerPos = new AMap.LngLat(item.lng, item.lat)
  477. let image ='https://rlfd.oss-cn-hangzhou.aliyuncs.com/smart_tool/beng.png';
  478. // 如果选中只显示围栏内设备
  479. if (that.isOnlyFenceDevice && that.checkedFenceObject) {
  480. if (!that.checkedFenceObject.contains(markerPos)) {
  481. return;
  482. }
  483. }
  484. var marker = new AMap.Marker({
  485. position: markerPos,
  486. icon: new AMap.Icon({
  487. size: new AMap.Size(40, 40),
  488. image: image,
  489. imageSize: new AMap.Size(40, 40),
  490. }), // 添加 Icon 图标 URL
  491. offset: new AMap.Pixel(-20,-40),
  492. extData: item
  493. })
  494. marker.on('click', function (e) {
  495. let data = e.target.getExtData();
  496. data.loc_mode = (data.loc_mode && data.loc_mode.toUpperCase())
  497. that.selectDevice = data;
  498. let str = (data.device_name != 'null' && data.device_name) ? data.device_name : data.imei;
  499. // str = str + '(' + data.imei + ')';
  500. let content = that.createInfowind(str);
  501. that.infoWindow.setContent(content);
  502. that.map.setCenter(e.target.getPosition())
  503. that.panelShow = true;
  504. that.infoWindow.open(that.map, marker.getPosition());
  505. });
  506. that.map.add(marker)
  507. markerArr.push(marker);
  508. });
  509. // if(!that.positionInfo.lng && !that.positionInfo.lat){
  510. // that.parsedAddress = '未曾上报位置,无法解析地址';
  511. // return;
  512. // }
  513. that.markersArr = markerArr;
  514. that.map.setFitView(markerArr);
  515. //默认选中一个
  516. if (that.markersArr.length) {
  517. let defaultData = that.markersArr[0].getExtData();
  518. defaultData.loc_mode = (defaultData.loc_mode && defaultData.loc_mode.toUpperCase())
  519. that.selectDevice = defaultData;
  520. // let defaultStr = defaultData.device_name + '(' + defaultData.imei + ')';
  521. let defaultStr = defaultData.device_name || defaultData.imei;
  522. let defaultContent = that.createInfowind(defaultStr);
  523. that.infoWindow.setContent(defaultContent);
  524. that.infoWindow.open(that.map, that.markersArr[0].getPosition());
  525. that.panelShow = true;
  526. }
  527. })
  528. .catch(res => {
  529. console.log(res);
  530. });
  531. }
  532. }
  533. }
  534. </script>
  535. <style lang="css" scoped>
  536. #route-map{
  537. height:calc(100vh - 1rem);
  538. width: 100%;
  539. -webkit-transform: translateZ(0);
  540. -moz-transform: translateZ(0);
  541. -ms-transform: translateZ(0);
  542. -o-transform: translateZ(0);
  543. transform: translateZ(0);
  544. }
  545. .map-info-window{
  546. background: white;
  547. box-shadow: 0 2px 4px rgba(0, 0, 0, .12), 0 0 6px rgba(0, 0, 0, .04);
  548. border-radius: 30px;
  549. max-width: 4rem;
  550. display: flex;
  551. justify-content: space-around;
  552. align-items: center;
  553. }
  554. .info-window-data{
  555. display: flex;
  556. flex-direction: column;
  557. align-items: center;
  558. }
  559. .info-panel{
  560. padding:0.4rem;
  561. }
  562. .panel-first-row{
  563. display: flex;
  564. margin-bottom:0.2rem;
  565. justify-content: space-between;
  566. align-items: center;
  567. }
  568. .panel-second-row{
  569. display: flex;
  570. justify-content: space-between;
  571. align-items: flex-end;
  572. }
  573. .second-row-left{
  574. display: flex;
  575. }
  576. .right-number{
  577. box-shadow: 0 2px 4px rgba(0, 0, 0, .12), 0 0 6px rgba(0, 0, 0, .04);
  578. border-radius: 6px;
  579. padding: 2px 4px 0 4px;
  580. font-weight: 600;
  581. }
  582. .row-left-layout{
  583. display: flex;
  584. flex-direction: column;
  585. margin-right: 0.5rem ;
  586. }
  587. .row-right-layout{
  588. display: flex;
  589. flex-direction: column;
  590. align-items: center;
  591. }
  592. .route-search-button{
  593. display: flex;
  594. justify-content: center;
  595. margin-top: 0.2rem;
  596. }
  597. .panel-plain-text{
  598. margin-top: 0.1rem;
  599. overflow:hidden;
  600. text-overflow:ellipsis;
  601. display:-webkit-box;
  602. -webkit-box-orient:vertical;
  603. -webkit-line-clamp:2;
  604. }
  605. .type-bar{
  606. position: absolute;
  607. right: 10px;
  608. z-index: 99;
  609. top: 2.5rem;
  610. }
  611. .change-button{
  612. display: flex;
  613. flex-direction: column;
  614. justify-content: space-between;
  615. }
  616. .fence-showoff{
  617. position: absolute;
  618. bottom: 240px;
  619. right: 15px;
  620. z-index: 99;
  621. }
  622. .icon-base{
  623. width: 0.6rem;
  624. height: 0.6rem;
  625. margin-bottom: 0.1rem;
  626. background: white;
  627. border-radius: 8px;
  628. display: flex;
  629. justify-content: center;
  630. align-items: center;
  631. padding: 4px;
  632. box-shadow: 0 2px 4px rgba(0, 0, 0, .12), 0 0 6px rgba(0, 0, 0, .04);
  633. color: darkgray;
  634. }
  635. .buttonActive{
  636. color: white;
  637. background: #2196f3;
  638. }
  639. .close-button-fold{
  640. /* position: absolute; */
  641. background: #ffffff;
  642. width: 40px;
  643. height: 20px;
  644. bottom: 4rem;
  645. left: 0;
  646. right: 0;
  647. margin: auto;
  648. border: 1px darkgray solid;
  649. border-bottom: none;
  650. border-radius: 100px 100px 0 0;
  651. display: flex;
  652. justify-content: center;
  653. align-items: center;
  654. z-index: 199;
  655. }
  656. .close-button-unfold{
  657. /* position: absolute; */
  658. background: #ffffff;
  659. width: 40px;
  660. height: 20px;
  661. bottom: 0;
  662. left: 0;
  663. right: 0;
  664. margin: auto;
  665. border: 1px darkgray solid;
  666. border-bottom: none;
  667. border-radius: 100px 100px 0 0;
  668. display: flex;
  669. justify-content: center;
  670. align-items: center;
  671. z-index: 199;
  672. }
  673. .normalColor{
  674. color: #2196f3;
  675. }
  676. .alarmColor{
  677. color: red;
  678. }
  679. .bottom-route-info{
  680. /* position: absolute; */
  681. bottom: 0;
  682. height: 4rem;
  683. background: #fff;
  684. width: 100%;
  685. }
  686. .route-button{
  687. background: #2196f3;
  688. border-radius: 5px;
  689. padding: 2px;
  690. width: 100px;
  691. text-align: center;
  692. color: #fff;
  693. }
  694. .address-info{
  695. display: flex;
  696. justify-content: space-between;
  697. margin-top: 10px;
  698. }
  699. .float-first{
  700. display: flex;
  701. align-items: center;
  702. position: relative;
  703. }
  704. .first-locinfo{
  705. display: flex;
  706. margin-left: 10px;
  707. }
  708. .first-avatar{
  709. border-radius: 500px;
  710. width: 48px;
  711. height: 48px;
  712. background: lightgray;
  713. display: flex;
  714. justify-content: center;
  715. align-content: center;
  716. }
  717. .select-butt{
  718. position: absolute;
  719. top: 1.2rem;
  720. z-index: 1;
  721. right: 0;
  722. left: 0;
  723. margin: auto;
  724. }
  725. .bottom-panel{
  726. position: absolute;
  727. bottom: 0.3rem;
  728. z-index: 99;
  729. width: 100%;
  730. display: flex;
  731. flex-direction: column;
  732. justify-content: center;
  733. align-items: center;
  734. }
  735. .first-loctime{
  736. text-align: left;
  737. }
  738. .panel-float{
  739. padding: 10px;
  740. background: white;
  741. border-radius: 10px;
  742. min-height: 2.4rem;
  743. width: 90%;
  744. box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1);
  745. }
  746. .refresh-button{
  747. position: absolute;
  748. bottom: 200px;
  749. right: 15px;
  750. z-index: 99;
  751. height: 30px;
  752. width: 30px;
  753. display: flex;
  754. justify-content: center;
  755. align-items: center;
  756. border-radius: 50px;
  757. padding: 2px;
  758. background: #fff;
  759. box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.2);
  760. }
  761. .setting-button {
  762. position: absolute;
  763. width: 30px;
  764. height: 30px;
  765. z-index: 1000;
  766. top: 15px;
  767. right: 15px;
  768. background-color: #fff;
  769. padding: 4px;
  770. border-radius: 6px;
  771. box-shadow: 0 2px 4px rgba(0, 0, 0, .12), 0 0 6px rgba(0, 0, 0, .04);
  772. text-align: center;
  773. line-height: 30px;
  774. }
  775. </style>