教你轻松制作Android音乐播放器
作者:lijiao 发布时间:2022-09-28 01:19:28
标签:android,播放器
欣赏一下我们清爽的界面吧~
如果是只用activity来制作这样的东西简直是太小儿科了,此处我们当然用的是service
首先我们先上service的代码:
1、如果我们要访问service的属性和方法,那么在activity肯定是以bindservice的方法实现的,而在service中的onbind方法也是必须要实现的,onbind返回的Ibinder对象在activity的serviceconnection中得到使用。
2、activity获取到Ibinder对象,可以进一步获取服务对象和player对象,来进行访问。
3、Environment.getExternalStorageDirectory()是获取sd中的内容的,不管是手机出场就已经内置的sd卡,还是用户后来自己添加的sd卡;而getExternalFilesDir()获取的真正是手机内部的存储空间,,/data/data/your_package/,随着应用的卸载存储的文件会被删除。
4、service通过发送广播与activity进行界面交互
public class MusicService extends Service{
private List<File> musicList;
private MediaPlayer player;
private int curPage;
public static final String MFILTER = "broadcast.intent.action.text";
public static final String NAME = "name";
public static final String TOTALTIME = "totaltime";
public static final String CURTIME = "curtime";
@Override
public IBinder onBind(Intent intent) {//1
// TODO Auto-generated method stub
return new MBinder();
}
public class MBinder extends Binder{//2
public MusicService getService(){
return MusicService.this;
}
public MediaPlayer getPlayer(){
return player;
}
}
@Override
public void onCreate() {
// TODO Auto-generated method stub
super.onCreate();
musicList = new ArrayList<File>();
File rootDir = Environment.getExternalStorageDirectory();//3
Log.d("rootname",rootDir.getName());
Log.d("rootname",rootDir.getAbsolutePath());
fillMusicList(rootDir);
Log.d("musiclist",String.valueOf(musicList.size()));
player = new MediaPlayer();
if (musicList.size() != 0) {
startPlay();
}
player.setOnCompletionListener(new OnCompletionListener() {
@Override
public void onCompletion(MediaPlayer mp) {
// TODO Auto-generated method stub
player.reset();
curPage = curPage==musicList.size()-1? (curPage+1)%musicList.size() : curPage+1;
startPlay();
}
});
}
/*迭代获取 音乐 文件*/
private void fillMusicList(File dir){
File[] sourceFiles = dir.listFiles();
Log.d("长度",String.valueOf(sourceFiles.length));
for(File file : sourceFiles){
if (file.isDirectory()) {
Log.d("文件夹名称",String.valueOf(file.getName()));
// if (!file.getName().equals("lost+found")) {
fillMusicList(file);
// }
}
else {
String name = file.getName();
Log.d("childname",file.getName());
if (name.endsWith(".mp3")||name.endsWith(".acc")) {//支持的格式
musicList.add(file);
}
}
}
}
private void startPlay(){
mSendBroadCast(NAME,musicList.get(curPage).getName());//4
try {
player.setDataSource(musicList.get(curPage).getAbsolutePath());
player.prepare();
player.start();
player.getDuration();
mSendBroadCast(TOTALTIME,player.getDuration());
Timer timer = new Timer();
timer.schedule(new TimerTask() {
@Override
public void run() {
// TODO Auto-generated method stub
mSendBroadCast(CURTIME,player.getCurrentPosition());
}
},0,1000);
} catch (IllegalArgumentException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (SecurityException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IllegalStateException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
public void playNext(){
curPage = curPage==musicList.size()-1? (curPage+1)%musicList.size() : curPage+1;
Log.d("curpage",String.valueOf(curPage));
player.reset();
startPlay();
}
public void playPrevious(){
curPage = curPage==0? 0 : curPage-1;
Log.d("curpage",String.valueOf(curPage));
player.reset();
startPlay();
}
public void parse(){
player.pause();
}
public void restart(){
player.start();
}
private void mSendBroadCast(String key, String value){
Intent intent = new Intent(MFILTER);
intent.putExtra(key,value);//发送广播
sendBroadcast(intent);
}
private void mSendBroadCast(String key, int value){
Intent intent = new Intent(MFILTER);
intent.putExtra(key,value);//发送广播
sendBroadcast(intent);
}
}
接下来上activity代码:
1、通过Ibinder对象获取服务对象
2、获取到服务对象以后,再访问服务的方法。
3、通过receiver刷新页面
public class MainActivity extends Activity implements OnClickListener{
SeekBar seekBar;
TextView curTime,totalTime;
TextView title;
private ServiceConnection sc;
private MusicService ms;
private boolean isStop;
private double totalTimeInt;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
IntentFilter filter = new IntentFilter(MusicService.MFILTER);
registerReceiver(new MusicReceiver(),filter);
sc = new ServiceConnection() {
@Override
public void onServiceDisconnected(ComponentName name) {
// TODO Auto-generated method stub
ms = null;
}
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
// TODO Auto-generated method stub
ms = ((MBinder)service).getService();//1
}
};
Button previous = (Button) findViewById(R.id.previous);
Button next = (Button) findViewById(R.id.next);
Button stop = (Button) findViewById(R.id.stop);
Button stopService = (Button) findViewById(R.id.stopService);
seekBar = (SeekBar) findViewById(R.id.mSeekbar);
curTime = (TextView) findViewById(R.id.curTime);
totalTime = (TextView) findViewById(R.id.totalTime);
title = (TextView) findViewById(R.id.title);
previous.setOnClickListener(this);
next.setOnClickListener(this);
stop.setOnClickListener(this);
stopService.setOnClickListener(this);
}
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
switch (v.getId()) {
case R.id.previous:
ms.playPrevious();//2
break;
case R.id.next:
ms.playNext();
break;
case R.id.stop:
if (isStop) {
ms.restart();
}
else {
ms.parse();
}
isStop = !isStop;
break;
case R.id.stopService:
Intent intent = new Intent("com.intent.musicplayer.MusicService");
unbindService(sc);
stopService(intent);
break;
default:
break;
}
}
@Override
protected void onStart() {
// TODO Auto-generated method stub
super.onStart();
Intent intent = new Intent("com.intent.musicplayer.MusicService");
bindService(intent,sc,Context.BIND_AUTO_CREATE);//当然你可以用startService的方式启动服务,这样结束了activity以后并不会结束service
}
private String transferMilliToTime(int millis){
DateFormat format = new SimpleDateFormat("mm:ss");
String result = format.format(new Date(millis));
return result;
}
private class MusicReceiver extends BroadcastReceiver{//3
@Override
public void onReceive(Context context, Intent intent) {
// TODO Auto-generated method stub
if (intent.getIntExtra(MusicService.CURTIME,0)!=0) {
double curTimeInt = intent.getIntExtra(MusicService.CURTIME,0);
curTime.setText(transferMilliToTime((int)curTimeInt));
double result = curTimeInt/totalTimeInt*100;
seekBar.setProgress((int) Math.floor(result));
}
else if(intent.getIntExtra(MusicService.TOTALTIME,0)!=0) {
totalTimeInt = intent.getIntExtra(MusicService.TOTALTIME,0);
totalTime.setText(transferMilliToTime((int)(totalTimeInt)));
}
else if (!TextUtils.isEmpty(intent.getStringExtra(MusicService.NAME))) {
title.setText(intent.getStringExtra(MusicService.NAME));
}
}
}
}
4、最后附上xml布局文件,算是代码上传完全了:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context="${relativePackage}.${activityClass}" >
<TextView
android:id="@+id/title"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_horizontal"
android:textSize="25sp"
android:textColor="#444444"
/>
<SeekBar
android:id="@+id/mSeekbar"
android:layout_gravity="center_horizontal"
android:layout_width="400dp"
android:layout_height="wrap_content"
android:max="100"
/>
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
>
<TextView
android:id="@+id/curTime"
android:layout_height="wrap_content"
android:layout_width="wrap_content"
android:layout_alignParentLeft="true"
/>
<TextView
android:id="@+id/totalTime"
android:layout_height="wrap_content"
android:layout_width="wrap_content"
android:layout_alignParentRight="true"
/>
</RelativeLayout>
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
>
<Button
android:id="@+id/previous"
android:layout_height="wrap_content"
android:layout_width="wrap_content"
android:text="上一曲"
android:layout_alignParentLeft="true"
/>
<Button
android:id="@+id/stop"
android:layout_height="wrap_content"
android:layout_width="wrap_content"
android:text="停止音乐"
android:layout_toRightOf="@id/previous"
/>
<Button
android:id="@+id/next"
android:layout_height="wrap_content"
android:layout_width="wrap_content"
android:text="下一曲"
android:layout_alignParentRight="true"
/>
<Button
android:id="@+id/stopService"
android:layout_height="wrap_content"
android:layout_width="wrap_content"
android:text="停止音乐服务"
android:layout_toLeftOf="@id/next"
/>
</RelativeLayout>
</LinearLayout>


