Home | History | Annotate | Download | only in cts
      1 /*
      2  * Copyright (C) 2016 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 package android.fragment.cts;
     17 
     18 import static org.junit.Assert.assertEquals;
     19 import static org.junit.Assert.assertTrue;
     20 
     21 import android.app.Activity;
     22 import android.app.Fragment;
     23 import android.app.FragmentController;
     24 import android.app.FragmentManager;
     25 import android.app.FragmentManagerNonConfig;
     26 import android.os.Looper;
     27 import android.os.Parcelable;
     28 import android.util.Pair;
     29 import android.view.View;
     30 import android.view.ViewGroup;
     31 import android.view.accessibility.AccessibilityNodeInfo;
     32 
     33 import androidx.test.rule.ActivityTestRule;
     34 
     35 import java.util.concurrent.CountDownLatch;
     36 import java.util.concurrent.TimeUnit;
     37 
     38 public class FragmentTestUtil {
     39     public static void waitForExecution(final ActivityTestRule<? extends Activity> rule) {
     40         // Wait for two cycles. When starting a postponed transition, it will post to
     41         // the UI thread and then the execution will be added onto the queue after that.
     42         // The two-cycle wait makes sure fragments have the opportunity to complete both
     43         // before returning.
     44         try {
     45             rule.runOnUiThread(() -> {
     46             });
     47             rule.runOnUiThread(() -> {
     48             });
     49         } catch (Throwable t) {
     50             throw new RuntimeException(t);
     51         }
     52     }
     53 
     54     private static void runOnUiThreadRethrow(ActivityTestRule<? extends Activity> rule,
     55             Runnable r) {
     56         if (Looper.getMainLooper() == Looper.myLooper()) {
     57             r.run();
     58         } else {
     59             try {
     60                 rule.runOnUiThread(r);
     61             } catch (Throwable t) {
     62                 throw new RuntimeException(t);
     63             }
     64         }
     65     }
     66 
     67     public static boolean executePendingTransactions(
     68             final ActivityTestRule<? extends Activity> rule) {
     69         return executePendingTransactions(rule, rule.getActivity().getFragmentManager());
     70     }
     71 
     72     public static boolean executePendingTransactions(
     73             final ActivityTestRule<? extends Activity> rule, final FragmentManager fm) {
     74         final boolean[] ret = new boolean[1];
     75         runOnUiThreadRethrow(rule, new Runnable() {
     76             @Override
     77             public void run() {
     78                 ret[0] = fm.executePendingTransactions();
     79             }
     80         });
     81         return ret[0];
     82     }
     83 
     84     public static boolean popBackStackImmediate(final ActivityTestRule<? extends Activity> rule) {
     85         return popBackStackImmediate(rule, rule.getActivity().getFragmentManager());
     86     }
     87 
     88     public static boolean popBackStackImmediate(final ActivityTestRule<? extends Activity> rule,
     89             final FragmentManager fm) {
     90         final boolean[] ret = new boolean[1];
     91         runOnUiThreadRethrow(rule, new Runnable() {
     92             @Override
     93             public void run() {
     94                 ret[0] = fm.popBackStackImmediate();
     95             }
     96         });
     97         return ret[0];
     98     }
     99 
    100     public static boolean popBackStackImmediate(final ActivityTestRule<FragmentTestActivity> rule,
    101             final int id, final int flags) {
    102         return popBackStackImmediate(rule, rule.getActivity().getFragmentManager(), id, flags);
    103     }
    104 
    105     public static boolean popBackStackImmediate(final ActivityTestRule<FragmentTestActivity> rule,
    106             final FragmentManager fm, final int id, final int flags) {
    107         final boolean[] ret = new boolean[1];
    108         runOnUiThreadRethrow(rule, new Runnable() {
    109             @Override
    110             public void run() {
    111                 ret[0] = fm.popBackStackImmediate(id, flags);
    112             }
    113         });
    114         return ret[0];
    115     }
    116 
    117     public static boolean popBackStackImmediate(final ActivityTestRule<FragmentTestActivity> rule,
    118             final String name, final int flags) {
    119         return popBackStackImmediate(rule, rule.getActivity().getFragmentManager(), name, flags);
    120     }
    121 
    122     public static boolean popBackStackImmediate(final ActivityTestRule<FragmentTestActivity> rule,
    123             final FragmentManager fm, final String name, final int flags) {
    124         final boolean[] ret = new boolean[1];
    125         runOnUiThreadRethrow(rule, new Runnable() {
    126             @Override
    127             public void run() {
    128                 ret[0] = fm.popBackStackImmediate(name, flags);
    129             }
    130         });
    131         return ret[0];
    132     }
    133 
    134     public static void setContentView(final ActivityTestRule<FragmentTestActivity> rule,
    135             final int layoutId) {
    136         final Activity activity = rule.getActivity();
    137         runOnUiThreadRethrow(rule, new Runnable() {
    138             @Override
    139             public void run() {
    140                 activity.setContentView(layoutId);
    141             }
    142         });
    143     }
    144 
    145     public static void assertChildren(ViewGroup container, Fragment... fragments) {
    146         final int numFragments = fragments == null ? 0 : fragments.length;
    147         assertEquals("There aren't the correct number of fragment Views in its container",
    148                 numFragments, container.getChildCount());
    149         for (int i = 0; i < numFragments; i++) {
    150             assertEquals("Wrong Fragment View order for [" + i + "]", container.getChildAt(i),
    151                     fragments[i].getView());
    152         }
    153     }
    154 
    155     public static FragmentController createController(ActivityTestRule<FragmentTestActivity> rule) {
    156         final FragmentController[] controller = new FragmentController[1];
    157         final FragmentTestActivity activity = rule.getActivity();
    158         runOnUiThreadRethrow(rule, () -> {
    159             HostCallbacks hostCallbacks = new HostCallbacks(activity, null, 0);
    160             controller[0] = FragmentController.createController(hostCallbacks);
    161         });
    162         return controller[0];
    163     }
    164 
    165 
    166     public static void resume(ActivityTestRule<FragmentTestActivity> rule,
    167             FragmentController fragmentController,
    168             Pair<Parcelable, FragmentManagerNonConfig> savedState) {
    169         runOnUiThreadRethrow(rule, () -> {
    170             fragmentController.attachHost(null);
    171             if (savedState != null) {
    172                 fragmentController.restoreAllState(savedState.first, savedState.second);
    173             }
    174             fragmentController.dispatchCreate();
    175             fragmentController.dispatchActivityCreated();
    176             fragmentController.noteStateNotSaved();
    177             fragmentController.execPendingActions();
    178             fragmentController.dispatchStart();
    179             fragmentController.reportLoaderStart();
    180             fragmentController.dispatchResume();
    181             fragmentController.execPendingActions();
    182         });
    183     }
    184 
    185     public static Pair<Parcelable, FragmentManagerNonConfig> destroy(
    186             ActivityTestRule<FragmentTestActivity> rule, FragmentController fragmentController) {
    187         final Pair<Parcelable, FragmentManagerNonConfig>[] result = new Pair[1];
    188         runOnUiThreadRethrow(rule, () -> {
    189             fragmentController.dispatchPause();
    190             final Parcelable savedState = fragmentController.saveAllState();
    191             final FragmentManagerNonConfig nonConfig = fragmentController.retainNestedNonConfig();
    192             fragmentController.dispatchStop();
    193             fragmentController.doLoaderStop(false);
    194             fragmentController.dispatchDestroy();
    195             fragmentController.doLoaderDestroy();
    196             result[0] = Pair.create(savedState, nonConfig);
    197         });
    198         return result[0];
    199     }
    200 
    201     public static boolean isVisible(Fragment fragment) {
    202         View view = fragment.getView();
    203         AccessibilityNodeInfo accessibilityNodeInfo = view.createAccessibilityNodeInfo();
    204         boolean isVisible = accessibilityNodeInfo.isVisibleToUser();
    205         accessibilityNodeInfo.recycle();
    206         return isVisible;
    207     }
    208 
    209     /**
    210      * Allocates until a garbage collection occurs.
    211      */
    212     public static void forceGC() {
    213         // This works on ART:
    214         Runtime.getRuntime().gc();
    215         Runtime.getRuntime().runFinalization();
    216         Runtime.getRuntime().gc();
    217         Runtime.getRuntime().runFinalization();
    218     }
    219 
    220     /**
    221      * Restarts the RecreatedActivity and waits for the new activity to be resumed.
    222      *
    223      * @return The newly-restarted Activity
    224      */
    225     public static <T extends RecreatedActivity> T recreateActivity(
    226             ActivityTestRule<? extends Activity> rule, T activity) throws InterruptedException {
    227         // Now switch the orientation
    228         RecreatedActivity.sResumed = new CountDownLatch(1);
    229         RecreatedActivity.sDestroyed = new CountDownLatch(1);
    230 
    231         runOnUiThreadRethrow(rule, () -> {
    232             activity.recreate();
    233         });
    234         assertTrue(RecreatedActivity.sResumed.await(1, TimeUnit.SECONDS));
    235         assertTrue(RecreatedActivity.sDestroyed.await(1, TimeUnit.SECONDS));
    236         T newActivity = (T) RecreatedActivity.sActivity;
    237 
    238         waitForExecution(rule);
    239 
    240         RecreatedActivity.clearState();
    241         return newActivity;
    242     }
    243 }
    244 
    245 
    246