2013-03-20 4 views
22

Я пытаюсь отправить информацию о треке через A2DP/AVRCP. Прямо сейчас, музыка прекрасна, но на «приемнике» (то есть: автомобильном аудио) экран «Информация о дорожке» пуст (это не так, если использовать популярных игроков там). Любая идея?отправить информацию об отслеживании через A2DP/AVRCP

ответ

18

Этот код работает для меня:

private static final String AVRCP_PLAYSTATE_CHANGED = "com.android.music.playstatechanged"; 
private static final String AVRCP_META_CHANGED = "com.android.music.metachanged"; 

private void bluetoothNotifyChange(String what) { 
    Intent i = new Intent(what); 
    i.putExtra("id", Long.valueOf(getAudioId())); 
    i.putExtra("artist", getArtistName()); 
    i.putExtra("album",getAlbumName()); 
    i.putExtra("track", getTrackName()); 
    i.putExtra("playing", isPlaying());   
    i.putExtra("ListSize", getQueue()); 
    i.putExtra("duration", duration()); 
    i.putExtra("position", position()); 
    sendBroadcast(i); 
} 

вызова bluetoothNotifyChange с соответствующим намерением (определено выше), в зависимости от статуса воспроизведения: пауза/воспроизведения/метаданные изменились.

+0

Этот код работает для меня с моей Bluetooth-гарнитурой, но показывает только название дорожки ... Будет проверять больше. Спасибо. – DavyJonesUA

+0

Этот код не работал для меня на гарнитуре Sony, но приложение Google Music по-прежнему может отображать информацию о треке на этой гарнитуре. – Wayne

+0

Можете ли вы дать мне образец ссылки с доступными метаданными? – Wayne

1

Чтобы отправить метаданные трека в головной блок, вам необходимо отправить намерение.

Intent avrcp = new Intent("com.android.music.metachanged"); 
avrcp.putExtra("track", "song title"); 
avrcp.putExtra("artist", "artist name"); 
avrcp.putExtra("album", "album name"); 
Context.sendBroadcast(avrcp); 

Когда песня исполнена, отправьте другое намерение пустым строкам для второго параметра метода putExtra.

+1

В то время как я не нашел еще один способ сделать это. Это похоже на хакерский подход. Похоже, вы полагаетесь на музыкальное приложение google, чтобы справиться с этим намерением. Как это делает музыкальное приложение google? –

+0

Действительно. Возможно ли, что музыка google является единственной, которая может использовать avrcp? Все приложения должны иметь эту возможность через RemoteControlClient? –

6

Это заняло у меня навсегда, чтобы понять. Просто трансляция намерения не сработала. Я получил AVRCP работать, посылая намерения И реализующих RemoteControlClient

Вот код, который я использовал:

public void onCreate(){ 
    super.onCreate(); 

    mAudioManager = (AudioManager) getSystemService(Context.AUDIO_SERVICE); 
    ComponentName rec = new ComponentName(getPackageName(), MyReceiver.class.getName()); 
    mAudioManager.registerMediaButtonEventReceiver(rec); 

    Intent i = new Intent(Intent.ACTION_MEDIA_BUTTON); 
    i.setComponent(rec); 
    PendingIntent pi = PendingIntent.getBroadcast(this, 0, i, 0); 
    mRemoteControlClient = new RemoteControlClient(pi); 
    mAudioManager.registerRemoteControlClient(mRemoteControlClient); 

    int flags = RemoteControlClient.FLAG_KEY_MEDIA_PREVIOUS 
      | RemoteControlClient.FLAG_KEY_MEDIA_NEXT 
      | RemoteControlClient.FLAG_KEY_MEDIA_PLAY 
      | RemoteControlClient.FLAG_KEY_MEDIA_PAUSE 
      | RemoteControlClient.FLAG_KEY_MEDIA_PLAY_PAUSE 
      | RemoteControlClient.FLAG_KEY_MEDIA_STOP 
      | RemoteControlClient.FLAG_KEY_MEDIA_FAST_FORWARD 
      | RemoteControlClient.FLAG_KEY_MEDIA_REWIND; 
    mRemoteControlClient.setTransportControlFlags(flags); 
} 

private void onTrackChanged(...) { 
    String title = ...; 
    String artist = ...; 
    String album = ...; 
    long duration = ...; 

    Intent i = new Intent("com.android.music.metachanged"); 
    i.putExtra("id", 1); 
    i.putExtra("track", title); 
    i.putExtra("artist", artist); 
    i.putExtra("album", album); 
    i.putExtra("playing", "true"); 
    sendStickyBroadcast(i); 

    RemoteControlClient.MetadataEditor ed = mRemoteControlClient.editMetadata(true); 
    ed.putString(MediaMetadataRetriever.METADATA_KEY_TITLE, title); 
    ed.putString(MediaMetadataRetriever.METADATA_KEY_ALBUM, album); 
    ed.putString(MediaMetadataRetriever.METADATA_KEY_ARTIST, artist); 
    ed.putLong(MediaMetadataRetriever.METADATA_KEY_DURATION, track.getDuration()); 
    ed.apply(); 
} 

