MediaStore:更新我的应用程序创建的文件后,无法轻易删除

MediaStore: After updating the file which is created by my app it cannot be deleted easily

下一个删除我的应用程序拥有的文件的代码工作正常,没有例外RecoverableSecurityException因为文件是由我的应用程序创建的(使用ContentResolver.insert(...)方法)

getVideoFileContentUri(context, file)?.let { uri ->
    try {
        context.contentResolver.delete(uri, null, null)
    } catch (securityException: RecoverableSecurityException) {
        val intentSender =
            securityException.userAction.actionIntent.intentSender
        intentSender?.let {
            activity.startIntentSenderForRecsult(
                intentSender,
                REQUEST_CODE,
                null,
                0,
                0,
                0,
                null
            )
        }
    }
}

fun getVideoFileContentUri(context: Context, file: File): Uri? {
    val filePath = file.absolutePath
    val cursor = context.contentResolver.query(
        MediaStore.Video.Media.EXTERNAL_CONTENT_URI, arrayOf(MediaStore.Video.Media._ID),
        MediaStore.Video.Media.DATA + "=? ", arrayOf(filePath), null
    )
    return if (cursor != null && cursor.moveToFirst()) {
        val id: Int = cursor.getInt(cursor.getColumnIndex(MediaStore.MediaColumns._ID))
        cursor.close()
        Uri.withAppendedPath(MediaStore.Video.Media.EXTERNAL_CONTENT_URI, "" + id)
    } else {
        null
    }
}

但是,如果我使用 ContentResolver.update(...) 方法更新由应用程序创建的文件,则删除该文件将需要权限 - 它会抛出 RecoverableSecurityException 并启动打开系统对话框以确认修改文件的意图

// here I change file name of the file
val contentValues = ContentValues(1).apply {
    put(MediaStore.Video.Media.DISPLAY_NAME, "SOME NEW FILE NAME")
}
contentResolver.update(uri, contentValues, null, null)

所以现在看起来我的应用程序并不拥有该文件,要删除它,我的用户必须确认删除每个文件

这真的很烦人,我该如何解决这个问题?

所以在 ContentResolver.update(...) 为您自己的文件创建后 ContentResolver.insert(...) 应用程序失去了修改该文件的权限并且需要请求它

我找到了解决方案。

当您创建文件时,我们需要将 put(MediaStore.*.Media.IS_PENDING, 1) 添加到内容值中:

val contentValues = ContentValues(5).apply {
    put(MediaStore.Video.Media.IS_PENDING, 1)
    put(MediaStore.Video.Media.DISPLAY_NAME, "INITIAL FILENAME")
    put(MediaStore.Video.Media.DATE_ADDED, System.currentTimeMillis() / 1000)
    put(MediaStore.Video.Media.MIME_TYPE, "video/mp4")
    put(MediaStore.Video.Media.RELATIVE_PATH, relativePath)
}

val uri = resolver.insert(
    MediaStore.Video.Media.EXTERNAL_CONTENT_URI,
    contentValues
)

然后当你完成一个文件(例如,使用 MediaRecorder 完成视频录制到那个文件)你可以像这样更新它(你必须添加 put(MediaStore.*.Media.IS_PENDING, 0):

val contentValues = ContentValues(2).apply {
    put(MediaStore.Video.Media.IS_PENDING, 0)
    put(MediaStore.Video.Media.DISPLAY_NAME, "NEW FILENAME, E.G. APPENDING END RECORDING TIMESTAMP TO PREVIOUS FILE NAME")
}
resolver.update(uri, contentValues, null, null)

然后在调用 ContentResolver.delete(...) 函数时,不再需要请求权限(不会抛出 RecoverableSecurityException

try {
    resolver.delete(uri, null, null)
} catch (securityException: RecoverableSecurityException) {
    val intentSender =
        securityException.userAction.actionIntent.intentSender
    intentSender?.let {
        activity.startIntentSenderForRecsult(
            intentSender,
            REQUEST_CODE,
            null,
            0,
            0,
            0,
            null
        )
}

用户不会生气 :)

虽然如果用户重新安装您的应用程序,您的应用程序之前创建的所有文件都需要权限请求,是的,从 Android Q 开始,这并不容易... :)