Android:使用Storage Access Framework获取的URI中的意向选择器打开文件

时间:2015-05-30 12:53:17

标签: android android-intent android-file storage-access-framework android-afilechooser

一开始,用户可以使用新的存储访问框架选择文件(假设应用程序是API> 19):

https://developer.android.com/guide/topics/providers/document-provider.html

然后我通过保存看起来像这样的URI来保存对这些所选文件的引用:

content://com.android.providers.downloads.documments/document/745

(在这种情况下,文件来自默认下载dir`)。

稍后,我想让用户打开这些文件(例如,他们的名字显示在UI列表中,用户选择一个)。

我想用Android着名的意图选择器功能来做这件事,而我所拥有的只是上面的URI对象......

谢谢,

2 个答案:

答案 0 :(得分:1)

编辑:我已修改此答案,以包含我最初称为"编写专门的ContentProvider"的方法示例代码。这应该完全满足问题的要求。可能会使答案太大,但现在它具有内部代码依赖性,所以让我们将其保留为整体。主要观点仍然适用:如果需要,可以使用下面的ContentPrvder,但尝试将file:// Uris提供给支持它们的应用程序,除非您想因某人的应用程序崩溃而受到指责。

原始回答

我会像现在这样远离Storage Access Framework。它没有得到Google的充分支持,而且应用程序中的支持非常糟糕,因此很难区分这些应用程序中的错误和SAF本身。如果你有足够的信心(这真的意味着"可以更好地使用try-catch块,那么平均Android开发人员"),请自己使用Storage Access Framework,但是只传递给其他好的file://路径。

您可以使用以下技巧从ParcelFileDescriptor获取文件系统路径(您可以通过调用openFileDescriptor从ContentResolver获取它):

class FdCompat {
 public static String getFdPath(ParcelFileDescriptor fd) {
  final String resolved;

  try {
   final File procfsFdFile = new File("/proc/self/fd/" + fd.getFd());

   if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
    // Returned name may be empty or "pipe:", "socket:", "(deleted)" etc.
    resolved = Os.readlink(procfsFdFile.getAbsolutePath());
   } else {
    // Returned name is usually valid or empty, but may start from
    // funny prefix if the file does not have a name
    resolved = procfsFdFile.getCanonicalPath();
   }

  if (TextUtils.isEmpty(resolved) || resolved.charAt(0) != '/'
                || resolved.startsWith("/proc/") || resolved.startsWith("/fd/"))
   return null;
  } catch (IOException ioe) {
   // This exception means, that given file DID have some name, but it is 
   // too long, some of symlinks in the path were broken or, most
   // likely, one of it's directories is inaccessible for reading.
   // Either way, it is almost certainly not a pipe.
   return "";
  } catch (Exception errnoe) {
   // Actually ErrnoException, but base type avoids VerifyError on old versions
   // This exception should be VERY rare and means, that the descriptor
   // was made unavailable by some Unix magic.
   return null;
  }

  return resolved;
 }
}

您必须做好准备,上面的方法将返回null(文件是管道或套接字,这是完全合法的)或空路径(对文件的父目录没有读取权限)。如果发生这种情况,将整个流复制到您可以访问的某个目录

完整解决方案

如果你真的想坚持使用内容提供商Uris,那么就去吧。以下是ContentProvider的代码。粘贴到您的应用中(并在AndroidManifest中注册)。使用下面的getShareableUri方法将收到的存储访问框架Uri转换为您自己的。将Uri传递给其他应用程序而不是原始的Uri。

下面的代码是不安全的(你可以很容易地使它安全,但解释这将扩大这个答案的长度超出想象)。如果您在意,请使用file:// Uris-Linux文件系统被广泛认为足够安全。

扩展下面的解决方案以提供没有相应Uri的任意文件描述符,这仍然是读者的练习。

public class FdProvider extends ContentProvider {
 private static final String ORIGINAL_URI = "o";
 private static final String FD = "fd";
 private static final String PATH = "p";

 private static final Uri BASE_URI = 
     Uri.parse("content://com.example.fdhelper/");

 // Create an Uri from some other Uri and (optionally) corresponding
 // file descriptor (if you don't plan to close it until your process is dead).
 public static Uri getShareableUri(@Nullable ParcelFileDescriptor fd,
                                   Uri trueUri) {
     String path = fd == null ? null : FdCompat.getFdPath(fd);
     String uri = trueUri.toString();

     Uri.Builder builder = BASE_URI.buildUpon();

     if (!TextUtils.isEmpty(uri))
         builder.appendQueryParameter(ORIGINAL_URI, uri);

     if (fd != null && !TextUtils.isEmpty(path))
         builder.appendQueryParameter(FD, String.valueOf(fd.getFd()))
                .appendQueryParameter(PATH, path);

     return builder.build();
 }

 public boolean onCreate() { return true; }

 public ParcelFileDescriptor openFile(Uri uri, String mode)
     throws FileNotFoundException {

     String o = uri.getQueryParameter(ORIGINAL_URI);
     String fd = uri.getQueryParameter(FD);
     String path = uri.getQueryParameter(PATH);

     if (TextUtils.isEmpty(o)) return null;

     // offer the descriptor directly, if our process still has it
     try {
         if (!TextUtils.isEmpty(fd) && !TextUtils.isEmpty(path)) {
             int intFd = Integer.parseInt(fd);

             ParcelFileDescriptor desc = ParcelFileDescriptor.fromFd(intFd);

             if (intFd >= 0 && path.equals(FdCompat.getFdPath(desc))) {
                 return desc;
             }
         }
     } catch (RuntimeException | IOException ignore) {}

     // otherwise just forward the call
     try {
         Uri trueUri = Uri.parse(o);

         return getContext().getContentResolver()
             .openFileDescriptor(trueUri, mode);
     }
     catch (RuntimeException ignore) {}

     throw new FileNotFoundException();
 }

 // all other calls are forwarded the same way as above
 public Cursor query(Uri uri, String[] projection, String selection,
     String[] selectionArgs, String sortOrder) {

     String o = uri.getQueryParameter(ORIGINAL_URI);

     if (TextUtils.isEmpty(o)) return null;

     try {
         Uri trueUri = Uri.parse(o);

         return getContext().getContentResolver().query(trueUri, projection,
             selection, selectionArgs, sortOrder);
     } catch (RuntimeException ignore) {}

     return null;
 }

 public String getType(Uri uri) {
     String o = uri.getQueryParameter(ORIGINAL_URI);

     if (TextUtils.isEmpty(o)) return "*/*";

     try {
         Uri trueUri = Uri.parse(o);

         return getContext().getContentResolver().getType(trueUri);
     } catch (RuntimeException e) { return null; }
 }

 public Uri insert(Uri uri, ContentValues values) {
     return null;
 }

 public int delete(Uri uri, String selection, String[] selectionArgs) {
     return 0;
 }

 public int update(Uri uri, ContentValues values, String selection,
     String[] selectionArgs) { return 0; }
}

答案 1 :(得分:1)

已经在SO上提供了解决方案,您只需要搜索它。

这是answer by Paul Burke。他编写了一个实用程序类,它返回这样一个内容路径的完整文件路径。

他说:

  

这将从MediaProvider,DownloadsProvider获取文件路径,   和ExternalStorageProvider,同时回到非官方   你提到的ContentProvider方法。

  

这些来自我的开源库aFileChooser

FileUtils.java是Paul Burke编写您正在寻找的方法的地方。