精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

HarmonyOS開發(fā)案例:【視頻播放器】

jf_46214456 ? 來(lái)源:jf_46214456 ? 作者:jf_46214456 ? 2024-04-24 14:52 ? 次閱讀

介紹

使用ArkTS語(yǔ)言實(shí)現(xiàn)視頻播放器,主要包括主頁(yè)面和視頻播放頁(yè)面,我們將一起完成以下功能:

  1. 獲取本地視頻和網(wǎng)絡(luò)視頻。
  2. 通過(guò)AVPlayer進(jìn)行視頻播放。
  3. 通過(guò)手勢(shì)調(diào)節(jié)屏幕亮度和視頻播放音量。

相關(guān)概念

  • [AVPlayer]:播放管理類,用于管理和播放媒體資源。
  • [XComponent]:可用于EGL/OpenGLES和媒體數(shù)據(jù)寫入,并顯示在XComponent組件。
  • [PanGesture]手勢(shì):用于觸發(fā)拖動(dòng)手勢(shì)事件,滑動(dòng)的最小距離為5vp時(shí)拖動(dòng)手勢(shì)識(shí)別成功。

相關(guān)權(quán)限

本篇Codelab使用了網(wǎng)絡(luò)連接,需要在配置文件module.json5文件里添加權(quán)限:ohos.permission.INTERNET。

環(huán)境搭建

軟件要求

  • [DevEco Studio]版本:DevEco Studio 3.1 Release。
  • OpenHarmony SDK版本:API version 9。

硬件要求

  • 開發(fā)板類型:[潤(rùn)和RK3568開發(fā)板]。
  • OpenHarmony系統(tǒng):3.2 Release。

環(huán)境搭建

完成本篇Codelab我們首先要完成開發(fā)環(huán)境的搭建,本示例以RK3568開發(fā)板為例,參照以下步驟進(jìn)行:

  1. [獲取OpenHarmony系統(tǒng)版本]:標(biāo)準(zhǔn)系統(tǒng)解決方案(二進(jìn)制)。以3.2 Release版本為例:
  2. 搭建燒錄環(huán)境。
    1. [完成DevEco Device Tool的安裝]
    2. [完成RK3568開發(fā)板的燒錄]
  3. 搭建開發(fā)環(huán)境。
    1. 開始前請(qǐng)參考[工具準(zhǔn)備],完成DevEco Studio的安裝和開發(fā)環(huán)境配置。
    2. 開發(fā)環(huán)境配置完成后,請(qǐng)參考[使用工程向?qū)創(chuàng)建工程(模板選擇“Empty Ability”)。
    3. 工程創(chuàng)建完成后,選擇使用[真機(jī)進(jìn)行調(diào)測(cè)]。
    4. 鴻蒙開發(fā)指導(dǎo)文檔:[gitee.com/li-shizhen-skin/harmony-os/blob/master/README.md]

代碼結(jié)構(gòu)解讀

本篇Codelab只對(duì)核心代碼進(jìn)行講解,對(duì)于完整代碼,我們會(huì)在gitee中提供。

搜狗高速瀏覽器截圖20240326151450.png

HarmonyOSOpenHarmony鴻蒙文檔籽料:mau123789v直接拿
├──entry/src/main/ets	                   // 代碼區(qū)
│  ├──common
│  │  ├──constants
│  │  │  ├──CommonConstants.ets	           // 公共常量類
│  │  │  ├──HomeConstants.ets	           // 首頁(yè)常量類
│  │  │  └──PlayConstants.ets	           // 視頻播放頁(yè)面常量類
│  │  ├──model
│  │  │  ├──HomeTabModel.ets	           // 首頁(yè)參數(shù)模型
│  │  │  └──PlayerModel.ets	               // 播放參數(shù)模型
│  │  └──util
│  │     ├──DateFormatUtil.ets	           // 日期工具類
│  │     ├──GlobalContext.ets	           // 全局工具類
│  │     ├──Logger.ets	                   // 日志工具類
│  │     └──ScreenUtil.ets                 // 屏幕工具類
│  ├──controller
│  │  └──VideoController.ets	           // 視頻控制類
│  ├──entryability
│  │  └──EntryAbility.ts                   // 程序入口類
│  ├──pages
│  │  ├──HomePage.ets                      // 首頁(yè)頁(yè)面
│  │  └──PlayPage.ets                      // 視頻播放頁(yè)面
│  ├──view
│  │  ├──HomeTabContent.ets                // 首頁(yè)Tab頁(yè)面
│  │  ├──HomeTabContentButton.ets          // 首頁(yè)按鈕子組件
│  │  ├──HomeTabContentDialog.ets          // 添加網(wǎng)絡(luò)視頻彈框子組件
│  │  ├──HomeTabContentList.ets            // 視頻列表子組件
│  │  ├──HomeTabContentListItem.ets        // 視頻對(duì)象子組件
│  │  ├──PlayControl.ets                   // 播放控制子組件
│  │  ├──PlayPlayer.ets                    // 視頻播放子組件
│  │  ├──PlayProgress.ets                  // 播放進(jìn)度子組件
│  │  ├──PlayTitle.ets                     // 播放標(biāo)題子組件
│  │  └──PlayTitleDialog.ets               // 播放速度設(shè)置子組件
│  └──viewmodel
│     ├──HomeDialogModel.ets         	   // 添加網(wǎng)絡(luò)視頻彈框類
│     ├──HomeVideoListModel.ets            // 獲取視頻列表數(shù)據(jù)類
│     ├──VideoItem.ets         	           // 視頻對(duì)象
│     └──VideoSpeed.ets                    // 播放速度類
└──entry/src/main/resource                 // 應(yīng)用靜態(tài)資源目錄

