ScreenElement.vue 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. <template>
  2. <div
  3. class="screen-element"
  4. :class="{ link: elementInfo.link }"
  5. :id="`screen-element-${elementInfo.id}`"
  6. :style="{
  7. zIndex: elementIndex,
  8. color: theme.fontColor,
  9. fontFamily: theme.fontName,
  10. fontSize: theme.fontSize,
  11. visibility: needWaitAnimation ? 'hidden' : 'visible'
  12. }"
  13. :title="elementInfo.link?.target || ''"
  14. @click="$event => openLink($event)"
  15. >
  16. <component :is="currentElementComponent" :needWaitAnimation="needWaitAnimation" :elementInfo="elementInfo" />
  17. </div>
  18. </template>
  19. <script lang="ts" setup>
  20. import { computed } from "vue"
  21. import { storeToRefs } from "pinia"
  22. import { useSlidesStore } from "@/store"
  23. import { ElementTypes, ElementSubtypeTypes, type PPTElement } from "@/types/slides"
  24. import BaseImageElement from "@/views/components/element/ImageElement/BaseImageElement.vue"
  25. import BaseTextElement from "@/views/components/element/TextElement/BaseTextElement.vue"
  26. import BaseShapeElement from "@/views/components/element/ShapeElement/BaseShapeElement.vue"
  27. import BaseLineElement from "@/views/components/element/LineElement/BaseLineElement.vue"
  28. import BaseChartElement from "@/views/components/element/ChartElement/BaseChartElement.vue"
  29. import BaseTableElement from "@/views/components/element/TableElement/BaseTableElement.vue"
  30. import BaseLatexElement from "@/views/components/element/LatexElement/BaseLatexElement.vue"
  31. import ScreenVideoElement from "@/views/components/element/VideoElement/ScreenVideoElement.vue"
  32. import ScreenAudioElement from "@/views/components/element/AudioElement/ScreenAudioElement.vue"
  33. import ScreenCloudCoachElement from "@/views/components/element/cloudCoachElement/ScreenCloudCoachElement.vue"
  34. import ScreenEnjoyElement from "@/views/components/element/enjoyElement/ScreenEnjoyElement.vue"
  35. import ScreenListeningPracticeElement from "@/views/components/element/listeningPracticeElement/ScreenListeningPracticeElement.vue"
  36. import ScreenRhythmPracticeElement from "@/views/components/element/rhythmPracticeElement/ScreenRhythmPracticeElement.vue"
  37. const props = defineProps<{
  38. elementInfo: PPTElement
  39. elementIndex: number
  40. animationIndex: number
  41. turnSlideToId: (id: string) => void
  42. manualExitFullscreen: () => void
  43. }>()
  44. const currentElementComponent = computed<unknown>(() => {
  45. const elementTypeMap = {
  46. [ElementTypes.IMAGE]: BaseImageElement,
  47. [ElementTypes.TEXT]: BaseTextElement,
  48. [ElementTypes.SHAPE]: BaseShapeElement,
  49. [ElementTypes.LINE]: BaseLineElement,
  50. [ElementTypes.CHART]: BaseChartElement,
  51. [ElementTypes.TABLE]: BaseTableElement,
  52. [ElementTypes.LATEX]: BaseLatexElement,
  53. [ElementTypes.ELF]: null
  54. }
  55. const elementSubtypeMap = {
  56. [ElementSubtypeTypes.AUDIO]: ScreenAudioElement,
  57. [ElementSubtypeTypes.VIDEO]: ScreenVideoElement,
  58. [ElementSubtypeTypes.SING_PLAY]: ScreenCloudCoachElement,
  59. [ElementSubtypeTypes.ENJOY]: ScreenEnjoyElement,
  60. [ElementSubtypeTypes.LISTENING_PRACTICE]: ScreenListeningPracticeElement,
  61. [ElementSubtypeTypes.RHYTHM_PRACTICE]: ScreenRhythmPracticeElement
  62. }
  63. return elementTypeMap[props.elementInfo.type] || elementSubtypeMap[props.elementInfo.subtype] || null
  64. })
  65. const { formatedAnimations, theme } = storeToRefs(useSlidesStore())
  66. // 判断元素是否需要等待执行入场动画:等待执行入场的元素需要先隐藏
  67. const needWaitAnimation = computed(() => {
  68. // 该元素在本页动画序列中的位置
  69. const elementIndexInAnimation = formatedAnimations.value.findIndex(item => {
  70. const elIds = item.animations.map(item => item.elId)
  71. return elIds.includes(props.elementInfo.id)
  72. })
  73. // 该元素未设置过动画
  74. if (elementIndexInAnimation === -1) return false
  75. // 若该元素已执行过动画,都无须隐藏
  76. // 具体来说:若已执行的最后一个动画为入场,显然无须隐藏;若已执行的最后一个动画为退场,由于保留了退场动画结束状态,也无需额外隐藏
  77. if (elementIndexInAnimation < props.animationIndex) return false
  78. // 若该元素未执行过动画,获取其将要执行的第一个动画
  79. // 若将要执行的第一个动画为入场,则需要隐藏,否则无须隐藏
  80. const firstAnimation = formatedAnimations.value[elementIndexInAnimation].animations.find(item => item.elId === props.elementInfo.id)
  81. if (firstAnimation?.type === "in") return true
  82. return false
  83. })
  84. // 打开元素绑定的超链接
  85. const openLink = (e: MouseEvent) => {
  86. if ((e.target as HTMLElement).tagName === "A") {
  87. props.manualExitFullscreen()
  88. return
  89. }
  90. const link = props.elementInfo.link
  91. if (!link) return
  92. if (link.type === "web") {
  93. props.manualExitFullscreen()
  94. window.open(link.target)
  95. } else if (link.type === "slide") {
  96. props.turnSlideToId(link.target)
  97. }
  98. }
  99. </script>
  100. <style lang="scss" scoped>
  101. .link {
  102. cursor: pointer;
  103. }
  104. </style>