App.vue 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249
  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. }
  104. else {
  105. url.searchParams.delete('mode')
  106. }
  107. // 使用 history.pushState 更新URL,不刷新页面
  108. window.history.pushState({}, '', url.toString())
  109. }
  110. // 使用provide提供切换函数,供子组件调用
  111. provide('switchViewMode', switchViewMode)
  112. if (import.meta.env.MODE !== 'development') {
  113. window.onbeforeunload = () => false
  114. }
  115. onMounted(async () => {
  116. const slides = await api.getFileData('slides')
  117. console.log(slides)
  118. slidesStore.setSlides(slides)
  119. // 初始化快照数据库
  120. // await deleteDiscardedDB()
  121. // snapshotStore.initSnapshotDatabase()
  122. // 监听视图模式切换事件
  123. window.addEventListener('viewModeChanged', (event: any) => {
  124. if (event.detail) {
  125. switchViewMode(event.detail)
  126. }
  127. })
  128. })
  129. // 应用注销时向 localStorage 中记录下本次 indexedDB 的数据库ID,用于之后清除数据库
  130. window.addEventListener('beforeunload', () => {
  131. const discardedDB = localStorage.getItem(LOCALSTORAGE_KEY_DISCARDED_DB)
  132. const discardedDBList: string[] = discardedDB ? JSON.parse(discardedDB) : []
  133. discardedDBList.push(databaseId.value)
  134. const newDiscardedDB = JSON.stringify(discardedDBList)
  135. localStorage.setItem(LOCALSTORAGE_KEY_DISCARDED_DB, newDiscardedDB)
  136. })
  137. </script>
  138. <style lang="scss">
  139. #app {
  140. height: 100%;
  141. }
  142. .image-preview {
  143. position: fixed;
  144. inset: 0;
  145. background: rgba(0, 0, 0, 0.85);
  146. display: flex;
  147. flex-direction: column;
  148. z-index: 6000;
  149. }
  150. .image-preview__toolbar {
  151. display: flex;
  152. gap: 8px;
  153. padding: 10px;
  154. justify-content: center;
  155. z-index: 9999;
  156. }
  157. .image-preview__toolbar button {
  158. padding: 8px 12px;
  159. border: none;
  160. background: linear-gradient(180deg, #3a8bff 0%, #2f80ed 100%);
  161. color: #fff;
  162. border-radius: 10px;
  163. cursor: pointer;
  164. transition: transform 0.15s ease, box-shadow 0.2s ease, background 0.2s ease;
  165. box-shadow: 0 2px 8px rgba(47, 128, 237, 0.3);
  166. }
  167. .image-preview__toolbar button:hover {
  168. transform: translateY(-1px);
  169. box-shadow: 0 6px 16px rgba(47, 128, 237, 0.35);
  170. }
  171. .image-preview__toolbar button:active {
  172. transform: translateY(0);
  173. box-shadow: 0 2px 8px rgba(47, 128, 237, 0.28);
  174. background: linear-gradient(180deg, #2f80ed 0%, #1b6dde 100%);
  175. }
  176. .image-preview__toolbar button:focus-visible {
  177. outline: 2px solid rgba(47, 128, 237, 0.6);
  178. outline-offset: 2px;
  179. }
  180. .image-preview__stage {
  181. flex: 1;
  182. display: flex;
  183. align-items: center;
  184. justify-content: center;
  185. cursor: grab;
  186. }
  187. .image-preview__stage:active {
  188. cursor: grabbing;
  189. }
  190. .image-preview__img {
  191. max-width: 92vw;
  192. max-height: 92vh;
  193. border-radius: 8px;
  194. box-shadow: 0 10px 30px rgba(0, 0, 0, 0.3);
  195. user-select: none;
  196. will-change: transform;
  197. }
  198. // 清空聊天记录确认弹窗样式
  199. .clear-confirm {
  200. padding-top: 4px;
  201. &__title {
  202. font-size: 16px;
  203. font-weight: 600;
  204. margin-bottom: 12px;
  205. color: #333;
  206. }
  207. &__content {
  208. font-size: 14px;
  209. color: #666;
  210. line-height: 1.6;
  211. margin-bottom: 20px;
  212. }
  213. &__footer {
  214. display: flex;
  215. justify-content: flex-end;
  216. gap: 8px;
  217. }
  218. }
  219. </style>