獲取視頻

視頻來(lái)源主要有本地視和網(wǎng)絡(luò)視頻兩種方式,效果如圖所示:

獲取本地視頻,通過(guò)resourceManager.getRawFd方法獲取rawfile文件夾中的視頻資源文件描述符,構(gòu)造本地視頻對(duì)象。

// HomeVideoListModel.ets
// 獲取本地視頻
async getLocalVideo() {
  this.videoLocalList = [];
  await this.assemblingVideoBean();
  GlobalContext.getContext().setObject('videoLocalList', this.videoLocalList);
  return this.videoLocalList;
}

// HomeVideoListModel.ets
// 組裝本地視頻對(duì)象
async assemblingVideoBean() {
  VIDEO_DATA.forEach(async (item: VideoItem) = > {
    let videoBean = await getContext().resourceManager.getRawFd(item.iSrc);
    let uri = videoBean;
    this.videoLocalList.push(new VideoItem(item.name, uri, ''));
  });
}

網(wǎng)絡(luò)視頻是通過(guò)手動(dòng)輸入地址,在有網(wǎng)的環(huán)境下點(diǎn)擊“鏈接校驗(yàn)”,通過(guò)地址獲取視頻時(shí)長(zhǎng),當(dāng)視頻時(shí)長(zhǎng)小于等于零時(shí)彈出“鏈接校驗(yàn)失敗”提示,否則彈出“鏈接校驗(yàn)成功”提示。

// HomeDialogModel.ets
// 設(shè)置網(wǎng)絡(luò)視頻路徑
async checkSrcValidity(checkFlag: number) {
  if (this.isLoading) {
    return;
  }
  this.isLoading = true;
  this.homeTabModel.linkCheck = $r('app.string.link_checking');
  this.homeTabModel.loadColor = $r('app.color.index_tab_unselected_font_color');
  this.checkFlag = checkFlag;
  this.createAvPlayer();
}

// 校驗(yàn)鏈接有效性
checkUrlValidity() {
  this.isLoading = false;
  this.homeTabModel.linkCheck = $r('app.string.link_check');
  this.homeTabModel.loadColor = $r('app.color.index_tab_selected_font_color');
  if (this.avPlayer !== null) {
    this.avPlayer.release();
  }
  if (this.duration === HomeConstants.DURATION_TWO) {
    // Failed to verify the link
    this.showPrompt($r('app.string.link_check_fail'));
  } else if (this.duration === HomeConstants.DURATION_ONE) {
    // The address is incorrect or no network is available
    this.showPrompt($r('app.string.link_check_address_internet'));
  } else {
    this.duration = 0;
    if (this.checkFlag === 0) {
      this.showPrompt($r('app.string.link_check_success'));
    } else {
      this.homeTabModel!.confirm();
      this.homeTabModel!.controller!.close();
    }
  }
}

視頻播放

視頻播放主要包括視頻的暫停、播放、切換、倍速播放、拖動(dòng)進(jìn)度條設(shè)置當(dāng)前進(jìn)度、顯示當(dāng)前播放時(shí)間、音量調(diào)節(jié)等功能,本章節(jié)主要針對(duì)播放管理類(下面簡(jiǎn)稱:AVPlayer)進(jìn)行講解,具體細(xì)節(jié)請(qǐng)參考gitee源碼,效果如圖所示:

