index.vue 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  1. <template>
  2. <div class="editor-header">
  3. <div class="left">
  4. <Popover trigger="click" placement="bottom-start" v-model:value="mainMenuVisible">
  5. <template #content>
  6. <FileInput
  7. accept="application/vnd.openxmlformats-officedocument.presentationml.presentation"
  8. @change="
  9. files => {
  10. spinType = 'import'
  11. importPPTXFile(files)
  12. mainMenuVisible = false
  13. }
  14. "
  15. >
  16. <PopoverMenuItem>导入pptx</PopoverMenuItem>
  17. </FileInput>
  18. <PopoverMenuItem @click="handleExport">导出pptx</PopoverMenuItem>
  19. <PopoverMenuItem
  20. @click="
  21. () => {
  22. resetSlides()
  23. mainMenuVisible = false
  24. }
  25. "
  26. >重置幻灯片</PopoverMenuItem
  27. >
  28. <PopoverMenuItem
  29. @click="
  30. () => {
  31. mainMenuVisible = false
  32. hotkeyDrawerVisible = true
  33. }
  34. "
  35. >快捷操作</PopoverMenuItem
  36. >
  37. </template>
  38. <div class="menuCon" :class="{ menuVisible: mainMenuVisible }">
  39. <img class="icon" src="./imgs/file.png" />
  40. </div>
  41. </Popover>
  42. <div class="line"></div>
  43. <div class="title">
  44. <div class="title-text">{{ title }}</div>
  45. </div>
  46. </div>
  47. <div class="right">
  48. <div class="cancelBtn" @click="handleClose">退出</div>
  49. <div class="saveBtn" @click="handleSave">保存课件</div>
  50. </div>
  51. <Drawer :width="320" v-model:visible="hotkeyDrawerVisible" placement="right">
  52. <HotkeyDoc />
  53. <template v-slot:title>快捷操作</template>
  54. </Drawer>
  55. <FullscreenSpin :loading="fullscreenSpinData.loading" :progress="fullscreenSpinData.progress" :tip="fullscreenSpinData.tip" />
  56. </div>
  57. </template>
  58. <script lang="ts" setup>
  59. import { storeToRefs } from "pinia"
  60. import { useSlidesStore } from "@/store"
  61. import useImport from "@/hooks/useImport"
  62. import useExport from "@/hooks/useExport"
  63. import useSlideHandler from "@/hooks/useSlideHandler"
  64. import HotkeyDoc from "./HotkeyDoc.vue"
  65. import FileInput from "@/components/FileInput.vue"
  66. import FullscreenSpin from "@/components/FullscreenSpin.vue"
  67. import Drawer from "@/components/Drawer.vue"
  68. import Popover from "@/components/Popover.vue"
  69. import PopoverMenuItem from "@/components/PopoverMenuItem.vue"
  70. import { ref, computed } from "vue"
  71. import { ElMessageBox } from "element-plus"
  72. import usePptWork from "@/store/pptWork"
  73. const slidesStore = useSlidesStore()
  74. const { title } = storeToRefs(slidesStore)
  75. const { resetSlides } = useSlideHandler()
  76. const mainMenuVisible = ref(false)
  77. const hotkeyDrawerVisible = ref(false)
  78. const spinType = ref<"import" | "export">("import")
  79. /* 导入 */
  80. const { importPPTXFile, importing, importProgress } = useImport()
  81. /* 导出 */
  82. const { exportPPTX, exporting, exportProgress } = useExport()
  83. function handleExport() {
  84. mainMenuVisible.value = false
  85. ElMessageBox.confirm("导出pptx可能会丢失一些内容,确认导出?", "提示", {
  86. confirmButtonText: "确认",
  87. cancelButtonText: "取消",
  88. type: "warning"
  89. })
  90. .then(() => {
  91. spinType.value = "export"
  92. exportPPTX()
  93. })
  94. .catch(() => {})
  95. }
  96. const fullscreenSpinData = computed(() => {
  97. if (spinType.value === "export") {
  98. return {
  99. loading: exporting.value,
  100. progress: exportProgress.value,
  101. tip: "正在导出中,请稍等…"
  102. }
  103. } else {
  104. return {
  105. loading: importing.value,
  106. progress: importProgress.value,
  107. tip: "正在导入中,请稍等…"
  108. }
  109. }
  110. })
  111. const usePptWorkHook = usePptWork()
  112. /* 保存课件 */
  113. function handleSave() {
  114. usePptWorkHook.updatePPT()
  115. }
  116. /* 关闭页面 */
  117. function handleClose() {
  118. window.close()
  119. window.parent.postMessage(
  120. {
  121. api: "iframe_exit",
  122. },
  123. "*"
  124. );
  125. }
  126. </script>
  127. <style lang="scss" scoped>
  128. .editor-header {
  129. background-color: #fff;
  130. user-select: none;
  131. border-bottom: 1px solid $borderColor;
  132. display: flex;
  133. justify-content: space-between;
  134. padding: 0 24px;
  135. }
  136. .left,
  137. .right {
  138. display: flex;
  139. justify-content: center;
  140. align-items: center;
  141. }
  142. .menuCon {
  143. display: flex;
  144. justify-content: center;
  145. align-items: center;
  146. cursor: pointer;
  147. margin-right: 8px;
  148. width: 34px;
  149. height: 34px;
  150. border-radius: 6px;
  151. margin-left: -8px;
  152. &:hover,
  153. &.menuVisible {
  154. background: rgba(34, 71, 133, 0.08);
  155. }
  156. & > img {
  157. width: 20px;
  158. height: 20px;
  159. }
  160. }
  161. .line {
  162. width: 1px;
  163. height: calc(100% - 26px);
  164. background-color: $borderColor;
  165. }
  166. .title {
  167. margin-left: 16px;
  168. .title-text {
  169. font-weight: 600;
  170. font-size: 15px;
  171. color: #131415;
  172. }
  173. }
  174. .cancelBtn {
  175. width: 88px;
  176. height: 32px;
  177. background: #f1f2f6;
  178. border-radius: 6px;
  179. line-height: 32px;
  180. font-weight: 600;
  181. font-size: 14px;
  182. color: #1e2022;
  183. text-align: center;
  184. cursor: pointer;
  185. &:hover {
  186. opacity: 0.8;
  187. }
  188. }
  189. .saveBtn {
  190. margin-left: 16px;
  191. width: 88px;
  192. height: 32px;
  193. background: linear-gradient(312deg, #1b7af8 0%, #3cbbff 100%);
  194. border-radius: 6px;
  195. line-height: 32px;
  196. font-weight: 600;
  197. font-size: 14px;
  198. color: #ffffff;
  199. text-align: center;
  200. cursor: pointer;
  201. &:hover {
  202. opacity: 0.8;
  203. }
  204. }
  205. </style>