App.vue 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. <template>
  2. <template v-if="slides.length">
  3. <Screen v-if="viewMode !== 'student' && screening" />
  4. <Editor
  5. v-if="viewMode === 'editor' && _isPC && !screening"
  6. :courseid="urlParams.courseid"
  7. />
  8. <Editor2
  9. v-else-if="viewMode === 'editor2' && _isPC && !screening"
  10. :courseid="urlParams.courseid"
  11. />
  12. <Editor3
  13. v-else-if="viewMode === 'editor3' && _isPC && !screening"
  14. :courseid="urlParams.courseid"
  15. />
  16. <Student
  17. v-else-if="viewMode === 'student'"
  18. :courseid="urlParams.courseid"
  19. :type="urlParams.type"
  20. :userid="urlParams.userid"
  21. :oid="urlParams.oid"
  22. :org="urlParams.org"
  23. :cid="urlParams.cid"
  24. />
  25. <Mobile v-else />
  26. </template>
  27. <FullscreenSpin :tip="lang.ssInitDataWait" v-else loading :mask="false" />
  28. </template>
  29. <script lang="ts" setup>
  30. import { onMounted, ref, provide } from "vue";
  31. import { storeToRefs } from "pinia";
  32. import {
  33. useScreenStore,
  34. useMainStore,
  35. useSnapshotStore,
  36. useSlidesStore,
  37. } from "@/store";
  38. import { lang } from "@/main";
  39. import { LOCALSTORAGE_KEY_DISCARDED_DB } from "@/configs/storage";
  40. import { deleteDiscardedDB } from "@/utils/database";
  41. import { isPC } from "@/utils/common";
  42. import api from "@/services";
  43. import Editor from "./views/Editor/index.vue";
  44. import Editor2 from "./views/Editor/index2.vue";
  45. import Editor3 from "./views/Editor/index3.vue";
  46. import Screen from "./views/Screen/index.vue";
  47. import Mobile from "./views/Mobile/index.vue";
  48. import Student from "./views/Student/index.vue";
  49. import FullscreenSpin from "@/components/FullscreenSpin.vue";
  50. const _isPC = isPC();
  51. const mainStore = useMainStore();
  52. const slidesStore = useSlidesStore();
  53. const snapshotStore = useSnapshotStore();
  54. const { databaseId } = storeToRefs(mainStore);
  55. const { slides } = storeToRefs(slidesStore);
  56. const { screening } = storeToRefs(useScreenStore());
  57. // 视图模式:'editor', 'student', 'screen'
  58. // 支持通过URL参数直接访问学生模式
  59. const getInitialViewMode = () => {
  60. // 检查URL参数
  61. const urlParams = new URLSearchParams(window.location.search);
  62. const modeFromUrl = urlParams.get("mode");
  63. console.log(modeFromUrl);
  64. if (modeFromUrl === "student") {
  65. return "student";
  66. }
  67. if (modeFromUrl === "editor2") {
  68. return "editor2";
  69. }
  70. if (modeFromUrl === "editor3") {
  71. return "editor3";
  72. }
  73. // 检查localStorage
  74. const modeFromStorage = localStorage.getItem("viewMode");
  75. if (modeFromStorage) {
  76. return modeFromStorage;
  77. }
  78. // 默认返回编辑模式
  79. return "editor";
  80. };
  81. // 获取URL参数中的courseid和type
  82. const getUrlParams = () => {
  83. const urlParams = new URLSearchParams(window.location.search);
  84. return {
  85. courseid: urlParams.get("courseid"),
  86. userid: urlParams.get("userid"),
  87. oid: urlParams.get("oid"),
  88. org: urlParams.get("org"),
  89. cid: urlParams.get("cid"),
  90. type: urlParams.get("type"),
  91. };
  92. };
  93. const urlParams = getUrlParams();
  94. const viewMode = ref(getInitialViewMode());
  95. // 全局切换视图模式的函数
  96. const switchViewMode = (mode: string) => {
  97. viewMode.value = mode;
  98. localStorage.setItem("viewMode", mode);
  99. // 更新URL参数
  100. const url = new URL(window.location.href);
  101. if (mode === "student") {
  102. url.searchParams.set("mode", "student");
  103. } else {
  104. url.searchParams.delete("mode");
  105. }
  106. // 使用 history.pushState 更新URL,不刷新页面
  107. window.history.pushState({}, "", url.toString());
  108. };
  109. // 使用provide提供切换函数,供子组件调用
  110. provide("switchViewMode", switchViewMode);
  111. if (import.meta.env.MODE !== "development") {
  112. window.onbeforeunload = () => false;
  113. }
  114. onMounted(async () => {
  115. const slides = await api.getFileData("slides");
  116. console.log(slides);
  117. slidesStore.setSlides(slides);
  118. // 初始化快照数据库
  119. // await deleteDiscardedDB()
  120. // snapshotStore.initSnapshotDatabase()
  121. // 监听视图模式切换事件
  122. window.addEventListener("viewModeChanged", (event: any) => {
  123. if (event.detail) {
  124. switchViewMode(event.detail);
  125. }
  126. });
  127. });
  128. // 应用注销时向 localStorage 中记录下本次 indexedDB 的数据库ID,用于之后清除数据库
  129. window.addEventListener("beforeunload", () => {
  130. const discardedDB = localStorage.getItem(LOCALSTORAGE_KEY_DISCARDED_DB);
  131. const discardedDBList: string[] = discardedDB ? JSON.parse(discardedDB) : [];
  132. discardedDBList.push(databaseId.value);
  133. const newDiscardedDB = JSON.stringify(discardedDBList);
  134. localStorage.setItem(LOCALSTORAGE_KEY_DISCARDED_DB, newDiscardedDB);
  135. });
  136. </script>
  137. <style lang="scss">
  138. #app {
  139. height: 100%;
  140. }
  141. .image-preview {
  142. position: fixed;
  143. inset: 0;
  144. background: rgba(0, 0, 0, 0.85);
  145. display: flex;
  146. flex-direction: column;
  147. z-index: 6000;
  148. }
  149. .image-preview__toolbar {
  150. display: flex;
  151. gap: 8px;
  152. padding: 10px;
  153. justify-content: center;
  154. z-index: 9999;
  155. }
  156. .image-preview__toolbar button {
  157. padding: 8px 12px;
  158. border: none;
  159. background: linear-gradient(180deg, #3a8bff 0%, #2f80ed 100%);
  160. color: #fff;
  161. border-radius: 10px;
  162. cursor: pointer;
  163. transition: transform 0.15s ease, box-shadow 0.2s ease, background 0.2s ease;
  164. box-shadow: 0 2px 8px rgba(47, 128, 237, 0.3);
  165. }
  166. .image-preview__toolbar button:hover {
  167. transform: translateY(-1px);
  168. box-shadow: 0 6px 16px rgba(47, 128, 237, 0.35);
  169. }
  170. .image-preview__toolbar button:active {
  171. transform: translateY(0);
  172. box-shadow: 0 2px 8px rgba(47, 128, 237, 0.28);
  173. background: linear-gradient(180deg, #2f80ed 0%, #1b6dde 100%);
  174. }
  175. .image-preview__toolbar button:focus-visible {
  176. outline: 2px solid rgba(47, 128, 237, 0.6);
  177. outline-offset: 2px;
  178. }
  179. .image-preview__stage {
  180. flex: 1;
  181. display: flex;
  182. align-items: center;
  183. justify-content: center;
  184. cursor: grab;
  185. }
  186. .image-preview__stage:active {
  187. cursor: grabbing;
  188. }
  189. .image-preview__img {
  190. max-width: 92vw;
  191. max-height: 92vh;
  192. border-radius: 8px;
  193. box-shadow: 0 10px 30px rgba(0, 0, 0, 0.3);
  194. user-select: none;
  195. will-change: transform;
  196. }
  197. </style>