播放的全流程包含:創(chuàng)建AVPlayer,設(shè)置播放資源,設(shè)置播放參數(shù)(音量/倍速),播放控制(播放/暫停/上一個(gè)視頻/下一個(gè)視頻),重置,銷毀資源。狀態(tài)機(jī)變化如圖所示:

視頻播放之前需要初始化XComponent組件用于展示視頻畫面。XComponent組件初始化成功之后在onLoad()中獲取surfaceID用于與AVPlayer實(shí)例關(guān)聯(lián)。

// PlayPlayer.ets
XComponent({
  ...
  controller: this.xComponentController
})
  .onLoad(async () = > {
    ...
    this.surfaceID = this.xComponentController.getXComponentSurfaceId();
    ...
  })
  ...

使用AVPlayer前需要通過(guò)createAVPlayer()構(gòu)建一個(gè)實(shí)例對(duì)象,并為AVPlayer實(shí)例綁定狀態(tài)機(jī),狀態(tài)機(jī)具體請(qǐng)參考[AVPlayerState]

// VideoController.ets
async createAVPlayer() {
  let avPlayer: media.AVPlayer = await media.createAVPlayer();
  this.avPlayer = avPlayer;
  this.bindState();
}

// VideoController.ets
async bindState() {
  if (this.avPlayer === null) {
    return;
  }
  this.avPlayer.on(Events.STATE_CHANGE, async (state: media.AVPlayerState) = > {
    let avplayerStatus: string = state;
    if (this.avPlayer === null) {
      return;
    }
    switch (avplayerStatus) {
      case AvplayerStatus.IDLE:
        ...
      case AvplayerStatus.INITIALIZED:
        ...
      case AvplayerStatus.PREPARED:
        ...
      case AvplayerStatus.PLAYING:
        ...
      case AvplayerStatus.PAUSED:
        ...
      case AvplayerStatus.COMPLETED:
        ...
      case AvplayerStatus.RELEASED:
        ...
      default:
        ...
    }
  });
  this.avPlayer.on(Events.TIME_UPDATE, (time: number) = > {
    this.initProgress(time);
  });
  this.avPlayer.on(Events.ERROR, () = > {
    this.playError();
  })
}

AVPlayer實(shí)例需設(shè)置播放路徑和XComponent中獲取的surfaceID,設(shè)置播放路徑之后AVPlayer狀態(tài)機(jī)變?yōu)閕nitialized狀態(tài),在此狀態(tài)下調(diào)用prepare(),進(jìn)入prepared狀態(tài)。

// VideoController.ets
async firstPlay(index: number, url: resourceManager.RawFileDescriptor, iUrl: string, surfaceId: string) {
  this.index = index;
  this.url = url;
  this.iUrl = iUrl;
  this.surfaceId = surfaceId;
  if (this.avPlayer === null) {
    await this.createAVPlayer();
  }
  if (this.avPlayer !== null) {
    if (this.iUrl) {
      this.avPlayer.url = this.iUrl;
    } else {
      this.avPlayer.fdSrc = this.url;
    }
  }
}

// VideoController.ets
async bindState() {
  ...
  this.avPlayer.on(Events.STATE_CHANGE, async (state: media.AVPlayerState) = > {
    let avplayerStatus: string = state;
    if (this.avPlayer === null) {
      return;
    }
    switch (avplayerStatus) {
      case AvplayerStatus.IDLE:
        ...
      case AvplayerStatus.INITIALIZED:
        this.avPlayer.surfaceId = this.surfaceId;
        this.avPlayer.prepare();
        break;
      ...
    }
  });
  ...
}

在prepared狀態(tài)下可獲取當(dāng)前播放路徑對(duì)應(yīng)視頻的總時(shí)長(zhǎng),并執(zhí)行play()進(jìn)行視頻播放。

// VideoController.ets
async bindState() {
  ...
  this.avPlayer.on(Events.STATE_CHANGE, async (state: media.AVPlayerState) = > {
    ...
    switch (avplayerStatus) {
      ...
      case AvplayerStatus.PREPARED:
        this.avPlayer.videoScaleType = 0;
        this.setVideoSize();
        this.avPlayer.play();
        this.duration = this.avPlayer.duration;
        break;
      ...
    }
  });
  ...
}

