Android媒体通知栏多系统适配怎么实现

其他教程   发布日期:2024年12月11日   浏览次数:238

今天小编给大家分享一下Android媒体通知栏多系统适配怎么实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。

需要考虑的问题如下:

1,通知栏适配,音乐播放需要常驻,所以要维护一个通知栏。

2,音控处理,在安卓7.0及以下,通过MediaSessionCompat可控制锁屏页音乐播放。

3,对于耳机的处理,不管是线耳机还是蓝牙耳机,耳机控制播放暂停,下一曲上一曲等操作。

4,打电话处理,在听音乐的同时如果电话进来后挂断,希望可以自动播放。

5,音频播放焦点处理,如果有别的应用抢占焦点可进行暂停播放。还有就是进入APP时想拥有音频焦点,都可以通过AudioManager进行处理。

实现方式

创建通知管理类NotifyBuilderManager代码如下:

  1. package com.idujing.myapplication.manager;
  2. import android.annotation.SuppressLint;
  3. import android.app.Notification;
  4. import android.app.NotificationChannel;
  5. import android.app.NotificationManager;
  6. import android.app.PendingIntent;
  7. import android.app.Service;
  8. import android.content.Context;
  9. import android.content.Intent;
  10. import android.os.Build;
  11. import androidx.core.app.NotificationCompat;
  12. import com.idujing.myapplication.R;
  13. /**
  14. * 音频播放通知栏管理
  15. */
  16. public class NotifyBuilderManager {
  17. private final String TAG = getClass().getSimpleName();
  18. public static final String ACTION_NEXT = "com.idujing.play.notify.next";// 下一首
  19. public static final String ACTION_PREV = "com.idujing.play.notify.prev";// 上一首
  20. public static final String ACTION_PLAY_PAUSE = "com.idujing.play.notify.play_state";// 播放暂停广播
  21. private static final int NOTIFICATION_ID = 0x123;
  22. private Service mContext;
  23. private Notification mNotification;
  24. private NotificationManager mNotificationManager;
  25. private NotificationCompat.Builder mNotificationBuilder;
  26. private MediaSessionManager mSessionManager;
  27. private PendingIntent mPendingPlay;
  28. private PendingIntent mPendingPre;
  29. private PendingIntent mPendingNext;
  30. private boolean isRunningForeground = false;
  31. public boolean isRunningForeground() {
  32. return isRunningForeground;
  33. }
  34. public NotifyBuilderManager(Service context) {
  35. this.mContext = context;
  36. mSessionManager = new MediaSessionManager(context, null);
  37. }
  38. /**
  39. * 初始化通知栏
  40. */
  41. private void initNotify() {
  42. mNotificationManager = (NotificationManager) mContext.getSystemService(Context.NOTIFICATION_SERVICE);
  43. Class<?> clazz = null;
  44. try {
  45. clazz = Class.forName("具体的播放器类名");
  46. } catch (ClassNotFoundException e) {
  47. e.printStackTrace();
  48. }
  49. // 适配12.0及以上
  50. int flag;
  51. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.S) {
  52. flag = PendingIntent.FLAG_IMMUTABLE;
  53. } else {
  54. flag = PendingIntent.FLAG_UPDATE_CURRENT;
  55. }
  56. //绑定事件通过创建的具体广播去接收即可。
  57. Intent infoIntent = new Intent(mContext, clazz);
  58. PendingIntent pendingInfo = PendingIntent.getActivity(mContext, 0, infoIntent, flag);
  59. Intent preIntent = new Intent();
  60. preIntent.setAction(ACTION_PREV);
  61. mPendingPre = PendingIntent.getBroadcast(mContext, 1, preIntent, flag);
  62. Intent playIntent = new Intent();
  63. playIntent.setAction(ACTION_PLAY_PAUSE);
  64. mPendingPlay = PendingIntent.getBroadcast(mContext, 2, playIntent, flag);
  65. Intent nextIntent = new Intent();
  66. nextIntent.setAction(ACTION_NEXT);
  67. mPendingNext = PendingIntent.getBroadcast(mContext, 3, nextIntent, PendingIntent.FLAG_IMMUTABLE);
  68. androidx.media.app.NotificationCompat.MediaStyle style = new androidx.media.app.NotificationCompat.MediaStyle()
  69. .setShowActionsInCompactView(0, 1, 2)
  70. .setMediaSession(mSessionManager.getMediaSession());
  71. mNotificationBuilder = new NotificationCompat.Builder(mContext, initChannelId())
  72. .setSmallIcon(R.mipmap.ic_launcher)
  73. .setPriority(NotificationCompat.PRIORITY_MAX)
  74. .setContentIntent(pendingInfo)
  75. .setVisibility(NotificationCompat.VISIBILITY_PUBLIC)
  76. .setStyle(style);
  77. isRunningForeground = true;
  78. }
  79. /**
  80. * 创建Notification ChannelID
  81. *
  82. * @return 频道id
  83. */
  84. private String initChannelId() {
  85. // 通知渠道的id
  86. String id = "music_01";
  87. // 用户可以看到的通知渠道的名字.
  88. CharSequence name = mContext.getString(R.string.app_name);
  89. // 用户可以看到的通知渠道的描述
  90. String description = "通知栏播放控制";
  91. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
  92. int importance = NotificationManager.IMPORTANCE_LOW;
  93. NotificationChannel channel = new NotificationChannel(id, name, importance);
  94. channel.setDescription(description);
  95. channel.enableLights(false);
  96. channel.enableVibration(false);
  97. mNotificationManager.createNotificationChannel(channel);
  98. }
  99. return id;
  100. }
  101. /**
  102. * 取消通知
  103. */
  104. public void cancelNotification() {
  105. if (mNotificationManager != null) {
  106. mContext.stopForeground(true);
  107. mNotificationManager.cancel(NOTIFICATION_ID);
  108. isRunningForeground = false;
  109. }
  110. }
  111. /**
  112. * 设置通知栏大图片
  113. */
  114. private void updateCoverSmall() {
  115. Glide.with(mContext).asBitmap()
  116. .load(url)
  117. .into(new CustomTarget<Bitmap>() {
  118. @Override
  119. public void onResourceReady(@NonNull Bitmap resource, @Nullable Transition<? super Bitmap> transition) {
  120. mNotificationBuilder.setLargeIcon(resource);
  121. mNotification = mNotificationBuilder.build();
  122. mNotificationManager.notify(NOTIFICATION_ID, mNotification);
  123. }
  124. @Override
  125. public void onLoadCleared(@Nullable Drawable placeholder) {
  126. }
  127. @Override
  128. public void onLoadFailed(@Nullable Drawable errorDrawable) {
  129. super.onLoadFailed(errorDrawable);
  130. Log.e(TAG, "onLoadFailed: ");
  131. }
  132. });
  133. }
  134. /**
  135. * 更新状态栏通知
  136. */
  137. @SuppressLint("RestrictedApi")
  138. public void updateNotification(boolean isMusicPlaying) {
  139. if (mNotification == null) {
  140. initNotify();
  141. }
  142. mSessionManager.updateMetaData();
  143. if (mNotificationBuilder != null) {
  144. int playButtonResId = isMusicPlaying
  145. ? android.R.drawable.ic_media_pause : android.R.drawable.ic_media_play;
  146. if (!mNotificationBuilder.mActions.isEmpty()) {
  147. mNotificationBuilder.mActions.clear();
  148. }
  149. mNotificationBuilder
  150. .addAction(android.R.drawable.ic_media_previous, "Previous", mPendingPre) // #0
  151. .addAction(playButtonResId, "Pause", mPendingPlay) // #1
  152. .addAction(android.R.drawable.ic_media_next, "Next", mPendingNext);
  153. mNotificationBuilder.setContentTitle("主标题");
  154. mNotificationBuilder.setContentText("副标题");
  155. updateCoverSmall();
  156. mNotification = mNotificationBuilder.build();
  157. mContext.startForeground(NOTIFICATION_ID, mNotification);
  158. mNotificationManager.notify(NOTIFICATION_ID, mNotification);
  159. }
  160. }
  161. }

