| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272 |
- <template>
- <div class="project-wayline-wrapper height-100">
- <a-spin :spinning="loading" :delay="300" tip="downloading" size="large">
- <div style="height: 50px; line-height: 50px; border-bottom: 1px solid #4f4f4f; font-weight: 450;">
- <a-row>
- <a-col :span="1"></a-col>
- <a-col :span="15">Flight Route Library</a-col>
- <a-col :span="8" v-if="importVisible" class="flex-row flex-justify-end flex-align-center">
- <a-upload name="file" :multiple="false" :before-upload="beforeUpload" :show-upload-list="false"
- :customRequest="uploadFile">
- <a-button type="text" style="color: white;">
- <SelectOutlined />
- </a-button>
- </a-upload>
- </a-col>
- </a-row>
- </div>
- <div :style="{ height: height + 'px' }" class="scrollbar">
- <div id="data" class="height-100 uranus-scrollbar" v-if="waylinesData.data.length !== 0" @scroll="onScroll">
- <div v-for="wayline in waylinesData.data" :key="wayline.id">
- <div class="wayline-panel" style="padding-top: 5px;" @click="selectRoute(wayline)">
- <div class="title">
- <a-tooltip :title="wayline.name">
- <div class="pr10"
- style="width: 120px; white-space: nowrap; text-overflow: ellipsis; overflow: hidden;">{{
- wayline.name }}</div>
- </a-tooltip>
- <div class="ml10">
- <UserOutlined />
- </div>
- <a-tooltip :title="wayline.user_name">
- <div class="ml5 pr10"
- style="width: 80px; white-space: nowrap; text-overflow: ellipsis; overflow: hidden;">{{
- wayline.user_name }}</div>
- </a-tooltip>
- <div class="fz20">
- <a-dropdown>
- <a style="color: white;">
- <EllipsisOutlined />
- </a>
- <template #overlay>
- <a-menu theme="dark" class="more" style="background: #3c3c3c;">
- <a-menu-item @click="downloadWayline(wayline.id, wayline.name)">
- <span>Download</span>
- </a-menu-item>
- <a-menu-item @click="showWaylineTip(wayline.id)">
- <span>Delete</span>
- </a-menu-item>
- </a-menu>
- </template>
- </a-dropdown>
- </div>
- </div>
- <div class="ml10 mt5" style="color: hsla(0,0%,100%,0.65);">
- <span>
- <RocketOutlined />
- </span>
- <span class="ml5">{{ DEVICE_NAME[wayline.drone_model_key] }}</span>
- <span class="ml10">
- <CameraFilled style="border-top: 1px solid; padding-top: -3px;" />
- </span>
- <span class="ml5" v-for="payload in wayline.payload_model_keys" :key="payload.id">
- {{ DEVICE_NAME[payload] }}
- </span>
- </div>
- <div class="mt5 ml10" style="color: hsla(0,0%,100%,0.35);">
- <span class="mr10">Update at {{ new Date(wayline.update_time).toLocaleString() }}</span>
- </div>
- </div>
- </div>
- </div>
- <div v-else>
- <a-empty :image-style="{ height: '60px', marginTop: '60px' }" />
- </div>
- <a-modal v-model:visible="deleteTip" width="450px" :closable="false" :maskClosable="false" centered
- :okButtonProps="{ danger: true }" @ok="deleteWayline">
- <p class="pt10 pl20" style="height: 50px;">Wayline file is unrecoverable once deleted. Continue?</p>
- <template #title>
- <div class="flex-row flex-justify-center">
- <span>Delete</span>
- </div>
- </template>
- </a-modal>
- </div>
- </a-spin>
- </div>
- </template>
- <script lang="ts" setup>
- import { message } from 'ant-design-vue'
- import { onMounted, reactive, ref } from 'vue'
- import { deleteWaylineFile, downloadWaylineFile, getWaylineFiles, importKmzFile } from '/@/api/wayline'
- import { ELocalStorageKey, ERouterName } from '/@/types'
- import { EllipsisOutlined, RocketOutlined, CameraFilled, UserOutlined, SelectOutlined } from '@ant-design/icons-vue'
- import { DEVICE_NAME } from '/@/types/device'
- import { useMyStore } from '/@/store'
- import { WaylineFile } from '/@/types/wayline'
- import { downloadFile } from '/@/utils/common'
- import { IPage } from '/@/api/http/type'
- import { load } from '@amap/amap-jsapi-loader'
- import { getRoot } from '/@/root'
- const loading = ref(false)
- const store = useMyStore()
- const pagination: IPage = {
- page: 1,
- total: -1,
- page_size: 10
- }
- const waylinesData = reactive({
- data: [] as WaylineFile[]
- })
- const root = getRoot()
- const workspaceId = localStorage.getItem(ELocalStorageKey.WorkspaceId)!
- const deleteTip = ref(false)
- const deleteWaylineId = ref<string>('')
- const canRefresh = ref(true)
- const importVisible = ref<boolean>(root.$router.currentRoute.value.name === ERouterName.WAYLINE)
- const height = ref()
- onMounted(() => {
- const parent = document.getElementsByClassName('scrollbar').item(0)?.parentNode as HTMLDivElement
- height.value = document.body.clientHeight - parent.firstElementChild!.clientHeight
- getWaylines()
- const key = setInterval(() => {
- const data = document.getElementById('data')?.lastElementChild as HTMLDivElement
- if (pagination.total === 0 || Math.ceil(pagination.total / pagination.page_size) <= pagination.page || height.value <= data?.clientHeight + data?.offsetTop) {
- clearInterval(key)
- return
- }
- pagination.page++
- getWaylines()
- }, 1000)
- })
- function getWaylines() {
- if (!canRefresh.value) {
- return
- }
- canRefresh.value = false
- getWaylineFiles(workspaceId, {
- page: pagination.page,
- page_size: pagination.page_size,
- order_by: 'update_time desc'
- }).then(res => {
- if (res.code !== 0) {
- return
- }
- waylinesData.data = [...waylinesData.data, ...res.data.list]
- pagination.total = res.data.pagination.total
- pagination.page = res.data.pagination.page
- }).finally(() => {
- canRefresh.value = true
- })
- }
- function showWaylineTip(waylineId: string) {
- deleteWaylineId.value = waylineId
- deleteTip.value = true
- }
- function deleteWayline() {
- deleteWaylineFile(workspaceId, deleteWaylineId.value).then(res => {
- if (res.code === 0) {
- message.success('Wayline file deleted')
- }
- deleteWaylineId.value = ''
- deleteTip.value = false
- pagination.total = 0
- pagination.page = 1
- waylinesData.data = []
- getWaylines()
- })
- }
- function downloadWayline(waylineId: string, fileName: string) {
- loading.value = true
- downloadWaylineFile(workspaceId, waylineId).then(res => {
- if (!res) {
- return
- }
- const data = new Blob([res], { type: 'application/zip' })
- downloadFile(data, fileName + '.kmz')
- }).finally(() => {
- loading.value = false
- })
- }
- function selectRoute(wayline: WaylineFile) {
- store.commit('SET_SELECT_WAYLINE_INFO', wayline)
- }
- function onScroll(e: any) {
- const element = e.srcElement
- if (element.scrollTop + element.clientHeight >= element.scrollHeight - 5 && Math.ceil(pagination.total / pagination.page_size) > pagination.page && canRefresh.value) {
- pagination.page++
- getWaylines()
- }
- }
- interface FileItem {
- uid: string;
- name?: string;
- status?: string;
- response?: string;
- url?: string;
- }
- interface FileInfo {
- file: FileItem;
- fileList: FileItem[];
- }
- const fileList = ref<FileItem[]>([])
- function beforeUpload(file: FileItem) {
- fileList.value = [file]
- loading.value = true
- return true
- }
- const uploadFile = async () => {
- fileList.value.forEach(async (file: FileItem) => {
- const fileData = new FormData()
- fileData.append('file', file, file.name)
- await importKmzFile(workspaceId, fileData).then((res) => {
- if (res.code === 0) {
- message.success(`${file.name} file uploaded successfully`)
- canRefresh.value = true
- pagination.total = 0
- pagination.page = 1
- waylinesData.data = []
- getWaylines()
- }
- }).finally(() => {
- loading.value = false
- fileList.value = []
- })
- })
- }
- </script>
- <style lang="scss" scoped>
- .wayline-panel {
- background: #3c3c3c;
- margin-left: auto;
- margin-right: auto;
- margin-top: 10px;
- height: 90px;
- width: 95%;
- font-size: 13px;
- border-radius: 2px;
- cursor: pointer;
- .title {
- display: flex;
- flex-direction: row;
- align-items: center;
- height: 30px;
- font-weight: bold;
- margin: 0px 10px 0 10px;
- }
- }
- .uranus-scrollbar {
- overflow: auto;
- scrollbar-width: thin;
- scrollbar-color: #c5c8cc transparent;
- }
- </style>
|