Android 4.1.2系统添加重启功能

时间:2022-05-30 21:17:29

对于Android的的手机或者平板长期使用,感觉会出现慢的情况,所以偶尔还是需要重启一下,而长按电源键弹出的菜单又没有重启选项,所以特在此记录自己添加这个功能的过程。

首先关机的那个弹出菜单是在frameworks/base/policy/src/com/android/internal/policy/impl/GlobalActions.java这个文件中创建的:

点击(此处)折叠或打开

  1. /**
  2. * Create the global actions dialog.
  3. * @return A new dialog.
  4. */
  5. private GlobalActionsDialog createDialog() {
  6. // Simple toggle style if there's no vibrator, otherwise use a tri-state
  7. if (!mHasVibrator) {
  8. mSilentModeAction = new SilentModeToggleAction();
  9. } else {
  10. mSilentModeAction = new SilentModeTriStateAction(mContext, mAudioManager, mHandler);
  11. }
  12. mAirplaneModeOn = new ToggleAction(
  13. R.drawable.ic_lock_airplane_mode,
  14. R.drawable.ic_lock_airplane_mode_off,
  15. R.string.global_actions_toggle_airplane_mode,
  16. R.string.global_actions_airplane_mode_on_status,
  17. R.string.global_actions_airplane_mode_off_status) {
  18. void onToggle(boolean on) {
  19. if (mHasTelephony && Boolean.parseBoolean(
  20. SystemProperties.get(TelephonyProperties.PROPERTY_INECM_MODE))) {
  21. mIsWaitingForEcmExit = true;
  22. // Launch ECM exit dialog
  23. Intent ecmDialogIntent =
  24. new Intent(TelephonyIntents.ACTION_SHOW_NOTICE_ECM_BLOCK_OTHERS, null);
  25. ecmDialogIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  26. mContext.startActivity(ecmDialogIntent);
  27. } else {
  28. changeAirplaneModeSystemSetting(on);
  29. }
  30. }
  31. @Override
  32. protected void changeStateFromPress(boolean buttonOn) {
  33. if (!mHasTelephony) return;
  34. // In ECM mode airplane state cannot be changed
  35. if (!(Boolean.parseBoolean(
  36. SystemProperties.get(TelephonyProperties.PROPERTY_INECM_MODE)))) {
  37. mState = buttonOn ? State.TurningOn : State.TurningOff;
  38. mAirplaneState = mState;
  39. }
  40. }
  41. public boolean showDuringKeyguard() {
  42. return true;
  43. }
  44. public boolean showBeforeProvisioning() {
  45. return false;
  46. }
  47. };
  48. onAirplaneModeChanged();
  49. mItems = new ArrayList<Action>();
  50. // first: power off
  51. mItems.add(
  52. new SinglePressAction(
  53. com.android.internal.R.drawable.ic_lock_power_off,
  54. R.string.global_action_power_off) {
  55. public void onPress() {
  56. // shutdown by making sure radio and power are handled accordingly.
  57. mWindowManagerFuncs.shutdown(true);
  58. }
  59. public boolean onLongPress() {
  60. mWindowManagerFuncs.rebootSafeMode(true);
  61. return true;
  62. }
  63. public boolean showDuringKeyguard() {
  64. return true;
  65. }
  66. public boolean showBeforeProvisioning() {
  67. return true;
  68. }
  69. });

我们可以看到mItems.add函数是添加一个选项,该菜单的第一个选项就是关机选项,我们可以在此之后添加重启选项,代码如下:

  1. mItems.add(
  2. new SinglePressAction(
  3. com.android.internal.R.drawable.ic_lock_power_off,
  4. R.string.global_action_reboot) {
  5. public void onPress() {
  6. // reboot
  7. mWindowManagerFuncs.reboot();
  8. }
  9. public boolean showDuringKeyguard() {
  10. return true;
  11. }
  12. public boolean showBeforeProvisioning() {
  13. return true;
  14. }
  15. });

上面的代码中使用了mWindowManagerFuncs.reboot函数和R.string.global_action_reboot资源,因此我们需要该资源并实现reboot函数。

首先在frameworks/base/core/java/android/view/WindowManagerPolicy.java中添加reboot接口:

  1. /**
  2. * Interface for calling back in to the window manager that is private
  3. * between it and the policy.
  4. */
  5. public interface WindowManagerFuncs {
  6. ...
  7. /**
  8. * Switch the keyboard layout for the given device.
  9. * Direction should be +1 or -1 to go to the next or previous keyboard layout.
  10. */
  11. public void switchKeyboardLayout(int deviceId, int direction);
  12. public void shutdown();
  13. public void reboot();
  14. public void rebootSafeMode();
  15. }

然后在frameworks/base/services/java/com/android/server/wm/WindowManagerService.java中实现该接口:

  1. // Called by window manager policy. Not exposed externally.
  2. @Override
  3. public void shutdown() {
  4. ShutdownThread.shutdown(mContext, true);
  5. }
  6. // Called by window manager policy. Not exposed externally.
  7. @Override
  8. public void reboot() {
  9. ShutdownThread.reboot(mContext, null, true);
  10. }
  11. // Called by window manager policy. Not exposed externally.
  12. @Override
  13. public void rebootSafeMode() {
  14. ShutdownThread.rebootSafeMode(mContext, true);
  15. }

接下来,为了在按下重启选项之后,能出现”重启“之类的提示,还需要修改frameworks/base/services/java/com/android/server/pm/ShutdownThread.java中的shutdownInner函数和beginShutdownSequence函数:

  1. static void shutdownInner(final Context context, boolean confirm) {
  2. // ensure that only one thread is trying to power down.
  3. // any additional calls are just returned
  4. synchronized (sIsStartedGuard) {
  5. if (sIsStarted) {
  6. Log.d(TAG, "Request
    to shutdown already running, returning.");
  7. return;
  8. }
  9. }
  10. final int longPressBehavior = context.getResources().getInteger(
  11. com.android.internal.R.integer.config_longPressOnPowerBehavior);
  12. final int resourceId = mRebootSafeMode
  13. ? com.android.internal.R.string.reboot_safemode_confirm
  14. : (longPressBehavior == 2
  15. ? com.android.internal.R.string.shutdown_confirm_question
  16. : (mReboot ? com.android.internal.R.string.reboot_confirm :
  17. com.android.internal.R.string.shutdown_confirm));
  18. Log.d(TAG, "Notifying
    thread to start shutdown longPressBehavior=" + longPressBehavior);
  19. if (confirm) {
  20. final CloseDialogReceiver
    closer = new CloseDialogReceiver(context);
  21. final AlertDialog dialog = new AlertDialog.Builder(context)
  22. .setTitle(mRebootSafeMode
  23. ? com.android.internal.R.string.reboot_safemode_title
  24. : (mReboot ? com.android.internal.R.string.reboot :
  25. com.android.internal.R.string.power_off))
  26. .setMessage(resourceId)
  27. .setPositiveButton(com.android.internal.R.string.yes, new DialogInterface.OnClickListener() {
  28. public void onClick(DialogInterface dialog, int which) {
  29. beginShutdownSequence(context);
  30. }
  31. })
  32. .setNegativeButton(com.android.internal.R.string.no, null)
  33. .create();
  34. closer.dialog = dialog;
  35. dialog.setOnDismissListener(closer);
  36. dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_KEYGUARD_DIALOG);
  37. dialog.show();
  38. } else {
  39. beginShutdownSequence(context);
  40. }
  41. }
  42. private static void beginShutdownSequence(Context context) {
  43. synchronized (sIsStartedGuard) {
  44. if (sIsStarted) {
  45. Log.d(TAG, "Shutdown
    sequence already running, returning.");
  46. return;
  47. }
  48. sIsStarted = true;
  49. }
  50. // throw up an indeterminate system dialog to indicate radio is
  51. // shutting down.
  52. ProgressDialog pd = new ProgressDialog(context);
  53. pd.setTitle(context.getText(com.android.internal.R.string.power_off));
  54. pd.setMessage(context.getText(com.android.internal.R.string.shutdown_progress));
  55. pd.setIndeterminate(true);
  56. pd.setCancelable(false);
  57. pd.getWindow().setType(WindowManager.LayoutParams.TYPE_KEYGUARD_DIALOG);
  58. pd.show();
  59. sInstance.mContext = context;
  60. sInstance.mPowerManager = (PowerManager)context.getSystemService(Context.POWER_SERVICE);
  61. // make sure we never fall asleep again
  62. sInstance.mCpuWakeLock = null;
  63. try {
  64. sInstance.mCpuWakeLock = sInstance.mPowerManager.newWakeLock(
  65. PowerManager.PARTIAL_WAKE_LOCK, TAG + "-cpu");
  66. sInstance.mCpuWakeLock.setReferenceCounted(false);
  67. sInstance.mCpuWakeLock.acquire();
  68. } catch (SecurityException e) {
  69. Log.w(TAG, "No
    permission to acquire wake lock", e);
  70. sInstance.mCpuWakeLock = null;
  71. }
  72. // also make sure the screen stays on for better user experience
  73. sInstance.mScreenWakeLock = null;
  74. if (sInstance.mPowerManager.isScreenOn()) {
  75. try {
  76. sInstance.mScreenWakeLock = sInstance.mPowerManager.newWakeLock(
  77. PowerManager.FULL_WAKE_LOCK, TAG + "-screen");
  78. sInstance.mScreenWakeLock.setReferenceCounted(false);
  79. sInstance.mScreenWakeLock.acquire();
  80. } catch (SecurityException e) {
  81. Log.w(TAG, "No
    permission to acquire wake lock", e);
  82. sInstance.mScreenWakeLock = null;
  83. }
  84. }
  85. // start the thread that initiates shutdown
  86. sInstance.mHandler = new Handler() {
  87. };
  88. sInstance.start();
  89. }

至此关于代码部分的改动全部完成,接下来就需要添加使用到的资源了,就是前面用到的字符串。首先需要在frameworks/base/core/res/res/values/strings.xml中添加一下字符串:

  1. <string name="reboot">Reboot</string>
  2. <string name="reboot_progress">Reboot\u2026</string>
  3. <string name="reboot_confirm" product="tablet">Your
    tablet will reboot.</string>
  4. <string name="reboot_confirm" product="default">Your
    phone will reboot.</string>
  5. <!-- label for item that reboot in phone options dialog -->
  6. <string name="global_action_reboot">Reboot</string>

而后需要在frameworks/base/core/res/res/values/public.xml中声明这些资源,否则编译的时候会出现找不到该资源的错误。

  1. <java-symbol type="string" name="reboot" />
  2. <java-symbol type="string" name="reboot_confirm" />
  3. <java-symbol type="string" name="reboot_progress" />
  4. <java-symbol type="string" name="global_action_reboot" />

至此,全部修改完成,编译烧写即可。

Android 4.1.2系统添加重启功能的更多相关文章

  1. android4&period;2添加重启菜单项

    本文主要是针对android4.2关机菜单添加重启功能 A.关机提示 android4.2/frameworks/base/policy/src/com/android/internal/policy ...

  2. Android系统移植与调试之------->如何修改Android设备添加重启、飞行模式、静音模式等功能(二)

    今天要说的是为Android设备添加重启.飞行模式.静音模式按钮,客户需求中需要添加这项功能,在长按电源键弹出的菜单中没有这些选项,谨以此文记录自己添加这个功能的过程. 首先找到长按电源键弹出的对话框 ...

  3. Android调用系统关机与重启功能

    我是在android源码里编译的package/apps/,因为需要调用的关机接口是不对上层开放的,在eclipse里面不能调用. 我主要是介绍调用android的关机功能,因为在调试过程中,关机的一 ...

  4. 探索Android调用系统的分享功能

    非常多的应用为了应用的推广和传播都会使用"分享"的功能,点击分享button.就能将想要分享的内容或者图片分享至QQ空间.微博.微信朋友圈等实现了分享功能的应用.这篇文章主要是为了 ...

  5. 【转】Android Service被关闭后自动重启&comma;解决被异常kill 服务

    http://www.kaifajie.cn/android/10182-2.html 每次调用startService(Intent)的时候,都会调用该Service对象的onStartComman ...

  6. Android6&period;0 源码修改之屏蔽系统短信功能和来电功能

    一.屏蔽系统短信功能 1.屏蔽所有短信 android 4.2 短信发送流程分析可参考这篇 戳这 源码位置 vendor\mediatek\proprietary\packages\apps\Mms\ ...

  7. 如何在本地搭建一个Android应用crashing跟踪系统-ACRA

    https://github.com/bboyfeiyu/android-tech-frontier/tree/master/others/%E5%A6%82%E4%BD%95%E5%9C%A8%E6 ...

  8. 嵌入式系统添加无线wifi模块

    开发环境:fl2440开发板,linux3.0内核,交叉编译器路径/opt/buildroot-2011.11/arm920t/usr/bin/arm-linux-,无线网卡RT3070 平时开发板联 ...

  9. Android端IM应用中的&commat;人功能实现:仿微博、QQ、微信,零入侵、高可扩展

    本文由“猫爸iYao”原创分享,感谢作者. 1.引言 最近有个需求:评论@人(没错,就是IM聊天或者微博APP里的@人功能),就像下图这样:   ▲ 微信群聊界面里的@人功能    ▲ QQ群聊界面里 ...

随机推荐

  1. SGU 131&period;Hardwood floor

    时间限制:0.25s 空间限制:4M 题意: 给出 n*m (1≤n.m≤9)的方格棋盘,用 1*2 的矩形的骨牌和 L 形的(2*2 的 去掉一个角)骨牌不重叠地覆盖,求覆盖满的方案数. Solut ...

  2. 【VB】StrConv函数 vbUnicode用法

    [VB]StrConv函数 StrConv(string, conversion, LCID) vbUnicode 64 根据系统的缺省码页将字符串转成Unicode. vbFromUnicode 1 ...

  3. &lbrack;物理学与PDEs&rsqb;第1章第4节 电磁能量和电磁动量&comma; 能量、动量守恒与转化定律 4&period;3 电磁能量 &lpar;动量&rpar; 密度&comma; 电磁能量流 &lpar;动量流&rpar; 密度

    1. 电磁能量密度: $\cfrac{1}{2}\sex{\ve_0E^2+\cfrac{1}{\mu_0}B^2}$. 2. 电磁能量流密度向量: ${\bf S}=\cfrac{1}{\mu_0} ...

  4. 练就Java24章真经—你所不知道的工厂方法

    前言 最近一直在Java方向奋斗<终于,我还是下决心学Java后台了>,今天抽空开始学习Java的设计模式了.计划有时间就去学习,你这么有时间,还不来一起上车吗? 之所以要学习Java模式 ...

  5. Django框架之第三篇模板语法&lpar;重要!!!&rpar;

    一.什么是模板? 只要是在html里面有模板语法就不是html文件了,这样的文件就叫做模板. 二.模板语法分类 一.模板语法之变量:语法为 {{ }}: 在 Django 模板中遍历复杂数据结构的关键 ...

  6. Docker概念学习系列之Docker的主要目标(2)

    不多说,直接上干货! Docker的主要目标: 见[博主]撰写的https://mp.weixin.qq.com/s/ELYUgMpQOhVvTsNCHQLELg 通过对应用组件的封装.分发.部署.运 ...

  7. linux中的通配符、元字符、转义符

    linux中的通配符.元字符.转义符 linux中的通配符元字符转义符 shell命令的构成 通配符 元字符meta 转义符 example reference shell命令的构成 每条linux命 ...

  8. 补发9&period;27&OpenCurlyDoubleQuote;天天向上”团队Scrum站立会议

    组长:王森 组员:张金生 张政 栾骄阳 时间:2016.09.27 地点:612寝 组员 已完成 未完成 王森 分析设计亮点 原型搭建 张金生 设计UI框架 美化完善 张政 学习C#语言初步应用 熟练 ...

  9. JavaScript 金额、数字、千分位、千分位、保留几位小数、舍入舍去、支持负数

    JavaScript 金额.数字 千分位格式化.保留指定位数小数.支持四舍五入.进一法.去尾法 字段说明: number:需要处理的数字: decimals:保留几位小数,默认两位,可不传: dec_ ...

  10. 持续集成、持续交付、持续部署、Jkens、git

    一  持续集成.持续交付.持续部署 1. 持续集成 持续集成指的是,频繁地(一天多次)将代码集成到主干.持续集成的目的,就是让产品可以快速迭代,同时还能 保持高质量.它的核心措施是,代码集成到主干之前 ...