1
1
package org .schabi .newpipe .player .mediabrowser ;
2
2
3
+ import android .net .Uri ;
3
4
import android .os .Bundle ;
4
- import android .support . v4 . media . MediaBrowserCompat ;
5
+ import android .os . ResultReceiver ;
5
6
import android .support .v4 .media .MediaBrowserCompat .MediaItem ;
7
+ import android .support .v4 .media .MediaDescriptionCompat ;
6
8
import android .support .v4 .media .session .MediaSessionCompat ;
9
+ import android .support .v4 .media .session .PlaybackStateCompat ;
7
10
import android .util .Log ;
8
11
9
12
import androidx .annotation .NonNull ;
10
13
import androidx .annotation .Nullable ;
14
+ import androidx .annotation .StringRes ;
11
15
import androidx .media .MediaBrowserServiceCompat ;
16
+ import androidx .media .utils .MediaConstants ;
12
17
18
+ import com .google .android .exoplayer2 .Player ;
13
19
import com .google .android .exoplayer2 .ext .mediasession .MediaSessionConnector ;
14
20
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 ;
15
27
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 ;
16
31
17
32
import java .util .ArrayList ;
18
33
import java .util .List ;
34
+ import java .util .stream .Collectors ;
19
35
36
+ import io .reactivex .rxjava3 .android .schedulers .AndroidSchedulers ;
20
37
import io .reactivex .rxjava3 .core .Single ;
38
+ import io .reactivex .rxjava3 .disposables .Disposable ;
21
39
22
- public class MediaBrowserConnector {
40
+ public class MediaBrowserConnector implements MediaSessionConnector . PlaybackPreparer {
23
41
private static final String TAG = MediaBrowserConnector .class .getSimpleName ();
24
42
25
43
private final PlayerService playerService ;
26
44
private final @ NonNull MediaSessionConnector sessionConnector ;
27
45
private final @ NonNull MediaSessionCompat mediaSession ;
28
46
47
+ private AppDatabase database ;
48
+ private LocalPlaylistManager localPlaylistManager ;
49
+ private Disposable prepareOrPlayDisposable ;
50
+
29
51
public MediaBrowserConnector (@ NonNull final PlayerService playerService ) {
30
52
this .playerService = playerService ;
31
53
mediaSession = new MediaSessionCompat (playerService , TAG );
32
54
sessionConnector = new MediaSessionConnector (mediaSession );
33
55
sessionConnector .setMetadataDeduplicationEnabled (true );
56
+ sessionConnector .setPlaybackPreparer (this );
34
57
playerService .setSessionToken (mediaSession .getSessionToken ());
35
58
}
36
59
@@ -39,11 +62,58 @@ public MediaBrowserConnector(@NonNull final PlayerService playerService) {
39
62
}
40
63
41
64
public void release () {
65
+ disposePrepareOrPlayCommands ();
42
66
mediaSession .release ();
43
67
}
44
68
45
69
@ 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
+ }
47
117
48
118
@ Nullable
49
119
public MediaBrowserServiceCompat .BrowserRoot onGetRoot (@ NonNull final String clientPackageName ,
@@ -52,14 +122,152 @@ public MediaBrowserServiceCompat.BrowserRoot onGetRoot(@NonNull final String cli
52
122
Log .d (TAG , String .format ("MediaBrowserService.onGetRoot(%s, %s, %s)" ,
53
123
clientPackageName , clientUid , rootHints ));
54
124
55
- return new MediaBrowserServiceCompat .BrowserRoot (MY_MEDIA_ROOT_ID , null );
125
+ return new MediaBrowserServiceCompat .BrowserRoot (ID_ROOT , null );
56
126
}
57
127
58
128
public Single <List <MediaItem >> onLoadChildren (@ NonNull final String parentId ) {
59
129
Log .d (TAG , String .format ("MediaBrowserService.onLoadChildren(%s)" , parentId ));
60
130
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 )));
62
138
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
+ }
63
150
return Single .just (mediaItems );
64
151
}
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
+ }
65
273
}
0 commit comments