7

In my app I gain sd card write permission using the following intent. If the user selects the sd card folder from the system file explorer then I have sd card write access.

Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT_TREE);
intent.putExtra("android.content.extra.SHOW_ADVANCED", true);
startActivityForResult(intent, 42);

After that I am able to modify files in th sd card using the DocumentFile class. But I'm having problem getting a DocumentFile for a random file path.

Document.fromFile(new File(path));
Document.fromSingleUri(Uri.fromFile(new File(path)));

both return a DocumentFile object that returns false on .canWrite(). Even though I allready have sd card permission.

So I wrote the method posted in the end of my question to get a DocumentFile that returns true on .canWrite(). But this is slow...And also feels very wrong! There has to be a better way to do this. I also wrote a method that returns the same string as

String docFileUriString = docFile.getUri().toString(); 

for any file, where docFile is the DocumentFile that is returned by the method below. But

DocumentFile.fromTreeUri(Uri.parse(docFileUriString ));

returns a DocumentFile that points to the root of the sd card instead of the DocumentFile path. Which is just weird. Can someone suggest a more elegant solution?

public static DocumentFile getDocumentFileIfAllowedToWrite(File file, Context con){ 

    List<UriPermission> permissionUris = con.getContentResolver().getPersistedUriPermissions();

    for(UriPermission permissionUri:permissionUris){

        Uri treeUri = permissionUri.getUri();
        DocumentFile rootDocFile = DocumentFile.fromTreeUri(con, treeUri);
        String rootDocFilePath = FileUtil.getFullPathFromTreeUri(treeUri, con);

        if(file.getAbsolutePath().startsWith(rootDocFilePath)){

            ArrayList<String> pathInRootDocParts = new ArrayList<String>();
            while(!rootDocFilePath.equals(file.getAbsolutePath())){
                pathInRootDocParts.add(file.getName());
                file = file.getParentFile();
            } 

            DocumentFile docFile = null;  

            if(pathInRootDocParts.size()==0){ 
                docFile = DocumentFile.fromTreeUri(con, rootDocFile.getUri()); 
            }
            else{
                for(int i=pathInRootDocParts.size()-1;i>=0;i--){
                    if(docFile==null){docFile = rootDocFile.findFile(pathInRootDocParts.get(i));}
                    else{docFile = docFile.findFile(pathInRootDocParts.get(i)); }  
                }
            }
            if(docFile!=null && docFile.canWrite()){ 
                return docFile; 
            }else{
                return null;
            }

        }
    }
    return null; 
}
4

2 に答える 2

2

Document.fromFile(File)最近のデバイスでは、この目的のために正しく機能していないようです

問題に関する最も多くの情報を入手した回答へのリンクは次のとおりです https://stackoverflow.com/a/26765884/971355

主な手順は次のとおりです。

  1. SDカードへの書き込みアクセスを取得し、電話の再起動後も永続化します

    public void onActivityResult(int requestCode, int resultCode, Intent resultData) {

    ...

    getContentResolver().takePersistableUriPermission(treeUri, Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

    }

  2. その後、ファイルやフォルダーがどれほど深くネストされていてもアクセスできますが、対処しなければならない複雑な URL に注意する必要があります。

あなたが私のような Linux の人間で、すべてがファイルであり、すべてが単純であると考えていた場合。今回はそうではありません。SD カード上のファイル。たとえば/storage/13E5-2604/testDir/test.txt13E5-2604SD カードのルート フォルダは次のようになります。

content://com.android.externalstorage.documents/tree/13E5-2604%3A/document/13E5-2604%3A%2FtestDir%2Ftest.txt

新しい現実で。なぜ彼らはそれをそれほど複雑にしたのか - 別の質問です...

通常の Linux パスをこの新しい現実のパスに透過的かつ簡単に変換する API メソッドを知りません。しかし、それを行った後 ( %3Ais : および%2Fis / 参照用にhttps://www.w3schools.com/tags/ref_urlencode.ASPを参照)、次のインスタンスをDocumentFile簡単に作成できます。

DocumentFile txtFile = DocumentFile.fromSingleUri(context,
Uri.parse("content://com.android.externalstorage.documents/tree/13E5-2604%3A/document/13E5-2604%3A%2FtestDir%2Ftest.txt"));

それに書き込みます。

于 2021-01-25T20:39:45.500 に答える