一、概述

在移动应用开发中,音乐播放器的进度控制是核心交互功能之一。本文将以HarmonyOS ArkUI的Slider组件为基础,详细讲解如何实现一个功能完备的音乐播放器进度条组件,涵盖时间同步、播放控制、手势交互等关键技术点。

二、实现效果

  • 实时显示播放进度(MM:SS格式)
  • 支持拖动滑块进行进度跳转
  • 自动时间更新与手动控制同步
  • 播放/暂停/停止功能集成
  • 自定义进度条样式
  • 效果图

 title=  title=

三、代码实现

3.1 状态管理

@State currentTime: number = 0;   // 当前播放时间(秒)
@State totalTime: number = 238;   // 总时长(秒)
@State isPlaying: boolean = false; // 播放状态
private timerId: number = 0;       // 计时器ID

使用ArkUI的@State装饰器实现响应式数据绑定,当这些状态变化时自动触发UI更新。

3.2 Slider组件配置

Slider({
        value: this.currentTime,
        min: 0,
        max: this.totalTime,
        step: 1,
        style: SliderStyle.OutSet // 进度条样式
      })
        .blockColor(Color.Blue) // 滑块颜色
        .trackColor(Color.Gray) // 背景轨道颜色
        .selectedColor(Color.Blue) // 进度颜色
        .showSteps(false) // 隐藏步进点
        .onChange((value: number) => { // 拖动事件
          this.onSliderChange(value);
        })
        .width('90%')

关键参数说明:

  • value:双向绑定当前播放时间
  • min/max:设置时间范围
  • step:1:按秒级精度控制
  • style:使用突出样式增强可视性

3.3 时间同步机制

// 定时器更新逻辑
this.timerId = setInterval(() => {
  if (this.currentTime < this.totalTime) {
    this.currentTime += 1;
  } else {
    clearInterval(this.timerId);
    this.isPlaying = false;
  }
}, 1000);

// 格式化显示
private formatTime(seconds: number): string {
  const mins = Math.floor(seconds / 60);
  const secs = seconds % 60;
  return `${mins.toString().padStart(2, '0')}:${secs.toString().padStart(2, '0')}`;
}

实现秒到分钟转换,使用padStart保证两位数显示格式。

3.4 交互事件处理

// 播放/暂停控制
  private togglePlay() {
    if (this.isPlaying) {
      // 暂停逻辑
      clearInterval(this.timerId);
    } else {
      // 播放逻辑
      this.timerId = setInterval(() => {
        if (this.currentTime < this.totalTime) {
          this.currentTime += 1;
        } else {
          clearInterval(this.timerId);
          this.isPlaying = false;
        }
      }, 1000);
    }
    this.isPlaying = !this.isPlaying;
  }

  // Slider拖动事件处理
  private onSliderChange(value: number) {
    this.currentTime = value;
    if (this.isPlaying) {
      // 如果正在播放,重启计时器保持同步
      clearInterval(this.timerId);
      this.timerId = setInterval(() => {
        if (this.currentTime < this.totalTime) {
          this.currentTime += 1;
        } else {
          clearInterval(this.timerId);
          this.isPlaying = false;
        }
      }, 1000);
    }
  }

处理三种交互场景:

  1. 自动播放时定时更新
  2. 手动拖动时即时跳转
  3. 状态切换时资源管理

3.5 完整代码

@Entry
@Component
struct MusicPlayer {
  // 状态管理
  @State currentTime: number = 0; // 当前播放时间(秒)
  @State totalTime: number = 238; // 总时长(秒)
  @State isPlaying: boolean = false; // 播放状态
  private timerId: number = 0; // 计时器ID

  // 格式化时间显示(秒转MM:SS)
  private formatTime(seconds: number): string {
    const mins = Math.floor(seconds / 60);
    const secs = Math.floor(seconds % 60);
    return `${mins.toString().padStart(2, '0')}:${secs.toString().padStart(2, '0')}`;
  }

  // 播放/暂停控制
  private togglePlay() {
    if (this.isPlaying) {
      // 暂停逻辑
      clearInterval(this.timerId);
    } else {
      // 播放逻辑
      this.timerId = setInterval(() => {
        if (this.currentTime < this.totalTime) {
          this.currentTime += 1;
        } else {
          clearInterval(this.timerId);
          this.isPlaying = false;
        }
      }, 1000);
    }
    this.isPlaying = !this.isPlaying;
  }

  // Slider拖动事件处理
  private onSliderChange(value: number) {
    this.currentTime = value;
    if (this.isPlaying) {
      // 如果正在播放,重启计时器保持同步
      clearInterval(this.timerId);
      this.timerId = setInterval(() => {
        if (this.currentTime < this.totalTime) {
          this.currentTime += 1;
        } else {
          clearInterval(this.timerId);
          this.isPlaying = false;
        }
      }, 1000);
    }
  }

  build() {
    Column() {
      // 歌曲信息
      Text('晴天 - 周杰伦')
        .fontSize(20)
        .margin({ bottom: 20 });

      // 进度条
      Slider({
        value: this.currentTime,
        min: 0,
        max: this.totalTime,
        step: 1,
        style: SliderStyle.OutSet // 进度条样式
      })
        .blockColor(Color.Blue) // 滑块颜色
        .trackColor(Color.Gray) // 背景轨道颜色
        .selectedColor(Color.Blue) // 进度颜色
        .showSteps(false) // 隐藏步进点
        .onChange((value: number) => { // 拖动事件
          this.onSliderChange(value);
        })
        .width('90%')

      // 时间显示
      Row() {
        Text(this.formatTime(this.currentTime))
          .fontColor(Color.Black)

        Blank() // 空白填充

        Text(this.formatTime(this.totalTime))
          .fontColor(Color.Gray)
      }
      .width('90%')
      .margin({ top: 8 })

      // 控制按钮
      Row() {
        Button(this.isPlaying ? '⏸️' : '▶️') // 播放/暂停按钮
          .onClick(() => this.togglePlay())
          .backgroundColor(Color.Transparent)
          .fontSize(32)

        Button('⏹️') // 停止按钮
          .onClick(() => {
            this.currentTime = 0;
            clearInterval(this.timerId);
            this.isPlaying = false;
          })
          .backgroundColor(Color.Transparent)
          .fontSize(32)
          .margin({ left: 20 })
      }
      .margin({ top: 20 })
    }
    .width('100%')
    .height('100%')
    .justifyContent(FlexAlign.Center)
    .backgroundColor(Color.White)
  }
}

饭特稀
1 声望0 粉丝