Skip to content

Commit 069cdf6

Browse files
committed
Media browser interface to show playlists on Android Auto
1 parent 9bdc970 commit 069cdf6

File tree

1 file changed

+213
-5
lines changed

1 file changed

+213
-5
lines changed
Original file line numberDiff line numberDiff line change
@@ -1,36 +1,59 @@
11
package org.schabi.newpipe.player.mediabrowser;
22

3+
import android.net.Uri;
34
import android.os.Bundle;
4-
import android.support.v4.media.MediaBrowserCompat;
5+
import android.os.ResultReceiver;
56
import android.support.v4.media.MediaBrowserCompat.MediaItem;
7+
import android.support.v4.media.MediaDescriptionCompat;
68
import android.support.v4.media.session.MediaSessionCompat;
9+
import android.support.v4.media.session.PlaybackStateCompat;
710
import android.util.Log;
811

912
import androidx.annotation.NonNull;
1013
import androidx.annotation.Nullable;
14+
import androidx.annotation.StringRes;
1115
import androidx.media.MediaBrowserServiceCompat;
16+
import androidx.media.utils.MediaConstants;
1217

18+
import com.google.android.exoplayer2.Player;
1319
import com.google.android.exoplayer2.ext.mediasession.MediaSessionConnector;
1420

21+
import org.schabi.newpipe.NewPipeDatabase;
22+
import org.schabi.newpipe.R;
23+
import org.schabi.newpipe.database.AppDatabase;
24+
import org.schabi.newpipe.database.playlist.PlaylistMetadataEntry;
25+
import org.schabi.newpipe.database.playlist.PlaylistStreamEntry;
26+
import org.schabi.newpipe.local.playlist.LocalPlaylistManager;
1527
import org.schabi.newpipe.player.PlayerService;
28+
import org.schabi.newpipe.player.playqueue.PlayQueue;
29+
import org.schabi.newpipe.player.playqueue.SinglePlayQueue;
30+
import org.schabi.newpipe.util.NavigationHelper;
1631

1732
import java.util.ArrayList;
1833
import java.util.List;
34+
import java.util.stream.Collectors;
1935

36+
import io.reactivex.rxjava3.android.schedulers.AndroidSchedulers;
2037
import io.reactivex.rxjava3.core.Single;
38+
import io.reactivex.rxjava3.disposables.Disposable;
2139

