Выпуск мультимедийных кнопок Android "O" (Oreo, 8)
Код для обработки мультимедийных кнопок из гарнитур, который я использую в своем приложении Text-to-Speech, прекрасно работает под Android API с 22 по 25 (в более старых версиях Android они обрабатываются другими, теперь устаревшими средствами). Однако под Android 8 "Oreo", как публичная бета-версия, так и финальная версия, она не работает. Вот соответствующий код:
Когда служба запускается, я создаю объект MediaSessionCompact:
mSession = new MediaSessionCompat(getApplicationContext(), "my.package.name._player_session");
mSession.setFlags(MediaSessionCompat.FLAG_HANDLES_MEDIA_BUTTONS | MediaSessionCompat.FLAG_HANDLES_TRANSPORT_CONTROLS);
mSession.setActive(true);
mSession.setCallback(myMediaSessionCallback);
PlaybackStateCompat state = new PlaybackStateCompat.Builder()
.setActions(ACTION_PLAY_PAUSE | ACTION_PLAY | ACTION_PAUSE |
ACTION_SKIP_TO_NEXT | ACTION_SKIP_TO_PREVIOUS |
ACTION_FAST_FORWARD | ACTION_REWIND
)
.setState(PlaybackStateCompat.STATE_PAUSED, 0 /*PlaybackStateCompat.PLAYBACK_POSITION_UNKNOWN*/, 1f)
.build();
mSession.setPlaybackState(state);
Конечно, определен обратный вызов медиа сеанса:
private MediaSessionCompat.Callback myMediaSessionCallback = new MediaSessionCompat.Callback() {
@Override
public boolean onMediaButtonEvent(Intent mediaButtonIntent) {
// The log output below never appears on "Oreo", nothing comes here.
Log.d(TAG, "callback onMediaButtonEvent() Compat");
MediaButtonIntentReceiver.handleIntent(mediaButtonIntent.getAction(), (KeyEvent) mediaButtonIntent.getParcelableExtra(Intent.EXTRA_KEY_EVENT));
return true;
}
@Override
public void onSkipToNext() {
//...
}
// etc. other overrides
};
Я также экспериментировал с PendingIntent, используя MediaButtonReceiver.buildMediaButtonPendingIntent() и установил mSession.setMediaButtonReceiver(pendingIntent) для всех интересующих меня действий, затем в моем сервисе onStartCommand() я вызываю MediaButtonReceiver.handleIntent(mSession,
// still in the same service:
mSession.setMediaButtonReceiver(
MediaButtonReceiver.buildMediaButtonPendingIntent(
this,
mMediaButtonReceiverComponentName,
ACTION_PLAY));
mSession.setMediaButtonReceiver(
MediaButtonReceiver.buildMediaButtonPendingIntent(
this,
mMediaButtonReceiverComponentName,
ACTION_PAUSE));
mSession.setMediaButtonReceiver(
MediaButtonReceiver.buildMediaButtonPendingIntent(
this,
mMediaButtonReceiverComponentName,
ACTION_PLAY_PAUSE));
и в сервисе onStartCommand():
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
// ...
if (intent != null) {
MediaButtonReceiver.handleIntent(mSession, intent);
// ...
}
return START_NOT_STICKY;
}
Ничего, совсем тупо для медиа кнопок нажимать на события. Что не так с "O" или моим кодом там??? Я полностью сбит с толку.
Обновление от 32.08.2017
Я также создал тривиальный, но работающий проект приложения, который демонстрирует проблему, см. По адресу: https://github.com/gregko/PlayerServiceSample. Этот проект отображает вывод LogCat при нажатии мультимедийной кнопки на гарнитуре под Android 5.x до 7.x, но полностью не работает под Android 8 "Oreo".
Обновление 9/1/2017 В настоящее время на Android-трекере существует открытая проблема, которую я отправил по адресу https://issuetracker.google.com/issues/65175978. Тем не менее кнопки мультимедиа работают в нескольких приложениях музыкального проигрывателя, которые я тестировал на Oreo, я просто не могу понять, что они делают по-другому, чтобы заставить их работать... Контекст моего приложения не воспроизводит музыку, а читает вслух текст с текстом к речевому сервису, поэтому большой код из примеров Music Player не применяется.
1 ответ
Решаемые. На странице Google " Изменения в поведении Android 8.0 " мы находим этот текст:
В Android 8.0 (уровень API 26) обработка событий мультимедийных кнопок отличается:
- Обработка мультимедийных кнопок в действии пользовательского интерфейса не изменилась: приоритетные действия по-прежнему имеют приоритет при обработке событий мультимедийных кнопок.
- Если действие переднего плана не обрабатывает событие мультимедийной кнопки, система направляет событие в приложение, которое воспроизводило аудио в последний раз локально. Активное состояние, флаги и состояние воспроизведения мультимедийного сеанса не учитываются при определении того, какое приложение получает события мультимедийных кнопок.
- Если сеанс мультимедиа приложения был отменен, система отправляет событие мультимедийной кнопки в MediaButtonReceiver приложения, если оно имеется.
- В любом другом случае система сбрасывает событие мультимедийной кнопки.
Все, что мне нужно было сделать, чтобы сделать мой тривиальный пример работы, это воспроизвести звук с помощью MediaPlayer. Видимо, воспроизведение звука с использованием API для преобразования текста в речь не подходит, что, на мой взгляд, является ошибкой.
Вот код, который я добавил к своему тривиальному образцу, чтобы он заработал, воспроизводя очень короткий и тихий WAV-файл из каталога Raw resources:
final MediaPlayer mMediaPlayer;
mMediaPlayer = MediaPlayer.create(this, R.raw.silent_sound);
mMediaPlayer.setOnCompletionListener(new MediaPlayer.OnCompletionListener() {
@Override
public void onCompletion(MediaPlayer mediaPlayer) {
mMediaPlayer.release();
}
});
mMediaPlayer.start();
Обновить
Отправил отчет об ошибке на устройство отслеживания проблем Android по адресу https://issuetracker.google.com/issues/65344811
Обновление 2, 10 октября 2017 г.
Google теперь говорит, что поведение Oreo в этом отношении "разработано" и не будет это исправлять. Прочитайте ответ ближе к концу сообщения трекера проблемы выше. Я должен сказать, что разочарован.