效果图

组件所需VUE3代码
TypeScript
<template>
<div class="video-dialog" :class="fullScreen && 'video-dialog-full-screen'">
<el-dialog
v-model="props.visible"
draggable
:show-close="false"
title=""
center
align-center
@close="changeVisible"
>
<!-- 视频容器 -->
<div
class="video-content"
:style="{
backgroundImage: firstFrameUrl ? `url(${firstFrameUrl})` : 'none',
}"
>
<!-- 背景磨砂层 -->
<div class="video-backdrop" v-if="firstFrameUrl"></div>
<!-- 视频加载中状态 -->
<div class="video-loading" v-if="isLoading">
<div class="loading-spinner"></div>
<p v-if="loadingText">{{ loadingText }}</p>
</div>
<!-- 视频标签 -->
<video
ref="videoRef"
:src="videoSrc"
@timeupdate="updateProgress"
@ended="handleEnded"
@click="togglePlay"
@volumechange="updateVolume"
@seeked="updateBackground"
@loadedmetadata="handleMetadataLoaded"
@error="handleVideoError"
@waiting="handleVideoBuffering"
@playing="handleVideoPlaying"
></video>
<!-- 播放错误提示 -->
<div class="video-error" v-if="videoError">
<img :src="DataUrl.ErrorIcon || 'https://picsum.photos/48/48'" alt="播放错误" class="error-icon" />
<p class="error-text">{{ errorMessage }}</p>
<p class="error-tips">建议使用MP4格式(H.264编码)重新上传</p>
</div>
<!-- 视频数据 -->
<ul class="video-data" v-if="previewInfo">
<li>
<img :src="DataUrl.ZanIcon" alt="" class="data-icon i-1" />
<p>760</p>
</li>
<li>
<img :src="DataUrl.PingIcon" alt="" class="data-icon i-2" />
<p>670</p>
</li>
<li>
<img :src="DataUrl.LookEyeIcon" alt="" class="data-icon i-3" />
<p>890</p>
</li>
</ul>
<!-- 视频基本信息 -->
<div class="video-base-info" v-if="previewInfo">
<div class="title flex">
<h1>网红名称</h1>
<span></span>
<p>2025.03.30</p>
</div>
<p class="note">
风掠过耳际时突然懂了:原来旅行不是赶路,是让山川湖海,把心里的褶皱慢慢烫平。
</p>
</div>
<img
@click="changeVisible"
:src="DataUrl.CloseRadiuIcon"
alt=""
class="close-icon"
/>
</div>
<!-- 视频进度条 -->
<div class="video-progress">
<div class="progress-bg" @click="seekToPosition">
<div class="progress-value" :style="{ width: progress + '%' }"></div>
<div class="progress-pointer" :style="{ left: progress + '%' }"></div>
</div>
</div>
<!-- 视频控制栏 -->
<div class="video-footer-tool">
<div class="flex">
<!-- 播放按钮 -->
<img
:src="isPlaying ? DataUrl.PauseIcon : DataUrl.PlayZIcon"
alt=""
class="icon-1"
@click="togglePlay"
/>
<!-- 下一个按钮 -->
<img
:src="DataUrl.PlayNextIcon"
alt=""
class="icon-2"
@click="playNextVideo"
/>
<!-- 音量控制 -->
<div class="volume-container">
<img
:src="isMuted ? DataUrl.MuteIcon : DataUrl.YlIcon"
alt=""
class="icon-3"
@click="toggleMute"
/>
<div class="hover-mute-content" @mouseenter="showVolumeSlider">
<div
class="hover-mute-content-value"
:style="{ width: volumePercent + '%' }"
@mousedown="startVolumeAdjust"
></div>
</div>
</div>
<!-- 视频时间 -->
<p>
{{ currentTime }} /<span>{{ durationTime }}</span>
</p>
</div>
<!-- 全屏按钮 -->
<div>
<img
:src="fullScreen ? DataUrl.UnFdIcon : DataUrl.FdIcon"
alt=""
class="icon-4"
@click="fullScreen = !fullScreen"
/>
</div>
</div>
</el-dialog>
</div>
</template>
<script setup lang="ts">
import { ref, onMounted, watch, nextTick } from "vue"
import DataUrl from "@/config/data-url.js"
// 定义props
const props = defineProps({
visible: {
type: Boolean,
default: false,
},
previewInfo: {
type: Boolean,
default: true,
},
videoInfo: {
type: Object,
default: {},
},
})
// 视频源
const videoSrc = ref(
props.videoInfo?.url ||
"https://umi-rise.oss-ap-southeast-1.aliyuncs.com/20250707/WeChat_20250707174726.mp4"
)
// 组件状态
const videoRef = ref(null)
const isPlaying = ref(false)
const isMuted = ref(false)
const progress = ref(0)
const currentTime = ref("00:00")
const durationTime = ref("00:00")
const showControls = ref(true)
const timer = ref(null)
const fullScreen = ref(false)
// 音量控制状态
const volumePercent = ref(70)
const isAdjustingVolume = ref(false)
const isVolumeSliderVisible = ref(false)
// 背景控制状态
const showLeftBackground = ref(false)
const showRightBackground = ref(false)
const videoAspectRatio = ref(0)
const firstFrameUrl = ref("")
const isFirstFrameLoaded = ref(false)
// 播放错误状态
const videoError = ref(false)
const errorMessage = ref("")
// 新增:加载状态
const isLoading = ref(true)
const loadingText = ref("准备播放...")
const emit = defineEmits(["update-visible", "confim"])
// 关闭对话框
const changeVisible = () => {
if (videoRef.value) {
videoRef.value.pause();
isPlaying.value = false; // 重置播放状态
}
emit("update-visible", false)
}
// 播放/暂停视频
const togglePlay = () => {
if (videoRef.value) {
if (isPlaying.value) {
videoRef.value.pause()
} else {
videoRef.value.play().catch((err) => {
console.log("自动播放失败,需要用户交互后才能播放", err)
})
}
isPlaying.value = !isPlaying.value
}
}
// 更新进度条
const updateProgress = () => {
if (videoRef.value) {
const percent = (videoRef.value.currentTime / videoRef.value.duration) * 100
progress.value = Math.min(100, percent)
currentTime.value = formatTime(videoRef.value.currentTime)
if (videoRef.value.duration > 0 && durationTime.value === "00:00") {
durationTime.value = formatTime(videoRef.value.duration)
}
}
}
// 格式化时间
const formatTime = (seconds: number) => {
const minutes = Math.floor(seconds / 60)
const secs = Math.floor(seconds % 60)
return `${minutes.toString().padStart(2, "0")}:${secs
.toString()
.padStart(2, "0")}`
}
// 进度条点击跳转
const seekToPosition = (e: MouseEvent) => {
if (videoRef.value) {
const progressBar = e.currentTarget as HTMLElement
const rect = progressBar.getBoundingClientRect()
const clickX = e.clientX - rect.left
const percent = (clickX / rect.width) * 100
videoRef.value.currentTime = (percent / 100) * videoRef.value.duration
}
}
// 音量控制
const toggleMute = () => {
if (videoRef.value) {
isMuted.value = !isMuted.value
videoRef.value.muted = isMuted.value
volumePercent.value = isMuted.value ? 0 : volumePercent.value
}
}
const showVolumeSlider = () => {
isVolumeSliderVisible.value = true
}
const startVolumeAdjust = (e: MouseEvent) => {
isAdjustingVolume.value = true
adjustVolume(e)
document.addEventListener("mousemove", adjustVolume)
document.addEventListener("mouseup", endVolumeAdjust)
e.preventDefault()
}
const adjustVolume = (e: MouseEvent) => {
if (!isAdjustingVolume.value || !videoRef.value) return
const volumeBar = document.querySelector(".hover-mute-content") as HTMLElement
if (!volumeBar) return
const rect = volumeBar.getBoundingClientRect()
const clickX = e.clientX - rect.left
let percent = (clickX / rect.width) * 100
percent = Math.max(0, Math.min(100, percent))
volumePercent.value = percent
videoRef.value.volume = percent / 100
if (isMuted.value) {
isMuted.value = false
videoRef.value.muted = false
}
}
const endVolumeAdjust = () => {
isAdjustingVolume.value = false
document.removeEventListener("mousemove", adjustVolume)
document.removeEventListener("mouseup", endVolumeAdjust)
}
// 更新音量显示
const updateVolume = () => {
if (videoRef.value) {
if (videoRef.value.muted !== isMuted.value) {
isMuted.value = videoRef.value.muted
}
if (!isMuted.value) {
volumePercent.value = Math.round(videoRef.value.volume * 100)
}
}
}
// 更新背景显示
const updateBackground = () => {
if (!videoRef.value || !videoRef.value.videoWidth) return
// 计算视频宽高比
videoAspectRatio.value =
videoRef.value.videoWidth / videoRef.value.videoHeight
// 获取容器尺寸
const container = videoRef.value.parentElement
if (!container) return
const containerWidth = container.offsetWidth
const containerHeight = container.offsetHeight
// 计算容器宽高比
const containerAspectRatio = containerWidth / containerHeight
// 根据宽高比差异调整视频显示逻辑
const isVideoWider = videoAspectRatio.value > containerAspectRatio
showLeftBackground.value = isVideoWider
showRightBackground.value = isVideoWider
}
// 加载第一帧
const loadFirstFrame = async () => {
if (!videoRef.value) return
try {
// 尝试获取视频编码信息(部分浏览器支持)
const videoTracks = videoRef.value.videoTracks
if (videoTracks && videoTracks.length > 0) {
const codec = videoTracks[0].codec || videoTracks[0].kind
if (codec && !codec.includes("avc1") && !codec.includes("h264")) {
console.warn("检测到非H.264编码,可能无法播放:", codec)
// 可在这里显示警告提示
}
}
// 尝试直接使用视频帧
await captureFrame()
} catch (error) {
console.error("直接捕获失败:", error)
// 尝试通过Fetch和Blob URL绕过跨域
try {
const response = await fetch(videoSrc.value)
const blob = await response.blob()
const blobUrl = URL.createObjectURL(blob)
const tempVideo = document.createElement("video")
tempVideo.crossOrigin = "anonymous"
tempVideo.src = blobUrl
await new Promise((resolve, reject) => {
tempVideo.onloadedmetadata = resolve
tempVideo.onerror = reject
})
tempVideo.currentTime = 0.1
await new Promise((resolve, reject) => {
tempVideo.onseeked = resolve
tempVideo.onerror = reject
})
const canvas = document.createElement("canvas")
canvas.width = tempVideo.videoWidth
canvas.height = tempVideo.videoHeight
const ctx = canvas.getContext("2d")
ctx.drawImage(tempVideo, 0, 0, canvas.width, canvas.height)
firstFrameUrl.value = canvas.toDataURL("image/jpeg", 0.8)
isFirstFrameLoaded.value = true
// 清理资源
URL.revokeObjectURL(blobUrl)
tempVideo.remove()
} catch (error) {
console.error("Blob方法失败:", error)
// 回退到默认占位图
firstFrameUrl.value =
"https://umi-rise.oss-ap-southeast-1.aliyuncs.com/20250707/cover-video-00008.png"
isFirstFrameLoaded.value = true
}
}
}
const captureFrame = () => {
return new Promise((resolve, reject) => {
if (!videoRef.value || !videoRef.value.videoWidth) {
reject(new Error("视频未加载"))
return
}
const video = videoRef.value
video.currentTime = 0.5
const handleSeeked = () => {
const canvas = document.createElement("canvas")
canvas.width = video.videoWidth
canvas.height = video.videoHeight
const ctx = canvas.getContext("2d")
ctx.drawImage(video, 0, 0, canvas.width, canvas.height)
try {
firstFrameUrl.value = canvas.toDataURL("image/jpeg", 0.8)
isFirstFrameLoaded.value = true
resolve()
} catch (error) {
reject(error)
} finally {
video.removeEventListener("seeked", handleSeeked)
}
}
video.addEventListener("seeked", handleSeeked)
})
}
// 处理视频元数据加载完成
const handleMetadataLoaded = () => {
loadingText.value = "加载中..."
// 视频元数据加载完成,但可能还需要缓冲
// 不立即隐藏loading,等待canplay或playing事件
}
// 处理视频缓冲
const handleVideoBuffering = () => {
if (!isLoading.value) {
isLoading.value = true
loadingText.value = "缓冲中..."
}
}
// 处理视频开始播放
const handleVideoPlaying = () => {
// 视频真正开始播放时,隐藏loading
isLoading.value = false
isPlaying.value = true
}
// 处理视频播放错误
const handleVideoError = () => {
if (!videoRef.value) return
const error = videoRef.value.error
if (!error) return
// 根据错误码判断原因
switch (error.code) {
case error.MEDIA_ERR_ABORTED:
errorMessage.value = "视频加载被中断"
break
case error.MEDIA_ERR_NETWORK:
errorMessage.value = "网络错误,无法加载视频"
break
case error.MEDIA_ERR_DECODE:
errorMessage.value = "视频编码不支持,无法播放"
break
case error.MEDIA_ERR_SRC_NOT_SUPPORTED:
errorMessage.value = "视频格式不支持"
break
default:
errorMessage.value = "播放失败,请重试"
}
// 显示错误提示,隐藏第一帧背景和loading
videoError.value = true
firstFrameUrl.value = ""
isLoading.value = false
isPlaying.value = false
}
// 播放下一个视频
const playNextVideo = () => {
console.log("播放下一个视频")
}
// 视频播放结束
const handleEnded = () => {
isPlaying.value = false
console.log("视频播放结束")
}
// 显示/隐藏控制栏
const toggleControls = () => {
showControls.value = !showControls.value
if (timer.value) clearTimeout(timer.value)
timer.value = setTimeout(() => {
showControls.value = false
}, 3000)
}
const playVideo = () => {
videoError.value = false
isLoading.value = true // 开始加载
loadingText.value = "准备播放..."
videoRef.value.removeEventListener("click", toggleControls)
videoRef.value.addEventListener("click", toggleControls)
videoRef.value.play().catch((err) => {
console.log("自动播放失败", err)
isLoading.value = false // 加载失败,隐藏loading
})
videoRef.value.addEventListener("click", toggleControls)
nextTick(updateProgress)
nextTick(updateBackground)
// 延迟加载第一帧,确保视频元素已初始化
setTimeout(loadFirstFrame, 100)
}
// 生命周期钩子
onMounted(() => {
if (videoRef.value) {
playVideo()
}
// 监听窗口大小变化
window.addEventListener("resize", updateBackground)
})
// 监听全屏状态变化
watch(
() => fullScreen.value,
() => {
nextTick(updateBackground)
}
)
// 监听视频源变化
watch(
() => props.videoInfo?.url,
(newUrl, oldUrl) => {
if (newUrl && newUrl !== oldUrl) {
// 更新视频源
videoSrc.value = newUrl
// 重置视频状态
videoError.value = false
isPlaying.value = false
progress.value = 0
currentTime.value = "00:00"
// 加载新视频时显示loading
isLoading.value = true
loadingText.value = "准备播放..."
// 重新加载并播放新视频
if (videoRef.value) {
videoRef.value.src = newUrl
videoRef.value.load()
videoRef.value.play().catch((err) => {
console.log("切换视频播放失败:", err)
isPlaying.value = false
isLoading.value = false // 加载失败,隐藏loading
})
// 重新加载第一帧
loadFirstFrame()
}
}
},
{ immediate: true }
)
// 组件卸载时清理
onUnmounted(() => {
if (timer.value) clearTimeout(timer.value)
window.removeEventListener("resize", updateBackground)
})
</script>
<style lang="scss" scoped>
.video-dialog,
.video-dialog-full-screen {
:deep(.el-overlay) {
background: rgba(0, 0, 0, 0.66);
}
:deep(.el-dialog) {
width: 900px;
--el-dialog-border-radius: 20px;
--el-dialog-padding-primary: 0;
background: transparent;
border-radius: 12px;
padding: 0;
overflow: hidden;
.video-content {
width: 100%;
height: 503px;
position: relative;
overflow: hidden;
background-size: cover;
background-position: center;
// 背景磨砂层
.video-backdrop {
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.4);
backdrop-filter: blur(10px);
z-index: 1;
}
video {
max-width: 100%;
max-height: 100%;
height: 100%;
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
object-fit: contain;
z-index: 2;
}
// 加载状态样式
.video-loading {
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
z-index: 5; // 高于视频但低于错误提示
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
background: rgba(0, 0, 0, 0.6);
.loading-spinner {
width: 48px;
height: 48px;
border: 4px solid rgba(255, 255, 255, 0.3);
border-radius: 50%;
border-top-color: #ff3f81;
animation: spin 1s linear infinite;
margin-bottom: 16px;
}
p {
color: white;
font-size: 16px;
font-family: 'PingFang SC', sans-serif;
}
}
@keyframes spin {
to { transform: rotate(360deg); }
}
// 播放错误提示
.video-error {
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
z-index: 10; // 最高层级
text-align: center;
width: 80%;
.error-icon {
width: 48px;
height: 48px;
margin-bottom: 16px;
}
.error-text {
font-size: 18px;
color: #ff4d4f;
margin-bottom: 8px;
}
.error-tips {
font-size: 14px;
color: rgba(255, 255, 255, 0.7);
}
}
// 视频数据
.video-data {
position: absolute;
right: 24px;
bottom: 49px;
z-index: 3;
li {
margin-bottom: 17px;
&:last-child {
margin-bottom: 0;
}
}
.data-icon {
width: 24px;
height: 24px;
display: block;
margin: auto;
}
p {
font-family: DIN, DIN;
color: #ffffff;
text-shadow: 1px 1px 2px rgba(0, 0, 0, 0.6);
margin-top: 15px;
font-weight: 400;
font-size: 16px;
}
}
.close-icon {
position: absolute;
right: 24px;
top: 35px;
width: 50px;
height: 50px;
z-index: 2;
cursor: pointer;
}
.video-base-info {
position: absolute;
left: 24px;
bottom: 12px;
z-index: 3;
.title {
align-items: center;
h1 {
font-family: PingFang SC, PingFang SC;
font-weight: 500;
font-size: 18px;
color: #ffffff;
text-shadow: 1px 1px 2px rgba(0, 0, 0, 0.6);
}
span {
margin: 0 8px;
display: inline-block;
width: 2px;
height: 2px;
background: #ffffff;
box-shadow: 1px 1px 2px 0px rgba(0, 0, 0, 0.6);
}
p {
font-family: DIN, DIN;
font-weight: 400;
font-size: 12px;
color: #eae9e8;
text-shadow: 1px 1px 2px rgba(0, 0, 0, 0.6);
}
}
.note {
margin-top: 2px;
width: 234px;
font-family: PingFang SC, PingFang SC;
font-weight: 400;
font-size: 14px;
color: #ffffff;
text-shadow: 1px 1px 2px rgba(0, 0, 0, 0.6);
line-height: 1.5;
}
}
}
.video-content::before {
content: "";
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
backdrop-filter: blur(8px);
background: rgba(255, 255, 255, 0.1);
z-index: 2;
pointer-events: none;
}
// 进度条样式
.video-progress {
position: absolute;
bottom: 56px;
left: 0;
right: 0;
height: 4px;
z-index: 3;
.progress-bg {
width: 100%;
height: 100%;
background: rgba(255, 255, 255, 0.2);
border-radius: 2px;
cursor: pointer;
.progress-value {
height: 100%;
background: #ff3f81;
border-radius: 2px;
width: 0;
transition: width 0.1s;
position: relative;
}
.progress-value::after {
content: "";
position: absolute;
right: 0;
top: 50%;
transform: translateY(-50%);
width: 12px;
height: 12px;
background: #ff3f81;
border-radius: 50%;
}
.progress-pointer {
position: absolute;
top: 50%;
transform: translateY(-50%);
width: 12px;
height: 12px;
background: #ff3f81;
border-radius: 50%;
margin-left: -6px;
box-shadow: 0 0 8px rgba(255, 63, 129, 0.5);
display: none;
}
}
}
// 底部控制栏样式
.video-footer-tool {
height: 56px;
width: 100%;
display: flex;
justify-content: space-between;
align-items: center;
background: #16110c;
padding: 0 24px;
.icon-1,
.icon-2,
.icon-3 {
width: 14.22px;
height: 16px;
margin-right: 24px;
cursor: pointer;
object-fit: contain;
}
.icon-3 {
width: 16px;
margin-right: 0;
}
.icon-4 {
width: 18px;
height: 18px;
cursor: pointer;
}
p {
font-family: PingFang SC, PingFang SC;
font-weight: 500;
font-size: 14px;
color: #ffffff;
span {
color: rgba($color: #ffffff, $alpha: 0.65);
}
}
// 音量控制样式
.volume-container {
display: flex;
align-items: center;
margin-right: 24px;
.hover-mute-content {
width: 68px;
height: 6px;
border-radius: 24px;
background: rgba(255, 255, 255, 0.37);
cursor: pointer;
position: relative;
display: none;
margin-left: 24px;
.hover-mute-content-value {
position: absolute;
top: 0;
left: 0;
width: 50%;
height: 100%;
background: #ffffff;
border-radius: 24px;
transition: width 0.1s;
&::after {
content: "";
position: absolute;
right: 0;
top: 50%;
transform: translateY(-50%);
width: 12px;
height: 12px;
background: #ffffff;
border-radius: 50%;
box-shadow: 0 0 8px rgba(255, 255, 255, 0.5);
}
}
}
}
.volume-container:hover {
width: calc(72px + 68px - 24px);
max-width: calc(72px + 68px - 24px);
.hover-mute-content {
display: block;
}
}
}
}
}
// 全屏模式样式
.video-dialog-full-screen {
:deep(.el-dialog) {
width: 100%;
height: 100%;
--el-dialog-border-radius: 0;
background: transparent;
border-radius: 0;
.video-content {
width: 100%;
height: calc(100vh - 56px);
video {
max-width: 100%;
max-height: 100%;
height: 100%;
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
object-fit: contain;
z-index: 2;
}
}
}
}
</style>
DataUrl Icon文件
新建或者自定义一个相关文件,将以下ICON.base64代码引入,这边用的是js文件
javascript
const dataUrl = {
"ErrorIcon": ``, "ZanIcon":``,
"PingIcon": ``,
"LookEyeIcon": ``,
"CloseRadiuIcon": ``,
"PauseIcon": ``,
"PlayZIcon": ``,
"PlayNextIcon": ``,
"MuteIcon": ``,
"YlIcon": ``,
"UnFdIcon": ``,
"FdIcon": ``,
}
export default dataUrl