2024-02-26 11:51:13 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<div class="geo-analysis">
|
2024-03-06 18:04:52 +08:00
|
|
|
|
<el-tabs v-model="activeTab" class="location-tabs">
|
2024-03-04 17:28:05 +08:00
|
|
|
|
<el-tab-pane :label="$t('location.locationMap')" name="locationMap"></el-tab-pane>
|
2024-03-06 18:04:52 +08:00
|
|
|
|
<el-tab-pane :label="$t('location.traceTracking')" name="traceTracking">
|
|
|
|
|
|
<template #label>
|
|
|
|
|
|
<div class="traceTracking-tabs_label">
|
|
|
|
|
|
{{$t('location.traceTracking')}}
|
|
|
|
|
|
<div class="traceTracking-num" :style="`opacity:${opacity};`">
|
|
|
|
|
|
{{trackingSubscribers.length}}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-tab-pane>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
</el-tabs>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
<!-- 右上角工具栏 -->
|
2024-02-26 11:51:13 +08:00
|
|
|
|
<div class="geo-tools">
|
2024-03-01 18:35:50 +08:00
|
|
|
|
<el-select
|
|
|
|
|
|
id="searchValue"
|
|
|
|
|
|
ref="searchValue"
|
|
|
|
|
|
v-model="dropDownValue"
|
2024-03-05 15:16:11 +08:00
|
|
|
|
v-show="activeTab === 'locationMap'"
|
2024-03-07 19:18:18 +08:00
|
|
|
|
size="small"
|
2024-03-01 18:35:50 +08:00
|
|
|
|
clearable
|
|
|
|
|
|
filterable
|
|
|
|
|
|
remote
|
2024-03-04 14:23:47 +08:00
|
|
|
|
:placeholder="$t('location.enterPhoneNumberToSearch')"
|
2024-03-01 18:35:50 +08:00
|
|
|
|
popper-class="search-select"
|
2024-03-07 19:18:18 +08:00
|
|
|
|
:teleported="false"
|
2024-03-01 18:35:50 +08:00
|
|
|
|
placement="bottom"
|
|
|
|
|
|
@visible-change="visibleChange"
|
|
|
|
|
|
:remote-method="dropDownSearch"
|
|
|
|
|
|
v-select-load-more="onLoadMore"
|
2024-03-07 15:35:59 +08:00
|
|
|
|
style="margin-right: 10px; width:250px;"
|
2024-03-01 18:35:50 +08:00
|
|
|
|
>
|
|
|
|
|
|
<el-option
|
|
|
|
|
|
v-for="item in searchValueListShow"
|
|
|
|
|
|
:key="item.subscriberId"
|
|
|
|
|
|
:value="item.phoneNumber"
|
|
|
|
|
|
>
|
|
|
|
|
|
<span class="search-active" :class="item.active === 1 ? 'active-icon' : 'inactive-icon'"></span>
|
2024-03-07 15:35:59 +08:00
|
|
|
|
<div class="search-content">
|
|
|
|
|
|
<span class="search-value">{{ item.phoneNumber }}</span>
|
|
|
|
|
|
<span class="" style="color: #999;">ID {{ item.subscriberId }}</span>
|
|
|
|
|
|
</div>
|
2024-03-01 18:35:50 +08:00
|
|
|
|
<span class="search-follow__icon" @click.stop="followSubscribers(item)">
|
2024-03-07 15:13:30 +08:00
|
|
|
|
<i class="cn-icon-follow-fill cn-icon" v-if="item.follow === 1"></i>
|
|
|
|
|
|
<i class="cn-icon-follow cn-icon" v-else></i>
|
2024-03-01 18:35:50 +08:00
|
|
|
|
</span>
|
|
|
|
|
|
</el-option>
|
2024-03-06 17:13:30 +08:00
|
|
|
|
<template #empty>
|
|
|
|
|
|
<div v-if="loading.searchLoading" style="height:40px;padding:10px 0;text-align: center;color:#909399;font-size:12px;display:flex;align-items: center;justify-content: center;">
|
|
|
|
|
|
<simple-loading style="padding-top:10px;" size="small" placement="top" :loading="loading.searchLoading" ></simple-loading>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div v-else-if="emptyTip !== ''" style="height:40px;padding:10px 0;text-align: center;color:#909399;font-size:12px;display:flex;align-items: center;justify-content: center;">
|
|
|
|
|
|
{{emptyTip}}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
2024-03-01 18:35:50 +08:00
|
|
|
|
</el-select>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
<div class="panel__time">
|
|
|
|
|
|
<date-time-range
|
|
|
|
|
|
class="date-time-range"
|
|
|
|
|
|
:start-time="timeFilter.startTime"
|
|
|
|
|
|
:end-time="timeFilter.endTime"
|
|
|
|
|
|
:date-range="timeFilter.dateRangeValue"
|
|
|
|
|
|
ref="dateTimeRange"
|
|
|
|
|
|
@change="reload"
|
|
|
|
|
|
/>
|
|
|
|
|
|
<time-refresh
|
|
|
|
|
|
class="date-time-range"
|
|
|
|
|
|
@change="timeRefreshChange"
|
|
|
|
|
|
:end-time="timeFilter.endTime"
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="geo-analysis__container">
|
2024-02-29 18:35:43 +08:00
|
|
|
|
<!-- 左侧地图 -->
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<div class="analysis-map">
|
|
|
|
|
|
<simple-loading size="small" placement="top-end" :loading="loading.mapLoading" v-if="activeTab === 'locationMap'"></simple-loading>
|
|
|
|
|
|
<simple-loading size="small" placement="top-end" :loading="loading.trackingMapLoading" v-else-if="activeTab === 'traceTracking'"></simple-loading>
|
|
|
|
|
|
<div id="analysisMap"></div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<!-- locationMap地图底部的时间轴 -->
|
|
|
|
|
|
<div class="map-time-line">
|
|
|
|
|
|
<time-line v-if="activeTab === 'locationMap'" :timeFilter="timeFilter" @change="mapTimeLineChange"></time-line>
|
|
|
|
|
|
</div>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
<!-- 右侧数据栏-map -->
|
2024-02-28 19:11:33 +08:00
|
|
|
|
<div class="analysis-statistics" v-show="activeTab === 'locationMap'">
|
2024-03-11 16:32:29 +08:00
|
|
|
|
<!-- 饼图 -->
|
2024-02-29 18:35:43 +08:00
|
|
|
|
<div class="analysis-statistics__chart">
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<simple-loading size="small" placement="top-end" :loading="loading.pieLoading"></simple-loading>
|
2024-03-04 17:28:05 +08:00
|
|
|
|
<div class="chart__header">{{$t('location.populationDensity')}}</div>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
<div class="chart__body">
|
|
|
|
|
|
<div class="chart__drawing" id="populationDensityChart"></div>
|
|
|
|
|
|
<div class="chart__legend">
|
|
|
|
|
|
<div v-for="legend in pieValueRamp" class="legend-item" :key="legend.color">
|
|
|
|
|
|
<div class="legend-icon" :style="`background:rgb(${legend.color});`"></div>
|
|
|
|
|
|
<div class="legend-range" >{{legend.start}}~{{legend.end}}</div>
|
|
|
|
|
|
<div class="legend-count">{{legend.count}}</div>
|
2024-02-28 07:38:41 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
</div>
|
2024-03-11 16:32:29 +08:00
|
|
|
|
<!-- 折线图 -->
|
2024-02-29 18:35:43 +08:00
|
|
|
|
<div class="analysis-statistics__chart">
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<simple-loading size="small" placement="top-end" :loading="loading.lineLoading"></simple-loading>
|
2024-03-04 17:28:05 +08:00
|
|
|
|
<div class="chart__header">{{$t('location.activeSubscribers')}}</div>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
<div class="chart__statistics">
|
|
|
|
|
|
<div class="statistics-number">{{activeCount}}</div>
|
2024-03-01 18:35:50 +08:00
|
|
|
|
<div class="statistics-trend">{{activeCountChain === '-' ? '-' : (activeCountChain < 0 ? '-'+valueToRangeValue(Math.abs(activeCountChain), unitTypes.percent).join(' '): valueToRangeValue(activeCountChain, unitTypes.percent).join(' '))}}</div>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
</div>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
<div class="chart-line__drawing" id="activeSubscribersChart"></div>
|
|
|
|
|
|
</div>
|
2024-03-06 17:44:13 +08:00
|
|
|
|
<div class="analysis-statistics__title">
|
|
|
|
|
|
{{$t('location.followedSubscribers')}}
|
|
|
|
|
|
<simple-loading :loading="loading.followSubscriberLoading" placement="right" size="small"></simple-loading>
|
|
|
|
|
|
</div>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
<div class="analysis-statistics__subscribers">
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<template v-for="item in followedSubscribersList" :key="item.subscriberId">
|
2024-03-11 16:32:29 +08:00
|
|
|
|
<div class="analysis-statistics__subscriber"
|
|
|
|
|
|
@click="subscriberListClick(item)"
|
|
|
|
|
|
@mouseenter="subscriberListMouseEnter(item)"
|
|
|
|
|
|
@mouseleave="subscriberListMouseLeave(item)"
|
|
|
|
|
|
:id="`locationMap-subscriberId-${item.subscriberId}`"
|
|
|
|
|
|
:class="highlightSubscriber.subscriberId === item.subscriberId ? 'analysis-statistics__subscriber--active' : ''"
|
|
|
|
|
|
>
|
2024-03-04 18:21:33 +08:00
|
|
|
|
<div class="subscriber__header" :class="item.active === 1 ? '' : 'subscriber__header-inactive'" >
|
2024-02-29 18:35:43 +08:00
|
|
|
|
<div class="header__icon">
|
|
|
|
|
|
<div class="icon__box">
|
2024-03-11 16:32:29 +08:00
|
|
|
|
<svg viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M366.689524 690.468571l87.283809 83.821715-75.434666 195.486476c-10.971429 27.794286-43.105524 42.081524-72.265143 32.036571-29.159619-10.24-44.080762-41.252571-33.450667-69.241904L366.689524 690.468571zM203.824762 476.306286l51.785143-95.183238a162.279619 162.279619 0 0 1 59.245714-59.977143c119.710476-68.266667 134.777905-67.291429 149.942857-66.218667l80.798476 5.168762c24.868571 0.975238 42.081524 7.314286 125.025524 124.14781a21.26019 21.26019 0 0 0 14.092191 8.289523l99.132952 14.482286c12.873143 1.852952 24.478476 8.582095 32.182857 18.67581a45.494857 45.494857 0 0 1 8.825905 35.108571 46.665143 46.665143 0 0 1-19.504762 30.866286 50.468571 50.468571 0 0 1-36.571429 8.435809l-99.181714-14.433524a119.954286 119.954286 0 0 1-79.774476-47.640381c-4.388571-6.241524-7.558095-11.361524-11.849143-16.579047l-63.634286 193.487238 88.405334 84.845714c12.970667 12.385524 23.698286 29.013333 30.232381 45.494857l67.876571 190.366477c5.022476 13.409524 4.193524 28.233143-2.291809 41.057523a54.613333 54.613333 0 0 1-32.182858 27.160381c-5.851429 2.048-12.092952 3.120762-18.383238 3.169524a58.075429 58.075429 0 0 1-53.930666-36.181333l-67.876572-190.366476c-1.024-2.096762-2.096762-3.120762-3.218285-5.168762L365.616762 623.177143a84.894476 84.894476 0 0 1-24.868572-80.700953l34.523429-146.919619c-3.413333 2.340571-7.070476 4.388571-10.776381 6.290286a58.806857 58.806857 0 0 0-22.674286 22.723048L290.133333 519.801905a49.834667 49.834667 0 0 1-43.105523 24.819809 59.977143 59.977143 0 0 1-22.674286-5.12 46.518857 46.518857 0 0 1-20.48-63.146666z m209.67619-360.448C420.08381 58.465524 473.86819 17.066667 533.650286 23.30819 593.383619 29.549714 636.537905 81.13981 630.00381 138.48381c-6.534095 57.392762-60.269714 98.840381-120.05181 92.550095-59.782095-6.241524-102.985143-57.782857-96.451048-115.175619z"></path></svg>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2024-03-04 18:21:33 +08:00
|
|
|
|
<div class="header__right">
|
|
|
|
|
|
<div class="header-msisdn">
|
|
|
|
|
|
<div class="header__title">MSISDN</div>
|
2024-03-05 11:19:47 +08:00
|
|
|
|
<div class="header__content">{{$_.get(item, 'subscriberDto.phoneNumber', '-')}}</div>
|
2024-03-04 18:21:33 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="header__operation">
|
2024-03-11 16:32:29 +08:00
|
|
|
|
<div class="trajectory-text" @click.stop="addOrRemoveTrackingSubscriber(item)">
|
2024-03-05 15:16:11 +08:00
|
|
|
|
<i class="cn-icon" :class="symbolClass(item)"></i>{{$t('location.track')}}
|
2024-03-04 18:21:33 +08:00
|
|
|
|
</div>
|
2024-03-05 15:16:11 +08:00
|
|
|
|
<div class="cancel-follow" @click.stop="cancelFollowSubscribers(item)">
|
2024-03-04 18:21:33 +08:00
|
|
|
|
<i class="cn-icon-close cn-icon"></i>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="subscriber__body">
|
2024-03-05 11:19:47 +08:00
|
|
|
|
<div class="body__item">
|
|
|
|
|
|
<div class="item__label">ID</div>
|
|
|
|
|
|
<div class="item__value">{{item.subscriberId}}</div>
|
|
|
|
|
|
</div>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
<div class="body__item">
|
2024-03-01 18:35:50 +08:00
|
|
|
|
<div class="item__label">{{$t('location.group')}}</div>
|
2024-03-05 11:19:47 +08:00
|
|
|
|
<div class="item__value">Terrorist</div>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="body__item">
|
2024-03-01 18:35:50 +08:00
|
|
|
|
<div class="item__label">{{$t('overall.info')}}</div>
|
2024-03-05 11:19:47 +08:00
|
|
|
|
<div class="item__value">Leader</div>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="body__item">
|
2024-03-01 18:35:50 +08:00
|
|
|
|
<div class="item__label">{{$t('location.location')}}</div>
|
2024-03-05 11:19:47 +08:00
|
|
|
|
<div class="item__value">China, Shanghai</div>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<!-- 右侧数据栏-trace -->
|
2024-03-05 21:57:46 +08:00
|
|
|
|
<div class="analysis-statistics" id="subscribersBlock" v-show="activeTab === 'traceTracking'">
|
2024-03-05 15:16:11 +08:00
|
|
|
|
<div class="analysis-statistics__no-tracking-tip" v-if="trackingSubscribers.length === 0" @click="activeTab = 'locationMap'">{{$t('location.noTrackingYet')}}</div>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<div class="analysis-statistics__subscribers">
|
|
|
|
|
|
<template v-for="(subscriber, index) in trackingSubscribers" :key="subscriber.subscriberId">
|
|
|
|
|
|
<div
|
|
|
|
|
|
class="analysis-statistics__subscriber"
|
|
|
|
|
|
:class="currentShowSubscriber && currentShowSubscriber.subscriberId === subscriber.subscriberId ? 'analysis-statistics__subscriber--active' : ''"
|
|
|
|
|
|
@click="changeCurrentShowSubscriber(subscriber)">
|
2024-03-06 16:59:07 +08:00
|
|
|
|
<div class="subscriber__header" :class="trackingSubscriberRecordMap[subscriber.subscriberId] && trackingSubscriberRecordMap[subscriber.subscriberId].length > 0 ? '' : 'subscriber__header-inactive'">
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<div class="header__icon">
|
|
|
|
|
|
<div class="icon__box">
|
2024-03-11 16:32:29 +08:00
|
|
|
|
<svg viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M366.689524 690.468571l87.283809 83.821715-75.434666 195.486476c-10.971429 27.794286-43.105524 42.081524-72.265143 32.036571-29.159619-10.24-44.080762-41.252571-33.450667-69.241904L366.689524 690.468571zM203.824762 476.306286l51.785143-95.183238a162.279619 162.279619 0 0 1 59.245714-59.977143c119.710476-68.266667 134.777905-67.291429 149.942857-66.218667l80.798476 5.168762c24.868571 0.975238 42.081524 7.314286 125.025524 124.14781a21.26019 21.26019 0 0 0 14.092191 8.289523l99.132952 14.482286c12.873143 1.852952 24.478476 8.582095 32.182857 18.67581a45.494857 45.494857 0 0 1 8.825905 35.108571 46.665143 46.665143 0 0 1-19.504762 30.866286 50.468571 50.468571 0 0 1-36.571429 8.435809l-99.181714-14.433524a119.954286 119.954286 0 0 1-79.774476-47.640381c-4.388571-6.241524-7.558095-11.361524-11.849143-16.579047l-63.634286 193.487238 88.405334 84.845714c12.970667 12.385524 23.698286 29.013333 30.232381 45.494857l67.876571 190.366477c5.022476 13.409524 4.193524 28.233143-2.291809 41.057523a54.613333 54.613333 0 0 1-32.182858 27.160381c-5.851429 2.048-12.092952 3.120762-18.383238 3.169524a58.075429 58.075429 0 0 1-53.930666-36.181333l-67.876572-190.366476c-1.024-2.096762-2.096762-3.120762-3.218285-5.168762L365.616762 623.177143a84.894476 84.894476 0 0 1-24.868572-80.700953l34.523429-146.919619c-3.413333 2.340571-7.070476 4.388571-10.776381 6.290286a58.806857 58.806857 0 0 0-22.674286 22.723048L290.133333 519.801905a49.834667 49.834667 0 0 1-43.105523 24.819809 59.977143 59.977143 0 0 1-22.674286-5.12 46.518857 46.518857 0 0 1-20.48-63.146666z m209.67619-360.448C420.08381 58.465524 473.86819 17.066667 533.650286 23.30819 593.383619 29.549714 636.537905 81.13981 630.00381 138.48381c-6.534095 57.392762-60.269714 98.840381-120.05181 92.550095-59.782095-6.241524-102.985143-57.782857-96.451048-115.175619z"></path></svg>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
</div>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
</div>
|
2024-03-05 15:16:11 +08:00
|
|
|
|
<div class="header__right">
|
|
|
|
|
|
<div class="header-msisdn">
|
|
|
|
|
|
<div class="header__title">MSISDN</div>
|
|
|
|
|
|
<div class="header__content">{{$_.get(subscriber, 'subscriberDto.phoneNumber', '-')}}</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="header__operation">
|
|
|
|
|
|
<div class="cancel-follow" @click.stop="removeTrackingSubscriber(subscriber)">
|
|
|
|
|
|
<i class="cn-icon-close cn-icon"></i>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
</div>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<div class="subscriber__body">
|
2024-03-05 11:19:47 +08:00
|
|
|
|
<div class="body__item">
|
|
|
|
|
|
<div class="item__label">ID</div>
|
|
|
|
|
|
<div class="item__value">{{subscriber.subscriberId}}</div>
|
|
|
|
|
|
</div>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<div class="body__item">
|
2024-03-04 14:23:47 +08:00
|
|
|
|
<div class="item__label">{{$t('entities.group')}}</div>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<div class="item__value">Terrorist</div>
|
2024-03-01 17:43:15 +08:00
|
|
|
|
</div>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<div class="body__item">
|
2024-03-04 14:23:47 +08:00
|
|
|
|
<div class="item__label">{{$t('overall.info')}}</div>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<div class="item__value">Leader</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="body__item">
|
2024-03-04 14:23:47 +08:00
|
|
|
|
<div class="item__label">{{$t('overall.location')}}</div>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<div class="item__value">China, Shanghai</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="body-item-record">
|
2024-03-05 10:44:43 +08:00
|
|
|
|
<div class="item-record__header">{{ $t('location.trackRecord') }}</div>
|
2024-03-06 16:59:07 +08:00
|
|
|
|
<template v-if="trackingSubscriberRecordMap[subscriber.subscriberId] && trackingSubscriberRecordMap[subscriber.subscriberId].length > 0">
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<div class="item-record__info">
|
|
|
|
|
|
<div class="circle"></div>
|
2024-03-01 17:43:15 +08:00
|
|
|
|
</div>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<div class="item-record__timeline">
|
|
|
|
|
|
<div class="timeline__info">
|
|
|
|
|
|
<div class="timeline__info--circle">
|
|
|
|
|
|
<div class="info__circle"></div>
|
|
|
|
|
|
<div class="info__line" v-show="subscriber.showLine"></div>
|
|
|
|
|
|
</div>
|
2024-03-06 16:59:07 +08:00
|
|
|
|
<div class="timeline__info--item" @mouseenter="timelineMouseEnter(subscriber, trackingSubscriberRecordMap[subscriber.subscriberId][0])" @mouseleave="timelineMouseLeave(subscriber, trackingSubscriberRecordMap[subscriber.subscriberId][0])">
|
2024-03-01 18:35:50 +08:00
|
|
|
|
<div>
|
2024-03-06 16:59:07 +08:00
|
|
|
|
<span>{{$t('overall.location')}}: </span><span class="info--item__value">{{trackingSubscriberRecordMap[subscriber.subscriberId][0].subscriberLongitude}}, {{trackingSubscriberRecordMap[subscriber.subscriberId][0].subscriberLatitude}}</span>
|
2024-03-01 17:43:15 +08:00
|
|
|
|
</div>
|
2024-03-01 18:35:50 +08:00
|
|
|
|
<div>
|
2024-03-06 16:59:07 +08:00
|
|
|
|
<span>{{ $t('location.timeOfArrival') }}: </span><span class="info--item__value">{{dateFormatByAppearance(Number(trackingSubscriberRecordMap[subscriber.subscriberId][0].time))}}</span>
|
2024-03-01 18:35:50 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div>
|
2024-03-05 10:44:43 +08:00
|
|
|
|
<span>{{ $t('location.residenceTime') }}: </span><span class="info--item__value">-</span>
|
2024-03-01 18:35:50 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
</div>
|
2024-03-08 18:03:21 +08:00
|
|
|
|
|
|
|
|
|
|
<div class="scroll-view" @scroll="onScroll" :id="subscriber.subscriberId" v-show="subscriber.show">
|
|
|
|
|
|
<!-- 虚拟列表 -->
|
|
|
|
|
|
<div class="virtual-scroller" :style="`height: ${subscriber.listHeight}px`"></div>
|
|
|
|
|
|
<div class="scroll-list" :style="`transform: translateY(${subscriber.startOffset}px)`">
|
|
|
|
|
|
<div class="scroll__item"
|
|
|
|
|
|
v-for="(record, index) in trackingSubscriberRecordMap[subscriber.subscriberId].slice(subscriber.scrollStartIndex, subscriber.scrollEndIndex)"
|
|
|
|
|
|
:key="index"
|
|
|
|
|
|
@mouseenter="timelineMouseEnter(subscriber, record)"
|
|
|
|
|
|
@mouseleave="timelineMouseLeave(subscriber, record)">
|
|
|
|
|
|
<div class="item-circle">
|
|
|
|
|
|
<div class="circle-circle"></div>
|
|
|
|
|
|
<div class="circle-line"></div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="item-content">
|
2024-03-03 22:10:02 +08:00
|
|
|
|
<div>
|
2024-03-06 12:43:45 +08:00
|
|
|
|
<span>{{$t('overall.location')}}: </span><span class="item__value">{{record.subscriberLongitude}}, {{record.subscriberLatitude}}</span>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div>
|
2024-03-05 10:44:43 +08:00
|
|
|
|
<span>{{ $t('location.timeOfArrival') }}: </span><span class="item__value">{{dateFormatByAppearance(Number(record.time))}}</span>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div>
|
2024-03-05 10:44:43 +08:00
|
|
|
|
<span>{{ $t('location.residenceTime') }}: </span><span class="item__value">{{record.stayTime}}</span>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2024-03-08 18:03:21 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2024-03-01 17:43:15 +08:00
|
|
|
|
|
2024-03-06 16:59:07 +08:00
|
|
|
|
<div :class="trackingSubscriberRecordMap[subscriber.subscriberId].length === 1 ? 'item-record__btn-disabled' : 'item-record__btn'" @click.stop="clickTrackBlock(index)">
|
|
|
|
|
|
<span v-if="!subscriber.show"><i class="cn-icon cn-icon-down2" :style="trackingSubscriberRecordMap[subscriber.subscriberId].length === 1 ? 'color: #C0C4CC' : ''"></i></span>
|
|
|
|
|
|
<span v-if="subscriber.show"><i class="cn-icon cn-icon-up2" :style="trackingSubscriberRecordMap[subscriber.subscriberId].length === 1 ? 'color: #C0C4CC' : ''"></i></span>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
2024-03-01 17:43:15 +08:00
|
|
|
|
</div>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2024-03-03 22:10:02 +08:00
|
|
|
|
</template>
|
|
|
|
|
|
</div>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
</div>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
</div>
|
2024-02-29 18:35:43 +08:00
|
|
|
|
<div class="geo-analysis__hexagon-tooltip" id="tooltip" :class="`geo-analysis__hexagon-tooltip--${tooltip.type}`" v-if="tooltip.showMarkerTooltip || tooltip.showPolygonTooltip" :style="{'left': `${tooltip.x}px`, 'top': `${tooltip.y}px`}" @mouseenter="tooltipMouseEnter" @mouseleave="tooltipMouseLeave">
|
2024-02-26 11:51:13 +08:00
|
|
|
|
<div class="hexagon-tooltip__header" :style="`background-color: ${tooltipHeaderColor}`">
|
|
|
|
|
|
<div class="header__icon">
|
2024-03-11 16:32:29 +08:00
|
|
|
|
<svg v-if="tooltip.type === tooltipType.hexagon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" width="36" height="36"><path d="M747.52 921.088H277.504L42.496 514.048l235.008-407.04H747.52l235.008 407.04-235.008 407.04z m-425.472-76.8h381.44l190.464-330.24-190.464-330.24h-381.44l-190.464 330.24 190.464 330.24z"></path></svg>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
<template v-else-if="tooltip.type === tooltipType.human">
|
|
|
|
|
|
<div class="icon__box">
|
2024-03-11 16:32:29 +08:00
|
|
|
|
<svg viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M366.689524 690.468571l87.283809 83.821715-75.434666 195.486476c-10.971429 27.794286-43.105524 42.081524-72.265143 32.036571-29.159619-10.24-44.080762-41.252571-33.450667-69.241904L366.689524 690.468571zM203.824762 476.306286l51.785143-95.183238a162.279619 162.279619 0 0 1 59.245714-59.977143c119.710476-68.266667 134.777905-67.291429 149.942857-66.218667l80.798476 5.168762c24.868571 0.975238 42.081524 7.314286 125.025524 124.14781a21.26019 21.26019 0 0 0 14.092191 8.289523l99.132952 14.482286c12.873143 1.852952 24.478476 8.582095 32.182857 18.67581a45.494857 45.494857 0 0 1 8.825905 35.108571 46.665143 46.665143 0 0 1-19.504762 30.866286 50.468571 50.468571 0 0 1-36.571429 8.435809l-99.181714-14.433524a119.954286 119.954286 0 0 1-79.774476-47.640381c-4.388571-6.241524-7.558095-11.361524-11.849143-16.579047l-63.634286 193.487238 88.405334 84.845714c12.970667 12.385524 23.698286 29.013333 30.232381 45.494857l67.876571 190.366477c5.022476 13.409524 4.193524 28.233143-2.291809 41.057523a54.613333 54.613333 0 0 1-32.182858 27.160381c-5.851429 2.048-12.092952 3.120762-18.383238 3.169524a58.075429 58.075429 0 0 1-53.930666-36.181333l-67.876572-190.366476c-1.024-2.096762-2.096762-3.120762-3.218285-5.168762L365.616762 623.177143a84.894476 84.894476 0 0 1-24.868572-80.700953l34.523429-146.919619c-3.413333 2.340571-7.070476 4.388571-10.776381 6.290286a58.806857 58.806857 0 0 0-22.674286 22.723048L290.133333 519.801905a49.834667 49.834667 0 0 1-43.105523 24.819809 59.977143 59.977143 0 0 1-22.674286-5.12 46.518857 46.518857 0 0 1-20.48-63.146666z m209.67619-360.448C420.08381 58.465524 473.86819 17.066667 533.650286 23.30819 593.383619 29.549714 636.537905 81.13981 630.00381 138.48381c-6.534095 57.392762-60.269714 98.840381-120.05181 92.550095-59.782095-6.241524-102.985143-57.782857-96.451048-115.175619z"></path></svg>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
<template v-else-if="tooltip.type === tooltipType.baseStation">
|
|
|
|
|
|
<div class="icon__box">
|
2024-03-11 16:32:29 +08:00
|
|
|
|
<svg viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M164.901926 519.585185a391.35763 391.35763 0 0 1-30.151111-152.38637c0-52.527407 10.05037-105.054815 30.151111-152.348445 15.094519-47.29363 45.283556-89.353481 80.516741-126.103703L174.990222 15.17037C129.706667 62.464 94.511407 114.991407 69.328593 178.062222 44.183704 241.095111 34.133333 304.165926 34.133333 367.198815c0 63.070815 10.05037 131.375407 35.19526 189.174518 25.182815 57.799111 60.378074 115.598222 105.661629 162.891852l70.428445-73.576296c-35.233185-36.788148-65.422222-78.810074-80.516741-126.103704z" p-id="8786"></path><path d="M255.469037 477.563259c15.094519 36.788148 35.195259 68.266667 60.340148 94.549334l70.428445-73.576297a233.168593 233.168593 0 0 1-40.201482-57.761185c-5.044148-26.282667-10.088296-47.29363-10.088296-73.576296 0-26.244741 5.044148-47.29363 15.094518-68.266667 5.006222-26.282667 20.100741-42.097778 35.19526-63.070815l-70.428445-73.576296c-25.144889 26.282667-45.24563 57.799111-60.340148 94.587259-15.094519 36.788148-20.100741 73.576296-20.100741 110.364445 0 36.788148 5.006222 73.53837 20.100741 110.326518zM436.527407 367.198815c0 43.538963 33.792 78.810074 75.472593 78.810074s75.472593-35.271111 75.472593-78.810074c0-43.501037-33.792-78.810074-75.472593-78.810074s-75.472593 35.271111-75.472593 78.810074zM637.76237 498.574222l70.428445 73.576297c25.144889-26.282667 45.24563-57.837037 60.340148-94.58726 15.094519-36.788148 20.100741-73.576296 20.100741-110.364444 0-36.788148-5.006222-73.576296-20.100741-110.364445-15.094519-36.750222-35.195259-68.266667-60.340148-94.549333l-70.428445 73.576296c15.094519 15.739259 30.189037 36.788148 40.201482 57.799111 10.088296 21.010963 15.132444 47.29363 15.132444 68.266667 0 26.282667-5.044148 47.331556-15.094518 68.342519-10.05037 31.516444-25.144889 47.29363-40.239408 68.266666z" p-id="8787"></path><path d="M954.671407 178.062222C929.488593 114.991407 894.293333 62.464 849.009778 15.17037L778.619259 88.746667c35.233185 36.788148 60.378074 78.810074 80.516741 126.103703 20.100741 47.255704 30.151111 99.821037 30.151111 152.348445 0 52.565333-10.05037 105.092741-30.151111 152.38637-20.100741 47.29363-45.283556 89.315556-80.516741 126.103704l70.428445 73.576296c45.283556-47.29363 80.478815-99.858963 105.661629-162.891852 25.144889-63.070815 35.195259-126.103704 35.19526-189.174518 0-63.032889-10.05037-131.337481-35.19526-189.136593zM210.185481 1024h603.629038L512 551.10163 210.185481 1024z m186.102519-105.054815L512 740.238222l115.674074 178.631111h-231.348148z"></path></svg>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="header__title">
|
|
|
|
|
|
<template v-if="tooltip.type === tooltipType.hexagon">HEX</template>
|
|
|
|
|
|
<template v-else-if="tooltip.type === tooltipType.human">MSISDN</template>
|
|
|
|
|
|
<template v-else-if="tooltip.type === tooltipType.baseStation">CID</template>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="header__content">
|
|
|
|
|
|
<template v-if="tooltip.type === tooltipType.hexagon">{{currentPolygon.hexId}}</template>
|
2024-03-04 14:23:47 +08:00
|
|
|
|
<template v-else-if="tooltip.type === tooltipType.human">{{currentSubscriber.subscriberDto.phoneNumber}}</template>
|
2024-02-28 19:11:33 +08:00
|
|
|
|
<template v-else-if="tooltip.type === tooltipType.baseStation">0xxa8805</template>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="hexagon-tooltip__body">
|
|
|
|
|
|
<template v-if="tooltip.type === tooltipType.hexagon">
|
2024-03-05 16:17:14 +08:00
|
|
|
|
<template v-if="activeTab === 'locationMap'">
|
|
|
|
|
|
<div class="body__item">
|
|
|
|
|
|
<div class="item__label">{{ $t('location.number') }}</div>
|
|
|
|
|
|
<div class="item__value">{{currentPolygon.number}}</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="body__item">
|
|
|
|
|
|
<div class="item__label">{{ $t('location.locals') }}</div>
|
|
|
|
|
|
<div class="item__value">{{currentPolygon.number}}</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="body__item">
|
|
|
|
|
|
<div class="item__label">{{ $t('location.visitors') }}</div>
|
|
|
|
|
|
<div class="item__value">{{currentPolygon.number}}</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="body__item">
|
|
|
|
|
|
<div class="item__label">{{ $t('location.roamers') }}</div>
|
|
|
|
|
|
<div class="item__value">{{currentPolygon.number}}</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
<template v-else>
|
2024-03-06 12:43:45 +08:00
|
|
|
|
<template v-for="(item, i) in JSON.parse(currentPolygon.locations)" :key="item.hexId">
|
|
|
|
|
|
<div class="body__timeline" v-if="i < 5">
|
|
|
|
|
|
<div class="timeline-symbol"></div>
|
|
|
|
|
|
<div>
|
|
|
|
|
|
<div class="body__item">
|
|
|
|
|
|
<div class="item__label">{{ $t('location.location') }}</div>
|
|
|
|
|
|
<div class="item__value">{{item.longitude}}, {{item.latitude}}</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="body__item">
|
|
|
|
|
|
<div class="item__label">Time</div>
|
|
|
|
|
|
<div class="item__value">{{dateFormatByAppearance(Number(item.time))}}</div>
|
|
|
|
|
|
</div>
|
2024-03-05 16:17:14 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2024-03-06 12:43:45 +08:00
|
|
|
|
</template>
|
|
|
|
|
|
<div class="body__timeline" v-if="JSON.parse(currentPolygon.locations).length > 5">...</div>
|
2024-03-05 16:17:14 +08:00
|
|
|
|
</template>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
</template>
|
|
|
|
|
|
<template v-else-if="tooltip.type === tooltipType.human">
|
2024-03-05 15:16:11 +08:00
|
|
|
|
<div class="body__item">
|
|
|
|
|
|
<div class="item__label">ID</div>
|
|
|
|
|
|
<div class="item__value">{{currentSubscriber.subscriberId}}</div>
|
|
|
|
|
|
</div>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
<div class="body__item">
|
2024-03-04 14:23:47 +08:00
|
|
|
|
<div class="item__label">{{$t('entities.group')}}</div>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
<div class="item__value">Terrorist</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="body__item">
|
2024-03-04 14:23:47 +08:00
|
|
|
|
<div class="item__label">{{$t('overall.info')}}</div>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
<div class="item__value">Leader</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="body__item">
|
2024-03-04 14:23:47 +08:00
|
|
|
|
<div class="item__label">{{$t('overall.location')}}</div>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
<div class="item__value">China, Shanghai</div>
|
|
|
|
|
|
</div>
|
2024-03-04 17:28:05 +08:00
|
|
|
|
<div class="body__tracking" @click="trackSubscriber(currentSubscriber)">{{$t('location.traceTracking')}}</div>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
</template>
|
|
|
|
|
|
<template v-else-if="tooltip.type === tooltipType.baseStation">
|
|
|
|
|
|
<div class="body__item">
|
2024-03-05 10:44:43 +08:00
|
|
|
|
<div class="item__label">{{ $t('location.locationAreaCode') }}</div>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
<div class="item__value">12</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="body__item">
|
2024-03-05 10:44:43 +08:00
|
|
|
|
<div class="item__label">{{ $t('location.mobileNetworkCode') }}</div>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
<div class="item__value">1</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="body__item">
|
2024-03-05 10:44:43 +08:00
|
|
|
|
<div class="item__label">{{ $t('location.communicationType') }}</div>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
<div class="item__value">4G</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="body__item">
|
2024-03-04 14:23:47 +08:00
|
|
|
|
<div class="item__label">{{$t('overall.location')}}</div>
|
2024-02-26 11:51:13 +08:00
|
|
|
|
<div class="item__value">China, Shanghai</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script>
|
|
|
|
|
|
import { ref, shallowRef } from 'vue'
|
2024-03-03 22:10:02 +08:00
|
|
|
|
import { dateFormatByAppearance, getNowTime, getSecond } from '@/utils/date-util'
|
2024-02-26 11:51:13 +08:00
|
|
|
|
import maplibregl from 'maplibre-gl'
|
|
|
|
|
|
import mapStyle from '@/views/charts2/charts/entityDetail/mapStyle'
|
|
|
|
|
|
import 'maplibre-gl/dist/maplibre-gl.css'
|
2024-03-03 22:10:02 +08:00
|
|
|
|
import unitConvert, { valueToRangeValue } from '@/utils/unit-convert'
|
2024-02-29 18:35:43 +08:00
|
|
|
|
import { unitTypes, storageKey } from '@/utils/constants'
|
2024-02-26 11:51:13 +08:00
|
|
|
|
import * as echarts from 'echarts'
|
|
|
|
|
|
import { appListChartOption } from '@/views/charts2/charts/options/echartOption'
|
|
|
|
|
|
import { pieOption } from '@/views/location/chartOption'
|
|
|
|
|
|
import _ from 'lodash'
|
|
|
|
|
|
import { useRoute, useRouter } from 'vue-router'
|
|
|
|
|
|
import { urlParamsHandler, overwriteUrl } from '@/utils/tools'
|
|
|
|
|
|
import axios from 'axios'
|
|
|
|
|
|
import { api } from '@/utils/api'
|
2024-02-28 19:11:33 +08:00
|
|
|
|
import { h3ToGeo, h3ToGeoBoundary } from 'h3-js'
|
2024-03-03 22:10:02 +08:00
|
|
|
|
import TimeLine from '@/components/common/TimeLine.vue'
|
|
|
|
|
|
import SimpleLoading from '@/components/common/SimpleLoading.vue'
|
2024-02-26 11:51:13 +08:00
|
|
|
|
|
2024-03-11 16:32:29 +08:00
|
|
|
|
const humanSvg = '<svg viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg"><path d="M366.689524 690.468571l87.283809 83.821715-75.434666 195.486476c-10.971429 27.794286-43.105524 42.081524-72.265143 32.036571-29.159619-10.24-44.080762-41.252571-33.450667-69.241904L366.689524 690.468571zM203.824762 476.306286l51.785143-95.183238a162.279619 162.279619 0 0 1 59.245714-59.977143c119.710476-68.266667 134.777905-67.291429 149.942857-66.218667l80.798476 5.168762c24.868571 0.975238 42.081524 7.314286 125.025524 124.14781a21.26019 21.26019 0 0 0 14.092191 8.289523l99.132952 14.482286c12.873143 1.852952 24.478476 8.582095 32.182857 18.67581a45.494857 45.494857 0 0 1 8.825905 35.108571 46.665143 46.665143 0 0 1-19.504762 30.866286 50.468571 50.468571 0 0 1-36.571429 8.435809l-99.181714-14.433524a119.954286 119.954286 0 0 1-79.774476-47.640381c-4.388571-6.241524-7.558095-11.361524-11.849143-16.579047l-63.634286 193.487238 88.405334 84.845714c12.970667 12.385524 23.698286 29.013333 30.232381 45.494857l67.876571 190.366477c5.022476 13.409524 4.193524 28.233143-2.291809 41.057523a54.613333 54.613333 0 0 1-32.182858 27.160381c-5.851429 2.048-12.092952 3.120762-18.383238 3.169524a58.075429 58.075429 0 0 1-53.930666-36.181333l-67.876572-190.366476c-1.024-2.096762-2.096762-3.120762-3.218285-5.168762L365.616762 623.177143a84.894476 84.894476 0 0 1-24.868572-80.700953l34.523429-146.919619c-3.413333 2.340571-7.070476 4.388571-10.776381 6.290286a58.806857 58.806857 0 0 0-22.674286 22.723048L290.133333 519.801905a49.834667 49.834667 0 0 1-43.105523 24.819809 59.977143 59.977143 0 0 1-22.674286-5.12 46.518857 46.518857 0 0 1-20.48-63.146666z m209.67619-360.448C420.08381 58.465524 473.86819 17.066667 533.650286 23.30819 593.383619 29.549714 636.537905 81.13981 630.00381 138.48381c-6.534095 57.392762-60.269714 98.840381-120.05181 92.550095-59.782095-6.241524-102.985143-57.782857-96.451048-115.175619z"></path></svg>'
|
|
|
|
|
|
const baseStationSvg = '<svg viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg"><path d="M164.901926 519.585185a391.35763 391.35763 0 0 1-30.151111-152.38637c0-52.527407 10.05037-105.054815 30.151111-152.348445 15.094519-47.29363 45.283556-89.353481 80.516741-126.103703L174.990222 15.17037C129.706667 62.464 94.511407 114.991407 69.328593 178.062222 44.183704 241.095111 34.133333 304.165926 34.133333 367.198815c0 63.070815 10.05037 131.375407 35.19526 189.174518 25.182815 57.799111 60.378074 115.598222 105.661629 162.891852l70.428445-73.576296c-35.233185-36.788148-65.422222-78.810074-80.516741-126.103704z" p-id="8786"></path><path d="M255.469037 477.563259c15.094519 36.788148 35.195259 68.266667 60.340148 94.549334l70.428445-73.576297a233.168593 233.168593 0 0 1-40.201482-57.761185c-5.044148-26.282667-10.088296-47.29363-10.088296-73.576296 0-26.244741 5.044148-47.29363 15.094518-68.266667 5.006222-26.282667 20.100741-42.097778 35.19526-63.070815l-70.428445-73.576296c-25.144889 26.282667-45.24563 57.799111-60.340148 94.587259-15.094519 36.788148-20.100741 73.576296-20.100741 110.364445 0 36.788148 5.006222 73.53837 20.100741 110.326518zM436.527407 367.198815c0 43.538963 33.792 78.810074 75.472593 78.810074s75.472593-35.271111 75.472593-78.810074c0-43.501037-33.792-78.810074-75.472593-78.810074s-75.472593 35.271111-75.472593 78.810074zM637.76237 498.574222l70.428445 73.576297c25.144889-26.282667 45.24563-57.837037 60.340148-94.58726 15.094519-36.788148 20.100741-73.576296 20.100741-110.364444 0-36.788148-5.006222-73.576296-20.100741-110.364445-15.094519-36.750222-35.195259-68.266667-60.340148-94.549333l-70.428445 73.576296c15.094519 15.739259 30.189037 36.788148 40.201482 57.799111 10.088296 21.010963 15.132444 47.29363 15.132444 68.266667 0 26.282667-5.044148 47.331556-15.094518 68.342519-10.05037 31.516444-25.144889 47.29363-40.239408 68.266666z" p-id="8787"></path><path d="M954.671407 178.062222C929.488593 114.991407 894.293333 62.464 849.009778 15.17037L778.619259 88.746667c35.233185 36.788148 60.378074 78.810074 80.516741 126.103703 20.100741 47.255704 30.151111 99.821037 30.151111 152.348445 0 52.565333-10.05037 105.092741-30.151111 152.38637-20.100741 47.29363-45.283556 89.315556-80.516741 126.103704l70.428445 73.576296c45.283556-47.29363 80.478815-99.858963 105.661629-162.891852 25.144889-63.070815 35.195259-126.103704 35.19526-189.174518 0-63.032889-10.05037-131.337481-35.19526-189.136593zM210.185481 1024h603.629038L512 551.10163 210.185481 1024z m186.102519-105.054815L512 740.238222l115.674074 178.631111h-231.348148z"></path></svg>'
|
2024-03-01 18:13:04 +08:00
|
|
|
|
|
2024-02-26 11:51:13 +08:00
|
|
|
|
export default {
|
|
|
|
|
|
name: 'Location',
|
|
|
|
|
|
data () {
|
|
|
|
|
|
return {
|
|
|
|
|
|
tooltipType: {
|
|
|
|
|
|
hexagon: 'hexagon',
|
|
|
|
|
|
baseStation: 'base-station',
|
|
|
|
|
|
human: 'human'
|
2024-02-28 07:38:41 +08:00
|
|
|
|
},
|
|
|
|
|
|
activeCount: '',
|
2024-03-01 17:43:15 +08:00
|
|
|
|
activeCountChain: '',
|
2024-03-01 18:35:50 +08:00
|
|
|
|
curPageNum: 1,
|
2024-03-04 18:21:33 +08:00
|
|
|
|
activeNames: '',
|
2024-03-06 17:13:30 +08:00
|
|
|
|
initFlag: true,
|
2024-03-06 18:04:52 +08:00
|
|
|
|
emptyTip: '',
|
2024-03-08 18:03:21 +08:00
|
|
|
|
opacity: 1,
|
|
|
|
|
|
scrollInfo: {
|
|
|
|
|
|
itemSize: 50, // 一个滚动item高度
|
|
|
|
|
|
containerHeight: 300 // 滚动列表
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-03-01 17:43:15 +08:00
|
|
|
|
components: {
|
2024-03-03 22:10:02 +08:00
|
|
|
|
TimeLine,
|
|
|
|
|
|
SimpleLoading
|
2024-03-01 17:43:15 +08:00
|
|
|
|
},
|
2024-02-26 11:51:13 +08:00
|
|
|
|
methods: {
|
2024-03-03 22:10:02 +08:00
|
|
|
|
dateFormatByAppearance,
|
2024-02-26 11:51:13 +08:00
|
|
|
|
valueToRangeValue,
|
|
|
|
|
|
async initMap () {
|
|
|
|
|
|
const _this = this
|
2024-02-29 18:35:43 +08:00
|
|
|
|
if (!this.mapChart) {
|
|
|
|
|
|
this.mapChart = new maplibregl.Map({
|
|
|
|
|
|
container: 'analysisMap',
|
|
|
|
|
|
style: mapStyle,
|
|
|
|
|
|
center: this.center,
|
|
|
|
|
|
maxZoom: this.maxZoom,
|
|
|
|
|
|
minZoom: this.minZoom,
|
|
|
|
|
|
zoom: this.defaultZoom
|
2024-02-26 11:51:13 +08:00
|
|
|
|
})
|
2024-02-29 18:35:43 +08:00
|
|
|
|
maplibregl.addProtocol('cn', (params, callback) => { // 切片显示接口 防止跨域的问题
|
|
|
|
|
|
fetch(`${params.url.split('://')[1]}`)
|
|
|
|
|
|
.then(t => {
|
|
|
|
|
|
if (t.status == 200) {
|
|
|
|
|
|
t.arrayBuffer().then(arr => {
|
|
|
|
|
|
callback(null, arr, null, null)
|
|
|
|
|
|
})
|
|
|
|
|
|
} else {
|
|
|
|
|
|
callback(new Error(`Tile fetch error: ${t.statusText}`))
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
.catch(e => {
|
|
|
|
|
|
callback(new Error(e))
|
|
|
|
|
|
})
|
|
|
|
|
|
return { cancel: () => { } }
|
2024-02-28 10:21:16 +08:00
|
|
|
|
})
|
2024-02-29 18:35:43 +08:00
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
|
2024-02-29 18:35:43 +08:00
|
|
|
|
this.mapChart.on('load', async function () {
|
|
|
|
|
|
// 加载地图上的基站,基站不随tab的切换而改变
|
2024-02-28 10:21:16 +08:00
|
|
|
|
const baseStationData = await _this.queryBaseStation()
|
2024-02-28 19:11:33 +08:00
|
|
|
|
_this.renderMarker(baseStationData, _this.tooltipType.baseStation)
|
|
|
|
|
|
|
2024-02-29 18:35:43 +08:00
|
|
|
|
if (_this.activeTab === 'locationMap') {
|
|
|
|
|
|
await _this.initLocationMapTab()
|
|
|
|
|
|
} else if (_this.activeTab === 'traceTracking') {
|
|
|
|
|
|
await _this.initTraceTrackingTab()
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
},
|
|
|
|
|
|
async initLocationMapTab () {
|
|
|
|
|
|
// 最先渲染右上角饼图
|
|
|
|
|
|
await this.renderDensityPie()
|
|
|
|
|
|
// 然后渲染地图的色块、基站、人(包括右侧关注列表),最后渲染右上角折线图
|
|
|
|
|
|
/* 地图色块 */
|
|
|
|
|
|
this.updateBoundaryBox()
|
|
|
|
|
|
const hexagonData = await this.queryHexagon()
|
|
|
|
|
|
// 将查到的h3hexagon数据转为geojson
|
2024-03-01 18:13:04 +08:00
|
|
|
|
const polygonSourceData = this.hexagonDataConverter(hexagonData, 'locationMap')
|
2024-02-29 18:35:43 +08:00
|
|
|
|
this.mapChart.addSource('hexGrid', {
|
|
|
|
|
|
type: 'geojson',
|
|
|
|
|
|
data: polygonSourceData
|
|
|
|
|
|
})
|
|
|
|
|
|
// TODO 六边形边框,考虑加一层line layer
|
|
|
|
|
|
this.mapChart.addLayer({
|
|
|
|
|
|
id: 'hexagon',
|
|
|
|
|
|
type: 'fill',
|
|
|
|
|
|
source: 'hexGrid',
|
|
|
|
|
|
layout: {},
|
|
|
|
|
|
paint: {
|
|
|
|
|
|
'fill-color': ['get', 'color'],
|
|
|
|
|
|
'fill-opacity': ['case', ['boolean', ['feature-state', 'hover'], false], 1, 0.6]
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
// 六边形的鼠标事件
|
|
|
|
|
|
this.bindHexagonEvents()
|
|
|
|
|
|
|
|
|
|
|
|
/* 地图上的人 */
|
|
|
|
|
|
const mapFollowedSubscriberData = await this.queryMapFollowedSubscriber()
|
|
|
|
|
|
this.renderMarker(mapFollowedSubscriberData, this.tooltipType.human)
|
2024-02-26 11:51:13 +08:00
|
|
|
|
|
2024-02-29 18:35:43 +08:00
|
|
|
|
/* 右侧关注列表 */
|
2024-03-05 11:19:47 +08:00
|
|
|
|
await this.queryFollowedList()
|
2024-02-29 18:35:43 +08:00
|
|
|
|
/* 右上角折线图 */
|
2024-03-05 11:19:47 +08:00
|
|
|
|
await this.renderActiveSubscribersLine()
|
2024-02-29 18:35:43 +08:00
|
|
|
|
},
|
|
|
|
|
|
async initTraceTrackingTab () {
|
2024-03-01 18:13:04 +08:00
|
|
|
|
await this.queryTraceTracking()
|
|
|
|
|
|
// 如果未指定展示谁的轨迹,则默认取追踪用户中的第一个
|
2024-02-29 18:35:43 +08:00
|
|
|
|
if (!this.currentShowSubscriber && this.trackingSubscribers.length > 0) {
|
2024-03-01 18:13:04 +08:00
|
|
|
|
this.currentShowSubscriber = this.trackingSubscribers[0]
|
|
|
|
|
|
}
|
2024-03-04 14:23:47 +08:00
|
|
|
|
const find = this.trackingSubscribers.find(s => s.subscriberId === this.currentShowSubscriber.subscriberId)
|
|
|
|
|
|
if (find) {
|
2024-03-05 21:57:46 +08:00
|
|
|
|
// 滚动条定位到id所在的dom
|
|
|
|
|
|
const findIndex = this.trackingSubscribers.findIndex(s => s.subscriberId === this.currentShowSubscriber.subscriberId)
|
|
|
|
|
|
const dom = document.getElementById('subscribersBlock')
|
|
|
|
|
|
if (findIndex && dom) {
|
2024-03-06 16:59:07 +08:00
|
|
|
|
await this.$nextTick(() => {
|
2024-03-05 21:57:46 +08:00
|
|
|
|
dom.scrollTop = 207 * findIndex
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
2024-03-04 14:23:47 +08:00
|
|
|
|
}
|
2024-03-03 22:10:02 +08:00
|
|
|
|
this.renderTrackingHexagon()
|
2024-02-26 11:51:13 +08:00
|
|
|
|
},
|
|
|
|
|
|
async renderDensityPie () {
|
|
|
|
|
|
const params = {
|
2024-03-06 12:43:45 +08:00
|
|
|
|
...this.timeFilter,
|
|
|
|
|
|
level: this.mapLevel
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
this.loading.pieLoading = true
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await axios.get(api.location.density, { params })
|
2024-02-28 19:11:33 +08:00
|
|
|
|
const densityData = response.data.data
|
2024-02-26 11:51:13 +08:00
|
|
|
|
// 按值的大小分为5组,并计算各组数量和颜色
|
|
|
|
|
|
this.pieValueRamp = this.calculateValueRamp(densityData)
|
|
|
|
|
|
const option = _.cloneDeep(pieOption)
|
|
|
|
|
|
option.color = this.pieColorRamp.map(c => `rgb(${c})`)
|
2024-03-04 18:21:33 +08:00
|
|
|
|
option.series[0].name = this.$t('location.populationDensity')
|
2024-02-26 11:51:13 +08:00
|
|
|
|
option.series[0].data = this.pieValueRamp.map((r, i) => ({
|
2024-02-28 10:50:45 +08:00
|
|
|
|
name: `${r.start}~${r.end}`,
|
2024-02-26 11:51:13 +08:00
|
|
|
|
value: r.count
|
|
|
|
|
|
}))
|
|
|
|
|
|
this.pieOption = option
|
|
|
|
|
|
if (!this.pieChart) {
|
|
|
|
|
|
this.pieChart = echarts.init(document.getElementById('populationDensityChart'))
|
|
|
|
|
|
}
|
2024-02-28 07:38:41 +08:00
|
|
|
|
this.$nextTick(() => {
|
|
|
|
|
|
this.pieChart.setOption(this.pieOption)
|
|
|
|
|
|
})
|
2024-02-26 11:51:13 +08:00
|
|
|
|
} catch (e) {
|
|
|
|
|
|
this.errorMsgHandler(e)
|
|
|
|
|
|
console.error(e)
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
this.loading.pieLoading = false
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-02-28 07:38:41 +08:00
|
|
|
|
async renderActiveSubscribersLine () {
|
|
|
|
|
|
const params = {
|
2024-03-06 12:43:45 +08:00
|
|
|
|
...this.timeFilter,
|
|
|
|
|
|
level: this.mapLevel
|
2024-02-28 07:38:41 +08:00
|
|
|
|
}
|
|
|
|
|
|
this.loading.lineLoading = true
|
|
|
|
|
|
try {
|
2024-02-28 19:11:33 +08:00
|
|
|
|
const curCountResponse = await axios.get(api.location.count, { params: { ...params, cycle: 0 } })// 当前周期活跃用户总数
|
|
|
|
|
|
const preCountResponse = await axios.get(api.location.count, { params: { ...params, cycle: 1 } })// 上一周期活跃用户总数
|
2024-02-28 07:38:41 +08:00
|
|
|
|
this.activeCount = curCountResponse.data.data.total
|
|
|
|
|
|
const preActiveCount = preCountResponse.data.data.total
|
|
|
|
|
|
if (preActiveCount !== 0) {
|
2024-03-01 18:35:50 +08:00
|
|
|
|
this.activeCountChain = (this.activeCount - preActiveCount) / preActiveCount
|
|
|
|
|
|
} else {
|
|
|
|
|
|
this.activeCountChain = '-'
|
2024-02-28 07:38:41 +08:00
|
|
|
|
}
|
|
|
|
|
|
const trendResponse = await axios.get(api.location.trend, { params })
|
2024-02-28 19:11:33 +08:00
|
|
|
|
const activeSubscribersData = trendResponse.data.data.result
|
2024-02-28 07:38:41 +08:00
|
|
|
|
const option = _.cloneDeep(appListChartOption)
|
|
|
|
|
|
option.series[0].data = activeSubscribersData.map(d => {
|
|
|
|
|
|
return [d[0], d[1], unitTypes.number]
|
|
|
|
|
|
})
|
|
|
|
|
|
this.lineOption = option
|
|
|
|
|
|
if (!this.lineChart) {
|
|
|
|
|
|
this.lineChart = echarts.init(document.getElementById('activeSubscribersChart'))
|
|
|
|
|
|
}
|
|
|
|
|
|
this.lineChart.setOption(this.lineOption)
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
this.errorMsgHandler(e)
|
|
|
|
|
|
console.error(e)
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
this.loading.lineLoading = false
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-03-01 18:35:50 +08:00
|
|
|
|
async queryFollowedList () {
|
|
|
|
|
|
const params = {
|
|
|
|
|
|
...this.timeFilter,
|
2024-03-06 12:43:45 +08:00
|
|
|
|
pageSize: -1,
|
|
|
|
|
|
level: this.mapLevel
|
2024-03-01 18:35:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
this.loading.followSubscriberLoading = true
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await axios.get(api.location.followedSubscriber, { params })
|
|
|
|
|
|
this.followedSubscribersList = response.data.data.list
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
this.errorMsgHandler(e)
|
|
|
|
|
|
console.error(e)
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
this.loading.followSubscriberLoading = false
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-02-26 11:51:13 +08:00
|
|
|
|
async queryHexagon () {
|
|
|
|
|
|
const params = {
|
|
|
|
|
|
...this.boundaryBox,
|
2024-03-06 12:43:45 +08:00
|
|
|
|
...this.timeFilter,
|
|
|
|
|
|
level: this.mapLevel
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
this.loading.hexagonLoading = true
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await axios.get(api.location.map, { params })
|
2024-02-28 19:11:33 +08:00
|
|
|
|
return response.data.data
|
2024-02-26 11:51:13 +08:00
|
|
|
|
} catch (e) {
|
|
|
|
|
|
this.errorMsgHandler(e)
|
|
|
|
|
|
console.error(e)
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
this.loading.hexagonLoading = false
|
|
|
|
|
|
}
|
|
|
|
|
|
return []
|
|
|
|
|
|
},
|
2024-02-28 10:21:16 +08:00
|
|
|
|
async queryBaseStation () {
|
|
|
|
|
|
this.loading.baseStationLoading = true
|
|
|
|
|
|
try {
|
2024-02-28 19:11:33 +08:00
|
|
|
|
// const response = await axios.get(api.location.baseStation)
|
|
|
|
|
|
const response = [
|
|
|
|
|
|
{
|
|
|
|
|
|
longitude: 116.38,
|
|
|
|
|
|
latitude: 39.9
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
longitude: 116.39,
|
|
|
|
|
|
latitude: 39.9
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
longitude: 116.383,
|
|
|
|
|
|
latitude: 39.886
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
longitude: 116.378,
|
|
|
|
|
|
latitude: 39.902
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
longitude: 116.369,
|
|
|
|
|
|
latitude: 39.91
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
longitude: 116.38,
|
|
|
|
|
|
latitude: 39.91
|
|
|
|
|
|
}
|
|
|
|
|
|
]
|
|
|
|
|
|
return response // response.data.data.list
|
2024-02-28 10:21:16 +08:00
|
|
|
|
} catch (e) {
|
|
|
|
|
|
this.errorMsgHandler(e)
|
|
|
|
|
|
console.error(e)
|
|
|
|
|
|
} finally {
|
2024-03-03 22:10:02 +08:00
|
|
|
|
this.loading.baseStationLoading = false
|
2024-02-28 10:21:16 +08:00
|
|
|
|
}
|
|
|
|
|
|
return []
|
|
|
|
|
|
},
|
2024-02-28 19:11:33 +08:00
|
|
|
|
async queryMapFollowedSubscriber () {
|
|
|
|
|
|
this.loading.timeBarLoading = true
|
2024-03-06 12:43:45 +08:00
|
|
|
|
const params = {
|
|
|
|
|
|
...this.minuteTimeFilter,
|
|
|
|
|
|
level: this.mapLevel
|
|
|
|
|
|
}
|
2024-02-28 19:11:33 +08:00
|
|
|
|
try {
|
2024-03-06 12:43:45 +08:00
|
|
|
|
const response = await axios.get(api.location.followedSubscriber, { params })
|
2024-02-28 19:11:33 +08:00
|
|
|
|
return response.data.data.list
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
this.errorMsgHandler(e)
|
|
|
|
|
|
console.error(e)
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
this.loading.timeBarLoading = false
|
|
|
|
|
|
}
|
|
|
|
|
|
return []
|
|
|
|
|
|
},
|
2024-02-29 18:35:43 +08:00
|
|
|
|
async queryTraceTracking () {
|
|
|
|
|
|
if (this.trackingSubscribers.length > 0) {
|
2024-03-03 22:10:02 +08:00
|
|
|
|
this.loading.trackingMapLoading = true
|
2024-02-29 18:35:43 +08:00
|
|
|
|
const params = {
|
|
|
|
|
|
...this.timeFilter,
|
2024-03-06 12:43:45 +08:00
|
|
|
|
subscriberIds: this.trackingSubscribers.map(item => `'${item.subscriberId}'`).join(','),
|
|
|
|
|
|
level: this.mapLevel
|
2024-02-29 18:35:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await axios.get(api.location.tracking, { params })
|
2024-03-06 16:59:07 +08:00
|
|
|
|
const trackingSubscribers = _.cloneDeep(this.trackingSubscribers)
|
2024-03-04 14:23:47 +08:00
|
|
|
|
if (response.data.data.result) {
|
2024-03-06 16:59:07 +08:00
|
|
|
|
trackingSubscribers.forEach(s => {
|
2024-03-04 14:23:47 +08:00
|
|
|
|
const find = response.data.data.result.find(item => item.subscriberId === s.subscriberId)
|
|
|
|
|
|
if (find) {
|
2024-03-06 16:59:07 +08:00
|
|
|
|
this.trackingSubscriberRecordMap[s.subscriberId] = find.trackRecords
|
2024-03-04 14:23:47 +08:00
|
|
|
|
} else {
|
2024-03-06 16:59:07 +08:00
|
|
|
|
this.trackingSubscriberRecordMap[s.subscriberId] = []
|
2024-03-04 14:23:47 +08:00
|
|
|
|
}
|
2024-03-05 21:57:46 +08:00
|
|
|
|
s.show = false // 切换到track tracking时就收起时间线
|
2024-03-04 14:23:47 +08:00
|
|
|
|
})
|
2024-03-06 16:59:07 +08:00
|
|
|
|
this.trackingSubscribers = trackingSubscribers
|
2024-03-04 14:23:47 +08:00
|
|
|
|
} else {
|
2024-03-06 16:59:07 +08:00
|
|
|
|
Object.keys(this.trackingSubscriberRecordMap).forEach(k => {
|
|
|
|
|
|
this.trackingSubscriberRecordMap[k] = []
|
2024-03-04 14:23:47 +08:00
|
|
|
|
})
|
|
|
|
|
|
}
|
2024-03-03 22:10:02 +08:00
|
|
|
|
// 计算停留时间
|
2024-03-08 18:03:21 +08:00
|
|
|
|
this.trackingSubscribers.forEach((s, index) => {
|
2024-03-06 16:59:07 +08:00
|
|
|
|
const trackRecords = this.trackingSubscriberRecordMap[s.subscriberId]
|
2024-03-08 18:03:21 +08:00
|
|
|
|
// 初始化时间线可视范围角标
|
|
|
|
|
|
if (trackRecords.length < 6) {
|
|
|
|
|
|
s.scrollStartIndex = 1
|
|
|
|
|
|
s.scrollEndIndex = trackRecords.length
|
2024-03-06 18:46:35 +08:00
|
|
|
|
} else {
|
2024-03-08 18:03:21 +08:00
|
|
|
|
s.scrollStartIndex = 1
|
|
|
|
|
|
s.scrollEndIndex = 6
|
2024-03-06 18:46:35 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-03-06 16:59:07 +08:00
|
|
|
|
if (trackRecords && trackRecords.length > 0) {
|
|
|
|
|
|
for (let i = 0; i < trackRecords.length; i++) {
|
2024-03-06 16:42:24 +08:00
|
|
|
|
if (i > 0) {
|
2024-03-06 16:59:07 +08:00
|
|
|
|
if ((trackRecords[i - 1].subscriberLongitude === trackRecords[i].subscriberLongitude) && (trackRecords[i - 1].subscriberLatitude === trackRecords[i].subscriberLatitude)) {
|
2024-03-06 16:42:24 +08:00
|
|
|
|
// 如果连续两条地址重复,则将时间累加,并将上一条删除,键值-1继续循环
|
2024-03-06 16:59:07 +08:00
|
|
|
|
if (i > 1 && trackRecords[i - 2]) {
|
|
|
|
|
|
const stayTime = unitConvert(trackRecords[i - 2].time - trackRecords[i].time, unitTypes.time, 's')
|
2024-03-06 16:42:24 +08:00
|
|
|
|
if (Number(stayTime[0]) === Number(Number(stayTime[0]).toFixed(0))) {
|
|
|
|
|
|
stayTime[0] = Number(stayTime[0]).toFixed(0)
|
|
|
|
|
|
}
|
2024-03-06 16:59:07 +08:00
|
|
|
|
trackRecords[i].stayTime = stayTime.join(' ')
|
2024-03-06 16:42:24 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
// 数据只有2条,或者第1条和第2条地点重复,删除1,合并时间
|
2024-03-06 16:59:07 +08:00
|
|
|
|
const stayTime = unitConvert(trackRecords[i - 1].time - trackRecords[i].time, unitTypes.time, 's')
|
2024-03-06 16:42:24 +08:00
|
|
|
|
if (Number(stayTime[0]) === Number(Number(stayTime[0]).toFixed(0))) {
|
|
|
|
|
|
stayTime[0] = Number(stayTime[0]).toFixed(0)
|
|
|
|
|
|
}
|
2024-03-06 16:59:07 +08:00
|
|
|
|
trackRecords[i].stayTime = stayTime.join(' ')
|
2024-03-05 21:57:46 +08:00
|
|
|
|
}
|
2024-03-06 16:59:07 +08:00
|
|
|
|
trackRecords.splice(i - 1, 1)
|
2024-03-06 16:42:24 +08:00
|
|
|
|
i = i - 1
|
2024-03-05 21:57:46 +08:00
|
|
|
|
} else {
|
2024-03-06 16:59:07 +08:00
|
|
|
|
const stayTime = unitConvert(trackRecords[i - 1].time - trackRecords[i].time, unitTypes.time, 's')
|
2024-03-05 21:57:46 +08:00
|
|
|
|
if (Number(stayTime[0]) === Number(Number(stayTime[0]).toFixed(0))) {
|
|
|
|
|
|
stayTime[0] = Number(stayTime[0]).toFixed(0)
|
|
|
|
|
|
}
|
2024-03-06 16:59:07 +08:00
|
|
|
|
trackRecords[i].stayTime = stayTime.join(' ')
|
2024-03-03 22:10:02 +08:00
|
|
|
|
}
|
2024-03-08 18:03:21 +08:00
|
|
|
|
if (i === trackRecords.length - 1) {
|
|
|
|
|
|
// 初始化数据时,重置偏移量和列表高度
|
|
|
|
|
|
s.startOffset = 0
|
|
|
|
|
|
s.listHeight = i * this.scrollInfo.itemSize
|
|
|
|
|
|
}
|
2024-03-03 22:10:02 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
2024-02-29 18:35:43 +08:00
|
|
|
|
} catch (e) {
|
|
|
|
|
|
this.errorMsgHandler(e)
|
|
|
|
|
|
console.error(e)
|
|
|
|
|
|
} finally {
|
2024-03-03 22:10:02 +08:00
|
|
|
|
this.loading.trackingMapLoading = false
|
2024-02-29 18:35:43 +08:00
|
|
|
|
}
|
2024-03-04 17:28:05 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
this.loading.trackingMapLoading = false
|
2024-02-29 18:35:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-03-03 22:10:02 +08:00
|
|
|
|
renderTrackingHexagon () {
|
2024-03-06 10:56:56 +08:00
|
|
|
|
if (!this.currentShowSubscriber) {
|
|
|
|
|
|
return true
|
|
|
|
|
|
}
|
2024-03-06 16:59:07 +08:00
|
|
|
|
const currentShowSubscriberRecords = this.trackingSubscriberRecordMap[this.currentShowSubscriber.subscriberId]
|
|
|
|
|
|
if (currentShowSubscriberRecords && currentShowSubscriberRecords.length > 0) {
|
2024-03-03 22:10:02 +08:00
|
|
|
|
// 六边形
|
2024-03-06 16:59:07 +08:00
|
|
|
|
this.trackingPolygonSourceData = this.hexagonDataConverter(this.trackingSubscriberRecordMap[this.currentShowSubscriber.subscriberId], 'traceTracking')
|
2024-03-03 22:10:02 +08:00
|
|
|
|
this.mapChart.addSource('trackingHexGrid', {
|
|
|
|
|
|
type: 'geojson',
|
2024-03-06 16:59:07 +08:00
|
|
|
|
data: this.trackingPolygonSourceData
|
2024-03-03 22:10:02 +08:00
|
|
|
|
})
|
|
|
|
|
|
this.mapChart.addLayer({
|
|
|
|
|
|
id: 'trackingHexagon',
|
|
|
|
|
|
type: 'fill',
|
|
|
|
|
|
source: 'trackingHexGrid',
|
|
|
|
|
|
layout: {},
|
|
|
|
|
|
paint: {
|
|
|
|
|
|
'fill-color': ['get', 'color'],
|
|
|
|
|
|
'fill-opacity': ['case', ['boolean', ['feature-state', 'hover'], false], 0.75, 0.5]
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
// 轨迹线
|
2024-03-06 16:59:07 +08:00
|
|
|
|
const mapLineSourceData = this.mapLineDataConverter()
|
2024-03-03 22:10:02 +08:00
|
|
|
|
this.mapChart.addSource('trackingLineSource', {
|
|
|
|
|
|
type: 'geojson',
|
|
|
|
|
|
data: mapLineSourceData
|
|
|
|
|
|
})
|
|
|
|
|
|
this.mapChart.addLayer({
|
|
|
|
|
|
id: 'trackingLine',
|
|
|
|
|
|
type: 'line',
|
|
|
|
|
|
source: 'trackingLineSource',
|
|
|
|
|
|
paint: {
|
|
|
|
|
|
'line-color': 'rgba(222, 52, 52, .8)',
|
|
|
|
|
|
'line-width': 3
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
// 最后所在地的图标
|
2024-03-06 16:59:07 +08:00
|
|
|
|
const coordinate = h3ToGeo(currentShowSubscriberRecords[0].hexId)
|
2024-03-03 22:10:02 +08:00
|
|
|
|
this.renderTrackingMarker([coordinate[1], coordinate[0]])
|
|
|
|
|
|
this.bindTrackingHexagonEvents()
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-02-28 19:11:33 +08:00
|
|
|
|
renderMarker (data, type) {
|
|
|
|
|
|
let svg
|
|
|
|
|
|
if (type === this.tooltipType.baseStation) {
|
2024-03-01 18:13:04 +08:00
|
|
|
|
svg = baseStationSvg
|
2024-02-28 19:11:33 +08:00
|
|
|
|
} else if (type === this.tooltipType.human) {
|
2024-03-01 18:13:04 +08:00
|
|
|
|
svg = humanSvg
|
2024-02-28 19:11:33 +08:00
|
|
|
|
}
|
2024-03-04 14:23:47 +08:00
|
|
|
|
try {
|
|
|
|
|
|
data.forEach(marker => {
|
2024-03-05 11:19:47 +08:00
|
|
|
|
if (type === this.tooltipType.human && marker.subscriberDto) {
|
2024-03-04 14:23:47 +08:00
|
|
|
|
const el = document.createElement('div')
|
|
|
|
|
|
el.className = `map-marker map-marker--${type}`
|
2024-03-11 16:32:29 +08:00
|
|
|
|
if (marker.subscriberId === this.highlightSubscriber.subscriberId) {
|
|
|
|
|
|
el.classList.add('map-marker--highlight')
|
|
|
|
|
|
}
|
2024-03-04 14:23:47 +08:00
|
|
|
|
el.innerHTML = svg
|
|
|
|
|
|
// 鼠标事件,控制tooltip显示和marker尺寸
|
|
|
|
|
|
this.bindMarkerEvent(el, marker, type)
|
|
|
|
|
|
const mapMarker = new maplibregl.Marker({ element: el })
|
2024-03-05 11:19:47 +08:00
|
|
|
|
.setLngLat([marker.subscriberDto.subscriberLongitude, marker.subscriberDto.subscriberLatitude])
|
2024-03-04 14:23:47 +08:00
|
|
|
|
.addTo(this.mapChart)
|
2024-03-05 16:42:04 +08:00
|
|
|
|
mapMarker.subscriberId = marker.subscriberId
|
2024-03-04 14:23:47 +08:00
|
|
|
|
this.humanMarkers.push(mapMarker)
|
|
|
|
|
|
} else if (type === this.tooltipType.baseStation) {
|
|
|
|
|
|
const el = document.createElement('div')
|
|
|
|
|
|
el.className = `map-marker map-marker--${type}`
|
|
|
|
|
|
el.innerHTML = svg
|
|
|
|
|
|
// 鼠标事件,控制tooltip显示和marker尺寸
|
|
|
|
|
|
this.bindMarkerEvent(el, marker, type)
|
|
|
|
|
|
const mapMarker = new maplibregl.Marker({ element: el })
|
|
|
|
|
|
.setLngLat([marker.longitude, marker.latitude])
|
|
|
|
|
|
.addTo(this.mapChart)
|
|
|
|
|
|
this.baseStationMarkers.push(mapMarker)
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
console.error(e)
|
|
|
|
|
|
}
|
2024-02-28 19:11:33 +08:00
|
|
|
|
},
|
2024-03-01 18:13:04 +08:00
|
|
|
|
renderTrackingMarker (coordinates) {
|
|
|
|
|
|
const el = document.createElement('div')
|
|
|
|
|
|
el.className = 'map-tracking-marker'
|
|
|
|
|
|
el.innerHTML = `<div class="tracking-marker__inner-circle">${humanSvg}</div>`
|
|
|
|
|
|
this.trackingHumanMarker = new maplibregl.Marker({ element: el })
|
|
|
|
|
|
.setLngLat(coordinates)
|
|
|
|
|
|
.addTo(this.mapChart)
|
|
|
|
|
|
},
|
2024-02-28 19:11:33 +08:00
|
|
|
|
updateBoundaryBox () {
|
2024-03-06 18:46:35 +08:00
|
|
|
|
/* const boundaryBox = this.mapChart.getBounds()
|
|
|
|
|
|
this.boundaryBox = {
|
2024-02-28 19:11:33 +08:00
|
|
|
|
maxLongitude: boundaryBox.getEast(),
|
|
|
|
|
|
maxLatitude: boundaryBox.getNorth(),
|
|
|
|
|
|
minLongitude: boundaryBox.getWest(),
|
|
|
|
|
|
minLatitude: boundaryBox.getSouth()
|
2024-02-29 18:35:43 +08:00
|
|
|
|
} */
|
2024-02-28 19:11:33 +08:00
|
|
|
|
this.boundaryBox = {
|
|
|
|
|
|
maxLongitude: 140,
|
|
|
|
|
|
maxLatitude: 50,
|
|
|
|
|
|
minLongitude: 100,
|
|
|
|
|
|
minLatitude: 30
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-02-26 11:51:13 +08:00
|
|
|
|
// 先使用min=0的等宽分组法,若后续出现特大或特小的异常值导致等宽分组效果不理想,考虑用分位数分组法
|
|
|
|
|
|
calculateValueRamp (data) {
|
2024-02-28 19:11:33 +08:00
|
|
|
|
const max = _.maxBy(data, d => Number(d.number))
|
2024-02-26 11:51:13 +08:00
|
|
|
|
const result = []
|
2024-02-29 18:35:43 +08:00
|
|
|
|
if (max) {
|
2024-03-06 12:43:45 +08:00
|
|
|
|
if (max.number < 10) {
|
|
|
|
|
|
for (let i = 1; i <= 5; i++) {
|
|
|
|
|
|
const item = {
|
|
|
|
|
|
start: i * 2 - 1,
|
|
|
|
|
|
end: i * 2,
|
|
|
|
|
|
color: this.pieColorRamp[i - 1]
|
|
|
|
|
|
}
|
|
|
|
|
|
item.count = data.filter(d => d.number >= item.start && d.number < item.end).length
|
|
|
|
|
|
result.push(item)
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
const maxLength = String(max.number).length
|
|
|
|
|
|
const maxLegend = Math.ceil(max.number / Math.pow(10, maxLength - 1)) * Math.pow(10, maxLength - 1)
|
|
|
|
|
|
for (let i = 1; i <= 5; i++) {
|
|
|
|
|
|
const item = {
|
|
|
|
|
|
start: maxLegend * (i - 1) / 5 + 1,
|
|
|
|
|
|
end: maxLegend * i / 5,
|
|
|
|
|
|
color: this.pieColorRamp[i - 1]
|
|
|
|
|
|
}
|
2024-03-06 14:23:48 +08:00
|
|
|
|
item.count = data.filter(d => d.number >= item.start && d.number <= item.end).length
|
2024-03-06 12:43:45 +08:00
|
|
|
|
result.push(item)
|
2024-02-29 18:35:43 +08:00
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return result
|
|
|
|
|
|
},
|
2024-03-01 18:13:04 +08:00
|
|
|
|
hexagonDataConverter (data, tab) {
|
|
|
|
|
|
const featureCollection = { type: 'FeatureCollection' }
|
|
|
|
|
|
if (tab === 'locationMap') {
|
|
|
|
|
|
featureCollection.features = data.map((d, i) => ({
|
|
|
|
|
|
id: i + 1,
|
|
|
|
|
|
type: 'Feature',
|
|
|
|
|
|
geometry: {
|
|
|
|
|
|
type: 'Polygon',
|
|
|
|
|
|
coordinates: [
|
|
|
|
|
|
h3ToGeoBoundary(d.hexId, true)
|
|
|
|
|
|
]
|
|
|
|
|
|
},
|
|
|
|
|
|
properties: {
|
|
|
|
|
|
hexId: d.hexId,
|
|
|
|
|
|
number: d.number,
|
|
|
|
|
|
color: this.getHexagonFillColor(d.number)
|
|
|
|
|
|
}
|
|
|
|
|
|
}))
|
|
|
|
|
|
} else if (tab === 'traceTracking') {
|
|
|
|
|
|
// 对hexId去重,将重复的hexId的时间合并
|
|
|
|
|
|
const hexagons = []
|
|
|
|
|
|
data.forEach(d => {
|
|
|
|
|
|
const find = hexagons.find(h => h.hexId === d.hexId)
|
|
|
|
|
|
if (!find) {
|
2024-03-05 16:17:14 +08:00
|
|
|
|
hexagons.push({ hexId: d.hexId, locations: [{ time: d.time, longitude: d.subscriberLongitude, latitude: d.subscriberLatitude }] })
|
2024-03-01 18:13:04 +08:00
|
|
|
|
} else {
|
2024-03-07 15:13:30 +08:00
|
|
|
|
if (find.locations.length < 6) {
|
2024-03-06 12:43:45 +08:00
|
|
|
|
find.locations.push({ time: d.time, longitude: d.subscriberLongitude, latitude: d.subscriberLatitude })
|
|
|
|
|
|
}
|
2024-03-01 18:13:04 +08:00
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
featureCollection.features = hexagons.map((d, i) => ({
|
|
|
|
|
|
id: i + 100000,
|
|
|
|
|
|
type: 'Feature',
|
|
|
|
|
|
geometry: {
|
|
|
|
|
|
type: 'Polygon',
|
|
|
|
|
|
coordinates: [
|
|
|
|
|
|
h3ToGeoBoundary(d.hexId, true)
|
|
|
|
|
|
]
|
|
|
|
|
|
},
|
|
|
|
|
|
properties: {
|
|
|
|
|
|
hexId: d.hexId,
|
|
|
|
|
|
number: d.number,
|
2024-03-05 16:17:14 +08:00
|
|
|
|
locations: d.locations,
|
2024-03-01 18:13:04 +08:00
|
|
|
|
color: [37, 55, 128]
|
|
|
|
|
|
}
|
|
|
|
|
|
}))
|
2024-02-28 10:21:16 +08:00
|
|
|
|
}
|
2024-03-01 18:13:04 +08:00
|
|
|
|
return featureCollection
|
2024-02-28 10:21:16 +08:00
|
|
|
|
},
|
2024-02-26 17:28:45 +08:00
|
|
|
|
getHexagonFillColor (number) {
|
2024-02-28 19:11:33 +08:00
|
|
|
|
const ramp = this.pieValueRamp.filter(r => Number(number) >= r.start && Number(number) <= r.end)
|
2024-02-26 17:28:45 +08:00
|
|
|
|
if (ramp.length > 0) {
|
|
|
|
|
|
return ramp[0].color.split(',').map(n => Number(n))
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
2024-03-01 18:13:04 +08:00
|
|
|
|
return [229, 229, 229]
|
|
|
|
|
|
},
|
2024-03-06 16:59:07 +08:00
|
|
|
|
mapLineDataConverter () {
|
|
|
|
|
|
const records = this.trackingSubscriberRecordMap[this.currentShowSubscriber.subscriberId]
|
2024-03-01 18:13:04 +08:00
|
|
|
|
const feature = {
|
|
|
|
|
|
type: 'Feature',
|
|
|
|
|
|
geometry: {
|
|
|
|
|
|
type: 'LineString',
|
2024-03-06 16:59:07 +08:00
|
|
|
|
coordinates: records.map(d => {
|
|
|
|
|
|
const cs = h3ToGeo(d.hexId)
|
2024-03-01 18:13:04 +08:00
|
|
|
|
return [cs[1], cs[0]]
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return {
|
|
|
|
|
|
type: 'FeatureCollection',
|
|
|
|
|
|
features: [feature]
|
|
|
|
|
|
}
|
2024-02-26 17:28:45 +08:00
|
|
|
|
},
|
2024-03-06 12:43:45 +08:00
|
|
|
|
hoverTrigger (source, id, hover) {
|
|
|
|
|
|
this.mapChart.setFeatureState({ source, id }, { hover })
|
|
|
|
|
|
},
|
2024-02-28 10:21:16 +08:00
|
|
|
|
bindHexagonEvents () {
|
2024-02-29 18:35:43 +08:00
|
|
|
|
this.mapChart.on('mouseenter', 'hexagon', this.hexagonMouseEnter)
|
|
|
|
|
|
this.mapChart.on('mouseleave', 'hexagon', this.hexagonMouseLeave)
|
|
|
|
|
|
this.mapChart.on('mousemove', 'hexagon', this.hexagonMouseMove)
|
|
|
|
|
|
},
|
|
|
|
|
|
unbindHexagonEvents () {
|
|
|
|
|
|
this.mapChart.off('mouseenter', this.hexagonMouseEnter)
|
|
|
|
|
|
this.mapChart.off('mouseleave', this.hexagonMouseLeave)
|
|
|
|
|
|
this.mapChart.off('mousemove', this.hexagonMouseMove)
|
|
|
|
|
|
},
|
2024-03-01 18:13:04 +08:00
|
|
|
|
bindTrackingHexagonEvents () {
|
|
|
|
|
|
this.mapChart.on('mouseenter', 'trackingHexagon', this.trackingHexagonMouseEnter)
|
|
|
|
|
|
this.mapChart.on('mouseleave', 'trackingHexagon', this.trackingHexagonMouseLeave)
|
|
|
|
|
|
this.mapChart.on('mousemove', 'trackingHexagon', this.trackingHexagonMouseMove)
|
|
|
|
|
|
},
|
|
|
|
|
|
unbindTrackingHexagonEvents () {
|
|
|
|
|
|
this.mapChart.off('mouseenter', this.trackingHexagonMouseEnter)
|
|
|
|
|
|
this.mapChart.off('mouseleave', this.trackingHexagonMouseLeave)
|
|
|
|
|
|
this.mapChart.off('mousemove', this.trackingHexagonMouseMove)
|
|
|
|
|
|
},
|
2024-02-29 18:35:43 +08:00
|
|
|
|
hexagonMouseEnter () {
|
|
|
|
|
|
this.tooltip.mouseIsInPolygon = true
|
|
|
|
|
|
},
|
|
|
|
|
|
hexagonMouseLeave () {
|
|
|
|
|
|
this.tooltip.showPolygonTooltip = false
|
|
|
|
|
|
this.tooltip.mouseIsInPolygon = false
|
|
|
|
|
|
// 去掉上一块的高亮
|
2024-03-06 12:43:45 +08:00
|
|
|
|
this.hoverTrigger('hexGrid', this.currentPolygon.id, false)
|
2024-02-29 18:35:43 +08:00
|
|
|
|
},
|
|
|
|
|
|
hexagonMouseMove (e) {
|
|
|
|
|
|
const { originalEvent, features } = e
|
|
|
|
|
|
if (!this.tooltip.mouseInMarkerOrTooltip) {
|
|
|
|
|
|
this.tooltip.showPolygonTooltip = true
|
|
|
|
|
|
this.tooltip.type = this.tooltipType.hexagon
|
|
|
|
|
|
if (this.tooltip.type === this.tooltipType.hexagon && this.currentPolygon.id && this.currentPolygon.id !== features[0].id) {
|
|
|
|
|
|
// 去掉上一块的高亮
|
2024-03-06 12:43:45 +08:00
|
|
|
|
this.hoverTrigger('hexGrid', this.currentPolygon.id, false)
|
2024-02-28 10:21:16 +08:00
|
|
|
|
}
|
2024-02-29 18:35:43 +08:00
|
|
|
|
this.currentPolygon = features[0].properties
|
|
|
|
|
|
this.currentPolygon.id = features[0].id
|
|
|
|
|
|
this.tooltip.x = originalEvent.clientX + 15
|
2024-03-06 12:43:45 +08:00
|
|
|
|
this.tooltip.y = (originalEvent.clientY + 5 + this.tooltipDomHeight.hexagon) > this.mapDomHeight ? (this.mapDomHeight - this.tooltipDomHeight.hexagon) : (originalEvent.clientY + 5)
|
2024-02-29 18:35:43 +08:00
|
|
|
|
|
|
|
|
|
|
// 鼠标滑过高亮
|
2024-03-06 12:43:45 +08:00
|
|
|
|
this.hoverTrigger('hexGrid', this.currentPolygon.id, true)
|
2024-02-26 17:28:45 +08:00
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
},
|
2024-03-01 18:13:04 +08:00
|
|
|
|
trackingHexagonMouseEnter () {
|
|
|
|
|
|
this.tooltip.mouseIsInPolygon = true
|
|
|
|
|
|
},
|
|
|
|
|
|
trackingHexagonMouseLeave () {
|
|
|
|
|
|
this.tooltip.showPolygonTooltip = false
|
|
|
|
|
|
this.tooltip.mouseIsInPolygon = false
|
|
|
|
|
|
// 去掉上一块的高亮
|
2024-03-06 12:43:45 +08:00
|
|
|
|
this.hoverTrigger('trackingHexGrid', this.currentPolygon.id, false)
|
2024-03-01 18:13:04 +08:00
|
|
|
|
},
|
|
|
|
|
|
trackingHexagonMouseMove (e) {
|
|
|
|
|
|
const { originalEvent, features } = e
|
|
|
|
|
|
if (!this.tooltip.mouseInMarkerOrTooltip) {
|
|
|
|
|
|
this.tooltip.showPolygonTooltip = true
|
|
|
|
|
|
this.tooltip.type = this.tooltipType.hexagon
|
|
|
|
|
|
if (this.tooltip.type === this.tooltipType.hexagon && this.currentPolygon.id && this.currentPolygon.id !== features[0].id) {
|
|
|
|
|
|
// 去掉上一块的高亮
|
2024-03-06 12:43:45 +08:00
|
|
|
|
this.hoverTrigger('trackingHexGrid', this.currentPolygon.id, false)
|
2024-03-01 18:13:04 +08:00
|
|
|
|
}
|
|
|
|
|
|
this.currentPolygon = features[0].properties
|
|
|
|
|
|
this.currentPolygon.id = features[0].id
|
2024-03-05 16:17:14 +08:00
|
|
|
|
this.currentPolygon.location = `${h3ToGeo(this.currentPolygon.hexId)[1]}, ${h3ToGeo(this.currentPolygon.hexId)[0]}`
|
2024-03-01 18:13:04 +08:00
|
|
|
|
this.tooltip.x = originalEvent.clientX + 15
|
|
|
|
|
|
this.tooltip.y = originalEvent.clientY + 5
|
2024-03-06 12:43:45 +08:00
|
|
|
|
/* this.$nextTick(() => {
|
|
|
|
|
|
const tooltipDom = document.getElementById('tooltip')
|
|
|
|
|
|
const tooltipDomHeight = tooltipDom.offsetHeight
|
|
|
|
|
|
this.tooltip.x = originalEvent.clientX + 15
|
|
|
|
|
|
this.tooltip.y = originalEvent.clientY + 5
|
|
|
|
|
|
this.tooltip.y = (originalEvent.clientY + 5 + tooltipDomHeight) > this.mapDomHeight ? (this.mapDomHeight - tooltipDomHeight) : (originalEvent.clientY + 5)
|
2024-03-06 18:46:35 +08:00
|
|
|
|
}) */
|
2024-03-01 18:13:04 +08:00
|
|
|
|
|
|
|
|
|
|
// 鼠标滑过高亮
|
2024-03-06 12:43:45 +08:00
|
|
|
|
this.hoverTrigger('trackingHexGrid', this.currentPolygon.id, true)
|
2024-03-01 18:13:04 +08:00
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-03-04 14:23:47 +08:00
|
|
|
|
bindMarkerEvent (el, markerData, type) {
|
2024-02-29 18:35:43 +08:00
|
|
|
|
el.addEventListener('mouseenter', e => {
|
|
|
|
|
|
this.currentMarkerDom = el
|
2024-03-04 14:23:47 +08:00
|
|
|
|
if (type === this.tooltipType.human) {
|
|
|
|
|
|
this.currentSubscriber = markerData
|
2024-03-06 12:43:45 +08:00
|
|
|
|
if (!this.tooltip.mouseInMarkerOrTooltip) {
|
|
|
|
|
|
this.tooltip.x = e.clientX + 15 - e.offsetX
|
|
|
|
|
|
this.tooltip.y = (e.clientY + 15 - e.offsetY + this.tooltipDomHeight.human) > this.mapDomHeight ? (this.mapDomHeight - this.tooltipDomHeight.human) : (e.clientY + 15 - e.offsetY)
|
|
|
|
|
|
}
|
2024-03-04 14:23:47 +08:00
|
|
|
|
} else if (type === this.tooltipType.baseStation) {
|
|
|
|
|
|
this.currentBaseStation = markerData
|
2024-03-06 12:43:45 +08:00
|
|
|
|
if (!this.tooltip.mouseInMarkerOrTooltip) {
|
|
|
|
|
|
this.tooltip.x = e.clientX + 15 - e.offsetX
|
|
|
|
|
|
this.tooltip.y = (e.clientY + 15 - e.offsetY + this.tooltipDomHeight.baseStation) > this.mapDomHeight ? (this.mapDomHeight - this.tooltipDomHeight.baseStation) : (e.clientY + 15 - e.offsetY)
|
|
|
|
|
|
}
|
2024-03-04 14:23:47 +08:00
|
|
|
|
}
|
2024-03-06 12:43:45 +08:00
|
|
|
|
|
2024-02-29 18:35:43 +08:00
|
|
|
|
this.tooltip.mouseInMarkerOrTooltip = true
|
2024-03-04 14:23:47 +08:00
|
|
|
|
this.tooltip.type = type
|
2024-02-29 18:35:43 +08:00
|
|
|
|
this.tooltip.showMarkerTooltip = true
|
|
|
|
|
|
el.classList.add('map-marker--hover')
|
|
|
|
|
|
})
|
|
|
|
|
|
el.addEventListener('mouseleave', event => {
|
|
|
|
|
|
const tooltipDom = document.getElementById('tooltip')
|
|
|
|
|
|
if (!tooltipDom.contains(event.relatedTarget)) {
|
|
|
|
|
|
el.classList.remove('map-marker--hover')
|
|
|
|
|
|
this.tooltip.mouseInMarkerOrTooltip = false
|
|
|
|
|
|
this.tooltip.showMarkerTooltip = false
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
2024-03-11 16:32:29 +08:00
|
|
|
|
if (type === this.tooltipType.human) {
|
|
|
|
|
|
el.addEventListener('click', e => {
|
|
|
|
|
|
this.humanMarkers.forEach(m => {
|
|
|
|
|
|
m.getElement().classList.remove('map-marker--highlight')
|
|
|
|
|
|
})
|
|
|
|
|
|
if (this.highlightSubscriber.subscriberId !== markerData.subscriberId) {
|
|
|
|
|
|
el.classList.add('map-marker--highlight')
|
|
|
|
|
|
this.highlightSubscriber = markerData
|
|
|
|
|
|
// 将滚动条跳转到对应位置
|
|
|
|
|
|
document.querySelector(`#locationMap-subscriberId-${markerData.subscriberId}`).scrollIntoView({ behavior: 'smooth', block: 'center' })
|
|
|
|
|
|
} else {
|
|
|
|
|
|
this.highlightSubscriber = {}
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
2024-02-29 18:35:43 +08:00
|
|
|
|
},
|
2024-02-26 11:51:13 +08:00
|
|
|
|
tooltipMouseEnter () {
|
2024-02-28 19:11:33 +08:00
|
|
|
|
this.tooltip.mouseInMarkerOrTooltip = true
|
2024-02-26 11:51:13 +08:00
|
|
|
|
},
|
2024-02-28 19:11:33 +08:00
|
|
|
|
tooltipMouseLeave (event) {
|
2024-02-29 18:35:43 +08:00
|
|
|
|
if (this.currentMarkerDom && !this.currentMarkerDom.contains(event.relatedTarget)) {
|
2024-02-28 19:11:33 +08:00
|
|
|
|
this.tooltip.mouseInMarkerOrTooltip = false
|
|
|
|
|
|
this.tooltip.showMarkerTooltip = false
|
2024-02-29 18:35:43 +08:00
|
|
|
|
this.currentMarkerDom.classList.remove('map-marker--hover')
|
2024-02-28 19:11:33 +08:00
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
},
|
|
|
|
|
|
reload (startTime, endTime, dateRangeValue) {
|
|
|
|
|
|
this.timeFilter = { startTime: getSecond(startTime), endTime: getSecond(endTime), dateRangeValue: dateRangeValue }
|
|
|
|
|
|
const { query } = this.$route
|
|
|
|
|
|
this.$store.commit('setTimeRangeArray', [this.timeFilter.startTime, this.timeFilter.endTime])
|
|
|
|
|
|
this.$store.commit('setTimeRangeFlag', dateRangeValue.value)
|
|
|
|
|
|
|
|
|
|
|
|
const newUrl = urlParamsHandler(window.location.href, query, {
|
|
|
|
|
|
startTime: this.timeFilter.startTime,
|
|
|
|
|
|
endTime: this.timeFilter.endTime,
|
|
|
|
|
|
range: dateRangeValue.value
|
|
|
|
|
|
})
|
|
|
|
|
|
overwriteUrl(newUrl)
|
|
|
|
|
|
},
|
2024-03-06 12:43:45 +08:00
|
|
|
|
timelineMouseEnter (subscriber, record) {
|
2024-03-08 18:18:20 +08:00
|
|
|
|
this.trackingPolygonSourceData.features.forEach(f => {
|
|
|
|
|
|
this.hoverTrigger('trackingHexGrid', f.id, false)
|
|
|
|
|
|
})
|
2024-03-06 12:43:45 +08:00
|
|
|
|
if (this.currentShowSubscriber.subscriberId === subscriber.subscriberId) {
|
|
|
|
|
|
const find = this.trackingPolygonSourceData.features.find(d => d.properties.hexId === record.hexId)
|
|
|
|
|
|
if (find) {
|
|
|
|
|
|
this.hoverTrigger('trackingHexGrid', find.id, true)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
timelineMouseLeave (subscriber, record) {
|
|
|
|
|
|
if (this.currentShowSubscriber.subscriberId === subscriber.subscriberId) {
|
|
|
|
|
|
const find = this.trackingPolygonSourceData.features.find(d => d.properties.hexId === record.hexId)
|
|
|
|
|
|
if (find) {
|
|
|
|
|
|
this.hoverTrigger('trackingHexGrid', find.id, false)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-03-11 16:32:29 +08:00
|
|
|
|
// subscriber列表点击后将地图上的人图标保持特殊高亮
|
|
|
|
|
|
subscriberListClick (subscriber) {
|
|
|
|
|
|
// 先删除当前所有高亮的
|
|
|
|
|
|
this.humanMarkers.forEach(m => {
|
|
|
|
|
|
m.getElement().classList.remove('map-marker--highlight')
|
|
|
|
|
|
})
|
|
|
|
|
|
if (this.highlightSubscriber.subscriberId === subscriber.subscriberId) {
|
|
|
|
|
|
this.highlightSubscriber = {}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
this.highlightSubscriber = subscriber
|
|
|
|
|
|
const target = this.humanMarkers.find(m => subscriber.subscriberId === m.subscriberId)
|
|
|
|
|
|
if (target) {
|
|
|
|
|
|
target.getElement().classList.add('map-marker--highlight')
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-03-06 17:44:13 +08:00
|
|
|
|
subscriberListMouseEnter (subscriber) {
|
|
|
|
|
|
const target = this.humanMarkers.find(m => subscriber.subscriberId === m.subscriberId)
|
|
|
|
|
|
if (target) {
|
|
|
|
|
|
target.getElement().classList.add('map-marker--hover')
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
subscriberListMouseLeave (subscriber) {
|
|
|
|
|
|
const target = this.humanMarkers.find(m => subscriber.subscriberId === m.subscriberId)
|
|
|
|
|
|
if (target) {
|
|
|
|
|
|
target.getElement().classList.remove('map-marker--hover')
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-02-26 11:51:13 +08:00
|
|
|
|
timeRefreshChange () {
|
|
|
|
|
|
// 不是自选时间
|
|
|
|
|
|
if (this.$refs.dateTimeRange) {
|
|
|
|
|
|
if (!this.$refs.dateTimeRange.isCustom) {
|
|
|
|
|
|
const value = this.timeFilter.dateRangeValue
|
|
|
|
|
|
this.$refs.dateTimeRange.quickChange(value)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
this.timeFilter = JSON.parse(JSON.stringify(this.timeFilter))
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
this.timeFilter = JSON.parse(JSON.stringify(this.timeFilter))
|
|
|
|
|
|
}
|
2024-03-01 17:43:15 +08:00
|
|
|
|
},
|
|
|
|
|
|
clickTrackBlock (i) {
|
2024-03-06 18:46:35 +08:00
|
|
|
|
const length = this.trackingSubscriberRecordMap[this.trackingSubscribers[i].subscriberId].length
|
|
|
|
|
|
if (length > 1) {
|
2024-03-05 21:57:46 +08:00
|
|
|
|
this.trackingSubscribers[i].show = !this.trackingSubscribers[i].show
|
|
|
|
|
|
if (this.trackingSubscribers[i].show) {
|
|
|
|
|
|
this.trackingSubscribers[i].showLine = true
|
2024-03-08 18:03:21 +08:00
|
|
|
|
this.trackingSubscribers[i].scrollStartIndex = 1
|
|
|
|
|
|
this.trackingSubscribers[i].scrollEndIndex = 6
|
|
|
|
|
|
this.trackingSubscribers[i].startOffset = 0
|
|
|
|
|
|
this.trackingSubscribers[i].listHeight = 0
|
|
|
|
|
|
// 高度置为0,是为了切换时间后再打开时间线,让滚动条置顶
|
|
|
|
|
|
const timer = setTimeout(() => {
|
|
|
|
|
|
this.trackingSubscribers[i].listHeight = this.trackingSubscriberRecordMap[this.trackingSubscribers[i].subscriberId].length * this.scrollInfo.itemSize
|
|
|
|
|
|
clearTimeout(timer)
|
|
|
|
|
|
}, 100)
|
2024-03-05 21:57:46 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
const timer = setTimeout(() => {
|
|
|
|
|
|
this.trackingSubscribers[i].showLine = false
|
|
|
|
|
|
clearTimeout(timer)
|
|
|
|
|
|
}, 200)
|
|
|
|
|
|
}
|
2024-03-01 17:43:15 +08:00
|
|
|
|
}
|
2024-03-01 18:35:50 +08:00
|
|
|
|
},
|
2024-03-03 22:10:02 +08:00
|
|
|
|
changeCurrentShowSubscriber (subscriber) {
|
2024-03-04 17:28:05 +08:00
|
|
|
|
if (subscriber.subscriberId !== this.currentShowSubscriber.subscriberId) {
|
|
|
|
|
|
this.currentShowSubscriber = subscriber
|
|
|
|
|
|
}
|
2024-03-03 22:10:02 +08:00
|
|
|
|
},
|
2024-03-05 15:16:11 +08:00
|
|
|
|
// 关注列表的添加、删除追踪
|
|
|
|
|
|
addOrRemoveTrackingSubscriber (subscriber) {
|
|
|
|
|
|
const find = this.trackingSubscribers.find(s => s.subscriberId === subscriber.subscriberId)
|
|
|
|
|
|
if (find) {
|
|
|
|
|
|
const index = this.trackingSubscribers.findIndex(s => s.subscriberId === subscriber.subscriberId)
|
|
|
|
|
|
this.trackingSubscribers.splice(index, 1)
|
|
|
|
|
|
} else {
|
2024-03-08 18:03:21 +08:00
|
|
|
|
this.trackingSubscribers.push({ ...subscriber, show: false, showLine: false, scrollStartIndex: 1, scrollEndIndex: 6, startOffset: 0, listHeight: 0 })
|
2024-03-05 15:16:11 +08:00
|
|
|
|
}
|
2024-03-06 18:04:52 +08:00
|
|
|
|
this.opacity = 0
|
|
|
|
|
|
setInterval(() => {
|
|
|
|
|
|
this.opacity += 0.05
|
2024-03-06 18:46:35 +08:00
|
|
|
|
if (this.opacity >= 1) {
|
2024-03-06 18:04:52 +08:00
|
|
|
|
this.opacity = 1
|
|
|
|
|
|
}
|
2024-03-06 18:46:35 +08:00
|
|
|
|
}, 16)
|
2024-03-05 15:16:11 +08:00
|
|
|
|
},
|
|
|
|
|
|
// 追踪页删除追踪
|
|
|
|
|
|
removeTrackingSubscriber (subscriber) {
|
|
|
|
|
|
const find = this.trackingSubscribers.find(s => s.subscriberId === subscriber.subscriberId)
|
|
|
|
|
|
if (find) {
|
|
|
|
|
|
const index = this.trackingSubscribers.findIndex(s => s.subscriberId === subscriber.subscriberId)
|
|
|
|
|
|
this.trackingSubscribers.splice(index, 1)
|
|
|
|
|
|
if (subscriber.subscriberId === this.currentShowSubscriber.subscriberId) {
|
|
|
|
|
|
// 如果删除的是当前正在地图上展示的,那么切换为展示第一个;如果删除后追踪列表清空了,则置为空
|
|
|
|
|
|
if (this.trackingSubscribers.length === 0) {
|
|
|
|
|
|
this.currentShowSubscriber = null
|
|
|
|
|
|
} else {
|
|
|
|
|
|
this.currentShowSubscriber = this.trackingSubscribers[0]
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
// 关注列表追踪图标class
|
|
|
|
|
|
symbolClass (subscriber) {
|
|
|
|
|
|
const find = this.trackingSubscribers.find(s => s.subscriberId === subscriber.subscriberId)
|
|
|
|
|
|
return find ? 'cn-icon-a-' : 'cn-icon-a-1'
|
|
|
|
|
|
},
|
2024-03-04 14:23:47 +08:00
|
|
|
|
// 地图上人图标鼠标悬浮框中点击追踪事件
|
|
|
|
|
|
trackSubscriber (subscriber) {
|
|
|
|
|
|
const find = this.trackingSubscribers.find(s => s.subscriberId === subscriber.subscriberId)
|
|
|
|
|
|
if (!find) {
|
2024-03-08 18:03:21 +08:00
|
|
|
|
this.trackingSubscribers.push({ ...subscriber, show: false, showLine: false, scrollStartIndex: 1, scrollEndIndex: 6, startOffset: 0, listHeight: 0 })
|
2024-03-04 14:23:47 +08:00
|
|
|
|
}
|
|
|
|
|
|
this.currentShowSubscriber = subscriber
|
|
|
|
|
|
this.activeTab = 'traceTracking'
|
2024-03-05 16:17:14 +08:00
|
|
|
|
this.tooltip.showMarkerTooltip = false
|
2024-03-04 14:23:47 +08:00
|
|
|
|
},
|
2024-03-01 18:35:50 +08:00
|
|
|
|
async initDropdownList (curSearchValue) {
|
2024-03-03 22:10:02 +08:00
|
|
|
|
if (curSearchValue !== '') {
|
2024-03-01 18:35:50 +08:00
|
|
|
|
const params = {
|
|
|
|
|
|
startTime: this.timeFilter.startTime,
|
|
|
|
|
|
endTime: this.timeFilter.endTime,
|
|
|
|
|
|
pageNo: this.curPageNum++,
|
|
|
|
|
|
pageSize: 10,
|
2024-03-06 18:46:35 +08:00
|
|
|
|
params: "phone_number like '%" + curSearchValue + "%'"
|
2024-03-01 18:35:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
try {
|
2024-03-06 17:13:30 +08:00
|
|
|
|
this.loading.searchLoading = true
|
|
|
|
|
|
this.emptyTip = ''
|
2024-03-04 18:21:33 +08:00
|
|
|
|
await axios.get(api.location.list, { params }).then(async response => {
|
2024-03-01 18:35:50 +08:00
|
|
|
|
if (response.status === 200) {
|
2024-03-06 17:13:30 +08:00
|
|
|
|
if (params.pageNo === 1) {
|
2024-03-04 18:21:33 +08:00
|
|
|
|
this.searchValueListShow = response.data.data
|
|
|
|
|
|
} else {
|
|
|
|
|
|
this.searchValueListShow = this.searchValueListShow.concat(response.data.data)
|
|
|
|
|
|
}
|
|
|
|
|
|
if (response.data.data.length === 0 && this.curPageNum > 1) {
|
2024-03-01 18:35:50 +08:00
|
|
|
|
this.curPageNum--
|
|
|
|
|
|
}
|
2024-03-05 17:31:44 +08:00
|
|
|
|
if (params.pageNo === 1) {
|
2024-03-04 18:21:33 +08:00
|
|
|
|
this.$nextTick(() => {
|
|
|
|
|
|
const selectDom = document.getElementsByClassName('el-select-dropdown search-select')
|
2024-03-06 18:46:35 +08:00
|
|
|
|
if (selectDom) {
|
|
|
|
|
|
const dom = selectDom[0].getElementsByClassName('el-scrollbar__thumb')
|
2024-03-04 18:21:33 +08:00
|
|
|
|
if (dom && dom[1]) {
|
2024-03-05 15:16:11 +08:00
|
|
|
|
dom[1].style = 'transform: translateY(0%) !important;'
|
2024-03-04 18:21:33 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
2024-03-01 18:35:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
this.errorMsgHandler(e)
|
|
|
|
|
|
console.error(e)
|
|
|
|
|
|
} finally {
|
2024-03-03 22:10:02 +08:00
|
|
|
|
this.loading.searchLoading = false
|
2024-03-06 18:46:35 +08:00
|
|
|
|
if (this.searchValueListShow.length === 0) {
|
2024-03-06 17:13:30 +08:00
|
|
|
|
this.emptyTip = 'No Data'
|
|
|
|
|
|
}
|
2024-03-01 18:35:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-03-03 22:10:02 +08:00
|
|
|
|
onLoadMore () {
|
2024-03-06 17:13:30 +08:00
|
|
|
|
if (!this.loading.searchLoading) {
|
2024-03-04 18:21:33 +08:00
|
|
|
|
this.initDropdownList(this.curSearchValue)
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-03-05 15:16:11 +08:00
|
|
|
|
cancelFollowSubscribers (item) {
|
2024-03-04 18:21:33 +08:00
|
|
|
|
axios.delete(api.location.follow + '?subscriberId=' + item.subscriberId).then(res => {
|
|
|
|
|
|
if (res.status === 200) {
|
|
|
|
|
|
this.$message({ duration: 2000, type: 'success', message: this.$t('location.cancleFollow.success') })
|
|
|
|
|
|
/* 刷新右侧关注列表 */
|
|
|
|
|
|
this.queryFollowedList()
|
2024-03-05 16:42:04 +08:00
|
|
|
|
// 删除地图中对应的人
|
|
|
|
|
|
this.humanMarkers.forEach(marker => {
|
|
|
|
|
|
if (marker.subscriberId === item.subscriberId) {
|
|
|
|
|
|
marker.remove()
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
2024-03-04 18:21:33 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
this.$message.error(res.data.message)
|
|
|
|
|
|
}
|
|
|
|
|
|
}).catch(e => {
|
|
|
|
|
|
this.$message.error(this.errorMsgHandler(e))
|
|
|
|
|
|
})
|
2024-03-01 18:35:50 +08:00
|
|
|
|
},
|
2024-03-03 22:10:02 +08:00
|
|
|
|
followSubscribers (item) {
|
|
|
|
|
|
if (item.follow === 1) {
|
2024-03-01 18:35:50 +08:00
|
|
|
|
axios.delete(api.location.follow + '?subscriberId=' + item.subscriberId).then(res => {
|
|
|
|
|
|
if (res.status === 200) {
|
|
|
|
|
|
this.$message({ duration: 2000, type: 'success', message: this.$t('location.cancleFollow.success') })
|
|
|
|
|
|
item.follow = 0
|
2024-03-05 16:42:04 +08:00
|
|
|
|
// 删除地图中对应的人
|
|
|
|
|
|
this.humanMarkers.forEach(marker => {
|
|
|
|
|
|
if (marker.subscriberId === item.subscriberId) {
|
|
|
|
|
|
marker.remove()
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
/* 刷新右侧关注列表 */
|
|
|
|
|
|
this.queryFollowedList()
|
2024-03-01 18:35:50 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
this.$message.error(res.data.message)
|
|
|
|
|
|
}
|
|
|
|
|
|
}).catch(e => {
|
|
|
|
|
|
this.$message.error(this.errorMsgHandler(e))
|
|
|
|
|
|
})
|
|
|
|
|
|
} else {
|
2024-03-05 16:42:04 +08:00
|
|
|
|
axios.post(api.location.follow, { subscriberId: item.subscriberId }).then(async res => {
|
2024-03-01 18:35:50 +08:00
|
|
|
|
if (res.status === 200) {
|
|
|
|
|
|
this.$message({ duration: 2000, type: 'success', message: this.$t('location.follow.success') })
|
|
|
|
|
|
item.follow = 1
|
2024-03-05 16:42:04 +08:00
|
|
|
|
/* 刷新右侧关注列表 */
|
|
|
|
|
|
await this.queryFollowedList()
|
|
|
|
|
|
// 刷新地图上的人
|
|
|
|
|
|
this.humanMarkers.forEach(marker => {
|
|
|
|
|
|
marker.remove && marker.remove()
|
|
|
|
|
|
})
|
|
|
|
|
|
this.humanMarkers = []
|
|
|
|
|
|
const mapFollowedSubscriberData = await this.queryMapFollowedSubscriber()
|
|
|
|
|
|
this.renderMarker(mapFollowedSubscriberData, this.tooltipType.human)
|
2024-03-01 18:35:50 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
this.$message.error(res.data.message)
|
|
|
|
|
|
}
|
|
|
|
|
|
}).catch(e => {
|
|
|
|
|
|
this.$message.error(this.errorMsgHandler(e))
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-03-03 22:10:02 +08:00
|
|
|
|
visibleChange (state) {
|
2024-03-01 18:35:50 +08:00
|
|
|
|
this.searchValueListShow = []
|
|
|
|
|
|
},
|
|
|
|
|
|
dropDownSearch (curVal) {
|
|
|
|
|
|
this.curSearchValue = curVal
|
|
|
|
|
|
this.curPageNum = 1
|
|
|
|
|
|
this.searchValueListShow = []
|
2024-03-06 17:13:30 +08:00
|
|
|
|
this.emptyTip = ''
|
2024-03-01 18:35:50 +08:00
|
|
|
|
this.initDropdownList(curVal)
|
|
|
|
|
|
},
|
2024-03-03 22:10:02 +08:00
|
|
|
|
mapTimeLineChange (timeFilter) {
|
|
|
|
|
|
this.minuteTimeFilter = {
|
|
|
|
|
|
startTime: getSecond(timeFilter.startTime),
|
|
|
|
|
|
endTime: getSecond(timeFilter.endTime)
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
async minuteTimeFilterChange () {
|
2024-03-06 10:27:38 +08:00
|
|
|
|
// 避免初始化时请求,造成人的图标会闪一下
|
|
|
|
|
|
if (this.initFlag) {
|
|
|
|
|
|
this.initFlag = false
|
2024-03-11 16:28:10 +08:00
|
|
|
|
}
|
2024-03-11 17:31:48 +08:00
|
|
|
|
this.humanMarkers.forEach(marker => {
|
|
|
|
|
|
marker.remove && marker.remove()
|
|
|
|
|
|
})
|
|
|
|
|
|
this.humanMarkers = []
|
2024-03-11 16:28:10 +08:00
|
|
|
|
const mapFollowedSubscriberData = await this.queryMapFollowedSubscriber()
|
|
|
|
|
|
if (mapFollowedSubscriberData.length > 0) {
|
|
|
|
|
|
this.renderMarker(mapFollowedSubscriberData, this.tooltipType.human)
|
2024-03-03 22:10:02 +08:00
|
|
|
|
}
|
2024-03-06 12:43:45 +08:00
|
|
|
|
},
|
|
|
|
|
|
onResize () {
|
|
|
|
|
|
this.$nextTick(() => {
|
|
|
|
|
|
this.mapDomHeight = document.getElementById('analysisMap').offsetHeight + 150
|
|
|
|
|
|
})
|
2024-03-08 18:03:21 +08:00
|
|
|
|
},
|
|
|
|
|
|
onScroll (e) {
|
|
|
|
|
|
const find = this.trackingSubscribers.find(d => d.subscriberId === e.target.id)
|
|
|
|
|
|
// 当前滚动位置
|
|
|
|
|
|
const scrollTop = e.target.scrollTop
|
|
|
|
|
|
// 列表开始索引
|
|
|
|
|
|
const startIndex = Math.floor(scrollTop / this.scrollInfo.itemSize) || 1
|
|
|
|
|
|
// 列表结束索引
|
|
|
|
|
|
const endIndex = Math.ceil((scrollTop + this.scrollInfo.containerHeight) / this.scrollInfo.itemSize)
|
|
|
|
|
|
find.scrollStartIndex = startIndex
|
|
|
|
|
|
find.scrollEndIndex = endIndex
|
|
|
|
|
|
// 列表距离顶部距离
|
|
|
|
|
|
find.startOffset = scrollTop - (scrollTop % this.scrollInfo.itemSize)
|
2024-03-03 22:10:02 +08:00
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
},
|
|
|
|
|
|
watch: {
|
2024-02-29 18:35:43 +08:00
|
|
|
|
async activeTab (n) {
|
2024-03-06 10:27:38 +08:00
|
|
|
|
this.initFlag = true
|
2024-03-06 20:23:23 +08:00
|
|
|
|
this.$store.state.headerMenuByTab = n
|
2024-02-29 18:35:43 +08:00
|
|
|
|
if (n === 'traceTracking') {
|
|
|
|
|
|
// 切换到轨迹追踪tab时,先移除地图上已有的图层和事件绑定、人型图标。基站予以保留
|
|
|
|
|
|
this.unbindHexagonEvents()
|
2024-03-01 18:13:04 +08:00
|
|
|
|
this.mapChart.getLayer('hexagon') && this.mapChart.removeLayer('hexagon')
|
|
|
|
|
|
this.mapChart.getSource('hexGrid') && this.mapChart.removeSource('hexGrid')
|
2024-02-29 18:35:43 +08:00
|
|
|
|
this.humanMarkers.forEach(marker => {
|
|
|
|
|
|
marker.remove && marker.remove()
|
|
|
|
|
|
})
|
|
|
|
|
|
this.humanMarkers = []
|
2024-03-03 22:10:02 +08:00
|
|
|
|
const newUrl = urlParamsHandler(`${window.location.protocol}//${window.location.host}/#/location/tracking`, {}, this.$route.query)
|
|
|
|
|
|
overwriteUrl(newUrl)
|
|
|
|
|
|
this.timeRefreshChange()
|
2024-02-29 18:35:43 +08:00
|
|
|
|
} else if (n === 'locationMap') {
|
2024-03-05 16:17:14 +08:00
|
|
|
|
this.unbindTrackingHexagonEvents()
|
2024-03-01 18:13:04 +08:00
|
|
|
|
this.mapChart.getLayer('trackingHexagon') && this.mapChart.removeLayer('trackingHexagon')
|
|
|
|
|
|
this.mapChart.getLayer('trackingLine') && this.mapChart.removeLayer('trackingLine')
|
|
|
|
|
|
this.mapChart.getSource('trackingHexGrid') && this.mapChart.removeSource('trackingHexGrid')
|
|
|
|
|
|
this.mapChart.getSource('trackingLineSource') && this.mapChart.removeSource('trackingLineSource')
|
|
|
|
|
|
this.trackingHumanMarker.remove && this.trackingHumanMarker.remove()
|
|
|
|
|
|
this.trackingHumanMarker = {}
|
2024-03-03 22:10:02 +08:00
|
|
|
|
|
|
|
|
|
|
const newUrl = urlParamsHandler(`${window.location.protocol}//${window.location.host}/#/location/map`, {}, this.$route.query)
|
|
|
|
|
|
overwriteUrl(newUrl)
|
|
|
|
|
|
this.timeRefreshChange()
|
2024-02-29 18:35:43 +08:00
|
|
|
|
}
|
2024-02-29 18:55:24 +08:00
|
|
|
|
},
|
|
|
|
|
|
// 时间轴改变时,重新查询人marker
|
|
|
|
|
|
async minuteTimeFilter (n) {
|
2024-03-05 16:17:14 +08:00
|
|
|
|
this.debounceFunc?.()
|
2024-03-03 22:10:02 +08:00
|
|
|
|
},
|
|
|
|
|
|
// 切换追踪的用户
|
|
|
|
|
|
currentShowSubscriber (n) {
|
|
|
|
|
|
this.mapChart.getLayer('trackingHexagon') && this.mapChart.removeLayer('trackingHexagon')
|
|
|
|
|
|
this.mapChart.getLayer('trackingLine') && this.mapChart.removeLayer('trackingLine')
|
|
|
|
|
|
this.mapChart.getSource('trackingHexGrid') && this.mapChart.removeSource('trackingHexGrid')
|
|
|
|
|
|
this.mapChart.getSource('trackingLineSource') && this.mapChart.removeSource('trackingLineSource')
|
|
|
|
|
|
this.trackingHumanMarker.remove && this.trackingHumanMarker.remove()
|
|
|
|
|
|
this.trackingHumanMarker = {}
|
2024-03-05 15:16:11 +08:00
|
|
|
|
if (n) {
|
|
|
|
|
|
this.renderTrackingHexagon()
|
|
|
|
|
|
}
|
2024-03-03 22:10:02 +08:00
|
|
|
|
},
|
|
|
|
|
|
timeFilter (n) {
|
|
|
|
|
|
if (this.activeTab === 'locationMap') {
|
2024-03-04 14:23:47 +08:00
|
|
|
|
this.unbindHexagonEvents()
|
|
|
|
|
|
this.mapChart.getLayer('hexagon') && this.mapChart.removeLayer('hexagon')
|
|
|
|
|
|
this.mapChart.getSource('hexGrid') && this.mapChart.removeSource('hexGrid')
|
|
|
|
|
|
this.humanMarkers.forEach(marker => {
|
|
|
|
|
|
marker.remove && marker.remove()
|
|
|
|
|
|
})
|
|
|
|
|
|
this.humanMarkers = []
|
2024-03-03 22:10:02 +08:00
|
|
|
|
this.initLocationMapTab()
|
|
|
|
|
|
} else if (this.activeTab === 'traceTracking') {
|
2024-03-05 16:17:14 +08:00
|
|
|
|
this.unbindTrackingHexagonEvents()
|
2024-03-04 14:23:47 +08:00
|
|
|
|
this.mapChart.getLayer('trackingHexagon') && this.mapChart.removeLayer('trackingHexagon')
|
|
|
|
|
|
this.mapChart.getLayer('trackingLine') && this.mapChart.removeLayer('trackingLine')
|
|
|
|
|
|
this.mapChart.getSource('trackingHexGrid') && this.mapChart.removeSource('trackingHexGrid')
|
|
|
|
|
|
this.mapChart.getSource('trackingLineSource') && this.mapChart.removeSource('trackingLineSource')
|
|
|
|
|
|
this.trackingHumanMarker.remove && this.trackingHumanMarker.remove()
|
|
|
|
|
|
this.trackingHumanMarker = {}
|
2024-03-03 22:10:02 +08:00
|
|
|
|
this.initTraceTrackingTab()
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-03-04 17:28:05 +08:00
|
|
|
|
trackingSubscribers: {
|
|
|
|
|
|
deep: true,
|
|
|
|
|
|
handler (n) {
|
2024-03-06 16:59:07 +08:00
|
|
|
|
sessionStorage.setItem(storageKey.trackingSubscribers, JSON.stringify(n.map(item => ({ subscriberId: item.subscriberId, subscriberDto: item.subscriberDto ? { phoneNumber: item.subscriberDto.phoneNumber } : null }))))
|
2024-03-04 17:28:05 +08:00
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-03-03 22:10:02 +08:00
|
|
|
|
// 控制map loading
|
|
|
|
|
|
'loading.hexagonLoading': {
|
|
|
|
|
|
handler (n) {
|
|
|
|
|
|
this.loading.mapLoading = n || this.loading.timeBarLoading || this.loading.baseStationLoading
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
'loading.timeBarLoading': {
|
|
|
|
|
|
handler (n) {
|
|
|
|
|
|
this.loading.mapLoading = n || this.loading.hexagonLoading || this.loading.baseStationLoading
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
'loading.baseStationLoading': {
|
|
|
|
|
|
handler (n) {
|
|
|
|
|
|
this.loading.mapLoading = n || this.loading.timeBarLoading || this.loading.hexagonLoading
|
2024-02-29 18:55:24 +08:00
|
|
|
|
}
|
2024-02-28 19:11:33 +08:00
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
},
|
|
|
|
|
|
computed: {
|
|
|
|
|
|
tooltipHeaderColor () {
|
|
|
|
|
|
if (this.tooltip.type === this.tooltipType.hexagon) {
|
2024-02-28 10:21:16 +08:00
|
|
|
|
const color = this.currentPolygon.color.split(',')
|
|
|
|
|
|
color[0] = color[0].split('[')[1]
|
|
|
|
|
|
color[2] = color[2].split(']')[0]
|
|
|
|
|
|
return `rgba(${color.join(',')},.8)`
|
2024-02-26 11:51:13 +08:00
|
|
|
|
} else if (this.tooltip.type === this.tooltipType.human) {
|
|
|
|
|
|
return '#38ACD2'
|
|
|
|
|
|
} else if (this.tooltip.type === this.tooltipType.baseStation) {
|
|
|
|
|
|
return '#233447'
|
|
|
|
|
|
}
|
|
|
|
|
|
return ''
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
async mounted () {
|
|
|
|
|
|
await this.initMap()
|
2024-03-03 22:10:02 +08:00
|
|
|
|
this.debounceFunc = _.debounce(this.minuteTimeFilterChange, 500)
|
2024-03-06 12:43:45 +08:00
|
|
|
|
this.debounceFunc2 = _.debounce(this.onResize, 500)
|
|
|
|
|
|
this.onResize()
|
2024-02-26 11:51:13 +08:00
|
|
|
|
},
|
|
|
|
|
|
setup () {
|
|
|
|
|
|
const { currentRoute } = useRouter()
|
|
|
|
|
|
const currentPath = currentRoute.value.path
|
|
|
|
|
|
const activeTab = ref('')
|
2024-03-01 18:35:50 +08:00
|
|
|
|
const dropDownValue = ref('')
|
|
|
|
|
|
const curSearchValue = ref('')
|
2024-02-26 11:51:13 +08:00
|
|
|
|
switch (currentPath) {
|
|
|
|
|
|
case ('/location/map'): {
|
|
|
|
|
|
activeTab.value = 'locationMap'
|
|
|
|
|
|
break
|
|
|
|
|
|
}
|
|
|
|
|
|
case ('/location/tracking'): {
|
|
|
|
|
|
activeTab.value = 'traceTracking'
|
|
|
|
|
|
break
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
const { query } = useRoute()
|
|
|
|
|
|
// 获取url携带的range、startTime、endTime
|
|
|
|
|
|
const rangeParam = query.range
|
|
|
|
|
|
const startTimeParam = query.startTime
|
|
|
|
|
|
const endTimeParam = query.endTime
|
|
|
|
|
|
|
|
|
|
|
|
// 优先级:url > config.js > 默认值。
|
|
|
|
|
|
const dateRangeValue = rangeParam ? parseInt(rangeParam) : (DEFAULT_TIME_FILTER_RANGE.dashboard || 60)
|
|
|
|
|
|
const timeFilter = ref({ dateRangeValue })
|
2024-03-04 17:28:05 +08:00
|
|
|
|
if (!startTimeParam || !endTimeParam || dateRangeValue > -1) {
|
2024-02-26 11:51:13 +08:00
|
|
|
|
const { startTime, endTime } = getNowTime(dateRangeValue)
|
|
|
|
|
|
timeFilter.value.startTime = getSecond(startTime)
|
|
|
|
|
|
timeFilter.value.endTime = getSecond(endTime)
|
2024-03-04 17:28:05 +08:00
|
|
|
|
// 将参数写入url
|
2024-02-26 11:51:13 +08:00
|
|
|
|
const newUrl = urlParamsHandler(window.location.href, useRoute().query, { startTime: timeFilter.value.startTime, endTime: timeFilter.value.endTime, range: dateRangeValue })
|
|
|
|
|
|
overwriteUrl(newUrl)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
timeFilter.value.startTime = parseInt(startTimeParam)
|
|
|
|
|
|
timeFilter.value.endTime = parseInt(endTimeParam)
|
|
|
|
|
|
}
|
2024-03-05 11:19:47 +08:00
|
|
|
|
const minuteTimeFilter = ref({})
|
2024-02-26 11:51:13 +08:00
|
|
|
|
const tooltip = ref({
|
|
|
|
|
|
type: ''
|
|
|
|
|
|
})
|
|
|
|
|
|
// const pieColorRamp = ['186,224,255', '105,177,255', '22,119,255', '0,62,179', '0,29,102']
|
2024-02-28 10:21:16 +08:00
|
|
|
|
// const pieColorRamp = ['156,174,29', '241,198,0', '89,202,242', '63,133,186', '37,55,128']
|
2024-03-06 17:15:22 +08:00
|
|
|
|
// const pieColorRamp = ['196,214,59', '190,230,255', '135,206,250', '63,133,186', '37,55,128']
|
|
|
|
|
|
const pieColorRamp = ['196,214,59', '135,206,250', '63,133,186', '42,60,135', '34,7,94']
|
2024-02-26 11:51:13 +08:00
|
|
|
|
const pieValueRamp = ref([])
|
2024-03-01 18:35:50 +08:00
|
|
|
|
const followedSubscribersList = ref([])
|
|
|
|
|
|
const searchValueListShow = ref([])
|
2024-02-26 11:51:13 +08:00
|
|
|
|
const boundaryBox = ref({}) // minLongitude、maxLongitude、minLatitude、maxLatitude
|
2024-02-29 18:35:43 +08:00
|
|
|
|
const mapChart = shallowRef(null)
|
|
|
|
|
|
const currentMarkerDom = shallowRef(null)
|
|
|
|
|
|
const humanMarkers = shallowRef([])
|
|
|
|
|
|
const baseStationMarkers = shallowRef([])
|
2024-03-01 18:13:04 +08:00
|
|
|
|
const trackingHumanMarker = shallowRef({})
|
2024-02-26 11:51:13 +08:00
|
|
|
|
const pieChart = shallowRef(null)
|
|
|
|
|
|
const pieOption = ref({})
|
|
|
|
|
|
const lineChart = shallowRef(null)
|
|
|
|
|
|
const lineOption = ref({})
|
|
|
|
|
|
const currentBaseStation = ref({})
|
|
|
|
|
|
const currentSubscriber = ref({})
|
|
|
|
|
|
const currentPolygon = ref({})
|
2024-03-11 16:32:29 +08:00
|
|
|
|
const highlightSubscriber = ref({})
|
2024-02-29 18:35:43 +08:00
|
|
|
|
|
|
|
|
|
|
// 从localStorage中获取数据
|
|
|
|
|
|
const trackingSubscribers = ref([])
|
2024-03-08 18:03:21 +08:00
|
|
|
|
sessionStorage.getItem(storageKey.trackingSubscribers) && (trackingSubscribers.value = JSON.parse(sessionStorage.getItem(storageKey.trackingSubscribers)).map(item => ({ ...item, show: false, showLine: false, scrollStartIndex: 1, scrollEndIndex: 6, startOffset: 0, listHeight: 0 })))
|
2024-03-05 10:44:43 +08:00
|
|
|
|
/* const test = ['gary6411', 'test6431', 'test6430', 'test6422']
|
2024-02-29 18:35:43 +08:00
|
|
|
|
test.forEach(id => {
|
2024-03-01 17:43:15 +08:00
|
|
|
|
trackingSubscribers.value.push({ subscriberId: id, show: false, showLine: false })
|
2024-03-05 10:44:43 +08:00
|
|
|
|
}) */
|
2024-02-29 18:35:43 +08:00
|
|
|
|
|
|
|
|
|
|
const currentShowSubscriber = ref(null)
|
2024-02-26 11:51:13 +08:00
|
|
|
|
const loading = ref({
|
2024-03-03 22:10:02 +08:00
|
|
|
|
mapLoading: true, // mapLoading控制location地图的loading,它状态同时受hexagonLoading、timeBarLoading、baseStationLoading影响
|
2024-02-26 11:51:13 +08:00
|
|
|
|
hexagonLoading: true, // 六边形加载状态
|
|
|
|
|
|
timeBarLoading: true, // 时间轴和地图上的人型图标的加载状态
|
|
|
|
|
|
baseStationLoading: true, // 基站加载状态
|
|
|
|
|
|
|
|
|
|
|
|
followSubscriberLoading: true, // 控制右侧关注用户列表加载状态
|
|
|
|
|
|
pieLoading: true, // 控制饼图加载状态
|
2024-03-01 18:35:50 +08:00
|
|
|
|
lineLoading: true, // 控制折线图加载状态
|
2024-03-03 22:10:02 +08:00
|
|
|
|
searchLoading: false, // 搜索框加载状态
|
|
|
|
|
|
trackingMapLoading: true // 控制追踪地图加载状态
|
2024-02-26 11:51:13 +08:00
|
|
|
|
})
|
2024-03-06 12:43:45 +08:00
|
|
|
|
|
|
|
|
|
|
const mapDomHeight = ref(0)
|
|
|
|
|
|
const tooltipDomHeight = {
|
|
|
|
|
|
hexagon: 153,
|
|
|
|
|
|
baseStation: 153,
|
|
|
|
|
|
human: 167
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
return {
|
|
|
|
|
|
activeTab,
|
2024-03-01 18:35:50 +08:00
|
|
|
|
dropDownValue,
|
|
|
|
|
|
curSearchValue,
|
2024-02-26 11:51:13 +08:00
|
|
|
|
timeFilter,
|
|
|
|
|
|
minuteTimeFilter, // 底下时间轴的时间
|
2024-03-03 22:10:02 +08:00
|
|
|
|
searchValueListShow, // 搜索框下拉列表
|
2024-02-26 11:51:13 +08:00
|
|
|
|
tooltip, // 控制鼠标悬浮框
|
|
|
|
|
|
pieColorRamp, // 六边形颜色坡度
|
|
|
|
|
|
pieValueRamp, // 饼图数值坡度,动态获取
|
2024-03-03 22:10:02 +08:00
|
|
|
|
followedSubscribersList, // Location关注用户列表
|
2024-02-26 11:51:13 +08:00
|
|
|
|
boundaryBox, // 查六边形的范围,minLongitude、maxLongitude、minLatitude、maxLatitude
|
|
|
|
|
|
mapChart, // 地图对象
|
2024-02-29 18:35:43 +08:00
|
|
|
|
currentMarkerDom, // 记录当前鼠标悬停的marker的dom
|
|
|
|
|
|
humanMarkers, // 储存人marker的引用
|
|
|
|
|
|
baseStationMarkers, // 储存基站marker的引用
|
2024-03-01 18:13:04 +08:00
|
|
|
|
trackingHumanMarker, // 追踪页的人marker
|
2024-02-26 11:51:13 +08:00
|
|
|
|
pieChart, // 饼图对象
|
|
|
|
|
|
pieOption,
|
|
|
|
|
|
lineChart, // 折线图对象
|
|
|
|
|
|
lineOption,
|
2024-03-06 12:43:45 +08:00
|
|
|
|
mapPolygonSourceData: shallowRef({}), // locationMap 的 maplibre sourceData
|
|
|
|
|
|
trackingPolygonSourceData: shallowRef({}), // traceTracking 的 maplibre sourceData
|
2024-02-26 11:51:13 +08:00
|
|
|
|
currentBaseStation, // 鼠标当前悬浮的基站
|
|
|
|
|
|
currentSubscriber, // 鼠标当前悬浮的Subscriber
|
|
|
|
|
|
currentPolygon, // 鼠标当前悬浮的六边形
|
2024-03-11 16:32:29 +08:00
|
|
|
|
highlightSubscriber, // locationMap页保持高亮的subscriber
|
2024-02-29 18:35:43 +08:00
|
|
|
|
trackingSubscribers, // 存放当前追踪的Subscriber列表
|
2024-03-06 17:15:22 +08:00
|
|
|
|
currentShowSubscriber, // 当前在地图上展示轨迹的Subscriber
|
2024-03-06 16:59:07 +08:00
|
|
|
|
trackingSubscriberRecordMap: [], // record数据量大时,vue监听性能开销太大,所以单独用非监听的数组来维护subscriberId与record的关系
|
2024-02-26 11:51:13 +08:00
|
|
|
|
loading, // 控制组件内各处loading图标
|
|
|
|
|
|
maxZoom: 14, // 地图最小缩放比例
|
|
|
|
|
|
minZoom: 3, // 地图最大缩放比例
|
2024-03-06 12:43:45 +08:00
|
|
|
|
mapLevel: 2, // 地图精度 1、2、3
|
2024-02-26 11:51:13 +08:00
|
|
|
|
unitTypes,
|
2024-03-06 12:43:45 +08:00
|
|
|
|
defaultZoom: 12, // 地图默认缩放比例
|
|
|
|
|
|
center: [116.38, 39.92], // 地图默认中心点。北京:[116.38, 39.9] 纽约:[-73.94539, 40.841843]
|
|
|
|
|
|
debounceFunc: shallowRef(null),
|
|
|
|
|
|
debounceFunc2: shallowRef(null),
|
|
|
|
|
|
mapDomHeight, // 地图dom的高度,用来计算悬浮框的位置
|
|
|
|
|
|
tooltipDomHeight // 计算悬浮框位置时默认的悬浮框高度
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
unmounted () {
|
2024-03-01 18:35:50 +08:00
|
|
|
|
if (this.mapChart && this.mapChart.remove) {
|
2024-02-26 11:51:13 +08:00
|
|
|
|
this.mapChart && this.mapChart.remove()
|
|
|
|
|
|
}
|
2024-03-01 17:43:15 +08:00
|
|
|
|
if (this.pieChart && this.pieChart.dispose) {
|
2024-02-26 11:51:13 +08:00
|
|
|
|
this.pieChart && this.pieChart.dispose()
|
|
|
|
|
|
}
|
2024-03-01 17:43:15 +08:00
|
|
|
|
if (this.lineChart && this.lineChart.dispose) {
|
2024-02-26 11:51:13 +08:00
|
|
|
|
this.lineChart && this.lineChart.dispose()
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
</script>
|
|
|
|
|
|
<style lang="scss">
|
|
|
|
|
|
.geo-analysis {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
padding: 0 20px 20px;
|
|
|
|
|
|
position: relative;
|
2024-03-06 18:04:52 +08:00
|
|
|
|
.location-tabs {
|
|
|
|
|
|
.traceTracking-tabs_label {
|
|
|
|
|
|
display:flex;
|
|
|
|
|
|
flex-display: row;
|
|
|
|
|
|
.traceTracking-num {
|
|
|
|
|
|
margin-top: 5px;
|
|
|
|
|
|
border-radius: 10px;
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: #FFFFFF;
|
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
|
height: 20px;
|
|
|
|
|
|
width: fit-content;
|
|
|
|
|
|
min-width: 20px;
|
|
|
|
|
|
padding:0 5px 0 5px;
|
|
|
|
|
|
line-height: 20px;
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
background: #E26154;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
|
2024-03-11 16:32:29 +08:00
|
|
|
|
svg {
|
|
|
|
|
|
fill: #fff;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-02-28 10:21:16 +08:00
|
|
|
|
.maplibregl-canvas:focus-visible {
|
|
|
|
|
|
outline: none;
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
.geo-tools {
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
top: 11px;
|
|
|
|
|
|
right: 20px;
|
2024-03-01 18:35:50 +08:00
|
|
|
|
.el-select .el-input__inner {
|
|
|
|
|
|
cursor: text;
|
|
|
|
|
|
}
|
2024-03-06 17:13:30 +08:00
|
|
|
|
.el-select .el-input .el-select__caret {
|
|
|
|
|
|
cursor: text;
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
.panel__time {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
}
|
2024-03-07 15:35:59 +08:00
|
|
|
|
.el-select {
|
|
|
|
|
|
.search-active {
|
|
|
|
|
|
float: left;
|
|
|
|
|
|
border-radius: 3px;
|
|
|
|
|
|
width: 6px;
|
|
|
|
|
|
height: 6px;
|
|
|
|
|
|
margin-right: 10px;
|
|
|
|
|
|
position:relative;
|
|
|
|
|
|
top:50%;
|
|
|
|
|
|
transform: translateY(-50%);
|
|
|
|
|
|
text-align:center;
|
|
|
|
|
|
}
|
|
|
|
|
|
.search-value {
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
margin-right:20px;
|
|
|
|
|
|
}
|
|
|
|
|
|
.search-follow__icon {
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
top: 50%;
|
|
|
|
|
|
right: 10px;
|
|
|
|
|
|
transform: translateY(-50%);
|
|
|
|
|
|
color: #6f6f6e;
|
|
|
|
|
|
margin-right: 10px;
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
i {
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.search-content {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
height: 100%;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
}
|
|
|
|
|
|
.active-icon {
|
|
|
|
|
|
background: #38ACD2;
|
|
|
|
|
|
}
|
|
|
|
|
|
.inactive-icon {
|
|
|
|
|
|
background: #CCCCCC;
|
|
|
|
|
|
}
|
|
|
|
|
|
.search-select {
|
|
|
|
|
|
max-height: 250px;
|
|
|
|
|
|
}
|
|
|
|
|
|
.search-select .el-scrollbar__wrap{
|
|
|
|
|
|
max-height: 250px;
|
|
|
|
|
|
overflow-y: auto;
|
|
|
|
|
|
}
|
|
|
|
|
|
.el-select-dropdown .el-scrollbar .el-select-dropdown__wrap .el-scrollbar__view.el-select-dropdown__list .el-select-dropdown__item {
|
|
|
|
|
|
position: relative;
|
|
|
|
|
|
height: 48px;
|
|
|
|
|
|
line-height: 16px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
.el-tabs {
|
|
|
|
|
|
.el-tabs__header {
|
|
|
|
|
|
margin-bottom: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
.el-tabs__nav-wrap::after {
|
|
|
|
|
|
height: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
.el-tabs__active-bar {
|
|
|
|
|
|
height: 3px;
|
|
|
|
|
|
background-color: #046eca;
|
2024-03-07 14:52:24 +08:00
|
|
|
|
bottom: 1px;
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
.el-tabs__item {
|
|
|
|
|
|
&.is-active {
|
|
|
|
|
|
color: #046eca;
|
|
|
|
|
|
}
|
|
|
|
|
|
height: 50px;
|
|
|
|
|
|
line-height: 50px;
|
|
|
|
|
|
font-size: 20px;
|
|
|
|
|
|
color: #353636;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.geo-analysis__container {
|
|
|
|
|
|
height: calc(100% - 50px);
|
|
|
|
|
|
display: flex;
|
2024-03-03 22:10:02 +08:00
|
|
|
|
position: relative;
|
2024-02-26 11:51:13 +08:00
|
|
|
|
|
|
|
|
|
|
.analysis-map {
|
|
|
|
|
|
flex: 1;
|
2024-03-03 22:10:02 +08:00
|
|
|
|
position: relative;
|
2024-02-26 11:51:13 +08:00
|
|
|
|
|
2024-03-03 22:10:02 +08:00
|
|
|
|
#analysisMap {
|
|
|
|
|
|
border-radius: 2px;
|
|
|
|
|
|
height: 100%;
|
|
|
|
|
|
width: 100%;
|
|
|
|
|
|
}
|
|
|
|
|
|
.maplibregl-ctrl-bottom-right {
|
|
|
|
|
|
top: 0;
|
|
|
|
|
|
left: 0;
|
|
|
|
|
|
bottom: unset;
|
|
|
|
|
|
right: unset;
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
.map-marker {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
width: 20px;
|
|
|
|
|
|
height: 20px;
|
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
|
cursor: default;
|
|
|
|
|
|
padding: 0;
|
2024-02-28 19:11:33 +08:00
|
|
|
|
transition: height .1s linear, width .1s linear;
|
|
|
|
|
|
|
|
|
|
|
|
svg {
|
|
|
|
|
|
transition: height .1s linear, width .1s linear;
|
2024-03-11 16:32:29 +08:00
|
|
|
|
fill: #fff;
|
2024-02-28 19:11:33 +08:00
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
|
|
|
|
|
|
&.map-marker--human {
|
|
|
|
|
|
background-color: #233447;
|
2024-03-11 16:32:29 +08:00
|
|
|
|
cursor: pointer;
|
2024-02-28 19:11:33 +08:00
|
|
|
|
|
|
|
|
|
|
svg {
|
|
|
|
|
|
width: 14px;
|
|
|
|
|
|
height: 14px;
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
&.map-marker--base-station {
|
|
|
|
|
|
background-color: #585B5F;
|
2024-02-28 19:11:33 +08:00
|
|
|
|
|
|
|
|
|
|
svg {
|
|
|
|
|
|
width: 12px;
|
|
|
|
|
|
height: 12px;
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
&.map-marker--hover {
|
2024-03-04 14:23:47 +08:00
|
|
|
|
width: 30px;
|
|
|
|
|
|
height: 30px;
|
2024-02-26 11:51:13 +08:00
|
|
|
|
border: 2px solid rgba(255,255,255,1);
|
2024-03-11 16:32:29 +08:00
|
|
|
|
z-index: 2;
|
2024-02-28 19:11:33 +08:00
|
|
|
|
|
|
|
|
|
|
&.map-marker--human svg {
|
2024-03-04 14:23:47 +08:00
|
|
|
|
width: 21px;
|
|
|
|
|
|
height: 21px;
|
2024-02-28 19:11:33 +08:00
|
|
|
|
}
|
|
|
|
|
|
&.map-marker--base-station svg {
|
2024-03-04 14:23:47 +08:00
|
|
|
|
width: 18px;
|
|
|
|
|
|
height: 18px;
|
2024-02-28 19:11:33 +08:00
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
2024-03-11 16:32:29 +08:00
|
|
|
|
&.map-marker--highlight {
|
|
|
|
|
|
width: 30px;
|
|
|
|
|
|
height: 30px;
|
|
|
|
|
|
border: 2px solid #fff;
|
|
|
|
|
|
background-color: rgb(204,68,68);
|
|
|
|
|
|
z-index: 3;
|
|
|
|
|
|
|
|
|
|
|
|
svg {
|
|
|
|
|
|
width: 21px;
|
|
|
|
|
|
height: 21px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
2024-03-01 18:13:04 +08:00
|
|
|
|
.map-tracking-marker {
|
|
|
|
|
|
width: 36px;
|
|
|
|
|
|
height: 36px;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
animation: pulse 2s infinite;
|
|
|
|
|
|
background-color: rgba(204,68,68,0.50);
|
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
|
|
|
|
|
|
|
.tracking-marker__inner-circle {
|
|
|
|
|
|
width: 25px;
|
|
|
|
|
|
height: 25px;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
background-color: rgba(204,68,68,0.80);
|
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
|
|
|
|
|
|
|
svg {
|
|
|
|
|
|
width: 20px;
|
|
|
|
|
|
height: 20px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
2024-03-03 22:10:02 +08:00
|
|
|
|
.map-time-line {
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
bottom: 0;
|
|
|
|
|
|
left: -6px;
|
|
|
|
|
|
width: calc(100% - 318px);
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
.analysis-statistics {
|
|
|
|
|
|
width: 330px;
|
|
|
|
|
|
overflow-y: auto;
|
|
|
|
|
|
padding-right: 5px;
|
|
|
|
|
|
|
|
|
|
|
|
.analysis-statistics__chart {
|
|
|
|
|
|
margin: 0 0 10px 20px;
|
|
|
|
|
|
height: 148px;
|
|
|
|
|
|
background: rgba(113,113,113,0.06);
|
|
|
|
|
|
border: 1px solid rgba(226,229,236,1);
|
|
|
|
|
|
border-radius: 4px;
|
2024-03-03 22:10:02 +08:00
|
|
|
|
position: relative;
|
2024-02-26 11:51:13 +08:00
|
|
|
|
|
|
|
|
|
|
.chart__header {
|
|
|
|
|
|
padding: 8px 0 0 10px;
|
|
|
|
|
|
color: #353636;
|
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
|
}
|
2024-02-28 07:38:41 +08:00
|
|
|
|
.chart__body {
|
|
|
|
|
|
display:flex;
|
|
|
|
|
|
flex-direction:row;
|
|
|
|
|
|
width:100%;
|
|
|
|
|
|
height:calc(100% - 32px);
|
|
|
|
|
|
.chart__legend {
|
|
|
|
|
|
width:calc(60% - 22px);
|
|
|
|
|
|
height: 100%;
|
|
|
|
|
|
display:flex;
|
|
|
|
|
|
flex-direction:column;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
.legend-item {
|
|
|
|
|
|
display:flex;
|
|
|
|
|
|
flex-direction:row;
|
|
|
|
|
|
justify-content: left;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
padding-top:2px;
|
|
|
|
|
|
padding-bottom:2px;
|
|
|
|
|
|
.legend-icon {
|
|
|
|
|
|
width:16px;
|
|
|
|
|
|
height:6px;
|
|
|
|
|
|
border-radius: 2px;
|
|
|
|
|
|
margin-right:10px;
|
|
|
|
|
|
}
|
|
|
|
|
|
.legend-range {
|
|
|
|
|
|
margin-right:40px;
|
|
|
|
|
|
width:54px;
|
|
|
|
|
|
font-family: Helvetica;
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: #575757;
|
|
|
|
|
|
line-height: 12px;
|
|
|
|
|
|
font-weight: 400;
|
|
|
|
|
|
}
|
|
|
|
|
|
.legend-count {
|
|
|
|
|
|
font-family: Helvetica;
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: #353636;
|
|
|
|
|
|
line-height: 12px;
|
|
|
|
|
|
font-weight: 400;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
.chart__statistics {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
height: 20px;
|
|
|
|
|
|
padding-left: 10px;
|
|
|
|
|
|
|
|
|
|
|
|
.statistics-number {
|
|
|
|
|
|
margin-right: 8px;
|
|
|
|
|
|
color: #353636;
|
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
|
line-height: 22px;
|
|
|
|
|
|
}
|
|
|
|
|
|
.statistics-trend {
|
|
|
|
|
|
padding: 0 10px;
|
|
|
|
|
|
background: #7E9F54;
|
|
|
|
|
|
border-radius: 10px;
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: white;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.chart__drawing {
|
2024-02-28 07:38:41 +08:00
|
|
|
|
height: 100%;
|
|
|
|
|
|
width:calc(40% + 12px);
|
|
|
|
|
|
margin-right:10px;
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
2024-02-28 09:58:26 +08:00
|
|
|
|
.chart-line__drawing {
|
|
|
|
|
|
height: calc(100% - 32px);
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
#activeSubscribersChart {
|
|
|
|
|
|
height: calc(100% - 52px);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.analysis-statistics__title {
|
2024-03-06 17:44:13 +08:00
|
|
|
|
position: relative;
|
2024-02-26 11:51:13 +08:00
|
|
|
|
margin-bottom: 10px;
|
|
|
|
|
|
padding-left: 20px;
|
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
|
color: #353636;
|
|
|
|
|
|
}
|
2024-03-05 15:16:11 +08:00
|
|
|
|
.analysis-statistics__no-tracking-tip {
|
|
|
|
|
|
padding: 0 20px;
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
color: #046eca;
|
|
|
|
|
|
text-decoration: underline;
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
.analysis-statistics__subscribers {
|
|
|
|
|
|
padding-left: 20px;
|
|
|
|
|
|
|
|
|
|
|
|
.analysis-statistics__subscriber {
|
|
|
|
|
|
margin-bottom: 10px;
|
2024-03-03 22:10:02 +08:00
|
|
|
|
|
2024-02-26 11:51:13 +08:00
|
|
|
|
&:last-of-type {
|
|
|
|
|
|
margin-bottom: 0;
|
|
|
|
|
|
}
|
2024-03-03 22:10:02 +08:00
|
|
|
|
&.analysis-statistics__subscriber--active {
|
2024-03-06 12:43:45 +08:00
|
|
|
|
border: 1px solid #38acd2;
|
2024-03-11 16:32:29 +08:00
|
|
|
|
box-shadow: 0 1px 3px 0 #38acd2;
|
2024-03-03 22:10:02 +08:00
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
|
2024-03-03 22:10:02 +08:00
|
|
|
|
background-color: #F7F7F7;
|
|
|
|
|
|
border: 1px solid rgb(226,229,236);
|
2024-02-26 11:51:13 +08:00
|
|
|
|
border-radius: 2px;
|
|
|
|
|
|
|
2024-03-03 22:10:02 +08:00
|
|
|
|
.subscriber__header {
|
2024-03-01 18:35:50 +08:00
|
|
|
|
background-color: #38ACD2;
|
2024-03-05 11:19:47 +08:00
|
|
|
|
|
|
|
|
|
|
&.subscriber__header-inactive {
|
|
|
|
|
|
background-color: #CCCCCC;
|
|
|
|
|
|
}
|
2024-03-01 18:35:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-02-26 11:51:13 +08:00
|
|
|
|
.subscriber__header {
|
|
|
|
|
|
position: relative;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
padding: 6px 0 0 70px;
|
|
|
|
|
|
color: white;
|
|
|
|
|
|
height: 58px;
|
|
|
|
|
|
|
|
|
|
|
|
.header__icon {
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
left: 20px;
|
|
|
|
|
|
top: 14px;
|
|
|
|
|
|
|
|
|
|
|
|
.icon__box {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
height: 30px;
|
|
|
|
|
|
width: 30px;
|
|
|
|
|
|
background-color: #233447;
|
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-03-04 18:21:33 +08:00
|
|
|
|
.header__right {
|
|
|
|
|
|
display:flex;
|
|
|
|
|
|
flex-direction: row;
|
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
|
.header-msisdn {
|
|
|
|
|
|
display:flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
.header__title {
|
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
|
}
|
|
|
|
|
|
.header__content {
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.header__operation {
|
|
|
|
|
|
display:flex;
|
|
|
|
|
|
flex-direction: row;
|
|
|
|
|
|
align-items: center;
|
2024-03-05 15:16:11 +08:00
|
|
|
|
|
2024-03-04 18:21:33 +08:00
|
|
|
|
.trajectory-text {
|
|
|
|
|
|
margin-right:12px;
|
|
|
|
|
|
font-family: Helvetica-Bold;
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: #233447;
|
|
|
|
|
|
font-weight: 700;
|
2024-03-05 15:16:11 +08:00
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
|
2024-03-04 18:21:33 +08:00
|
|
|
|
i {
|
|
|
|
|
|
font-size: 12px !important;
|
|
|
|
|
|
font-weight: bolder;
|
|
|
|
|
|
margin-right:3px;
|
|
|
|
|
|
color: #233447;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-03-05 10:44:43 +08:00
|
|
|
|
.cancel-follow {
|
2024-03-04 18:21:33 +08:00
|
|
|
|
margin-right:14px;
|
|
|
|
|
|
i {
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
font-size:8px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.subscriber__body {
|
|
|
|
|
|
padding: 10px 18px;
|
|
|
|
|
|
.body__item {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
|
|
|
|
|
|
.item__label {
|
|
|
|
|
|
padding-right: 10px;
|
|
|
|
|
|
text-align: right;
|
|
|
|
|
|
width: 60px;
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: #353636;
|
|
|
|
|
|
}
|
|
|
|
|
|
.item__value {
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
color: #233447;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-03-01 17:43:15 +08:00
|
|
|
|
|
|
|
|
|
|
.body-item-record {
|
2024-03-03 22:10:02 +08:00
|
|
|
|
margin-top: 10px;
|
|
|
|
|
|
|
2024-03-01 17:43:15 +08:00
|
|
|
|
.item-record__header {
|
|
|
|
|
|
font-family: Helvetica;
|
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
|
color: #353636;
|
|
|
|
|
|
font-weight: 400;
|
|
|
|
|
|
height: 38px;
|
|
|
|
|
|
line-height: 38px;
|
|
|
|
|
|
}
|
|
|
|
|
|
.item-record__info {
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
.item-record__timeline {
|
2024-03-03 22:10:02 +08:00
|
|
|
|
margin-left: 6px;
|
|
|
|
|
|
|
2024-03-01 17:43:15 +08:00
|
|
|
|
.el-timeline {
|
|
|
|
|
|
padding-left: 0;
|
2024-03-08 18:03:21 +08:00
|
|
|
|
//min-height: 300px;
|
|
|
|
|
|
height: 300px;
|
2024-03-03 22:10:02 +08:00
|
|
|
|
overflow: auto;
|
|
|
|
|
|
|
|
|
|
|
|
&.el-timeline--hide {
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-03-01 17:43:15 +08:00
|
|
|
|
.el-timeline-item {
|
2024-03-08 18:03:21 +08:00
|
|
|
|
padding-bottom: 0;
|
2024-03-01 17:43:15 +08:00
|
|
|
|
.el-timeline-item__tail {
|
|
|
|
|
|
border-left: 2px dotted #cccccc;
|
|
|
|
|
|
margin-left: 2px;
|
|
|
|
|
|
}
|
|
|
|
|
|
.el-timeline-item__node--normal {
|
|
|
|
|
|
background-image: radial-gradient(#DE3434 20%, transparent);
|
|
|
|
|
|
outline: #F7F7F7 solid 6px;
|
|
|
|
|
|
margin-left: 2px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.el-timeline-item:last-child {
|
|
|
|
|
|
padding-bottom: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.timeline__info {
|
|
|
|
|
|
display: flex;
|
2024-03-08 18:03:21 +08:00
|
|
|
|
padding-bottom: 10px;
|
2024-03-03 22:10:02 +08:00
|
|
|
|
|
2024-03-01 17:43:15 +08:00
|
|
|
|
.timeline__info--circle {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
.info__circle {
|
|
|
|
|
|
width: 17px;
|
|
|
|
|
|
height: 17px;
|
|
|
|
|
|
margin-left: -2px;
|
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
|
background-image: radial-gradient(#DE3434 20%, transparent);
|
|
|
|
|
|
outline: rgba(222,52,52,0.30) solid 4px;
|
|
|
|
|
|
margin-top: 2px;
|
|
|
|
|
|
}
|
|
|
|
|
|
.info__line {
|
|
|
|
|
|
border-left: 2px #cccccc dotted;
|
|
|
|
|
|
height: 34px;
|
|
|
|
|
|
margin-left: 6px;
|
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.timeline__info--item {
|
|
|
|
|
|
padding-left: 13px;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: #666666;
|
|
|
|
|
|
|
|
|
|
|
|
.info--item__value {
|
|
|
|
|
|
color: #333;
|
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
|
margin-left: 4px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.timeline__item {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: #666666;
|
|
|
|
|
|
|
|
|
|
|
|
.item__value {
|
|
|
|
|
|
color: #333;
|
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
|
margin-left: 4px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-03-05 21:57:46 +08:00
|
|
|
|
.item-record__btn, .item-record__btn-disabled {
|
2024-03-03 22:10:02 +08:00
|
|
|
|
padding-right: 6px;
|
2024-03-01 17:43:15 +08:00
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
}
|
2024-03-05 21:57:46 +08:00
|
|
|
|
.item-record__btn-disabled {
|
|
|
|
|
|
cursor: no-drop;
|
|
|
|
|
|
}
|
2024-03-08 18:03:21 +08:00
|
|
|
|
.scroll-view {
|
|
|
|
|
|
width: 100%;
|
|
|
|
|
|
height: 300px;
|
|
|
|
|
|
overflow-y: scroll;
|
|
|
|
|
|
position: relative;
|
|
|
|
|
|
|
|
|
|
|
|
.scroll__item {
|
|
|
|
|
|
width: 100%;
|
|
|
|
|
|
height: 58px;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
|
|
|
|
|
|
.item-circle {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
.circle-circle {
|
|
|
|
|
|
width: 10px;
|
|
|
|
|
|
height: 10px;
|
|
|
|
|
|
margin-left: 2px;
|
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
|
background-color: #DE3434;
|
|
|
|
|
|
}
|
|
|
|
|
|
.circle-line {
|
|
|
|
|
|
border-left: 2px #cccccc dotted;
|
|
|
|
|
|
height: 34px;
|
|
|
|
|
|
margin-left: 6px;
|
|
|
|
|
|
margin-top: 6px;
|
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.item-content {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: #666666;
|
|
|
|
|
|
padding-left: 16px;
|
|
|
|
|
|
|
|
|
|
|
|
.item__value {
|
|
|
|
|
|
color: #333;
|
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
|
margin-left: 4px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.scroll-list {
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
top: 0;
|
|
|
|
|
|
left: 0;
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-03-01 17:43:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.geo-analysis__hexagon-tooltip {
|
|
|
|
|
|
position: fixed;
|
|
|
|
|
|
background-color: rgba(255,255,255,0.80);
|
|
|
|
|
|
box-shadow: 0 1px 10px 0 rgba(0,0,0,0.5);
|
|
|
|
|
|
border-radius: 2px;
|
|
|
|
|
|
min-width: 185px;
|
2024-03-11 16:32:29 +08:00
|
|
|
|
z-index: 3;
|
2024-02-26 11:51:13 +08:00
|
|
|
|
|
|
|
|
|
|
/*&.geo-analysis__hexagon-tooltip--hexagon {
|
|
|
|
|
|
}*/
|
|
|
|
|
|
&.geo-analysis__hexagon-tooltip--human {
|
|
|
|
|
|
.icon__box {
|
|
|
|
|
|
background-color: #233447;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
&.geo-analysis__hexagon-tooltip--base-station {
|
|
|
|
|
|
.icon__box {
|
|
|
|
|
|
background-color: #585B5F;
|
|
|
|
|
|
}
|
|
|
|
|
|
.hexagon-tooltip__body {
|
|
|
|
|
|
.body__item .item__label {
|
|
|
|
|
|
width: 140px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.hexagon-tooltip__header {
|
|
|
|
|
|
position: relative;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
padding: 10px 0 10px 63px;
|
|
|
|
|
|
color: white;
|
|
|
|
|
|
|
|
|
|
|
|
.header__icon {
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
left: 14px;
|
|
|
|
|
|
top: 16px;
|
|
|
|
|
|
|
|
|
|
|
|
.icon__box {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
height: 32px;
|
|
|
|
|
|
width: 32px;
|
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.header__title {
|
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
|
}
|
|
|
|
|
|
.header__content {
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.hexagon-tooltip__body {
|
|
|
|
|
|
padding: 8px 18px;
|
|
|
|
|
|
|
2024-03-05 16:17:14 +08:00
|
|
|
|
.body__timeline {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: row;
|
|
|
|
|
|
padding: 3px 0;
|
|
|
|
|
|
|
|
|
|
|
|
.timeline-symbol {
|
|
|
|
|
|
margin-top: 6px;
|
|
|
|
|
|
margin-right: 6px;
|
|
|
|
|
|
width: 8px;
|
|
|
|
|
|
height: 8px;
|
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
|
background-color: #DE3434;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
.body__item {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
|
|
|
|
|
|
.item__label {
|
|
|
|
|
|
padding-right: 10px;
|
|
|
|
|
|
text-align: right;
|
|
|
|
|
|
width: 60px;
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: #353636;
|
|
|
|
|
|
}
|
|
|
|
|
|
.item__value {
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
color: #233447;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.body__tracking {
|
|
|
|
|
|
padding-top: 6px;
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: #38ACD2;
|
|
|
|
|
|
text-decoration: underline;
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-03-01 18:13:04 +08:00
|
|
|
|
@keyframes pulse {
|
|
|
|
|
|
0% {
|
|
|
|
|
|
opacity: 1;
|
|
|
|
|
|
}
|
|
|
|
|
|
70% {
|
2024-03-03 22:10:02 +08:00
|
|
|
|
opacity: 0.6;
|
2024-03-01 18:13:04 +08:00
|
|
|
|
}
|
|
|
|
|
|
100% {
|
|
|
|
|
|
opacity: 1;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-02-26 11:51:13 +08:00
|
|
|
|
</style>
|