blob: c95849e4a6748c4420b166d167628226459f0bfa [file] [log] [blame]
Lucas Dupin9e3fa102017-11-08 17:16:55 -08001/*
2 * Copyright (C) 2017 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
17package com.android.systemui.statusbar.phone;
18
Lucas Dupin82aa1632017-12-13 00:13:57 -080019import static com.android.systemui.statusbar.phone.ScrimController.VISIBILITY_FULLY_OPAQUE;
20import static com.android.systemui.statusbar.phone.ScrimController.VISIBILITY_FULLY_TRANSPARENT;
21import static com.android.systemui.statusbar.phone.ScrimController.VISIBILITY_SEMI_TRANSPARENT;
22
23import static org.mockito.ArgumentMatchers.any;
24import static org.mockito.ArgumentMatchers.anyInt;
25import static org.mockito.ArgumentMatchers.anyLong;
Lucas Dupin9e3fa102017-11-08 17:16:55 -080026import static org.mockito.Mockito.mock;
27import static org.mockito.Mockito.never;
Lucas Dupin82aa1632017-12-13 00:13:57 -080028import static org.mockito.Mockito.reset;
Lucas Dupin067136c2018-03-27 18:03:25 -070029import static org.mockito.Mockito.spy;
Lucas Dupin9e3fa102017-11-08 17:16:55 -080030import static org.mockito.Mockito.verify;
Lucas Dupineea53b32017-12-18 13:47:14 -080031import static org.mockito.Mockito.verifyZeroInteractions;
Lucas Dupin9e3fa102017-11-08 17:16:55 -080032import static org.mockito.Mockito.when;
33
34import android.animation.Animator;
Lucas Dupin80a3fcc2018-02-07 10:49:55 -080035import android.animation.ValueAnimator;
Lucas Dupin82aa1632017-12-13 00:13:57 -080036import android.app.AlarmManager;
Lucas Dupin9e3fa102017-11-08 17:16:55 -080037import android.graphics.Color;
38import android.os.Handler;
39import android.os.Looper;
40import android.support.test.filters.SmallTest;
41import android.testing.AndroidTestingRunner;
42import android.testing.TestableLooper;
43import android.view.View;
44
shawnlin317db372018-04-09 19:49:48 +080045import com.android.internal.colorextraction.ColorExtractor.GradientColors;
Lucas Dupin7517b5d2017-08-22 12:51:25 -070046import com.android.keyguard.KeyguardUpdateMonitor;
Lucas Dupin9e3fa102017-11-08 17:16:55 -080047import com.android.systemui.SysuiTestCase;
48import com.android.systemui.statusbar.ScrimView;
49import com.android.systemui.util.wakelock.WakeLock;
50import com.android.systemui.utils.os.FakeHandler;
51
52import org.junit.Assert;
53import org.junit.Before;
54import org.junit.Test;
55import org.junit.runner.RunWith;
56
Lucas Dupin78949b82018-04-03 18:54:39 -070057import java.util.Arrays;
58import java.util.HashSet;
Lucas Dupin9e3fa102017-11-08 17:16:55 -080059import java.util.function.Consumer;
60
61@RunWith(AndroidTestingRunner.class)
62@TestableLooper.RunWithLooper
63@SmallTest
64public class ScrimControllerTest extends SysuiTestCase {
65
66 private SynchronousScrimController mScrimController;
67 private ScrimView mScrimBehind;
68 private ScrimView mScrimInFront;
shawnlin317db372018-04-09 19:49:48 +080069 private Consumer<Float> mScrimBehindAlphaCallback;
70 private Consumer<GradientColors> mScrimInFrontColorCallback;
Lucas Dupin82aa1632017-12-13 00:13:57 -080071 private Consumer<Integer> mScrimVisibilityCallback;
shawnlin317db372018-04-09 19:49:48 +080072 private float mScrimBehindAlpha;
73 private GradientColors mScrimInFrontColor;
Lucas Dupin82aa1632017-12-13 00:13:57 -080074 private int mScrimVisibility;
Lucas Dupin9e3fa102017-11-08 17:16:55 -080075 private DozeParameters mDozeParamenters;
76 private WakeLock mWakeLock;
77 private boolean mAlwaysOnEnabled;
Lucas Dupin82aa1632017-12-13 00:13:57 -080078 private AlarmManager mAlarmManager;
Lucas Dupin9e3fa102017-11-08 17:16:55 -080079
80 @Before
81 public void setup() {
Lucas Dupin067136c2018-03-27 18:03:25 -070082 mScrimBehind = spy(new ScrimView(getContext()));
Lucas Dupin9e3fa102017-11-08 17:16:55 -080083 mScrimInFront = new ScrimView(getContext());
Lucas Dupin9e3fa102017-11-08 17:16:55 -080084 mWakeLock = mock(WakeLock.class);
Lucas Dupin82aa1632017-12-13 00:13:57 -080085 mAlarmManager = mock(AlarmManager.class);
Lucas Dupin9e3fa102017-11-08 17:16:55 -080086 mAlwaysOnEnabled = true;
shawnlin317db372018-04-09 19:49:48 +080087 mScrimBehindAlphaCallback = (Float alpha) -> mScrimBehindAlpha = alpha;
88 mScrimInFrontColorCallback = (GradientColors color) -> mScrimInFrontColor = color;
Lucas Dupin82aa1632017-12-13 00:13:57 -080089 mScrimVisibilityCallback = (Integer visible) -> mScrimVisibility = visible;
Lucas Dupin9e3fa102017-11-08 17:16:55 -080090 mDozeParamenters = mock(DozeParameters.class);
91 when(mDozeParamenters.getAlwaysOn()).thenAnswer(invocation -> mAlwaysOnEnabled);
Lucas Dupin43d0d732017-11-16 11:23:49 -080092 when(mDozeParamenters.getDisplayNeedsBlanking()).thenReturn(true);
shawnlin317db372018-04-09 19:49:48 +080093 mScrimController = new SynchronousScrimController(mScrimBehind, mScrimInFront,
94 mScrimBehindAlphaCallback, mScrimInFrontColorCallback, mScrimVisibilityCallback,
95 mDozeParamenters, mAlarmManager);
Lucas Dupin9e3fa102017-11-08 17:16:55 -080096 }
97
98 @Test
99 public void initialState() {
100 Assert.assertEquals("ScrimController should start initialized",
101 mScrimController.getState(), ScrimState.UNINITIALIZED);
102 }
103
104 @Test
105 public void transitionToKeyguard() {
106 mScrimController.transitionTo(ScrimState.KEYGUARD);
107 mScrimController.finishAnimationsImmediately();
108 // Front scrim should be transparent
109 // Back scrim should be visible without tint
Lucas Dupin82aa1632017-12-13 00:13:57 -0800110 assertScrimVisibility(VISIBILITY_FULLY_TRANSPARENT, VISIBILITY_SEMI_TRANSPARENT);
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800111 assertScrimTint(mScrimBehind, false /* tinted */);
112 }
113
114 @Test
Lucas Dupin7517b5d2017-08-22 12:51:25 -0700115 public void transitionToAod_withRegularWallpaper() {
116 mScrimController.transitionTo(ScrimState.AOD);
117 mScrimController.finishAnimationsImmediately();
118 // Front scrim should be transparent
119 // Back scrim should be visible with tint
Lucas Dupin82aa1632017-12-13 00:13:57 -0800120 assertScrimVisibility(VISIBILITY_FULLY_TRANSPARENT, VISIBILITY_FULLY_OPAQUE);
Lucas Dupin7517b5d2017-08-22 12:51:25 -0700121 assertScrimTint(mScrimBehind, true /* tinted */);
122 assertScrimTint(mScrimInFront, true /* tinted */);
123 }
124
125 @Test
126 public void transitionToAod_withAodWallpaper() {
127 mScrimController.setWallpaperSupportsAmbientMode(true);
128 mScrimController.transitionTo(ScrimState.AOD);
129 mScrimController.finishAnimationsImmediately();
130 // Front scrim should be transparent
131 // Back scrim should be transparent
Lucas Dupin82aa1632017-12-13 00:13:57 -0800132 assertScrimVisibility(VISIBILITY_FULLY_TRANSPARENT, VISIBILITY_FULLY_TRANSPARENT);
133
134 // Move on to PULSING and check if the back scrim is still transparent
135 mScrimController.transitionTo(ScrimState.PULSING);
136 mScrimController.finishAnimationsImmediately();
137 assertScrimVisibility(VISIBILITY_FULLY_TRANSPARENT, VISIBILITY_FULLY_TRANSPARENT);
Lucas Dupin7517b5d2017-08-22 12:51:25 -0700138 }
139
140 @Test
141 public void transitionToAod_withAodWallpaperAndLockScreenWallpaper() {
142 ScrimState.AOD.mKeyguardUpdateMonitor = new KeyguardUpdateMonitor(getContext()) {
143 @Override
144 public boolean hasLockscreenWallpaper() {
145 return true;
146 }
147 };
148 mScrimController.setWallpaperSupportsAmbientMode(true);
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800149 mScrimController.transitionTo(ScrimState.AOD);
150 mScrimController.finishAnimationsImmediately();
151 // Front scrim should be transparent
152 // Back scrim should be visible with tint
Lucas Dupin82aa1632017-12-13 00:13:57 -0800153 assertScrimVisibility(VISIBILITY_FULLY_TRANSPARENT, VISIBILITY_FULLY_OPAQUE);
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800154 assertScrimTint(mScrimBehind, true /* tinted */);
155 assertScrimTint(mScrimInFront, true /* tinted */);
156 }
157
158 @Test
159 public void transitionToPulsing() {
Lucas Dupin82aa1632017-12-13 00:13:57 -0800160 // Pre-condition
161 // Need to go to AoD first because PULSING doesn't change
162 // the back scrim opacity - otherwise it would hide AoD wallpapers.
163 mScrimController.setWallpaperSupportsAmbientMode(false);
164 mScrimController.transitionTo(ScrimState.AOD);
165 mScrimController.finishAnimationsImmediately();
166 assertScrimVisibility(VISIBILITY_FULLY_TRANSPARENT, VISIBILITY_FULLY_OPAQUE);
167
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800168 mScrimController.transitionTo(ScrimState.PULSING);
169 mScrimController.finishAnimationsImmediately();
170 // Front scrim should be transparent
171 // Back scrim should be visible with tint
172 // Pulse callback should have been invoked
Lucas Dupin82aa1632017-12-13 00:13:57 -0800173 assertScrimVisibility(VISIBILITY_FULLY_TRANSPARENT, VISIBILITY_FULLY_OPAQUE);
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800174 assertScrimTint(mScrimBehind, true /* tinted */);
175 }
176
177 @Test
Lucas Dupinbc9aac12018-03-04 20:18:15 -0800178 public void transitionToKeyguardBouncer() {
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800179 mScrimController.transitionTo(ScrimState.BOUNCER);
180 mScrimController.finishAnimationsImmediately();
181 // Front scrim should be transparent
182 // Back scrim should be visible without tint
Lucas Dupinbc9aac12018-03-04 20:18:15 -0800183 assertScrimVisibility(VISIBILITY_FULLY_TRANSPARENT, VISIBILITY_SEMI_TRANSPARENT);
184 assertScrimTint(mScrimBehind, false /* tinted */);
185 }
186
187 @Test
188 public void transitionToBouncer() {
Lucas Dupin05726cd2018-03-13 14:00:24 -0700189 mScrimController.transitionTo(ScrimState.BOUNCER_SCRIMMED);
Lucas Dupinbc9aac12018-03-04 20:18:15 -0800190 mScrimController.finishAnimationsImmediately();
191 // Front scrim should be transparent
192 // Back scrim should be visible without tint
193 assertScrimVisibility(VISIBILITY_SEMI_TRANSPARENT, VISIBILITY_FULLY_TRANSPARENT);
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800194 assertScrimTint(mScrimBehind, false /* tinted */);
195 }
196
197 @Test
198 public void transitionToUnlocked() {
Lucas Dupin80a3fcc2018-02-07 10:49:55 -0800199 mScrimController.setPanelExpansion(0f);
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800200 mScrimController.transitionTo(ScrimState.UNLOCKED);
201 mScrimController.finishAnimationsImmediately();
202 // Front scrim should be transparent
203 // Back scrim should be transparent
Lucas Dupin82aa1632017-12-13 00:13:57 -0800204 assertScrimVisibility(VISIBILITY_FULLY_TRANSPARENT, VISIBILITY_FULLY_TRANSPARENT);
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800205 assertScrimTint(mScrimBehind, false /* tinted */);
206 assertScrimTint(mScrimInFront, false /* tinted */);
207
208 // Back scrim should be visible after start dragging
209 mScrimController.setPanelExpansion(0.5f);
Lucas Dupin82aa1632017-12-13 00:13:57 -0800210 assertScrimVisibility(VISIBILITY_FULLY_TRANSPARENT, VISIBILITY_SEMI_TRANSPARENT);
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800211 }
212
213 @Test
shawnlin317db372018-04-09 19:49:48 +0800214 public void panelExpansion() {
215 mScrimController.setPanelExpansion(0f);
216 mScrimController.setPanelExpansion(0.5f);
217 mScrimController.transitionTo(ScrimState.UNLOCKED);
218 mScrimController.finishAnimationsImmediately();
219
220 reset(mScrimBehind);
221 mScrimController.setPanelExpansion(0f);
222 mScrimController.setPanelExpansion(1.0f);
223 mScrimController.onPreDraw();
224
225 Assert.assertEquals("Scrim alpha should change after setPanelExpansion",
226 mScrimBehindAlpha, mScrimBehind.getViewAlpha(), 0.01f);
227
228 mScrimController.setPanelExpansion(0f);
229 mScrimController.onPreDraw();
230
231 Assert.assertEquals("Scrim alpha should change after setPanelExpansion",
232 mScrimBehindAlpha, mScrimBehind.getViewAlpha(), 0.01f);
233 }
234
235 @Test
Lucas Dupin67f02632018-03-12 11:08:31 -0700236 public void panelExpansionAffectsAlpha() {
237 mScrimController.setPanelExpansion(0f);
238 mScrimController.setPanelExpansion(0.5f);
239 mScrimController.transitionTo(ScrimState.UNLOCKED);
240 mScrimController.finishAnimationsImmediately();
241
242 final float scrimAlpha = mScrimBehind.getViewAlpha();
Lucas Dupin067136c2018-03-27 18:03:25 -0700243 reset(mScrimBehind);
Lucas Dupin67f02632018-03-12 11:08:31 -0700244 mScrimController.setExpansionAffectsAlpha(false);
245 mScrimController.setPanelExpansion(0.8f);
Lucas Dupin067136c2018-03-27 18:03:25 -0700246 verifyZeroInteractions(mScrimBehind);
Lucas Dupin67f02632018-03-12 11:08:31 -0700247 Assert.assertEquals("Scrim opacity shouldn't change when setExpansionAffectsAlpha "
248 + "is false", scrimAlpha, mScrimBehind.getViewAlpha(), 0.01f);
249
250 mScrimController.setExpansionAffectsAlpha(true);
251 mScrimController.setPanelExpansion(0.1f);
252 Assert.assertNotEquals("Scrim opacity should change when setExpansionAffectsAlpha "
253 + "is true", scrimAlpha, mScrimBehind.getViewAlpha(), 0.01f);
254 }
255
256 @Test
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800257 public void transitionToUnlockedFromAod() {
258 // Simulate unlock with fingerprint
259 mScrimController.transitionTo(ScrimState.AOD);
Lucas Dupin80a3fcc2018-02-07 10:49:55 -0800260 mScrimController.setPanelExpansion(0f);
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800261 mScrimController.finishAnimationsImmediately();
262 mScrimController.transitionTo(ScrimState.UNLOCKED);
263 // Immediately tinted after the transition starts
264 assertScrimTint(mScrimInFront, true /* tinted */);
265 assertScrimTint(mScrimBehind, true /* tinted */);
266 mScrimController.finishAnimationsImmediately();
267 // Front scrim should be transparent
268 // Back scrim should be transparent
269 // Neither scrims should be tinted anymore after the animation.
Lucas Dupin82aa1632017-12-13 00:13:57 -0800270 assertScrimVisibility(VISIBILITY_FULLY_TRANSPARENT, VISIBILITY_FULLY_TRANSPARENT);
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800271 assertScrimTint(mScrimInFront, false /* tinted */);
272 assertScrimTint(mScrimBehind, false /* tinted */);
273 }
274
275 @Test
Lucas Dupinea0116e2018-04-05 10:09:29 -0700276 public void scrimBlanksBeforeLeavingAod() {
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800277 // Simulate unlock with fingerprint
278 mScrimController.transitionTo(ScrimState.AOD);
279 mScrimController.finishAnimationsImmediately();
280 mScrimController.transitionTo(ScrimState.UNLOCKED,
281 new ScrimController.Callback() {
282 @Override
283 public void onDisplayBlanked() {
284 // Front scrim should be black in the middle of the transition
285 Assert.assertTrue("Scrim should be visible during transition. Alpha: "
286 + mScrimInFront.getViewAlpha(), mScrimInFront.getViewAlpha() > 0);
287 assertScrimTint(mScrimInFront, true /* tinted */);
Lucas Dupin82aa1632017-12-13 00:13:57 -0800288 Assert.assertSame("Scrim should be visible during transition.",
289 mScrimVisibility, VISIBILITY_FULLY_OPAQUE);
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800290 }
291 });
292 mScrimController.finishAnimationsImmediately();
293 }
294
295 @Test
296 public void testScrimCallback() {
297 int[] callOrder = {0, 0, 0};
298 int[] currentCall = {0};
299 mScrimController.transitionTo(ScrimState.AOD, new ScrimController.Callback() {
300 @Override
301 public void onStart() {
302 callOrder[0] = ++currentCall[0];
303 }
304
305 @Override
306 public void onDisplayBlanked() {
307 callOrder[1] = ++currentCall[0];
308 }
309
310 @Override
311 public void onFinished() {
312 callOrder[2] = ++currentCall[0];
313 }
314 });
315 mScrimController.finishAnimationsImmediately();
316 Assert.assertEquals("onStart called in wrong order", 1, callOrder[0]);
317 Assert.assertEquals("onDisplayBlanked called in wrong order", 2, callOrder[1]);
318 Assert.assertEquals("onFinished called in wrong order", 3, callOrder[2]);
319 }
320
321 @Test
322 public void testScrimCallbacksWithoutAmbientDisplay() {
323 mAlwaysOnEnabled = false;
324 testScrimCallback();
325 }
326
327 @Test
328 public void testScrimCallbackCancelled() {
329 boolean[] cancelledCalled = {false};
330 mScrimController.transitionTo(ScrimState.AOD, new ScrimController.Callback() {
331 @Override
332 public void onCancelled() {
333 cancelledCalled[0] = true;
334 }
335 });
336 mScrimController.transitionTo(ScrimState.PULSING);
337 Assert.assertTrue("onCancelled should have been called", cancelledCalled[0]);
338 }
339
340 @Test
Lucas Dupineea53b32017-12-18 13:47:14 -0800341 public void testHoldsWakeLock_whenAOD() {
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800342 mScrimController.transitionTo(ScrimState.AOD);
Lucas Dupin82aa1632017-12-13 00:13:57 -0800343 verify(mWakeLock).acquire();
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800344 verify(mWakeLock, never()).release();
345 mScrimController.finishAnimationsImmediately();
Lucas Dupin82aa1632017-12-13 00:13:57 -0800346 verify(mWakeLock).release();
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800347 }
348
Lucas Dupin19aba8e2017-12-11 12:42:26 -0800349 @Test
Lucas Dupineea53b32017-12-18 13:47:14 -0800350 public void testDoesNotHoldWakeLock_whenUnlocking() {
351 mScrimController.transitionTo(ScrimState.UNLOCKED);
352 mScrimController.finishAnimationsImmediately();
353 verifyZeroInteractions(mWakeLock);
354 }
355
356 @Test
Lucas Dupin19aba8e2017-12-11 12:42:26 -0800357 public void testCallbackInvokedOnSameStateTransition() {
358 mScrimController.transitionTo(ScrimState.UNLOCKED);
359 mScrimController.finishAnimationsImmediately();
360 ScrimController.Callback callback = mock(ScrimController.Callback.class);
361 mScrimController.transitionTo(ScrimState.UNLOCKED, callback);
Lucas Dupin82aa1632017-12-13 00:13:57 -0800362 verify(callback).onFinished();
363 }
364
365 @Test
366 public void testHoldsAodWallpaperAnimationLock() {
367 // Pre-conditions
368 mScrimController.transitionTo(ScrimState.AOD);
369 mScrimController.finishAnimationsImmediately();
370 reset(mWakeLock);
371
372 mScrimController.onHideWallpaperTimeout();
373 verify(mWakeLock).acquire();
374 verify(mWakeLock, never()).release();
375 mScrimController.finishAnimationsImmediately();
376 verify(mWakeLock).release();
377 }
378
379 @Test
Lucas Dupin16cfe452018-02-08 13:14:50 -0800380 public void testWillHideAodWallpaper() {
Lucas Dupin82aa1632017-12-13 00:13:57 -0800381 mScrimController.setWallpaperSupportsAmbientMode(true);
382 mScrimController.transitionTo(ScrimState.AOD);
383 verify(mAlarmManager).setExact(anyInt(), anyLong(), any(), any(), any());
384 mScrimController.transitionTo(ScrimState.KEYGUARD);
385 verify(mAlarmManager).cancel(any(AlarmManager.OnAlarmListener.class));
Lucas Dupin19aba8e2017-12-11 12:42:26 -0800386 }
387
Lucas Dupin80a3fcc2018-02-07 10:49:55 -0800388 @Test
389 public void testConservesExpansionOpacityAfterTransition() {
390 mScrimController.transitionTo(ScrimState.UNLOCKED);
391 mScrimController.setPanelExpansion(0.5f);
392 mScrimController.finishAnimationsImmediately();
393
394 final float expandedAlpha = mScrimBehind.getViewAlpha();
395
396 mScrimController.transitionTo(ScrimState.BRIGHTNESS_MIRROR);
397 mScrimController.finishAnimationsImmediately();
398 mScrimController.transitionTo(ScrimState.UNLOCKED);
399 mScrimController.finishAnimationsImmediately();
400
401 Assert.assertEquals("Scrim expansion opacity wasn't conserved when transitioning back",
402 expandedAlpha, mScrimBehind.getViewAlpha(), 0.01f);
403 }
404
405 @Test
406 public void cancelsOldAnimationBeforeBlanking() {
407 mScrimController.transitionTo(ScrimState.AOD);
408 mScrimController.finishAnimationsImmediately();
409 // Consume whatever value we had before
410 mScrimController.wasAnimationJustCancelled();
411
412 mScrimController.transitionTo(ScrimState.KEYGUARD);
413 mScrimController.finishAnimationsImmediately();
414 Assert.assertTrue(mScrimController.wasAnimationJustCancelled());
415 }
416
Lucas Dupinb380c882018-02-25 21:57:17 -0800417 /**
418 * Number of visible notifications affects scrim opacity.
419 */
420 @Test
421 public void testNotificationDensity() {
422 mScrimController.transitionTo(ScrimState.KEYGUARD);
423 mScrimController.finishAnimationsImmediately();
424
425 mScrimController.setNotificationCount(0);
426 mScrimController.finishAnimationsImmediately();
427 Assert.assertEquals("lower density when no notifications",
428 ScrimController.GRADIENT_SCRIM_ALPHA, mScrimBehind.getViewAlpha(), 0.01f);
429
430 mScrimController.setNotificationCount(3);
431 mScrimController.finishAnimationsImmediately();
432 Assert.assertEquals("stronger density when notifications are visible",
433 ScrimController.GRADIENT_SCRIM_ALPHA_BUSY, mScrimBehind.getViewAlpha(), 0.01f);
434 }
435
436 /**
437 * Moving from/to states conserves old notification density.
438 */
439 @Test
440 public void testConservesNotificationDensity() {
441 testConservesNotificationDensity(0 /* count */, ScrimController.GRADIENT_SCRIM_ALPHA);
442 testConservesNotificationDensity(3 /* count */, ScrimController.GRADIENT_SCRIM_ALPHA_BUSY);
443 }
444
Lucas Dupin3daf3b02018-02-27 17:23:41 -0800445 @Test
Lucas Dupin38962d72018-03-14 12:41:39 -0700446 public void testScrimFocus() {
447 mScrimController.transitionTo(ScrimState.AOD);
448 Assert.assertFalse("Should not be focusable on AOD", mScrimBehind.isFocusable());
449 Assert.assertFalse("Should not be focusable on AOD", mScrimInFront.isFocusable());
450
451 mScrimController.transitionTo(ScrimState.KEYGUARD);
452 Assert.assertTrue("Should be focusable on keyguard", mScrimBehind.isFocusable());
453 Assert.assertTrue("Should be focusable on keyguard", mScrimInFront.isFocusable());
454 }
455
Lucas Dupind5107302018-03-19 15:30:29 -0700456 @Test
457 public void testHidesShowWhenLockedActivity() {
458 mScrimController.setWallpaperSupportsAmbientMode(true);
459 mScrimController.setKeyguardOccluded(true);
460 mScrimController.transitionTo(ScrimState.AOD);
461 mScrimController.finishAnimationsImmediately();
462 assertScrimVisibility(VISIBILITY_FULLY_TRANSPARENT, VISIBILITY_FULLY_OPAQUE);
463
464 mScrimController.transitionTo(ScrimState.PULSING);
465 mScrimController.finishAnimationsImmediately();
466 assertScrimVisibility(VISIBILITY_FULLY_TRANSPARENT, VISIBILITY_FULLY_OPAQUE);
467 }
468
Lucas Dupin78949b82018-04-03 18:54:39 -0700469 @Test
470 public void testEatsTouchEvent() {
471 HashSet<ScrimState> eatsTouches =
472 new HashSet<>(Arrays.asList(ScrimState.AOD, ScrimState.PULSING));
473 for (ScrimState state : ScrimState.values()) {
474 if (state == ScrimState.UNINITIALIZED) {
475 continue;
476 }
477 mScrimController.transitionTo(state);
478 mScrimController.finishAnimationsImmediately();
479 Assert.assertEquals("Should be clickable unless AOD or PULSING, was: " + state,
480 mScrimBehind.getViewAlpha() != 0 && !eatsTouches.contains(state),
481 mScrimBehind.isClickable());
482 }
483 }
484
Lucas Dupinea0116e2018-04-05 10:09:29 -0700485 @Test
486 public void testAnimatesTransitionToAod() {
487 when(mDozeParamenters.shouldControlScreenOff()).thenReturn(false);
488 ScrimState.AOD.prepare(ScrimState.KEYGUARD);
489 Assert.assertFalse("No animation when ColorFade kicks in",
490 ScrimState.AOD.getAnimateChange());
491
492 reset(mDozeParamenters);
493 when(mDozeParamenters.shouldControlScreenOff()).thenReturn(true);
494 ScrimState.AOD.prepare(ScrimState.KEYGUARD);
495 Assert.assertTrue("Animate scrims when ColorFade won't be triggered",
496 ScrimState.AOD.getAnimateChange());
497 }
498
Lucas Dupinb380c882018-02-25 21:57:17 -0800499 /**
500 * Conserves old notification density after leaving state and coming back.
501 *
502 * @param count How many notification.
503 * @param expectedAlpha Expected alpha.
504 */
505 private void testConservesNotificationDensity(int count, float expectedAlpha) {
506 mScrimController.setNotificationCount(count);
507 mScrimController.transitionTo(ScrimState.UNLOCKED);
508 mScrimController.finishAnimationsImmediately();
509
510 mScrimController.transitionTo(ScrimState.KEYGUARD);
511 mScrimController.finishAnimationsImmediately();
512
513 Assert.assertEquals("Doesn't respect notification busyness after transition",
514 expectedAlpha, mScrimBehind.getViewAlpha(), 0.01f);
515 }
516
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800517 private void assertScrimTint(ScrimView scrimView, boolean tinted) {
518 final boolean viewIsTinted = scrimView.getTint() != Color.TRANSPARENT;
519 final String name = scrimView == mScrimInFront ? "front" : "back";
520 Assert.assertEquals("Tint test failed at state " + mScrimController.getState()
521 +" with scrim: " + name + " and tint: " + Integer.toHexString(scrimView.getTint()),
522 tinted, viewIsTinted);
523 }
524
Lucas Dupin82aa1632017-12-13 00:13:57 -0800525 private void assertScrimVisibility(int inFront, int behind) {
526 boolean inFrontVisible = inFront != ScrimController.VISIBILITY_FULLY_TRANSPARENT;
527 boolean behindVisible = behind != ScrimController.VISIBILITY_FULLY_TRANSPARENT;
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800528 Assert.assertEquals("Unexpected front scrim visibility. Alpha is "
Lucas Dupin82aa1632017-12-13 00:13:57 -0800529 + mScrimInFront.getViewAlpha(), inFrontVisible, mScrimInFront.getViewAlpha() > 0);
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800530 Assert.assertEquals("Unexpected back scrim visibility. Alpha is "
Lucas Dupin82aa1632017-12-13 00:13:57 -0800531 + mScrimBehind.getViewAlpha(), behindVisible, mScrimBehind.getViewAlpha() > 0);
532
533 final int visibility;
534 if (inFront == VISIBILITY_FULLY_OPAQUE || behind == VISIBILITY_FULLY_OPAQUE) {
535 visibility = VISIBILITY_FULLY_OPAQUE;
536 } else if (inFront > VISIBILITY_FULLY_TRANSPARENT || behind > VISIBILITY_FULLY_TRANSPARENT) {
537 visibility = VISIBILITY_SEMI_TRANSPARENT;
538 } else {
539 visibility = VISIBILITY_FULLY_TRANSPARENT;
540 }
541 Assert.assertEquals("Invalid visibility.", visibility, mScrimVisibility);
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800542 }
543
544 /**
545 * Special version of ScrimController where animations have 0 duration for test purposes.
546 */
547 private class SynchronousScrimController extends ScrimController {
548
549 private FakeHandler mHandler;
Lucas Dupin80a3fcc2018-02-07 10:49:55 -0800550 private boolean mAnimationCancelled;
Lucas Dupin16cfe452018-02-08 13:14:50 -0800551 boolean mOnPreDrawCalled;
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800552
shawnlin317db372018-04-09 19:49:48 +0800553 SynchronousScrimController(ScrimView scrimBehind, ScrimView scrimInFront,
554 Consumer<Float> scrimBehindAlphaListener,
555 Consumer<GradientColors> scrimInFrontColorListener,
Lucas Dupin82aa1632017-12-13 00:13:57 -0800556 Consumer<Integer> scrimVisibleListener, DozeParameters dozeParameters,
557 AlarmManager alarmManager) {
shawnlin317db372018-04-09 19:49:48 +0800558 super(scrimBehind, scrimInFront, scrimBehindAlphaListener, scrimInFrontColorListener,
Lucas Dupin82aa1632017-12-13 00:13:57 -0800559 scrimVisibleListener, dozeParameters, alarmManager);
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800560 mHandler = new FakeHandler(Looper.myLooper());
561 }
562
Lucas Dupin16cfe452018-02-08 13:14:50 -0800563 @Override
564 public boolean onPreDraw() {
565 mOnPreDrawCalled = true;
566 return super.onPreDraw();
567 }
568
Lucas Dupin3503c5f2018-03-02 19:04:00 -0800569 void finishAnimationsImmediately() {
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800570 boolean[] animationFinished = {false};
571 setOnAnimationFinished(()-> animationFinished[0] = true);
572
573 // Execute code that will trigger animations.
574 onPreDraw();
575
576 // Force finish screen blanking.
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800577 mHandler.dispatchQueuedMessages();
578 // Force finish all animations.
579 endAnimation(mScrimBehind, TAG_KEY_ANIM);
580 endAnimation(mScrimInFront, TAG_KEY_ANIM);
581
582 if (!animationFinished[0]) {
583 throw new IllegalStateException("Animation never finished");
584 }
585 }
586
Lucas Dupin3503c5f2018-03-02 19:04:00 -0800587 boolean wasAnimationJustCancelled() {
Lucas Dupin80a3fcc2018-02-07 10:49:55 -0800588 final boolean wasCancelled = mAnimationCancelled;
589 mAnimationCancelled = false;
590 return wasCancelled;
591 }
592
Lucas Dupin3daf3b02018-02-27 17:23:41 -0800593 private void endAnimation(View scrimView, int tag) {
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800594 Animator animator = (Animator) scrimView.getTag(tag);
595 if (animator != null) {
596 animator.end();
597 }
598 }
599
600 @Override
Lucas Dupin80a3fcc2018-02-07 10:49:55 -0800601 protected void cancelAnimator(ValueAnimator previousAnimator) {
602 super.cancelAnimator(previousAnimator);
603 mAnimationCancelled = true;
604 }
605
606 @Override
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800607 protected Handler getHandler() {
608 return mHandler;
609 }
610
611 @Override
612 protected WakeLock createWakeLock() {
613 return mWakeLock;
614 }
Lucas Dupin8c7cb022018-02-05 10:49:03 -0800615
616 /**
617 * Do not wait for a frame since we're in a test environment.
618 * @param callback What to execute.
619 */
620 @Override
Lucas Dupin0791d972018-03-26 13:32:16 -0700621 protected void doOnTheNextFrame(Runnable callback) {
622 callback.run();
Lucas Dupin8c7cb022018-02-05 10:49:03 -0800623 }
Lucas Dupin9e3fa102017-11-08 17:16:55 -0800624 }
625
626}