创建音控管理类MediaSessionManager代码如下:

  1. package com.idujing.myapplication.manager;
  2. import android.content.Context;
  3. import android.os.Handler;
  4. import android.support.v4.media.MediaMetadataCompat;
  5. import android.support.v4.media.session.MediaSessionCompat;
  6. import android.support.v4.media.session.PlaybackStateCompat;
  7. /**
  8. * 主要管理Android 5.0以后线控和蓝牙远程控制播放
  9. */
  10. public class MediaSessionManager {
  11. private static final String TAG = "MediaSessionManager";
  12. //指定可以接收的来自锁屏页面的按键信息
  13. private static final long MEDIA_SESSION_ACTIONS =
  14. PlaybackStateCompat.ACTION_PLAY
  15. | PlaybackStateCompat.ACTION_PAUSE
  16. | PlaybackStateCompat.ACTION_PLAY_PAUSE
  17. | PlaybackStateCompat.ACTION_SKIP_TO_NEXT
  18. | PlaybackStateCompat.ACTION_SKIP_TO_PREVIOUS
  19. | PlaybackStateCompat.ACTION_STOP
  20. | PlaybackStateCompat.ACTION_SEEK_TO;
  21. private final Context mContext;
  22. private MediaSessionCompat mMediaSession;
  23. private Handler mHandler;
  24. public MediaSessionManager(Context context, Handler handler) {
  25. this.mContext = context;
  26. this.mHandler = handler;
  27. setupMediaSession();
  28. }
  29. /**
  30. * 是否在播放
  31. *
  32. * @return
  33. */
  34. protected boolean isPlaying() {
  35. //具体去实现
  36. return false;
  37. }
  38. /**
  39. * 初始化并激活 MediaSession
  40. */
  41. private void setupMediaSession() {
  42. mMediaSession = new MediaSessionCompat(mContext, TAG);
  43. //指明支持的按键信息类型
  44. mMediaSession.setFlags(
  45. MediaSessionCompat.FLAG_HANDLES_MEDIA_BUTTONS |
  46. MediaSessionCompat.FLAG_HANDLES_TRANSPORT_CONTROLS
  47. );
  48. mMediaSession.setCallback(callback, mHandler);
  49. mMediaSession.setActive(true);
  50. }
  51. /**
  52. * 更新正在播放的音乐信息,切换歌曲时调用
  53. */
  54. public void updateMetaData() {
  55. MediaMetadataCompat.Builder metaDta = new MediaMetadataCompat.Builder()
  56. .putString(MediaMetadataCompat.METADATA_KEY_TITLE, "title")
  57. .putString(MediaMetadataCompat.METADATA_KEY_ARTIST, "Artist")
  58. .putString(MediaMetadataCompat.METADATA_KEY_ALBUM, "Album")
  59. .putString(MediaMetadataCompat.METADATA_KEY_ALBUM_ARTIST, "Artist")
  60. .putLong(MediaMetadataCompat.METADATA_KEY_DURATION, 100);
  61. mMediaSession.setMetadata(metaDta.build());
  62. int state = isPlaying() ? PlaybackStateCompat.STATE_PLAYING :
  63. PlaybackStateCompat.STATE_PAUSED;
  64. mMediaSession.setPlaybackState(new PlaybackStateCompat.Builder()
  65. .setActions(MEDIA_SESSION_ACTIONS)
  66. .setState(state, 0, 1)
  67. .build());
  68. //锁屏页封面设置,高本版没有效果,因为通知栏权限调整。
  69. Glide.with(mContext).asBitmap().
  70. load(url)
  71. .into(new CustomTarget<Bitmap>() {
  72. @Override
  73. public void onResourceReady(@NonNull Bitmap resource, @Nullable Transition<? super Bitmap> transition) {
  74. metaDta.putBitmap(MediaMetadataCompat.METADATA_KEY_ALBUM_ART, resource);
  75. mMediaSession.setMetadata(metaDta.build());
  76. }
  77. @Override
  78. public void onLoadCleared(@Nullable Drawable placeholder) {
  79. }
  80. });
  81. }
  82. public MediaSessionCompat.Token getMediaSession() {
  83. return mMediaSession.getSessionToken();
  84. }
  85. /**
  86. * 释放MediaSession,退出播放器时调用
  87. */
  88. public void release() {
  89. mMediaSession.setCallback(null);
  90. mMediaSession.setActive(false);
  91. mMediaSession.release();
  92. }
  93. /**
  94. * API 21 以上 耳机多媒体按钮监听 MediaSessionCompat.Callback
  95. */
  96. private MediaSessionCompat.Callback callback = new MediaSessionCompat.Callback() {
  97. @Override
  98. public void onPlay() {
  99. //具体自己实现
  100. }
  101. @Override
  102. public void onPause() {
  103. }
  104. @Override
  105. public void onSkipToNext() {
  106. }
  107. @Override
  108. public void onSkipToPrevious() {
  109. }
  110. @Override
  111. public void onStop() {
  112. }
  113. @Override
  114. public void onSeekTo(long pos) {
  115. }
  116. };
  117. }

