HarmonyOS元服务开发系列教程(三):实现音乐播放和封面旋转

首先诚邀大家参加学习鸿蒙拿好礼活动,即日起,只要加入班级考取华为开发者基础/高级证书,并发表一篇技术文章,就有机会获得官方发放的精美礼品,数量有限,先到先得。幽蓝君的班级链接如下:​​https://developer.huawei.com/consumer/cn/training/classDetail/7b706ca975bd42e98b3bb51aa6b0be5a?type=1?ha_source=hmosclass&ha_sourceId=89000248​

上一篇文章中分享了如何开发音乐元服务的页面,今天继续完善应用功能,完成音乐的播放和控制,以及音乐列表弹窗等功能。

本项目使用本地音频,音频文件和封面都存放在rawfile文件夹。

由于本项目使用V2版本的状态管理,定义变量的装饰器也发生了变化,比如定使用@Local定义歌曲列表和当前播放序号:

复制代码
private avPlayer?: media.AVPlayer = undefined;
@Local musicList:MusicClass[] = [
  new MusicClass('Never Really Easy','Stephanie Poetri','NeverReallyEasy.mp3','NeverReallyEasy.png'),
  new MusicClass('Only Lovers','Ronan Keating','onlylovers.mp3','onlylovers.png'),
  new MusicClass('Bumping Up and Down','Raffi','BumpingUpandDown.mp3','BumpingUpandDown.png')
]
@Local currentIndex:number = 0
然后读取本地文件并初始化播放器:
// 创建avPlayer实例对象
let avPlayer: media.AVPlayer = await media.createAVPlayer();
// 创建状态机变化回调函数
this.setAVPlayerCallback(avPlayer);
// 通过UIAbilityContext的resourceManager成员的getRawFd接口获取媒体资源播放地址
// 返回类型为{fd,offset,length},fd为HAP包fd地址,offset为媒体资源偏移量,length为播放长度
let context = getContext(this) as common.UIAbilityContext;
let fileDescriptor = await context.resourceManager.getRawFd(this.musicList[this.currentIndex].url);
let avFileDescriptor: media.AVFileDescriptor =
  { fd: fileDescriptor.fd, offset: fileDescriptor.offset, length: fileDescriptor.length };
this.isSeek = false; // 支持seek操作
// 为fdSrc赋值触发initialized状态机上报
avPlayer.fdSrc = avFileDescriptor;
this.avPlayer = avPlayer

Local大家可以理解为V1版本的State,区别是Local是组件内部的状态管理,不允许在外部进行初始化,并且Local能够实现深度状态检测。更多的V2版本装饰器会在项目中陆续介绍。

接下来注册播放器的回调,里面包含播放、暂停、完成等状态回调:

复制代码
avPlayer.on('timeUpdate', (seekDoneTime: number) => {

  if(this.duration > 0){
    let progress = seekDoneTime/this.duration
    this.progressNow = progress*100
  }
  this.progressTimeString = this.durationToTimeString(seekDoneTime)
})
// seek操作结果回调函数
avPlayer.on('seekDone', (seekDoneTime: number) => {
  console.info(`AVPlayer seek succeeded, seek time is ${seekDoneTime}`);
})

// 状态机变化回调函数
avPlayer.on('stateChange', async (state: string, reason: media.StateChangeReason) => {
  switch (state) {
    case 'idle': // 成功调用reset接口后触发该状态机上报
      console.info('AVPlayer state idle called.');
      avPlayer.release(); // 调用release接口销毁实例对象
      break;
    case 'initialized': // avplayer 设置播放源后触发该状态上报
      console.info('AVPlayer state initialized called.');
      avPlayer.prepare();
      break;
    case 'prepared': // prepare调用成功后上报该状态机
      console.info('AVPlayer state prepared called.');
      if(this.isPlay){
        avPlayer.play();
      }
      this.duration = avPlayer.duration
      this.durationTimeString = this.durationToTimeString(this.duration)
      console.log('duration:',avPlayer.duration)

      break;
    case 'playing': // play成功调用后触发该状态机上报
      console.info('AVPlayer state playing called.');
      if (this.count !== 0) {
        if (this.isSeek) {
          console.info('AVPlayer start to seek.');
          avPlayer.seek(avPlayer.duration); //seek到音频末尾
        } else {
          // 当播放模式不支持seek操作时继续播放到结尾
          console.info('AVPlayer wait to play end.');
        }
      } else {
        // avPlayer.pause(); // 调用暂停接口暂停播放
      }
      this.count++;
      break;
    case 'paused': // pause成功调用后触发该状态机上报
      console.info('AVPlayer state paused called.');
      // avPlayer.play(); // 再次播放接口开始播放
      break;
    case 'completed': // 播放结束后触发该状态机上报
      console.info('AVPlayer state completed called.');
      // avPlayer.stop(); //调用播放结束接口
      // this.endRotate();
      if(this.currentIndex < this.musicList.length - 1){
        this.currentIndex += 1
        this.changeSong()
      }else {
        this.endRotate();
      }
      break;
    case 'stopped': // stop接口成功调用后触发该状态机上报
      console.info('AVPlayer state stopped called.');
      // avPlayer.reset(); // 调用reset接口初始化avplayer状态
      break;
    case 'released':
      console.info('AVPlayer state released called.');
      break;
    default:
      console.info('AVPlayer state unknown called.');
      break;
  }
})

然后可以调用播放器的play方法进行音乐播放:

复制代码
this.avPlayer.play()

音乐播放的同时封面图片也随之转动,这里使用计时器不断修改图片角度来实现,首先定义计时器和角度变量:

复制代码
private timer?: number;
@Local rotateAngle:number=0;

实现计时器并修改角度:

复制代码
this.timer = setInterval(() => {
// 保留2位小数
this.rotateAngle = this.rotateAngle + 0.005

}, 30);

今天的内容就是这样,感谢阅读。

相关推荐
liulian091638 分钟前
Flutter for OpenHarmony 跨平台开发:BMI计算器功能实战指南
flutter·华为
xmdy58664 小时前
Flutter+开源鸿蒙实战|智安盾电商溯源平台Day1 项目搭建与整体方案拆解
flutter·开源·harmonyos
nashane5 小时前
HarmonyOS 6学习:应用签名文件丢失处理与更新完全指南
学习·华为·harmonyos·harmonyos 5
笔触狂放6 小时前
【项目】基于ArkTS的老年人智能应用开发(1)
harmonyos·arkts·鸿蒙
24白菜头8 小时前
【无标题】
c++·笔记·学习·harmonyos
LeesonWong9 小时前
Neo 构建鸿蒙应用【二】:技术路线全解
harmonyos
LeesonWong9 小时前
Neo 构建鸿蒙应用【三】:实战社交应用与工程感悟
harmonyos
xmdy586610 小时前
Flutter+开源鸿蒙实战|智联邻里Day6 引入GetX全局架构+升级版下拉刷新+Toast弹窗+网络状态监听
flutter·开源·harmonyos
斯班奇的好朋友阿法法10 小时前
鸿蒙 vs iOS vs 微信小程序:开发平台全面对比
ios·微信小程序·harmonyos
xmdy586610 小时前
Flutter+开源鸿蒙实战|智联邻里Day5 闲置详情页+删除功能+下拉刷新+交互优化
flutter·开源·harmonyos