从源代码here我们可以看到扫描器的最终实现有两个步骤来扫描音频文件。如果这两个步骤中的任何一个失败,音频文件将不会插入到媒体提供程序中。
第一步检查文件扩展名
static bool FileHasAcceptableExtension(const char *extension) {
static const char *kValidExtensions[] = {
".mp3", ".mp4", ".m4a", ".3gp", ".3gpp", ".3g2", ".3gpp2",
".mpeg", ".ogg", ".mid", ".smf", ".imy", ".wma", ".aac",
".wav", ".amr", ".midi", ".xmf", ".rtttl", ".rtx", ".ota",
".mkv", ".mka", ".webm", ".ts", ".fl", ".flac", ".mxmf",
".avi", ".mpeg", ".mpg"
};
static const size_t kNumValidExtensions =
sizeof(kValidExtensions) / sizeof(kValidExtensions[0]);
for (size_t i = 0; i < kNumValidExtensions; ++i) {
if (!strcasecmp(extension, kValidExtensions[i])) {
return true;
}
}
return false;
}
自 Android 5.0 以来添加了更多扩展。 opus编解码器的通用容器是ogg,这个扩展在Android 5.0之前就存在。假设您的音频文件扩展名是ogg,则此步骤的扫描过程很好。
step2检索元数据
第一步通过后,扫描仪需要检索媒体的元数据以供以后插入数据库。我认为扫描仪会在这一步进行编解码器级别检查。
sp<MediaMetadataRetriever> mRetriever(new MediaMetadataRetriever);
int fd = open(path, O_RDONLY | O_LARGEFILE);
status_t status;
if (fd < 0) {
// couldn't open it locally, maybe the media server can?
status = mRetriever->setDataSource(path);
} else {
status = mRetriever->setDataSource(fd, 0, 0x7ffffffffffffffL);
close(fd);
}
if (status) {
return MEDIA_SCAN_RESULT_ERROR;
}
对于 5.0 之前的 Android 版本,扫描仪可能会在此步骤失败。由于缺乏内置的 opus 编解码器支持,setDataSource 最后会失败。媒体文件最终不会被添加到媒体提供者中。
建议的解决方案
因为我们知道音频文件会被添加到
MediaStore.Audio.Media.EXTERNAL_CONTENT_URI
我们可以手动进行数据库操作。如果您希望您的音频文件与数据库中的其他音频文件保持一致,您必须自己检索所有元数据。既然你可以播放作品文件,我认为检索元数据很容易。
// retrieve more metadata, duration etc.
ContentValues contentValues = new ContentValues();
contentValues.put(MediaStore.Audio.AudioColumns.DATA, "/mnt/sdcard/Music/example.opus");
contentValues.put(MediaStore.Audio.AudioColumns.TITLE, "Example track");
contentValues.put(MediaStore.Audio.AudioColumns.DISPLAY_NAME, "example");
// more columns should be filled from here
Uri uri = getContentResolver().insert(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, contentValues);
Log.d(TAG, uri.toString());
之后,您的应用可以找到音频文件。
getContentResolver().query(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI...