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;
此外还有其他的问题也会导致关机闹钟的失效,有时间再来记录一下。
总结
到此这篇关于Android平台预置GMS包后,关机闹钟失效的文章就介绍到这了,更多相关android 预置GMS包关机闹钟失效内容请搜索自由互联以前的文章或继续浏览下面的相关文章希望大家以后多多支持自由互联!