123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576 |
- <template>
- <div
- class="thumbnails"
- @mousedown="() => setThumbnailsFocus(true)"
- v-click-outside="() => setThumbnailsFocus(false)"
- v-contextmenu="contextmenusThumbnails"
- >
- <div class="add-slide">
- <div class="btn" @click="createSlide()"><IconPlus class="icon" />添加幻灯片</div>
- <Popover trigger="click" placement="bottom-start" v-model:value="presetLayoutPopoverVisible" center>
- <template #content>
- <LayoutPool
- @select="
- slide => {
- createSlideByTemplate(slide)
- presetLayoutPopoverVisible = false
- }
- "
- />
- </template>
- <div class="select-btn">
- <img src="./imgs/list.png" alt="" />
- </div>
- </Popover>
- </div>
- <Draggable
- class="thumbnail-list"
- ref="thumbnailsRef"
- :modelValue="slides"
- :animation="200"
- :scroll="true"
- :scrollSensitivity="50"
- :disabled="editingSectionId"
- @end="handleDragEnd"
- itemKey="id"
- >
- <template #item="{ element, index }">
- <div class="thumbnail-container">
- <div
- class="section-title"
- :data-section-id="element?.sectionTag?.id || ''"
- v-if="element.sectionTag || (hasSection && index === 0)"
- v-contextmenu="contextmenusSection"
- >
- <input
- :id="`section-title-input-${element?.sectionTag?.id || 'default'}`"
- type="text"
- :value="element?.sectionTag?.title || ''"
- placeholder="输入节名称"
- @blur="$event => saveSection($event)"
- @keydown.enter.stop="$event => saveSection($event)"
- v-if="editingSectionId === element?.sectionTag?.id || (index === 0 && editingSectionId === 'default')"
- />
- <span class="text" v-else>
- <div class="text-content">{{ element?.sectionTag ? element?.sectionTag?.title || "无标题节" : "默认节" }}</div>
- </span>
- </div>
- <div
- class="thumbnail-item"
- :class="{
- active: slideIndex === index,
- selected: selectedSlidesIndex.includes(index)
- }"
- @mousedown="$event => handleClickSlideThumbnail($event, index)"
- @dblclick="enterScreening()"
- v-contextmenu="contextmenusThumbnailItem"
- >
- <div class="label" :class="{ 'offset-left': index >= 99 }">{{ fillDigit(index + 1, 2) }}</div>
- <div class="thumbnail">
- <ThumbnailSlide :id="`thumbnailSlide_${index}`" :slide="element" :size="180" :visible="index < slidesLoadLimit" />
- <div class="tools" v-if="slideIndex === index">
- <img v-tooltip="'预览'" src="./imgs/play.png" @click="enterScreening" alt="" />
- <img v-tooltip="'添加幻灯片'" src="./imgs/add.png" @click="createSlide" alt="" />
- </div>
- </div>
- <div class="note-flag" v-if="element.notes && element.notes.length" @click="openNotesPanel()">{{ element.notes.length }}</div>
- </div>
- </div>
- </template>
- </Draggable>
- <div class="page-number">幻灯片 {{ slideIndex + 1 }} / {{ slides.length }}</div>
- </div>
- </template>
- <script lang="ts" setup>
- import { computed, nextTick, onMounted, ref, watch } from "vue"
- import { storeToRefs } from "pinia"
- import { useMainStore, useSlidesStore, useKeyboardStore } from "@/store"
- import { fillDigit } from "@/utils/common"
- import { isElementInViewport } from "@/utils/element"
- import type { ContextmenuItem } from "@/components/Contextmenu/types"
- import useSlideHandler from "@/hooks/useSlideHandler"
- import useSectionHandler from "@/hooks/useSectionHandler"
- import useScreening from "@/hooks/useScreening"
- import useLoadSlides from "@/hooks/useLoadSlides"
- import ThumbnailSlide from "@/views/components/ThumbnailSlide/index.vue"
- import LayoutPool from "./LayoutPool.vue"
- import Popover from "@/components/Popover.vue"
- import Draggable from "vuedraggable"
- const mainStore = useMainStore()
- const slidesStore = useSlidesStore()
- const keyboardStore = useKeyboardStore()
- const { selectedSlidesIndex: _selectedSlidesIndex, thumbnailsFocus } = storeToRefs(mainStore)
- const { slides, slideIndex, currentSlide } = storeToRefs(slidesStore)
- const { ctrlKeyState, shiftKeyState } = storeToRefs(keyboardStore)
- const { slidesLoadLimit } = useLoadSlides()
- const selectedSlidesIndex = computed(() => [..._selectedSlidesIndex.value, slideIndex.value])
- const presetLayoutPopoverVisible = ref(false)
- const hasSection = computed(() => {
- return slides.value.some(item => item.sectionTag)
- })
- const { copySlide, pasteSlide, createSlide, createSlideByTemplate, copyAndPasteSlide, deleteSlide, cutSlide, selectAllSlide, sortSlides } =
- useSlideHandler()
- const { createSection, removeSection, removeAllSection, removeSectionSlides, updateSectionTitle } = useSectionHandler()
- // 页面被切换时
- const thumbnailsRef = ref<InstanceType<typeof Draggable>>()
- watch(
- () => slideIndex.value,
- () => {
- // 清除多选状态的幻灯片
- if (selectedSlidesIndex.value.length) {
- mainStore.updateSelectedSlidesIndex([])
- }
- // 检查当前页缩略图是否在可视范围,不在的话需要滚动到对应的位置
- nextTick(() => {
- const activeThumbnailRef: HTMLElement = thumbnailsRef.value?.$el?.querySelector(".thumbnail-item.active")
- if (thumbnailsRef.value && activeThumbnailRef && !isElementInViewport(activeThumbnailRef, thumbnailsRef.value.$el)) {
- setTimeout(() => {
- activeThumbnailRef.scrollIntoView({ behavior: "smooth" })
- }, 100)
- }
- })
- }
- )
- // 从预览切换回来的时候 滚动到对应的位置
- onMounted(() => {
- const activeThumbnailRef: HTMLElement = thumbnailsRef.value?.$el?.querySelector(".thumbnail-item.active")
- if (thumbnailsRef.value && activeThumbnailRef && !isElementInViewport(activeThumbnailRef, thumbnailsRef.value.$el)) {
- setTimeout(() => {
- activeThumbnailRef.scrollIntoView()
- }, 100)
- }
- })
- // 切换页面
- const changeSlideIndex = (index: number) => {
- mainStore.setActiveElementIdList([])
- if (slideIndex.value === index) return
- slidesStore.updateSlideIndex(index)
- }
- // 点击缩略图
- const handleClickSlideThumbnail = (e: MouseEvent, index: number) => {
- if (editingSectionId.value) return
- const isMultiSelected = selectedSlidesIndex.value.length > 1
- if (isMultiSelected && selectedSlidesIndex.value.includes(index) && e.button !== 0) return
- // 按住Ctrl键,点选幻灯片,再次点击已选中的页面则取消选中
- // 如果被取消选中的页面刚好是当前激活页面,则需要从其他被选中的页面中选择第一个作为当前激活页面
- if (ctrlKeyState.value) {
- if (slideIndex.value === index) {
- if (!isMultiSelected) return
- const newSelectedSlidesIndex = selectedSlidesIndex.value.filter(item => item !== index)
- mainStore.updateSelectedSlidesIndex(newSelectedSlidesIndex)
- changeSlideIndex(selectedSlidesIndex.value[0])
- } else {
- if (selectedSlidesIndex.value.includes(index)) {
- const newSelectedSlidesIndex = selectedSlidesIndex.value.filter(item => item !== index)
- mainStore.updateSelectedSlidesIndex(newSelectedSlidesIndex)
- } else {
- const newSelectedSlidesIndex = [...selectedSlidesIndex.value, index]
- mainStore.updateSelectedSlidesIndex(newSelectedSlidesIndex)
- }
- }
- }
- // 按住Shift键,选择范围内的全部幻灯片
- else if (shiftKeyState.value) {
- if (slideIndex.value === index && !isMultiSelected) return
- let minIndex = Math.min(...selectedSlidesIndex.value)
- let maxIndex = index
- if (index < minIndex) {
- maxIndex = Math.max(...selectedSlidesIndex.value)
- minIndex = index
- }
- const newSelectedSlidesIndex = []
- for (let i = minIndex; i <= maxIndex; i++) newSelectedSlidesIndex.push(i)
- mainStore.updateSelectedSlidesIndex(newSelectedSlidesIndex)
- }
- // 正常切换页面
- else {
- mainStore.updateSelectedSlidesIndex([])
- changeSlideIndex(index)
- }
- }
- // 设置缩略图工具栏聚焦状态(只有聚焦状态下,该部分的快捷键才能生效)
- const setThumbnailsFocus = (focus: boolean) => {
- if (thumbnailsFocus.value === focus) return
- mainStore.setThumbnailsFocus(focus)
- if (!focus) mainStore.updateSelectedSlidesIndex([])
- }
- // 拖拽调整顺序后进行数据的同步
- const handleDragEnd = (eventData: { newIndex: number; oldIndex: number }) => {
- const { newIndex, oldIndex } = eventData
- if (newIndex === undefined || oldIndex === undefined || newIndex === oldIndex) return
- sortSlides(newIndex, oldIndex)
- }
- // 打开批注面板
- const openNotesPanel = () => {
- mainStore.setNotesPanelState(true)
- }
- const editingSectionId = ref("")
- const editSection = (id: string) => {
- mainStore.setDisableHotkeysState(true)
- editingSectionId.value = id || "default"
- nextTick(() => {
- const inputRef = document.querySelector(`#section-title-input-${id || "default"}`) as HTMLInputElement
- inputRef.focus()
- })
- }
- const saveSection = (e: FocusEvent | KeyboardEvent) => {
- const title = (e.target as HTMLInputElement).value
- updateSectionTitle(editingSectionId.value, title)
- editingSectionId.value = ""
- mainStore.setDisableHotkeysState(false)
- }
- const contextmenusSection = (el: HTMLElement): ContextmenuItem[] => {
- const sectionId = el.dataset.sectionId!
- return [
- {
- text: "删除节",
- handler: () => removeSection(sectionId)
- },
- {
- text: "删除节和幻灯片",
- handler: () => {
- mainStore.setActiveElementIdList([])
- removeSectionSlides(sectionId)
- }
- },
- {
- text: "删除所有节",
- handler: removeAllSection
- },
- {
- text: "重命名节",
- handler: () => editSection(sectionId)
- }
- ]
- }
- const { enterScreening, enterScreeningFromStart } = useScreening()
- const contextmenusThumbnails = (): ContextmenuItem[] => {
- return [
- {
- text: "粘贴",
- subText: "Ctrl + V",
- handler: pasteSlide
- },
- {
- text: "全选",
- subText: "Ctrl + A",
- handler: selectAllSlide
- },
- {
- text: "新建页面",
- subText: "Enter",
- handler: createSlide
- },
- {
- text: "幻灯片放映",
- subText: "F5",
- handler: enterScreeningFromStart
- }
- ]
- }
- const contextmenusThumbnailItem = (): ContextmenuItem[] => {
- return [
- {
- text: "剪切",
- subText: "Ctrl + X",
- handler: cutSlide
- },
- {
- text: "复制",
- subText: "Ctrl + C",
- handler: copySlide
- },
- {
- text: "粘贴",
- subText: "Ctrl + V",
- handler: pasteSlide
- },
- {
- text: "全选",
- subText: "Ctrl + A",
- handler: selectAllSlide
- },
- { divider: true },
- {
- text: "新建页面",
- subText: "Enter",
- handler: createSlide
- },
- {
- text: "复制页面",
- subText: "Ctrl + D",
- handler: copyAndPasteSlide
- },
- {
- text: "删除页面",
- subText: "Delete",
- handler: () => deleteSlide()
- },
- {
- text: "增加节",
- handler: createSection,
- disable: !!currentSlide.value.sectionTag
- },
- { divider: true },
- {
- text: "从当前放映",
- subText: "Shift + F5",
- handler: enterScreening
- }
- ]
- }
- </script>
- <style lang="scss" scoped>
- .thumbnails {
- border-right: solid 1px $borderColor;
- background-color: #fff;
- display: flex;
- flex-direction: column;
- user-select: none;
- }
- .add-slide {
- margin-top: 16px;
- margin-bottom: 8px;
- font-size: 12px;
- display: flex;
- flex-shrink: 0;
- justify-content: center;
- align-items: center;
- .btn {
- cursor: pointer;
- flex-shrink: 1;
- display: flex;
- justify-content: center;
- align-items: center;
- width: 180px;
- height: 30px;
- background: #ffffff;
- border-radius: 4px;
- border: 1px solid #c4c4c4;
- &:hover {
- background-color: $lightGray;
- }
- }
- .select-btn {
- cursor: pointer;
- margin-left: 12px;
- width: 30px;
- height: 100%;
- display: flex;
- justify-content: center;
- align-items: center;
- width: 30px;
- height: 30px;
- background: #ffffff;
- border-radius: 4px;
- border: 1px solid #c4c4c4;
- &:hover {
- background-color: $lightGray;
- }
- & > img {
- width: 20px;
- height: 20px;
- }
- }
- .icon {
- margin-right: 3px;
- font-size: 14px;
- }
- }
- .thumbnail-list {
- flex: 1;
- overflow: auto;
- }
- .thumbnail-item {
- display: flex;
- justify-content: center;
- align-items: center;
- padding: 8px 0;
- position: relative;
- .thumbnail {
- overflow: hidden;
- border-radius: 4px;
- outline: 1px solid #dedede;
- position: relative;
- .tools {
- position: absolute;
- bottom: 8px;
- left: 50%;
- transform: translateX(-50%);
- display: flex;
- & > img {
- cursor: pointer;
- width: 28px;
- height: 28px;
- &:hover {
- opacity: 0.8;
- }
- &:last-child {
- margin-left: 40px;
- }
- }
- }
- }
- &.active {
- .label {
- color: $themeColor;
- }
- .thumbnail {
- outline-color: $themeColor;
- }
- }
- &.selected {
- .thumbnail {
- outline: 2px solid $themeColor;
- }
- .note-flag {
- background-color: $themeColor;
- &::after {
- border-top-color: $themeColor;
- }
- }
- }
- .note-flag {
- width: 16px;
- height: 12px;
- border-radius: 1px;
- position: absolute;
- left: 22px;
- top: 26px;
- font-size: 8px;
- background-color: rgba($color: $themeColor, $alpha: 0.75);
- color: #fff;
- text-align: center;
- line-height: 12px;
- cursor: pointer;
- &::after {
- content: "";
- width: 0;
- height: 0;
- position: absolute;
- top: 10px;
- left: 4px;
- border: 4px solid transparent;
- border-top-color: rgba($color: $themeColor, $alpha: 0.75);
- }
- }
- }
- .label {
- font-size: 12px;
- color: #999;
- width: 20px;
- font-weight: 600;
- font-size: 14px;
- cursor: grab;
- margin-right: 18px;
- &.offset-left {
- position: relative;
- left: -4px;
- }
- &:active {
- cursor: grabbing;
- }
- }
- .page-number {
- height: 50px;
- border-top: 1px solid $borderColor;
- line-height: 50px;
- padding-left: 24px;
- font-weight: 400;
- font-size: 12px;
- color: #131415;
- }
- .section-title {
- height: 26px;
- font-size: 12px;
- padding: 6px 8px 2px 18px;
- color: #555;
- &.contextmenu-active {
- color: $themeColor;
- .text::before {
- border-bottom-color: $themeColor;
- border-right-color: $themeColor;
- }
- }
- .text {
- display: flex;
- align-items: center;
- position: relative;
- &::before {
- content: "";
- width: 0;
- height: 0;
- border-top: 3px solid transparent;
- border-left: 3px solid transparent;
- border-bottom: 3px solid #555;
- border-right: 3px solid #555;
- margin-right: 5px;
- }
- .text-content {
- display: inline-block;
- @include ellipsis-oneline();
- }
- }
- input {
- width: 100%;
- border: 0;
- outline: 0;
- padding: 0;
- font-size: 12px;
- }
- }
- </style>
|