|
<template> |
|
<div class="base-view" :class="{ 'laser-pen': laserPen }"> |
|
<ScreenSlideList |
|
:slideWidth="slideWidth" |
|
:slideHeight="slideHeight" |
|
:animationIndex="animationIndex" |
|
:turnSlideToId="turnSlideToId" |
|
:manualExitFullscreen="manualExitFullscreen" |
|
@wheel="$event => mousewheelListener($event)" |
|
@touchstart="$event => touchStartListener($event)" |
|
@touchend="$event => touchEndListener($event)" |
|
v-contextmenu="contextmenus" |
|
/> |
|
|
|
<SlideThumbnails |
|
v-if="slideThumbnailModelVisible" |
|
:turnSlideToIndex="turnSlideToIndex" |
|
@close="slideThumbnailModelVisible = false" |
|
/> |
|
|
|
<WritingBoardTool |
|
:slideWidth="slideWidth" |
|
:slideHeight="slideHeight" |
|
v-if="writingBoardToolVisible" |
|
@close="writingBoardToolVisible = false" |
|
/> |
|
|
|
<CountdownTimer |
|
v-if="timerlVisible" |
|
@close="timerlVisible = false" |
|
/> |
|
|
|
<div class="tools-left"> |
|
<IconLeftTwo class="tool-btn" theme="two-tone" :fill="['#111', '#fff']" @click="execPrev()" /> |
|
<IconRightTwo class="tool-btn" theme="two-tone" :fill="['#111', '#fff']" @click="execNext()" /> |
|
</div> |
|
|
|
<div |
|
class="tools-right" :class="{ 'visible': rightToolsVisible }" |
|
@mouseleave="rightToolsVisible = false" |
|
@mouseenter="rightToolsVisible = true" |
|
> |
|
<div class="content"> |
|
<div class="tool-btn page-number" @click="slideThumbnailModelVisible = true">幻灯片 {{slideIndex + 1}} / {{slides.length}}</div> |
|
<IconWrite class="tool-btn" v-tooltip="'画笔工具'" @click="writingBoardToolVisible = true" /> |
|
<IconMagic class="tool-btn" v-tooltip="'激光笔'" :class="{ 'active': laserPen }" @click="laserPen = !laserPen" /> |
|
<IconStopwatchStart class="tool-btn" v-tooltip="'计时器'" :class="{ 'active': timerlVisible }" @click="timerlVisible = !timerlVisible" /> |
|
<IconListView class="tool-btn" v-tooltip="'演讲者视图'" @click="changeViewMode('presenter')" /> |
|
<IconOffScreenOne class="tool-btn" v-tooltip="'退出全屏'" v-if="fullscreenState" @click="manualExitFullscreen()" /> |
|
<IconFullScreenOne class="tool-btn" v-tooltip="'进入全屏'" v-else @click="enterFullscreen()" /> |
|
<IconPower class="tool-btn" v-tooltip="'结束放映'" @click="exitScreening()" /> |
|
</div> |
|
</div> |
|
|
|
<BottomThumbnails v-if="bottomThumbnailsVisible" /> |
|
</div> |
|
</template> |
|
|
|
<script lang="ts" setup> |
|
import { ref } from 'vue' |
|
import { storeToRefs } from 'pinia' |
|
import { useSlidesStore } from '@/store' |
|
import type { ContextmenuItem } from '@/components/Contextmenu/types' |
|
import { enterFullscreen } from '@/utils/fullscreen' |
|
import useScreening from '@/hooks/useScreening' |
|
import useExecPlay from './hooks/useExecPlay' |
|
import useSlideSize from './hooks/useSlideSize' |
|
import useFullscreen from './hooks/useFullscreen' |
|
|
|
import ScreenSlideList from './ScreenSlideList.vue' |
|
import SlideThumbnails from './SlideThumbnails.vue' |
|
import WritingBoardTool from './WritingBoardTool.vue' |
|
import CountdownTimer from './CountdownTimer.vue' |
|
import BottomThumbnails from './BottomThumbnails.vue' |
|
|
|
const props = defineProps<{ |
|
changeViewMode: (mode: 'base' | 'presenter') => void |
|
}>() |
|
|
|
const { slides, slideIndex } = storeToRefs(useSlidesStore()) |
|
|
|
const { |
|
autoPlayTimer, |
|
autoPlay, |
|
closeAutoPlay, |
|
autoPlayInterval, |
|
setAutoPlayInterval, |
|
loopPlay, |
|
setLoopPlay, |
|
mousewheelListener, |
|
touchStartListener, |
|
touchEndListener, |
|
turnPrevSlide, |
|
turnNextSlide, |
|
turnSlideToIndex, |
|
turnSlideToId, |
|
execPrev, |
|
execNext, |
|
animationIndex, |
|
} = useExecPlay() |
|
|
|
const { slideWidth, slideHeight } = useSlideSize() |
|
const { exitScreening } = useScreening() |
|
const { fullscreenState, manualExitFullscreen } = useFullscreen() |
|
|
|
const rightToolsVisible = ref(false) |
|
const writingBoardToolVisible = ref(false) |
|
const timerlVisible = ref(false) |
|
const slideThumbnailModelVisible = ref(false) |
|
const bottomThumbnailsVisible = ref(false) |
|
const laserPen = ref(false) |
|
|
|
const contextmenus = (): ContextmenuItem[] => { |
|
return [ |
|
{ |
|
text: '上一页', |
|
subText: '↑ ←', |
|
disable: slideIndex.value <= 0, |
|
handler: () => turnPrevSlide(), |
|
}, |
|
{ |
|
text: '下一页', |
|
subText: '↓ →', |
|
disable: slideIndex.value >= slides.value.length - 1, |
|
handler: () => turnNextSlide(), |
|
}, |
|
{ |
|
text: '第一页', |
|
disable: slideIndex.value === 0, |
|
handler: () => turnSlideToIndex(0), |
|
}, |
|
{ |
|
text: '最后一页', |
|
disable: slideIndex.value === slides.value.length - 1, |
|
handler: () => turnSlideToIndex(slides.value.length - 1), |
|
}, |
|
{ divider: true }, |
|
{ |
|
text: autoPlayTimer.value ? '取消自动放映' : '自动放映', |
|
handler: autoPlayTimer.value ? closeAutoPlay : autoPlay, |
|
children: [ |
|
{ |
|
text: '2.5秒', |
|
subText: autoPlayInterval.value === 2500 ? '√' : '', |
|
handler: () => setAutoPlayInterval(2500), |
|
}, |
|
{ |
|
text: '5秒', |
|
subText: autoPlayInterval.value === 5000 ? '√' : '', |
|
handler: () => setAutoPlayInterval(5000), |
|
}, |
|
{ |
|
text: '7.5秒', |
|
subText: autoPlayInterval.value === 7500 ? '√' : '', |
|
handler: () => setAutoPlayInterval(7500), |
|
}, |
|
{ |
|
text: '10秒', |
|
subText: autoPlayInterval.value === 10000 ? '√' : '', |
|
handler: () => setAutoPlayInterval(10000), |
|
}, |
|
], |
|
}, |
|
{ |
|
text: '循环放映', |
|
subText: loopPlay.value ? '√' : '', |
|
handler: () => setLoopPlay(!loopPlay.value), |
|
}, |
|
{ divider: true }, |
|
{ |
|
text: '显示工具栏', |
|
handler: () => rightToolsVisible.value = true, |
|
}, |
|
{ |
|
text: '查看所有幻灯片', |
|
handler: () => slideThumbnailModelVisible.value = true, |
|
}, |
|
{ |
|
text: '触底显示缩略图', |
|
subText: bottomThumbnailsVisible.value ? '√' : '', |
|
handler: () => bottomThumbnailsVisible.value = !bottomThumbnailsVisible.value, |
|
}, |
|
{ |
|
text: '画笔工具', |
|
handler: () => writingBoardToolVisible.value = true, |
|
}, |
|
{ |
|
text: '演讲者视图', |
|
handler: () => props.changeViewMode('presenter'), |
|
}, |
|
{ divider: true }, |
|
{ |
|
text: '结束放映', |
|
subText: 'ESC', |
|
handler: exitScreening, |
|
}, |
|
] |
|
} |
|
</script> |
|
|
|
<style lang="scss" scoped> |
|
.base-view { |
|
width: 100%; |
|
height: 100%; |
|
|
|
&.laser-pen { |
|
cursor: url() 20 20, default !important; |
|
} |
|
} |
|
.tools-left { |
|
position: fixed; |
|
bottom: 8px; |
|
left: 8px; |
|
font-size: 25px; |
|
color: |
|
z-index: 10; |
|
|
|
.tool-btn { |
|
opacity: .3; |
|
cursor: pointer; |
|
transition: opacity $transitionDelay; |
|
|
|
&:hover { |
|
opacity: .95; |
|
} |
|
& + .tool-btn { |
|
margin-left: 8px; |
|
} |
|
} |
|
} |
|
.tools-right { |
|
height: 66px; |
|
position: fixed; |
|
bottom: -66px; |
|
right: 0; |
|
z-index: 5; |
|
padding: 8px; |
|
transition: bottom $transitionDelay; |
|
|
|
&.visible { |
|
bottom: 0; |
|
} |
|
|
|
&::after { |
|
content: ''; |
|
width: 100%; |
|
height: 66px; |
|
position: absolute; |
|
left: 0; |
|
top: -66px; |
|
} |
|
|
|
.content { |
|
width: 100%; |
|
height: 100%; |
|
display: flex; |
|
justify-content: center; |
|
align-items: center; |
|
border-radius: $borderRadius; |
|
font-size: 25px; |
|
background-color: |
|
color: $textColor; |
|
padding: 8px 10px; |
|
box-shadow: 0 2px 12px 0 rgb(56, 56, 56, .2); |
|
border: 1px solid |
|
} |
|
|
|
.tool-btn { |
|
cursor: pointer; |
|
|
|
&:hover, &.active { |
|
color: $themeColor; |
|
} |
|
|
|
& + .tool-btn { |
|
margin-left: 15px; |
|
} |
|
} |
|
.page-number { |
|
font-size: 12px; |
|
padding: 0 12px; |
|
cursor: pointer; |
|
} |
|
} |
|
</style> |