成人怡红院-成人怡红院视频在线观看-成人影视大全-成人影院203nnxyz-美女毛片在线看-美女免费黄

站長資訊網
最全最豐富的資訊網站

手把手教你用Vue3寫播放器

本篇文章給大家帶來了關于Vue3的相關知識,其中主要跟大家聊一聊怎么用Vue3寫個播放器,感興趣的朋友下面一起來看一下吧,希望對大家有幫助。

ps:音樂可能播放失敗。原因是 audio 的鏈接是臨時的,手動替換下即可。

手把手教你用Vue3寫播放器

TODO

  • 實現播放/暫停;
  • 實現開始/結束時間及開始時間和滾動條動態跟隨播放動態變化;
  • 實現點擊進度條跳轉指定播放位置;
  • 實現點擊圓點拖拽滾動條。

頁面布局及 css 樣式如下

<template>   <div class="song-item">     <audio src="" />     <!-- 進度條 -->     <div class="audio-player">       <span>00:00</span>       <div class="progress-wrapper">         <div class="progress-inner">           <div class="progress-dot" />         </div>       </div>       <span>00:00</span>       <!-- 播放/暫停 -->       <div style="margin-left: 10px; color: #409eff; cursor: pointer;" >         播放      </div>     </div>   </div></template><style lang="scss">   * { font-size: 14px; }  .song-item {    display: flex;    flex-direction: column;    justify-content: center;    height: 100px;    padding: 0 20px;    transition: all ease .2s;    border-bottom: 1px solid #ddd;    /* 進度條樣式 */     .audio-player {      display: flex;      height: 18px;      margin-top: 10px;      align-items: center;      font-size: 12px;      color: #666;      .progress-wrapper {        flex: 1;        height: 4px;        margin: 0 20px 0 20px;        border-radius: 2px;        background-color: #e9e9eb;        cursor: pointer;        .progress-inner {          position: relative;          width: 0%;          height: 100%;          border-radius: 2px;          background-color: #409eff;          .progress-dot {            position: absolute;            top: 50%;            right: 0;            z-index: 1;            width: 10px;            height: 10px;            border-radius: 50%;            background-color: #409eff;            transform: translateY(-50%);           }         }       }     }   }</style>
登錄后復制

實現播放/暫停

思路:給 ”播放“ 注冊點擊事件,在點擊事件中通過 audio 的屬性及方法來判定當前歌曲是什么狀態,是否播放或暫停,然后聲明一個屬性同步這個狀態,在模板中做出判斷當前應該顯示 ”播放/暫停“。

關鍵性 api:

audio.paused:當前播放器是否為暫停狀態

audio.play():播放

audio.pause():暫停

const audioIsPlaying = ref(false); // 用于同步當前的播放狀態const audioEle = ref<HTMLAudioElement | null>(null); // audio 元素/**  * @description 播放/暫停音樂  */const togglePlayer = () => {  if (audioEle.value) {    if (audioEle.value?.paused) {       audioEle.value.play();       audioIsPlaying.value = true;     }    else {       audioEle.value?.pause();       audioIsPlaying.value = false;     }   } };onMounted(() => {  // 頁面點擊的時候肯定是加載完成了,這里獲取一下沒毛病   audioEle.value = document.querySelector('audio'); });
登錄后復制

最后把屬性及事件應用到模板中去。

<div    style="margin-left: 10px; color: #409eff; cursor: pointer;"   @click="togglePlayer">    {{ audioIsPlaying ? '暫停' : '播放'}}</div>
登錄后復制

實現開始/結束時間及開始時間和滾動條動態跟隨播放動態變化

思路:獲取當前已經播放的時間及總時長,然后再拿當前時長 / 總時長及得到歌曲播放的百分比即滾動條的百分比。通過偵聽 audio 元素的 timeupdate 事件以做到每次當前時間改變時,同步把 DOM 也進行更新。最后播放完成后把狀態初始化。

關鍵性api:

audio.currentTime:當前的播放時間;單位(s)

audio.duration:音頻的總時長;單位(s)

timeupdatecurrentTime 變更時會觸發該事件。

import dayjs from 'dayjs';const audioCurrentPlayTime = ref('00:00'); // 當前播放時長const audioCurrentPlayCountTime = ref('00:00'); // 總時長const pgsInnerEle = ref<HTMLDivElement | null>(null);/**  * @description 更新進度條與當前播放時間  */const updateProgress = () => {  const currentProgress = audioEle.value!.currentTime / audioEle.value!.duration;    pgsInnerEle.value!.style.width = `${currentProgress * 100}%`;  // 設置進度時長   if (audioEle.value)     audioCurrentPlayTime.value = dayjs(audioEle.value.currentTime * 1000).format('mm:ss'); };/**  * @description 播放完成重置播放狀態  */const audioPlayEnded = () => {   audioCurrentPlayTime.value = '00:00';   pgsInnerEle.value!.style.width = '0%';   audioIsPlaying.value = false; };onMounted(() => {   pgsInnerEle.value = document.querySelector('.progress-inner');     // 設置總時長   if (audioEle.value)     audioCurrentPlayCountTime.value = dayjs(audioEle.value.duration * 1000).format('mm:ss');       // 偵聽播放中事件   audioEle.value?.addEventListener('timeupdate', updateProgress, false);  // 播放結束 event   audioEle.value?.addEventListener('ended', audioPlayEnded, false); });
登錄后復制

實現點擊進度條跳轉指定播放位置

思路:給滾動條注冊鼠標點擊事件,每次點擊的時候獲取當前的 offsetX 以及滾動條的寬度,用寬度 / offsetX 最后用總時長 * 前面的商就得到了我們想要的進度,再次更新進度條即可。

關鍵性api:

event.offsetX:鼠標指針相較于觸發事件對象的 x 坐標。

/**  * @description 點擊滾動條同步更新音樂進度  */const clickProgressSync = (event: MouseEvent) => {  if (audioEle.value) {    // 保證是正在播放或者已經播放的狀態     if (!audioEle.value.paused || audioEle.value.currentTime !== 0) {      const pgsWrapperWidth = pgsWrapperEle.value!.getBoundingClientRect().width;      const rate = event.offsetX / pgsWrapperWidth;      // 同步滾動條和播放進度       audioEle.value.currentTime = audioEle.value.duration * rate;      updateProgress();     }   } };onMounted({   pgsWrapperEle.value = document.querySelector('.progress-wrapper');  // 點擊進度條 event   pgsWrapperEle.value?.addEventListener('mousedown', clickProgressSync, false); });// 別忘記統一移除偵聽onBeforeUnmount(() => {   audioEle.value?.removeEventListener('timeupdate', updateProgress);   audioEle.value?.removeEventListener('ended', audioPlayEnded);   pgsWrapperEle.value?.removeEventListener('mousedown', clickProgressSync); });
登錄后復制

實現點擊圓點拖拽滾動條。

思路:使用 hook 管理這個拖動的功能,偵聽這個滾動條的 鼠標點擊、鼠標移動、鼠標抬起事件。

點擊時: 獲取鼠標在窗口的 x 坐標,圓點距離窗口的 left 距離及最大的右移距離(滾動條寬度 – 圓點距離窗口的 left)。為了讓移動式不隨便開始計算,在開始的時候可以弄一個開關 flag

移動時: 實時獲取鼠標在窗口的 x 坐標減去 點擊時獲取的 x 坐標。然后根據最大移動距離做出判斷,不要讓它越界。最后: 總時長 * (圓點距離窗口的 left + 計算得出的 x / 滾動條長度) 得出百分比更新滾動條及進度

抬起時:將 flag 重置。

/**  * @method useSongProgressDrag  * @param audioEle  * @param pgsWrapperEle  * @param updateProgress 更新滾動條方法  * @param startSongDragDot 是否開啟拖拽滾動  * @description 拖拽更新歌曲播放進度  */const useSongProgressDrag = (   audioEle: Ref<HTMLAudioElement | null>,   pgsWrapperEle: Ref<HTMLDivElement | null>,   updateProgress: () => void,   startSongDragDot: Ref<boolean>) => {  const audioPlayer = ref<HTMLDivElement | null>(null);  const audioDotEle = ref<HTMLDivElement | null>(null);  const dragFlag = ref(false);  const position = ref({    startOffsetLeft: 0,    startX: 0,    maxLeft: 0,    maxRight: 0,   });  /**    * @description 鼠標點擊 audioPlayer    */   const mousedownProgressHandle = (event: MouseEvent) => {    if (audioEle.value) {      if (!audioEle.value.paused || audioEle.value.currentTime !== 0) {         dragFlag.value = true;          position.value.startOffsetLeft = audioDotEle.value!.offsetLeft;         position.value.startX = event.clientX;         position.value.maxLeft = position.value.startOffsetLeft;         position.value.maxRight = pgsWrapperEle.value!.offsetWidth - position.value.startOffsetLeft;       }     }     event.preventDefault();     event.stopPropagation();   };  /**    * @description 鼠標移動 audioPlayer    */   const mousemoveProgressHandle = (event: MouseEvent) => {    if (dragFlag.value) {      const clientX = event.clientX;      let x = clientX - position.value.startX;      if (x > position.value.maxRight)         x = position.value.maxRight;      if (x < -position.value.maxLeft)         x = -position.value.maxLeft;      const pgsWidth = pgsWrapperEle.value?.getBoundingClientRect().width;      const reat = (position.value.startOffsetLeft + x) / pgsWidth!;        audioEle.value!.currentTime = audioEle.value!.duration * reat;      updateProgress();     }   };  /**    * @description 鼠標取消點擊    */   const mouseupProgressHandle = () => dragFlag.value = false;  onMounted(() => {    if (startSongDragDot.value) {       audioPlayer.value = document.querySelector('.audio-player');       audioDotEle.value = document.querySelector('.progress-dot');      // 在捕獲中去觸發點擊 dot 事件. fix: 點擊原點 offset 回到原點 bug       audioDotEle.value?.addEventListener('mousedown', mousedownProgressHandle, true);       audioPlayer.value?.addEventListener('mousemove', mousemoveProgressHandle, false);      document.addEventListener('mouseup', mouseupProgressHandle, false);     }   });  onBeforeUnmount(() => {    if (startSongDragDot.value) {       audioPlayer.value?.removeEventListener('mousedown', mousedownProgressHandle);       audioPlayer.value?.removeEventListener('mousemove', mousemoveProgressHandle);      document.removeEventListener('mouseup', mouseupProgressHandle);     }   }); };
登錄后復制

最后調用這個 hook

// 是否顯示可拖拽 dot// 可以在原點元素上增加 v-if 用來判定是否需要拖動功能const startSongDragDot = ref(true);useSongProgressDrag(audioEle, pgsWrapperEle, updateProgress, startSongDragDot);
登錄后復制

贊(0)
分享到: 更多 (0)
?
網站地圖   滬ICP備18035694號-2    滬公網安備31011702889846號
天天AV天天爽无码中文| 日本精品一线二线三线区别在哪里| 国产AV无码区亚洲AV欧美| 玩小雪跪趴把腿分到最大影视| 欧美肥胖老太大喷水| 久久久久久精品免费看SSS | 人与野鲁交XXXⅩ视频| 久久综合噜噜激激的五月天| 黑人巨大猛烈捣出白浆视频在线 | 中文字幕亚洲一区二区VA在线| 亚洲国产AV高清无码| 天堂VA在线高清一区| 亚洲AV无码国产永久播放蜜芽 | 亚洲精品国产精品乱码不99| 无码一区二区三区中文字幕| 少妇WWB搡BBBB搡BBBB| 强行无套内谢大学生初次| 男生把感叹号放进女生的括号| 久久久久久精品免费免费直播| 免费拗女网站1300部| 久久婷婷五月综合色99啪AK| 精品深夜AV无码一区二区老年| 国精产品一区一区三区有限| 国产台湾无码AV片在线观看| 国产精品秘 入口A级熟女| GOGO全球高清专业大尺度摄影| 337P日本欧洲亚洲大胆裸体艺| JLZZZJLZZZ国产免费观| 国产熟女乱子视频正在播放| GV天堂GV无码男同在线观看| 国产SUV精品一区二区| 狠狠精品久久久无码中文字幕 | 亚洲中文自拍另类AV片| 亚洲AV永久无码精品一区| 亚洲AⅤ日韩久久久久久| 无码人妻丝袜在线视频| 无人区一码二码三码四码| 无码一区二区三区中文字幕| 野花新免费高清完整在线观看 | 高黄暴H日本在线观看| 初小VIDEOS第一次摘花| 嗯啊WW免费视频网站| 国产边做边吃奶AⅤ视频免费| 夫目前侵犯一区二区三区| 精品国产AV色欲果冻传媒| 久久精品99国产精品日本| 久久午夜夜伦鲁鲁片免费无码影视| 久久精品国产精品国产一区 | YY8男人的天堂| А√天堂资源地址在线官网BT| YW尤物AV无码| 狠狠人妻熟妇av又粗又大| 欧洲另类二三四区| 亚洲MV国产MV在线MV综合试| 亚洲人成网站色7799| 中国鲜肉GAY高中XX禁18网| 99久久精品免费观看国产| 成年无码AV片完整版| 国产成人乱色伦区小说| 国产女人高潮抽搐喷水嗷嗷叫| 精品国产国语对白久久免费| 日本强好片久久久久久AAA | 天天躁日日躁狠狠躁欧美老妇小说| 一边做一边说国语对白| 中文字幕一区二区三区乱码人妻 | 草莓丝瓜芭乐鸭脖奶茶发型| 差差差不多视频30分钟轮滑| 久久99精品久久水蜜桃| 奶头被民工们吸得又红又肿怎么办 | 亚洲娇小与黑人巨大video| 亚洲婷婷综合色高清在线| 2022久久国产精品免费热麻豆| 国产日产欧洲无码视频| 人妻丰满熟妇aⅴ无码HD| 天天天天做夜夜夜夜做无码| 13小箩利洗澡无码视频网站| 成人精品一二三区| 巨爆乳寡妇中文在线观看| 午夜麻豆国产精品无码| 波多野结衣TORRENT| 国产欧美日韩专区发布| 日本丰满少妇毛茸茸| 在线观看亚洲AV每日更新| 成人网站V片免费观看| 国产亚洲色欲色一色WWW| 鲁鲁夜夜天天综合视频| 日韩揉捏奶头高潮不断视频| 亚洲AV永久无码精品一区二区| 低调看JRS直播| 精品国产第一国产综合精品| 日韩人妻无码精品系列| 80岁老熟妇乱子伦牲交| 国产浮力第一页草草影院| 久久久久噜噜噜亚洲熟女综合| 人人妻人人澡人人爽人人到DVD| 性一交一乱一伦一在线小视频| 中文无码日韩欧免费视频APP| 狠狠色丁香婷婷久久综合不卡| 舌头伸进去添的我好爽高潮电影 | 激情五月综合 香亚洲| 欧美高清VIDEOS36OP| 亚洲A∨精品无码一区二区| 高潮奶水涨喷在线播放| 日本高清在线观看视频WWW色 | 亚洲色偷无码一区二区| 边吃奶边扎下面动态| 男女交性视频无遮挡全过程 | 裸体跳舞XXXX裸体跳舞| 亚洲熟女丰满多毛XXXXX| 狠狠躁夜夜躁人人爽碰AV| 午夜福制92视频| 丰满爆乳BBWBBWBBW| 久久精品午夜一区二区福利| 日韩免费无码专区精品观看| GOGO全球大胆高清人体131| 免费午夜爽爽爽WWW视频十八禁 | 国产精品无码久久综合| 双乳被一左一右吃着的小说| 尤物永久免费AV无码网站| 国产97色在线 | 免| 老外和中国女人毛片免费视频| 调教狠扇打肿私密跪撅屁股作文| 真人新婚之夜破苞第一次视频 | 无码国产精品一区二区免费式直播 | 性一交一乱一伦一色一情| 国产精华精华液一二三区别| 狠狠色伊人亚洲综合网站野外| 无码囯产精品一区二区免费| 在线观看亚洲AV电影网站| 老熟妇乱子伦牲交视频| 亚洲综合色婷婷在线观看| 成人国内精品视频在线观看| 欧美疯狂做受XXXX高潮| A在线视频播放观看免费观看| 黑人大雞巴ⅩⅩⅩ| 两个病娇男友轮流爱我资源| 亚洲色成人网站WWW永久四虎| 精品人妻一区二区三区三区换着玩 | 性欧美大战久久久久久久久 | 男朋友一晚弄了我5次正常吗| 中文在线最新版天堂| 奶头被客人玩的又红又肿| av资源在线观看少妇| 久久成人国产精品免费| 野花电影3在线观看免费| 麻花传媒MD0076在线入口| 亚洲AV综合色区无码一二三区| 寂寞骚妇被后入式爆草抓爆| 熟女体下毛毛黑森林| 国产精品亚洲产品一区二区三区| 人体艺术大胆图片| 成视频年人黄网站免费视频| 天堂AV无码大芭蕉伊人AV孕妇 | 亚洲亚洲人成无码网WWW| 狂猛欧美激情性XXXX大豆行情| 重口老太大和小伙乱| 秋霞在线观看视频| 1000部又爽又黄无遮挡的视频| 欧美精品第1页WWW| 第一次挺进苏雨萌| 香港三日本8A三级少妇三级99| 国产av一区二区三区| 性丰满ⅩXXOOO性FREE| 精品无码一区二区三区爱欲九九| 又爽又刺激免费男女视频| 欧美成在线精品视频| 吃奶呻吟打开双腿做受是免费视频| 无码人妻一区二区三区免费视频| 激情综合婷婷丁香五月情| 在线A毛片免费视频观看| 人妻无码不卡在线视频| 国产精品久久成人网站| 他揉捏她两乳不停呻吟人妻| 哈昂~哈昂够了太多了动图| 性一交一乱一伦一| 久久香蕉综合色一综合色88| XXXⅩ少妇少妇XXXX范冰冰| 他将头埋进双腿间吮小核故事 | 亚洲AV永久无码精品一区二区| 老汉粗大不带套怀孕| 大黑大巴大战欧洲美女图片| 亚洲AV无码国产精品永久一区| 巨大黑人极品videos精品| 成 人色 网 站 欧美大片在线| 亚洲AV无码国产一区二区三区| 免费观看添你到高潮视频| 丰满妇女强高潮18ⅩXXXHD| 亚洲丰满熟妇在线播放电影全集| 韩国三级HD中文字幕叫床| 中文字幕AV无码不卡免费| 熟妇高潮一区二区精品午夜无码| 精品国产一区二区三区久久| 99精品国产成人一区二区| 强 暴 疼 哭 处 女| 国产精品久免费的黄网站| 又黄又爽又无遮挡免费的网站| 色99久久久久高潮综合影院| 国产高清在线A视频大全| 伊人久久久AV老熟妇色| 男女嘿咻发声动态图|