• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2013 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 com.android.deskclock.alarms;
17 
18 import android.annotation.TargetApi;
19 import android.app.AlarmManager;
20 import android.app.AlarmManager.AlarmClockInfo;
21 import android.app.PendingIntent;
22 import android.content.BroadcastReceiver;
23 import android.content.ContentResolver;
24 import android.content.Context;
25 import android.content.Intent;
26 import android.net.Uri;
27 import android.os.Build;
28 import android.os.Handler;
29 import android.os.PowerManager;
30 import android.provider.Settings;
31 import android.support.v4.app.NotificationManagerCompat;
32 import android.text.format.DateFormat;
33 import android.widget.Toast;
34 
35 import com.android.deskclock.AlarmAlertWakeLock;
36 import com.android.deskclock.AlarmClockFragment;
37 import com.android.deskclock.AlarmUtils;
38 import com.android.deskclock.AsyncHandler;
39 import com.android.deskclock.DeskClock;
40 import com.android.deskclock.LogUtils;
41 import com.android.deskclock.R;
42 import com.android.deskclock.Utils;
43 import com.android.deskclock.data.DataModel;
44 import com.android.deskclock.events.Events;
45 import com.android.deskclock.provider.Alarm;
46 import com.android.deskclock.provider.AlarmInstance;
47 
48 import java.util.Calendar;
49 import java.util.Collections;
50 import java.util.Comparator;
51 import java.util.List;
52 
53 import static android.content.Context.ALARM_SERVICE;
54 import static android.provider.Settings.System.NEXT_ALARM_FORMATTED;
55 
56 /**
57  * This class handles all the state changes for alarm instances. You need to
58  * register all alarm instances with the state manager if you want them to
59  * be activated. If a major time change has occurred (ie. TIMEZONE_CHANGE, TIMESET_CHANGE),
60  * then you must also re-register instances to fix their states.
61  *
62  * Please see {@link #registerInstance) for special transitions when major time changes
63  * occur.
64  *
65  * Following states:
66  *
67  * SILENT_STATE:
68  * This state is used when the alarm is activated, but doesn't need to display anything. It
69  * is in charge of changing the alarm instance state to a LOW_NOTIFICATION_STATE.
70  *
71  * LOW_NOTIFICATION_STATE:
72  * This state is used to notify the user that the alarm will go off
73  * {@link AlarmInstance#LOW_NOTIFICATION_HOUR_OFFSET}. This
74  * state handles the state changes to HIGH_NOTIFICATION_STATE, HIDE_NOTIFICATION_STATE and
75  * DISMISS_STATE.
76  *
77  * HIDE_NOTIFICATION_STATE:
78  * This is a transient state of the LOW_NOTIFICATION_STATE, where the user wants to hide the
79  * notification. This will sit and wait until the HIGH_PRIORITY_NOTIFICATION should go off.
80  *
81  * HIGH_NOTIFICATION_STATE:
82  * This state behaves like the LOW_NOTIFICATION_STATE, but doesn't allow the user to hide it.
83  * This state is in charge of triggering a FIRED_STATE or DISMISS_STATE.
84  *
85  * SNOOZED_STATE:
86  * The SNOOZED_STATE behaves like a HIGH_NOTIFICATION_STATE, but with a different message. It
87  * also increments the alarm time in the instance to reflect the new snooze time.
88  *
89  * FIRED_STATE:
90  * The FIRED_STATE is used when the alarm is firing. It will start the AlarmService, and wait
91  * until the user interacts with the alarm via SNOOZED_STATE or DISMISS_STATE change. If the user
92  * doesn't then it might be change to MISSED_STATE if auto-silenced was enabled.
93  *
94  * MISSED_STATE:
95  * The MISSED_STATE is used when the alarm already fired, but the user could not interact with
96  * it. At this point the alarm instance is dead and we check the parent alarm to see if we need
97  * to disable or schedule a new alarm_instance. There is also a notification shown to the user
98  * that he/she missed the alarm and that stays for
99  * {@link AlarmInstance#MISSED_TIME_TO_LIVE_HOUR_OFFSET} or until the user acknownledges it.
100  *
101  * DISMISS_STATE:
102  * This is really a transient state that will properly delete the alarm instance. Use this state,
103  * whenever you want to get rid of the alarm instance. This state will also check the alarm
104  * parent to see if it should disable or schedule a new alarm instance.
105  */
106 public final class AlarmStateManager extends BroadcastReceiver {
107     // Intent action to trigger an instance state change.
108     public static final String CHANGE_STATE_ACTION = "change_state";
109 
110     // Intent action to show the alarm and dismiss the instance
111     public static final String SHOW_AND_DISMISS_ALARM_ACTION = "show_and_dismiss_alarm";
112 
113     // Intent action for an AlarmManager alarm serving only to set the next alarm indicators
114     private static final String INDICATOR_ACTION = "indicator";
115 
116     // System intent action to notify AppWidget that we changed the alarm text.
117     public static final String ACTION_ALARM_CHANGED = "com.android.deskclock.ALARM_CHANGED";
118 
119     // Extra key to set the desired state change.
120     public static final String ALARM_STATE_EXTRA = "intent.extra.alarm.state";
121 
122     // Extra key to indicate the state change was launched from a notification.
123     public static final String FROM_NOTIFICATION_EXTRA = "intent.extra.from.notification";
124 
125     // Extra key to set the global broadcast id.
126     private static final String ALARM_GLOBAL_ID_EXTRA = "intent.extra.alarm.global.id";
127 
128     // Intent category tags used to dismiss, snooze or delete an alarm
129     public static final String ALARM_DISMISS_TAG = "DISMISS_TAG";
130     public static final String ALARM_SNOOZE_TAG = "SNOOZE_TAG";
131     public static final String ALARM_DELETE_TAG = "DELETE_TAG";
132 
133     // Intent category tag used when schedule state change intents in alarm manager.
134     private static final String ALARM_MANAGER_TAG = "ALARM_MANAGER";
135 
136     // Buffer time in seconds to fire alarm instead of marking it missed.
137     public static final int ALARM_FIRE_BUFFER = 15;
138 
139     // A factory for the current time; can be mocked for testing purposes.
140     private static CurrentTimeFactory sCurrentTimeFactory;
141 
142     // Schedules alarm state transitions; can be mocked for testing purposes.
143     private static StateChangeScheduler sStateChangeScheduler =
144             new AlarmManagerStateChangeScheduler();
145 
getCurrentTime()146     private static Calendar getCurrentTime() {
147         return sCurrentTimeFactory == null
148                 ? DataModel.getDataModel().getCalendar()
149                 : sCurrentTimeFactory.getCurrentTime();
150     }
151 
setCurrentTimeFactory(CurrentTimeFactory currentTimeFactory)152     static void setCurrentTimeFactory(CurrentTimeFactory currentTimeFactory) {
153         sCurrentTimeFactory = currentTimeFactory;
154     }
155 
setStateChangeScheduler(StateChangeScheduler stateChangeScheduler)156     static void setStateChangeScheduler(StateChangeScheduler stateChangeScheduler) {
157         if (stateChangeScheduler == null) {
158             stateChangeScheduler = new AlarmManagerStateChangeScheduler();
159         }
160         sStateChangeScheduler = stateChangeScheduler;
161     }
162 
163     /**
164      * Update the next alarm stored in framework. This value is also displayed in digital widgets
165      * and the clock tab in this app.
166      */
updateNextAlarm(Context context)167     private static void updateNextAlarm(Context context) {
168         final AlarmInstance nextAlarm = getNextFiringAlarm(context);
169 
170         if (Utils.isPreL()) {
171             updateNextAlarmInSystemSettings(context, nextAlarm);
172         } else {
173             updateNextAlarmInAlarmManager(context, nextAlarm);
174         }
175     }
176 
177     /**
178      * Returns an alarm instance of an alarm that's going to fire next.
179      *
180      * @param context application context
181      * @return an alarm instance that will fire earliest relative to current time.
182      */
getNextFiringAlarm(Context context)183     public static AlarmInstance getNextFiringAlarm(Context context) {
184         final ContentResolver cr = context.getContentResolver();
185         final String activeAlarmQuery = AlarmInstance.ALARM_STATE + "<" + AlarmInstance.FIRED_STATE;
186         final List<AlarmInstance> alarmInstances = AlarmInstance.getInstances(cr, activeAlarmQuery);
187 
188         AlarmInstance nextAlarm = null;
189         for (AlarmInstance instance : alarmInstances) {
190             if (nextAlarm == null || instance.getAlarmTime().before(nextAlarm.getAlarmTime())) {
191                 nextAlarm = instance;
192             }
193         }
194         return nextAlarm;
195     }
196 
197     /**
198      * Used in pre-L devices, where "next alarm" is stored in system settings.
199      */
200     @SuppressWarnings("deprecation")
201     @TargetApi(Build.VERSION_CODES.KITKAT)
updateNextAlarmInSystemSettings(Context context, AlarmInstance nextAlarm)202     private static void updateNextAlarmInSystemSettings(Context context, AlarmInstance nextAlarm) {
203         // Format the next alarm time if an alarm is scheduled.
204         String time = "";
205         if (nextAlarm != null) {
206             time = AlarmUtils.getFormattedTime(context, nextAlarm.getAlarmTime());
207         }
208 
209         try {
210             // Write directly to NEXT_ALARM_FORMATTED in all pre-L versions
211             Settings.System.putString(context.getContentResolver(), NEXT_ALARM_FORMATTED, time);
212 
213             LogUtils.i("Updated next alarm time to: \'" + time + '\'');
214 
215             // Send broadcast message so pre-L AppWidgets will recognize an update.
216             context.sendBroadcast(new Intent(ACTION_ALARM_CHANGED));
217         } catch (SecurityException se) {
218             // The user has most likely revoked WRITE_SETTINGS.
219             LogUtils.e("Unable to update next alarm to: \'" + time + '\'', se);
220         }
221     }
222 
223     /**
224      * Used in L and later devices where "next alarm" is stored in the Alarm Manager.
225      */
226     @TargetApi(Build.VERSION_CODES.LOLLIPOP)
updateNextAlarmInAlarmManager(Context context, AlarmInstance nextAlarm)227     private static void updateNextAlarmInAlarmManager(Context context, AlarmInstance nextAlarm) {
228         // Sets a surrogate alarm with alarm manager that provides the AlarmClockInfo for the
229         // alarm that is going to fire next. The operation is constructed such that it is ignored
230         // by AlarmStateManager.
231 
232         final AlarmManager alarmManager = (AlarmManager) context.getSystemService(ALARM_SERVICE);
233 
234         final int flags = nextAlarm == null ? PendingIntent.FLAG_NO_CREATE : 0;
235         final PendingIntent operation = PendingIntent.getBroadcast(context, 0 /* requestCode */,
236                 AlarmStateManager.createIndicatorIntent(context), flags);
237 
238         if (nextAlarm != null) {
239             LogUtils.i("Setting upcoming AlarmClockInfo for alarm: " + nextAlarm.mId);
240             long alarmTime = nextAlarm.getAlarmTime().getTimeInMillis();
241 
242             // Create an intent that can be used to show or edit details of the next alarm.
243             PendingIntent viewIntent = PendingIntent.getActivity(context, nextAlarm.hashCode(),
244                     AlarmNotifications.createViewAlarmIntent(context, nextAlarm),
245                     PendingIntent.FLAG_UPDATE_CURRENT);
246 
247             final AlarmClockInfo info = new AlarmClockInfo(alarmTime, viewIntent);
248             Utils.updateNextAlarm(alarmManager, info, operation);
249         } else if (operation != null) {
250             LogUtils.i("Canceling upcoming AlarmClockInfo");
251             alarmManager.cancel(operation);
252         }
253     }
254 
255     /**
256      * Used by dismissed and missed states, to update parent alarm. This will either
257      * disable, delete or reschedule parent alarm.
258      *
259      * @param context  application context
260      * @param instance to update parent for
261      */
updateParentAlarm(Context context, AlarmInstance instance)262     private static void updateParentAlarm(Context context, AlarmInstance instance) {
263         ContentResolver cr = context.getContentResolver();
264         Alarm alarm = Alarm.getAlarm(cr, instance.mAlarmId);
265         if (alarm == null) {
266             LogUtils.e("Parent has been deleted with instance: " + instance.toString());
267             return;
268         }
269 
270         if (!alarm.daysOfWeek.isRepeating()) {
271             if (alarm.deleteAfterUse) {
272                 LogUtils.i("Deleting parent alarm: " + alarm.id);
273                 Alarm.deleteAlarm(cr, alarm.id);
274             } else {
275                 LogUtils.i("Disabling parent alarm: " + alarm.id);
276                 alarm.enabled = false;
277                 Alarm.updateAlarm(cr, alarm);
278             }
279         } else {
280             // Schedule the next repeating instance which may be before the current instance if a
281             // time jump has occurred. Otherwise, if the current instance is the next instance
282             // and has already been fired, schedule the subsequent instance.
283             AlarmInstance nextRepeatedInstance = alarm.createInstanceAfter(getCurrentTime());
284             if (instance.mAlarmState > AlarmInstance.FIRED_STATE
285                     && nextRepeatedInstance.getAlarmTime().equals(instance.getAlarmTime())) {
286                 nextRepeatedInstance = alarm.createInstanceAfter(instance.getAlarmTime());
287             }
288 
289             LogUtils.i("Creating new instance for repeating alarm " + alarm.id + " at " +
290                     AlarmUtils.getFormattedTime(context, nextRepeatedInstance.getAlarmTime()));
291             AlarmInstance.addInstance(cr, nextRepeatedInstance);
292             registerInstance(context, nextRepeatedInstance, true);
293         }
294     }
295 
296     /**
297      * Utility method to create a proper change state intent.
298      *
299      * @param context  application context
300      * @param tag      used to make intent differ from other state change intents.
301      * @param instance to change state to
302      * @param state    to change to.
303      * @return intent that can be used to change an alarm instance state
304      */
createStateChangeIntent(Context context, String tag, AlarmInstance instance, Integer state)305     public static Intent createStateChangeIntent(Context context, String tag,
306             AlarmInstance instance, Integer state) {
307         // This intent is directed to AlarmService, though the actual handling of it occurs here
308         // in AlarmStateManager. The reason is that evidence exists showing the jump between the
309         // broadcast receiver (AlarmStateManager) and service (AlarmService) can be thwarted by the
310         // Out Of Memory killer. If clock is killed during that jump, firing an alarm can fail to
311         // occur. To be safer, the call begins in AlarmService, which has the power to display the
312         // firing alarm if needed, so no jump is needed.
313         Intent intent = AlarmInstance.createIntent(context, AlarmService.class, instance.mId);
314         intent.setAction(CHANGE_STATE_ACTION);
315         intent.addCategory(tag);
316         intent.putExtra(ALARM_GLOBAL_ID_EXTRA, DataModel.getDataModel().getGlobalIntentId());
317         if (state != null) {
318             intent.putExtra(ALARM_STATE_EXTRA, state.intValue());
319         }
320         return intent;
321     }
322 
323     /**
324      * Schedule alarm instance state changes with {@link AlarmManager}.
325      *
326      * @param ctx      application context
327      * @param time     to trigger state change
328      * @param instance to change state to
329      * @param newState to change to
330      */
scheduleInstanceStateChange(Context ctx, Calendar time, AlarmInstance instance, int newState)331     private static void scheduleInstanceStateChange(Context ctx, Calendar time,
332             AlarmInstance instance, int newState) {
333         sStateChangeScheduler.scheduleInstanceStateChange(ctx, time, instance, newState);
334     }
335 
336     /**
337      * Cancel all {@link AlarmManager} timers for instance.
338      *
339      * @param ctx      application context
340      * @param instance to disable all {@link AlarmManager} timers
341      */
cancelScheduledInstanceStateChange(Context ctx, AlarmInstance instance)342     private static void cancelScheduledInstanceStateChange(Context ctx, AlarmInstance instance) {
343         sStateChangeScheduler.cancelScheduledInstanceStateChange(ctx, instance);
344     }
345 
346 
347     /**
348      * This will set the alarm instance to the SILENT_STATE and update
349      * the application notifications and schedule any state changes that need
350      * to occur in the future.
351      *
352      * @param context  application context
353      * @param instance to set state to
354      */
setSilentState(Context context, AlarmInstance instance)355     public static void setSilentState(Context context, AlarmInstance instance) {
356         LogUtils.i("Setting silent state to instance " + instance.mId);
357 
358         // Update alarm in db
359         ContentResolver contentResolver = context.getContentResolver();
360         instance.mAlarmState = AlarmInstance.SILENT_STATE;
361         AlarmInstance.updateInstance(contentResolver, instance);
362 
363         // Setup instance notification and scheduling timers
364         AlarmNotifications.clearNotification(context, instance);
365         scheduleInstanceStateChange(context, instance.getLowNotificationTime(),
366                 instance, AlarmInstance.LOW_NOTIFICATION_STATE);
367     }
368 
369     /**
370      * This will set the alarm instance to the LOW_NOTIFICATION_STATE and update
371      * the application notifications and schedule any state changes that need
372      * to occur in the future.
373      *
374      * @param context  application context
375      * @param instance to set state to
376      */
setLowNotificationState(Context context, AlarmInstance instance)377     public static void setLowNotificationState(Context context, AlarmInstance instance) {
378         LogUtils.i("Setting low notification state to instance " + instance.mId);
379 
380         // Update alarm state in db
381         ContentResolver contentResolver = context.getContentResolver();
382         instance.mAlarmState = AlarmInstance.LOW_NOTIFICATION_STATE;
383         AlarmInstance.updateInstance(contentResolver, instance);
384 
385         // Setup instance notification and scheduling timers
386         AlarmNotifications.showLowPriorityNotification(context, instance);
387         scheduleInstanceStateChange(context, instance.getHighNotificationTime(),
388                 instance, AlarmInstance.HIGH_NOTIFICATION_STATE);
389     }
390 
391     /**
392      * This will set the alarm instance to the HIDE_NOTIFICATION_STATE and update
393      * the application notifications and schedule any state changes that need
394      * to occur in the future.
395      *
396      * @param context  application context
397      * @param instance to set state to
398      */
setHideNotificationState(Context context, AlarmInstance instance)399     public static void setHideNotificationState(Context context, AlarmInstance instance) {
400         LogUtils.i("Setting hide notification state to instance " + instance.mId);
401 
402         // Update alarm state in db
403         ContentResolver contentResolver = context.getContentResolver();
404         instance.mAlarmState = AlarmInstance.HIDE_NOTIFICATION_STATE;
405         AlarmInstance.updateInstance(contentResolver, instance);
406 
407         // Setup instance notification and scheduling timers
408         AlarmNotifications.clearNotification(context, instance);
409         scheduleInstanceStateChange(context, instance.getHighNotificationTime(),
410                 instance, AlarmInstance.HIGH_NOTIFICATION_STATE);
411     }
412 
413     /**
414      * This will set the alarm instance to the HIGH_NOTIFICATION_STATE and update
415      * the application notifications and schedule any state changes that need
416      * to occur in the future.
417      *
418      * @param context  application context
419      * @param instance to set state to
420      */
setHighNotificationState(Context context, AlarmInstance instance)421     public static void setHighNotificationState(Context context, AlarmInstance instance) {
422         LogUtils.i("Setting high notification state to instance " + instance.mId);
423 
424         // Update alarm state in db
425         ContentResolver contentResolver = context.getContentResolver();
426         instance.mAlarmState = AlarmInstance.HIGH_NOTIFICATION_STATE;
427         AlarmInstance.updateInstance(contentResolver, instance);
428 
429         // Setup instance notification and scheduling timers
430         AlarmNotifications.showHighPriorityNotification(context, instance);
431         scheduleInstanceStateChange(context, instance.getAlarmTime(),
432                 instance, AlarmInstance.FIRED_STATE);
433     }
434 
435     /**
436      * This will set the alarm instance to the FIRED_STATE and update
437      * the application notifications and schedule any state changes that need
438      * to occur in the future.
439      *
440      * @param context  application context
441      * @param instance to set state to
442      */
setFiredState(Context context, AlarmInstance instance)443     public static void setFiredState(Context context, AlarmInstance instance) {
444         LogUtils.i("Setting fire state to instance " + instance.mId);
445 
446         // Update alarm state in db
447         ContentResolver contentResolver = context.getContentResolver();
448         instance.mAlarmState = AlarmInstance.FIRED_STATE;
449         AlarmInstance.updateInstance(contentResolver, instance);
450 
451         if (instance.mAlarmId != null) {
452             // if the time changed *backward* and pushed an instance from missed back to fired,
453             // remove any other scheduled instances that may exist
454             AlarmInstance.deleteOtherInstances(context, contentResolver, instance.mAlarmId,
455                     instance.mId);
456         }
457 
458         Events.sendAlarmEvent(R.string.action_fire, 0);
459 
460         Calendar timeout = instance.getTimeout();
461         if (timeout != null) {
462             scheduleInstanceStateChange(context, timeout, instance, AlarmInstance.MISSED_STATE);
463         }
464 
465         // Instance not valid anymore, so find next alarm that will fire and notify system
466         updateNextAlarm(context);
467     }
468 
469     /**
470      * This will set the alarm instance to the SNOOZE_STATE and update
471      * the application notifications and schedule any state changes that need
472      * to occur in the future.
473      *
474      * @param context  application context
475      * @param instance to set state to
476      */
setSnoozeState(final Context context, AlarmInstance instance, boolean showToast)477     public static void setSnoozeState(final Context context, AlarmInstance instance,
478             boolean showToast) {
479         // Stop alarm if this instance is firing it
480         AlarmService.stopAlarm(context, instance);
481 
482         // Calculate the new snooze alarm time
483         final int snoozeMinutes = DataModel.getDataModel().getSnoozeLength();
484         Calendar newAlarmTime = Calendar.getInstance();
485         newAlarmTime.add(Calendar.MINUTE, snoozeMinutes);
486 
487         // Update alarm state and new alarm time in db.
488         LogUtils.i("Setting snoozed state to instance " + instance.mId + " for "
489                 + AlarmUtils.getFormattedTime(context, newAlarmTime));
490         instance.setAlarmTime(newAlarmTime);
491         instance.mAlarmState = AlarmInstance.SNOOZE_STATE;
492         AlarmInstance.updateInstance(context.getContentResolver(), instance);
493 
494         // Setup instance notification and scheduling timers
495         AlarmNotifications.showSnoozeNotification(context, instance);
496         scheduleInstanceStateChange(context, instance.getAlarmTime(),
497                 instance, AlarmInstance.FIRED_STATE);
498 
499         // Display the snooze minutes in a toast.
500         if (showToast) {
501             final Handler mainHandler = new Handler(context.getMainLooper());
502             final Runnable myRunnable = new Runnable() {
503                 @Override
504                 public void run() {
505                     String displayTime = String.format(context.getResources().getQuantityText
506                             (R.plurals.alarm_alert_snooze_set, snoozeMinutes).toString(),
507                             snoozeMinutes);
508                     Toast.makeText(context, displayTime, Toast.LENGTH_LONG).show();
509                 }
510             };
511             mainHandler.post(myRunnable);
512         }
513 
514         // Instance time changed, so find next alarm that will fire and notify system
515         updateNextAlarm(context);
516     }
517 
518     /**
519      * This will set the alarm instance to the MISSED_STATE and update
520      * the application notifications and schedule any state changes that need
521      * to occur in the future.
522      *
523      * @param context  application context
524      * @param instance to set state to
525      */
setMissedState(Context context, AlarmInstance instance)526     public static void setMissedState(Context context, AlarmInstance instance) {
527         LogUtils.i("Setting missed state to instance " + instance.mId);
528         // Stop alarm if this instance is firing it
529         AlarmService.stopAlarm(context, instance);
530 
531         // Check parent if it needs to reschedule, disable or delete itself
532         if (instance.mAlarmId != null) {
533             updateParentAlarm(context, instance);
534         }
535 
536         // Update alarm state
537         ContentResolver contentResolver = context.getContentResolver();
538         instance.mAlarmState = AlarmInstance.MISSED_STATE;
539         AlarmInstance.updateInstance(contentResolver, instance);
540 
541         // Setup instance notification and scheduling timers
542         AlarmNotifications.showMissedNotification(context, instance);
543         scheduleInstanceStateChange(context, instance.getMissedTimeToLive(),
544                 instance, AlarmInstance.DISMISSED_STATE);
545 
546         // Instance is not valid anymore, so find next alarm that will fire and notify system
547         updateNextAlarm(context);
548     }
549 
550     /**
551      * This will set the alarm instance to the PREDISMISSED_STATE and schedule an instance state
552      * change to DISMISSED_STATE at the regularly scheduled firing time.
553      *
554      * @param context  application context
555      * @param instance to set state to
556      */
setPreDismissState(Context context, AlarmInstance instance)557     public static void setPreDismissState(Context context, AlarmInstance instance) {
558         LogUtils.i("Setting predismissed state to instance " + instance.mId);
559 
560         // Update alarm in db
561         final ContentResolver contentResolver = context.getContentResolver();
562         instance.mAlarmState = AlarmInstance.PREDISMISSED_STATE;
563         AlarmInstance.updateInstance(contentResolver, instance);
564 
565         // Setup instance notification and scheduling timers
566         AlarmNotifications.clearNotification(context, instance);
567         scheduleInstanceStateChange(context, instance.getAlarmTime(), instance,
568                 AlarmInstance.DISMISSED_STATE);
569 
570         // Check parent if it needs to reschedule, disable or delete itself
571         if (instance.mAlarmId != null) {
572             updateParentAlarm(context, instance);
573         }
574 
575         updateNextAlarm(context);
576     }
577 
578     /**
579      * This just sets the alarm instance to DISMISSED_STATE.
580      */
setDismissState(Context context, AlarmInstance instance)581     public static void setDismissState(Context context, AlarmInstance instance) {
582         LogUtils.i("Setting dismissed state to instance " + instance.mId);
583         instance.mAlarmState = AlarmInstance.DISMISSED_STATE;
584         final ContentResolver contentResolver = context.getContentResolver();
585         AlarmInstance.updateInstance(contentResolver, instance);
586     }
587 
588     /**
589      * This will delete the alarm instance, update the application notifications, and schedule
590      * any state changes that need to occur in the future.
591      *
592      * @param context  application context
593      * @param instance to set state to
594      */
deleteInstanceAndUpdateParent(Context context, AlarmInstance instance)595     public static void deleteInstanceAndUpdateParent(Context context, AlarmInstance instance) {
596         LogUtils.i("Deleting instance " + instance.mId + " and updating parent alarm.");
597 
598         // Remove all other timers and notifications associated to it
599         unregisterInstance(context, instance);
600 
601         // Check parent if it needs to reschedule, disable or delete itself
602         if (instance.mAlarmId != null) {
603             updateParentAlarm(context, instance);
604         }
605 
606         // Delete instance as it is not needed anymore
607         AlarmInstance.deleteInstance(context.getContentResolver(), instance.mId);
608 
609         // Instance is not valid anymore, so find next alarm that will fire and notify system
610         updateNextAlarm(context);
611     }
612 
613     /**
614      * This will set the instance state to DISMISSED_STATE and remove its notifications and
615      * alarm timers.
616      *
617      * @param context  application context
618      * @param instance to unregister
619      */
unregisterInstance(Context context, AlarmInstance instance)620     public static void unregisterInstance(Context context, AlarmInstance instance) {
621         LogUtils.i("Unregistering instance " + instance.mId);
622         // Stop alarm if this instance is firing it
623         AlarmService.stopAlarm(context, instance);
624         AlarmNotifications.clearNotification(context, instance);
625         cancelScheduledInstanceStateChange(context, instance);
626         setDismissState(context, instance);
627     }
628 
629     /**
630      * This registers the AlarmInstance to the state manager. This will look at the instance
631      * and choose the most appropriate state to put it in. This is primarily used by new
632      * alarms, but it can also be called when the system time changes.
633      *
634      * Most state changes are handled by the states themselves, but during major time changes we
635      * have to correct the alarm instance state. This means we have to handle special cases as
636      * describe below:
637      *
638      * <ul>
639      *     <li>Make sure all dismissed alarms are never re-activated</li>
640      *     <li>Make sure pre-dismissed alarms stay predismissed</li>
641      *     <li>Make sure firing alarms stayed fired unless they should be auto-silenced</li>
642      *     <li>Missed instance that have parents should be re-enabled if we went back in time</li>
643      *     <li>If alarm was SNOOZED, then show the notification but don't update time</li>
644      *     <li>If low priority notification was hidden, then make sure it stays hidden</li>
645      * </ul>
646      *
647      * If none of these special case are found, then we just check the time and see what is the
648      * proper state for the instance.
649      *
650      * @param context  application context
651      * @param instance to register
652      */
registerInstance(Context context, AlarmInstance instance, boolean updateNextAlarm)653     public static void registerInstance(Context context, AlarmInstance instance,
654             boolean updateNextAlarm) {
655         LogUtils.i("Registering instance: " + instance.mId);
656         final ContentResolver cr = context.getContentResolver();
657         final Alarm alarm = Alarm.getAlarm(cr, instance.mAlarmId);
658         final Calendar currentTime = getCurrentTime();
659         final Calendar alarmTime = instance.getAlarmTime();
660         final Calendar timeoutTime = instance.getTimeout();
661         final Calendar lowNotificationTime = instance.getLowNotificationTime();
662         final Calendar highNotificationTime = instance.getHighNotificationTime();
663         final Calendar missedTTL = instance.getMissedTimeToLive();
664 
665         // Handle special use cases here
666         if (instance.mAlarmState == AlarmInstance.DISMISSED_STATE) {
667             // This should never happen, but add a quick check here
668             LogUtils.e("Alarm Instance is dismissed, but never deleted");
669             deleteInstanceAndUpdateParent(context, instance);
670             return;
671         } else if (instance.mAlarmState == AlarmInstance.FIRED_STATE) {
672             // Keep alarm firing, unless it should be timed out
673             boolean hasTimeout = timeoutTime != null && currentTime.after(timeoutTime);
674             if (!hasTimeout) {
675                 setFiredState(context, instance);
676                 return;
677             }
678         } else if (instance.mAlarmState == AlarmInstance.MISSED_STATE) {
679             if (currentTime.before(alarmTime)) {
680                 if (instance.mAlarmId == null) {
681                     LogUtils.i("Cannot restore missed instance for one-time alarm");
682                     // This instance parent got deleted (ie. deleteAfterUse), so
683                     // we should not re-activate it.-
684                     deleteInstanceAndUpdateParent(context, instance);
685                     return;
686                 }
687 
688                 // TODO: This will re-activate missed snoozed alarms, but will
689                 // use our normal notifications. This is not ideal, but very rare use-case.
690                 // We should look into fixing this in the future.
691 
692                 // Make sure we re-enable the parent alarm of the instance
693                 // because it will get activated by by the below code
694                 alarm.enabled = true;
695                 Alarm.updateAlarm(cr, alarm);
696             }
697         } else if (instance.mAlarmState == AlarmInstance.PREDISMISSED_STATE) {
698             if (currentTime.before(alarmTime)) {
699                 setPreDismissState(context, instance);
700             } else {
701                 deleteInstanceAndUpdateParent(context, instance);
702             }
703             return;
704         }
705 
706         // Fix states that are time sensitive
707         if (currentTime.after(missedTTL)) {
708             // Alarm is so old, just dismiss it
709             deleteInstanceAndUpdateParent(context, instance);
710         } else if (currentTime.after(alarmTime)) {
711             // There is a chance that the TIME_SET occurred right when the alarm should go off, so
712             // we need to add a check to see if we should fire the alarm instead of marking it
713             // missed.
714             Calendar alarmBuffer = Calendar.getInstance();
715             alarmBuffer.setTime(alarmTime.getTime());
716             alarmBuffer.add(Calendar.SECOND, ALARM_FIRE_BUFFER);
717             if (currentTime.before(alarmBuffer)) {
718                 setFiredState(context, instance);
719             } else {
720                 setMissedState(context, instance);
721             }
722         } else if (instance.mAlarmState == AlarmInstance.SNOOZE_STATE) {
723             // We only want to display snooze notification and not update the time,
724             // so handle showing the notification directly
725             AlarmNotifications.showSnoozeNotification(context, instance);
726             scheduleInstanceStateChange(context, instance.getAlarmTime(),
727                     instance, AlarmInstance.FIRED_STATE);
728         } else if (currentTime.after(highNotificationTime)) {
729             setHighNotificationState(context, instance);
730         } else if (currentTime.after(lowNotificationTime)) {
731             // Only show low notification if it wasn't hidden in the past
732             if (instance.mAlarmState == AlarmInstance.HIDE_NOTIFICATION_STATE) {
733                 setHideNotificationState(context, instance);
734             } else {
735                 setLowNotificationState(context, instance);
736             }
737         } else {
738             // Alarm is still active, so initialize as a silent alarm
739             setSilentState(context, instance);
740         }
741 
742         // The caller prefers to handle updateNextAlarm for optimization
743         if (updateNextAlarm) {
744             updateNextAlarm(context);
745         }
746     }
747 
748     /**
749      * This will delete and unregister all instances associated with alarmId, without affect
750      * the alarm itself. This should be used whenever modifying or deleting an alarm.
751      *
752      * @param context application context
753      * @param alarmId to find instances to delete.
754      */
deleteAllInstances(Context context, long alarmId)755     public static void deleteAllInstances(Context context, long alarmId) {
756         LogUtils.i("Deleting all instances of alarm: " + alarmId);
757         ContentResolver cr = context.getContentResolver();
758         List<AlarmInstance> instances = AlarmInstance.getInstancesByAlarmId(cr, alarmId);
759         for (AlarmInstance instance : instances) {
760             unregisterInstance(context, instance);
761             AlarmInstance.deleteInstance(context.getContentResolver(), instance.mId);
762         }
763         updateNextAlarm(context);
764     }
765 
766     /**
767      * Delete and unregister all instances unless they are snoozed. This is used whenever an alarm
768      * is modified superficially (label, vibrate, or ringtone change).
769      */
deleteNonSnoozeInstances(Context context, long alarmId)770     public static void deleteNonSnoozeInstances(Context context, long alarmId) {
771         LogUtils.i("Deleting all non-snooze instances of alarm: " + alarmId);
772         ContentResolver cr = context.getContentResolver();
773         List<AlarmInstance> instances = AlarmInstance.getInstancesByAlarmId(cr, alarmId);
774         for (AlarmInstance instance : instances) {
775             if (instance.mAlarmState == AlarmInstance.SNOOZE_STATE) {
776                 continue;
777             }
778             unregisterInstance(context, instance);
779             AlarmInstance.deleteInstance(context.getContentResolver(), instance.mId);
780         }
781         updateNextAlarm(context);
782     }
783 
784     /**
785      * Fix and update all alarm instance when a time change event occurs.
786      *
787      * @param context application context
788      */
fixAlarmInstances(Context context)789     public static void fixAlarmInstances(Context context) {
790         LogUtils.i("Fixing alarm instances");
791         // Register all instances after major time changes or when phone restarts
792         final ContentResolver contentResolver = context.getContentResolver();
793         final Calendar currentTime = getCurrentTime();
794 
795         // Sort the instances in reverse chronological order so that later instances are fixed or
796         // deleted before re-scheduling prior instances (which may re-create or update the later
797         // instances).
798         final List<AlarmInstance> instances = AlarmInstance.getInstances(
799                 contentResolver, null /* selection */);
800         Collections.sort(instances, new Comparator<AlarmInstance>() {
801             @Override
802             public int compare(AlarmInstance lhs, AlarmInstance rhs) {
803                 return rhs.getAlarmTime().compareTo(lhs.getAlarmTime());
804             }
805         });
806 
807         for (AlarmInstance instance : instances) {
808             final Alarm alarm = Alarm.getAlarm(contentResolver, instance.mAlarmId);
809             if (alarm == null) {
810                 unregisterInstance(context, instance);
811                 AlarmInstance.deleteInstance(contentResolver, instance.mId);
812                 LogUtils.e("Found instance without matching alarm; deleting instance %s", instance);
813                 continue;
814             }
815             final Calendar priorAlarmTime = alarm.getPreviousAlarmTime(instance.getAlarmTime());
816             final Calendar missedTTLTime = instance.getMissedTimeToLive();
817             if (currentTime.before(priorAlarmTime) || currentTime.after(missedTTLTime)) {
818                 final Calendar oldAlarmTime = instance.getAlarmTime();
819                 final Calendar newAlarmTime = alarm.getNextAlarmTime(currentTime);
820                 final CharSequence oldTime = DateFormat.format("MM/dd/yyyy hh:mm a", oldAlarmTime);
821                 final CharSequence newTime = DateFormat.format("MM/dd/yyyy hh:mm a", newAlarmTime);
822                 LogUtils.i("A time change has caused an existing alarm scheduled to fire at %s to" +
823                         " be replaced by a new alarm scheduled to fire at %s", oldTime, newTime);
824 
825                 // The time change is so dramatic the AlarmInstance doesn't make any sense;
826                 // remove it and schedule the new appropriate instance.
827                 AlarmStateManager.deleteInstanceAndUpdateParent(context, instance);
828             } else {
829                 registerInstance(context, instance, false /* updateNextAlarm */);
830             }
831         }
832 
833         updateNextAlarm(context);
834     }
835 
836     /**
837      * Utility method to set alarm instance state via constants.
838      *
839      * @param context  application context
840      * @param instance to change state on
841      * @param state    to change to
842      */
setAlarmState(Context context, AlarmInstance instance, int state)843     private static void setAlarmState(Context context, AlarmInstance instance, int state) {
844         if (instance == null) {
845             LogUtils.e("Null alarm instance while setting state to %d", state);
846             return;
847         }
848         switch (state) {
849             case AlarmInstance.SILENT_STATE:
850                 setSilentState(context, instance);
851                 break;
852             case AlarmInstance.LOW_NOTIFICATION_STATE:
853                 setLowNotificationState(context, instance);
854                 break;
855             case AlarmInstance.HIDE_NOTIFICATION_STATE:
856                 setHideNotificationState(context, instance);
857                 break;
858             case AlarmInstance.HIGH_NOTIFICATION_STATE:
859                 setHighNotificationState(context, instance);
860                 break;
861             case AlarmInstance.FIRED_STATE:
862                 setFiredState(context, instance);
863                 break;
864             case AlarmInstance.SNOOZE_STATE:
865                 setSnoozeState(context, instance, true /* showToast */);
866                 break;
867             case AlarmInstance.MISSED_STATE:
868                 setMissedState(context, instance);
869                 break;
870             case AlarmInstance.PREDISMISSED_STATE:
871                 setPreDismissState(context, instance);
872                 break;
873             case AlarmInstance.DISMISSED_STATE:
874                 deleteInstanceAndUpdateParent(context, instance);
875                 break;
876             default:
877                 LogUtils.e("Trying to change to unknown alarm state: " + state);
878         }
879     }
880 
881     @Override
onReceive(final Context context, final Intent intent)882     public void onReceive(final Context context, final Intent intent) {
883         if (INDICATOR_ACTION.equals(intent.getAction())) {
884             return;
885         }
886 
887         final PendingResult result = goAsync();
888         final PowerManager.WakeLock wl = AlarmAlertWakeLock.createPartialWakeLock(context);
889         wl.acquire();
890         AsyncHandler.post(new Runnable() {
891             @Override
892             public void run() {
893                 handleIntent(context, intent);
894                 result.finish();
895                 wl.release();
896             }
897         });
898     }
899 
handleIntent(Context context, Intent intent)900     public static void handleIntent(Context context, Intent intent) {
901         final String action = intent.getAction();
902         LogUtils.v("AlarmStateManager received intent " + intent);
903         if (CHANGE_STATE_ACTION.equals(action)) {
904             Uri uri = intent.getData();
905             AlarmInstance instance = AlarmInstance.getInstance(context.getContentResolver(),
906                     AlarmInstance.getId(uri));
907             if (instance == null) {
908                 LogUtils.e("Can not change state for unknown instance: " + uri);
909                 return;
910             }
911 
912             int globalId = DataModel.getDataModel().getGlobalIntentId();
913             int intentId = intent.getIntExtra(ALARM_GLOBAL_ID_EXTRA, -1);
914             int alarmState = intent.getIntExtra(ALARM_STATE_EXTRA, -1);
915             if (intentId != globalId) {
916                 LogUtils.i("IntentId: " + intentId + " GlobalId: " + globalId + " AlarmState: " +
917                         alarmState);
918                 // Allows dismiss/snooze requests to go through
919                 if (!intent.hasCategory(ALARM_DISMISS_TAG) &&
920                         !intent.hasCategory(ALARM_SNOOZE_TAG)) {
921                     LogUtils.i("Ignoring old Intent");
922                     return;
923                 }
924             }
925 
926             if (intent.getBooleanExtra(FROM_NOTIFICATION_EXTRA, false)) {
927                 if (intent.hasCategory(ALARM_DISMISS_TAG)) {
928                     Events.sendAlarmEvent(R.string.action_dismiss, R.string.label_notification);
929                 } else if (intent.hasCategory(ALARM_SNOOZE_TAG)) {
930                     Events.sendAlarmEvent(R.string.action_snooze, R.string.label_notification);
931                 }
932             }
933 
934             if (alarmState >= 0) {
935                 setAlarmState(context, instance, alarmState);
936             } else {
937                 registerInstance(context, instance, true);
938             }
939         } else if (SHOW_AND_DISMISS_ALARM_ACTION.equals(action)) {
940             Uri uri = intent.getData();
941             AlarmInstance instance = AlarmInstance.getInstance(context.getContentResolver(),
942                     AlarmInstance.getId(uri));
943 
944             if (instance == null) {
945                 LogUtils.e("Null alarminstance for SHOW_AND_DISMISS");
946                 // dismiss the notification
947                 final int id = intent.getIntExtra(AlarmNotifications.EXTRA_NOTIFICATION_ID, -1);
948                 if (id != -1) {
949                     NotificationManagerCompat.from(context).cancel(id);
950                 }
951                 return;
952             }
953 
954             long alarmId = instance.mAlarmId == null ? Alarm.INVALID_ID : instance.mAlarmId;
955             final Intent viewAlarmIntent = Alarm.createIntent(context, DeskClock.class, alarmId)
956                     .putExtra(AlarmClockFragment.SCROLL_TO_ALARM_INTENT_EXTRA, alarmId)
957                     .addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
958 
959             // Open DeskClock which is now positioned on the alarms tab.
960             context.startActivity(viewAlarmIntent);
961 
962             deleteInstanceAndUpdateParent(context, instance);
963         }
964     }
965 
966     /**
967      * Creates an intent that can be used to set an AlarmManager alarm to set the next alarm
968      * indicators.
969      */
createIndicatorIntent(Context context)970     public static Intent createIndicatorIntent(Context context) {
971         return new Intent(context, AlarmStateManager.class).setAction(INDICATOR_ACTION);
972     }
973 
974     /**
975      * Abstract away how the current time is computed. If no implementation of this interface is
976      * given the default is to return {@link Calendar#getInstance()}. Otherwise, the factory
977      * instance is consulted for the current time.
978      */
979     interface CurrentTimeFactory {
getCurrentTime()980         Calendar getCurrentTime();
981     }
982 
983     /**
984      * Abstracts away how state changes are scheduled. The {@link AlarmManagerStateChangeScheduler}
985      * implementation schedules callbacks within the system AlarmManager. Alternate
986      * implementations, such as test case mocks can subvert this behavior.
987      */
988     interface StateChangeScheduler {
scheduleInstanceStateChange(Context context, Calendar time, AlarmInstance instance, int newState)989         void scheduleInstanceStateChange(Context context, Calendar time,
990                 AlarmInstance instance, int newState);
991 
cancelScheduledInstanceStateChange(Context context, AlarmInstance instance)992         void cancelScheduledInstanceStateChange(Context context, AlarmInstance instance);
993     }
994 
995     /**
996      * Schedules state change callbacks within the AlarmManager.
997      */
998     private static class AlarmManagerStateChangeScheduler implements StateChangeScheduler {
999         @Override
scheduleInstanceStateChange(Context context, Calendar time, AlarmInstance instance, int newState)1000         public void scheduleInstanceStateChange(Context context, Calendar time,
1001                 AlarmInstance instance, int newState) {
1002             final long timeInMillis = time.getTimeInMillis();
1003             LogUtils.i("Scheduling state change %d to instance %d at %s (%d)", newState,
1004                     instance.mId, AlarmUtils.getFormattedTime(context, time), timeInMillis);
1005             final Intent stateChangeIntent =
1006                     createStateChangeIntent(context, ALARM_MANAGER_TAG, instance, newState);
1007             // Treat alarm state change as high priority, use foreground broadcasts
1008             stateChangeIntent.addFlags(Intent.FLAG_RECEIVER_FOREGROUND);
1009             PendingIntent pendingIntent = PendingIntent.getService(context, instance.hashCode(),
1010                     stateChangeIntent, PendingIntent.FLAG_UPDATE_CURRENT);
1011 
1012             final AlarmManager am = (AlarmManager) context.getSystemService(ALARM_SERVICE);
1013             if (Utils.isMOrLater()) {
1014                 // Ensure the alarm fires even if the device is dozing.
1015                 am.setExactAndAllowWhileIdle(AlarmManager.RTC_WAKEUP, timeInMillis, pendingIntent);
1016             } else {
1017                 am.setExact(AlarmManager.RTC_WAKEUP, timeInMillis, pendingIntent);
1018             }
1019         }
1020 
1021         @Override
cancelScheduledInstanceStateChange(Context context, AlarmInstance instance)1022         public void cancelScheduledInstanceStateChange(Context context, AlarmInstance instance) {
1023             LogUtils.v("Canceling instance " + instance.mId + " timers");
1024 
1025             // Create a PendingIntent that will match any one set for this instance
1026             PendingIntent pendingIntent = PendingIntent.getService(context, instance.hashCode(),
1027                     createStateChangeIntent(context, ALARM_MANAGER_TAG, instance, null),
1028                     PendingIntent.FLAG_NO_CREATE);
1029 
1030             if (pendingIntent != null) {
1031                 AlarmManager am = (AlarmManager) context.getSystemService(ALARM_SERVICE);
1032                 am.cancel(pendingIntent);
1033                 pendingIntent.cancel();
1034             }
1035         }
1036     }
1037 }
1038