猜你喜欢
- 一. 概述参考开源项目https://github.com/xkcoding/spring-boot-demo在系统运维中, 有时候为了避免
- java接口返回参数按照请求参数进行排序在项目实际开发中可能遇到过这种问题,接口请求参数顺序是[a,b,c],结果返回的数据是[bObjec
- java Long类型转为String类型1、Long.ValueOf("String")返回Long包装类型数据包装类
- 官方办法 JAVA语言提供的一个关键字“FINAL”可以用来履行该任务。看看下面的源代码范例://FinalDemo.java public
- 前言 SpringCloud 是微服务中的翘楚,最佳的落地方案。 在微服务架构中多层服务之间会相互调用,如果其中有一
- 本人新手,有什么问题 还请指出来,大家一起学习进步,话不多说。首先,搭建dubbo项目,要有三个工程。它们分别是:maven java工程
- 目录特性引入依赖使用特性Kotlin + Flow 实现的 Android 应用初始化任务启动库。支持模块化,按模块加载任务可指定工作进程名
- 一、系统介绍本系统实现扑克的分发,抢地主,电脑自动出牌等功能。二、系统展示1.扑克分发2.抢地主3.出牌4.游戏胜利三、系统实现Card.j
- 前言前阵子有同学反馈Flutter中的http请求无法通过fiddler抓包,作者喜欢使用Charles抓包工具,于是抽时间写了个小demo
- 由于Android对单个应用所施加的内存限制,比如16MB,这导致加载Bitmap的时候很容易出现内存溢出,本文主要包含2个方面的内容分析B
- 配置文件形式pom.xml<?xml version="1.0" encoding="UTF-8&quo
- 本文实例讲述了Java简单验证身份证功能。分享给大家供大家参考,具体如下:package org.cxy.csdn.example;impo
- 在使用spring框架中我们都知道,某个类如果使用了@Service、@Autowire 这种依赖注入的方式引用了其他对象,在另外一个类中,
- 解决Spring in action @valid验证不生效按照书上的示例代码来实现但是,添加了验证但是没有生效。Spring提供了校验Ap
- 在本文中,我们将看到如何将double转换为int。在Java编程中,您将有一个double原语值(例如82.14),但是要执行进一步的操作
- Java 堆是用来存储对象实例的, 因此如果我们不断地创建对象, 并且保证 GC Root 和创建的对象之间有可达路径以免对象被垃圾回收,
- 本文实例讲述了Android实现的数字格式化用法。分享给大家供大家参考,具体如下:package formatnumber;import j
- 做多媒体项目时,经常会最后来个客户签名并保存之类的,签名保存之前的博客Unity3d截图方法合集有介绍过了,今天闲着把断笔写字的也贴出来吧,
- 在不用Maven的时候,比如说以前我们用Ant构建项目,在项目目录下,往往会看到一个名为/lib的子目录,那里存放着各类第三方依赖jar文件
- 在 Windows 有一些字符是不能作为文件名,尝试重命名一个文件,输入/ 就可以看到windows 提示的不能作为文件名的字符那么具体是包