366 lines
10 KiB
366 lines
10 KiB
<template> |
|
<div class="header">计划库</div> |
|
<div class="plan-panel-wrapper"> |
|
<a-table class="plan-table" :columns="columns" :data-source="plansData.data" row-key="job_id" |
|
:pagination="paginationProp" :scroll="{ x: true, y: `calc(100vh - 196px)`}" @change="refreshData"> |
|
<!-- 执行时间 --> |
|
<template #duration="{ record }"> |
|
<div class="flex-row" style="white-space: pre-wrap"> |
|
<div> |
|
<div>{{ formatTaskTime(record.begin_time) }}</div> |
|
<div>{{ formatTaskTime(record.end_time) }}</div> |
|
</div> |
|
<div class="ml10"> |
|
<div>{{ formatTaskTime(record.execute_time) }}</div> |
|
<div>{{ formatTaskTime(record.completed_time) }}</div> |
|
</div> |
|
</div> |
|
</template> |
|
<!-- 状态 --> |
|
<template #status="{ record }"> |
|
<div> |
|
<div class="flex-display flex-align-center"> |
|
<span class="circle-icon" :style="{backgroundColor: formatTaskStatus(record).color}"></span> |
|
{{ formatTaskStatus(record).text }} |
|
<a-tooltip v-if="!!record.code" placement="bottom" arrow-point-at-center > |
|
<template #title> |
|
<div>{{ getCodeMessage(record.code) }}</div> |
|
</template> |
|
<exclamation-circle-outlined class="ml5" :style="{color: commonColor.WARN, fontSize: '16px' }"/> |
|
</a-tooltip> |
|
</div> |
|
<div v-if="record.status === TaskStatus.Carrying"> |
|
<a-progress :percent="record.progress || 0" /> |
|
</div> |
|
</div> |
|
</template> |
|
<!-- 任务类型 --> |
|
<template #taskType="{ record }"> |
|
<div>{{ formatTaskType(record) }}</div> |
|
</template> |
|
<!-- 失控动作 --> |
|
<template #lostAction="{ record }"> |
|
<div>{{ formatLostAction(record) }}</div> |
|
</template> |
|
<!-- 媒体上传状态 --> |
|
<template #media_upload="{ record }"> |
|
<div> |
|
<div class="flex-display flex-align-center"> |
|
<span class="circle-icon" :style="{backgroundColor: formatMediaTaskStatus(record).color}"></span> |
|
{{ formatMediaTaskStatus(record).text }} |
|
</div> |
|
<div class="pl15"> |
|
{{ formatMediaTaskStatus(record).number }} |
|
<a-tooltip v-if="formatMediaTaskStatus(record).status === MediaStatus.ToUpload" placement="bottom" arrow-point-at-center > |
|
<template #title> |
|
<div>立即下载</div> |
|
</template> |
|
<UploadOutlined class="ml5" :style="{color: commonColor.BLUE, fontSize: '16px' }" @click="onUploadMediaFileNow(record.job_id)"/> |
|
</a-tooltip> |
|
</div> |
|
</div> |
|
</template> |
|
<!-- 操作 --> |
|
<template #action="{ record }"> |
|
<div class="action-area"> |
|
<a-popconfirm |
|
v-if="record.status === TaskStatus.Wait" |
|
title="Are you sure you want to delete flight task?" |
|
ok-text="Yes" |
|
cancel-text="No" |
|
@confirm="onDeleteTask(record.job_id)" |
|
> |
|
<a-button type="primary" size="small">删除</a-button> |
|
</a-popconfirm> |
|
<a-popconfirm |
|
v-if="record.status === TaskStatus.Carrying" |
|
title="Are you sure you want to suspend?" |
|
ok-text="Yes" |
|
cancel-text="No" |
|
@confirm="onSuspendTask(record.job_id)" |
|
> |
|
<a-button type="primary" size="small">暂停</a-button> |
|
</a-popconfirm> |
|
<a-popconfirm |
|
v-if="record.status === TaskStatus.Paused" |
|
title="Are you sure you want to resume?" |
|
ok-text="Yes" |
|
cancel-text="No" |
|
@confirm="onResumeTask(record.job_id)" |
|
> |
|
<a-button type="primary" size="small">开始</a-button> |
|
</a-popconfirm> |
|
</div> |
|
</template> |
|
</a-table> |
|
</div> |
|
</template> |
|
|
|
<script setup lang="ts"> |
|
import { reactive, ref } from '@vue/reactivity' |
|
import { message } from 'ant-design-vue' |
|
import { TableState } from 'ant-design-vue/lib/table/interface' |
|
import { onMounted } from 'vue' |
|
import { IPage } from '/@/api/http/type' |
|
import { deleteTask, updateTaskStatus, UpdateTaskStatus, getWaylineJobs, Task, uploadMediaFileNow } from '/@/api/wayline' |
|
import { useMyStore } from '/@/store' |
|
import { ELocalStorageKey } from '/@/types/enums' |
|
import { useFormatTask } from './use-format-task' |
|
import { TaskStatus, TaskProgressInfo, TaskProgressStatus, TaskProgressWsStatusMap, MediaStatus, MediaStatusProgressInfo, TaskMediaHighestPriorityProgressInfo } from '/@/types/task' |
|
import { useTaskWsEvent } from './use-task-ws-event' |
|
import { getErrorMessage } from '/@/utils/error-code/index' |
|
import { commonColor } from '/@/utils/color' |
|
import { ExclamationCircleOutlined, UploadOutlined } from '@ant-design/icons-vue' |
|
|
|
const store = useMyStore() |
|
const workspaceId = localStorage.getItem(ELocalStorageKey.WorkspaceId)! |
|
|
|
const body: IPage = { |
|
page: 1, |
|
total: 0, |
|
page_size: 50 |
|
} |
|
const paginationProp = reactive({ |
|
pageSizeOptions: ['20', '50', '100'], |
|
showQuickJumper: true, |
|
showSizeChanger: true, |
|
pageSize: 50, |
|
current: 1, |
|
total: 0 |
|
}) |
|
|
|
const columns = [ |
|
{ |
|
title: '计划|实际时间', |
|
dataIndex: 'duration', |
|
width: 200, |
|
slots: { customRender: 'duration' }, |
|
}, |
|
{ |
|
title: '执行状态', |
|
key: 'status', |
|
width: 150, |
|
slots: { customRender: 'status' } |
|
}, |
|
{ |
|
title: '计划名称', |
|
dataIndex: 'job_name', |
|
width: 100, |
|
}, |
|
{ |
|
title: '类型', |
|
dataIndex: 'taskType', |
|
width: 100, |
|
slots: { customRender: 'taskType' }, |
|
}, |
|
{ |
|
title: '航线名称', |
|
dataIndex: 'file_name', |
|
width: 100, |
|
}, |
|
{ |
|
title: '机场名称', |
|
dataIndex: 'dock_name', |
|
width: 100, |
|
ellipsis: true |
|
}, |
|
{ |
|
title: '返航高度(m)', |
|
dataIndex: 'rth_altitude', |
|
width: 120, |
|
}, |
|
{ |
|
title: '失联动作', |
|
dataIndex: 'out_of_control_action', |
|
width: 120, |
|
slots: { customRender: 'lostAction' }, |
|
}, |
|
{ |
|
title: '创建者', |
|
dataIndex: 'username', |
|
width: 120, |
|
}, |
|
{ |
|
title: '媒体文件下载', |
|
key: 'media_upload', |
|
width: 160, |
|
slots: { customRender: 'media_upload' } |
|
}, |
|
{ |
|
title: '操作', |
|
width: 120, |
|
slots: { customRender: 'action' } |
|
} |
|
] |
|
type Pagination = TableState['pagination'] |
|
|
|
const plansData = reactive({ |
|
data: [] as Task[] |
|
}) |
|
|
|
const { formatTaskType, formatTaskTime, formatLostAction, formatTaskStatus, formatMediaTaskStatus } = useFormatTask() |
|
|
|
// 设备任务执行进度更新 |
|
function onTaskProgressWs (data: TaskProgressInfo) { |
|
const { bid, output } = data |
|
if (output) { |
|
const { status, progress } = output || {} |
|
const taskItem = plansData.data.find(task => task.job_id === bid) |
|
if (!taskItem) return |
|
if (status) { |
|
taskItem.status = TaskProgressWsStatusMap[status] |
|
// 执行中,更新进度 |
|
if (status === TaskProgressStatus.Sent || status === TaskProgressStatus.inProgress) { |
|
taskItem.progress = progress?.percent || 0 |
|
} else if ([TaskProgressStatus.Rejected, TaskProgressStatus.Canceled, TaskProgressStatus.Timeout, TaskProgressStatus.Failed, TaskProgressStatus.OK].includes(status)) { |
|
getPlans() |
|
} |
|
} |
|
} |
|
} |
|
|
|
// 媒体上传进度更新 |
|
function onTaskMediaProgressWs (data: MediaStatusProgressInfo) { |
|
const { media_count: mediaCount, uploaded_count: uploadedCount, job_id: jobId } = data |
|
if (isNaN(mediaCount) || isNaN(uploadedCount) || !jobId) { |
|
return |
|
} |
|
const taskItem = plansData.data.find(task => task.job_id === jobId) |
|
if (!taskItem) return |
|
if (mediaCount === uploadedCount) { |
|
taskItem.uploading = false |
|
} else { |
|
taskItem.uploading = true |
|
} |
|
taskItem.media_count = mediaCount |
|
taskItem.uploaded_count = uploadedCount |
|
} |
|
|
|
function onoTaskMediaHighestPriorityWS (data: TaskMediaHighestPriorityProgressInfo) { |
|
const { pre_job_id: preJobId, job_id: jobId } = data |
|
const preTaskItem = plansData.data.find(task => task.job_id === preJobId) |
|
const taskItem = plansData.data.find(task => task.job_id === jobId) |
|
if (preTaskItem) { |
|
preTaskItem.uploading = false |
|
} |
|
if (taskItem) { |
|
taskItem.uploading = true |
|
} |
|
} |
|
|
|
function getCodeMessage (code: number) { |
|
return getErrorMessage(code) + `(code: ${code})` |
|
} |
|
|
|
useTaskWsEvent({ |
|
onTaskProgressWs, |
|
onTaskMediaProgressWs, |
|
onoTaskMediaHighestPriorityWS, |
|
}) |
|
|
|
onMounted(() => { |
|
getPlans() |
|
}) |
|
|
|
function getPlans () { |
|
getWaylineJobs(workspaceId, body).then(res => { |
|
if (res.code !== 0) { |
|
return |
|
} |
|
plansData.data = res.data.list |
|
paginationProp.total = res.data.pagination.total |
|
paginationProp.current = res.data.pagination.page |
|
}) |
|
} |
|
|
|
function refreshData (page: Pagination) { |
|
body.page = page?.current! |
|
body.page_size = page?.pageSize! |
|
getPlans() |
|
} |
|
|
|
// 删除任务 |
|
async function onDeleteTask (jobId: string) { |
|
const { code } = await deleteTask(workspaceId, { |
|
job_id: jobId |
|
}) |
|
if (code === 0) { |
|
message.success('Deleted successfully') |
|
getPlans() |
|
} |
|
} |
|
|
|
// 挂起任务 |
|
async function onSuspendTask (jobId: string) { |
|
const { code } = await updateTaskStatus(workspaceId, { |
|
job_id: jobId, |
|
status: UpdateTaskStatus.Suspend |
|
}) |
|
if (code === 0) { |
|
message.success('Suspended successfully') |
|
getPlans() |
|
} |
|
} |
|
|
|
// 解除挂起任务 |
|
async function onResumeTask (jobId: string) { |
|
const { code } = await updateTaskStatus(workspaceId, { |
|
job_id: jobId, |
|
status: UpdateTaskStatus.Resume |
|
}) |
|
if (code === 0) { |
|
message.success('Resumed successfully') |
|
getPlans() |
|
} |
|
} |
|
|
|
// 立即上传媒体 |
|
async function onUploadMediaFileNow (jobId: string) { |
|
const { code } = await uploadMediaFileNow(workspaceId, jobId) |
|
if (code === 0) { |
|
message.success('Upload Media File successfully') |
|
getPlans() |
|
} |
|
} |
|
</script> |
|
|
|
<style lang="scss" scoped> |
|
.plan-panel-wrapper { |
|
width: 100%; |
|
padding: 16px; |
|
.plan-table { |
|
background: #fff; |
|
margin-top: 10px; |
|
width:100%; |
|
} |
|
.action-area { |
|
|
|
&::v-deep { |
|
.ant-btn { |
|
margin-right: 10px; |
|
margin-bottom: 10px; |
|
} |
|
} |
|
} |
|
|
|
.circle-icon { |
|
display: inline-block; |
|
width: 12px; |
|
height: 12px; |
|
margin-right: 3px; |
|
border-radius: 50%; |
|
vertical-align: middle; |
|
flex-shrink: 0; |
|
} |
|
} |
|
.header { |
|
width: 100%; |
|
height: 60px; |
|
background: #fff; |
|
padding: 16px; |
|
font-size: 20px; |
|
font-weight: bold; |
|
text-align: start; |
|
color: #000; |
|
} |
|
</style>
|
|
|