音乐播放器
功能分析
播放
- 点击歌曲播放(recyclerView的Item的点击事件)
- 点击按钮播放(按钮的点击事件)
暂停
- 点击按钮暂停(按钮的点击事件)
上一首
- 点击按钮切换到上一首(按钮的点击事件)
下一首
- 点击按钮切换到下一首(按钮的点击事件)
布局
页面主布局
采用约束布局。
recyclerView放在上部分,下部分作为当前正在播放歌曲的信息展示和切换、播放、暂停歌曲的按钮。
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@mipmap/bg2">
<ImageView
android:id="@+id/iv_splitline"
android:layout_width="match_parent"
android:layout_height="0.5dp"
android:background="@color/violet"
app:layout_constraintBottom_toBottomOf="parent"
android:layout_marginBottom="75dp"
/>
<ImageView
android:id="@+id/iv_current_icon"
android:layout_width="60dp"
android:layout_height="60dp"
android:layout_centerVertical="true"
android:background="@mipmap/a1"
android:src="@mipmap/icon_song"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@+id/iv_current_last"
app:layout_constraintHorizontal_bias="0.036"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/iv_splitline"
app:layout_constraintVertical_bias="0.448" />
<TextView
android:id="@+id/tv_current_title"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginStart="8dp"
android:text=""
android:textSize="18sp"
android:textStyle="bold"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintStart_toEndOf="@+id/iv_current_icon"
app:layout_constraintTop_toBottomOf="@+id/iv_splitline"
app:layout_constraintVertical_bias="0.2" />
<TextView
android:id="@+id/tv_current_singer"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginStart="8dp"
android:text=""
android:textSize="12sp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintStart_toEndOf="@+id/iv_current_icon"
app:layout_constraintTop_toBottomOf="@+id/tv_current_title"
app:layout_constraintVertical_bias="0.277" />
<ImageView
android:id="@+id/iv_current_next"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentEnd="true"
android:layout_centerVertical="true"
android:src="@mipmap/icon_next"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintTop_toBottomOf="@+id/iv_splitline"
android:layout_marginEnd="20dp"/>
<ImageView
android:id="@+id/iv_current_play"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerVertical="true"
android:layout_marginEnd="20dp"
android:layout_toStartOf="@id/iv_current_next"
android:src="@mipmap/icon_play"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@+id/iv_current_next"
app:layout_constraintTop_toBottomOf="@+id/iv_splitline" />
<ImageView
android:id="@+id/iv_current_last"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerVertical="true"
android:layout_marginEnd="20dp"
android:layout_toStartOf="@id/iv_current_play"
android:src="@mipmap/icon_last"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@+id/iv_current_play"
app:layout_constraintTop_toBottomOf="@+id/iv_splitline" />
<androidx.recyclerview.widget.RecyclerView
android:id="@+id/rv"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_marginBottom="0dp"
app:layout_constraintBottom_toTopOf="@+id/iv_splitline"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintVertical_bias="0.0"
tools:ignore="MissingConstraints" />
</androidx.constraintlayout.widget.ConstraintLayout>
recyclerView中的ite布局
采用 卡片式布局,把每一个item作为一张卡片。
卡片中主要展示:歌曲排序的序列号,歌曲名,歌手名,专辑名,时长。
<?xml version="1.0" encoding="utf-8"?>
<androidx.cardview.widget.CardView xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginLeft="10dp"
android:layout_marginRight="10dp"
android:layout_marginTop="10dp"
app:contentPadding="10dp"
app:cardCornerRadius="10dp"
app:cardElevation="1dp"
app:cardBackgroundColor="@color/pink">
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView
android:id="@+id/tv_number"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerVertical="true"
android:text=""
android:textSize="24sp"
android:textStyle="bold"
android:layout_marginStart="10dp"/>
<TextView
android:id="@+id/tv_title"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginStart="20dp"
android:layout_marginTop="10dp"
android:layout_toEndOf="@id/tv_number"
android:singleLine="true"
android:text=""
android:textSize="18sp"
android:textStyle="bold"
/>
<TextView
android:id="@+id/tv_singer"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_below="@id/tv_title"
android:layout_alignStart="@id/tv_title"
android:layout_marginTop="5dp"
android:text=""
/>
<TextView
android:id="@+id/tv_line"
android:layout_width="2dp"
android:layout_height="18dp"
android:background="@color/textcolor"
android:layout_toEndOf="@id/tv_singer"
android:layout_marginLeft="10dp"
android:layout_marginRight="10dp"
android:layout_alignTop="@id/tv_singer"
/>
<TextView
android:id="@+id/tv_album"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignTop="@id/tv_singer"
android:layout_toEndOf="@id/tv_line"
android:ellipsize="end"
android:singleLine="true"
android:text=""
android:textColor="@color/textcolor"
android:textSize="14sp"
/>
<TextView
android:id="@+id/tv_duration"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_below="@id/tv_singer"
android:layout_alignParentEnd="true"
android:text=""
android:textSize="14sp"
android:textColor="@color/textcolor"
android:layout_centerVertical="true"/>
</RelativeLayout>
</androidx.cardview.widget.CardView>
数据载入
初始化布局
private void initView() {
lastIv = findViewById(R.id.iv_current_last);
playIv = findViewById(R.id.iv_current_play);
nextIv = findViewById(R.id.iv_current_next);
songTv = findViewById(R.id.tv_current_title);
singerTv = findViewById(R.id.tv_current_singer);
musicRv = findViewById(R.id.rv);
lastIv.setOnClickListener(this);
playIv.setOnClickListener(this);
nextIv.setOnClickListener(this);
}
编写音乐实体类
目的:存放之后查询到的音乐的信息。
//音乐实体
LocalMusicBean musicBean;
public class LocalMusicBean {
private String id;//歌曲id
private String song;//歌曲名称
private String singer;//歌手名称
private String album;//专辑名称
private String duration;//歌曲时长
private String path;//歌曲存储路径
public LocalMusicBean() {
}
public LocalMusicBean(String id, String song, String singer, String album, String duration, String path) {
this.id = id;
this.song = song;
this.singer = singer;
this.album = album;
this.duration = duration;
this.path = path;
}
public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public String getSong() {
return song;
}
public void setSong(String song) {
this.song = song;
}
public String getSinger() {
return singer;
}
public void setSinger(String singer) {
this.singer = singer;
}
public String getAlbum() {
return album;
}
public void setAlbum(String album) {
this.album = album;
}
public String getDuration() {
return duration;
}
public void setDuration(String duration) {
this.duration = duration;
}
public String getPath() {
return path;
}
public void setPath(String path) {
this.path = path;
}
}
设置数据源
查找出手机中的音乐文件,把所有音乐文件存入数据源当中。
//数据源
List<LocalMusicBean> mData;
//获取ContentResolver对象
String[] selectionArgs = new String[]{getString(R.string.music)};
String selection = MediaStore.Audio.Media.DATA + getString(R.string.like);
// 媒体库查询语句
@SuppressLint("Recycle") Cursor cursor = getContentResolver().query(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, null, selection, selectionArgs, MediaStore.Audio.AudioColumns.IS_MUSIC);
//遍历Cursor
int id = 0;
while (cursor.moveToNext()) {
String song = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Audio.Media.TITLE));
String singer = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Audio.Media.ARTIST));
String album = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Audio.Media.ALBUM));
long duration = cursor.getLong(cursor.getColumnIndexOrThrow(MediaStore.Audio.Media.DURATION));
@SuppressLint("SimpleDateFormat") SimpleDateFormat dateFormat = new SimpleDateFormat(getString(R.string.timeformat));
String time = dateFormat.format(new Date(duration));
//将一行当中的数据封装到对象当中
if (!time.equals(getString(R.string.timeiszeero))) {
id++;
String sid = String.valueOf(id);
String path = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Audio.Media.DATA));
musicBean = new LocalMusicBean(sid, song, singer, album, time, path);
mData.add(musicBean);
}
}
//数据源发生变化,提示适配器更新
musicAdapter.notifyDataSetChanged();
设置适配器
把数据源中的数据加载到布局中显示出来。
//适配器
private LocalMusicAdapter musicAdapter;
public class LocalMusicAdapter extends RecyclerView.Adapter<LocalMusicAdapter.localMusicViewHolder> {
//上下文
Context context;
//数据源
List<LocalMusicBean> mDatas;
//点击事件
onItemClickListener onItemClickListener;
public void setOnItemClickListener(LocalMusicAdapter.onItemClickListener onItemClickListener) {
this.onItemClickListener = onItemClickListener;
}
//自定义的点击事件的接口,后面通过回调写点击事件
public interface onItemClickListener{
void OnItemClick(View view,int position);
}
public LocalMusicAdapter(Context context, List<LocalMusicBean> mDatas) {
this.context = context;
this.mDatas = mDatas;
}
//加载recyclerView的item布局
@NonNull
@Override
public localMusicViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
View view = LayoutInflater.from(context).inflate(R.layout.item_local_music, parent, false);
return new localMusicViewHolder(view);
}
//item布局初始化并且设置点击事件
@Override
public void onBindViewHolder(@NonNull localMusicViewHolder holder, int position) {
LocalMusicBean musicBean = mDatas.get(position);
holder.idTv.setText(musicBean.getId());
holder.songTv.setText(musicBean.getSong());
holder.singerTv.setText(musicBean.getSinger());
holder.albumTv.setText(musicBean.getAlbum());
holder.timeTv.setText(musicBean.getDuration());
holder.itemView.setOnClickListener(view -> {
onItemClickListener.OnItemClick(view,position);
});
}
//获得数据源列表的长度,即有多少首歌
@Override
public int getItemCount() {
return mDatas.size();
}
//加载item的布局
static class localMusicViewHolder extends RecyclerView.ViewHolder {
TextView idTv,songTv,singerTv,albumTv,timeTv;
public localMusicViewHolder(@NonNull View itemView) {
super(itemView);
idTv = itemView.findViewById(R.id.tv_number);
songTv = itemView.findViewById(R.id.tv_title);
singerTv = itemView.findViewById(R.id.tv_singer);
albumTv = itemView.findViewById(R.id.tv_album);
timeTv = itemView.findViewById(R.id.tv_duration);
}
}
}
点击事件
点击歌曲播放对应歌曲,点击上一首按钮,播放上一首歌曲;点击下一首歌曲,播放下一首歌曲。
//记录当前正在播放的音乐的位置
private int currentPosition = -1;
//记录当前播放的音乐播放到哪了
private int currentPositionInMusic = 0;
MediaPlayer mediaPlayer;
//设置recyclerView的item点击事件
private void setEventListener() {
musicAdapter.setOnItemClickListener((view, position) -> {
currentPosition = position;
musicBean = mData.get(position);
playMusicInMusicBean(musicBean);
});
}
//根据歌曲来进行播放
public void playMusicInMusicBean(LocalMusicBean musicBean) {
songTv.setText(musicBean.getSong());
singerTv.setText(musicBean.getSinger());
stopMusic();
mediaPlayer.reset();
try {
currentPositionInMusic = 0;
mediaPlayer.setDataSource(musicBean.getPath());
playMusic();
} catch (IOException ioException) {
ioException.printStackTrace();
}
}
//播放音乐
private void playMusic(){
if (currentPositionInMusic == 0){
//从头开始播放
if (mediaPlayer != null && !mediaPlayer.isPlaying()){
try {
mediaPlayer.prepare();
} catch (IOException e) {
e.printStackTrace();
}
mediaPlayer.start();
}
}else{
//从暂停处开始播放
mediaPlayer.seekTo(currentPositionInMusic);
mediaPlayer.start();
}
playIv.setImageResource(R.mipmap.icon_pause);
}
//暂停播放音乐
private void pauseMusic() {
if (mediaPlayer != null && mediaPlayer.isPlaying()){
currentPositionInMusic = mediaPlayer.getCurrentPosition();
mediaPlayer.pause();
playIv.setImageResource(R.mipmap.icon_play);
}
}
//停止播放
private void stopMusic() {
if (mediaPlayer != null){
mediaPlayer.pause();
mediaPlayer.seekTo(0);
mediaPlayer.stop();
playIv.setImageResource(R.mipmap.icon_play);
}
}
@Override
protected void onDestroy() {
super.onDestroy();
// stopMusic();
}
//按钮的点击事件
@SuppressLint("NonConstantResourceId")
@Override
public void onClick(View view) {
switch (view.getId()){
case R.id.iv_current_last:
if (currentPosition == -1){
//没有选中要播放的音乐
Toast.makeText(this,"请选择要播放的音乐",Toast.LENGTH_LONG).show();
return;
}
if (currentPosition == 0){
Toast.makeText(this,"已经是第一首了 ",Toast.LENGTH_LONG).show();
}
else {
currentPosition--;
currentPositionInMusic = 0;
LocalMusicBean lastMusicBean = mData.get(currentPosition);
playMusicInMusicBean(lastMusicBean);
}
break;
case R.id.iv_current_play:
if (currentPosition == -1){
//没有选中要播放的音乐
Toast.makeText(this,"请选择要播放的音乐",Toast.LENGTH_LONG).show();
return;
}
if (mediaPlayer.isPlaying()){
//处于播放状态,需要暂停音乐
pauseMusic();
}else {
playMusic();
}
break;
case R.id.iv_current_next:
if (currentPosition == -1){
Toast.makeText(this,"请选择要播放的音乐 ",Toast.LENGTH_LONG).show();
return;
}else if (currentPosition == mData.size()-1){
Toast.makeText(this,"已经是最后一首了",Toast.LENGTH_LONG).show();
}else {
currentPosition++;
currentPositionInMusic = 0;
LocalMusicBean nextMusicBean = mData.get(currentPosition);
playMusicInMusicBean(nextMusicBean);
}
break;
default:
break;
}
}
代码逻辑:先通过recyclerView的点击事件拿到歌曲在列表中的位置和对应歌曲实体,然后再根据歌曲实体拿到歌曲信息加载到布局中。最后根据所拿信息调用MediaPlayer的方法进行播放。
注意:暂停功能中用到了播放位置,在第一次播放时currentPositionInMusic对播放无影响,但是如果按了暂停,currentPositionInMusic发生改变,如果这时候点击事件不是继续播放,且没有对currentPositionInMusic进行重置,那么会进入继续播放逻辑的代码中,但是继续播放逻辑的代码没有对歌曲实体进行加载和准备,所以会报错。因此,在非继续播放事件发生前需要对currentPositionInMusic进行重置。
整个项目已经在gitee上开源,如有需要请自取。gitee地址:https://gitee.com/luozhaosong/new-music-app