22-
public class MediaBrowserConnector {
40+
public class MediaBrowserConnector implements MediaSessionConnector.PlaybackPreparer {
2341
private static final String TAG = MediaBrowserConnector.class.getSimpleName();
2442

2543
private final PlayerService playerService;
2644
private final @NonNull MediaSessionConnector sessionConnector;
2745
private final @NonNull MediaSessionCompat mediaSession;
2846

47+
private AppDatabase database;
48+
private LocalPlaylistManager localPlaylistManager;
49+
private Disposable prepareOrPlayDisposable;
50+
2951
public MediaBrowserConnector(@NonNull final PlayerService playerService) {
3052
this.playerService = playerService;
3153
mediaSession = new MediaSessionCompat(playerService, TAG);
3254
sessionConnector = new MediaSessionConnector(mediaSession);
3355
sessionConnector.setMetadataDeduplicationEnabled(true);
56+
sessionConnector.setPlaybackPreparer(this);
3457
playerService.setSessionToken(mediaSession.getSessionToken());
3558
}
3659

@@ -39,11 +62,58 @@ public MediaBrowserConnector(@NonNull final PlayerService playerService) {
3962
}
4063

4164
public void release() {
65+
disposePrepareOrPlayCommands();
4266
mediaSession.release();
4367
}
4468

4569
@NonNull
46-
private static final String MY_MEDIA_ROOT_ID = "media_root_id";
70+
private static final String ID_ROOT = "//${BuildConfig.APPLICATION_ID}/r";
71+
@NonNull
72+
private static final String ID_BOOKMARKS = ID_ROOT + "/playlists";
73+
74+
private MediaItem createRootMediaItem(final String mediaId, final String folderName) {
75+
final var builder = new MediaDescriptionCompat.Builder();
76+
builder.setMediaId(mediaId);
77+
builder.setTitle(folderName);
78+
79+
final var extras = new Bundle();
80+
extras.putString(MediaConstants.DESCRIPTION_EXTRAS_KEY_CONTENT_STYLE_GROUP_TITLE,
81+
"NewPipe");
82+
builder.setExtras(extras);
83+
return new MediaItem(builder.build(), MediaItem.FLAG_BROWSABLE);
84+
}
85+
86+
private MediaItem createPlaylistMediaItem(final PlaylistMetadataEntry playlist) {
87+
final var builder = new MediaDescriptionCompat.Builder();
88+
builder.setMediaId(createMediaIdForPlaylist(playlist.uid))
89+
.setTitle(playlist.name)
90+
.setIconUri(Uri.parse(playlist.thumbnailUrl));
91+
92+
final var extras = new Bundle();
93+
extras.putString(MediaConstants.DESCRIPTION_EXTRAS_KEY_CONTENT_STYLE_GROUP_TITLE,
94+
playerService.getResources().getString(R.string.tab_bookmarks));
95+
builder.setExtras(extras);
96+
return new MediaItem(builder.build(), MediaItem.FLAG_BROWSABLE);
97+
}
98+
99+
private String createMediaIdForPlaylist(final long playlistId) {
100+
return ID_BOOKMARKS + '/' + playlistId;
101+
}
102+
103+
private MediaItem createPlaylistStreamMediaItem(final long playlistId,
104+
final PlaylistStreamEntry item,
105+
final int index) {
106+
final var builder = new MediaDescriptionCompat.Builder();
107+
builder.setMediaId(createMediaIdForPlaylistIndex(playlistId, index))
108+
.setTitle(item.getStreamEntity().getTitle())
109+
.setIconUri(Uri.parse(item.getStreamEntity().getThumbnailUrl()));
110+
111+
return new MediaItem(builder.build(), MediaItem.FLAG_PLAYABLE);
112+
}
113+
114+
private String createMediaIdForPlaylistIndex(final long playlistId, final int index) {
115+
return createMediaIdForPlaylist(playlistId) + '/' + index;
116+
}
47117

48118
@Nullable
49119
public MediaBrowserServiceCompat.BrowserRoot onGetRoot(@NonNull final String clientPackageName,
@@ -52,14 +122,152 @@ public MediaBrowserServiceCompat.BrowserRoot onGetRoot(@NonNull final String cli
52122
Log.d(TAG, String.format("MediaBrowserService.onGetRoot(%s, %s, %s)",
53123
clientPackageName, clientUid, rootHints));
54124

55-
return new MediaBrowserServiceCompat.BrowserRoot(MY_MEDIA_ROOT_ID, null);
125+
return new MediaBrowserServiceCompat.BrowserRoot(ID_ROOT, null);
56126
}
57127

58128
public Single<List<MediaItem>> onLoadChildren(@NonNull final String parentId) {
59129
Log.d(TAG, String.format("MediaBrowserService.onLoadChildren(%s)", parentId));
60130

61-
final List<MediaBrowserCompat.MediaItem> mediaItems = new ArrayList<>();
131+
final List<MediaItem> mediaItems = new ArrayList<>();
132+
final var parentIdUri = Uri.parse(parentId);
133+
134+
if (parentId.equals(ID_ROOT)) {
135+
mediaItems.add(
136+
createRootMediaItem(ID_BOOKMARKS,
137+
playerService.getResources().getString(R.string.tab_bookmarks)));
62138

139+
} else if (parentId.startsWith(ID_BOOKMARKS)) {
140+
final var path = parentIdUri.getPathSegments();
141+
if (path.size() == 2) {
142+
return populateBookmarks();
143+
} else if (path.size() == 3) {
144+
final var playlistId = Long.parseLong(path.get(2));
145+
return populatePlaylist(playlistId);
146+
} else {
147+
Log.w(TAG, "Unknown playlist uri " + parentId);
148+
}
149+
}
63150
return Single.just(mediaItems);
64151
}
152+
153+
private LocalPlaylistManager getPlaylistManager() {
154+
if (database == null) {
155+
database = NewPipeDatabase.getInstance(playerService);
156+
}
157+
if (localPlaylistManager == null) {
158+
localPlaylistManager = new LocalPlaylistManager(database);
159+
}
160+
return localPlaylistManager;
161+
}
162+
163+
private Single<List<MediaItem>> populateBookmarks() {
164+
final var playlists = getPlaylistManager().getPlaylists().firstOrError();
165+
return playlists.map(playlist ->
166+
playlist.stream().map(this::createPlaylistMediaItem).collect(Collectors.toList()));
167+
}
168+
169+
private Single<List<MediaItem>> populatePlaylist(final long playlistId) {
170+
final var playlist = getPlaylistManager().getPlaylistStreams(playlistId).firstOrError();
171+
return playlist.map(items -> {
172+
final List<MediaItem> results = new ArrayList<>();
173+
int index = 0;
174+
for (final var item : items) {
175+
results.add(createPlaylistStreamMediaItem(playlistId, item, index));
176+
++index;
177+
}
178+
return results;
179+
});
180+
}
181+
182+
private void playbackError(@StringRes final int resId, final int code) {
183+
playerService.stopForImmediateReusing();
184+
sessionConnector.setCustomErrorMessage(playerService.getString(resId), code);
185+
}
186+
187+
private Single<PlayQueue> extractPlayQueueFromMediaId(final String mediaId) {
188+
final Uri mediaIdUri = Uri.parse(mediaId);
189+
if (mediaIdUri == null) {
190+
return Single.error(new NullPointerException());
191+
}
192+
if (mediaId.startsWith(ID_BOOKMARKS)) {
193+
final var path = mediaIdUri.getPathSegments();
194+
if (path.size() == 4) {
195+
final long playlistId = Long.parseLong(path.get(2));
196+
final int index = Integer.parseInt(path.get(3));
197+
198+
return getPlaylistManager()
199+
.getPlaylistStreams(playlistId)
200+
.firstOrError()
201+
.map(items -> {
202+
final var infoItems = items.stream()
203+
.map(PlaylistStreamEntry::toStreamInfoItem)
204+
.collect(Collectors.toList());
205+
return new SinglePlayQueue(infoItems, index);
206+
});
207+
}
208+
}
209+
210+
return Single.error(new NullPointerException());
211+
}
212+
213+
@Override
214+
public long getSupportedPrepareActions() {
215+
return PlaybackStateCompat.ACTION_PLAY_FROM_MEDIA_ID;
216+
}
217+
218+
private void disposePrepareOrPlayCommands() {
219+
if (prepareOrPlayDisposable != null) {
220+
prepareOrPlayDisposable.dispose();
221+
prepareOrPlayDisposable = null;
222+
}
223+
}
224+
225+
@Override
226+
public void onPrepare(final boolean playWhenReady) {
227+
disposePrepareOrPlayCommands();
228+
// No need to prepare
229+
}
230+
231+
@Override
232+
public void onPrepareFromMediaId(@NonNull final String mediaId, final boolean playWhenReady,
233+
@Nullable final Bundle extras) {
234+
Log.d(TAG, String.format("MediaBrowserConnector.onPrepareFromMediaId(%s, %s, %s)",
235+
mediaId, playWhenReady, extras));
236+
237+
disposePrepareOrPlayCommands();
238+
prepareOrPlayDisposable = extractPlayQueueFromMediaId(mediaId)
239+
.observeOn(AndroidSchedulers.mainThread())
240+
.subscribe(
241+
playQueue -> {
242+
sessionConnector.setCustomErrorMessage(null);
243+
NavigationHelper.playOnBackgroundPlayer(playerService, playQueue,
244+
playWhenReady);
245+
},
246+
throwable -> {
247+
playbackError(R.string.error_http_not_found,
248+
PlaybackStateCompat.ERROR_CODE_NOT_SUPPORTED);
249+
250+
}
251+
);
252+
}
253+
254+
@Override
255+
public void onPrepareFromSearch(@NonNull final String query, final boolean playWhenReady,
256+
@Nullable final Bundle extras) {
257+
disposePrepareOrPlayCommands();
258+
playbackError(R.string.content_not_supported, PlaybackStateCompat.ERROR_CODE_NOT_SUPPORTED);
259+
}
260+
261+
@Override
262+
public void onPrepareFromUri(@NonNull final Uri uri, final boolean playWhenReady,
263+
@Nullable final Bundle extras) {
264+
disposePrepareOrPlayCommands();
265+
playbackError(R.string.content_not_supported, PlaybackStateCompat.ERROR_CODE_NOT_SUPPORTED);
266+
}
267+
268+
@Override
269+
public boolean onCommand(@NonNull final Player player, @NonNull final String command,
270+
@Nullable final Bundle extras, @Nullable final ResultReceiver cb) {
271+
return false;
272+
}
65273
}

0 commit comments

Comments
 (0)