public void onDestroy(){ 
    mAudioManager.unregisterRemoteControlClient(mRemoteControlClient); 
    super.onDestroy(); 
} 
+0

вы можете объяснить строку 'ComponentName rec = new ComponentName (getPackageName(), MyReceiver.class.getName());' – Distwo

+0

Вот как вы смотрите BroadcastReceiver с помощью отражения ... –

+0

Я пробовал вам код, который действительно соответствует тому, что мы можем найти в https://android.googlesource.com/platform/packages/apps/Music/+/android -5.1.1_r13/src/com/android/music/MediaPlaybackService.java, но это не работает. Не может быть, потому что нам нужно иметь дело с AudioManager.AUDIOFOCUS? –

10

Если вы просто хотите, чтобы отправить информацию о метаданных с телефона подключенного AVRCP совместимого аудио устройства Bluetooth и НЕ ЗАПУСКАЙТЕ, чтобы управлять своим приложением с устройства Bluetooth вообще, вы можете найти код ниже полезного. И есть NO необходимо реализовать и зарегистрировать MediaButtonEventReceiver с AudioManager.

Я также включил код для API версии 21 (LOLLIPOP, 5.0). Из API 21 использование RemoteControlClient устарело и рекомендуется использовать MediaSession.

Init фаза:

if (mAudioManager == null) { 
     mAudioManager = (AudioManager) getSystemService(Context.AUDIO_SERVICE); 
    } 

    if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) { 
     if (mRemoteControlClient == null) { 
      Log.d("init()", "API " + Build.VERSION.SDK_INT + " lower then " + Build.VERSION_CODES.LOLLIPOP); 
      Log.d("init()", "Using RemoteControlClient API."); 

      mRemoteControlClient = new RemoteControlClient(PendingIntent.getBroadcast(this, 0, new Intent(Intent.ACTION_MEDIA_BUTTON), 0)); 
      mAudioManager.registerRemoteControlClient(mRemoteControlClient); 
     } 
    } else { 
     if (mMediaSession == null) { 
      Log.d("init()", "API " + Build.VERSION.SDK_INT + " greater or equals " + Build.VERSION_CODES.LOLLIPOP); 
      Log.d("init()", "Using MediaSession API."); 

      mMediaSession = new MediaSession(this, "PlayerServiceMediaSession"); 
      mMediaSession.setFlags(MediaSession.FLAG_HANDLES_TRANSPORT_CONTROLS); 
      mMediaSession.setActive(true); 

     } 
    } 

Способ передачи информации метаданных песни AVRCP совместимого Bluetooth Audio устройство:

private void onTrackChanged(String title, String artist, String album, long duration, long position, long trackNumber) { 

    if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) { 

     RemoteControlClient.MetadataEditor ed = mRemoteControlClient.editMetadata(true); 
     ed.putString(MediaMetadataRetriever.METADATA_KEY_TITLE, title); 
     ed.putString(MediaMetadataRetriever.METADATA_KEY_ARTIST, artist); 
     ed.putString(MediaMetadataRetriever.METADATA_KEY_ALBUM, album); 
     ed.putLong(MediaMetadataRetriever.METADATA_KEY_DURATION, duration); 
     ed.putLong(MediaMetadataRetriever.METADATA_KEY_CD_TRACK_NUMBER, trackNumber); 
     ed.apply(); 

     mRemoteControlClient.setPlaybackState(RemoteControlClient.PLAYSTATE_PLAYING, position, 1.0f); 
    } else { 

     MediaMetadata metadata = new MediaMetadata.Builder() 
       .putString(MediaMetadata.METADATA_KEY_TITLE, title) 
       .putString(MediaMetadata.METADATA_KEY_ARTIST, artist) 
       .putString(MediaMetadata.METADATA_KEY_ALBUM, album) 
       .putLong(MediaMetadata.METADATA_KEY_DURATION, duration) 
       .putLong(MediaMetadata.METADATA_KEY_TRACK_NUMBER, trackNumber) 
       .build(); 

     mMediaSession.setMetadata(metadata); 

     PlaybackState state = new PlaybackState.Builder() 
       .setActions(PlaybackState.ACTION_PLAY) 
       .setState(PlaybackState.STATE_PLAYING, position, 1.0f, SystemClock.elapsedRealtime()) 
       .build(); 

     mMediaSession.setPlaybackState(state); 
    } 
} 