視頻播放后,變?yōu)閜laying狀態(tài),可通過(guò)“播放/暫停”按鈕切換播放狀態(tài),當(dāng)視頻暫停時(shí)狀態(tài)機(jī)變?yōu)閜aused狀態(tài)。

// VideoController.ets
switchPlayOrPause() {
  if (this.avPlayer === null) {
    return;
  }
  if (this.status === CommonConstants.STATUS_START) {
    this.avPlayer.pause();
  } else {
    this.avPlayer.play();
  }
}

// VideoController.ets
async bindState() {
  ...
  this.avPlayer.on(Events.STATE_CHANGE, async (state: media.AVPlayerState) = > {
    ...
    switch (avplayerStatus) {
      ...
      case AvplayerStatus.PLAYING:
        this.avPlayer.setVolume(this.playerModel.volume);
        this.setBright();
        this.status = CommonConstants.STATUS_START;
        this.watchStatus();
        break;
      ...
    }
  });
  ...
}

可拖動(dòng)進(jìn)度條設(shè)置視頻播放位置,也可滑動(dòng)音量調(diào)節(jié)區(qū)域設(shè)置視頻播放音量、設(shè)置播放速度。

// VideoController.ets
// 設(shè)置當(dāng)前播放位置
setSeekTime(value: number, mode: SliderChangeMode) {
  if (mode === Number(SliderMode.MOVING)) {
    this.playerModel.progressVal = value;
    this.playerModel.currentTime = DateFormatUtil.secondToTime(Math.floor(value * this.duration /
    CommonConstants.ONE_HUNDRED / CommonConstants.A_THOUSAND));
  }
  if (mode === Number(SliderMode.END) || mode === Number(SliderMode.CLICK)) {
    this.seekTime = value * this.duration / CommonConstants.ONE_HUNDRED;
    if (this.avPlayer !== null) {
      this.avPlayer.seek(this.seekTime, media.SeekMode.SEEK_PREV_SYNC);
    }
  }
}

// VideoController.ets
// 設(shè)置播放音量
onVolumeActionUpdate(event?: GestureEvent) {
  if (!event) {
    return;
  }
  if (this.avPlayer === null) {
    return;
  }
  if (CommonConstants.OPERATE_STATE.indexOf(this.avPlayer.state) === -1) {
    return;
  }
  if (this.playerModel.brightShow === false) {
    this.playerModel.volumeShow = true;
    let screenWidth = GlobalContext.getContext().getObject('screenWidth') as number;
    let changeVolume = (event.offsetX - this.positionX) / screenWidth;
    let volume: number = this.playerModel.volume;
    let currentVolume = volume + changeVolume;
    let volumeMinFlag = currentVolume <= PlayConstants.MIN_VALUE;
    let volumeMaxFlag = currentVolume > PlayConstants.MAX_VALUE;
    this.playerModel.volume = volumeMinFlag ? PlayConstants.MIN_VALUE :
      (volumeMaxFlag ? PlayConstants.MAX_VALUE : currentVolume);
    this.avPlayer.setVolume(this.playerModel.volume);
    this.positionX = event.offsetX;
  }
}

// VideoController.ets
// 設(shè)置播放速度
setSpeed(playSpeed: number) {
  if (this.avPlayer === null) {
    return;
  }
  if (CommonConstants.OPERATE_STATE.indexOf(this.avPlayer.state) === -1) {
    return;
  }
  this.playerModel.playSpeed = playSpeed;
  this.avPlayer.setSpeed(this.playerModel.playSpeed);
}

視頻播放完成之后,進(jìn)入completed狀態(tài),需調(diào)用reset()對(duì)視頻進(jìn)行重置,此時(shí)變?yōu)閕dle轉(zhuǎn)態(tài),在idle狀態(tài)下設(shè)置下一個(gè)視頻的播放地址,又會(huì)進(jìn)入initialized狀態(tài)。

// VideoController.ets 
sync bindState() {
  ...
  this.avPlayer.on(Events.STATE_CHANGE, async (state: media.AVPlayerState) = > {
    let avplayerStatus: string = state;
    ...
    switch (avplayerStatus) {
      case AvplayerStatus.IDLE:
        this.resetProgress();
        if (this.iUrl) {
          this.avPlayer.url = this.iUrl;
        } else {
          this.avPlayer.fdSrc = this.url;
        }
        break;
      case AvplayerStatus.INITIALIZED:
        this.avPlayer.surfaceId = this.surfaceId;
        this.avPlayer.prepare();
        break;
      ...
      case AvplayerStatus.COMPLETED:
        ...
        this.avPlayer.reset();
        break;
      ...
    }
  });
  ...
}

