项目已开源,开源地址: https://gitcode.com/nutpi/HarmonyosNextCaseStudyTutorial , 欢迎fork & star
在现代音乐播放器应用中,侧边栏是展示播放列表和歌单的重要界面元素。通过HarmonyOS NEXT的SideBarContainer
组件,我们可以轻松实现一个功能完善的音乐播放器侧边栏,为用户提供流畅的音乐浏览和播放体验。
SideBarContainer
是HarmonyOS NEXT提供的一个容器组件,专门用于创建带有侧边栏的布局。它支持三种显示模式:
在本案例中,我们使用Embed模式来实现音乐播放器的侧边栏,这样用户可以同时看到播放列表和当前播放的歌曲信息。
一个功能完善的音乐播放器侧边栏通常需要满足以下需求:
接下来,我们将通过代码实现这些功能,并详细讲解每个部分的实现原理。
首先,让我们看一下音乐播放器侧边栏的基本结构:
@Entry
@Component
struct MusicPlayer {
@State isSideBarShow: boolean = true
@State currentSongIndex: number = 0
@State isPlaying: boolean = false
@State currentTime: number = 0
@State totalTime: number = 280
@State currentPlaylist: string = '最近播放'
build() {
SideBarContainer(SideBarContainerType.Embed) {
// 侧边栏内容 - 播放列表和歌单
this.SideBarContent()
// 主内容区 - 当前播放歌曲信息和控制
this.MainContent()
}
.controlButton({
icons: {
shown: $r('app.media.ic_sidebar_shown'),
hidden: $r('app.media.ic_sidebar_hidden'),
}
})
.sideBarWidth(300)
.minSideBarWidth(150)
.maxSideBarWidth(500)
.onChange((isShow: boolean) => {
this.isSideBarShow = isShow
})
}
// 侧边栏内容构建器
@Builder SideBarContent() {
// 侧边栏内容实现...
}
// 主内容区构建器
@Builder MainContent() {
// 主内容区实现...
}
}
这个基本结构包含了:
SideBarContainer
的Embed
模式创建侧边栏布局@Builder
装饰器定义了侧边栏内容和主内容区的构建器接下来,我们实现侧边栏内容,包括播放列表和歌单:
@Builder SideBarContent() {
Column() {
// 标题
Text('音乐库')
.fontSize(24)
.fontWeight(FontWeight.Bold)
.margin({ top: 20, bottom: 20, left: 16 })
// 歌单列表
List() {
// 歌单分类标题
ListItem() {
Text('我的歌单')
.fontSize(16)
.fontWeight(FontWeight.Medium)
.margin({ left: 16, top: 10, bottom: 10 })
}
// 歌单列表项
ForEach(this.playlists, (playlist: PlaylistInfo) => {
ListItem() {
Row() {
Image(playlist.cover)
.width(40)
.height(40)
.borderRadius(8)
.margin({ right: 12 })
Column() {
Text(playlist.name)
.fontSize(16)
.fontColor(this.currentPlaylist === playlist.name ? '#007DFF' : '#000000')
Text(`${playlist.count}首`)
.fontSize(12)
.fontColor('#888888')
.margin({ top: 4 })
}
.alignItems(HorizontalAlign.Start)
.layoutWeight(1)
}
.width('100%')
.padding({ left: 16, right: 16, top: 8, bottom: 8 })
.borderRadius(8)
.backgroundColor(this.currentPlaylist === playlist.name ? '#F0F0F0' : '#FFFFFF')
}
.onClick(() => {
this.currentPlaylist = playlist.name
// 在实际应用中,这里会加载对应歌单的歌曲
})
})
// 当前播放列表标题
ListItem() {
Text('当前播放')
.fontSize(16)
.fontWeight(FontWeight.Medium)
.margin({ left: 16, top: 20, bottom: 10 })
}
// 当前播放列表歌曲
ForEach(this.songs, (song: SongInfo, index: number) => {
ListItem() {
Row() {
Text(`${index + 1}`)
.fontSize(16)
.fontColor('#888888')
.width(30)
Column() {
Text(song.title)
.fontSize(16)
.fontColor(this.currentSongIndex === index ? '#007DFF' : '#000000')
.maxLines(1)
.textOverflow({ overflow: TextOverflow.Ellipsis })
Text(song.artist)
.fontSize(12)
.fontColor('#888888')
.margin({ top: 4 })
.maxLines(1)
.textOverflow({ overflow: TextOverflow.Ellipsis })
}
.alignItems(HorizontalAlign.Start)
.layoutWeight(1)
Text(this.formatDuration(song.duration))
.fontSize(14)
.fontColor('#888888')
}
.width('100%')
.padding({ left: 16, right: 16, top: 8, bottom: 8 })
.borderRadius(8)
.backgroundColor(this.currentSongIndex === index ? '#F0F0F0' : '#FFFFFF')
}
.onClick(() => {
this.currentSongIndex = index
this.currentTime = 0
this.totalTime = song.duration
this.isPlaying = true
})
})
}
.layoutWeight(1)
}
.width('100%')
.height('100%')
}
在这个侧边栏内容中,我们实现了:
接下来,我们实现主内容区,显示当前播放歌曲的信息和控制:
@Builder MainContent() {
Column() {
// 当前播放歌曲信息
Column() {
// 歌曲封面
Image(this.songs[this.currentSongIndex].cover)
.width('60%')
.aspectRatio(1)
.borderRadius(12)
.margin({ top: 40 })
// 歌曲标题和艺术家
Text(this.songs[this.currentSongIndex].title)
.fontSize(24)
.fontWeight(FontWeight.Bold)
.margin({ top: 20 })
Text(this.songs[this.currentSongIndex].artist)
.fontSize(16)
.fontColor('#888888')
.margin({ top: 8 })
// 播放进度条
Column() {
Slider({
value: this.currentTime,
min: 0,
max: this.totalTime,
step: 1,
style: SliderStyle.OutSet
})
.width('90%')
.onChange((value: number) => {
this.currentTime = value
})
// 时间显示
Row() {
Text(this.formatDuration(this.currentTime))
.fontSize(12)
.fontColor('#888888')
Text(this.formatDuration(this.totalTime))
.fontSize(12)
.fontColor('#888888')
}
.width('90%')
.justifyContent(FlexAlign.SpaceBetween)
.margin({ top: 8 })
}
.margin({ top: 40 })
// 播放控制按钮
Row() {
// 上一首按钮
Image($r('app.media.ic_previous'))
.width(40)
.height(40)
.onClick(() => {
this.playPrevious()
})
// 播放/暂停按钮
Image(this.isPlaying ? $r('app.media.ic_pause') : $r('app.media.ic_play'))
.width(60)
.height(60)
.margin({ left: 40, right: 40 })
.onClick(() => {
this.isPlaying = !this.isPlaying
})
// 下一首按钮
Image($r('app.media.ic_next'))
.width(40)
.height(40)
.onClick(() => {
this.playNext()
})
}
.margin({ top: 40 })
}
.width('100%')
.layoutWeight(1)
.justifyContent(FlexAlign.Start)
.alignItems(HorizontalAlign.Center)
}
.width('100%')
.height('100%')
.backgroundColor('#FFFFFF')
}
在主内容区中,我们实现了:
Slider
组件显示播放进度,并允许用户调整进度为了支持上述功能,我们需要定义一些辅助方法和数据模型:
// 歌单信息数据模型
interface PlaylistInfo {
name: string
cover: Resource
count: number
}
// 歌曲信息数据模型
interface SongInfo {
title: string
artist: string
cover: Resource
duration: number
}
@Entry
@Component
struct MusicPlayer {
// 状态变量...
// 歌单数据
private playlists: PlaylistInfo[] = [
{ name: '最近播放', cover: $r('app.media.playlist1'), count: 28 },
{ name: '我喜欢的', cover: $r('app.media.playlist2'), count: 45 },
{ name: '流行音乐', cover: $r('app.media.playlist3'), count: 32 },
{ name: '轻音乐', cover: $r('app.media.playlist4'), count: 18 },
{ name: '摇滚', cover: $r('app.media.playlist5'), count: 24 }
]
// 歌曲数据
private songs: SongInfo[] = [
{ title: '夜曲', artist: '周杰伦', cover: $r('app.media.song1'), duration: 280 },
{ title: '起风了', artist: '买辣椒也用券', cover: $r('app.media.song2'), duration: 312 },
{ title: '海阔天空', artist: 'Beyond', cover: $r('app.media.song3'), duration: 326 },
{ title: '稻香', artist: '周杰伦', cover: $r('app.media.song4'), duration: 234 },
{ title: '光辉岁月', artist: 'Beyond', cover: $r('app.media.song5'), duration: 302 },
{ title: '晴天', artist: '周杰伦', cover: $r('app.media.song6'), duration: 269 },
{ title: '平凡之路', artist: '朴树', cover: $r('app.media.song7'), duration: 293 },
{ title: '青花瓷', artist: '周杰伦', cover: $r('app.media.song8'), duration: 241 }
]
// 格式化时长
private formatDuration(seconds: number): string {
const minutes = Math.floor(seconds / 60)
const remainingSeconds = Math.floor(seconds % 60)
return `${minutes}:${remainingSeconds.toString().padStart(2, '0')}`
}
// 播放上一首
private playPrevious() {
this.currentSongIndex = (this.currentSongIndex - 1 + this.songs.length) % this.songs.length
this.currentTime = 0
this.totalTime = this.songs[this.currentSongIndex].duration
this.isPlaying = true
}
// 播放下一首
private playNext() {
this.currentSongIndex = (this.currentSongIndex + 1) % this.songs.length
this.currentTime = 0
this.totalTime = this.songs[this.currentSongIndex].duration
this.isPlaying = true
}
// 构建器...
}
在这部分代码中,我们定义了:
PlaylistInfo
和SongInfo
接口,用于描述歌单和歌曲的数据结构formatDuration
:将秒数格式化为分:秒格式playPrevious
:播放上一首歌曲playNext
:播放下一首歌曲SideBarContainer(SideBarContainerType.Embed) {
// 侧边栏内容
this.SideBarContent()
// 主内容区
this.MainContent()
}
.controlButton({
icons: {
shown: $r('app.media.ic_sidebar_shown'),
hidden: $r('app.media.ic_sidebar_hidden'),
}
})
.sideBarWidth(300)
.minSideBarWidth(150)
.maxSideBarWidth(500)
.onChange((isShow: boolean) => {
this.isSideBarShow = isShow
})
这段代码配置了SideBarContainer
组件:
SideBarContainerType.Embed
,将侧边栏嵌入在主内容区内controlButton
属性设置显示和隐藏状态的图标sideBarWidth
:设置默认宽度为300像素minSideBarWidth
:设置最小宽度为150像素maxSideBarWidth
:设置最大宽度为500像素onChange
回调函数监听侧边栏显示状态的变化侧边栏内容主要包括两部分:歌单列表和当前播放列表。
// 歌单列表
List() {
// 歌单分类标题
ListItem() {
Text('我的歌单')
.fontSize(16)
.fontWeight(FontWeight.Medium)
.margin({ left: 16, top: 10, bottom: 10 })
}
// 歌单列表项
ForEach(this.playlists, (playlist: PlaylistInfo) => {
ListItem() {
Row() {
Image(playlist.cover)
.width(40)
.height(40)
.borderRadius(8)
.margin({ right: 12 })
Column() {
Text(playlist.name)
.fontSize(16)
.fontColor(this.currentPlaylist === playlist.name ? '#007DFF' : '#000000')
Text(`${playlist.count}首`)
.fontSize(12)
.fontColor('#888888')
.margin({ top: 4 })
}
.alignItems(HorizontalAlign.Start)
.layoutWeight(1)
}
.width('100%')
.padding({ left: 16, right: 16, top: 8, bottom: 8 })
.borderRadius(8)
.backgroundColor(this.currentPlaylist === playlist.name ? '#F0F0F0' : '#FFFFFF')
}
.onClick(() => {
this.currentPlaylist = playlist.name
// 在实际应用中,这里会加载对应歌单的歌曲
})
})
// ...
}
这段代码实现了歌单列表:
List
和ListItem
组件创建列表ForEach
遍历playlists
数组,为每个歌单创建一个列表项currentPlaylist
和playlist.name
,为当前选中的歌单应用不同的样式currentPlaylist
状态变量// 当前播放列表标题
ListItem() {
Text('当前播放')
.fontSize(16)
.fontWeight(FontWeight.Medium)
.margin({ left: 16, top: 20, bottom: 10 })
}
// 当前播放列表歌曲
ForEach(this.songs, (song: SongInfo, index: number) => {
ListItem() {
Row() {
Text(`${index + 1}`)
.fontSize(16)
.fontColor('#888888')
.width(30)
Column() {
Text(song.title)
.fontSize(16)
.fontColor(this.currentSongIndex === index ? '#007DFF' : '#000000')
.maxLines(1)
.textOverflow({ overflow: TextOverflow.Ellipsis })
Text(song.artist)
.fontSize(12)
.fontColor('#888888')
.margin({ top: 4 })
.maxLines(1)
.textOverflow({ overflow: TextOverflow.Ellipsis })
}
.alignItems(HorizontalAlign.Start)
.layoutWeight(1)
Text(this.formatDuration(song.duration))
.fontSize(14)
.fontColor('#888888')
}
.width('100%')
.padding({ left: 16, right: 16, top: 8, bottom: 8 })
.borderRadius(8)
.backgroundColor(this.currentSongIndex === index ? '#F0F0F0' : '#FFFFFF')
}
.onClick(() => {
this.currentSongIndex = index
this.currentTime = 0
this.totalTime = song.duration
this.isPlaying = true
})
})
这段代码实现了当前播放列表:
ForEach
遍历songs
数组,为每首歌曲创建一个列表项currentSongIndex
和index
,为当前播放的歌曲应用不同的样式currentSongIndex
、currentTime
、totalTime
和isPlaying
状态变量主内容区主要显示当前播放歌曲的信息和控制。
// 歌曲封面
Image(this.songs[this.currentSongIndex].cover)
.width('60%')
.aspectRatio(1)
.borderRadius(12)
.margin({ top: 40 })
// 歌曲标题和艺术家
Text(this.songs[this.currentSongIndex].title)
.fontSize(24)
.fontWeight(FontWeight.Bold)
.margin({ top: 20 })
Text(this.songs[this.currentSongIndex].artist)
.fontSize(16)
.fontColor('#888888')
.margin({ top: 8 })
这段代码显示了当前播放歌曲的信息:
Image
组件显示歌曲封面,设置宽度为父容器的60%,并保持1:1的宽高比Text
组件显示歌曲标题,设置较大的字体大小和粗体Text
组件显示艺术家名称,设置较小的字体大小和灰色字体// 播放进度条
Column() {
Slider({
value: this.currentTime,
min: 0,
max: this.totalTime,
step: 1,
style: SliderStyle.OutSet
})
.width('90%')
.onChange((value: number) => {
this.currentTime = value
})
// 时间显示
Row() {
Text(this.formatDuration(this.currentTime))
.fontSize(12)
.fontColor('#888888')
Text(this.formatDuration(this.totalTime))
.fontSize(12)
.fontColor('#888888')
}
.width('90%')
.justifyContent(FlexAlign.SpaceBetween)
.margin({ top: 8 })
}
.margin({ top: 40 })
这段代码实现了播放进度条:
Slider
组件显示播放进度,设置最小值为0,最大值为歌曲总时长onChange
回调函数,在用户调整进度条时更新currentTime
状态变量formatDuration
方法格式化时间// 播放控制按钮
Row() {
// 上一首按钮
Image($r('app.media.ic_previous'))
.width(40)
.height(40)
.onClick(() => {
this.playPrevious()
})
// 播放/暂停按钮
Image(this.isPlaying ? $r('app.media.ic_pause') : $r('app.media.ic_play'))
.width(60)
.height(60)
.margin({ left: 40, right: 40 })
.onClick(() => {
this.isPlaying = !this.isPlaying
})
// 下一首按钮
Image($r('app.media.ic_next'))
.width(40)
.height(40)
.onClick(() => {
this.playNext()
})
}
.margin({ top: 40 })
这段代码实现了播放控制按钮:
Image
组件显示上一首、播放/暂停和下一首按钮isPlaying
状态变量显示不同的图标playPrevious
方法isPlaying
状态变量playNext
方法// 格式化时长
private formatDuration(seconds: number): string {
const minutes = Math.floor(seconds / 60)
const remainingSeconds = Math.floor(seconds % 60)
return `${minutes}:${remainingSeconds.toString().padStart(2, '0')}`
}
这个方法将秒数格式化为分:秒格式:
Math.floor(seconds / 60)
计算分钟数Math.floor(seconds % 60)
计算剩余的秒数padStart
方法确保秒数始终显示为两位数// 播放上一首
private playPrevious() {
this.currentSongIndex = (this.currentSongIndex - 1 + this.songs.length) % this.songs.length
this.currentTime = 0
this.totalTime = this.songs[this.currentSongIndex].duration
this.isPlaying = true
}
// 播放下一首
private playNext() {
this.currentSongIndex = (this.currentSongIndex + 1) % this.songs.length
this.currentTime = 0
this.totalTime = this.songs[this.currentSongIndex].duration
this.isPlaying = true
}
这两个方法实现了播放上一首和下一首的功能:
currentTime
为0,表示从头开始播放totalTime
为新歌曲的时长isPlaying
为true,表示开始播放在音乐播放器应用中,侧边栏宽度的设置非常重要,它影响用户浏览播放列表的体验。我们使用了以下属性来控制侧边栏宽度:
.sideBarWidth(300) // 默认宽度
.minSideBarWidth(150) // 最小宽度
.maxSideBarWidth(500) // 最大宽度
这种设置有以下优点:
在歌单和歌曲列表项中,我们使用了Row
和Column
组件来创建灵活的布局:
Row() {
Image(playlist.cover)
.width(40)
.height(40)
.borderRadius(8)
.margin({ right: 12 })
Column() {
Text(playlist.name)
.fontSize(16)
.fontColor(this.currentPlaylist === playlist.name ? '#007DFF' : '#000000')
Text(`${playlist.count}首`)
.fontSize(12)
.fontColor('#888888')
.margin({ top: 4 })
}
.alignItems(HorizontalAlign.Start)
.layoutWeight(1)
}
这种布局有以下优点:
Row
组件将封面图片和文本信息水平排列Column
组件将标题和副标题垂直排列layoutWeight(1)
使文本区域占用剩余空间alignItems(HorizontalAlign.Start)
使文本左对齐在显示歌曲标题和艺术家名称时,我们使用了文本溢出处理:
Text(song.title)
.fontSize(16)
.fontColor(this.currentSongIndex === index ? '#007DFF' : '#000000')
.maxLines(1)
.textOverflow({ overflow: TextOverflow.Ellipsis })
这种处理有以下优点:
maxLines(1)
限制文本只显示一行textOverflow({ overflow: TextOverflow.Ellipsis })
在文本溢出时显示省略号在播放控制部分,我们使用了不同大小的按钮来突出主要功能:
// 上一首按钮
Image($r('app.media.ic_previous'))
.width(40)
.height(40)
// 播放/暂停按钮
Image(this.isPlaying ? $r('app.media.ic_pause') : $r('app.media.ic_play'))
.width(60)
.height(60)
.margin({ left: 40, right: 40 })
// 下一首按钮
Image($r('app.media.ic_next'))
.width(40)
.height(40)
这种设计有以下优点:
margin
属性为按钮之间添加适当的间距,提高可点击性isPlaying
状态显示不同的图标,提供直观的状态反馈我们的布局设计考虑了不同屏幕尺寸的适配:
width('60%')
等百分比值,使布局能够适应不同宽度的屏幕layoutWeight(1)
使组件能够根据可用空间调整大小在本教程中,我们学习了如何使用HarmonyOS NEXT的SideBarContainer
组件创建一个功能完善的音乐播放器侧边栏。 通过这个案例,我们展示了SideBarContainer
组件在音乐播放器应用中的应用,以及如何使用HarmonyOS NEXT的其他组件(如List
、Image
、Text
、Slider
等)创建丰富的用户界面。
在进阶篇中,我们将进一步探讨如何为音乐播放器添加更多高级功能,如歌词显示、均衡器设置、播放模式切换等,以及如何优化应用的性能和用户体验。