blob: bab675308f0e41fd98ebbf4300dbefa75618cab7 [file] [log] [blame]
/*
* Copyright (C) 2015 The Android Open Source Project
*
* 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.
*/
package com.android.documentsui.files;
import static com.android.documentsui.base.DocumentInfo.getCursorInt;
import static com.android.documentsui.base.DocumentInfo.getCursorString;
import static com.android.documentsui.base.Shared.MAX_DOCS_IN_INTENT;
import static com.android.documentsui.base.SharedMinimal.DEBUG;
import android.content.ClipData;
import android.content.ClipDescription;
import android.content.Context;
import android.content.Intent;
import android.content.QuickViewConstants;
import android.content.pm.PackageManager;
import android.content.res.Resources;
import android.database.Cursor;
import android.net.Uri;
import android.provider.DocumentsContract;
import android.provider.DocumentsContract.Document;
import android.text.TextUtils;
import android.util.Log;
import android.util.Range;
import androidx.annotation.Nullable;
import com.android.documentsui.Model;
import com.android.documentsui.R;
import com.android.documentsui.base.DebugFlags;
import com.android.documentsui.base.DocumentInfo;
import com.android.documentsui.base.UserId;
import com.android.documentsui.roots.RootCursorWrapper;
import java.util.ArrayList;
import java.util.List;
/**
* Provides support for gather a list of quick-viewable files into a quick view intent.
*/
public final class QuickViewIntentBuilder {
// trusted quick view package can be set via system property on debug builds.
// Unfortunately when the value is set, it interferes with testing (supercedes
// any value set in the resource system).
// For that reason when trusted quick view package is set to this magic value
// we won't honor the system property.
public static final String IGNORE_DEBUG_PROP = "*disabled*";
private static final String TAG = "QuickViewIntentBuilder";
private static final String[] IN_ARCHIVE_FEATURES = {};
private static final String[] FULL_FEATURES = {
QuickViewConstants.FEATURE_VIEW,
QuickViewConstants.FEATURE_EDIT,
QuickViewConstants.FEATURE_DELETE,
QuickViewConstants.FEATURE_SEND,
QuickViewConstants.FEATURE_DOWNLOAD,
QuickViewConstants.FEATURE_PRINT
};
private static final String[] PICKER_FEATURES = {
QuickViewConstants.FEATURE_VIEW
};
private final DocumentInfo mDocument;
private final Model mModel;
private final PackageManager mPackageMgr;
private final Resources mResources;
private final boolean mFromPicker;
public QuickViewIntentBuilder(
Context context,
Resources resources,
DocumentInfo doc,
Model model,
boolean fromPicker) {
assert(context != null);
assert(resources != null);
assert(doc != null);
assert(model != null);
mPackageMgr = doc.userId.getPackageManager(context);
mResources = resources;
mDocument = doc;
mModel = model;
mFromPicker = fromPicker;
}
/**
* Builds the intent for quick viewing. Short circuits building if a handler cannot
* be resolved; in this case {@code null} is returned.
*/
@Nullable public Intent build() {
if (DEBUG) {
Log.d(TAG, "Preparing intent for doc:" + mDocument.documentId);
}
String trustedPkg = getQuickViewPackage();
if (!TextUtils.isEmpty(trustedPkg)) {
Intent intent = new Intent(Intent.ACTION_QUICK_VIEW);
intent.setDataAndType(mDocument.getDocumentUri(), mDocument.mimeType);
intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION
| Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
intent.setPackage(trustedPkg);
if (hasRegisteredHandler(intent)) {
includeQuickViewFeaturesFlag(intent, mDocument, mFromPicker);
final ArrayList<Uri> uris = new ArrayList<>();
final int documentLocation = collectViewableUris(uris);
final Range<Integer> range = computeSiblingsRange(uris, documentLocation);
ClipData clipData = null;
ClipData.Item item;
Uri uri;
for (int i = range.getLower(); i <= range.getUpper(); i++) {
uri = uris.get(i);
item = new ClipData.Item(uri);
if (DEBUG) {
Log.d(TAG, "Including file: " + uri);
}
if (clipData == null) {
clipData = new ClipData(
"URIs", new String[] { ClipDescription.MIMETYPE_TEXT_URILIST },
item);
} else {
clipData.addItem(item);
}
}
// The documentLocation variable contains an index in "uris". However,
// ClipData contains a slice of "uris", so we need to shift the location
// so it points to the same Uri.
intent.putExtra(Intent.EXTRA_INDEX, documentLocation - range.getLower());
intent.setClipData(clipData);
return intent;
} else {
Log.e(TAG, "Can't resolve trusted quick view package: " + trustedPkg);
}
}
return null;
}
private String getQuickViewPackage() {
String resValue = mResources.getString(R.string.trusted_quick_viewer_package);
// Allow automated tests to hard-disable quick viewing.
if (IGNORE_DEBUG_PROP.equals(resValue)) {
return "";
}
// Allow users of debug devices to override default quick viewer
// for the purposes of testing.
if (DEBUG) {
String quickViewer = DebugFlags.getQuickViewer();
if (quickViewer != null) {
return quickViewer;
}
}
return resValue;
}
private int collectViewableUris(ArrayList<Uri> uris) {
final String[] siblingIds = mModel.getModelIds();
uris.ensureCapacity(siblingIds.length);
int documentLocation = 0;
Cursor cursor;
String mimeType;
String id;
String authority;
UserId userId;
Uri uri;
boolean hasNonMatchingDocumentUser = false;
// Cursor's are not guaranteed to be immutable. Hence, traverse it only once.
for (int i = 0; i < siblingIds.length; i++) {
cursor = mModel.getItem(siblingIds[i]);
if (cursor == null) {
if (DEBUG) {
Log.d(TAG,
"Unable to obtain cursor for sibling document, modelId: "
+ siblingIds[i]);
}
continue;
}
mimeType = getCursorString(cursor, Document.COLUMN_MIME_TYPE);
if (Document.MIME_TYPE_DIR.equals(mimeType)) {
if (DEBUG) {
Log.d(TAG,
"Skipping directory, not supported by quick view. modelId: "
+ siblingIds[i]);
}
continue;
}
userId = UserId.of(getCursorInt(cursor, RootCursorWrapper.COLUMN_USER_ID));
if (!userId.equals(mDocument.userId)) {
// If there is any document in the model does not have the same user as
// mDocument, we will not add any siblings and the user for security reason.
// Although the quick view package is trusted, the trusted quick view package may
// not notice it is a cross-profile uri and may allow other app to handle this uri.
if (DEBUG) {
Log.d(TAG,
"Skipping document from the other user. modelId: "
+ siblingIds[i]);
}
hasNonMatchingDocumentUser = true;
continue;
}
id = getCursorString(cursor, Document.COLUMN_DOCUMENT_ID);
authority = getCursorString(cursor, RootCursorWrapper.COLUMN_AUTHORITY);
if (UserId.CURRENT_USER.equals(userId)) {
uri = DocumentsContract.buildDocumentUri(authority, id);
} else {
uri = userId.buildDocumentUriAsUser(authority, id);
}
if (id.equals(mDocument.documentId)) {
uris.add(uri);
documentLocation = uris.size() - 1; // Position in "uris", not in the model.
if (DEBUG) {
Log.d(TAG, "Found starting point for QV. " + documentLocation);
}
} else if (!hasNonMatchingDocumentUser) {
uris.add(uri);
}
}
if (!uris.isEmpty() && hasNonMatchingDocumentUser) {
if (DEBUG) {
Log.d(TAG,
"Remove all other uris except the document uri");
}
Uri documentUri = uris.get(documentLocation);
uris.clear();
uris.add(documentUri);
return 0; // index of the item in a singleton list is 0.
}
return documentLocation;
}
private boolean hasRegisteredHandler(Intent intent) {
// Try to resolve the intent. If a matching app isn't installed, it won't resolve.
return intent.resolveActivity(mPackageMgr) != null;
}
private static void includeQuickViewFeaturesFlag(Intent intent, DocumentInfo doc,
boolean fromPicker) {
intent.putExtra(
Intent.EXTRA_QUICK_VIEW_FEATURES,
doc.isInArchive() ? IN_ARCHIVE_FEATURES
: fromPicker ? PICKER_FEATURES : FULL_FEATURES);
}
private static Range<Integer> computeSiblingsRange(List<Uri> uris, int documentLocation) {
// Restrict number of siblings to avoid hitting the IPC limit.
// TODO: Remove this restriction once ClipData can hold an arbitrary number of
// items.
int firstSibling;
int lastSibling;
if (documentLocation < uris.size() / 2) {
firstSibling = Math.max(0, documentLocation - MAX_DOCS_IN_INTENT / 2);
lastSibling = Math.min(uris.size() - 1, firstSibling + MAX_DOCS_IN_INTENT - 1);
} else {
lastSibling = Math.min(uris.size() - 1, documentLocation + MAX_DOCS_IN_INTENT / 2);
firstSibling = Math.max(0, lastSibling - MAX_DOCS_IN_INTENT + 1);
}
if (DEBUG) {
Log.d(TAG, "Copmuted siblings from index: " + firstSibling
+ " to: " + lastSibling);
}
return new Range(firstSibling, lastSibling);
}
}