hunterliy小作品之 HunterMusic音乐播放器(Day3-自定义通知栏实现)

1.显示普通通知栏。新建一个 module,在界面上写两个按钮,一个显示通知,一个隐藏通知

MainActivity.java

 import android.app.Activity;
 import android.app.Notification;
 import android.app.NotificationManager;
 import android.app.PendingIntent;
 import android.content.Intent;
 import android.os.Bundle;
 import android.view.Menu;
 import android.view.MenuItem;
 import android.view.View;
 import android.widget.Toast;
 public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.activity_main);
 String msg=getIntent().getStringExtra("msg");
 if (msg!=null){
 Toast.makeText(MainActivity.this, msg, Toast.LENGTH_SHORT).show();
 }
 }
 public void show(View view){
 showNotification();
 }
 /** 显示通知栏*/
 private void showNotification() {
 Notification notification=new Notification(R.drawable.icon,"正在播放:
 dida",System.currentTimeMillis());
 notification.setLatestEventInfo(this, "温柔", "五月天", getPendingIntent());
 //不允许删除通知栏
 notification.flags=Notification.FLAG_ONGOING_EVENT;
 NotificationManager notificationManger= (NotificationManager)
 getSystemService(NOTIFICATION_SERVICE);
 notificationManger.notify(0,notification);
 }
 private PendingIntent getPendingIntent() {
 Intent intent=new Intent(this,MainActivity.class);
 intent.putExtra("msg","notification 发出的通知");
 return PendingIntent.getActivity(this,0,
 intent,PendingIntent.FLAG_UPDATE_CURRENT);
 }
 public void cancel(View view){
 cancelNotification();
 }
 /** 取消通知栏*/
 private void cancelNotification() {
 NotificationManager notificationManger= (NotificationManager)
 getSystemService(NOTIFICATION_SERVICE);
 notificationManger.cancel(0);
 }
 }

2.使用新的 API 显示通知栏

/** 使用新的 api*/
2. private Notification getNewNotification() {
3. Notification.Builder notification=new Notification.Builder(this);
 notification.setSmallIcon(R.drawable.icon);
 notification.setTicker("正在播放:温柔");
 notification.setOngoing(true);
 notification.setContentTitle("温柔");
 notification.setContentText("五月天");
 notification.setWhen(System.currentTimeMillis());
 notification.setContentIntent(getPendingIntent());
 return notification.getNotification();
 }

3.自定义通知栏。写一个通知栏的布局,分别填充上面的文字和点击事件

notification.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="wrap_content"
 android:padding="6dp"
 android:id="@+id/audioplay_ll_notification"
 android:gravity="center_vertical"
 android:orientation="horizontal">
 <ImageView
 android:layout_width="wrap_content"
 android:layout_height="wrap_content"
 android:src="@drawable/icon"/>
 <LinearLayout
 android:layout_width="0dp"
 android:layout_height="wrap_content"
 android:layout_weight="1"
 android:orientation="vertical">
 <TextView
 android:id="@+id/audioplay_tv_notification_name"
 android:layout_width="wrap_content"
 android:layout_height="wrap_content"
 android:text="歌曲名"
 android:textColor="@color/white"
 android:textSize="18sp"/>
 <TextView
 android:id="@+id/audioplay_tv_notification_artist"
 android:layout_marginTop="4dp"
 android:layout_width="wrap_content"
 android:layout_height="wrap_content"
 android:text="歌手名"
 android:textColor="@color/halfwhite"
 android:textSize="16sp"/>
 </LinearLayout>
 <ImageView
 android:id="@+id/audioplay_iv_notification_pre"
 android:layout_width="wrap_content"
 android:layout_height="wrap_content"
 android:layout_marginLeft="10dp"
 android:layout_marginRight="10dp"
 android:src="@drawable/icon_notification_pre"/>
 <ImageView
 android:id="@+id/audioplay_iv_notification_next"
 android:layout_width="wrap_content"
 android:layout_height="wrap_content"
 android:src="@drawable/icon_notification_next"/>
 </LinearLayout>

自定义通知栏代码的实现

