音乐播放器实战
学习目标
学习安卓的广播机制,课堂上讲解了系统广播和自定义广播,并学习了静态注册和动态注册两种。
在此完成安卓音乐播放器,该播放器包括功能:播放、暂停、取消播放、音乐上一首与下一首歌切换、显示当前播放英语的信息。
Broadcast的讲解
项目实战
短信来时播放音乐
我完成的是音乐播放器的实现,这里就先直接放上别的博客了。
这里有相关的案例
音乐播放器
项目目录
这个项目编码需要完成的内容包括:
- MainActivity 主活动
- MusicService 音乐播放器
- main.xml 页面
需要修改的内容包括:
- AndroidManifest.xml 对于Service进行配置、还需在此需要配置intent相关信息
- assets 存放音乐
- res.drawable
相关编码
MainActivity.java
完成任务
在此需要完成的是:界面按钮的监听工作(当按钮按下时发送对应的广播)、接收广播(接收Service传送过来的有关歌曲的信息);对于上下首歌的功能,在此增加了两个按钮的监听。
代码编写
package com.example.maibenben.delta_vi_musicbox;
import android.app.Activity;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.ImageButton;
import android.widget.TextView;
public class MainActivity extends Activity implements OnClickListener
{
// 获取界面中显示歌曲标题、作者文本框
TextView title, author;
// 播放/暂停、停止按钮
ImageButton play, stop, prev,next;
ActivityReceiver activityReceiver;
public static final String CTL_ACTION =
"org.crazyit.action.CTL_ACTION";
public static final String UPDATE_ACTION =
"org.crazyit.action.UPDATE_ACTION";
// 定义音乐的播放状态,0x11代表没有播放;0x12代表正在播放;0x13代表暂停
int status = 0x11;
String[] titleStrs = new String[] { "心愿", "约定", "美丽新世界" };
String[] authorStrs = new String[] { "未知艺术家", "周蕙", "伍佰" };
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
// 获取程序界面界面中的两个按钮
play = (ImageButton) this.findViewById(R.id.play);
stop = (ImageButton) this.findViewById(R.id.stop);
next = (ImageButton) this.findViewById(R.id.next);
prev = (ImageButton) this.findViewById(R.id.prev);
title = (TextView) findViewById(R.id.title);
author = (TextView) findViewById(R.id.author);
// 为两个按钮的单击事件添加监听器
play.setOnClickListener(this);
stop.setOnClickListener(this);
next.setOnClickListener(this);
prev.setOnClickListener(this);
activityReceiver = new ActivityReceiver();
// 创建IntentFilter
IntentFilter filter = new IntentFilter();
// 指定BroadcastReceiver监听的Action
filter.addAction(UPDATE_ACTION);
// 注册BroadcastReceiver
registerReceiver(activityReceiver, filter);
Intent intent = new Intent(this, MusicService.class);
// 启动后台Service
startService(intent);
}
// 自定义的BroadcastReceiver,负责监听从Service传回来的广播
public class ActivityReceiver extends BroadcastReceiver
{
@Override
public void onReceive(Context context, Intent intent)
{
// 获取Intent中的update消息,update代表播放状态
int update = intent.getIntExtra("update", -1);
// 获取Intent中的current消息,current代表当前正在播放的歌曲
int current = intent.getIntExtra("current", -1);
if (current >= 0)
{
title.setText(titleStrs[current]);
author.setText(authorStrs[current]);
}
switch (update)
{
case 0x11:
play.setBackgroundResource(R.drawable.play);
status = 0x11;
break;
// 控制系统进入播放状态
case 0x12:
// 播放状态下设置使用暂停图标
play.setBackgroundResource(R.drawable.pause);
// 设置当前状态
status = 0x12;
break;
// 控制系统进入暂停状态
case 0x13:
// 暂停状态下设置使用播放图标
play.setBackgroundResource(R.drawable.play);
// 设置当前状态
status = 0x13;
break;
}
}
}
@Override
public void onClick(View source)
{
// 创建Intent
Intent intent = new Intent("org.crazyit.action.CTL_ACTION");
switch (source.getId())
{
// 按下播放/暂停按钮
case R.id.play:
intent.putExtra("control", 1);
break;
// 按下停止按钮
case R.id.stop:
intent.putExtra("control", 2);
break;
case R.id.next:
intent.putExtra("control", 3);
break;
case R.id.prev:
intent.putExtra("control", 4);
break;
}
// 发送广播,将被Service组件中的BroadcastReceiver接收到
sendBroadcast(intent);
}
}
MusicService.java
完成任务
在此接收监听按钮发送过来的广播,并根据所传来的信息对播放器进行相关操作(在此学习了媒体鹦音乐的播放),最后发送包含歌曲信息的广播给MainActibity。
对于课后的上下首歌切换用的类似于循环队列,例如上一首歌:current = (current+musics.length-1)%musics.length
代码编写
package com.example.maibenben.delta_vi_musicbox;
import android.app.Service;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.content.res.AssetFileDescriptor;
import android.content.res.AssetManager;
import android.media.MediaPlayer;
import android.media.MediaPlayer.OnCompletionListener;
import android.os.IBinder;
import java.io.IOException;
public class MusicService extends Service
{
MyReceiver serviceReceiver;
AssetManager am;
String[] musics = new String[] { "wish.mp3", "promise.mp3",
"beautiful.mp3" };
MediaPlayer mPlayer;
// 当前的状态,0x11代表没有播放;0x12代表正在播放;0x13代表暂停
int status = 0x11;
// 记录当前正在播放的音乐
int current = 0;
@Override
public IBinder onBind(Intent intent)
{
return null;
}
@Override
public void onCreate()
{
super.onCreate();
am = getAssets();
// 创建BroadcastReceiver
serviceReceiver = new MyReceiver();
// 创建IntentFilter
IntentFilter filter = new IntentFilter();
filter.addAction(MainActivity.CTL_ACTION);
registerReceiver(serviceReceiver, filter);
// 创建MediaPlayer
mPlayer = new MediaPlayer();
// 为MediaPlayer播放完成事件绑定监听器
mPlayer.setOnCompletionListener(new OnCompletionListener() // ①
{
@Override
public void onCompletion(MediaPlayer mp)
{
current++;
if (current >= 3)
{
current = 0;
}
//发送广播通知Activity更改文本框
Intent sendIntent = new Intent(MainActivity.UPDATE_ACTION);
sendIntent.putExtra("current", current);
// 发送广播,将被Activity组件中的BroadcastReceiver接收到
sendBroadcast(sendIntent);
// 准备并播放音乐
prepareAndPlay(musics[current]);
}
});
}
public class MyReceiver extends BroadcastReceiver
{
@Override
public void onReceive(final Context context, Intent intent)
{
int control = intent.getIntExtra("control", -1);
switch (control)
{
// 播放或暂停
case 1:
// 原来处于没有播放状态
if (status == 0x11)
{
// 准备并播放音乐
prepareAndPlay(musics[current]);
status = 0x12;
}
// 原来处于播放状态
else if (status == 0x12)
{
// 暂停
mPlayer.pause();
// 改变为暂停状态
status = 0x13;
}
// 原来处于暂停状态
else if (status == 0x13)
{
// 播放
mPlayer.start();
// 改变状态
status = 0x12;
}
break;
// 停止声音
case 2:
// 如果原来正在播放或暂停
if (status == 0x12 || status == 0x13)
{
// 停止播放
mPlayer.stop();
status = 0x11;
}
break;
case 3:
current = (current+1)%musics.length;
prepareAndPlay(musics[current]);
// 如果原来正在播放或暂停
if (status == 0x12 || status == 0x13)
{
// 停止播放
mPlayer.stop();
status = 0x11;
}
break;
case 4:
current = (current+musics.length-1)%musics.length;
prepareAndPlay(musics[current]);
// 如果原来正在播放或暂停
if (status == 0x12 || status == 0x13)
{
// 停止播放
mPlayer.stop();
status = 0x11;
}
break;
}
// 广播通知Activity更改图标、文本框
Intent sendIntent = new Intent(MainActivity.UPDATE_ACTION);
sendIntent.putExtra("update", status);
sendIntent.putExtra("current", current);
// 发送广播,将被Activity组件中的BroadcastReceiver接收到
sendBroadcast(sendIntent);
}
}
private void prepareAndPlay(String music)
{
try
{
// 打开指定音乐文件
AssetFileDescriptor afd = am.openFd(music);
mPlayer.reset();
// 使用MediaPlayer加载指定的声音文件。
mPlayer.setDataSource(afd.getFileDescriptor(),
afd.getStartOffset(), afd.getLength());
// 准备声音
mPlayer.prepare();
// 播放
mPlayer.start();
}
catch (IOException e)
{
e.printStackTrace();
}
}
}
main.xml
最后一个按钮的暂停QQ音乐找不到配套的。。有点不搭
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<LinearLayout
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">
<ImageView
android:layout_marginTop="30px"
android:layout_width="wrap_content"
android:layout_height="150px"
android:id="@+id/imageView"
android:src="@drawable/music"/>
<TextView
android:id="@+id/title"
android:layout_height="wrap_content"
android:textSize="20dp"
android:textColor="#000"
android:layout_width="wrap_content"
android:ellipsize="marquee"
android:layout_weight="1"
android:layout_gravity="center"
android:marqueeRepeatLimit="marquee_forever"/>
<TextView
android:id="@+id/author"
android:textSize="12dp"
android:layout_gravity="center"
android:layout_weight="1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>
</LinearLayout>
<LinearLayout
android:orientation="horizontal"
android:layout_width="match_parent"
android:layout_height="match_parent">
<ImageButton
android:id="@+id/prev"
android:layout_width="80px"
android:layout_height="80px"
android:background="@drawable/prev"/>
<ImageButton
android:id="@+id/play"
android:layout_width="80px"
android:layout_height="80px"
android:background="@drawable/play"/>
<ImageButton
android:id="@+id/next"
android:layout_width="80px"
android:layout_height="80px"
android:background="@drawable/next"/>
<ImageButton
android:id="@+id/stop"
android:layout_width="70px"
android:layout_height="70px"
android:src="@drawable/stop"/>
</LinearLayout>
</LinearLayout>
遇到过的问题
对于按上下首歌播放状态不匹配
对于点上下首歌后仍然需要状态判断,如果之前播放暂停,则切换完歌曲后此时也该暂停
current = (current+musics.length-1)%musics.length;
prepareAndPlay(musics[current]);
// 如果原来正在播放或暂停
if (status == 0x12 || status == 0x13)
{
// 停止播放
mPlayer.stop();
status = 0x11;
}
对于静态注册时的intent筛选
没有加过滤器(一开始以为可以收到所有的intent)导致没有接收到广播
<receiver android:name=".MyReceiver02" android:enabled="true" android:exported="true"> <intent-filter> <action android:name="com.example.maibenben.delta_vi_musicbox.borad" /> </intent-filter> </receiver>
对于图片资源的使用问题
不能用jpg的图片,然后png图片不能在as里直接将jpg改为png,好像要求格式不严谨,需要as外用图片软件保存为png再导入。
写这篇文章也是格外的艰难
今天不知道是我的电脑还是这个CSDN抽风了,文档一直保存失败。学习以后写文档还是得备份,不然丢失后心情难受(╯﹏╰)。后面用手机写的,格式有点问题,之后电脑能用了再修改。
结果截图