问题描述:
本周在处理下载保存一段视频时,在系统相册中未能及时更新显示,导致一些自动化脚本(自动点击排序最前的视频)执行失误的问题。
发现问题
部分手机上(如 小米 note 5),下载后的视频未能排在最前,于是同事使用com.googlecode.mp4parser:isoparser库修改了视频文件的创建时间、修改时间。并且使用发广播的方式,通知系统刷新多媒体文件。但是出现部分手机,视频无法打开。
fun modifyMp4Date(fullPath: String, newDate: Date = Date()): Boolean {
try {
Logger.i(TAG, "modifyMp4Date()")
if (!fullPath.endsWith(".mp4")) return false
val videoFile = File(fullPath)
if (!videoFile.exists()) return false
val isoFile = IsoFile(fullPath)
val movieBox = isoFile.movieBox ?: return false
val movieHeaderBox = movieBox.movieHeaderBox ?: return false
movieHeaderBox.creationTime = newDate
movieHeaderBox.modificationTime = newDate
val baos = BetterByteArrayOutputStream()
movieBox.getBox(Channels.newChannel(baos))
isoFile.close()
val fc = RandomAccessFile(videoFile, "rw").channel
fc.write(ByteBuffer.wrap(baos.buffer, 0, baos.size()))
fc.close()
return true
} catch (e: Exception) {
Logger.e(TAG, "modifyMp4Date error path=$fullPath", e)
}
return false
}
解决问题
知识点
- 多媒体文件在系统相册中立即展示是需要手动通知系统的。
-
MediaProvider,一个系统app(数据库),提供了在内部和外部存储设备上所有可用媒体的元数据。即所有文件的索引。 通过MediaProvider,我们可以快速获取到手机内的不同的文件类型的数据。
比如说相册的展示、微信聊天选择视频图片的发送。
MediaProvider源码:
http://androidxref.com/8.1.0_r33/xref/packages/providers/MediaProvider/
- android.provider.MediaStore则提供了一些对于MediaProvider的API
手动更新 MediaProvider的方式。
- 通过操作 MediaStore 类。
MediaStore.Images.Media.insertImage(
contentResolver,
mShareBitmap!!,
"image_file",
"file")
- 发送广播更新。
val saveAs = "Your_Created_Image_File_Path"
val contentUri = Uri.fromFile(File(saveAs))
val mediaScanIntent = Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE,contentUri)
sendBroadcast(mediaScanIntent)
- 通过操作 MediaScannerConnection 类。
MediaScannerConnection.scanFile(this
, arrayOf(picFile.absolutePath)
, arrayOf("image/jpeg")) { path, uri ->
Log.i("cxmyDev", "onScanCompleted : " + path)
}
以上三种方法代码,转自承香墨影的文章,详细请参考 :
https://www.cnblogs.com/plokmju/p/android_mediastore.html
- 直接操作MediaProvider数据库
ContentValues values = new ContentValues();
values.put(MediaStore.Video.Media.TITLE, "Title1");
values.put(MediaStore.Video.Media.MIME_TYPE, "video/mp4");
values.put(MediaStore.Video.Media.DATA, videoPath);
Uri uri = cr.insert(MediaStore.Video.Media.EXTERNAL_CONTENT_URI, values);
这里插入的数据越详尽当然越好,表现在于,我们在文件管理器中,查看该文件详情所展示出来的数据。否则就会出现诸如时长未知、宽高未知等。。
上述三种方式,最终走的都是这个插库流程。
如MediaStore.Images.Media.insertImage()
public static final String insertImage(ContentResolver cr, Bitmap source,
String title, String description) {
ContentValues values = new ContentValues();
values.put(Images.Media.TITLE, title);
values.put(Images.Media.DESCRIPTION, description);
values.put(Images.Media.MIME_TYPE, "image/jpeg");
Uri url = null;
String stringUrl = null; /* value to be returned */
try {
url = cr.insert(EXTERNAL_CONTENT_URI, values);
... //保存缩略图
}
}
不过很奇怪的是这里并没有处理我所关心的文件的时间。。
踩过的坑
主要是针对视频
- 使用的广播方式:
部分手机,发送广播后,相册并没有更新!这里可能是由于部分厂商对于MediaProvider的所在的包的接受广播的类做了某些更改,或者限制。
小米 note 5,android 8.1
在系统相册中的视频排序,不是按照视频的修改时间,甚至不是按照数据库里面的使用DATETAKEN字段(拍摄时间),而是文件的真正的创建时间。。所以这也是有下面那条的尝试。修改文件的创建时间:(直接对文件进行流操作)
部分手机,文件解析直接出了问题,视频打不开了,查看文件详情,宽高变成了错误的参数,比如(27*1956),估计是因为不同解码器?
结果
虽然坑依然有,但是实际上面的分析已经满足我需求,因为,我并不需要某个下载的媒体文件在系统相册中排在最前,只需要在我所关心的app,比如在微信的聊天发送第一个视频,抖音上传第一条视频。
- 如微信使用的是文件修改时间作为顺序
- 而抖音以拍摄时间来排序
所以我的需求显然通过直接插库的方法可以处理
values.put(MediaStore.Video.Media.DATETAKEN, curTime);
values.put(MediaStore.Video.Media.DATE_MODIFIED, curTime);
即便在自己app里面,下载视频后,喜欢使用什么排序,只要我们在查询数据库的时候,给一个排序类类型即可
Cursor cursor = context.getContentResolver().query(
MediaStore.Video.Media.EXTERNAL_CONTENT_URI, projection, null,
null, "datetaken");
疑问
综上,我依然有两个疑问,就是关于linux的时间的问题。
当新建一个文件的时候(比如下载或者复制),这个文件的创建时间是哪一个时间点?
比如说上述小米 note 5 的视频时间的问题,下载图片后却没有此问题,究竟是不是小米的锅?
另外,关于linux的时间,我觉得这篇还是蛮不错的。
https://blog.csdn.net/h106140873/article/details/78858344大厂他们是如何处理。。比如说,同样的手机,微信或者浏览器保存一个小视频,并没有出现此问题、、、好想知道他们怎么处理的。。
补充一点:
后续实验中,发现就是,如果对文件进行真正的编辑,比如剪辑甚至使用adb cp 命令,都不会出现上述问题
个人觉得大厂是很可能有对文件创建时间进行处理的,毕竟不就是一个MP4文件的编解码库而已嘛,甚至我之所以踩坑,也可能只是IsoParse库的使用不到位
2019.06.25