Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Android Q: file.mkdirs() returns false

We have an app that uses external storage to store some temporary files: images, binary data. The code for that has been working for a few years without big changes until recently. On Android Q it doesn't work:

File f = new File(Environment.getExternalStorageDirectory().toString() + File.separator + MainActivity.APP_DIR) f.mkdirs(); // do sth with f 

The mkdirs now returns just false.

Required permission is provided in the manifest:

    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> 

The code works fine on previous versions of Android. Is there some system level change to this type of access? If so, what is the workaround?

like image 776
khusrav Avatar asked Jun 29 '19 21:06

khusrav


2 Answers

There was huge privacy change in android Q by introducing Scoped Storage.

Since Q beta 4 it's possible to opt-out of that feature by:

  • targeting API 28 (or lower)
  • using requestLegacyExternalStorage manifest attribute (while targetting API 29):

<manifest ... >   <!-- This attribute is "false" by default on apps targeting Android Q. -->   <application android:requestLegacyExternalStorage="true" ... >     ...   </application> </manifest> 

edit: as mentioned in other answer this does not work if app is targeting API 30 - Android 11 devices will ignore legacy storage flag.

edit 2: heads up for anyone planning to publish on play store - soon usage of this flag will be restricted (new and updated apps won't be accepted) unless its required for core functionality (e.g. file manager)

like image 167
Pawel Avatar answered Sep 28 '22 05:09

Pawel


UPDATE: Since Android 11 scoped storage is enforced. Apps that target Android 10 (API level 29) can still request the requestLegacyExternalStorage attribute. This flag allows apps to temporarily opt-out of the changes associated with scoped storage, such as granting access to different directories and different types of media files. After you update your app to target Android 11, the system ignores the requestLegacyExternalStorage flag.


In API level 29 direct access to shared/external storage devices is deprecated. When an app targets Build.VERSION_CODES.Q, the path returned from getExternalStorageDirectory() method is no longer directly accessible to apps. Apps can continue to access content stored on shared/external storage by migrating to alternatives such as Context#getExternalFilesDir(String), MediaStore, or Intent#ACTION_OPEN_DOCUMENT.

It's a best practice to use scoped storage unless your app needs access to a file that doesn't reside in the app-specific directory.

Those who face the issue with managing files in Android-Q may read through this article to know further.

like image 22
Anoop M Maddasseri Avatar answered Sep 28 '22 07:09

Anoop M Maddasseri