教你轻松制作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>
0
投稿
猜你喜欢
- 实践过程效果代码public partial class Form1 : Form{ public Form1()
- 一、什么时候会加载类?使用到类中的内容时加载:有三种情况1.创建对象:new StaticCode();2.使用类中的静态成员:Static
- 一. 可变字符串1. 简介在Java中,我们除了可以通过String类创建和处理字符串之外,还可以使用StringBuffer和String
- 前言我们在配置Spring Xml配置文件的时候,可以在文件路径字符串中加入 ${} 占位符,Spring会自动帮我们解析占位符,这么神奇的
- 前言:在纯 Java 代码里 我们一般都会用class.getResource(String name) 或者 class.getClass
- 序本文主要研究下迁移到java9的一些注意事项。迁移种类1、代码不模块化,先迁移到jdk9上,好利用jdk9的api2、代码同时也模块化迁移
- Java 垃圾回收与对象生命周期详解Java中的垃圾回收与对象生命周期1. 垃圾回收 垃圾回收是Java程序设计中
- 本文介绍C# lock关键字,C#提供了一个关键字lock,它可以把一段代码定义为互斥段(critical section),互斥段在一个时
- IntelliJ IDEA是很多程序员必备且在业界被公认为最好的Java开发工具,有很多小伙伴在安装完IDEA并且tomcat之后,启动to
- 报错org.springframework.web.util.NestedServletException: Request process
- 目标:list中有0到39共40个元素,删除其中索引是10、20、30的元素方案一:使用普通for循环从前往后遍历再删除//初始化List列
- 使用httpclient检测url状态及链接是否能打开有时候我们需要检测某个url返回的状态码是不是200或者页面能不能正常打开响应可使用如
- ArrayList线程不安全怎么办?有三种解决方法:使用对应的 Vector 类,这个类中的所有方法都加上了 synchronized 关键
- 本文实例为大家分享了Java基于Socket实现简易版多人聊天室的具体代码,供大家参考,具体内容如下一、 聊天室需求1、一个服务端,多个客户
- Spring5路径匹配器PathPatternPathPattern 对url地址匹配的处理更加快速,它和AntPathMatcher 主要
- Random类介绍Random类一个用于产生 伪随机 数字的类。这里的伪随机表示有随机性但是可以基于算法模拟出随机规律。Random类的构造
- Java 开发语言中实现HTTP请求的方法主要有两种:一种是JAVA的标准类HttpUrlConnection,比较原生的实现方法;另一种是
- 本文实例讲述了Java实现的模糊匹配某文件夹下的文件并删除功能。分享给大家供大家参考,具体如下:package com.wyebd.gis;
- 如何使用struts2 * ,或者自定义 * 。特别注意,在使用 * 的时候,在Action里面必须最后一定要引用struts2自带的 *
- 前言 短时间提升自己最快的手段就是背面试题,最近总结了Java常用的面试题,分享给大家,希望大家都能圆梦大厂,加油,我命由我不由天