创建音频焦点控制类AudioAndFocusManager

通过音频焦点控制,不管是别的应用抢占焦点,还是打电话都可以接收到状态。

  1. package com.idujing.myapplication.manager;
  2. import android.content.Context;
  3. import android.media.AudioFocusRequest;
  4. import android.media.AudioManager;
  5. import android.os.Build;
  6. import android.util.Log;
  7. import androidx.annotation.RequiresApi;
  8. /**
  9. * Description: 主要用来管理音频焦点
  10. */
  11. public class AudioAndFocusManager {
  12. private static final String TAG = "AudioAndFocusManager";
  13. private AudioManager mAudioManager;
  14. @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
  15. public AudioAndFocusManager(Context mContext) {
  16. mAudioManager = (AudioManager) mContext.getSystemService(Context.AUDIO_SERVICE);
  17. }
  18. /**
  19. * 请求音频焦点
  20. */
  21. public void requestAudioFocus() {
  22. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
  23. AudioFocusRequest mAudioFocusRequest = new AudioFocusRequest.Builder(AudioManager.AUDIOFOCUS_GAIN)
  24. .setOnAudioFocusChangeListener(audioFocusChangeListener)
  25. .build();
  26. int res = mAudioManager.requestAudioFocus(mAudioFocusRequest);
  27. if (res == 1) {
  28. Log.e(TAG, "res=" + true);
  29. }
  30. } else {
  31. if (audioFocusChangeListener != null) {
  32. boolean result = AudioManager.AUDIOFOCUS_REQUEST_GRANTED ==
  33. mAudioManager.requestAudioFocus(audioFocusChangeListener,
  34. AudioManager.STREAM_MUSIC,
  35. AudioManager.AUDIOFOCUS_GAIN);
  36. Log.e(TAG, "requestAudioFocus result=" + result);
  37. }
  38. }
  39. }
  40. /**
  41. * 关闭音频焦点
  42. */
  43. public void abandonAudioFocus() {
  44. if (audioFocusChangeListener != null) {
  45. boolean result = AudioManager.AUDIOFOCUS_REQUEST_GRANTED ==
  46. mAudioManager.abandonAudioFocus(audioFocusChangeListener);
  47. Log.e(TAG, "abandonAudioFocus result=" + result);
  48. }
  49. }
  50. /**
  51. * 音频焦点改变监听器
  52. */
  53. private AudioManager.OnAudioFocusChangeListener audioFocusChangeListener = focusChange -> {
  54. switch (focusChange) {
  55. case AudioManager.AUDIOFOCUS_LOSS://失去音频焦点
  56. case AudioManager.AUDIOFOCUS_LOSS_TRANSIENT://暂时失去焦点
  57. break;
  58. case AudioManager.AUDIOFOCUS_GAIN://获取焦点
  59. break;
  60. default:
  61. }
  62. };
  63. }

以上就是Android媒体通知栏多系统适配怎么实现的详细内容,更多关于Android媒体通知栏多系统适配怎么实现的资料请关注九品源码其它相关文章!