/* * Copyright (C) 2013 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; import static com.android.documentsui.Shared.DEBUG; import static com.android.documentsui.Shared.TAG; import static com.android.documentsui.State.SORT_ORDER_DISPLAY_NAME; import static com.android.documentsui.State.SORT_ORDER_LAST_MODIFIED; import static com.android.documentsui.State.SORT_ORDER_SIZE; import android.content.AsyncTaskLoader; import android.content.ContentProviderClient; import android.content.ContentResolver; import android.content.Context; import android.database.Cursor; import android.net.Uri; import android.os.CancellationSignal; import android.os.OperationCanceledException; import android.os.RemoteException; import android.provider.DocumentsContract; import android.provider.DocumentsContract.Document; import android.util.Log; import com.android.documentsui.dirlist.DirectoryFragment; import com.android.documentsui.model.DocumentInfo; import com.android.documentsui.model.RootInfo; import libcore.io.IoUtils; import java.io.FileNotFoundException; public class DirectoryLoader extends AsyncTaskLoader { private static final String[] SEARCH_REJECT_MIMES = new String[] { Document.MIME_TYPE_DIR }; private final ForceLoadContentObserver mObserver = new ForceLoadContentObserver(); private final int mType; private final RootInfo mRoot; private final Uri mUri; private final int mUserSortOrder; private final boolean mSearchMode; private DocumentInfo mDoc; private CancellationSignal mSignal; private DirectoryResult mResult; public DirectoryLoader(Context context, int type, RootInfo root, DocumentInfo doc, Uri uri, int userSortOrder, boolean inSearchMode) { super(context, ProviderExecutor.forAuthority(root.authority)); mType = type; mRoot = root; mUri = uri; mUserSortOrder = userSortOrder; mDoc = doc; mSearchMode = inSearchMode; } @Override public final DirectoryResult loadInBackground() { synchronized (this) { if (isLoadInBackgroundCanceled()) { throw new OperationCanceledException(); } mSignal = new CancellationSignal(); } final ContentResolver resolver = getContext().getContentResolver(); final String authority = mUri.getAuthority(); final DirectoryResult result = new DirectoryResult(); result.doc = mDoc; // Use default document when searching if (mSearchMode) { final Uri docUri = DocumentsContract.buildDocumentUri( mRoot.authority, mRoot.documentId); try { mDoc = DocumentInfo.fromUri(resolver, docUri); } catch (FileNotFoundException e) { Log.w(TAG, "Failed to query", e); result.exception = e; return result; } } if (mUserSortOrder != State.SORT_ORDER_UNKNOWN) { result.sortOrder = mUserSortOrder; } else { if ((mDoc.flags & Document.FLAG_DIR_PREFERS_LAST_MODIFIED) != 0) { result.sortOrder = State.SORT_ORDER_LAST_MODIFIED; } else { result.sortOrder = State.SORT_ORDER_DISPLAY_NAME; } } // Search always uses ranking from provider if (mSearchMode) { result.sortOrder = State.SORT_ORDER_UNKNOWN; } if (DEBUG) Log.d(TAG, "userSortOrder=" + mUserSortOrder + ", sortOrder=" + result.sortOrder); ContentProviderClient client = null; Cursor cursor = null; try { client = DocumentsApplication.acquireUnstableProviderOrThrow(resolver, authority); cursor = client.query( mUri, null, null, null, getQuerySortOrder(result.sortOrder), mSignal); if (cursor == null) { throw new RemoteException("Provider returned null"); } cursor.registerContentObserver(mObserver); cursor = new RootCursorWrapper(mUri.getAuthority(), mRoot.rootId, cursor, -1); if (mSearchMode) { // Filter directories out of search results, for now cursor = new FilteringCursorWrapper(cursor, null, SEARCH_REJECT_MIMES); } result.client = client; result.cursor = cursor; } catch (Exception e) { Log.w(TAG, "Failed to query", e); result.exception = e; ContentProviderClient.releaseQuietly(client); } finally { synchronized (this) { mSignal = null; } } return result; } @Override public void cancelLoadInBackground() { super.cancelLoadInBackground(); synchronized (this) { if (mSignal != null) { mSignal.cancel(); } } } @Override public void deliverResult(DirectoryResult result) { if (isReset()) { IoUtils.closeQuietly(result); return; } DirectoryResult oldResult = mResult; mResult = result; if (isStarted()) { super.deliverResult(result); } if (oldResult != null && oldResult != result) { IoUtils.closeQuietly(oldResult); } } @Override protected void onStartLoading() { if (mResult != null) { deliverResult(mResult); } if (takeContentChanged() || mResult == null) { forceLoad(); } } @Override protected void onStopLoading() { cancelLoad(); } @Override public void onCanceled(DirectoryResult result) { IoUtils.closeQuietly(result); } @Override protected void onReset() { super.onReset(); // Ensure the loader is stopped onStopLoading(); IoUtils.closeQuietly(mResult); mResult = null; getContext().getContentResolver().unregisterContentObserver(mObserver); } public static String getQuerySortOrder(int sortOrder) { switch (sortOrder) { case SORT_ORDER_DISPLAY_NAME: return Document.COLUMN_DISPLAY_NAME + " ASC"; case SORT_ORDER_LAST_MODIFIED: return Document.COLUMN_LAST_MODIFIED + " DESC"; case SORT_ORDER_SIZE: return Document.COLUMN_SIZE + " DESC"; default: return null; } } }