Flutter[Android]:无法发布具有管理外部存储权限的应用程序

问题描述 投票:0回答:4

我们需要允许用户将文件存储在外部存储中,为此,我们在应用程序中使用

MANAGE_EXTERNAL_STORAGE
权限。

理想情况下,对于 Android SDK 版本 30 及以上,我们使用

Permission.manageExternalStorage
,对于低于 30 的 Android SDK 版本,我们使用
Permission.storage
,如下面的代码所示

  // This func is added to access scope storage to export csv files
  static Future<bool> externalStoragePermission(BuildContext context) async {
    final androidVersion = await DeviceInfoPlugin().androidInfo;

    if ((androidVersion.version.sdkInt ?? 0) >= 30) {
      return await checkManageStoragePermission(context);
    } else {
      return await checkStoragePermission(context);
    }
  }

  static Future<bool> checkManageStoragePermission(BuildContext context) async {
    return (await Permission.manageExternalStorage.isGranted ||
        await Permission.manageExternalStorage.request().isGranted);
  }

  static Future<bool> checkStoragePermission(BuildContext context,
      {String? storageTitle, String? storageSubMessage}) async {
    if (await Permission.storage.isGranted ||
        await Permission.storage.request().isGranted) {
      return true;
    } else {
      openBottomSheet(
        title: storageTitle ?? Str.of(context).storagePermissionRequired,
        message: storageSubMessage ?? Str.of(context).storageSubMessage,
      ).show(context);
      return false;
    }
  }

通过上述实现,在开发和内部发布时一切正常,但 Google Play 控制台拒绝了该应用程序,并显示以下拒绝信息(我们还提交了原因以及管理存储权限)。

android flutter dart android-external-storage
4个回答
1
投票

我找到了这个问题的解决方案

在 Android 级别低于 11 或高于 11 时进行了正确测试

首先请检查您的 onpressd 按钮上的 Android 设备是否像这样

if(Platform.isAndroid)
{
  downloadAndSavePDF('your network link');
}

=> 然后方法

Future<void> downloadAndSavePDF(String pdfUrl, String fileName) async {
final response = await http.get(Uri.parse(pdfUrl));
if (response.contentLength == 0) {
  return Future.value(null);
}
try {
  if (response.statusCode == 200) {
    // Get the downloads directory
    Directory downloadsDirectory =
        Directory('/storage/emulated/0/Download/');
    final File file = File('${downloadsDirectory.path}/$fileName');
    if (file.existsSync()) {
      Navigator.pop(context);
      CustomSnakeBar(errorMessage: 'File already downloaded')
          .snakeBar(context);
      return;
    }
    // Write the PDF content to the file
    await file.writeAsBytes(response.bodyBytes);
    Navigator.pop(context);
    CustomSnakeBar(
            errorMessage: 'PDF downloaded', colorText: ConstColor.green)
        .snakeBar(context);
  } else {
    CustomSnakeBar(errorMessage: ConstString.somethingWentWrong)
        .snakeBar(context);
  }
} catch (e) {
  Navigator.pop(context);
  CustomSnakeBar(errorMessage: ConstString.somethingWentWrong)
      .snakeBar(context);
}
}

0
投票

我在我的项目中找到并应用了以下解决方案来解决上述问题。

  1. 我们必须使用SAF(存储访问框架)代表存储权限shared_storage,这将允许我们授予共享存储中特定目录的权限,并且我们可以将文件存储在那里。我还添加了下面相同的代码示例。
  Future<void> exportFile({
    required String csvData,
    required String fileName,
  }) async {
    Uri? selectedUriDir;
    final pref = await SharedPreferences.getInstance();
    final scopeStoragePersistUrl = pref.getString('scopeStoragePersistUrl');
   
    // Check User has already grant permission to any directory or not
    if (scopeStoragePersistUrl != null &&
        await isPersistedUri(Uri.parse(scopeStoragePersistUrl)) &&
        (await exists(Uri.parse(scopeStoragePersistUrl)) ?? false)) {
      selectedUriDir = Uri.parse(scopeStoragePersistUrl);
    } else {
      selectedUriDir = await openDocumentTree();
      await pref.setString('scopeStoragePersistUrl', selectedUriDir.toString());
    }
    if (selectedUriDir == null) {
      return false;
    }
    try {
      final existingFile = await findFile(selectedUriDir, fileName);
      if (existingFile != null && existingFile.isFile) {
        debugPrint("Found existing file ${existingFile.uri}");
        await delete(existingFile.uri);
      }
      final newDocumentFile = await createFileAsString(
        selectedUriDir,
        mimeType: AppConstants.csvMimeTypeWhileExport,
        content: csvData,
        displayName: fileName,
      );
      return newDocumentFile != null;
    } catch (e) {
      debugPrint("Exception while create new file: ${e.toString()}");
      return false;
    }
  }

0
投票

如果不是需要访问设备外部存储的应用程序(例如防病毒应用程序),并且您想在 Flutter 中下载文件,您可以在安装过程中使用包 path_provider 来使用您的应用程序特定目录提供程序。用它来下载文件

final response = await http.get(Uri.parse(URL!));
final appDir = await getApplicationDocumentsDirectory();
File file = File(appDir.path + '/${filename}');
try {
  await file.writeAsBytes(response.bodyBytes);
  print('success');
} catch (e) {
  print(e.toString());
}

或者您可以使用 dio 包下载文件,它管理为您创建开箱即用的文件所需的一切

try {
  Response response = await Dio().download(
    downloadURL!,
    appDir.path + '/${filename}',
    onReceiveProgress: (received, total) {
      // You can handle download progress here
      print('Received: $received, Total: $total');
    },
  );
  print('success');
} catch (e) {
  print('Error: $e');
}

-1
投票

我们需要允许用户将文件存储在外部存储中,为此,我们在应用程序中使用 MANAGE_EXTERNAL_STORAGE 权限。

您不需要该权限即可在公共目录(例如下载、文档、DCIM、图片等)中创建自己的子目录。

此外,您不需要该权限即可在这些目录和子目录中创建文件。

© www.soinside.com 2019 - 2024. All rights reserved.