ImageTask.vue 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. <template>
  2. <el-card class="dr-task" body-class="task-card" shadow="never">
  3. <template #header>绘画任务</template>
  4. <ImageTaskCard
  5. v-for="image in imageList"
  6. :key="image"
  7. :image-detail="image"
  8. @on-btn-click="handlerImageBtnClick"
  9. @on-mj-btn-click="handlerImageMjBtnClick"/>
  10. </el-card>
  11. <!-- 图片 detail 抽屉 -->
  12. <ImageDetailDrawer
  13. :show="isShowImageDetail"
  14. :id="showImageDetailId"
  15. @handler-drawer-close="handlerDrawerClose"
  16. />
  17. </template>
  18. <script setup lang="ts">
  19. import {ImageApi, ImageDetailVO, ImageMjActionVO, ImageMjButtonsVO} from '@/api/ai/image';
  20. import ImageDetailDrawer from './ImageDetailDrawer.vue'
  21. import ImageTaskCard from './ImageTaskCard.vue'
  22. const message = useMessage() // 消息弹窗
  23. const imageList = ref<ImageDetailVO[]>([]) // image 列表
  24. const imageListInterval = ref<any>() // image 列表定时器,刷新列表
  25. const isShowImageDetail = ref<boolean>(false) // 是否显示 task 详情
  26. const showImageDetailId = ref<number>(0) // 是否显示 task 详情
  27. /** 抽屉 - close */
  28. const handlerDrawerClose = async () => {
  29. isShowImageDetail.value = false
  30. }
  31. /** 任务 - detail */
  32. const handlerDrawerOpen = async () => {
  33. isShowImageDetail.value = true
  34. }
  35. /**
  36. * 获取 - image 列表
  37. */
  38. const getImageList = async () => {
  39. const { list } = await ImageApi.getImageList({pageNo: 1, pageSize: 20})
  40. imageList.value = list
  41. }
  42. /** 图片 - btn click */
  43. const handlerImageBtnClick = async (type, imageDetail: ImageDetailVO) => {
  44. // 获取 image detail id
  45. showImageDetailId.value = imageDetail.id
  46. // 处理不用 btn
  47. if (type === 'more') {
  48. await handlerDrawerOpen()
  49. } else if (type === 'delete') {
  50. await message.confirm(`是否删除照片?`)
  51. await ImageApi.deleteImage(imageDetail.id)
  52. await getImageList()
  53. await message.success("删除成功!")
  54. } else if (type === 'download') {
  55. await downloadImage(imageDetail.picUrl)
  56. }
  57. }
  58. /** 图片 - mj btn click */
  59. const handlerImageMjBtnClick = async (button: ImageMjButtonsVO, imageDetail: ImageDetailVO) => {
  60. // 1、构建 params 参数
  61. const params = {
  62. id: imageDetail.id,
  63. customId: button.customId,
  64. } as ImageMjActionVO
  65. // 2、发送 action
  66. await ImageApi.midjourneyAction(params)
  67. // 3、刷新列表
  68. await getImageList()
  69. }
  70. /** 下载 - image */
  71. // TODO @fan:貌似可以考虑抽到 download 里面,作为一个方法
  72. const downloadImage = async (imageUrl) => {
  73. const image = new Image()
  74. image.setAttribute('crossOrigin', 'anonymous')
  75. image.src = imageUrl
  76. image.onload = () => {
  77. const canvas = document.createElement('canvas')
  78. canvas.width = image.width
  79. canvas.height = image.height
  80. const ctx = canvas.getContext('2d') as CanvasDrawImage
  81. ctx.drawImage(image, 0, 0, image.width, image.height)
  82. const url = canvas.toDataURL('image/png')
  83. const a = document.createElement('a')
  84. a.href = url
  85. a.download = 'image.png'
  86. a.click()
  87. }
  88. }
  89. /** 暴露组件方法 */
  90. defineExpose({getImageList})
  91. /** 组件挂在的时候 */
  92. onMounted(async () => {
  93. // 获取 image 列表
  94. await getImageList()
  95. // 自动刷新 image 列表
  96. imageListInterval.value = setInterval(async () => {
  97. await getImageList()
  98. }, 1000 * 20)
  99. })
  100. /** 组件取消挂在的时候 */
  101. onUnmounted(async () => {
  102. if (imageListInterval.value) {
  103. clearInterval(imageListInterval.value)
  104. }
  105. })
  106. </script>
  107. <style lang="scss">
  108. .task-card {
  109. display: flex;
  110. flex-direction: row;
  111. flex-wrap: wrap;
  112. align-content: flex-start;
  113. height: 100%;
  114. overflow: auto;
  115. >div {
  116. margin-right: 20px;
  117. margin-bottom: 20px;
  118. }
  119. }
  120. </style>
  121. <style scoped lang="scss">
  122. .dr-task {
  123. width: 100%;
  124. height: 100%;
  125. }
  126. </style>