[automerger skipped] Merge Coral/Flame into AOSP master am: 4d33f1a127 -s ours am: 0cf898fc34 -s ours
am: 5ce68c862f -s ours
am skip reason: change_id I28864c5b8fdd4cb10628672ff5285ddf8391b2d3 with SHA1 819ff9ef6e is in history
Change-Id: Ic4577f3b651d300d67127a192783e747fc17cd91
diff --git a/perf-tests/Android.bp b/perf-tests/Android.bp
index 963a6da..e745fdf 100644
--- a/perf-tests/Android.bp
+++ b/perf-tests/Android.bp
@@ -20,6 +20,7 @@
static_libs: [
"androidx.legacy_legacy-support-v4",
+ "androidx.test.rules",
"androidx.test.espresso.core",
"mockito-target",
"ub-janktesthelper",
diff --git a/res/values/styles.xml b/res/values/styles.xml
index 10f9b05..2409ed6 100644
--- a/res/values/styles.xml
+++ b/res/values/styles.xml
@@ -91,8 +91,14 @@
<item name="buttonBarNegativeButtonStyle">@style/DialogTextButton</item>
</style>
+ <style name="MaterialAlertDialogStyle" parent="@style/MaterialAlertDialog.MaterialComponents">
+ <item name="backgroundInsetTop">12dp</item>
+ <item name="backgroundInsetBottom">12dp</item>
+ </style>
+
<style name="MaterialAlertDialogTheme" parent="@style/ThemeOverlay.MaterialComponents.MaterialAlertDialog.Centered">
<item name="android:dialogCornerRadius">@dimen/grid_item_radius</item>
+ <item name="alertDialogStyle">@style/MaterialAlertDialogStyle</item>
<item name="buttonBarPositiveButtonStyle">@style/DialogTextButton</item>
<item name="buttonBarNegativeButtonStyle">@style/DialogTextButton</item>
<item name="materialAlertDialogTitleTextStyle">@style/MaterialAlertDialogTitleStyle</item>
diff --git a/src/com/android/documentsui/AbstractActionHandler.java b/src/com/android/documentsui/AbstractActionHandler.java
index 423565d..824d209 100644
--- a/src/com/android/documentsui/AbstractActionHandler.java
+++ b/src/com/android/documentsui/AbstractActionHandler.java
@@ -576,11 +576,15 @@
Context context = mActivity;
if (mState.stack.isRecents()) {
+ final LockingContentObserver observer = new LockingContentObserver(
+ mContentLock, AbstractActionHandler.this::loadDocumentsForCurrentStack);
+ MultiRootDocumentsLoader loader;
+
if (mSearchMgr.isSearching()) {
if (DEBUG) {
Log.d(TAG, "Creating new GlobalSearchLoader.");
}
- return new GlobalSearchLoader(
+ loader = new GlobalSearchLoader(
context,
mProviders,
mState,
@@ -591,13 +595,15 @@
if (DEBUG) {
Log.d(TAG, "Creating new loader recents.");
}
- return new RecentsLoader(
+ loader = new RecentsLoader(
context,
mProviders,
mState,
mExecutors,
mInjector.fileTypeLookup);
}
+ loader.setObserver(observer);
+ return loader;
} else {
Uri contentsUri = mSearchMgr.isSearching()
? DocumentsContract.buildSearchDocumentsUri(
diff --git a/src/com/android/documentsui/CreateDirectoryFragment.java b/src/com/android/documentsui/CreateDirectoryFragment.java
index 756b555..dcaa87a 100644
--- a/src/com/android/documentsui/CreateDirectoryFragment.java
+++ b/src/com/android/documentsui/CreateDirectoryFragment.java
@@ -68,7 +68,6 @@
@Override
public Dialog onCreateDialog(Bundle savedInstanceState) {
final Context context = getActivity();
- final ContentResolver resolver = context.getContentResolver();
final MaterialAlertDialogBuilder builder = new MaterialAlertDialogBuilder(context);
final LayoutInflater dialogInflater = LayoutInflater.from(builder.getContext());
diff --git a/src/com/android/documentsui/DirectoryLoader.java b/src/com/android/documentsui/DirectoryLoader.java
index d65b4c3..63c3bf6 100644
--- a/src/com/android/documentsui/DirectoryLoader.java
+++ b/src/com/android/documentsui/DirectoryLoader.java
@@ -16,20 +16,16 @@
package com.android.documentsui;
-import static com.android.documentsui.base.SharedMinimal.DEBUG;
import static com.android.documentsui.base.SharedMinimal.VERBOSE;
import android.content.ContentProviderClient;
import android.content.ContentResolver;
import android.content.Context;
-import android.database.ContentObserver;
import android.database.Cursor;
import android.net.Uri;
import android.os.Bundle;
import android.os.CancellationSignal;
import android.os.FileUtils;
-import android.os.Handler;
-import android.os.Looper;
import android.os.OperationCanceledException;
import android.os.RemoteException;
import android.provider.DocumentsContract.Document;
@@ -241,28 +237,4 @@
getContext().getContentResolver().unregisterContentObserver(mObserver);
}
-
- private static final class LockingContentObserver extends ContentObserver {
- private final ContentLock mLock;
- private final Runnable mContentChangedCallback;
-
- public LockingContentObserver(ContentLock lock, Runnable contentChangedCallback) {
- super(new Handler(Looper.getMainLooper()));
- mLock = lock;
- mContentChangedCallback = contentChangedCallback;
- }
-
- @Override
- public boolean deliverSelfNotifications() {
- return true;
- }
-
- @Override
- public void onChange(boolean selfChange) {
- if (DEBUG) {
- Log.d(TAG, "Directory content updated.");
- }
- mLock.runWhenUnlocked(mContentChangedCallback);
- }
- }
}
diff --git a/src/com/android/documentsui/LockingContentObserver.java b/src/com/android/documentsui/LockingContentObserver.java
new file mode 100644
index 0000000..bbd7a98
--- /dev/null
+++ b/src/com/android/documentsui/LockingContentObserver.java
@@ -0,0 +1,53 @@
+/*
+ * Copyright (C) 2019 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.base.SharedMinimal.DEBUG;
+import static com.android.documentsui.base.SharedMinimal.TAG;
+
+import android.database.ContentObserver;
+import android.os.Handler;
+import android.os.Looper;
+import android.util.Log;
+
+/**
+ * A custom {@link ContentObserver} which constructed by a {@link ContentLock}
+ * and a {@link Runnable} callback. It will callback when it's onChange and ContentLock is unlock.
+ */
+public final class LockingContentObserver extends ContentObserver {
+ private final ContentLock mLock;
+ private final Runnable mContentChangedCallback;
+
+ public LockingContentObserver(ContentLock lock, Runnable contentChangedCallback) {
+ super(new Handler(Looper.getMainLooper()));
+ mLock = lock;
+ mContentChangedCallback = contentChangedCallback;
+ }
+
+ @Override
+ public boolean deliverSelfNotifications() {
+ return true;
+ }
+
+ @Override
+ public void onChange(boolean selfChange) {
+ if (DEBUG) {
+ Log.d(TAG, "Content updated.");
+ }
+ mLock.runWhenUnlocked(mContentChangedCallback);
+ }
+}
diff --git a/src/com/android/documentsui/MultiRootDocumentsLoader.java b/src/com/android/documentsui/MultiRootDocumentsLoader.java
index 777efe0..11208c1 100644
--- a/src/com/android/documentsui/MultiRootDocumentsLoader.java
+++ b/src/com/android/documentsui/MultiRootDocumentsLoader.java
@@ -17,7 +17,6 @@
package com.android.documentsui;
import static com.android.documentsui.base.SharedMinimal.DEBUG;
-import static com.android.documentsui.base.SharedMinimal.TAG;
import android.app.ActivityManager;
import android.content.ContentProviderClient;
@@ -65,6 +64,9 @@
* and return the combined result.
*/
public abstract class MultiRootDocumentsLoader extends AsyncTaskLoader<DirectoryResult> {
+
+ private static final String TAG = "MultiRootDocsLoader";
+
// TODO: clean up cursor ownership so background thread doesn't traverse
// previously returned cursors for filtering/sorting; this currently races
// with the UI thread.
@@ -83,6 +85,7 @@
private final ProvidersAccess mProviders;
private final Lookup<String, Executor> mExecutors;
private final Lookup<String, String> mFileTypeMap;
+ private LockingContentObserver mObserver;
@GuardedBy("mTasks")
/** A authority -> QueryTask map */
@@ -101,6 +104,8 @@
* @param state current state
* @param executors the executors of authorities
* @param fileTypeMap the map of mime types and file types.
+ * @param lock the selection lock
+ * @param contentChangedCallback callback when content changed
*/
public MultiRootDocumentsLoader(Context context, ProvidersAccess providers, State state,
Lookup<String, Executor> executors, Lookup<String, String> fileTypeMap) {
@@ -126,6 +131,10 @@
}
}
+ public void setObserver(LockingContentObserver observer) {
+ mObserver = observer;
+ }
+
private DirectoryResult loadInBackgroundLocked() {
if (mFirstPassLatch == null) {
// First time through we kick off all the recent tasks, and wait
@@ -322,6 +331,10 @@
FileUtils.closeQuietly(mResult);
mResult = null;
+
+ if (mObserver != null) {
+ getContext().getContentResolver().unregisterContentObserver(mObserver);
+ }
}
// TODO: create better transfer of ownership around cursor to ensure its
@@ -406,6 +419,9 @@
mState.sortModel.addQuerySortArgs(queryArgs);
addQueryArgs(queryArgs);
res[i] = client.query(uri, null, queryArgs, null);
+ if (mObserver != null) {
+ res[i].registerContentObserver(mObserver);
+ }
mCursors[i] = generateResultCursor(rootInfos.get(i), res[i]);
} catch (Exception e) {
Log.w(TAG, "Failed to load " + authority + ", " + rootInfos.get(i).rootId,
diff --git a/tests/common/com/android/documentsui/DialogFragmentTest.java b/tests/common/com/android/documentsui/DialogFragmentTest.java
new file mode 100644
index 0000000..05ad5fb
--- /dev/null
+++ b/tests/common/com/android/documentsui/DialogFragmentTest.java
@@ -0,0 +1,320 @@
+/*
+ * Copyright (C) 2019 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 org.junit.Assert.assertNotNull;
+import static org.junit.Assert.assertNull;
+import static org.junit.Assert.assertTrue;
+import static org.junit.Assert.fail;
+
+import android.app.Activity;
+import android.app.Instrumentation;
+import android.app.UiAutomation;
+import android.content.Intent;
+import android.content.pm.ActivityInfo;
+import android.graphics.Paint;
+import android.os.Build;
+import android.os.ParcelFileDescriptor;
+
+import androidx.fragment.app.FragmentManager;
+import androidx.test.filters.MediumTest;
+import androidx.test.platform.app.InstrumentationRegistry;
+import androidx.test.rule.ActivityTestRule;
+import androidx.test.runner.AndroidJUnit4;
+
+import com.android.documentsui.files.FilesActivity;
+
+import com.google.android.material.textfield.TextInputEditText;
+
+import org.junit.After;
+import org.junit.Before;
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+
+import java.io.FileInputStream;
+import java.io.IOException;
+
+@MediumTest
+@RunWith(AndroidJUnit4.class)
+public class DialogFragmentTest {
+
+ private static final String CREATE_FRAGEMENT_TAG = "create_directory";
+ CreateDirectoryFragment mCreateDirectoryFragment;
+ FragmentManager mFragmentManager;
+ ScreenDensitySession mScreenDensitySession;
+ Intent mFileActivityIntent;
+
+ @Rule
+ public ActivityTestRule<FilesActivity> mActivityTestRule = new ActivityTestRule<>(
+ FilesActivity.class);
+
+ @Before
+ public void setup() {
+ mFileActivityIntent = new Intent(Intent.ACTION_OPEN_DOCUMENT_TREE);
+ mFragmentManager = mActivityTestRule.getActivity().getSupportFragmentManager();
+ mScreenDensitySession = new ScreenDensitySession();
+ }
+
+ @After
+ public void tearDown() {
+ mScreenDensitySession.close();
+ mCreateDirectoryFragment = null;
+ }
+
+ @Test
+ public void testCreateDialogShows() throws Throwable {
+ mActivityTestRule.runOnUiThread(() -> CreateDirectoryFragment.show(mFragmentManager));
+ InstrumentationRegistry.getInstrumentation().waitForIdleSync();
+ mCreateDirectoryFragment =
+ (CreateDirectoryFragment) mFragmentManager.findFragmentByTag(CREATE_FRAGEMENT_TAG);
+
+ assertNotNull("Dialog was null", mCreateDirectoryFragment.getDialog());
+ assertTrue("Dialog was not being shown", mCreateDirectoryFragment.getDialog().isShowing());
+ }
+
+ @Test
+ public void testCreateDialogShowsDismiss() throws Throwable {
+ mActivityTestRule.runOnUiThread(() -> CreateDirectoryFragment.show(mFragmentManager));
+ InstrumentationRegistry.getInstrumentation().waitForIdleSync();
+ mCreateDirectoryFragment =
+ (CreateDirectoryFragment) mFragmentManager.findFragmentByTag(CREATE_FRAGEMENT_TAG);
+
+ assertNotNull("Dialog was null", mCreateDirectoryFragment.getDialog());
+ assertTrue("Dialog was not being shown", mCreateDirectoryFragment.getDialog().isShowing());
+
+ mActivityTestRule.runOnUiThread(() -> mCreateDirectoryFragment.dismiss());
+ InstrumentationRegistry.getInstrumentation().waitForIdleSync();
+
+ assertNull("Dialog should be null after dismiss()", mCreateDirectoryFragment.getDialog());
+ }
+
+ @Test
+ public void testCreateDialogShows_textInputEditText_shouldNotTruncateOnPortrait()
+ throws Throwable {
+ mActivityTestRule.runOnUiThread(() -> CreateDirectoryFragment.show(mFragmentManager));
+ InstrumentationRegistry.getInstrumentation().waitForIdleSync();
+ mCreateDirectoryFragment =
+ (CreateDirectoryFragment) mFragmentManager.findFragmentByTag(CREATE_FRAGEMENT_TAG);
+
+ final TextInputEditText inputView =
+ mCreateDirectoryFragment.getDialog().findViewById(android.R.id.text1);
+
+ assertTrue(inputView.getHeight() > getInputTextHeight(inputView));
+ }
+
+ @Test
+ public void testCreateDialog_textInputEditText_shouldNotTruncateOnLargeDensity()
+ throws Throwable {
+
+ mScreenDensitySession.setLargeDensity();
+ mActivityTestRule.finishActivity();
+ mActivityTestRule.launchActivity(mFileActivityIntent);
+ mFragmentManager = mActivityTestRule.getActivity().getSupportFragmentManager();
+
+ mActivityTestRule.runOnUiThread(() -> CreateDirectoryFragment.show(mFragmentManager));
+ InstrumentationRegistry.getInstrumentation().waitForIdleSync();
+ mCreateDirectoryFragment =
+ (CreateDirectoryFragment) mFragmentManager.findFragmentByTag(CREATE_FRAGEMENT_TAG);
+
+ final TextInputEditText inputView =
+ mCreateDirectoryFragment.getDialog().getWindow().findViewById(android.R.id.text1);
+
+ assertTrue(inputView.getHeight() > getInputTextHeight(inputView));
+
+ }
+
+ @Test
+ public void testCreateDialog_textInputEditText_shouldNotTruncateOnLargerDensity()
+ throws Throwable {
+
+ mScreenDensitySession.setLargerDensity();
+ mActivityTestRule.finishActivity();
+ mActivityTestRule.launchActivity(mFileActivityIntent);
+ mFragmentManager = mActivityTestRule.getActivity().getSupportFragmentManager();
+
+ mActivityTestRule.runOnUiThread(() -> CreateDirectoryFragment.show(mFragmentManager));
+ InstrumentationRegistry.getInstrumentation().waitForIdleSync();
+ mCreateDirectoryFragment =
+ (CreateDirectoryFragment) mFragmentManager.findFragmentByTag(CREATE_FRAGEMENT_TAG);
+
+ final TextInputEditText inputView =
+ mCreateDirectoryFragment.getDialog().getWindow().findViewById(android.R.id.text1);
+
+ assertTrue(inputView.getHeight() > getInputTextHeight(inputView));
+ }
+
+ @Test
+ public void testCreateDialog_textInputEditText_shouldNotTruncateOnLargestDensity()
+ throws Throwable {
+
+ mScreenDensitySession.setLargestDensity();
+ mActivityTestRule.finishActivity();
+ mActivityTestRule.launchActivity(mFileActivityIntent);
+ mFragmentManager = mActivityTestRule.getActivity().getSupportFragmentManager();
+
+ mActivityTestRule.runOnUiThread(() -> CreateDirectoryFragment.show(mFragmentManager));
+ InstrumentationRegistry.getInstrumentation().waitForIdleSync();
+ mCreateDirectoryFragment =
+ (CreateDirectoryFragment) mFragmentManager.findFragmentByTag(CREATE_FRAGEMENT_TAG);
+
+ final TextInputEditText inputView =
+ mCreateDirectoryFragment.getDialog().getWindow().findViewById(android.R.id.text1);
+
+ assertTrue(inputView.getHeight() > getInputTextHeight(inputView));
+ }
+
+ @Test
+ public void testCreateDirectoryFragmentShows_textInputEditText_shouldNotTruncateOnLandscape()
+ throws Throwable {
+ switchOrientation(mActivityTestRule.getActivity());
+ mScreenDensitySession.setLargestDensity();
+ mActivityTestRule.finishActivity();
+ mActivityTestRule.launchActivity(mFileActivityIntent);
+ mFragmentManager = mActivityTestRule.getActivity().getSupportFragmentManager();
+
+ mActivityTestRule.runOnUiThread(() -> CreateDirectoryFragment.show(mFragmentManager));
+ InstrumentationRegistry.getInstrumentation().waitForIdleSync();
+ mCreateDirectoryFragment =
+ (CreateDirectoryFragment) mFragmentManager.findFragmentByTag(CREATE_FRAGEMENT_TAG);
+
+ final TextInputEditText inputView =
+ mCreateDirectoryFragment.getDialog().getWindow().findViewById(android.R.id.text1);
+ Paint paint = inputView.getPaint();
+ final float textSize = paint.getTextSize();
+
+ assertTrue(inputView.getHeight() > Math.round(textSize));
+
+ switchOrientation(mActivityTestRule.getActivity());
+ }
+
+ private static int getInputTextHeight(TextInputEditText v) {
+ Paint paint = v.getPaint();
+ final float textSize = paint.getTextSize();
+ final float textSpace = paint.getFontSpacing();
+ return Math.round(textSize + textSpace);
+ }
+
+ private static void switchOrientation(final Activity activity) {
+ final int[] orientations = getOrientations(activity);
+ activity.setRequestedOrientation(orientations[1]);
+ }
+
+ private static int[] getOrientations(final Activity activity) {
+ final int originalOrientation = activity.getResources().getConfiguration().orientation;
+ final int newOrientation = originalOrientation == ActivityInfo.SCREEN_ORIENTATION_PORTRAIT
+ ? ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE
+ : ActivityInfo.SCREEN_ORIENTATION_PORTRAIT;
+ return new int[]{originalOrientation, newOrientation};
+ }
+
+ private static class ScreenDensitySession implements AutoCloseable {
+ private static final String DENSITY_PROP_DEVICE = "ro.sf.lcd_density";
+ private static final String DENSITY_PROP_EMULATOR = "qemu.sf.lcd_density";
+
+ private static final float DENSITY_DEFAULT = 1f;
+ private static final float DENSITY_LARGE = 1.09f;
+ private static final float DENSITY_LARGER = 1.19f;
+ private static final float DENSITY_LARGEST = 1.29f;
+
+ void setDefaultDensity() {
+ final int stableDensity = getStableDensity();
+ final int targetDensity = (int) (stableDensity * DENSITY_DEFAULT);
+ setDensity(targetDensity);
+ }
+
+ void setLargeDensity() {
+ final int stableDensity = getStableDensity();
+ final int targetDensity = (int) (stableDensity * DENSITY_LARGE);
+ setDensity(targetDensity);
+ }
+
+ void setLargerDensity() {
+ final int stableDensity = getStableDensity();
+ final int targetDensity = (int) (stableDensity * DENSITY_LARGER);
+ setDensity(targetDensity);
+ }
+
+ void setLargestDensity() {
+ final int stableDensity = getStableDensity();
+ final int targetDensity = (int) (stableDensity * DENSITY_LARGEST);
+ setDensity(targetDensity);
+ }
+
+ @Override
+ public void close() {
+ resetDensity();
+ }
+
+ private int getStableDensity() {
+ final String densityProp;
+ if (Build.IS_EMULATOR) {
+ densityProp = DENSITY_PROP_EMULATOR;
+ } else {
+ densityProp = DENSITY_PROP_DEVICE;
+ }
+
+ return Integer.parseInt(executeShellCommand("getprop " + densityProp).trim());
+ }
+
+ private void setDensity(int targetDensity) {
+ executeShellCommand("wm density " + targetDensity);
+
+ // Verify that the density is changed.
+ final String output = executeShellCommand("wm density");
+ final boolean success = output.contains("Override density: " + targetDensity);
+
+ assertTrue("Failed to set density to " + targetDensity, success);
+ }
+
+ private void resetDensity() {
+ executeShellCommand("wm density reset");
+ }
+ }
+
+ public static String executeShellCommand(String cmd) {
+ try {
+ return runShellCommand(InstrumentationRegistry.getInstrumentation(), cmd);
+ } catch (IOException e) {
+ fail("Failed reading command output: " + e);
+ return "";
+ }
+ }
+
+ public static String runShellCommand(Instrumentation instrumentation, String cmd)
+ throws IOException {
+ return runShellCommand(instrumentation.getUiAutomation(), cmd);
+ }
+
+ public static String runShellCommand(UiAutomation automation, String cmd)
+ throws IOException {
+ if (cmd.startsWith("pm grant ") || cmd.startsWith("pm revoke ")) {
+ throw new UnsupportedOperationException("Use UiAutomation.grantRuntimePermission() "
+ + "or revokeRuntimePermission() directly, which are more robust.");
+ }
+ ParcelFileDescriptor pfd = automation.executeShellCommand(cmd);
+ byte[] buf = new byte[512];
+ int bytesRead;
+ FileInputStream fis = new ParcelFileDescriptor.AutoCloseInputStream(pfd);
+ StringBuffer stdout = new StringBuffer();
+ while ((bytesRead = fis.read(buf)) != -1) {
+ stdout.append(new String(buf, 0, bytesRead));
+ }
+ fis.close();
+ return stdout.toString();
+ }
+}
diff --git a/tests/common/com/android/documentsui/testing/TestCursor.java b/tests/common/com/android/documentsui/testing/TestCursor.java
new file mode 100644
index 0000000..56f4269
--- /dev/null
+++ b/tests/common/com/android/documentsui/testing/TestCursor.java
@@ -0,0 +1,30 @@
+/*
+ * Copyright (C) 2019 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.testing;
+
+import android.database.MatrixCursor;
+
+public class TestCursor extends MatrixCursor {
+
+ public TestCursor(String[] columnNames) {
+ super(columnNames);
+ }
+
+ public void mockOnChange() {
+ onChange(false);
+ }
+}
diff --git a/tests/common/com/android/documentsui/testing/TestDocumentsProvider.java b/tests/common/com/android/documentsui/testing/TestDocumentsProvider.java
index 794d1c9..3e4dffd 100644
--- a/tests/common/com/android/documentsui/testing/TestDocumentsProvider.java
+++ b/tests/common/com/android/documentsui/testing/TestDocumentsProvider.java
@@ -117,7 +117,7 @@
}
private Cursor createDocumentsCursor(DocumentInfo... docs) {
- MatrixCursor cursor = new MatrixCursor(DOCUMENTS_PROJECTION);
+ TestCursor cursor = new TestCursor(DOCUMENTS_PROJECTION);
for (DocumentInfo doc : docs) {
cursor.newRow()
.add(Document.COLUMN_DOCUMENT_ID, doc.documentId)
diff --git a/tests/unit/com/android/documentsui/RecentsLoaderTests.java b/tests/unit/com/android/documentsui/RecentsLoaderTests.java
index f044b92..73ba36f 100644
--- a/tests/unit/com/android/documentsui/RecentsLoaderTests.java
+++ b/tests/unit/com/android/documentsui/RecentsLoaderTests.java
@@ -29,6 +29,7 @@
import com.android.documentsui.base.DocumentInfo;
import com.android.documentsui.base.State;
import com.android.documentsui.testing.ActivityManagers;
+import com.android.documentsui.testing.TestCursor;
import com.android.documentsui.testing.TestEnv;
import com.android.documentsui.testing.TestFileTypeLookup;
import com.android.documentsui.testing.TestImmediateExecutor;
@@ -38,6 +39,9 @@
import org.junit.Test;
import org.junit.runner.RunWith;
+import java.util.concurrent.CountDownLatch;
+import java.util.concurrent.TimeUnit;
+
@RunWith(AndroidJUnit4.class)
@MediumTest
public class RecentsLoaderTests {
@@ -45,6 +49,7 @@
private TestEnv mEnv;
private TestActivity mActivity;
private RecentsLoader mLoader;
+ private boolean mContentChanged;
@Before
public void setUp() {
@@ -105,4 +110,28 @@
assertEquals(0, flags & Document.FLAG_SUPPORTS_MOVE);
}
}
+
+ @Test
+ public void testContentsUpdate_observable() throws Exception {
+ CountDownLatch latch = new CountDownLatch(1);
+ Runnable callback = () -> {
+ latch.countDown();
+ mContentChanged = true;
+ };
+ mLoader.setObserver(new LockingContentObserver(new ContentLock(), callback));
+
+ final DocumentInfo doc = mEnv.model.createFile("freddy.jpg");
+ doc.lastModified = System.currentTimeMillis();
+ mEnv.mockProviders.get(TestProvidersAccess.HOME.authority)
+ .setNextRecentDocumentsReturns(doc);
+
+ mLoader.loadInBackground();
+
+ TestCursor c = (TestCursor) mEnv.mockProviders.get(TestProvidersAccess.HOME.authority)
+ .queryRecentDocuments(null, null);
+ c.mockOnChange();
+
+ latch.await(1, TimeUnit.SECONDS);
+ assertTrue(mContentChanged);
+ }
}