图片播放器模仿视频播放器

arm设备,安装的浏览器无法播放视频。快速播放图片,模仿播放视频

第一版:直接快速播放网络图片,效果非常差,1秒5帧都无法满足

第二版:使用双缓冲策略,网页上使用两个image标签轮流显示,显示效果有提升,但是不大

第三版:加入预加载策略,增加一个长度为5的预加载队列,用于预先去网络请求图片,每显示一张图片的时候,都会去更新预加载队列,删除最早加入的图片,并开始预加载新的图片。效果提升明显,内网环境下可满足每秒20帧的要求

优化点:双缓冲+预加载

<template>
  <div class="image-player">
    <!-- 播放器主体区域 -->
    <div class="player-container">
      <img
        :src="bufferA.src"
        alt="播放图片"
        v-show="bufferA.isActive"
        @load="handleImageLoad('A')"
        :class="{ 'fade-in': bufferA.isActive }"
      />
      <img
        :src="bufferB.src"
        alt="播放图片"
        v-show="bufferB.isActive"
        @load="handleImageLoad('B')"
        :class="{ 'fade-in': bufferB.isActive }"
      />
      <div v-if="isLoading" class="loading-indicator">加载中...</div>
    </div>


    <!-- 控制按钮区域 -->
    <div class="controls">
      <button @click="togglePlay">
        {{ isPlaying ? "暂停" : "播放" }}
      </button>
      <span class="progress">
        {{ currentIndex + 1 }} / {{ images.length }}
      </span>
      <!-- 添加帧率选择器 -->
      <div class="fps-selector">
        <span>帧率:</span>
        <select v-model="selectedFps">
          <option v-for="fps in fpsOptions" :key="fps" :value="fps">
            {{ fps }} FPS
          </option>
        </select>
      </div>
    </div>
  </div>
</template>

<script>
export default {
  name: "ImagePlayer",


  data() {
    return {
      // 图片列表数组
      images: [
        // 添加更多图片URL
      ],
      currentIndex: 0, // 当前播放的图片索引
      isPlaying: false, // 播放状态
      playIntervalTimer: null, // 播放定时器
      // 添加双缓冲相关的数据
      bufferA: {
        src: "",
        isActive: true,
        isLoaded: false,
      },
      bufferB: {
        src: "",
        isActive: false,
        isLoaded: false,
      },
      // 添加帧率相关数据
      fpsOptions: [1, 2, 5, 10, 15, 20, 24],
      selectedFps: 5, // 默认5帧每秒
      preloadQueue: [], // 预加载队列
      maxPreloadCount: 5, // 最大预加载数量
      isLoading: false,
    };
  },

  computed: {
    currentImage() {
      return this.images[this.currentIndex];
    },
    // 根据选择的帧率计算播放间隔(毫秒)
    playInterval() {
      // 向下取整
      return Math.floor(1000 / this.selectedFps);
    },
  },
  created() {
    // 生成100张图片
    for (let i = 0; i < 1000; i++) {
      this.images.push(`https://picsum.photos/200?t=${i}`);
    }
    // 初始化第一张图片
    this.bufferA.src = this.images[0];
    // 初始化预加载队列
    this.initPreloadQueue();
  },
  methods: {
    // 初始化预加载队列
    initPreloadQueue() {
      const startIdx = (this.currentIndex + 1) % this.images.length;
      for (let i = 0; i < this.maxPreloadCount; i++) {
        const idx = (startIdx + i) % this.images.length;
        this.preloadImage(this.images[idx]);
      }
    },

    // 预加载图片
    preloadImage(src) {
      if (!this.preloadQueue.includes(src)) {
        const img = new Image();
        img.src = src;
        this.preloadQueue.push(src);
        if (this.preloadQueue.length > this.maxPreloadCount) {
          this.preloadQueue.shift(); // 移除最早的预加载图片
        }
      }
    },

    // 处理图片加载
    handleImageLoad(buffer) {
      if (buffer === "A") {
        this.bufferA.isLoaded = true;
      } else {
        this.bufferB.isLoaded = true;
      }
      this.isLoading = false;
    },

    // 优化后的播放下一张图片方法
    async nextImage() {
      this.isLoading = true;
      const nextIndex = (this.currentIndex + 1) % this.images.length;

      // 切换缓冲区
      this.bufferA.isActive = !this.bufferA.isActive;
      this.bufferB.isActive = !this.bufferB.isActive;

      // 更新当前索引
      this.currentIndex = nextIndex;

      // 预加载下一张图片
      const nextBuffer = this.bufferA.isActive ? this.bufferB : this.bufferA;
      nextBuffer.isLoaded = false;
      nextBuffer.src = this.images[nextIndex];

      // 更新预加载队列
      const preloadIndex =
        (nextIndex + this.maxPreloadCount) % this.images.length;
      this.preloadImage(this.images[preloadIndex]);
    },

    // 优化后的播放控制
    play() {
      if (!this.isPlaying) {
        this.isPlaying = true;
        this.playIntervalTimer = setInterval(() => {
          if (!this.isLoading) {
            this.nextImage();
          }
        }, this.playInterval);
      }
    },

    // 切换播放/暂停状态
    togglePlay() {
      if (this.isPlaying) {
        this.pause();
      } else {
        this.play();
      }
    },

    // 暂停播放
    pause() {
      this.isPlaying = false;
      if (this.playIntervalTimer) {
        clearInterval(this.playIntervalTimer);
        this.playIntervalTimer = null;
      }
    },
  },

  // 组件销毁时清理定时器
  beforeDestroy() {
    // 清理预加载队列
    this.preloadQueue = [];
    this.pause();
  },

  // 监听帧率变化
  watch: {
    selectedFps() {
      // 如果正在播放,重新开始播放以应用新的帧率
      if (this.isPlaying) {
        this.pause();
        this.$nextTick(() => {
          this.play();
        });
      }
    },
  },
};
</script>