手勢(shì)控制

播放頁(yè)面通過(guò)綁定平移手勢(shì)(PanGesture),上下滑動(dòng)調(diào)節(jié)屏幕亮度,左右滑動(dòng)調(diào)節(jié)視頻音量,效果如圖所示:

// PlayPage.ets
Column() {
  ...
  Column()
    ...
    .gesture(
      PanGesture(this.panOptionBright)
        .onActionStart((event?: GestureEvent) = > {
          this.playVideoModel.onBrightActionStart(event);
        })
        .onActionUpdate((event?: GestureEvent) = > {
          this.playVideoModel.onBrightActionUpdate(event);
        })
        .onActionEnd(() = > {
          this.playVideoModel.onActionEnd();
        })
    )
  ...
  Column()
    ...
    .gesture(
      PanGesture(this.panOptionVolume)
        .onActionStart((event?: GestureEvent) = > {
          this.playVideoModel.onVolumeActionStart(event);
        })
        .onActionUpdate((event?: GestureEvent) = > {
          this.playVideoModel.onVolumeActionUpdate(event);
        })
        .onActionEnd(() = > {
          this.playVideoModel.onActionEnd();
        })
    )
  ...
}
...

本章節(jié)以音量調(diào)節(jié)介紹手勢(shì)控制,當(dāng)手指觸摸音量調(diào)節(jié)區(qū)域時(shí)獲取當(dāng)前屏幕坐標(biāo),滑動(dòng)手指實(shí)時(shí)獲取屏幕坐標(biāo)并計(jì)算音量。

// VideoController.ets
// 手指觸摸到音量調(diào)節(jié)區(qū)域
onVolumeActionStart(event?: GestureEvent) {
  if (!event) {
    return;
  }
  this.positionX = event.offsetX;
}

// 手指在音量調(diào)節(jié)區(qū)域水平滑動(dòng)
onVolumeActionUpdate(event?: GestureEvent) {
  if (!event) {
    return;
  }
  if (this.avPlayer === null) {
    return;
  }
  if (CommonConstants.OPERATE_STATE.indexOf(this.avPlayer.state) === -1) {
    return;
  }
  if (this.playerModel.brightShow === false) {
    this.playerModel.volumeShow = true;
    let screenWidth = GlobalContext.getContext().getObject('screenWidth') as number;
    let changeVolume = (event.offsetX - this.positionX) / screenWidth;
    let volume: number = this.playerModel.volume;
    let currentVolume = volume + changeVolume;
    let volumeMinFlag = currentVolume <= PlayConstants.MIN_VALUE;
    let volumeMaxFlag = currentVolume > PlayConstants.MAX_VALUE;
    this.playerModel.volume = volumeMinFlag ? PlayConstants.MIN_VALUE :
      (volumeMaxFlag ? PlayConstants.MAX_VALUE : currentVolume);
    this.avPlayer.setVolume(this.playerModel.volume);
    this.positionX = event.offsetX;
  }
}

