首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >保存文本文件-作用域存储Android 11

保存文本文件-作用域存储Android 11
EN

Stack Overflow用户
提问于 2022-01-02 03:11:15
回答 1查看 1.6K关注 0票数 0

有没有一种方法可以在外部存储的Android/data文件夹之外创建和保存文本文件?我正在创建一个音乐应用程序,我添加了一个预置管理器,这样用户就可以保存、加载和编辑预置文件(这在Android 11之前工作得很好),我不希望这些文件在卸载时被自动删除。这就像Photoshop在卸载Photoshop时删除了所有Photoshop文档一样,这太可怕了!这些是用户保存的文件,如果用户愿意,它们可以单独删除。

必须有办法解决这个问题,但我还没有找到任何可行的方法。ACTION_OPEN_DOCUMENT_TREE看起来很有希望,直到我看到Android也删除了这个选项。

https://developer.android.com/about/versions/11/privacy/storage

您不能再使用ACTION_OPEN_DOCUMENT_TREE意图操作请求对以下目录的访问:

  • 是内部存储卷的根目录。
  • 是设备制造商认为可靠的每个SD卡卷的根目录,无论该卡是模拟的还是可移动的。一个可靠的卷是一个应用程序大部分时间都可以成功访问的卷。
  • 下载目录.

我读过Google只允许MANAGE_EXTERNAL_STORAGE在需要它的应用程序(如文件浏览器、反病毒等)中使用,这些应用程序在我的情况下可能不起作用。我不想只依赖于针对老API的API,所以requestLegacyExternalStorage也不能工作。

我查过的一切似乎都是死胡同。还有什么我能做的吗?

下面是一个简短的测试程序(我正在使用LibGDX),它目前只能保存到根位置:

代码语言:javascript
运行
复制
Android/data/com.mygdx.filetest/files/

核FileTest.java

代码语言:javascript
运行
复制
package com.mygdx.filetest;

import com.badlogic.gdx.ApplicationListener;
import com.badlogic.gdx.Files;
import com.badlogic.gdx.Gdx;
import com.badlogic.gdx.files.FileHandle;
import com.badlogic.gdx.utils.ScreenUtils;

public class FileTest implements ApplicationListener {
    private NativePermissions permissions;
    private Files.FileType fileType;
    private String directory = "TestDir/";
    private String name = "text.txt";
       
    public FileTest(final NativePermissions permissions){     
        this.permissions = permissions;
    }
    
    @Override
    public void create(){
        fileType = getFileType();
        if (permissions != null){
            permissions.checkExternalStoragePermission();
        } else {
            permissionGranted();
        }
    }
    private Files.FileType getFileType(){
        switch(Gdx.app.getType()) {
            case Android:
                return Files.FileType.External;
            default:
                return Files.FileType.Local;
        }
    }
    
    @Override public void render(){ ScreenUtils.clear(0.4f, 0.4f, 0.4f, 1); }
    @Override public void resize(int width, int height) {}
    @Override public void pause(){}
    @Override public void resume(){}
    @Override public void dispose (){}
    
    public void permissionGranted() {
        FileHandle fileHandle = Gdx.files.getFileHandle(directory+name, fileType); 
        if (fileHandle!=null) fileHandle.writeString("test", false);
    }
}

android AndroidLauncher.java

代码语言:javascript
运行
复制
package com.mygdx.filetest;

import android.Manifest;
import android.app.AlertDialog;
import android.content.DialogInterface;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.widget.Toast;

import androidx.core.app.ActivityCompat;

import com.badlogic.gdx.backends.android.AndroidApplication;
import com.badlogic.gdx.backends.android.AndroidApplicationConfiguration;

public class AndroidLauncher extends AndroidApplication {
    private final FileTest application;
    private final int STORAGE_PERMISSION_CODE = 1;
    private boolean dialogBoxShowing = false;
    
    public AndroidLauncher(){
        final AndroidPermissions permissions = new AndroidPermissions(this);
        application = new FileTest(permissions);
    }
    
    @Override
    protected void onCreate (Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        AndroidApplicationConfiguration config = new AndroidApplicationConfiguration();
        initialize(application, config);
    }
    @Override
    public void onRequestPermissionsResult(final int requestCode, final String permissions[], final int[] grantResults) {
        dialogBoxShowing = false;
        if (requestCode == STORAGE_PERMISSION_CODE) {
            if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                Toast.makeText(this, "Permission GRANTED", Toast.LENGTH_SHORT).show();
                permissionGranted();
            } else {
                Toast.makeText(this, "Permission DENIED", Toast.LENGTH_SHORT).show();
            }
        }
    }
    public void promptExternalStoragePermission() {
        if (dialogBoxShowing) return;
        dialogBoxShowing = true;
        this.runOnUiThread(new Runnable() {
            @Override
            public void run() {
                final AlertDialog.Builder builder = new AlertDialog.Builder(AndroidLauncher.this);
                builder.setMessage("To save user presets and custom settings, allow access to your phone’s storage.");
                builder.setCancelable(false);
                // reverse these buttons to put "NO" on left and "YES" on right
                builder.setPositiveButton("NOT NOW", new DialogInterface.OnClickListener(){
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        dialogBoxShowing = false;
                        dialog.dismiss();
                    }
                });
                builder.setNegativeButton("CONTINUE", new DialogInterface.OnClickListener(){
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        ActivityCompat.requestPermissions(AndroidLauncher.this, new String[] {Manifest.permission.WRITE_EXTERNAL_STORAGE}, STORAGE_PERMISSION_CODE);
                    }
                });
                builder.create().show();
            }
        });
    }
    public void permissionGranted(){ application.permissionGranted(); }
}

核NativePermissions.java

代码语言:javascript
运行
复制
package com.mygdx.filetest;

public interface NativePermissions {
    public void checkExternalStoragePermission();
}

android AndroidPermissions.java

代码语言:javascript
运行
复制
package com.mygdx.filetest;

import android.Manifest;
import android.content.pm.PackageManager;

import androidx.core.content.ContextCompat;

public class AndroidPermissions implements NativePermissions {
    private final AndroidLauncher context;
    
    public AndroidPermissions(final AndroidLauncher context){
        this.context = context;
    }
    
    @Override
    public void checkExternalStoragePermission() {
        if (ContextCompat.checkSelfPermission(context, Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED){
            context.permissionGranted();
        } else {
            context.promptExternalStoragePermission();
        }
    }
}

android AndroidManifest.xml

代码语言:javascript
运行
复制
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.mygdx.filetest">
    
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
    
    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:isGame="true"
        android:appCategory="game"
        android:label="@string/app_name"
        android:theme="@style/GdxTheme" >
        <activity
            android:name="com.mygdx.filetest.AndroidLauncher"
            android:label="@string/app_name" 
            android:screenOrientation="fullUser"
            android:configChanges="keyboard|keyboardHidden|navigation|orientation|screenSize|screenLayout">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
</manifest>
EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2022-01-02 06:58:23

一堆小题大做。

您可以以传统的方式将文件保存到公共文档目录中。

或者将SAF与ACTION_OPEN_DOCUMENT_TREE一起用于该目录。

两者都不需要“所有文件访问”。

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/70553358

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档