/** 使用新的 API 自定义一个通知*/
 private Notification getCustomNotification() {
 Notification.Builder notification=new Notification.Builder(this);
 notification.setSmallIcon(R.drawable.icon);
 notification.setTicker("正在播放:我相信");
 notification.setOngoing(true);
 // notification.setWhen(System.currentTimeMillis());
 // notification.setContentTitle("北京北京");
 // notification.setContentText("汪峰");
 // notification.setContentIntent(getPendingIntent());、
 notification.setContent(getRemoteView());
 return notification.getNotification();
 }
 /** 自定义通知栏显示的布局*/
 private RemoteViews getRemoteView() {
 RemoteViews remoteViews=new
 RemoteViews(getPackageName(),R.layout.notification);
 //填充文字
 remoteViews.setTextViewText(R.id.audioplay_tv_notification_name,"我相信");
 remoteViews.setTextViewText(R.id.audioplay_tv_notification_artist,"unknow");
 //设置点击事件
 remoteViews.setOnClickPendingIntent(R.id.audioplay_ll_notification,
 getPendingIntent());
 remoteViews.setOnClickPendingIntent(R.id.audioplay_iv_notification_pre,
 getPrePendingIntent());
 remoteViews.setOnClickPendingIntent(R.id.audioplay_iv_notification_next,
 getNextPendingIntent());
 return remoteViews;
 }
 /** 下一首*/
 private PendingIntent getNextPendingIntent() {
 Intent intent=new Intent(this,MainActivity.class);
 intent.putExtra("msg", "下一首");
 return PendingIntent.getActivity(this,2,
 intent,PendingIntent.FLAG_UPDATE_CURRENT);
 }
 /** 上一首*/
 private PendingIntent getPrePendingIntent() {
 Intent intent=new Intent(this,MainActivity.class);
 intent.putExtra("msg", "上一首");
 //注意有多个 PendingIntent 的时候第二个参数不能一样
 return PendingIntent.getActivity(this,1,
 intent,PendingIntent.FLAG_UPDATE_CURRENT);
 }
 private PendingIntent getPendingIntent() {
 Intent intent=new Intent(this,MainActivity.class);
 intent.putExtra("msg", "notification 发出的通知");
 return PendingIntent.getActivity(this,0, intent,PendingIntent.FLAG_UPDATE_CURRENT);
 }

4.将自定义的通知集成到项目中去。将布局文件和代码直接拷贝到项目中,代码需要放在服务中。最后将歌名和歌手名的数据初始化一下

/** 音乐暂停*/
 public void pause(){
 mediaPlayer.pause();
 cancelNotification();
}
/** 音乐开始*/
 public void start(){
 mediaPlayer.start();
 showNotification();
 }
 private class OnAudioPreparedListener implements MediaPlayer.OnPreparedListener {
 @Override
 public void onPrepared(MediaPlayer mediaPlayer) {
 //准备完成,开始播放
 mediaPlayer.start();
 //服务发广播来更新界面的图片
 Intent intent=new Intent("com.itheima.mediaplay.updateplaybtn");
 //获取当前正在播放的音乐
 AudioItem audioItem=audioItems.get(position);
 intent.putExtra("audioItem",audioItem);
 sendBroadcast(intent);
 //显示通知
 showNotification();
 }
 }

