Android 11 媒体存储 - 无法在相对路径上录制新视频。它适用于 Environment.DIRECTORY_MOVIES 路径但不适用于子文件夹

Android 11 Media store - cannot record new videos on relative path. It works with Environment.DIRECTORY_MOVIES path but not not with sub folder

我的代码在低于 Android 11 的任何 Android 上运行完美。 在 Android 11(特别是在 Pixel2 和 Pixel3 模拟器和真实设备上),文件描述符在将文件插入到媒体存储后找不到文件。

如果我将第二行的相对路径更改为仅 Environment.DIRECTORY_MOVIES 路径 - 它有效

String relativeLocation = Environment.DIRECTORY_MOVIES;

这是我的代码:

String mimeType = "video/mp4";
String relativeLocation = Environment.DIRECTORY_MOVIES + File.separator + SUB_DIRECTORY_NAME ;
shortFileName = getDateStamp() + selectedExtension;
videoContentValues = new ContentValues();
videoContentValues.put(MediaStore.Video.Media.TITLE, shortFileName);
videoContentValues.put(MediaStore.MediaColumns.DISPLAY_NAME, shortFileName);
videoContentValues.put(MediaStore.MediaColumns.MIME_TYPE, mimeType);
videoContentValues.put(MediaStore.Video.Media.DATE_ADDED, System.currentTimeMillis() / 1000);
videoContentValues.put(MediaStore.Video.Media.DATE_TAKEN, System.currentTimeMillis());

videoContentValues.put(MediaStore.MediaColumns.RELATIVE_PATH, relativeLocation);
videoContentValues.put(MediaStore.MediaColumns.IS_PENDING, 1);
videResolver = mContext.getContentResolver();

Uri contentUri = MediaStore.Video.Media.getContentUri(MediaStore.VOLUME_EXTERNAL_PRIMARY);
videoUri = videResolver.insert(contentUri, videoContentValues);
if (videoUri == null) {
    throw new IOException("Failed to create new MediaStore record.");
}
pfd = mContext.getContentResolver().openFileDescriptor(videoUri, "w");
mMediaRecorder.setOutputFile(pfd.getFileDescriptor());

代码运行正常,直到它到达 return 出现以下错误的 openFileDescriptor:

java.io.FileNotFoundException: java.io.FileNotFoundException: No item at content://media/external_primary/video/media/47

我做错了什么?

更新 1: 更多信息: 我仍在请求 WRITE_EXTERNAL_STORAGE 并确保在开始录制之前允许它

以下代码适用于 Android 11 Pixel 2 模拟器上的同一相机应用程序以保存照片:

    String mimeType = "image/jpeg";
    final String relativeLocation = Environment.DIRECTORY_PICTURES + File.separator + SUB_DIRECTORY_NAME;
    ContentValues contentValues = new ContentValues();
    contentValues.put(MediaStore.MediaColumns.DISPLAY_NAME, createImageFileNameQ());
    contentValues.put(MediaStore.MediaColumns.MIME_TYPE, mimeType);
    contentValues.put(MediaStore.MediaColumns.RELATIVE_PATH, relativeLocation);
    ContentResolver resolver = mContext.getContentResolver();
    OutputStream stream = null;
    Uri uri = null;
    try {
          final Uri contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
          uri = resolver.insert(contentUri, contentValues);
      try {
                OutputStream fos = resolver.openOutputStream(uri);
                fos.write(data);
                fos.flush();
                    fos.close();
           }
     catch (Exception ex) {
        Log.e(TAG, ex.toString());
      }

更新 2:

如果我在第二行写“String relativeLocation = Environment.DIRECTORY_PICTURES + File.separator + SUB_DIRECTORY_NAME;”而不是 DIRECTORY_MOVIES : 有用!@!! 视频文件保存在 Videos\Sub_Directory

更新 3: 现在在模拟器上:

没有技巧可以帮助创建相对路径。 而且图片不能保存到相对路径

String mimeType = "image/jpeg"; String relativeLocation = Environment.DIRECTORY_PICTURES + File.separator + SUB_DIRECTORY_NAME;
ContentValues contentValues = new ContentValues();
    
 contentValues.put(MediaStore.MediaColumns.DISPLAY_NAME, createImageFileNameQ());
  contentValues.put(MediaStore.MediaColumns.MIME_TYPE, mimeType);
  contentValues.put(MediaStore.MediaColumns.RELATIVE_PATH, relativeLocation);

  ContentResolver resolver = mContext.getContentResolver();


  OutputStream stream = null;
  Uri uri = null;

  try {
    final Uri contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
     uri = resolver.insert(contentUri, contentValues);
 try {
     OutputStream fos = resolver.openOutputStream(uri);
     fos.write(data);
     fos.flush();
     fos.close();
 } catch (Exception ex) {
 Log.e(TAG, ex.toString());
 }
 } catch (Exception e) {}

我已经打开了 Google/Android 的票证,并按照他们的要求完整记录了模拟器调试消息。他们说:"有趣,他们会调查的"。他们没有回答 - 但他们已经为最新的 Android 11 版本修复了它。安全补丁 2020 年 11 月。所以 - 问题已解决。 我在我的代码中留下了一个错误修复,以防有人没有得到最后的 android 11 更新。如果出现此错误 - 我将 IsAndroid11SaveOnSubDirectoryEnabled 变量设置为 false

String relativeLocation = Environment.DIRECTORY_MOVIES + File.separator + SUB_DIRECTORY_NAME;
if (Build.VERSION.SDK_INT > Build.VERSION_CODES.Q && IsAndroid11SaveOnSubDirectoryEnabled == false)
{
relativeLocation = Environment.DIRECTORY_MOVIES + File.separator + "$" + SUB_DIRECTORY_NAME;
Log.e(TAG, "Recording on $SubDirectory");
}

编辑二: 我从 Google/Android 得到一个答案,也许我在目录中有一个“.nomedia”文件,所以 - 媒体存储不能 insert/index 新文件。 (.nomedia 是一个文件,告诉画廊不要读取此文件夹中的媒体,它适用于 Android 10,在 Android 11 - 无法在电影文件夹中创建或删除此文件,因为它不是媒体)。但它不正确 - 我发送到 Google/Android 的日志文件和记录来自 Android 11 模拟器,它没有这个文件。

编辑 III - 由于 Dilijeet 询问,IsAndroid11SaveOnSubDirectoryEnabled 是我保存到 SharedPreference 的布尔值。这是代码 - 每当 IOExcpetion & Android 11 上的记录崩溃时 - 我假设这是子目录问题,并设置此布尔值以进行进一步记录。

catch (IOException ex)
{

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R && IsAndroid11SaveOnSubDirectoryEnabled == true) {
    mSharedPreferences.edit().putBoolean("chkSaveOnSubDirectory", false).apply();
    IsAndroid11SaveOnSubDirectoryEnabled = false;    
}