вызова, если изменения метаданных, но проверить, если у нас есть подключение A2DP к аудио Bluetooth устройство. Нет необходимости посылать информацию метаданных, если мы не связаны:

if (mAudioManager.isBluetoothA2dpOn()) { 
    Log.d("AudioManager", "isBluetoothA2dpOn() = true"); 
    onTrackChanged(getTitle(), getArtist(), getAlbum(), getDuration(), getCurrentPosition(), getId()); 
} 

Очистка на уничтожить:

@Override 
public void onDestroy() { 
    super.onDestroy(); 

[..]  

if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) { 
     mAudioManager.unregisterRemoteControlClient(mRemoteControlClient); 
    } else { 
     mMediaSession.release(); 
    } 
} 

This как это выглядит на моей машине стерео

+0

как насчет режима повтора в случайном порядке? Вы тоже справлялись с этим? спасибо – issamux

+0

@issamux Взгляните на этот http://stackoverflow.com/questions/31194180/how-do-i-make-shuffle-playlist-button-and-repeat-button-in-android-studio –

0

Вам не нужно для управления SDK_INT, если вы используете версию компонентов совместимости. Ниже код протестирован на многих автомобильных устройствах Bluetooth и работает как шарм. Некоторые устройства не понимают некоторых КЛЮЧЕЙ, поэтому лучше использовать KEY. Reference. Не забывайте.строить() после того, как putBitmap не ранее

public static void sendTrackInfo() { 
if(audioManager == null) { 
    audioManager = (AudioManager) getSystemService(Context.AUDIO_SERVICE); 
} 

if (mMediaSession == null) { 
    mMediaSession = new MediaSessionCompat(this, "PlayerServiceMediaSession"); 
    mMediaSession.setFlags(MediaSessionCompat.FLAG_HANDLES_TRANSPORT_CONTROLS); 
    mMediaSession.setActive(true); 
} 

if (audioManager.isBluetoothA2dpOn()) { 
    try { 
     String songTitle = getTitle(); 
     String artistTitle = getArtist(); 
     String radioImageUri = getImagesArr().get(0); 
     String songImageUri = getImagesArr().get(1); 
     long duration = getDuration(); 

     final MediaMetadataCompat.Builder metadata = new MediaMetadataCompat.Builder(); 

     metadata.putString(MediaMetadataCompat.METADATA_KEY_TITLE, songTitle); 
     metadata.putString(MediaMetadataCompat.METADATA_KEY_DISPLAY_TITLE, songTitle); 
     metadata.putString(MediaMetadataCompat.METADATA_KEY_ARTIST, artistTitle); 
     metadata.putString(MediaMetadataCompat.METADATA_KEY_ALBUM_ARTIST, artistTitle); 
     metadata.putString(MediaMetadataCompat.METADATA_KEY_ART_URI, radioImageUri); 
     metadata.putString(MediaMetadataCompat.METADATA_KEY_DISPLAY_ICON_URI, radioImageUri); 
     metadata.putString(MediaMetadataCompat.METADATA_KEY_ALBUM_ART_URI, songImageUri); 
     metadata.putLong(MediaMetadataCompat.METADATA_KEY_DURATION, duration); 

     imageCounter = 0; 

     Glide.with(act) 
       .load(Uri.parse(radioImageUri)) 
       .asBitmap() 
       .into(new SimpleTarget<Bitmap>(250, 250) { 
        @Override 
        public void onResourceReady(Bitmap bitmap, GlideAnimation anim) { 
         metadata.putBitmap(MediaMetadataCompat.METADATA_KEY_ART, bitmap); 
         metadata.putBitmap(MediaMetadataCompat.METADATA_KEY_DISPLAY_ICON, bitmap); 

         imageCounter = imageCounter + 1; 

         if(imageCounter == 2) { 
          mMediaSession.setMetadata(metadata.build()); 
         } 
        } 
       }); 

     Glide.with(act) 
       .load(Uri.parse(songImageUri)) 
       .asBitmap() 
       .into(new SimpleTarget<Bitmap>(250, 250) { 
        @Override 
        public void onResourceReady(Bitmap bitmap, GlideAnimation anim) { 
         metadata.putBitmap(MediaMetadataCompat.METADATA_KEY_ALBUM_ART, bitmap); 

         imageCounter = imageCounter + 1; 

         if(imageCounter == 2) { 
          mMediaSession.setMetadata(metadata.build()); 
         } 
        } 
       }); 
    } 
    catch (JSONException e) { 
     e.printStackTrace(); 
    } 
} 

}