5.通知栏中上一首和下一首功能的实现。 在 onStartCommand 中先去判断点击是不是通知栏发起的,然后再去进行操作

 private static final int NOTIFY_CONTENT = 0;
 private static final int NOTIFY_PRE = 1;
 private static final int NOTIFY_NEXT = 2;
 private static final String NOTIFY_TYPE = "notifyType";
  @Override
 public int onStartCommand(Intent intent, int flags, int startId) {
 int notifyType=intent.getIntExtra(NOTIFY_TYPE,-1);
 if (notifyType==-1){
 //不是通知栏发起的
 LogUtil.e(TAG,"AudioPlayService:onStartCommand");
 audioItems = (List<AudioItem>) intent.getSerializableExtra("audioItems");
 position = intent.getIntExtra("position", -1);
 binder.play();
 }else{
 //从通知栏点击发起
 switch (notifyType){
 case NOTIFY_NEXT:
 binder.next();
 break;
 case NOTIFY_PRE:
 binder.pre();
 break;
 }
 }
 return super.onStartCommand(intent, flags, startId);
 }
 /** 下一首*/
 private PendingIntent getNextPendingIntent() {
 Intent intent=new Intent(this,AudioPlayService.class);
 intent.putExtra(NOTIFY_TYPE, NOTIFY_NEXT);
 return PendingIntent.getService(this, 2, intent, PendingIntent.FLAG_UPDATE_CURRENT);
 }
 /** 上一首*/
 private PendingIntent getPrePendingIntent() {
 Intent intent=new Intent(this,AudioPlayService.class);
 intent.putExtra(NOTIFY_TYPE, NOTIFY_PRE);
 //注意有多个 PendingIntent 的时候第二个参数不能一样
 return PendingIntent.getService(this, 1, intent, PendingIntent.FLAG_UPDATE_CURRENT);
 }

6.点击通知栏的空白部分的处理。当点击空白区域的时候需要去开启播放音乐的界面,但是发现界面没有数据,所以我们需要将界面刷新一下,而且注意要将 AudioPlayerActivity 在清单文件中设为 SingleTop 模式,否则点击多次通知会弹出多个页面

@Override
 public int onStartCommand(Intent intent, int flags, int startId) {
 int notifyType=intent.getIntExtra(NOTIFY_TYPE,-1);
 //从通知栏点击返回的不是-1
 switch (notifyType){
 case NOTIFY_NEXT:
 binder.next();
 break;
 case NOTIFY_PRE:
 binder.pre();
 case NOTIFY_CONTENT:
 notifyUpdateUI();
 break;
 default:
 LogUtil.e(TAG,"AudioPlayService:onStartCommand");
 audioItems = (List<AudioItem>)
 intent.getSerializableExtra("audioItems");
 position = intent.getIntExtra("position", -1);
 binder.play();
 break;
 }
 return super.onStartCommand(intent, flags, startId);
 }
 private class OnAudioPreparedListener implements MediaPlayer.OnPreparedListener {
 @Override
 public void onPrepared(MediaPlayer mediaPlayer) {
 //准备完成,开始播放
 mediaPlayer.start();
 //更新界面
 notifyUpdateUI();
 //显示通知
 showNotification();
 }
 }
 /** 更新界面*/
 private void notifyUpdateUI() {
 //服务发广播来更新界面的图片
 Intent intent=new Intent("com.itheima.mediaplay.updateplaybtn");
 //获取当前正在播放的音乐
 AudioItem audioItem=audioItems.get(position);
 intent.putExtra("audioItem",audioItem);
 sendBroadcast(intent);
 }

7.重复播放不再重新开始播放。判断获取的 position 和之前记录的 position 是否相同,如果相同就是同一首歌,只需要重新加载数据,没有开始播放的话直接让他播放就可以了,如果不是同一首歌曲,将当前的 position 记录下来

@Override
 public int onStartCommand(Intent intent, int flags, int startId) {
 int notifyType=intent.getIntExtra(NOTIFY_TYPE,-1);
 //从通知栏点击返回的不是-1
 switch (notifyType){
 case NOTIFY_NEXT:
 binder.next();
 break;
 case NOTIFY_PRE:
 binder.pre();
 case NOTIFY_CONTENT:
 notifyUpdateUI();
 break;
 default:
 LogUtil.e(TAG,"AudioPlayService:onStartCommand");
 int positions = intent.getIntExtra("position", -1);
 if (positions==this.position){
 //重复播放
 notifyUpdateUI();
 if (!binder.isPlaying()){
 binder.start();
 }
 }else{
 audioItems = (List<AudioItem>)
 intent.getSerializableExtra("audioItems");
 this.position=positions;
 binder.play();
 }
 break;
 }
 return super.onStartCommand(intent, flags, startId);
 }

小结

这一篇主要是实现了自定义通知栏,大家也可以根据自己所要的界面来添加不同的东西,下一篇则是实现歌词显示,希望大家能继续看下去。

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值