223 lines
5.7 KiB
Vue
223 lines
5.7 KiB
Vue
<script setup lang="ts">
|
||
import { danmus as danmusData, getDanmuData } from "./danmu.js";
|
||
import { onMounted, onUnmounted, reactive, ref } from "vue";
|
||
import VueDanmaku from "vue3-danmaku";
|
||
|
||
defineOptions({
|
||
name: "Danmaku"
|
||
});
|
||
|
||
const danmaku = ref();
|
||
const danmus = ref<any[]>(getDanmuData());
|
||
const danmuMsg = ref<string>("");
|
||
let timer = 0;
|
||
const config = reactive({
|
||
channels: 5, // 轨道数量,为0则弹幕轨道数会撑满容器
|
||
useSlot: true, // 是否开启slot
|
||
loop: true, // 是否开启弹幕循环
|
||
speeds: 200, // 弹幕速度,实际为弹幕滚动完一整屏的秒数,值越小速度越快
|
||
fontSize: 20, // 文本模式下的字号
|
||
top: 10, // 弹幕轨道间的垂直间距
|
||
right: 0, // 同一轨道弹幕的水平间距
|
||
debounce: 100, // 弹幕刷新频率(多少毫秒插入一条弹幕,建议不小于50)
|
||
randomChannel: true // 随机弹幕轨道
|
||
});
|
||
|
||
onMounted(() => {
|
||
window.onresize = () => resizeHandler();
|
||
});
|
||
|
||
onUnmounted(() => {
|
||
window.onresize = null;
|
||
});
|
||
|
||
function play(type: string) {
|
||
switch (type) {
|
||
case "play":
|
||
danmaku.value.play();
|
||
break;
|
||
case "pause":
|
||
danmaku.value.pause();
|
||
break;
|
||
case "stop":
|
||
danmaku.value.stop();
|
||
break;
|
||
case "show":
|
||
danmaku.value.show();
|
||
break;
|
||
case "hide":
|
||
danmaku.value.hide();
|
||
break;
|
||
case "reset":
|
||
danmaku.value.reset();
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
}
|
||
|
||
function switchSlot(slot: boolean) {
|
||
config.useSlot = slot;
|
||
danmus.value = slot ? getDanmuData() : danmusData;
|
||
|
||
setTimeout(() => {
|
||
danmaku.value.stop();
|
||
danmaku.value.play();
|
||
});
|
||
}
|
||
function speedsChange(val: number) {
|
||
if (config.speeds <= 10 && val === -10) {
|
||
return;
|
||
}
|
||
config.speeds += val;
|
||
danmaku.value.reset();
|
||
}
|
||
function fontChange(val: number) {
|
||
config.fontSize += val;
|
||
danmaku.value.reset();
|
||
}
|
||
function channelChange(val: number) {
|
||
if (!config.channels && val === -1) {
|
||
return;
|
||
}
|
||
config.channels += val;
|
||
}
|
||
function resizeHandler() {
|
||
if (timer) clearTimeout(timer);
|
||
timer = window.setTimeout(() => {
|
||
danmaku.value.resize();
|
||
}, 500);
|
||
}
|
||
function addDanmu() {
|
||
if (!danmuMsg.value) return;
|
||
const _danmuMsg = config.useSlot
|
||
? {
|
||
avatar: "https://i.loli.net/2021/01/17/xpwbm3jKytfaNOD.jpg",
|
||
name: "你",
|
||
text: danmuMsg.value
|
||
}
|
||
: danmuMsg.value;
|
||
danmaku.value.add(_danmuMsg);
|
||
danmuMsg.value = "";
|
||
}
|
||
</script>
|
||
<template>
|
||
<el-card shadow="never">
|
||
<template #header>
|
||
<div class="card-header">
|
||
<span class="font-medium">
|
||
弹幕组件,采用开源的
|
||
<el-link
|
||
href="https://github.com/hellodigua/vue-danmaku/tree/vue3"
|
||
target="_blank"
|
||
style="margin: 0 4px 5px; font-size: 16px"
|
||
>
|
||
vue3-danmaku
|
||
</el-link>
|
||
</span>
|
||
</div>
|
||
<el-link
|
||
class="mt-2"
|
||
href="https://github.com/pure-admin/vue-pure-admin/blob/main/src/views/able/danmaku"
|
||
target="_blank"
|
||
>
|
||
代码位置 src/views/able/danmaku
|
||
</el-link>
|
||
</template>
|
||
<div class="flex gap-5">
|
||
<vue-danmaku
|
||
ref="danmaku"
|
||
v-model:danmus="danmus"
|
||
class="demo"
|
||
isSuspend
|
||
v-bind="config"
|
||
>
|
||
<!-- 弹幕slot -->
|
||
<template v-slot:dm="{ danmu, index }">
|
||
<div class="danmu-item">
|
||
<img class="img" :src="danmu.avatar" />
|
||
<span>{{ index }}{{ danmu.name }}:</span>
|
||
<span>{{ danmu.text }}</span>
|
||
</div>
|
||
</template>
|
||
</vue-danmaku>
|
||
<div class="main">
|
||
<p>
|
||
播放:
|
||
<el-button @click="play('play')">播放</el-button>
|
||
<el-button @click="play('pause')">暂停</el-button>
|
||
<el-button @click="play('stop')">停止</el-button>
|
||
</p>
|
||
<p>
|
||
模式:
|
||
<el-button @click="switchSlot(true)">弹幕 slot</el-button>
|
||
<el-button @click="switchSlot(false)">普通文本</el-button>
|
||
</p>
|
||
<p>
|
||
显示:
|
||
<el-button @click="play('show')">显示</el-button>
|
||
<el-button @click="play('hide')">隐藏</el-button>
|
||
</p>
|
||
<p>
|
||
速度:
|
||
<el-button @click="speedsChange(-10)">减速</el-button>
|
||
<el-button @click="speedsChange(10)">增速</el-button>
|
||
<span class="ml-5">当前速度:{{ config.speeds }}像素/s</span>
|
||
</p>
|
||
<p>
|
||
字号:
|
||
<el-button :disabled="config.useSlot" @click="fontChange(-1)">
|
||
缩小
|
||
</el-button>
|
||
<el-button :disabled="config.useSlot" @click="fontChange(1)">
|
||
放大
|
||
</el-button>
|
||
<span class="ml-5">当前字号:{{ config.fontSize }}px</span>
|
||
</p>
|
||
<p>
|
||
轨道:
|
||
<el-button @click="channelChange(-1)">-1</el-button>
|
||
<el-button @click="channelChange(1)">+1</el-button>
|
||
<el-button @click="channelChange(-config.channels)"> 填满 </el-button>
|
||
<span class="ml-5">当前轨道:{{ config.channels }}</span>
|
||
</p>
|
||
<p class="flex">
|
||
<el-input
|
||
v-model="danmuMsg"
|
||
type="text"
|
||
placeholder="输入评论后,回车发送弹幕"
|
||
@keyup.enter="addDanmu"
|
||
/>
|
||
</p>
|
||
</div>
|
||
</div>
|
||
</el-card>
|
||
</template>
|
||
<style lang="scss" scoped>
|
||
.demo {
|
||
flex: 1;
|
||
height: 600px;
|
||
background: linear-gradient(45deg, #5ac381, #20568b);
|
||
|
||
.danmu-item {
|
||
display: flex;
|
||
align-items: center;
|
||
|
||
.img {
|
||
width: 25px;
|
||
height: 25px;
|
||
margin-right: 5px;
|
||
border-radius: 50%;
|
||
}
|
||
}
|
||
}
|
||
|
||
.main {
|
||
flex: 1;
|
||
|
||
p {
|
||
margin-top: 10px;
|
||
}
|
||
}
|
||
</style>
|