使用MediaPlayer+Service+Broadcast完成音乐播放器
1,封装歌曲的类
package com.example.service_broadcast_music;
public class Song {
private String title;
private String artist;
private int duration;
private String data;
private int size;
public Song(String title, String artist, int duration, String data, int size) {
this.title = title;
this.artist = artist;
this.duration = duration;
this.data = data;
this.size = size;
}
public Song() {
}
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
public String getArtist() {
return artist;
}
public void setArtist(String artist) {
this.artist = artist;
}
public int getDuration() {
return duration;
}
public void setDuration(int duration) {
this.duration = duration;
}
public String getData() {
return data;
}
public void setData(String data) {
this.data = data;
}
public int getSize() {
return size;
}
public void setSize(int size) {
this.size = size;
}
}
2 读取所有的歌曲
package com.example.service_broadcast_music;
import android.content.ContentResolver;
import android.content.Context;
import android.database.Cursor;
import android.net.Uri;
import android.provider.MediaStore;
import java.util.ArrayList;
import java.util.List;
public class Utils {
private Context context;
private List<Song> list;
public Utils(Context context, List<Song> list) {
this.context = context;
this.list = list;
}
public static List<Song> init(Context context){
List<Song> list=new ArrayList<>();
Uri uri= MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
String[] strings={MediaStore.Audio.Media.TITLE,MediaStore.Audio.Media.ARTIST,MediaStore.Audio.Media.DURATION,MediaStore.Audio.Media.DATA,MediaStore.Audio.Media.SIZE};
ContentResolver contentResolver = context.getContentResolver();
Cursor cursor = contentResolver.query(uri, strings, null, null, null);
while (cursor.moveToNext()){
String title = cursor.getString(cursor.getColumnIndex(MediaStore.Audio.Media.TITLE));
String artist = cursor.getString(cursor.getColumnIndex(MediaStore.Audio.Media.ARTIST));
int duration = cursor.getInt(cursor.getColumnIndex(MediaStore.Audio.Media.DURATION));
String data = cursor.getString(cursor.getColumnIndex(MediaStore.Audio.Media.DATA));
int size = cursor.getInt(cursor.getColumnIndex(MediaStore.Audio.Media.SIZE));
Song song = new Song(title,artist,duration,data,size);
list.add(song);
}
cursor.close();
return list;
}
}
3 创建listview布局适配器
package com.example.service_broadcast_music;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;
import java.util.List;
public class Myadpater extends BaseAdapter {
private Context context;
private List<Song> list;
public Myadpater(Context context, List<Song> list) {
this.context = context;
this.list = list;
}
@Override
public int getCount() {
return list.size();
}
@Override
public Object getItem(int position) {
return list.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
Viewholder viewholder;
if (convertView==null){
convertView = LayoutInflater.from(context).inflate(R.layout.layout, null);
viewholder=new Viewholder();
viewholder.title=convertView.findViewById(R.id.titleid);
viewholder.artist = convertView.findViewById(R.id.artistid);
viewholder.duration=convertView.findViewById(R.id.durationid);
convertView.setTag(viewholder);
}else {
viewholder = (Viewholder) convertView.getTag();
}
viewholder.title.setText(list.get(position).getTitle());
viewholder.artist.setText(list.get(position).getArtist());
int time = list.get(position).getDuration();
String s = formatTime(time);
viewholder.duration.setText(s);
return convertView;
}
class Viewholder{
TextView title;
TextView artist;
TextView duration;
}
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;
}
}
}
4 service中的代码
package com.example.service_broadcast_music;
import android.app.Notification;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.app.Service;
import android.content.Intent;
import android.media.MediaPlayer;
import android.os.Binder;
import android.os.IBinder;
import android.widget.RemoteViews;
import java.io.IOException;
import java.util.List;
public class MyService extends Service {
private MediaPlayer mediaPlayer;
private List<Song> list;
private int index;
private Notification.Builder builder;
public MyService() {
}
@Override
public IBinder onBind(Intent intent) {
return new Musicbiner();
}
@Override
public void onCreate() {
super.onCreate();
mediaPlayer = new MediaPlayer();
list = Utils.init(this);
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
NotificationManager manager= (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
builder=new Notification.Builder(this);
builder.setSmallIcon(R.mipmap.dog);
//自定义广播中的布局
RemoteViews views = new RemoteViews(getPackageName(),R.layout.activity_main);
Intent intent1 = new Intent();
intent1.setAction("com.notify.stop");
PendingIntent pendingIntent = PendingIntent.getBroadcast(this, 800, intent1, PendingIntent.FLAG_UPDATE_CURRENT);
views.setOnClickPendingIntent(R.id.stopid,pendingIntent);
Intent intent2 = new Intent();
intent1.setAction("com.notify.continue");
PendingIntent pendingIntent2 = PendingIntent.getBroadcast(this, 810, intent1, PendingIntent.FLAG_UPDATE_CURRENT);
views.setOnClickPendingIntent(R.id.continueid,pendingIntent2);
Intent intent3 = new Intent();
intent1.setAction("com.notify.back");
PendingIntent pendingIntent3 = PendingIntent.getBroadcast(this, 820, intent1, PendingIntent.FLAG_UPDATE_CURRENT);
views.setOnClickPendingIntent(R.id.backid,pendingIntent3);
Intent intent4 = new Intent();
intent1.setAction("com.notify.next");
PendingIntent pendingIntent4 = PendingIntent.getBroadcast(this, 830, intent1, PendingIntent.FLAG_UPDATE_CURRENT);
views.setOnClickPendingIntent(R.id.nextid,pendingIntent4);
//设置通知的布局
builder.setCustomContentView(views);
//开始一个前台服务
startForeground(1,builder.build());
return super.onStartCommand(intent,flags,startId);
}
public class Musicbiner extends Binder{
public void callplay(int position){
playsong(position);
index=position;
}
public void callpause(){
pause();
}
public void callcontinue(){
playcontinue();
}
public void callnextsong(){
if (++index>list.size()-1){
index=0;
}
playsong(index);
}
public void callbacksong(){
if (--index<0){
index=0;
}
playsong(index);
}
}
private void playcontinue() {
mediaPlayer.start();
}
private void pause() {
if (mediaPlayer.isPlaying()){
mediaPlayer.pause();
}
}
private void playsong(int position){
if (mediaPlayer.isPlaying()){
mediaPlayer.stop();
}
//重置
mediaPlayer.reset();
//找到位置
Song song = list.get(position);
try {
mediaPlayer.setDataSource(song.getData());
mediaPlayer.prepareAsync();
mediaPlayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
@Override
public void onPrepared(MediaPlayer mp) {
mediaPlayer.start();
}
});
} catch (IOException e) {
e.printStackTrace();
}
}
}
5 service中通知的布局文件
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout 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"
tools:context=".MainActivity">
<ListView
android:id="@+id/listid"
android:layout_width="match_parent"
android:layout_height="match_parent"></ListView>
<LinearLayout
android:gravity="center"
android:layout_alignParentBottom="true"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<Button
android:id="@+id/continueid"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="继续播放"
/>
<Button
android:id="@+id/stopid"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="暂停播放"
/>
<Button
android:id="@+id/backid"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="上一首"
/>
<Button
android:id="@+id/nextid"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="下一首"
/>
</LinearLayout>
</RelativeLayout>
6 广播的内容
package com.example.service_broadcast_music;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;
public class MyReceiver extends BroadcastReceiver {
private static final String TAG = "MyReceiver";
@Override
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
Log.i(TAG, "onReceive: "+action);
if (action.equals("com.notify.stop")){
MainActivity.handler.sendEmptyMessage(110);
}else if (action.equals("com.notify.continue")){
MainActivity.handler.sendEmptyMessage(120);
}else if (action.equals("com.notify.back")){
MainActivity.handler.sendEmptyMessage(130);
}else if (action.equals("com.notify.next")){
MainActivity.handler.sendEmptyMessage(140);
}
}
}
7 MainActivity中的代码
package com.example.service_broadcast_music;
import android.Manifest;
import android.app.Service;
import android.content.ComponentName;
import android.content.Intent;
import android.content.IntentFilter;
import android.content.ServiceConnection;
import android.content.pm.PackageManager;
import android.os.Build;
import android.os.Handler;
import android.os.IBinder;
import android.os.Message;
import android.support.annotation.NonNull;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.AdapterView;
import android.widget.Button;
import android.widget.ListView;
import java.util.List;
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
private ListView listid;
private Button continueid;
private Button stopid;
private Button backid;
private Button nextid;
private List<Song> list;
private ServiceConnection connection;
private static MyService.Musicbiner biner;
private Myadpater myadpater;
private static final String TAG = "MainActivity";
public static Handler handler=new Handler(){
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
Log.i(TAG, "handleMessage: "+msg.what);
if (msg.what==110){
biner.callpause();
}else if (msg.what==120){
biner.callcontinue();
}else if (msg.what==130){
biner.callbacksong();
}else if (msg.what==140){
biner.callnextsong();
}
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
viewinit();
if (Build.VERSION.SDK_INT>=Build.VERSION_CODES.M){
requestPermissions(new String[]{Manifest.permission.READ_EXTERNAL_STORAGE,Manifest.permission.WRITE_EXTERNAL_STORAGE},101);
}
listid.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
biner.callplay(position);
}
});
Intent intent = new Intent(this,MyService.class);
startService(intent);
connection=new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
biner=(MyService.Musicbiner)service;
}
@Override
public void onServiceDisconnected(ComponentName name) {
}
};
bindService(intent,connection, Service.BIND_AUTO_CREATE);
}
private void viewinit() {
listid = (ListView) findViewById(R.id.listid);
continueid = (Button) findViewById(R.id.continueid);
continueid.setOnClickListener(this);
stopid = (Button) findViewById(R.id.stopid);
stopid.setOnClickListener(this);
backid = (Button) findViewById(R.id.backid);
backid.setOnClickListener(this);
nextid = (Button) findViewById(R.id.nextid);
nextid.setOnClickListener(this);
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction("com.notify.stop");
intentFilter.addAction("com.notify.continue");
intentFilter.addAction("com.notify.back");
intentFilter.addAction("com.notify.next");
MyReceiver myReceiver = new MyReceiver();
registerReceiver(myReceiver,intentFilter);
}
@Override
public void onClick(View v) {
int id = v.getId();
switch (id){
case R.id.stopid:
biner.callpause();
break;
case R.id.continueid:
biner.callcontinue();
break;
case R.id.nextid:
biner.callnextsong();
break;
case R.id.backid:
biner.callbacksong();
break;
}
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (requestCode==101&&grantResults[0]== PackageManager.PERMISSION_GRANTED){
List<Song> init = Utils.init(this);
myadpater = new Myadpater(this, init);
listid.setAdapter(myadpater);
}
}
@Override
protected void onDestroy() {
super.onDestroy();
if (connection!=null){
unbindService(connection);
}
}
}
8 MainActivity中的布局
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout 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"
tools:context=".MainActivity">
<ListView
android:id="@+id/listid"
android:layout_width="match_parent"
android:layout_height="match_parent"></ListView>
<LinearLayout
android:gravity="center"
android:layout_alignParentBottom="true"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<Button
android:id="@+id/continueid"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="继续播放"
/>
<Button
android:id="@+id/stopid"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="暂停播放"
/>
<Button
android:id="@+id/backid"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="上一首"
/>
<Button
android:id="@+id/nextid"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="下一首"
/>
</LinearLayout>
</RelativeLayout>
9 listView的布局文件
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent"
android:layout_height="match_parent">
<LinearLayout
android:layout_weight="1"
android:layout_width="0dp"
android:orientation="vertical"
android:layout_height="wrap_content">
<TextView
android:id="@+id/titleid"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="我是歌名"
android:layout_margin="10dp"
/>
<TextView
android:id="@+id/artistid"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="我是歌手"
android:layout_margin="10dp"
/>
</LinearLayout>
<LinearLayout
android:layout_width="0dp"
android:layout_height="78dp"
android:layout_weight="0.5"
android:gravity="center">
<TextView
android:id="@+id/durationid"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="20dp"
android:text="我是时长" />
</LinearLayout>
</LinearLayout>
10 清单文件
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.service_broadcast_music">
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"></uses-permission>
<uses-permission android:name="android.permission.WRITE_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">
<receiver
android:name=".MyReceiver"
android:enabled="true"
android:exported="true"></receiver>
<service
android:name=".MyService"
android:enabled="true"
android:exported="true" />
<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>