blob: 122ef2a760b15971550b6ede4e4bd0a537ccb7d4 [file] [log] [blame]
/*
* Copyright (C) 2020 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 android.server.wm;
import static android.content.Intent.FLAG_ACTIVITY_NEW_TASK;
import static android.server.wm.app.Components.TEST_ACTIVITY;
import static android.view.Display.DEFAULT_DISPLAY;
import static org.junit.Assert.assertTrue;
import static org.junit.Assume.assumeFalse;
import static org.junit.Assume.assumeTrue;
import android.app.Activity;
import android.app.ActivityOptions;
import android.app.Instrumentation;
import android.content.ComponentName;
import android.content.Intent;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.os.SystemClock;
import android.platform.test.annotations.Presubmit;
import android.server.wm.cts.R;
import android.util.Range;
import androidx.test.platform.app.InstrumentationRegistry;
import com.android.compatibility.common.util.SystemUtil;
import org.junit.Test;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicLong;
/**
* <p>Build/Install/Run:
* atest CtsWindowManagerDeviceTestCases:ActivityTransitionTests
*/
@Presubmit
public class ActivityTransitionTests extends ActivityManagerTestBase {
// See WindowManagerService.DISABLE_CUSTOM_TASK_ANIMATION_PROPERTY
static final String DISABLE_CUSTOM_TASK_ANIMATION_PROPERTY =
"persist.wm.disable_custom_task_animation";
static final boolean DISABLE_CUSTOM_TASK_ANIMATION_DEFAULT = true;
// Duration of the default wallpaper close animation
static final long DEFAULT_ANIMATION_DURATION = 275L;
// Duration of the R.anim.alpha animation
static final long CUSTOM_ANIMATION_DURATION = 2000L;
private static boolean customTaskAnimationDisabled() {
try {
return Integer.parseInt(executeShellCommand(
"getprop " + DISABLE_CUSTOM_TASK_ANIMATION_PROPERTY).replace("\n", "")) != 0;
} catch (NumberFormatException e) {
return DISABLE_CUSTOM_TASK_ANIMATION_DEFAULT;
}
}
@Test
public void testActivityTransitionDurationNoShortenAsExpected() throws Exception {
final long expectedDurationMs = CUSTOM_ANIMATION_DURATION - 100L;
final long minDurationMs = expectedDurationMs;
final long maxDurationMs = expectedDurationMs + 300L;
final Range<Long> durationRange = new Range<>(minDurationMs, maxDurationMs);
final CountDownLatch latch = new CountDownLatch(1);
AtomicLong transitionStartTime = new AtomicLong();
AtomicLong transitionEndTime = new AtomicLong();
final ActivityOptions.OnAnimationStartedListener startedListener = () -> {
transitionStartTime.set(SystemClock.elapsedRealtime());
};
final ActivityOptions.OnAnimationFinishedListener finishedListener = () -> {
transitionEndTime.set(SystemClock.elapsedRealtime());
latch.countDown();
};
final Intent intent = new Intent(mContext, LauncherActivity.class)
.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
final Instrumentation instrumentation = InstrumentationRegistry.getInstrumentation();
final LauncherActivity launcherActivity =
(LauncherActivity) instrumentation.startActivitySync(intent);
final Bundle bundle = ActivityOptions.makeCustomAnimation(mContext,
R.anim.alpha, 0, new Handler(Looper.getMainLooper()), startedListener,
finishedListener).toBundle();
launcherActivity.startTransitionActivity(bundle);
mWmState.waitForAppTransitionIdleOnDisplay(DEFAULT_DISPLAY);
waitAndAssertTopResumedActivity(new ComponentName(mContext, TransitionActivity.class),
DEFAULT_DISPLAY, "Activity must be launched");
latch.await(2, TimeUnit.SECONDS);
final long totalTime = transitionEndTime.get() - transitionStartTime.get();
assertTrue("Actual transition duration should be in the range "
+ "<" + minDurationMs + ", " + maxDurationMs + "> ms, "
+ "actual=" + totalTime, durationRange.contains(totalTime));
}
@Test
public void testTaskTransitionDurationNoShortenAsExpected() throws Exception {
assumeFalse(customTaskAnimationDisabled());
final long expectedDurationMs = CUSTOM_ANIMATION_DURATION - 100L;
final long minDurationMs = expectedDurationMs;
final long maxDurationMs = expectedDurationMs + 300L;
final Range<Long> durationRange = new Range<>(minDurationMs, maxDurationMs);
final CountDownLatch latch = new CountDownLatch(1);
AtomicLong transitionStartTime = new AtomicLong();
AtomicLong transitionEndTime = new AtomicLong();
final ActivityOptions.OnAnimationStartedListener startedListener = () -> {
transitionStartTime.set(SystemClock.elapsedRealtime());
};
final ActivityOptions.OnAnimationFinishedListener finishedListener = () -> {
transitionEndTime.set(SystemClock.elapsedRealtime());
latch.countDown();
};
final Bundle bundle = ActivityOptions.makeCustomAnimation(mContext,
R.anim.alpha, 0, new Handler(Looper.getMainLooper()), startedListener,
finishedListener).toBundle();
final Intent intent = new Intent().setComponent(TEST_ACTIVITY)
.addFlags(FLAG_ACTIVITY_NEW_TASK);
mContext.startActivity(intent, bundle);
mWmState.waitForAppTransitionIdleOnDisplay(DEFAULT_DISPLAY);
waitAndAssertTopResumedActivity(TEST_ACTIVITY, DEFAULT_DISPLAY,
"Activity must be launched");
latch.await(2, TimeUnit.SECONDS);
final long totalTime = transitionEndTime.get() - transitionStartTime.get();
assertTrue("Actual transition duration should be in the range "
+ "<" + minDurationMs + ", " + maxDurationMs + "> ms, "
+ "actual=" + totalTime, durationRange.contains(totalTime));
}
@Test
public void testTaskTransitionOverrideDisabled() throws Exception {
assumeTrue(customTaskAnimationDisabled());
final long expectedDurationMs = DEFAULT_ANIMATION_DURATION - 100L;
final long minDurationMs = expectedDurationMs;
final long maxDurationMs = expectedDurationMs + 1000L;
final Range<Long> durationRange = new Range<>(minDurationMs, maxDurationMs);
final CountDownLatch latch = new CountDownLatch(1);
AtomicLong transitionStartTime = new AtomicLong();
AtomicLong transitionEndTime = new AtomicLong();
final ActivityOptions.OnAnimationStartedListener startedListener = () -> {
transitionStartTime.set(SystemClock.elapsedRealtime());
};
final ActivityOptions.OnAnimationFinishedListener finishedListener = () -> {
transitionEndTime.set(SystemClock.elapsedRealtime());
latch.countDown();
};
// Overriding task transit animation is disabled, so default wallpaper close animation
// is played.
final Bundle bundle = ActivityOptions.makeCustomAnimation(mContext,
R.anim.alpha, 0, new Handler(Looper.getMainLooper()), startedListener,
finishedListener).toBundle();
final Intent intent = new Intent().setComponent(TEST_ACTIVITY)
.addFlags(FLAG_ACTIVITY_NEW_TASK);
mContext.startActivity(intent, bundle);
mWmState.waitForAppTransitionIdleOnDisplay(DEFAULT_DISPLAY);
waitAndAssertTopResumedActivity(TEST_ACTIVITY, DEFAULT_DISPLAY,
"Activity must be launched");
latch.await(2, TimeUnit.SECONDS);
final long totalTime = transitionEndTime.get() - transitionStartTime.get();
assertTrue("Actual transition duration should be in the range "
+ "<" + minDurationMs + ", " + maxDurationMs + "> ms, "
+ "actual=" + totalTime, durationRange.contains(totalTime));
}
@Test
public void testTaskTransitionOverride() throws Exception {
assumeTrue(customTaskAnimationDisabled());
final long expectedDurationMs = CUSTOM_ANIMATION_DURATION - 100L;
final long minDurationMs = expectedDurationMs;
final long maxDurationMs = expectedDurationMs + 1000L;
final Range<Long> durationRange = new Range<>(minDurationMs, maxDurationMs);
final CountDownLatch latch = new CountDownLatch(1);
AtomicLong transitionStartTime = new AtomicLong();
AtomicLong transitionEndTime = new AtomicLong();
final ActivityOptions.OnAnimationStartedListener startedListener = () -> {
transitionStartTime.set(SystemClock.elapsedRealtime());
};
final ActivityOptions.OnAnimationFinishedListener finishedListener = () -> {
transitionEndTime.set(SystemClock.elapsedRealtime());
latch.countDown();
};
SystemUtil.runWithShellPermissionIdentity(() -> {
// Overriding task transit animation is enabled, so custom animation is played.
final Bundle bundle = ActivityOptions.makeCustomTaskAnimation(mContext,
R.anim.alpha, 0, new Handler(Looper.getMainLooper()), startedListener,
finishedListener).toBundle();
final Intent intent = new Intent().setComponent(TEST_ACTIVITY)
.addFlags(FLAG_ACTIVITY_NEW_TASK);
mContext.startActivity(intent, bundle);
mWmState.waitForAppTransitionIdleOnDisplay(DEFAULT_DISPLAY);
waitAndAssertTopResumedActivity(TEST_ACTIVITY, DEFAULT_DISPLAY,
"Activity must be launched");
latch.await(2, TimeUnit.SECONDS);
final long totalTime = transitionEndTime.get() - transitionStartTime.get();
assertTrue("Actual transition duration should be in the range "
+ "<" + minDurationMs + ", " + maxDurationMs + "> ms, "
+ "actual=" + totalTime, durationRange.contains(totalTime));
});
}
public static class LauncherActivity extends Activity {
public void startTransitionActivity(Bundle bundle) {
startActivity(new Intent(this, TransitionActivity.class), bundle);
}
}
public static class TransitionActivity extends Activity {
}
}