成人怡红院-成人怡红院视频在线观看-成人影视大全-成人影院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號
高清毛茸茸的中国少妇| 国产精品VA在线播放| 第九理论午夜电影院| 国产AV导航大全精品| 国产毛片精选好视频| 娇妻在客厅被朋友玩得呻吟动漫 | 亚洲精品无码成人片久久| 亚洲色欲久久久久综合网| 中文字幕人妻高清乱码| YIN荡护士揉捏乱P办公室视频| 粗大的内捧猛烈进出动态图 | 亚洲欧洲AV综合色无码| 中美日韩精品激情无码AV| TOUGHTESTED硬汉系列| 国产成人精品久久久久| 黑人大雞巴XⅩⅩ| 两根大肉大捧一进一出好爽视频| 全部AV―极品视觉盛宴| 天堂中文在线最新版WWW| 亚洲国产成人精品无码区在线| 一区二区三区无码在线观看| WWW久久只有这里有精品| 国产精品成熟老妇女| 久久发布国产伦子伦精品| 欧美人妻久久精品| 无码人妻出轨与黑人中文字幕| 亚洲人成未满十八禁网站| 99久久国产露脸国语对白| 丰满的继牳3中文字幕系列| 激情欧美日韩一区二区| 女神被啪进深处娇喘在线观看| 熟悉妇人妻av无码毛片| 亚洲一区二区三区无码蜜桃 | 余生请多指教在线观看免费全集| 草木影视在线视频免费观看| 国内精品自国内精品自线| 男同桌上课用手指进去了好爽| 色欲人妻AAAAAAA无码| 亚洲精品中文字幕无码专区| 99久久精品费精品国产| 国产精品亚洲А∨无码播放| 久久久受WWW免费人成| 日本ⅩXXX色视频在线观看| 亚洲AV无码一区二区三区天堂 | 伊人久久大香线蕉AV仙人| 差差差很疼免费软件APP下载 | 日产精品乱码卡一卡2卡三网站| 亚洲AV五十路在线观看| XXX.日本学生妹.COM| 狠狠色成人一区二区三区| 人妻巨大乳一二三区麻豆| 亚洲成人免费av| 必看无人区一码二码三码| 精品一区二区三区东京热| 日韩精品人妻系列无码专区| 亚洲综合精品第一页| 国产成人18黄网站免费观看| 老旺大肉蟒进进出出电影免费| 乌克兰ZOOMKOOL| 99国产精品无码| 交换配乱吟粗大SNS84O| 色噜噜精品一区二区三区| 又色又爽的无遮挡免费网址| 国产精品免费_区二区三区观看| 男男野战爆了我的菊BL| 亚洲AV综合色区无码专区桃色| 波多野美乳人妻HD电影欧美 | 97免费公开在线视频| 和老师做H无码动漫| 日韩人妻无码一区2区3区里沙 | 午夜免费无码福利视频| 啊灬啊灬啊灬快灬深用力 | 岳女二人名器共侍一夫的出处 | 午夜精品无人区乱码1区2区| OM老熟女DHXⅩXXX69| 久久国产欧美日韩精品| 我的娇妻QUEEN| ワンピースのエロ.WWW在线| 久久精品国产亚洲AV忘忧草18| 偷拍 亚洲 卡通 另类 小说| FREE俄罗斯免费视频| 久久五月丁香综合中文亚洲| 亚州中文字幕无码中文字幕| 吃奶呻吟打开双腿做受视频免费| 美女裸露双奶头尿口无遮挡网站| 亚洲AV无码兔费综合| 国产AⅤ精品一区三区| 欧美一区二期三区视频| 在火车和后妈妈谁在一起是什么书| 韩国午夜理伦三级理论三级| 天天躁日日躁很很很躁| 成人H视频在线观看| 年轻漂亮岳每4乱理2| 亚洲最大成人一区久久久| 国产又色又爽又刺激在线播放| 少妇特黄Av一区二区三区 | 国产仑乱无码内谢| 色欲AⅤ蜜臀AV免费观看| CAOPORN免费视频在线| 乱中年女人伦AV三区| 亚洲欧美VR色区| 国内精品久久久久久久久电影网| 他的白月光H1∨1笔趣阁| 车内挺进尤物少妇紧窄| 欧美胖老太牲交XXⅩXXX| 中国人熟女HDFREEHDXⅩ| 久精品夜色国产亚洲AV| 亚洲AV无码传区国产乱码O| 国产精品国产三级欧美二区| 少妇高潮惨叫久久久久电影69| 啊灬啊灬啊灬快高潮了网站 | 办公室欧美大尺寸SUV| 欧美一级草B内射| 337P日本欧洲亚洲大胆裸体艺 | OLDGRAANNY日本老熟妇| 欧美Z0ZO人禽交| …日韩人妻无码精品一专区| 麻豆文化传媒WWW| 中国人妻XXXXX免费看| 巨大欧美黑人XXXXBBBB| 一本久久A精品一区二区| 国产精品原创AV片国产日韩| 精品国产人成亚洲区| 亚洲AV图片一亚洲AV| 国产乱妇乱子在线视频| 无码超级大爆乳在线播放| 国产成人牲交在线观看视频| 四虎库影必出精品8848| 国产成人无码AV麻豆| 无码 有码 日韩 人妻| 国产精品欧美一区二区三区| 无码一区二区波多野结衣播放搜索| 国产成人久久AV免费| 我的真實亂倫故事| 国产免费又爽又色又粗视频| 亚洲AV成人无码久久WWW| 国产伦精品一区二区三区免费| 午夜性色一区二区三区不卡视频| 国产男女爽爽爽免费视频| 亚洲AV无码成人网站国产网站| 黄 色 网 站 在 线 免费| 亚洲精品自偷自拍无码| 久久久97精品国产一区蜜桃| 岳把腿扒开让我添| 欧美精品VIDEOSSEX少妇| MM1313亚洲国产精品无码试| 日韩欧美操逼视频| 国产精品久久无码不卡黑寡妇| 亚洲AV成人噜噜无码网站 | 午夜无码无遮挡在线视频| 国产最新无码AⅤ在线播放| 亚洲国产精品久久久天堂不卡海量| 久久精品国产亚洲AV麻豆甜| 中文天堂在线WWW最新版官网| 欧美熟妇ⅤIDEOS| 儿子比老公更大更硬朗| 小SAO货撅起屁股扒开GIF动| 精品无码人妻一区二区免费蜜桃| 樱花YY私人在线影院| 欧美国产激情二区三区| 大学生酒店呻吟在线观看| 无码人妻一区二区三区免费视频 | 亚洲人成网站18禁止大| 蜜芽亚洲AV无码精品国产| 啊灬啊灬啊灬快灬高潮少妇软件 | 色狠狠色噜噜AV综合五区| 国产内射老熟女AAAA∵| 亚洲熟妇在线观看| 欧美人与动牲交A免费观看| 公和我做好爽添厨房在线观看| 亚洲AV无码国产一区二区三区不| 久久亚洲私人国产精品VA| 扒开腿挺进湿润的花苞| 无人区码一码二码三码区别在哪里| 精品人妻一区二区三区四区| 717午夜伦伦电影理论片| 试看120秒男女啪啪免费| 精品久久亚洲中文字幕| 97人妻精品一区二区三区| 玩弄JaPan白嫩少妇HD小说| 久久99热精品免费观看| CAOPORN超碰进入页面| 午夜福利1000集无码| 麻豆AV字幕无码中文| 成年女人午夜毛片免费视频| 亚洲AV片一区二区三区| 内谢XXXXX8老| 国产免费AV片在线无码免费看 | 欧美一区二区放荡人妇| 国产精品揄拍100视频| 在线天堂中文最新版| 特大黑人巨交吊性XXXX| 久久精品人妻一区二区三区av| 波多野结衣初尝黑人巨大| 亚洲国产成人爱AV网站| 欧美一级137片内射亚洲| 国自产精品手机在线观看视频| 99在线精品国自产拍| 亚洲AV女人18毛片水真多| 欧美一区二期三区视频|