1.介绍
关机闹钟为Android中默认支持的功能,实现起来则需要满足一定的条件:自动开机、开机后响铃。对于自动开机来说,自动关机可以在应用层通过设置alarm来实现,而自动开机需要底层rtc时钟的支持;开机后检查时间,到点响铃。
2.原理
一般智能手机的硬件架构都是分为RF,BB,AP这三个部分, RF射频部分的功能主要是接收和发射射频,大家不用关注这个。而AP部分就是应用程序处理器,其实也就是CPU模块;BB部分则是基带模块。通常手机关机都是AP模块完全断电,而BB虽然处在关机状态,但实际上闹钟和时间部分并没有断电。而当所设置闹钟到点时BB模块会自动给AP模块上电,这样系统也就能开机并激活闹钟功能了。
3.流程
在闹钟设置一个时间合适且有效响铃的闹钟后,向PowerOffAlarm发送设定关机闹钟广播并传入闹钟时间参数,PowerOffAlarm接收到广播后,根据预设提前开机时间和闹钟时间往rtc中写入时间,并将该时间写入文件中暂存。
设置好闹钟后关机,机器会根据rtc中的alrm_time时间参数来开机,开机后AlarmManagerService会检查闹钟,并在闹钟时间发送广播后弹出响铃界面。
点击关闭闹钟后,会发送取消关机闹钟广播,PowerOffAlarm会对传入时间与暂存的时间对比,如果相同则会取消该关机闹钟。
4.遇到问题及解决方法
4.1预置GMS包后,关机闹钟失效
在对比平台闹钟分析后,发现谷歌闹钟在设置闹钟后并没有向PowerOffAlarm发送广播,但是由于谷歌闹钟无法进行修改。
在分析了平台的闹钟设置闹钟步骤后,发现在设置闹钟后都会调用AlarmManager.setAlarmClock()这个方法,然后再AlarmStateManager的updateNextAlarm方法中发送设置关机闹钟的广播。
谷歌闹钟在设置闹钟也会调用AlarmManager.setAlarmClock()这个方法,所以在该方法中直接发送广播到PowerOffAlarm中。
@@ -615,8 +633,34 @@ public class AlarmManager {
public void setAlarmClock(AlarmClockInfo info, PendingIntent operation) {
setImpl(RTC_WAKEUP, info.getTriggerTime(), WINDOW_EXACT, 0, 0, operation,
null, null, null, null, info);
+ setPowerOffAlarm(mContext,info.getTriggerTime());
}
+
+ /*add by hxj at Aug.6,for PowerOffAlarm*/
+ private void setPowerOffAlarm(Context context, long time) {
+ Log.d(TAG, "setPowerOffAlarm: saveAlarmToPreference and sendBroadcast to setPowerOffAlarm,the time is "+time);
+ DEFAULT_ALARM_TIME = time;
+ Intent intent = new Intent(ACTION_SET_POWEROFF_ALARM);
+ intent.addFlags(Intent.FLAG_RECEIVER_FOREGROUND);
+ intent.setPackage(POWER_OFF_ALARM_PACKAGE);
+ intent.putExtra(TIME, time);
+ context.sendBroadcast(intent);
+ }
+
+ private void cancelPowerOffAlarm(Context context) {
+ //long time = getAlarmFromPreference(context);
+ Log.d(TAG, "cancelPowerOffAlarm: cancel alarm and cancelPowerOffAlarm(mContext),time is "+DEFAULT_ALARM_TIME );
+ Intent intent = new Intent(ACTION_CANCEL_POWEROFF_ALARM);
+ intent.addFlags(Intent.FLAG_RECEIVER_FOREGROUND);
+ intent.putExtra(TIME, DEFAULT_ALARM_TIME);
+ intent.setPackage(POWER_OFF_ALARM_PACKAGE);
+ context.sendBroadcast(intent);
+ DEFAULT_ALARM_TIME = 0L;
+ }
+
+
/** @hide */
@SystemApi
@RequiresPermission(android.Manifest.permission.UPDATE_DEVICE_STATS)
@@ -930,6 +974,7 @@ public class AlarmManager {
try {
mService.remove(operation, null);
+ cancelPowerOffAlarm(mContext);
} catch (RemoteException ex) {
throw ex.rethrowFromSystemServer();
}
4.2 PowerOffAlarm无法接收广播
在分析log后,发现接受该广播需要在应用中添加特定的权限,即"org.codeaurora.permission.POWER_OFF_ALARM"权限,但是由于谷歌闹钟中未添加并且无法修改,所以在PowerOffAlarm中接收管广播处移除该权限。
@@ -29,7 +29,7 @@
android:defaultToDeviceProtectedStorage="true">
<receiver android:name=".PowerOffAlarmBroadcastReceiver"
- android:permission="org.codeaurora.permission.POWER_OFF_ALARM"
+
android:exported="true"
android:directBootAware="true"
android:label="PowerOffAlarmBroadcastReceiver">
4.3 开机时间过长导致开机后闹钟过期
提前开机时间平台预设默认为90000毫秒,项目中开机时间较长,所以增长开机时间,改为150000毫秒。
@@ -42,7 +42,7 @@ public class PowerOffAlarmUtils {
private static final int FAILURE = -1;
- public static final long MS_IN_ONE_MIN = 90000L;
+ public static final long MS_IN_ONE_MIN = 150000L;
private static final long SEC_TO_MS = 1000L;
此外还有其他的问题也会导致关机闹钟的失效,有时间再来记录一下。
网友评论