android多媒体库的实时更新

2019-06-25  本文已影响0人  真心czx

问题描述:


本周在处理下载保存一段视频时,在系统相册中未能及时更新显示,导致一些自动化脚本(自动点击排序最前的视频)执行失误的问题。

发现问题


部分手机上(如 小米 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源码:
http://androidxref.com/8.1.0_r33/xref/packages/providers/MediaProvider/

手动更新 MediaProvider的方式。
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.scanFile(this
        , arrayOf(picFile.absolutePath)
        , arrayOf("image/jpeg")) { path, uri ->

    Log.i("cxmyDev", "onScanCompleted : " + path)
}

以上三种方法代码,转自承香墨影的文章,详细请参考 :
https://www.cnblogs.com/plokmju/p/android_mediastore.html

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);
                    ... //保存缩略图
          }
}

不过很奇怪的是这里并没有处理我所关心的文件的时间。。

踩过的坑


主要是针对视频

  1. 使用的广播方式:
    部分手机,发送广播后,相册并没有更新!这里可能是由于部分厂商对于MediaProvider的所在的包的接受广播的类做了某些更改,或者限制。

接收广播的源码:
http://androidxref.com/8.1.0_r33/xref/packages/providers/MediaProvider/src/com/android/providers/media/MediaScannerReceiver.java

  1. 小米 note 5,android 8.1
    在系统相册中的视频排序,不是按照视频的修改时间,甚至不是按照数据库里面的使用DATETAKEN字段(拍摄时间),而是文件的真正的创建时间。。所以这也是有下面那条的尝试。

  2. 修改文件的创建时间:(直接对文件进行流操作)
    部分手机,文件解析直接出了问题,视频打不开了,查看文件详情,宽高变成了错误的参数,比如(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的时间的问题。

  1. 当新建一个文件的时候(比如下载或者复制),这个文件的创建时间是哪一个时间点?
    比如说上述小米 note 5 的视频时间的问题,下载图片后却没有此问题,究竟是不是小米的锅?
    另外,关于linux的时间,我觉得这篇还是蛮不错的。
    https://blog.csdn.net/h106140873/article/details/78858344

  2. 大厂他们是如何处理。。比如说,同样的手机,微信或者浏览器保存一个小视频,并没有出现此问题、、、好想知道他们怎么处理的。。

补充一点:
后续实验中,发现就是,如果对文件进行真正的编辑,比如剪辑甚至使用adb cp 命令,都不会出现上述问题
个人觉得大厂是很可能有对文件创建时间进行处理的,毕竟不就是一个MP4文件的编解码库而已嘛,甚至我之所以踩坑,也可能只是IsoParse库的使用不到位

2019.06.25

上一篇 下一篇

猜你喜欢

热点阅读