Доступ к файлам SD-карты
Используйте диалоговое окно DOCUMENT_TREE
, чтобы получить Uri
.
. Сообщите пользователю о том, как выбрать sd-card
вдиалог.(с изображениями или gif-анимацией)
// call for document tree dialog
Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT_TREE);
startActivityForResult(intent, REQUEST_CODE_OPEN_DOCUMENT_TREE);
На onActivityResult
у вас будет выбранный каталог Uri
.(sdCardUri)
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
switch (requestCode) {
case REQUEST_CODE_OPEN_DOCUMENT_TREE:
if (resultCode == Activity.RESULT_OK) {
sdCardUri = data.getData();
}
break;
}
}
Теперь необходимо проверить, является ли пользователь
a.выбрал sd-карту
б.выбрал sd-карту, на которой находится наш файл (на некоторых устройствах может быть несколько sd-карт).
Мы проверяем и a, и b, находя файл по иерархии, от корня sd до нашего файла.,Если файл найден, оба условия a и b получены.
//First we get `DocumentFile` from the `TreeUri` which in our case is `sdCardUri`.
DocumentFile documentFile = DocumentFile.fromTreeUri(this, sdCardUri);
//Then we split file path into array of strings.
//ex: parts:{"", "storage", "extSdCard", "MyFolder", "MyFolder", "myImage.jpg"}
// There is a reason for having two similar names "MyFolder" in
//my exmple file path to show you similarity in names in a path will not
//distract our hiarchy search that is provided below.
String[] parts = (file.getPath()).split("\\/");
// findFile method will search documentFile for the first file
// with the expected `DisplayName`
// We skip first three items because we are already on it.(sdCardUri = /storage/extSdCard)
for (int i = 3; i < parts.length; i++) {
if (documentFile != null) {
documentFile = documentFile.findFile(parts[i]);
}
}
if (documentFile == null) {
// File not found on tree search
// User selected a wrong directory as the sd-card
// Here must inform user about how to get the correct sd-card
// and invoke file chooser dialog again.
} else {
// File found on sd-card and it is a correct sd-card directory
// save this path as a root for sd-card on your database(SQLite, XML, txt,...)
// Now do whatever you like to do with documentFile.
// Here I do deletion to provide an example.
if (documentFile.delete()) {// if delete file succeed
// Remove information related to your media from ContentResolver,
// which documentFile.delete() didn't do the trick for me.
// Must do it otherwise you will end up with showing an empty
// ImageView if you are getting your URLs from MediaStore.
//
Uri mediaContentUri = ContentUris.withAppendedId(
MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
longMediaId);
getContentResolver().delete(mediaContentUri , null, null);
}
}
Примечание:
Вы должны предоставить разрешение на доступ к внешнему хранилищу внутри манифеста и дляos> = Зефир внутри приложения.https://stackoverflow.com/a/32175771/2123400
Редактирование файлов SD-карты
Для редактирования существующего изображения на SD-карте вам не нужно выполнять какие-либо из вышеперечисленных шагов, если выхочу вызвать другое приложение, чтобы сделать это для вас.
Здесь мы вызываем все действия (из всех установленных приложений) с возможностью редактирования изображений.(Программисты помечают свои приложения в манифесте, чтобы обеспечить возможность доступа из других приложений (действий).)
в вашем событии click editButton:
String mimeType = getMimeTypeFromMediaContentUri(mediaContentUri);
startActivityForResult(Intent.createChooser(new Intent(Intent.ACTION_EDIT).setDataAndType(mediaContentUri, mimeType).putExtra(Intent.EXTRA_STREAM, mediaContentUri).addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION), "Edit"), REQUEST_CODE_SHARE_EDIT_SET_AS_INTENT);
, и вот как получить mimeType:
public String getMimeTypeFromMediaContentUri(Uri uri) {
String mimeType;
if (uri.getScheme().equals(ContentResolver.SCHEME_CONTENT)) {
ContentResolver cr = getContentResolver();
mimeType = cr.getType(uri);
} else {
String fileExtension = MimeTypeMap.getFileExtensionFromUrl(uri
.toString());
mimeType = MimeTypeMap.getSingleton().getMimeTypeFromExtension(
fileExtension.toLowerCase());
}
return mimeType;
}
Примечание:
В Android KitKat (4.4) не просите пользователя выбрать sd-карту, поскольку в этой версии Android DocumentProvider
не применимо, поэтому у нас нет шансов получить доступ к SD-карте при таком подходе.Посмотрите на уровень API для DocumentProvider
https://developer.android.com/reference/android/provider/DocumentsProvider.html
Я не смог найти ничего, что работает на Android KitKat (4.4).Если вы нашли что-нибудь полезное с KitKat, пожалуйста, поделитесь с нами.
В версиях ниже KitKat доступ к SD-карте уже предоставляется ОС.