web_ppt / frontend /src /views /Screen /PresenterView.vue
CatPtain's picture
Upload 339 files
89ce340 verified
<template>
<div class="presenter-view">
<div class="toolbar">
<div class="tool-btn" @click="changeViewMode('base')"><IconListView class="tool-icon" /><span>普通视图</span></div>
<div class="tool-btn" :class="{ 'active': writingBoardToolVisible }" @click="writingBoardToolVisible = !writingBoardToolVisible"><IconWrite class="tool-icon" /><span>画笔</span></div>
<div class="tool-btn" :class="{ 'active': laserPen }" @click="laserPen = !laserPen"><IconMagic class="tool-icon" /><span>激光笔</span></div>
<div class="tool-btn" :class="{ 'active': timerlVisible }" @click="timerlVisible = !timerlVisible"><IconStopwatchStart class="tool-icon" /><span>计时器</span></div>
<div class="tool-btn" @click="() => fullscreenState ? manualExitFullscreen() : enterFullscreen()">
<IconOffScreenOne class="tool-icon" v-if="fullscreenState" />
<IconFullScreenOne class="tool-icon" v-else />
<span>{{ fullscreenState ? '退出全屏' : '全屏' }}</span>
</div>
<Divider class="divider" />
<div class="tool-btn" @click="exitScreening()"><IconPower class="tool-icon" /><span>结束放映</span></div>
</div>
<div class="content">
<div
class="slide-list-wrap"
:class="{ 'laser-pen': laserPen }"
ref="slideListWrapRef"
>
<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"
/>
<WritingBoardTool
:slideWidth="slideWidth"
:slideHeight="slideHeight"
:left="-365"
:top="-155"
v-if="writingBoardToolVisible"
@close="writingBoardToolVisible = false"
/>
<CountdownTimer
v-if="timerlVisible"
:left="75"
@close="timerlVisible = false"
/>
</div>
<div class="thumbnails"
ref="thumbnailsRef"
@wheel.prevent="$event => handleMousewheelThumbnails($event)"
>
<div
class="thumbnail"
:class="{ 'active': index === slideIndex }"
v-for="(slide, index) in slides"
:key="slide.id"
@click="turnSlideToIndex(index)"
>
<ThumbnailSlide :slide="slide" :size="120 / viewportRatio" :visible="index < slidesLoadLimit" />
</div>
</div>
</div>
<div class="remark">
<div class="header">
<span>演讲者备注</span>
<span>P {{slideIndex + 1}} / {{slides.length}}</span>
</div>
<div class="remark-content ProseMirror-static" :class="{ 'empty': !currentSlideRemark }" :style="{ fontSize: remarkFontSize + 'px' }" v-html="currentSlideRemark || '无备注'"></div>
<div class="remark-scale">
<div :class="['scale-btn', { 'disable': remarkFontSize === 12 }]" @click="setRemarkFontSize(remarkFontSize - 2)"><IconMinus /></div>
<div :class="['scale-btn', { 'disable': remarkFontSize === 40 }]" @click="setRemarkFontSize(remarkFontSize + 2)"><IconPlus /></div>
</div>
</div>
</div>
</template>
<script lang="ts" setup>
import { computed, nextTick, ref, watch } from 'vue'
import { storeToRefs } from 'pinia'
import { useSlidesStore } from '@/store'
import type { ContextmenuItem } from '@/components/Contextmenu/types'
import { enterFullscreen } from '@/utils/fullscreen'
import { parseText2Paragraphs } from '@/utils/textParser'
import useScreening from '@/hooks/useScreening'
import useLoadSlides from '@/hooks/useLoadSlides'
import useExecPlay from './hooks/useExecPlay'
import useSlideSize from './hooks/useSlideSize'
import useFullscreen from './hooks/useFullscreen'
import ThumbnailSlide from '@/views/components/ThumbnailSlide/index.vue'
import ScreenSlideList from './ScreenSlideList.vue'
import WritingBoardTool from './WritingBoardTool.vue'
import CountdownTimer from './CountdownTimer.vue'
import Divider from '@/components/Divider.vue'
const props = defineProps<{
changeViewMode: (mode: 'base' | 'presenter') => void
}>()
const { slides, slideIndex, viewportRatio, currentSlide } = storeToRefs(useSlidesStore())
const slideListWrapRef = ref<HTMLElement>()
const thumbnailsRef = ref<HTMLElement>()
const writingBoardToolVisible = ref(false)
const timerlVisible = ref(false)
const laserPen = ref(false)
const {
mousewheelListener,
touchStartListener,
touchEndListener,
turnPrevSlide,
turnNextSlide,
turnSlideToIndex,
turnSlideToId,
animationIndex,
} = useExecPlay()
const { slideWidth, slideHeight } = useSlideSize(slideListWrapRef)
const { exitScreening } = useScreening()
const { slidesLoadLimit } = useLoadSlides()
const { fullscreenState, manualExitFullscreen } = useFullscreen()
const remarkFontSize = ref(16)
const currentSlideRemark = computed(() => {
if (!currentSlide.value.remark) return ''
return parseText2Paragraphs(currentSlide.value.remark)
})
const handleMousewheelThumbnails = (e: WheelEvent) => {
if (!thumbnailsRef.value) return
thumbnailsRef.value.scrollBy(e.deltaY, 0)
}
const setRemarkFontSize = (fontSize: number) => {
if (fontSize < 12 || fontSize > 40) return
remarkFontSize.value = fontSize
}
watch(slideIndex, () => {
nextTick(() => {
if (!thumbnailsRef.value) return
const activeThumbnailRef: HTMLElement | null = thumbnailsRef.value.querySelector('.thumbnail.active')
if (!activeThumbnailRef) return
const width = thumbnailsRef.value.offsetWidth
const offsetLeft = activeThumbnailRef.offsetLeft + activeThumbnailRef.clientWidth / 2
thumbnailsRef.value.scrollTo({ left: offsetLeft - width / 2, behavior: 'smooth' })
})
})
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: '画笔工具',
handler: () => writingBoardToolVisible.value = true,
},
{
text: '普通视图',
handler: () => props.changeViewMode('base'),
},
{ divider: true },
{
text: '结束放映',
subText: 'ESC',
handler: exitScreening,
},
]
}
</script>
<style lang="scss" scoped>
.presenter-view {
width: 100%;
height: 100%;
display: flex;
}
.toolbar {
width: 70px;
height: 100%;
background-color: #fff;
border-right: solid 1px #eee;
font-size: 12px;
margin: 20px 0;
.tool-btn {
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
cursor: pointer;
& + .tool-btn {
margin-top: 22px;
}
&:hover, &.active {
color: $themeColor;
}
}
.divider {
width: 70%;
margin: 24px 15% !important;
}
.tool-icon {
margin-bottom: 8px;
font-size: 22px;
}
}
.content {
width: calc(100% - 430px);
height: 100%;
background-color: #1d1d1d;
}
.slide-list-wrap {
height: calc(100% - 190px);
margin: 20px;
overflow: hidden;
position: relative;
&.laser-pen {
cursor: url() 20 20, default !important;
}
}
.thumbnails {
height: 150px;
padding: 15px;
white-space: nowrap;
overflow-x: auto;
overflow-y: hidden;
border-top: solid 1px #3a3a3a;
position: relative;
}
.thumbnail {
display: inline-block;
outline: 2px solid #aaa;
& + .thumbnail {
margin-left: 10px;
}
&:hover {
outline-color: $themeColor;
}
&.active {
outline-width: 3px;
outline-color: $themeColor;
}
}
.remark {
width: 360px;
height: 100%;
position: relative;
background-color: #2a2a2a;
border-left: solid 1px #3a3a3a;
color: #fff;
.header {
height: 60px;
padding: 0 20px;
display: flex;
justify-content: space-between;
align-items: center;
font-size: 18px;
border-bottom: 1px solid #3a3a3a;
}
.remark-content {
height: calc(100% - 60px);
padding: 20px;
line-height: 1.5;
@include overflow-overlay();
&.empty {
color: #999;
font-style: italic;
}
}
.remark-scale {
position: absolute;
right: 5px;
bottom: 5px;
font-size: 22px;
display: flex;
}
.scale-btn {
width: 40px;
height: 40px;
display: flex;
justify-content: center;
align-items: center;
user-select: none;
cursor: pointer;
&.disable {
color: #666;
cursor: no-drop;
}
&:not(.disable):hover {
background-color: #333;
}
}
}
::-webkit-scrollbar {
width: 0;
height: 0;
}
</style>