index.vue 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  1. <template>
  2. <div class="editor-header">
  3. <div class="left">
  4. <Dropdown :trigger="['click']">
  5. <div class="menu-item"><IconEdit /> 编辑</div>
  6. <template #overlay>
  7. <Menu>
  8. <MenuItem @click="undo()">撤销</MenuItem>
  9. <MenuItem @click="redo()">重做</MenuItem>
  10. <MenuItem @click="createSlide()">添加页面</MenuItem>
  11. <MenuItem @click="deleteSlide()">删除页面</MenuItem>
  12. <MenuItem @click="toggleGridLines()">{{ showGridLines ? '关闭网格线' : '打开网格线' }}</MenuItem>
  13. <MenuItem @click="resetSlides()">重置幻灯片</MenuItem>
  14. </Menu>
  15. </template>
  16. </Dropdown>
  17. <Dropdown :trigger="['click']">
  18. <div class="menu-item"><IconPpt /> 演示</div>
  19. <template #overlay>
  20. <Menu>
  21. <MenuItem @click="enterScreeningFromStart()">从头开始</MenuItem>
  22. <MenuItem @click="enterScreening()">从当前页开始</MenuItem>
  23. </Menu>
  24. </template>
  25. </Dropdown>
  26. <Dropdown :trigger="['click']">
  27. <div class="menu-item"><IconHelpcenter /> 帮助</div>
  28. <template #overlay>
  29. <Menu>
  30. <MenuItem @click="openDoc()">开发文档</MenuItem>
  31. <MenuItem @click="hotkeyDrawerVisible = true">快捷键</MenuItem>
  32. </Menu>
  33. </template>
  34. </Dropdown>
  35. </div>
  36. <div class="right">
  37. <Tooltip :mouseLeaveDelay="0" title="幻灯片放映">
  38. <div class="menu-item" @click="enterScreening()">
  39. <IconPpt size="18" fill="#666" style="margin-top: 2px;" />
  40. </div>
  41. </Tooltip>
  42. <Tooltip :mouseLeaveDelay="0" title="Github 仓库">
  43. <div class="menu-item"><IconGithub size="18" fill="#666" /></div>
  44. </Tooltip>
  45. </div>
  46. <Drawer
  47. width="320"
  48. placement="right"
  49. :visible="hotkeyDrawerVisible"
  50. @close="hotkeyDrawerVisible = false"
  51. >
  52. <div class="hotkeys">
  53. <template v-for="item in hotkeys" :key="item.type">
  54. <div class="title">{{item.type}}</div>
  55. <div class="hotkey-item" v-for="hotkey in item.children" :key="hotkey.label">
  56. <div class="label">{{hotkey.label}}</div>
  57. <div class="value">{{hotkey.value}}</div>
  58. </div>
  59. </template>
  60. </div>
  61. </Drawer>
  62. </div>
  63. </template>
  64. <script lang="ts">
  65. import { computed, defineComponent, ref } from 'vue'
  66. import { MutationTypes, useStore } from '@/store'
  67. import { createRandomCode } from '@/utils/common'
  68. import useScreening from '@/hooks/useScreening'
  69. import useSlideHandler from '@/hooks/useSlideHandler'
  70. import useHistorySnapshot from '@/hooks/useHistorySnapshot'
  71. import { message } from 'ant-design-vue'
  72. const hotkeys = [
  73. {
  74. type: '通用',
  75. children: [
  76. { label: '剪切', value: 'Ctrl + X' },
  77. { label: '复制', value: 'Ctrl + C' },
  78. { label: '粘贴', value: 'Ctrl + V' },
  79. { label: '快速复制粘贴', value: 'Ctrl + D' },
  80. { label: '全选', value: 'Ctrl + A' },
  81. { label: '撤销', value: 'Ctrl + Z' },
  82. { label: '恢复', value: 'Ctrl + Y' },
  83. { label: '删除', value: 'Delete' },
  84. ],
  85. },
  86. {
  87. type: '幻灯片放映',
  88. children: [
  89. { label: '开始放映幻灯片', value: 'Ctrl + F' },
  90. { label: '切换上一页', value: '↑ / ←' },
  91. { label: '切换下一页', value: '↓ / → / Enter / Space' },
  92. { label: '退出放映', value: 'ESC' },
  93. ],
  94. },
  95. {
  96. type: '幻灯片编辑',
  97. children: [
  98. { label: '新建幻灯片', value: 'Enter' },
  99. { label: '缩放画布', value: 'Ctrl + 鼠标滚动' },
  100. { label: '放大画布', value: 'Ctrl + =' },
  101. { label: '缩小画布', value: 'Ctrl + -' },
  102. { label: '缩放画布到合适大小', value: 'Ctrl + 0' },
  103. { label: '编辑上一页', value: '↑ / ←' },
  104. { label: '编辑下一页', value: '↓ / →' },
  105. ],
  106. },
  107. {
  108. type: '元素操作',
  109. children: [
  110. { label: '移动', value: '↑ / ← / ↓ / →' },
  111. { label: '锁定', value: 'Ctrl + L' },
  112. { label: '组合', value: 'Ctrl + G' },
  113. { label: '取消组合', value: 'Ctrl + Shift + G' },
  114. { label: '多选', value: '按住 Ctrl 或 Shift' },
  115. { label: '锁定宽高比例', value: '按住 Ctrl 或 Shift' },
  116. { label: '创建水平 / 垂直线条', value: '按住 Ctrl 或 Shift' },
  117. { label: '确认图片裁剪', value: 'Enter' },
  118. ],
  119. },
  120. {
  121. type: '表格编辑',
  122. children: [
  123. { label: '聚焦到下一个单元格', value: 'Tab' },
  124. ],
  125. },
  126. {
  127. type: '文本编辑',
  128. children: [
  129. { label: '加粗', value: 'Ctrl + B' },
  130. { label: '斜体', value: 'Ctrl + I' },
  131. { label: '下划线', value: 'Ctrl + U' },
  132. { label: '删除线', value: 'Ctrl + D' },
  133. ],
  134. },
  135. ]
  136. export default defineComponent({
  137. name: 'editor-header',
  138. setup() {
  139. const store = useStore()
  140. const { enterScreening, enterScreeningFromStart } = useScreening()
  141. const { createSlide, deleteSlide } = useSlideHandler()
  142. const { redo, undo } = useHistorySnapshot()
  143. const showGridLines = computed(() => store.state.showGridLines)
  144. const toggleGridLines = () => {
  145. store.commit(MutationTypes.SET_GRID_LINES_STATE, !showGridLines.value)
  146. }
  147. const resetSlides = () => {
  148. store.commit(MutationTypes.SET_ACTIVE_ELEMENT_ID_LIST, [])
  149. store.commit(MutationTypes.SET_SLIDES, [{
  150. id: createRandomCode(),
  151. elements: [],
  152. }])
  153. }
  154. const openDoc = () => {
  155. message.warning('作者努力编写中...')
  156. }
  157. const hotkeyDrawerVisible = ref(false)
  158. return {
  159. enterScreening,
  160. enterScreeningFromStart,
  161. createSlide,
  162. deleteSlide,
  163. redo,
  164. undo,
  165. toggleGridLines,
  166. showGridLines,
  167. resetSlides,
  168. openDoc,
  169. hotkeyDrawerVisible,
  170. hotkeys,
  171. }
  172. },
  173. })
  174. </script>
  175. <style lang="scss" scoped>
  176. .editor-header {
  177. background-color: #fff;
  178. user-select: none;
  179. border-bottom: 1px solid $borderColor;
  180. display: flex;
  181. justify-content: space-between;
  182. padding: 0 10px;
  183. }
  184. .left, .right {
  185. display: flex;
  186. justify-content: center;
  187. align-items: center;
  188. }
  189. .menu-item {
  190. font-size: 13px;
  191. margin: 0 10px;
  192. cursor: pointer;
  193. }
  194. .hotkeys {
  195. height: 100%;
  196. overflow: auto;
  197. font-size: 12px;
  198. }
  199. .title {
  200. font-size: 14px;
  201. font-weight: 700;
  202. border-bottom: 1px solid #e5e5e5;
  203. padding: 15px 0 5px 0;
  204. }
  205. .hotkey-item {
  206. border-bottom: 1px solid #e5e5e5;
  207. padding: 15px 0 5px 0;
  208. display: flex;
  209. align-items: center;
  210. }
  211. .label {
  212. width: 140px;
  213. @include ellipsis();
  214. }
  215. </style>