问题描述
我目前正在尝试实施 MediabrowserService
以构建适用于 Android Auto 的媒体应用。
我按照官方 Android Auto 文档 (https://developer.android.com/training/cars/media#onLoadChildren) 来实现onLoadChildren
功能。
以下是我尝试在 Android Auto 屏幕上显示内容的代码片段:
override fun onLoadChildren(parentId: String,result: Result<MutableList<MediabrowserCompat.MediaItem>>) {
...
if (parentId == NODE_LIBRARY_ALBUMS) {
val items = mutablelistof<MediabrowserCompat.MediaItem>()
val albumList = LibraryManager.getAlbumList()
for (it in albumList) {
val descriptionBuilder = MediaDescriptionCompat.Builder()
.setTitle(it.albumName)
items.add(MediabrowserCompat.MediaItem(descriptionBuilder.build(),MediabrowserCompat.MediaItem.FLAG_broWSABLE))
}
result.sendResult(items)
}
...
}
当项目数量足够小时,这很有效。 但是当item数量很大时(比如5000左右),会出现如下错误:
E/JavaBinder: !!! Failed BINDER TRANSACTION !!! (parcel size = 1339384)
我发现支持 Android Auto 的其他几个媒体应用程序(例如三星音乐)可以显示大量项目。
有没有什么办法可以在onLoadChildren
函数上返回大量的项目,或者有没有其他办法可以解决这个问题?
谢谢!
解决方法
可能您必须将大数据拆分为小块。例如,您有一个包含 5000 个项目的列表。所以在你的内心,MediaBrowserService
中的 onLoadChildren
做这样的事情
public fun onLoadChildren(parentId: String,result: Result<List<MediaBrowserCompat.MediaItem>>) {
if (MEDIA_ID_ROOT == parentId || itemsList.size > 100) {
fillMediaBrowsableResult(parentId,result);
}
else {
fillMediaPlayableResult(parentId,result);
}
}
//Split the large number of content to a parts
private fun fillMediaBrowsableResult(parentId: String,result: Result<List<MediaBrowserCompat.MediaItem>>) {
// Detect count of parts
val partsCount = itemsList.size / 100
if(itemsList.size % 100 > 0){
partsCount ++
}
val mediaItems = mutableListOf<MediaBrowserCompat.MediaItem>()
//Create parts in a loop
for(i in 0 until partsCount){
val mediaDescription = MediaDescriptionCompat.Builder()
.setMediaId(i) // This is your next parent in onLoadChildren when you click on it in AA
.setTitle("Part ${i + 1}")
.build();
val mediaItem =. MediaBrowserCompat.MediaItem(mediaDescription,MediaBrowserCompat.MediaItem.FLAG_BROWSABLE)
mediaItems.add(mediaItem)
}
result.sendResult(mediaItems)
}
private fun fillMediaPlayableResult(parentId: String,result: Result<List<MediaBrowserCompat.MediaItem>>){
val intParent = parentId.toInt()
val startPosition = intParent * 100 // where to start for this part
val stopPosition = (intParent + 1) * 100 // where to finish for this part
if(stopPosition > itemsList.size){
stopPosition = itemsList.size
}
val mediaItems = mutableListOf<MediaBrowserCompat.MediaItem>()
for(i in startPosition..stopPosition){
//Generate playable content for this part
val item = itemsList[i]
val mediaDescription = MediaDescriptionCompat.Builder()
.setMediaId(item.id)
.setTitle(item.albumTitle)
.build();
val mediaItem =. MediaBrowserCompat.MediaItem(mediaDescription,MediaBrowserCompat.MediaItem.FLAG_PLAYABLE)
mediaItems.add(mediaItem)
}
result.sendResult(mediaItems)
}
我没有检查这段代码,但我认为这个想法很清楚。
,如果您查看 Android SDK document:
注意:Android Auto 和 Android Automotive OS 对如何 他们可以在菜单的每个级别中显示许多媒体项。这些 限制最大程度地减少对驾驶员的干扰并帮助他们操作您的 带有语音命令的应用程序。 ...
所以,我认为最好的方法是限制用户体验设计中媒体项目的数量。对于您在处理大量媒体项目的其他应用中看到的情况,他们可能将 Jetpack Media 2 用于 pagination。