Vue3实现视频播放弹窗组件,支持全屏播放,音量控制,进度条自定义样式,适配浏览器小窗播放,视频大小自适配,缓冲loading,代码复制即用

效果图

组件所需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
相关推荐
byzh_rc5 分钟前
[微机原理与系统设计-从入门到入土] 微型计算机基础
开发语言·javascript·ecmascript
m0_471199635 分钟前
【小程序】订单数据缓存 以及针对海量库存数据的 懒加载+数据分片 的具体实现方式
前端·vue.js·小程序
编程大师哥6 分钟前
Java web
java·开发语言·前端
A小码哥8 分钟前
Vibe Coding 提示词优化的四个实战策略
前端
Murrays8 分钟前
【React】01 初识 React
前端·javascript·react.js
大喜xi11 分钟前
ReactNative 使用百分比宽度时,aspectRatio 在某些情况下无法正确推断出高度,导致图片高度为 0,从而无法显示
前端
helloCat12 分钟前
你的前端代码应该怎么写
前端·javascript·架构
电商API_1800790524712 分钟前
大麦网API实战指南:关键字搜索与详情数据获取全解析
java·大数据·前端·人工智能·spring·网络爬虫
康一夏13 分钟前
CSS盒模型(Box Model) 原理
前端·css
web前端12314 分钟前
React Hooks 介绍与实践要点
前端·react.js