CatPtain's picture
Upload 339 files
89ce340 verified
<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: #666;
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: #fff;
color: $textColor;
padding: 8px 10px;
box-shadow: 0 2px 12px 0 rgb(56, 56, 56, .2);
border: 1px solid #e2e6ed;
}
.tool-btn {
cursor: pointer;
&:hover, &.active {
color: $themeColor;
}
& + .tool-btn {
margin-left: 15px;
}
}
.page-number {
font-size: 12px;
padding: 0 12px;
cursor: pointer;
}
}
</style>