blob: bf067947f2b9cbf6acc975d3f2496eac4f2c83c9 [file] [log] [blame]
Lucas Dupin957e50c2017-10-10 11:23:27 -07001/*
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.keyguard;
18
Lucas Dupin1f7374a2018-02-26 18:08:33 -080019import static org.mockito.ArgumentMatchers.any;
20import static org.mockito.ArgumentMatchers.anyString;
21import static org.mockito.ArgumentMatchers.eq;
Lucas Dupin7e171e22018-12-20 11:29:35 -080022import static org.mockito.Mockito.mock;
Lucas Dupin8b77a3b2018-05-01 13:23:03 -070023import static org.mockito.Mockito.never;
Lucas Dupin3d560e42019-01-07 13:52:44 -080024import static org.mockito.Mockito.reset;
Lucas Dupin8b77a3b2018-05-01 13:23:03 -070025import static org.mockito.Mockito.spy;
Lucas Dupin1f7374a2018-02-26 18:08:33 -080026import static org.mockito.Mockito.verify;
Lucas Dupinc9548242019-04-11 10:59:13 -070027import static org.mockito.Mockito.when;
Lucas Dupin1f7374a2018-02-26 18:08:33 -080028
29import android.app.AlarmManager;
30import android.content.ContentResolver;
Lucas Dupin6a03a9f2018-12-20 17:13:52 -080031import android.media.MediaMetadata;
Lucas Dupin6b40d5e2019-05-15 19:47:11 -070032import android.media.session.PlaybackState;
Lucas Dupin957e50c2017-10-10 11:23:27 -070033import android.net.Uri;
Lucas Dupin8b77a3b2018-05-01 13:23:03 -070034import android.provider.Settings;
Lucas Dupin957e50c2017-10-10 11:23:27 -070035import android.testing.AndroidTestingRunner;
36import android.testing.TestableLooper;
37import android.testing.TestableLooper.RunWithLooper;
Lucas Dupin627ad372018-11-27 10:28:25 +010038
39import androidx.slice.Slice;
40import androidx.slice.SliceItem;
41import androidx.slice.SliceProvider;
42import androidx.slice.SliceSpecs;
43import androidx.slice.builders.ListBuilder;
44import androidx.slice.core.SliceQuery;
Brett Chabot84151d92019-02-27 15:37:59 -080045import androidx.test.filters.SmallTest;
Lucas Dupin957e50c2017-10-10 11:23:27 -070046
Lucas Dupin7e171e22018-12-20 11:29:35 -080047import com.android.keyguard.KeyguardUpdateMonitor;
Lucas Dupin957e50c2017-10-10 11:23:27 -070048import com.android.systemui.SysuiTestCase;
Beverly8fdb5332019-02-04 14:29:49 -050049import com.android.systemui.plugins.statusbar.StatusBarStateController;
Lucas Dupin6a03a9f2018-12-20 17:13:52 -080050import com.android.systemui.statusbar.NotificationMediaManager;
Lucas Dupin2d345232019-06-20 14:07:57 -070051import com.android.systemui.statusbar.phone.KeyguardBypassController;
52import com.android.systemui.statusbar.policy.ZenModeController;
53import com.android.systemui.util.wakelock.SettableWakeLock;
Lucas Dupin957e50c2017-10-10 11:23:27 -070054
55import org.junit.Assert;
56import org.junit.Before;
57import org.junit.Test;
58import org.junit.runner.RunWith;
Lucas Dupin1f7374a2018-02-26 18:08:33 -080059import org.mockito.Mock;
60import org.mockito.MockitoAnnotations;
Lucas Dupin957e50c2017-10-10 11:23:27 -070061
Jason Monk459fee32018-01-09 20:37:27 -050062import java.util.Arrays;
Jeff Gastonc2100af2018-04-04 01:54:52 -040063import java.util.HashSet;
Lucas Dupin8b77a3b2018-05-01 13:23:03 -070064import java.util.concurrent.TimeUnit;
Jason Monk459fee32018-01-09 20:37:27 -050065
Lucas Dupin957e50c2017-10-10 11:23:27 -070066@SmallTest
67@RunWith(AndroidTestingRunner.class)
Jason Monka716bac2018-12-05 15:48:21 -050068@RunWithLooper
Lucas Dupin957e50c2017-10-10 11:23:27 -070069public class KeyguardSliceProviderTest extends SysuiTestCase {
70
Lucas Dupin1f7374a2018-02-26 18:08:33 -080071 @Mock
72 private ContentResolver mContentResolver;
73 @Mock
74 private AlarmManager mAlarmManager;
Lucas Dupin6a03a9f2018-12-20 17:13:52 -080075 @Mock
76 private NotificationMediaManager mNotificationMediaManager;
Lucas Dupin3d560e42019-01-07 13:52:44 -080077 @Mock
78 private StatusBarStateController mStatusBarStateController;
Lucas Dupin2d345232019-06-20 14:07:57 -070079 @Mock
80 private KeyguardBypassController mKeyguardBypassController;
81 @Mock
82 private ZenModeController mZenModeController;
83 @Mock
84 private SettableWakeLock mMediaWakeLock;
85 @Mock
86 private KeyguardUpdateMonitor mKeyguardUpdateMonitor;
Lucas Dupin957e50c2017-10-10 11:23:27 -070087 private TestableKeyguardSliceProvider mProvider;
Lucas Dupin8b77a3b2018-05-01 13:23:03 -070088 private boolean mIsZenMode;
Lucas Dupin957e50c2017-10-10 11:23:27 -070089
90 @Before
91 public void setup() {
Lucas Dupin1f7374a2018-02-26 18:08:33 -080092 MockitoAnnotations.initMocks(this);
Lucas Dupin8b77a3b2018-05-01 13:23:03 -070093 mIsZenMode = false;
Lucas Dupin957e50c2017-10-10 11:23:27 -070094 mProvider = new TestableKeyguardSliceProvider();
95 mProvider.attachInfo(getContext(), null);
Lucas Dupin940e4902019-06-03 15:43:20 -070096 mProvider.initDependencies(mNotificationMediaManager, mStatusBarStateController,
97 mKeyguardBypassController);
Jeff Gastonc2100af2018-04-04 01:54:52 -040098 SliceProvider.setSpecs(new HashSet<>(Arrays.asList(SliceSpecs.LIST)));
Lucas Dupin957e50c2017-10-10 11:23:27 -070099 }
100
101 @Test
102 public void registersClockUpdate() {
103 Assert.assertTrue("registerClockUpdate should have been called during initialization.",
104 mProvider.isRegistered());
105 }
106
107 @Test
Lucas Dupin957e50c2017-10-10 11:23:27 -0700108 public void returnsValidSlice() {
Lucas Dupin1f7374a2018-02-26 18:08:33 -0800109 Slice slice = mProvider.onBindSlice(mProvider.getUri());
Lucas Dupin6bd86012017-12-05 17:58:57 -0800110 SliceItem text = SliceQuery.find(slice, android.app.slice.SliceItem.FORMAT_TEXT,
111 android.app.slice.Slice.HINT_TITLE,
Lucas Dupin957e50c2017-10-10 11:23:27 -0700112 null /* nonHints */);
113 Assert.assertNotNull("Slice must provide a title.", text);
114 }
115
116 @Test
Lucas Dupin940e4902019-06-03 15:43:20 -0700117 public void onBindSlice_readsMedia_withoutBypass() {
Lucas Dupin6a03a9f2018-12-20 17:13:52 -0800118 MediaMetadata metadata = mock(MediaMetadata.class);
Lucas Dupinc9548242019-04-11 10:59:13 -0700119 when(metadata.getText(any())).thenReturn("metadata");
Lucas Dupin3d560e42019-01-07 13:52:44 -0800120 mProvider.onDozingChanged(true);
Lucas Dupin6b40d5e2019-05-15 19:47:11 -0700121 mProvider.onMetadataOrStateChanged(metadata, PlaybackState.STATE_PLAYING);
Lucas Dupin6a03a9f2018-12-20 17:13:52 -0800122 mProvider.onBindSlice(mProvider.getUri());
123 verify(metadata).getText(eq(MediaMetadata.METADATA_KEY_TITLE));
124 verify(metadata).getText(eq(MediaMetadata.METADATA_KEY_ARTIST));
125 verify(mNotificationMediaManager).getMediaIcon();
126 }
127
128 @Test
Lucas Dupin940e4902019-06-03 15:43:20 -0700129 public void onBindSlice_readsMedia_withBypass_notDozing() {
130 MediaMetadata metadata = mock(MediaMetadata.class);
131 when(metadata.getText(any())).thenReturn("metadata");
132 when(mKeyguardBypassController.getBypassEnabled()).thenReturn(true);
133 mProvider.onMetadataOrStateChanged(metadata, PlaybackState.STATE_PLAYING);
134 mProvider.onBindSlice(mProvider.getUri());
135 verify(metadata).getText(eq(MediaMetadata.METADATA_KEY_TITLE));
136 verify(metadata).getText(eq(MediaMetadata.METADATA_KEY_ARTIST));
137 verify(mNotificationMediaManager).getMediaIcon();
138 }
139
140 @Test
Lucas Dupin957e50c2017-10-10 11:23:27 -0700141 public void cleansDateFormat() {
Lucas Dupin7e171e22018-12-20 11:29:35 -0800142 mProvider.mKeyguardUpdateMonitorCallback.onTimeZoneChanged(null);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700143 TestableLooper.get(this).processAllMessages();
144 Assert.assertEquals("Date format should have been cleaned.", 1 /* expected */,
145 mProvider.mCleanDateFormatInvokations);
146 }
147
148 @Test
149 public void updatesClock() {
Lucas Dupin7e171e22018-12-20 11:29:35 -0800150 mProvider.mKeyguardUpdateMonitorCallback.onTimeChanged();
Lucas Dupin957e50c2017-10-10 11:23:27 -0700151 TestableLooper.get(this).processAllMessages();
Lucas Dupin1f7374a2018-02-26 18:08:33 -0800152 verify(mContentResolver).notifyChange(eq(mProvider.getUri()), eq(null));
153 }
154
155 @Test
156 public void schedulesAlarm12hBefore() {
157 long in16Hours = System.currentTimeMillis() + TimeUnit.HOURS.toHours(16);
158 AlarmManager.AlarmClockInfo alarmClockInfo = new AlarmManager.AlarmClockInfo(in16Hours, null);
159 mProvider.onNextAlarmChanged(alarmClockInfo);
160
161 long twelveHours = TimeUnit.HOURS.toMillis(KeyguardSliceProvider.ALARM_VISIBILITY_HOURS);
162 long triggerAt = in16Hours - twelveHours;
163 verify(mAlarmManager).setExact(eq(AlarmManager.RTC), eq(triggerAt), anyString(), any(),
164 any());
165 }
166
167 @Test
168 public void updatingNextAlarmInvalidatesSlice() {
169 long in16Hours = System.currentTimeMillis() + TimeUnit.HOURS.toHours(8);
170 AlarmManager.AlarmClockInfo alarmClockInfo = new AlarmManager.AlarmClockInfo(in16Hours, null);
171 mProvider.onNextAlarmChanged(alarmClockInfo);
172
173 verify(mContentResolver).notifyChange(eq(mProvider.getUri()), eq(null));
Lucas Dupin957e50c2017-10-10 11:23:27 -0700174 }
175
Lucas Dupin8b77a3b2018-05-01 13:23:03 -0700176 @Test
177 public void onZenChanged_updatesSlice() {
178 mProvider.onZenChanged(Settings.Global.ZEN_MODE_ALARMS);
179 verify(mContentResolver).notifyChange(eq(mProvider.getUri()), eq(null));
180 }
181
182 @Test
183 public void addZenMode_addedToSlice() {
Jason Monk1045e0b2018-08-06 09:42:10 -0400184 ListBuilder listBuilder = spy(new ListBuilder(getContext(), mProvider.getUri(),
185 ListBuilder.INFINITY));
Lucas Dupin6a03a9f2018-12-20 17:13:52 -0800186 mProvider.addZenModeLocked(listBuilder);
Lucas Dupin8b77a3b2018-05-01 13:23:03 -0700187 verify(listBuilder, never()).addRow(any(ListBuilder.RowBuilder.class));
188
189 mIsZenMode = true;
Lucas Dupin6a03a9f2018-12-20 17:13:52 -0800190 mProvider.addZenModeLocked(listBuilder);
Lucas Dupin8b77a3b2018-05-01 13:23:03 -0700191 verify(listBuilder).addRow(any(ListBuilder.RowBuilder.class));
192 }
193
Lucas Dupin6a03a9f2018-12-20 17:13:52 -0800194 @Test
195 public void onMetadataChanged_updatesSlice() {
Lucas Dupin3d560e42019-01-07 13:52:44 -0800196 mProvider.onDozingChanged(true);
Lucas Dupin76527ee2019-02-20 10:46:23 -0800197 reset(mContentResolver);
Lucas Dupin6b40d5e2019-05-15 19:47:11 -0700198 mProvider.onMetadataOrStateChanged(mock(MediaMetadata.class), PlaybackState.STATE_PLAYING);
Lucas Dupin6a03a9f2018-12-20 17:13:52 -0800199 verify(mContentResolver).notifyChange(eq(mProvider.getUri()), eq(null));
Lucas Dupin3d560e42019-01-07 13:52:44 -0800200
201 // Hides after waking up
202 reset(mContentResolver);
203 mProvider.onDozingChanged(false);
204 verify(mContentResolver).notifyChange(eq(mProvider.getUri()), eq(null));
Lucas Dupin3d560e42019-01-07 13:52:44 -0800205 }
206
207 @Test
208 public void onDozingChanged_updatesSliceIfMedia() {
Lucas Dupin6b40d5e2019-05-15 19:47:11 -0700209 mProvider.onMetadataOrStateChanged(mock(MediaMetadata.class), PlaybackState.STATE_PLAYING);
Lucas Dupin76527ee2019-02-20 10:46:23 -0800210 reset(mContentResolver);
211 // Show media when dozing
Lucas Dupin3d560e42019-01-07 13:52:44 -0800212 mProvider.onDozingChanged(true);
213 verify(mContentResolver).notifyChange(eq(mProvider.getUri()), eq(null));
214
215 // Do not notify again if nothing changed
216 reset(mContentResolver);
217 mProvider.onDozingChanged(true);
218 verify(mContentResolver, never()).notifyChange(eq(mProvider.getUri()), eq(null));
Lucas Dupin6a03a9f2018-12-20 17:13:52 -0800219 }
220
Lucas Dupin2d345232019-06-20 14:07:57 -0700221 @Test
222 public void onDestroy_noCrash() {
223 mProvider.onDestroy();
224 }
225
226 @Test
227 public void onDestroy_unregisterListeners() {
228 mProvider.registerClockUpdate();
229 mProvider.onDestroy();
230 verify(mMediaWakeLock).setAcquired(eq(false));
231 verify(mAlarmManager).cancel(any(AlarmManager.OnAlarmListener.class));
232 verify(mKeyguardUpdateMonitor).removeCallback(any());
233 }
234
Lucas Dupin957e50c2017-10-10 11:23:27 -0700235 private class TestableKeyguardSliceProvider extends KeyguardSliceProvider {
236 int mCleanDateFormatInvokations;
Lucas Dupin1f7374a2018-02-26 18:08:33 -0800237 private int mCounter;
Lucas Dupin957e50c2017-10-10 11:23:27 -0700238
Lucas Dupin1f7374a2018-02-26 18:08:33 -0800239 Uri getUri() {
240 return mSliceUri;
241 }
242
243 @Override
244 public boolean onCreateSliceProvider() {
245 super.onCreateSliceProvider();
246 mAlarmManager = KeyguardSliceProviderTest.this.mAlarmManager;
247 mContentResolver = KeyguardSliceProviderTest.this.mContentResolver;
Lucas Dupin2d345232019-06-20 14:07:57 -0700248 mZenModeController = KeyguardSliceProviderTest.this.mZenModeController;
249 mMediaWakeLock = KeyguardSliceProviderTest.this.mMediaWakeLock;
Lucas Dupin1f7374a2018-02-26 18:08:33 -0800250 return true;
251 }
252
Lucas Dupin957e50c2017-10-10 11:23:27 -0700253 @Override
Lucas Dupin627ad372018-11-27 10:28:25 +0100254 protected boolean isDndOn() {
Lucas Dupin8b77a3b2018-05-01 13:23:03 -0700255 return mIsZenMode;
256 }
257
258 @Override
Lucas Dupin6a03a9f2018-12-20 17:13:52 -0800259 void cleanDateFormatLocked() {
260 super.cleanDateFormatLocked();
Lucas Dupin957e50c2017-10-10 11:23:27 -0700261 mCleanDateFormatInvokations++;
262 }
263
264 @Override
Lucas Dupin7e171e22018-12-20 11:29:35 -0800265 public KeyguardUpdateMonitor getKeyguardUpdateMonitor() {
Lucas Dupin2d345232019-06-20 14:07:57 -0700266 return mKeyguardUpdateMonitor;
Lucas Dupin7e171e22018-12-20 11:29:35 -0800267 }
268
269 @Override
Lucas Dupin6a03a9f2018-12-20 17:13:52 -0800270 protected String getFormattedDateLocked() {
271 return super.getFormattedDateLocked() + mCounter++;
272 }
Lucas Dupin957e50c2017-10-10 11:23:27 -0700273 }
274
275}