对于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. 为Android系统定制重启功能

    按计划每周更新一篇技术博文,第二篇:<为Android系统定制重启功能> 一.Android系统重启的实现方式 1.广播方式 之前的博文介绍过这种方式<使用广播实现的Android关 ...

  2. Android 电量监控、关机、重启功能的实现

    本文主要是介绍Android电量监控.关机.重启功能的实现,需要具备的条件是手机需要root过,才能实现相关操作. 1.MainActivity.java import java.io.IOExcep ...

  3. Django by Example·第二章|Enhancing Your Blog with Advanced Features(为博客系统添加高级功能)@笔记

    Django by Example·第二章|Enhancing Your Blog with Advanced Features(为博客系统添加高级功能)@笔记 这本书的结构确实很不错,如果能够坚持看 ...

  4. Android 5.1长按电源键添加重启功能

    原址:http://blog.csdn.net/zhoumushui 现在长按Power键只有一个关机键,需要添加一个重启,以下是我的添加步骤: 1.在frameworks/base/core/res ...

  5. HEXO博客系统添加搜索功能

    Hexo 可以通过接入第三方搜索引擎来为博客添加搜索功能.这里给出一种基于 Algolia 搜索引擎的实现方式. 流程 Algolia 是一家搜索解决方案提供商,提供强大的搜索服务和 API 接口.在 ...

  6. android 4.0.4系统添加波斯语

    1 确定好语言和国家对照表,波斯与对照如下: fa_IR 2 根据build\target\product\sdk.mk $(call inherit-product-if-exists, frame ...

  7. android 9.0 屏蔽系统所有通知功能实现

    目录 1.概述 2.屏蔽系统所有通知功能实现的核心类 3.屏蔽系统所有通知功能实现的核心功能实现和分析

  8. 机房收费系统-添加密保功能

    起因 之前曾经和小伙伴讨论过关于机房收费系统的安全性的问题,如果忘了密码怎么办? 于是脑洞就来了,就联想到了QQ的密保功能!于是就开始干了 思路 密保用来找回密码,所以需要在数据库user_info中 ...

  9. Android Q及以上系统音频捕获功能(声音内录)的简单实现

    前言 现在越来越多的视频类APP,如抖音.快手.B站等等,都开放了音频捕获配置,也就是android:allowAudioPlaybackCapture="true".因此学习如何 ...

最新文章

  1. 【nginx】从主页搭建看nginx常用配置
  2. javaScript中变量作用域
  3. ubuntu shell实现99乘法表
  4. linux添加三权,基于SELinux的三权分离技术的研究
  5. 蓝桥杯 ALGO-46 算法训练 Hanoi问题
  6. DNS解释问题:java.net.UnknownHostException
  7. java swt 双屏_SWT(JFace)体验之打开多个Form
  8. 《JAVA程序设计教程 (第二版)雍俊海编著》pdf 附下载链接
  9. cuda和cudnn下载地址
  10. Python 搜狗词库的批量下载
  11. 魔百盒九联UNT402H,(芒果、南传、百视通)等通刷刷机固件
  12. 由于找不到 MSVCR120.dll,无法继续执行代码终极解决方法
  13. python-opencv标定相机内参(针孔+鱼目)
  14. bigdecimal 平均数_用Java计算平均值
  15. 全国计算机四级之网络工程师知识点(四)
  16. ArrayList删除某元素的几种方法
  17. PD866EZ-12D/YCZ多用户预付费电表 上传至西安市能耗平台
  18. STM32、NBIOT、Lora模块烧写方法-Hex文件烧录步骤详解-新大陆物联网设备-NEWLab开发板
  19. Android 系统编译技巧
  20. 【Appium】基于 Appium 的 iOS 自动化

热门文章

  1. 043_Card卡片
  2. 035_jQuery Ajax的ajaxSetup方法
  3. 087_html5表单元素
  4. 008_logback配置语法
  5. 福建高职单招计算机基础知识,福建高职单招计算机类试题及标准答案
  6. 从Linux上查看接口路由IP(吓一跳ip)
  7. 第六章 ORacle权限设置、schema、通过profile 限制用户恶意登录、通过profile限制使用期限
  8. Unity各个版本的离线文档下载和配置方法
  9. 向量空间和计算机科学与技术,向量空间
  10. 设计一个名为complex的类来表示复数_天线波束设计