1 /*
2  * Copyright (C) 2020 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 
17 package com.android.deskclock.alarms
18 
19 import android.app.Service
20 import android.content.BroadcastReceiver
21 import android.content.ContentResolver
22 import android.content.Context
23 import android.content.Intent
24 import android.content.IntentFilter
25 import android.os.Binder
26 import android.os.IBinder
27 import android.telephony.PhoneStateListener
28 import android.telephony.TelephonyManager
29 
30 import com.android.deskclock.AlarmAlertWakeLock
31 import com.android.deskclock.LogUtils
32 import com.android.deskclock.R
33 import com.android.deskclock.events.Events
34 import com.android.deskclock.provider.AlarmInstance
35 import com.android.deskclock.provider.ClockContract.InstancesColumns
36 
37 /**
38  * This service is in charge of starting/stopping the alarm. It will bring up and manage the
39  * [AlarmActivity] as well as [AlarmKlaxon].
40  *
41  * Registers a broadcast receiver to listen for snooze/dismiss intents. The broadcast receiver
42  * exits early if AlarmActivity is bound to prevent double-processing of the snooze/dismiss intents.
43  */
44 class AlarmService : Service() {
45     /** Binder given to AlarmActivity.  */
46     private val mBinder: IBinder = Binder()
47 
48     /** Whether the service is currently bound to AlarmActivity  */
49     private var mIsBound = false
50 
51     /** Listener for changes in phone state.  */
52     private val mPhoneStateListener = PhoneStateChangeListener()
53 
54     /** Whether the receiver is currently registered  */
55     private var mIsRegistered = false
56 
onBindnull57     override fun onBind(intent: Intent?): IBinder {
58         mIsBound = true
59         return mBinder
60     }
61 
onUnbindnull62     override fun onUnbind(intent: Intent?): Boolean {
63         mIsBound = false
64         return super.onUnbind(intent)
65     }
66 
67     private lateinit var mTelephonyManager: TelephonyManager
68     private var mCurrentAlarm: AlarmInstance? = null
69 
startAlarmnull70     private fun startAlarm(instance: AlarmInstance) {
71         LogUtils.v("AlarmService.start with instance: " + instance.mId)
72         if (mCurrentAlarm != null) {
73             AlarmStateManager.setMissedState(this, mCurrentAlarm!!)
74             stopCurrentAlarm()
75         }
76 
77         AlarmAlertWakeLock.acquireCpuWakeLock(this)
78 
79         mCurrentAlarm = instance
80         AlarmNotifications.showAlarmNotification(this, mCurrentAlarm!!)
81         mTelephonyManager.listen(mPhoneStateListener.init(), PhoneStateListener.LISTEN_CALL_STATE)
82         AlarmKlaxon.start(this, mCurrentAlarm!!)
83         sendBroadcast(Intent(ALARM_ALERT_ACTION))
84     }
85 
stopCurrentAlarmnull86     private fun stopCurrentAlarm() {
87         if (mCurrentAlarm == null) {
88             LogUtils.v("There is no current alarm to stop")
89             return
90         }
91 
92         val instanceId = mCurrentAlarm!!.mId
93         LogUtils.v("AlarmService.stop with instance: %s", instanceId)
94 
95         AlarmKlaxon.stop(this)
96         mTelephonyManager.listen(mPhoneStateListener, PhoneStateListener.LISTEN_NONE)
97         sendBroadcast(Intent(ALARM_DONE_ACTION))
98 
99         stopForeground(true /* removeNotification */)
100 
101         mCurrentAlarm = null
102         AlarmAlertWakeLock.releaseCpuLock()
103     }
104 
105     private val mActionsReceiver: BroadcastReceiver = object : BroadcastReceiver() {
onReceivenull106         override fun onReceive(context: Context, intent: Intent) {
107             val action: String? = intent.getAction()
108             LogUtils.i("AlarmService received intent %s", action)
109             if (mCurrentAlarm == null ||
110                     mCurrentAlarm!!.mAlarmState != InstancesColumns.FIRED_STATE) {
111                 LogUtils.i("No valid firing alarm")
112                 return
113             }
114 
115             if (mIsBound) {
116                 LogUtils.i("AlarmActivity bound; AlarmService no-op")
117                 return
118             }
119 
120             when (action) {
121                 ALARM_SNOOZE_ACTION -> {
122                     // Set the alarm state to snoozed.
123                     // If this broadcast receiver is handling the snooze intent then AlarmActivity
124                     // must not be showing, so always show snooze toast.
125                     AlarmStateManager.setSnoozeState(context, mCurrentAlarm!!, true /* showToast */)
126                     Events.sendAlarmEvent(R.string.action_snooze, R.string.label_intent)
127                 }
128                 ALARM_DISMISS_ACTION -> {
129                     // Set the alarm state to dismissed.
130                     AlarmStateManager.deleteInstanceAndUpdateParent(context, mCurrentAlarm!!)
131                     Events.sendAlarmEvent(R.string.action_dismiss, R.string.label_intent)
132                 }
133             }
134         }
135     }
136 
onCreatenull137     override fun onCreate() {
138         super.onCreate()
139         mTelephonyManager = getSystemService(Context.TELEPHONY_SERVICE) as TelephonyManager
140 
141         // Register the broadcast receiver
142         val filter = IntentFilter(ALARM_SNOOZE_ACTION)
143         filter.addAction(ALARM_DISMISS_ACTION)
144         registerReceiver(mActionsReceiver, filter, Context.RECEIVER_EXPORTED)
145         mIsRegistered = true
146     }
147 
onStartCommandnull148     override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
149         LogUtils.v("AlarmService.onStartCommand() with %s", intent)
150         if (intent == null) {
151             return Service.START_NOT_STICKY
152         }
153 
154         val instanceId = AlarmInstance.getId(intent.getData()!!)
155         when (intent.getAction()) {
156             AlarmStateManager.CHANGE_STATE_ACTION -> {
157                 AlarmStateManager.handleIntent(this, intent)
158 
159                 // If state is changed to firing, actually fire the alarm!
160                 val alarmState: Int = intent.getIntExtra(AlarmStateManager.ALARM_STATE_EXTRA, -1)
161                 if (alarmState == InstancesColumns.FIRED_STATE) {
162                     val cr: ContentResolver = this.getContentResolver()
163                     val instance: AlarmInstance? = AlarmInstance.getInstance(cr, instanceId)
164                     if (instance == null) {
165                         LogUtils.e("No instance found to start alarm: %d", instanceId)
166                         if (mCurrentAlarm != null) {
167                             // Only release lock if we are not firing alarm
168                             AlarmAlertWakeLock.releaseCpuLock()
169                         }
170                     } else if (mCurrentAlarm != null && mCurrentAlarm!!.mId == instanceId) {
171                         LogUtils.e("Alarm already started for instance: %d", instanceId)
172                     } else {
173                         startAlarm(instance)
174                     }
175                 }
176             }
177             STOP_ALARM_ACTION -> {
178                 if (mCurrentAlarm != null && mCurrentAlarm!!.mId != instanceId) {
179                     LogUtils.e("Can't stop alarm for instance: %d because current alarm is: %d",
180                             instanceId, mCurrentAlarm!!.mId)
181                 } else {
182                     stopCurrentAlarm()
183                     stopSelf()
184                 }
185             }
186         }
187 
188         return Service.START_NOT_STICKY
189     }
190 
onDestroynull191     override fun onDestroy() {
192         LogUtils.v("AlarmService.onDestroy() called")
193         super.onDestroy()
194         if (mCurrentAlarm != null) {
195             stopCurrentAlarm()
196         }
197 
198         if (mIsRegistered) {
199             unregisterReceiver(mActionsReceiver)
200             mIsRegistered = false
201         }
202     }
203 
204     private inner class PhoneStateChangeListener : PhoneStateListener() {
205         private var mPhoneCallState = 0
206 
initnull207         fun init(): PhoneStateChangeListener {
208             mPhoneCallState = -1
209             return this
210         }
211 
onCallStateChangednull212         override fun onCallStateChanged(state: Int, ignored: String?) {
213             if (mPhoneCallState == -1) {
214                 mPhoneCallState = state
215             }
216 
217             if (state != TelephonyManager.CALL_STATE_IDLE && state != mPhoneCallState) {
218                 startService(AlarmStateManager.createStateChangeIntent(this@AlarmService,
219                         "AlarmService", mCurrentAlarm!!, InstancesColumns.MISSED_STATE))
220             }
221         }
222     }
223 
224     companion object {
225         /**
226          * AlarmActivity and AlarmService (when unbound) listen for this broadcast intent
227          * so that other applications can snooze the alarm (after ALARM_ALERT_ACTION and before
228          * ALARM_DONE_ACTION).
229          */
230         const val ALARM_SNOOZE_ACTION = "com.android.deskclock.ALARM_SNOOZE"
231 
232         /**
233          * AlarmActivity and AlarmService listen for this broadcast intent so that other
234          * applications can dismiss the alarm (after ALARM_ALERT_ACTION and before ALARM_DONE_ACTION).
235          */
236         const val ALARM_DISMISS_ACTION = "com.android.deskclock.ALARM_DISMISS"
237 
238         /** A public action sent by AlarmService when the alarm has started.  */
239         const val ALARM_ALERT_ACTION = "com.android.deskclock.ALARM_ALERT"
240 
241         /** A public action sent by AlarmService when the alarm has stopped for any reason.  */
242         const val ALARM_DONE_ACTION = "com.android.deskclock.ALARM_DONE"
243 
244         /** Private action used to stop an alarm with this service.  */
245         const val STOP_ALARM_ACTION = "STOP_ALARM"
246 
247         /**
248          * Utility method to help stop an alarm properly. Nothing will happen, if alarm is not firing
249          * or using a different instance.
250          *
251          * @param context application context
252          * @param instance you are trying to stop
253          */
254         @JvmStatic
stopAlarmnull255         fun stopAlarm(context: Context, instance: AlarmInstance) {
256             val intent: Intent =
257                     AlarmInstance.createIntent(context, AlarmService::class.java, instance.mId)
258                             .setAction(STOP_ALARM_ACTION)
259 
260             // We don't need a wake lock here, since we are trying to kill an alarm
261             context.startService(intent)
262         }
263     }
264 }