/***
Copyright (c) 2016 CommonsWare, LLC
Licensed under the Apache License, Version 2.0 (the "License"); you may not
use this file except in compliance with the License. You may obtain a copy
of the License at http://www.apache.org/licenses/LICENSE-2.0. Unless required
by applicable law or agreed to in writing, software distributed under the
License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS
OF ANY KIND, either express or implied. See the License for the specific
language governing permissions and limitations under the License.
Covered in detail in the book _The Busy Coder's Guide to Android Development_
https://commonsware.com/Android
*/
package com.commonsware.android.tte;
import android.app.IntentService;
import android.content.ContentResolver;
import android.content.Context;
import android.content.Intent;
import android.content.UriPermission;
import android.net.Uri;
import android.support.v4.provider.DocumentFile;
import android.util.Log;
import org.greenrobot.eventbus.EventBus;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.Reader;
public class DocumentStorageService extends IntentService {
private static final String EXTRA_CLOSING="isClosing";
public static void loadDocument(Context ctxt, Uri document) {
Intent i=new Intent(ctxt, DocumentStorageService.class)
.setAction(Intent.ACTION_OPEN_DOCUMENT)
.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
.setData(document);
ctxt.startService(i);
}
public static void saveDocument(Context ctxt, Uri document,
String text, boolean isClosing) {
Intent i=new Intent(ctxt, DocumentStorageService.class)
.setAction(Intent.ACTION_EDIT)
.setData(document)
.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION|
Intent.FLAG_GRANT_WRITE_URI_PERMISSION)
.putExtra(Intent.EXTRA_TEXT, text)
.putExtra(EXTRA_CLOSING, isClosing);
ctxt.startService(i);
}
public DocumentStorageService() {
super("DocumentStorageService");
}
@Override
protected void onHandleIntent(Intent intent) {
if (Intent.ACTION_OPEN_DOCUMENT.equals(intent.getAction())) {
load(intent.getData());
}
else if (Intent.ACTION_EDIT.equals(intent.getAction())) {
save(intent.getData(),
intent.getStringExtra(Intent.EXTRA_TEXT),
intent.getBooleanExtra(EXTRA_CLOSING, false));
}
}
private void load(Uri document) {
try {
boolean weHavePermission=false;
boolean isContent=
ContentResolver.SCHEME_CONTENT.equals(document.getScheme());
if (isContent) {
int perms=Intent.FLAG_GRANT_READ_URI_PERMISSION
| Intent.FLAG_GRANT_WRITE_URI_PERMISSION;
getContentResolver()
.takePersistableUriPermission(document, perms);
for (UriPermission perm :
getContentResolver().getPersistedUriPermissions()) {
if (perm.getUri().equals(document)) {
weHavePermission=true;
}
}
}
else {
weHavePermission=true;
}
if (weHavePermission) {
try {
InputStream is=
getContentResolver().openInputStream(document);
try {
String text=slurp(is);
DocumentFile docFile;
if (isContent) {
docFile=DocumentFile.fromSingleUri(this, document);
}
else {
docFile=DocumentFile.fromFile(new File(document.getPath()));
}
EventBus
.getDefault()
.post(
new DocumentLoadedEvent(document, text,
docFile.getName(), docFile.canWrite()));
}
finally {
is.close();
}
}
catch (Exception e) {
Log.e(getClass().getSimpleName(),
"Exception loading "+document.toString(), e);
EventBus
.getDefault()
.post(new DocumentLoadErrorEvent(document, e));
}
}
else {
Log.e(getClass().getSimpleName(),
"We failed to get permissions for "+document.toString());
EventBus
.getDefault()
.post(new DocumentPermissionFailureEvent(document));
}
}
catch (SecurityException e) {
Log.e(getClass().getSimpleName(),
"Exception getting permissions for "+document.toString(), e);
EventBus
.getDefault()
.post(new DocumentPermissionFailureEvent(document));
}
}
private void save(Uri document, String text, boolean isClosing) {
boolean isContent=
ContentResolver.SCHEME_CONTENT.equals(document.getScheme());
try {
OutputStream os=
getContentResolver().openOutputStream(document, "w");
OutputStreamWriter osw=new OutputStreamWriter(os);
try {
osw.write(text);
osw.flush();
if (isClosing && isContent) {
int perms=Intent.FLAG_GRANT_READ_URI_PERMISSION
| Intent.FLAG_GRANT_WRITE_URI_PERMISSION;
getContentResolver()
.releasePersistableUriPermission(document, perms);
}
EventBus
.getDefault()
.post(new DocumentSavedEvent(document));
}
finally {
osw.close();
}
}
catch (Exception e) {
Log.e(getClass().getSimpleName(),
"Exception saving "+document.toString(), e);
EventBus
.getDefault()
.post(new DocumentSaveErrorEvent(document, e));
}
}
// based on http://stackoverflow.com/a/309718/115145
private static String slurp(final InputStream is)
throws IOException {
final char[] buffer=new char[8192];
final StringBuilder out=new StringBuilder();
final Reader in=new InputStreamReader(is, "UTF-8");
int rsz=in.read(buffer, 0, buffer.length);
while (rsz>0) {
out.append(buffer, 0, rsz);
rsz=in.read(buffer, 0, buffer.length);
}
return(out.toString());
}
public static class DocumentEvent {
public final Uri document;
DocumentEvent(Uri document) {
this.document=document;
}
}
public static class DocumentLoadedEvent extends DocumentEvent {
public final String text;
public final String displayName;
public final boolean canWrite;
DocumentLoadedEvent(Uri document, String text,
String displayName, boolean canWrite) {
super(document);
this.text=text;
this.displayName=displayName;
this.canWrite=canWrite;
}
}
public static class DocumentErrorEvent extends DocumentEvent {
public final Exception e;
DocumentErrorEvent(Uri document, Exception e) {
super(document);
this.e=e;
}
}
public static class DocumentLoadErrorEvent
extends DocumentErrorEvent {
DocumentLoadErrorEvent(Uri document, Exception e) {
super(document, e);
}
}
public static class DocumentSaveErrorEvent
extends DocumentErrorEvent {
DocumentSaveErrorEvent(Uri document, Exception e) {
super(document, e);
}
}
public static class DocumentSavedEvent extends DocumentEvent {
DocumentSavedEvent(Uri document) {
super(document);
}
}
public static class DocumentPermissionFailureEvent
extends DocumentEvent {
DocumentPermissionFailureEvent(Uri document) {
super(document);
}
}
}