| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227 |
- <template>
- <el-card class="dr-task" body-class="task-card" shadow="never">
- <template #header>
- 视频任务
- <!-- TODO @fan:看看,怎么优化下这个样子哈。 -->
- <!-- <el-button @click="handleViewPublic">视频作品</el-button>-->
- </template>
- <!-- 视频列表 -->
- <div class="task-video-list" ref="videoListRef">
- <VideoCard
- v-for="video in videoList"
- :key="video.id"
- :detail="video"
- @on-btn-click="handleVideoButtonClick"
- />
- </div>
- <div class="task-video-pagination">
- <Pagination
- :total="pageTotal"
- v-model:page="queryParams.pageNo"
- v-model:limit="queryParams.pageSize"
- @pagination="getVideoList"
- />
- </div>
- </el-card>
- <!-- 视频详情 -->
- <VideoDetail
- :show="isShowVideoDetail"
- :id="showVideoDetailId"
- @handle-drawer-close="handleDetailClose"
- />
- </template>
- <script setup lang="ts">
- import {
- VideoApi,
- VideoVO
- } from '@/api/ai/video'
- import VideoDetail from './VideoDetail.vue'
- import VideoCard from './VideoCard.vue'
- import { ElLoading, LoadingOptionsResolved } from 'element-plus'
- import { AiVideoStatusEnum } from '@/views/ai/utils/constants'
- import download from '@/utils/download'
- const message = useMessage() // 消息弹窗
- const router = useRouter() // 路由
- // 视频分页相关的参数
- const queryParams = reactive({
- pageNo: 1,
- pageSize: 10
- })
- const pageTotal = ref<number>(0) // page size
- const videoList = ref<VideoVO[]>([]) // video 列表
- const videoListLoadingInstance = ref<any>() // video 列表是否正在加载中
- const videoListRef = ref<any>() // ref
- // 视频轮询相关的参数(正在生成中的)
- const inProgressVideoMap = ref<{}>({}) // 监听的 video 映射,一般是生成中(需要轮询),key 为 video 编号,value 为 video
- const inProgressTimer = ref<any>() // 生成中的 video 定时器,轮询生成进展
- // 视频详情相关的参数
- const isShowVideoDetail = ref<boolean>(false) // 视频详情是否展示
- const showVideoDetailId = ref<number>(0) // 视频详情的视频编号
- /** 处理查看视频作品 */
- const handleViewPublic = () => {
- router.push({
- name: 'AiVideoSquare'
- })
- }
- /** 查看视频的详情 */
- const handleDetailOpen = async () => {
- isShowVideoDetail.value = true
- }
- /** 关闭视频的详情 */
- const handleDetailClose = async () => {
- isShowVideoDetail.value = false
- }
- /** 获得 video 视频列表 */
- const getVideoList = async () => {
- try {
- // 1. 加载视频列表
- videoListLoadingInstance.value = ElLoading.service({
- target: videoListRef.value,
- text: '加载中...'
- } as LoadingOptionsResolved)
- const { list, total } = await VideoApi.getVideoPageMy(queryParams)
- videoList.value = list
- pageTotal.value = total
- // 2. 计算需要轮询的视频
- const newWatVideos = {}
- videoList.value.forEach((item) => {
- if (item.status === AiVideoStatusEnum.IN_PROGRESS) {
- newWatVideos[item.id] = item
- }
- })
- inProgressVideoMap.value = newWatVideos
- } finally {
- // 关闭正在“加载中”的 Loading
- if (videoListLoadingInstance.value) {
- videoListLoadingInstance.value.close()
- videoListLoadingInstance.value = null
- }
- }
- }
- /** 轮询生成中的 video 列表 */
- const refreshWatchVideos = async () => {
- const videoIds = Object.keys(inProgressVideoMap.value).map(Number)
- if (videoIds.length == 0) {
- return
- }
- const list = (await VideoApi.getVideoListMyByIds(videoIds)) as VideoVO[]
- const newWatchVideos = {}
- list.forEach((video) => {
- if (video.status === AiVideoStatusEnum.IN_PROGRESS) {
- newWatchVideos[video.id] = video
- } else {
- const index = videoList.value.findIndex((oldVideo) => video.id === oldVideo.id)
- if (index >= 0) {
- // 更新 videoList
- videoList.value[index] = video
- }
- }
- })
- inProgressVideoMap.value = newWatchVideos
- }
- /** 视频的点击事件 */
- const handleVideoButtonClick = async (type: string, videoDetail: VideoVO) => {
- // 详情
- if (type === 'more') {
- showVideoDetailId.value = videoDetail.id
- await handleDetailOpen()
- return
- }
- // 删除
- if (type === 'delete') {
- await message.confirm(`是否删除照片?`)
- await VideoApi.deleteVideoMy(videoDetail.id)
- await getVideoList()
- message.success('删除成功!')
- return
- }
- // 下载
- if (type === 'download') {
- window.open(videoDetail.videoUrl)
- // await download.video({ url: videoDetail.videoUrl })
- return
- }
- // 重新生成
- if (type === 'regeneration') {
- // await emits('onRegeneration', videoDetail)
- return
- }
- }
- defineExpose({ getVideoList }) // 暴露组件方法
- const emits = defineEmits(['onRegeneration'])
- /** 组件挂在的时候 */
- onMounted(async () => {
- // 获取 video 列表
- await getVideoList()
- // 自动刷新 video 列表
- inProgressTimer.value = setInterval(async () => {
- await refreshWatchVideos()
- }, 1000 * 3)
- })
- /** 组件取消挂在的时候 */
- onUnmounted(async () => {
- if (inProgressTimer.value) {
- clearInterval(inProgressTimer.value)
- }
- })
- </script>
- <style lang="scss">
- .dr-task {
- width: 100%;
- height: 100%;
- }
- .task-card {
- margin: 0;
- padding: 0;
- height: 100%;
- position: relative;
- }
- .task-video-list {
- position: relative;
- display: flex;
- flex-direction: row;
- flex-wrap: wrap;
- align-content: flex-start;
- height: 100%;
- overflow: auto;
- padding: 20px 20px 140px;
- box-sizing: border-box; /* 确保内边距不会增加高度 */
- > div {
- margin-right: 20px;
- margin-bottom: 20px;
- }
- > div:last-of-type {
- //margin-bottom: 100px;
- }
- }
- .task-video-pagination {
- position: absolute;
- bottom: 60px;
- height: 50px;
- line-height: 90px;
- width: 100%;
- z-index: 999;
- background-color: #ffffff;
- display: flex;
- flex-direction: row;
- justify-content: center;
- align-items: center;
- }
- </style>
|