<style scoped>
.image-player {
  width: 100%;
  max-width: 800px;
  margin: 0 auto;
}


.player-container {
  width: 100%;
  aspect-ratio: 16/9;
  background: #f0f0f0;
  display: flex;
  align-items: center;
  justify-content: center;
  overflow: hidden;
}

.player-container img {
  position: absolute;
  max-width: 100%;
  max-height: 100%;
  object-fit: contain;
  will-change: transform; /* 优化渲染性能 */
  transition: opacity 0.2s ease-in-out;
}

.controls {
  margin-top: 10px;
  display: flex;
  align-items: center;
  gap: 20px; /* 增加间距 */
}

button {
  padding: 8px 16px;
  cursor: pointer;
}

.progress {
  font-size: 14px;
}

/* 添加帧率选择器样式 */
.fps-selector {
  display: flex;
  align-items: center;
  gap: 8px;
}

.fps-selector select {
  padding: 4px 8px;
  border-radius: 4px;
  border: 1px solid #ccc;
  cursor: pointer;
}

.loading-indicator {
  position: absolute;
  top: 50%;
  left: 50%;
  transform: translate(-50%, -50%);
  background: rgba(0, 0, 0, 0.5);
  color: white;
  padding: 10px;
  border-radius: 4px;
}

.fade-in {
  /* animation: fadeIn 0.2s ease-in-out; */
}

@keyframes fadeIn {
  from {
    opacity: 0;
  }
  to {
    opacity: 1;
  }
}
</style>
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 222,252评论 6 516
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,886评论 3 399
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 168,814评论 0 361
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,869评论 1 299
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,888评论 6 398
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,475评论 1 312
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 41,010评论 3 422
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,924评论 0 277
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,469评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,552评论 3 342
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,680评论 1 353
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,362评论 5 351
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 42,037评论 3 335
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,519评论 0 25
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,621评论 1 274
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 49,099评论 3 378
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,691评论 2 361

推荐阅读更多精彩内容