審核編輯 黃宇

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 鴻蒙
    +關(guān)注

    關(guān)注

    57

    文章

    2321

    瀏覽量

    42749
  • HarmonyOS
    +關(guān)注

    關(guān)注

    79

    文章

    1967

    瀏覽量

    30035
  • OpenHarmony
    +關(guān)注

    關(guān)注

    25

    文章

    3665

    瀏覽量

    16161
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    在(Linux)ubuntu下通過(guò)GTK調(diào)用libvlc開發(fā)視頻播放器

    本項(xiàng)目實(shí)現(xiàn)了一個(gè)基于GTK和libvlc的視頻播放器。使用GTK創(chuàng)建GUI界面,使用libvlc播放視頻。用戶可以通過(guò)選擇視頻文件,然后啟動(dòng)
    的頭像 發(fā)表于 06-01 15:42 ?2034次閱讀
    在(Linux)ubuntu下通過(guò)GTK調(diào)用libvlc<b class='flag-5'>開發(fā)</b><b class='flag-5'>視頻</b><b class='flag-5'>播放器</b>

    鴻蒙開發(fā)-視頻播放器方案

    HarmonyOS系統(tǒng)中,提供兩種視頻播放開發(fā)的方案: [AVPlayer]:功能較完善的音視頻播放ArkTS/JS API,集成了流媒體
    發(fā)表于 02-19 17:20

    靈活電影播放器視頻/音頻播放軟件

    東京,2008年4月24日--瑞薩科技公司(Renesas Techno logy Corp.)今天宣布,推出為采用移動(dòng)電話SH-Mobile*1應(yīng)用處理的嵌入式系統(tǒng)開發(fā)的靈活電影播放器
    發(fā)表于 03-06 19:52

    HarmonyOS應(yīng)用開發(fā)-視頻播放

    HarmonyOS應(yīng)用開發(fā)E2E體驗(yàn),學(xué)到了:如何創(chuàng)建一個(gè)HarmonyOS Demo Project如何構(gòu)建一個(gè)Hap并且將其部署到真機(jī)在HarmonyOS上如何使用
    發(fā)表于 09-11 17:25

    HarmonyOS IoT 硬件開發(fā)案例分享

    ``許思維老師HiSpark Wi-Fi IoT 開發(fā)案例分享:案例一:AHT20溫濕度傳感開發(fā)、調(diào)試;案例二:oled屏驅(qū)動(dòng)庫(kù)移植,調(diào)試;案例三:用OLED屏播放
    發(fā)表于 10-27 17:30

    基于HarmonyOS ets開發(fā)的簡(jiǎn)易視頻播放器

    這是我們使用HarmonyOS的codelab簡(jiǎn)易視頻播放器的codelab簡(jiǎn)易播放器。內(nèi)容就是一個(gè)主視頻界面,包括頂部的
    發(fā)表于 04-18 10:41

    網(wǎng)頁(yè)視頻播放器代碼

    網(wǎng)頁(yè)視頻播放器代碼
    發(fā)表于 01-10 11:23 ?102次下載
    網(wǎng)頁(yè)<b class='flag-5'>視頻</b><b class='flag-5'>播放器</b>代碼

    flv視頻播放器代碼

    flv視頻播放器代碼 FlV視頻播放器代碼 代碼如下這里只是介紹幾個(gè)例子,現(xiàn)在把代碼公布一下,大家可以參考著做,也可以把你喜歡的視頻連接
    發(fā)表于 01-10 12:36 ?2054次閱讀

    MP4播放器視頻播放格式有哪些?

    MP4播放器視頻播放格式有哪些?        
    發(fā)表于 12-21 15:51 ?2.4w次閱讀

    LXE播放器

    lxe視頻播放器軟件是免費(fèi)軟件,可以完全免費(fèi)使用、可以自由傳播,exe視頻播放器用于播放屏幕錄像專家錄制的LXE和EXE格式的錄像文件,安裝
    發(fā)表于 11-23 16:14 ?0次下載

    關(guān)于VR電影視頻播放器 盤點(diǎn)12款VR播放器

    VR電影和視頻那個(gè)播放器好,哪些播放器更為實(shí)用,由于視頻資源格式多樣,在一個(gè)播放器播放不了的
    發(fā)表于 06-27 15:50 ?12.6w次閱讀

    數(shù)碼播放器開發(fā)案

    數(shù)碼播放器開發(fā)案例說(shuō)明。
    發(fā)表于 05-19 11:07 ?6次下載

    HarmonyOS開發(fā)案例:【視頻播放器

    基于video、swiper和slider組件,實(shí)現(xiàn)簡(jiǎn)單的視頻播放器,可支持海報(bào)輪播、視頻播放等功能。
    的頭像 發(fā)表于 04-22 21:06 ?424次閱讀
    <b class='flag-5'>HarmonyOS</b><b class='flag-5'>開發(fā)案</b>例:【<b class='flag-5'>視頻</b><b class='flag-5'>播放器</b>】

    HarmonyOS開發(fā)案例:【音樂播放器

    使用ArkTS語(yǔ)言實(shí)現(xiàn)了一個(gè)簡(jiǎn)易的音樂播放器應(yīng)用
    的頭像 發(fā)表于 04-23 15:44 ?939次閱讀
    <b class='flag-5'>HarmonyOS</b><b class='flag-5'>開發(fā)案</b>例:【音樂<b class='flag-5'>播放器</b>】

    HarmonyOS開發(fā)案例:【視頻播放器

    使用ArkTS語(yǔ)言實(shí)現(xiàn)視頻播放器,主要包括主界面和視頻播放界面,
    的頭像 發(fā)表于 04-23 17:25 ?637次閱讀
    <b class='flag-5'>HarmonyOS</b><b class='flag-5'>開發(fā)案</b>例:【<b class='flag-5'>視頻</b><b class='flag-5'>播放器</b>】