从这里的源代码中,我们可以看到扫描程序的最终实现有两个步骤来扫描音频文件。如果其中任何一个步骤失败,则该音频文件将不会插入媒体提供程序。
步骤1 检查文件扩展名
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
,则扫描过程在此步骤中正常。
第二步 检索元数据
在第一步通过后,扫描器需要检索媒体的元数据以备后续的数据库插入。我认为在此步骤中,扫描器进行编解码级别的检查。
sp<MediaMetadataRetriever> mRetriever(new MediaMetadataRetriever);
int fd = open(path, O_RDONLY | O_LARGEFILE);
status_t status;
if (fd < 0) {
status = mRetriever->setDataSource(path);
} else {
status = mRetriever->setDataSource(fd, 0, 0x7ffffffffffffffL);
close(fd);
}
if (status) {
return MEDIA_SCAN_RESULT_ERROR;
}
对于 Android 5.0 版本之前的设备,在此步骤可能会出现扫描器失败的问题。因为缺少内置的 Opus 编解码器支持,setDataSource
最终可能会失败。媒体文件最终不会被添加到媒体提供程序中。
建议解决方案
因为我们知道音频文件将被添加到
MediaStore.Audio.Media.EXTERNAL_CONTENT_URI
我们可以手动进行数据库操作。如果您希望您的音频文件与数据库中的其他音频文件保持一致,您需要自己检索所有元数据。由于您可以播放opus文件,我认为检索元数据很容易。
// 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...