使用MediaPlayer+Service完成音乐播放器
清单文件记得加权限
1,先把歌曲封装成一个类
package com.example.service_music;
public class Song {
/** * 歌手 */
private String singer;
/** * 歌曲名 */
private String song;
/** * 歌曲的地址 */
private String path;
/** * 歌曲长度 */
private int duration;
/** * 歌曲的大小 */
private long size;
/** 当前歌曲在list中的位置*/
private int position;
public int getPosition() {
return position;
}
public void setPosition(int position) {
this.position = position;
}
public String getSinger() {
return singer;
}
public void setSinger(String singer) {
this.singer = singer;
}
public String getSong() {
return song;
}
public void setSong(String song) {
this.song = song;
}
public String getPath() {
return path;
}
public void setPath(String path) {
this.path = path;
}
public int getDuration() {
return duration;
}
public void setDuration(int duration) {
this.duration = duration;
}
public long getSize() {
return size;
}
public void setSize(long size) {
this.size = size;
}
}
2,封装查询本地歌曲的工具类
package com.example.service_music;
import android.content.Context;
import android.database.Cursor;
import android.provider.MediaStore;
import android.util.Log;
import android.widget.Toast;
import java.util.ArrayList;
import java.util.List;
/**
* ${FENG}
* 2019-07-24
*/
public class MusicUtils {
private static final String TAG = "MusicUtils";
/**
* 扫描系统里面的音频文件,返回一个list集合
*/
public static List<Song> getMusicData(Context context) {
List<Song> list = new ArrayList<>();
int i = 0;
Cursor cursor = context.getContentResolver().query(
MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, null, null, null,
MediaStore.Audio.AudioColumns.IS_MUSIC);
Log.i(TAG, "getMusicData: "+MediaStore.Audio.Media.EXTERNAL_CONTENT_URI.getPath());
if (cursor != null) {
while (cursor.moveToNext()) {
Song song = new Song();
song.setSong( cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Audio.Media.TITLE)));
song.setSinger( cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Audio.Media.ARTIST)));
song.setPath(cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Audio.Media.DATA)));
song.setDuration( cursor.getInt(cursor.getColumnIndexOrThrow(MediaStore.Audio.Media.DURATION)));
song.setSize( cursor.getLong(cursor.getColumnIndexOrThrow(MediaStore.Audio.Media.SIZE)));
song.setPosition(i);
i++;
// if (song.getSize() > 1000 * 800) {//过滤掉短音频
// // 分离出歌曲名和歌手
// if (song.getSong().contains("-")) {
// String[] str = song.getSong().split("-");
// song.setSinger( str[0]);
// song.setSong( str[1]);
// }
list.add(song);
// }
}
// 释放资源
cursor.close();
}else{
Toast.makeText(context, "没有找到文件", Toast.LENGTH_SHORT).show();
Log.i(TAG, "getMusicData: 没有");
}
return list;
}
//格式化时间
public static String formatTime(int time) {
if (time / 1000 % 60 < 10) {
return time / 1000 / 60 + ":0" + time / 1000 % 60;
} else {
return time / 1000 / 60 + ":" + time / 1000 % 60;
}
}
}
3 ,封装一个adapter类
package com.example.service_music;
import android.content.Context;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;
import java.util.List;
/**
* ${FENG}
* 2019-07-24
*/
public class MyAdapter extends BaseAdapter {
private Context context;
private List<Song> list;
private int position_flag = 0;
public MyAdapter(MainActivity mainActivity, List<Song> list) {
this.context = mainActivity;
this.list = list;
}
@Override
public int getCount() {
return list.size();
}
@Override
public Object getItem(int i) {
return list.get(i);
}
@Override
public long getItemId(int i) {
return i;
}
@Override
public View getView(int i, View view, ViewGroup viewGroup) {
ViewHolder holder = null;
if (view == null) {
holder = new ViewHolder();
// 引入布局
view = View.inflate(context, R.layout.list_item, null);
// 实例化对象
holder.song = (TextView) view.findViewById(R.id.item_mymusic_song);
holder.singer = (TextView) view
.findViewById(R.id.item_mymusic_singer);
holder.duration = (TextView) view
.findViewById(R.id.item_mymusic_duration);
holder.position = (TextView) view
.findViewById(R.id.item_mymusic_postion);
view.setTag(holder);
} else {
holder = (ViewHolder) view.getTag();
}
// 给控件赋值
String string_song = list.get(i).getSong();
if (string_song.length() >= 5
&& string_song.substring(string_song.length() - 4,
string_song.length()).equals(".mp3")) {
holder.song.setText(string_song.substring(0,
string_song.length() - 4).trim());
} else {
holder.song.setText(string_song.trim());
}
holder.singer.setText(list.get(i).getSinger().toString().trim());
// 时间转换为时分秒
int duration = list.get(i).getDuration();
String time = MusicUtils.formatTime(duration);
holder.duration.setText(time);
return view;
}
class ViewHolder {
TextView song;// 歌曲名
TextView singer;// 歌手
TextView duration;// 时长
TextView position;// 序号
}
}
4, adapter中的布局
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="60dp"
android:orientation="horizontal">
<TextView
android:id="@+id/item_mymusic_postion"
android:layout_width="30dp"
android:layout_height="30dp"
android:layout_alignBottom="@+id/item_mymusic_singer"
android:layout_gravity="center_vertical"
android:gravity="center"
android:layout_margin="10dp"
android:text="1"
android:textSize="18sp" />
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="60dp">
<TextView
android:id="@+id/item_mymusic_song"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_alignParentTop="true"
android:ellipsize="end"
android:maxLines="1"
android:layout_marginRight="10dp"
android:layout_marginTop="5dp"
android:text="歌曲名"
android:textSize="18sp" />
<TextView
android:id="@+id/item_mymusic_singer"
android:layout_width="wrap_content"
android:layout_height="30dp"
android:layout_alignParentBottom="true"
android:layout_alignParentLeft="true"
android:layout_toLeftOf="@+id/item_mymusic_duration"
android:gravity="bottom"
android:text="歌手"
android:ellipsize="end"
android:maxLines="1"
android:layout_marginBottom="5dp"
android:textSize="16sp" />
<TextView
android:id="@+id/item_mymusic_duration"
android:layout_width="wrap_content"
android:layout_height="30dp"
android:layout_alignParentBottom="true"
android:layout_alignParentRight="true"
android:gravity="bottom"
android:layout_marginRight="5dp"
android:layout_marginBottom="5dp"
android:text="歌曲时间"
android:textSize="16sp" />
</RelativeLayout>
</LinearLayout>
5,服务中的所有代码
package com.example.service_music.service;
import android.app.Service;
import android.content.Intent;
import android.media.MediaPlayer;
import android.os.Binder;
import android.os.IBinder;
import android.util.Log;
import android.widget.Toast;
import com.example.service_music.MusicUtils;
import com.example.service_music.MyAdapter;
import com.example.service_music.Song;
import java.io.IOException;
import java.util.List;
public class MusicService extends Service {
private MediaPlayer mediaPlayer;
private static final String TAG = "MusicService";
private List<Song> listSong;
int index;
public MusicService() {
}
@Override
public void onCreate() {
super.onCreate();
mediaPlayer = new MediaPlayer();
listSong = MusicUtils.getMusicData(this);
}
public class MusicBinder extends Binder{
public void callPlay(int position){
playSong(position);
index = position;
}
public void callPause(){
pause();
}
public void callPlayNextSong(){
if(++index > listSong.size() - 1){
index = 0;
}
playSong(index);
}
public void callPlayNavSong(){
playSong(index);
}
public void callRestart(){
restart();
}
}
private void restart() {
mediaPlayer.start();
}
private void pause() {
if(mediaPlayer.isPlaying()){
mediaPlayer.pause();
}
}
@Override
public IBinder onBind(Intent intent) {
return new MusicBinder();
}
//播放一个歌曲
private void playSong(int position) {
if(mediaPlayer.isPlaying()){
mediaPlayer.stop();
}
//先重置
mediaPlayer.reset();
//取到点击歌曲文件的位置
Song song = listSong.get(position);
try {
mediaPlayer.setDataSource(song.getPath());
// mediaPlayer.prepare();
mediaPlayer.prepareAsync();
mediaPlayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
@Override
public void onPrepared(MediaPlayer mp) {
mediaPlayer.start();
}
});
} catch (IOException e) {
e.printStackTrace();
}
}
private void playNextSong(int position) {
Log.i(TAG, "playNextSong: "+position);
Log.i(TAG, "size: "+listSong.size());
playSong(position);
}
//播放上一首
private void playNavSong() {
if (index <= 0){
index = 0;
}
playSong(index);
}
}
6 Activity中的布局.
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout 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:orientation="vertical"
android:layout_height="match_parent"
tools:context=".MainActivity">
<ListView
android:id="@+id/lv"
android:layout_weight="8"
android:layout_width="match_parent"
android:layout_height="0dp">
</ListView>
<LinearLayout
android:layout_weight="1"
android:layout_width="match_parent"
android:orientation="horizontal"
android:layout_height="0dp">
<Button
android:layout_weight="1"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:id="@+id/pause"
android:text="暂停"
/>
<Button
android:layout_weight="1"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:id="@+id/restart"
android:text="继续播放"
/>
<Button
android:layout_weight="1"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:id="@+id/nav"
android:text="上一首"
/>
<Button
android:layout_weight="1"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:id="@+id/next"
android:text="下一首"
/>
</LinearLayout>
</LinearLayout>
7 Activity中的代码
package com.example.service_music;
import android.Manifest;
import android.app.Service;
import android.content.ComponentName;
import android.content.Intent;
import android.content.ServiceConnection;
import android.content.pm.PackageManager;
import android.media.MediaPlayer;
import android.os.Bundle;
import android.os.IBinder;
import android.support.annotation.NonNull;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.AdapterView;
import android.widget.Button;
import android.widget.ListView;
import com.example.service_music.service.MusicService;
import java.io.IOException;
import java.util.List;
public class MainActivity extends AppCompatActivity implements View.OnClickListener{
private static final String TAG = "MainActivity";
private ListView lv;
private Button pause;
private Button restart;
private Button nav;
private Button next;
private Song song;
private List<Song> listSong;
private int index;
private ServiceConnection connection;
private MusicService.MusicBinder binder;
@Override
protected void onCreate(final Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
//请求授权
requestPermissions(new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},100);
lv.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
binder.callPlay(position);
}
});
Intent intent = new Intent(this,MusicService.class);
//1启动
startService(intent);
connection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
binder = (MusicService.MusicBinder)service;
}
@Override
public void onServiceDisconnected(ComponentName name) {
}
};
//2绑定
bindService(intent,connection,Service.BIND_AUTO_CREATE);
}
@Override
public void onClick(View v) {
int id = v.getId();
switch (id){
case R.id.pause:
binder.callPause();
break;
case R.id.restart:
binder.callRestart();
break;
case R.id.nav:
binder.callPlayNav();
break;
case R.id.next:
binder.callPlayNext();
break;
}
}
private void initView() {
lv = findViewById(R.id.lv);
pause = findViewById(R.id.pause);
pause.setOnClickListener(this);
restart = findViewById(R.id.restart);
restart.setOnClickListener(this);
nav = findViewById(R.id.nav);
nav.setOnClickListener(this);
next = findViewById(R.id.next);
next.setOnClickListener(this);
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if(requestCode == 100 && grantResults[0] == PackageManager.PERMISSION_GRANTED){
initAdapter();
}else{
//结束当前App
finish();
}
}
//初始化适配器
public void initAdapter(){
listSong = MusicUtils.getMusicData(this);
MyAdapter myAdapter = new MyAdapter(this, listSong);
lv.setAdapter(myAdapter);
}
@Override
protected void onDestroy() {
super.onDestroy();
//3 在解绑
if(connection != null){
unbindService(connection);
}
}
}
8 清单文件
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.service_music">
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"></uses-permission>
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<service
android:name=".service.MusicService"
android:enabled="true"
android:exported="true"></service>
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>