按计划每周更新一篇技术博文,第二篇:《为Android系统定制重启功能
一、Android系统重启的实现方式 1、广播方式 之前的博文介绍过这种方式《使用广播实现的Android关机及重启》,注意应用要在源码中编译,并且应用需要系统权限。
2、通过init.rc启动系统服务来运行sh文件 a.重启shell文件(system_reboot.sh),放在system/etc/目录下
#!/system/bin/sh  reboot

b.init.rc中加入一个Service的定义
   service system_reboot /system/etc/system_reboot.sh     oneshot     disabled

c.启动系统服务
SystemProperties.set("ctl.start", "system_reboot");


3、直接代码中执行命令行 需要取得root权限
Process proc = Runtime.getRuntime().exec(cmd);

以上三种方法可以实现重启,但现在需要实现长按关机键提示对话框中有重启选项,这就设计修改系统源码,感觉修改源码,是个很难的工作,其实找到关机提示框的代码位置,一切都变得很简单。
二、修改源码定制重启功能 1、跟踪长按Power键相关代码 首先要了解长按电源键弹出对话框的相关源码位置,通过跟踪找到PhoneWindowManager.java中有截断长按电源键的方法,继续跟踪代码找到RunnablemPowerLongPress= new Runnable(),在该runnable里边调用showGlobalActionsDialog()方法便是显示长按power键弹出的关机提示对话框,如果你的机器是原生态的,默认提示“飞行模式”,“静音”和“关机”选项。 showGlobalActionsDialog创建关机对话框的代码在GlobalActions.java里边,主要代码如下:
private AlertDialog createDialog() {……mItems = new ArrayList<Action>();        // first: power off        mItems.add(            new SinglePressAction(                    com.android.internal.R.drawable.ic_lock_power_off,                    R.string.global_action_power_off) {                public void onPress() {                    // shutdown by making sure radio and power are handled accordingly.                    ShutdownThread.shutdown(mContext, true);                }                public boolean showDuringKeyguard() {                    return true;                }                public boolean showBeforeProvisioning() {                    return true;                }            });        // next: airplane mode        mItems.add(mAirplaneModeOn);        // last: silent mode        if (SHOW_SILENT_TOGGLE) {            mItems.add(mSilentModeAction);        }……}


2、重启方法 跟踪源码到此处,需要加入重启部分的源码也是往AlertDialog添加多一项,显示部分的代码直接造这power off即可,需要加入重启逻辑的代码直接修改回调方法onPress里边(上边代码粗体部分),一开始,我尝试了前文中提示的三种方法中的方法一和方法三均行不通,后来也没再进一步排查原因,估计权限问题。 本打算使用方法二再尝试一下,突然想到为何不去看看关机的逻辑代码ShutdownThread.shutdown(),打开ShutdownThread.java,很快就发现源码里边也有reboot的实现方法,如下:
/**     * Request a clean shutdown, waiting for subsystems to clean up their     * state etc.  Must be called from a Looper thread in which its UI     * is shown.     *     * @param context Context used to display the shutdown progress dialog.     * @param reason code to pass to the kernel (e.g. "recovery"), or null.     * @param confirm true if user confirmation is needed before shutting down.     */    public static void reboot(final Context context, String reason, boolean confirm){     ……     }

该方法的使用直接看注释知道,reboot方法的后两个参数解释如下: reason 如果值为是null,正常重启;如果是recovery,系统重启进入recovery mode confirm true显示关机提示框,需要用户【确认】;false不显示提示框,直接关机 3、代码修改 a.在关机提示框中加入重启选项 弄清楚关机相关的源码,直接在GlobalActions.java的createDialog方法中加入如下代码即可:
      ... // next: reboot global_action_reboot        mItems.add(         new SinglePressAction(                    com.android.internal.R.drawable.ic_lock_power_off,                    R.string.global_action_reboot) {                public void onPress() {              // reboot                 ShutdownThread.reboot(mContext,null,false);                              }                public boolean showDuringKeyguard() {                    return true;                }                public boolean showBeforeProvisioning() {                    return true;                }            });...

b.修改重启提示框,以区分关机提示框 因为提示框默认是“关机”,在ShutdownThread.java的beginShutdownSequence(context)方法中代码修改如下:
/*///修改前:…… pd.setTitle(context.getText(com.android.internal.R.string.power_off)); pd.setMessage(context.getText(com.android.internal.R.string.shutdown_progress));……/*///修改后:……if(mReboot){     // reboot progress         pd.setTitle(context.getText(com.android.internal.R.string.global_action_reboot));         pd.setMessage(context.getText(com.android.internal.R.string.reboot_progress));     }else{          // shutdown progress         pd.setTitle(context.getText(com.android.internal.R.string.power_off));         pd.setMessage(context.getText(com.android.internal.R.string.shutdown_progress));     }……//*/




重启功能实现了,但是不能到此结束,定制Android系统的重启功能,添加选项和修改重启提示框时加入了“重启”和“正在重启”的字符串,所以会涉及到在系统资源文件中添加新的字符串,源码中资源文件涉及到的多国语言直接忽略,我只在values/strings.xml和values-zh-rCN/strings.xml两个文件中加入对应的字符串
    <string name="global_action_reboot">重启</string>    <string name="reboot_progress">"正在重启..."</string>


c.最后编译 因为修改涉及到系统资源文件又涉及到policy.jar包,经过多次尝试,正确的编译顺序如下: 步骤1.编译frameworks/base/res,在out/target/product/X设备名X/system/framework/目录下生成framework-res.apk 步骤2.编译frameworks/base/,在o同样目录下生成framework.jar包(不能忽略,不然步骤3编译报错) 步骤3.编译frameworks/base/policy,在同样目录下生成policy.jar包

附录: 本例源码以Android 4.0.4为准(2.3一样),关机相关的源码路径: ics/frameworks/base/policy/src/com/android/internal/policy/impl/目录下: PhoneWindowManager.java GlobalActions.java
ics/frameworks/base/core/java/com/android/internal/app/目录下: ShutdownThread.java

PS:4.1.1 关机相关方法位置有些变化 源码已经把关机的接口整合到: public class WindowManagerService extends IWindowManager.Stub
implements Watchdog.Monitor, WindowManagerPolicy.WindowManagerFuncs{...}中
不过可以直接使用: import com.android.server.pm.ShutdownThread;

ShutdownThread.shutdown(mContext, true);
ShutdownThread.rebootSafeMode(mContext, true);

文件路径: Android4.1.1/jb/frameworks/base/services/java/com/android/server/pm/
ShutdownThread.java

Android4.1.1/jb/frameworks/base/services/java/com/android/server/wm
WindowManagerService.java

...
参考引用: 1、Android应用程序获得root权限

更多相关文章

  1. Android(安卓)ActionBar的源代码分析(二)
  2. Android5.1系统通过包名给应用开放系统权限的方法
  3. Android(安卓)SQLite使用入门
  4. Android中执行java命令的方法及java代码执行并解析shell命令
  5. Android(安卓)MP4取得播放时长的方法
  6. Android(安卓)AOP(二):AspectJ在Android中实现Aop
  7. 写给Android开发者的ThreadLocal介绍
  8. 浅谈Java中Collections.sort对List排序的两种方法
  9. Python list sort方法的具体使用

随机推荐

  1. Android如何在service中弹出对话框
  2. Android(安卓)Application Task Activiti
  3. Android中LinearLayout布局的常用属性总
  4. ArcGIS for Android地图控件的5大常见操
  5. Android 监听home键(android:launchMode="
  6. 下载android sdk, adt
  7. 【Android-002】【常见布局】
  8. Android Edittext焦点处理;
  9. Android超链接去下划线--Android学习笔记
  10. Android学习笔记(9)——ListView简介