android源码探索之定制android关机界面的方法

 更新时间:2015年10月08日 11:59:18   作者:pgalxx  
这篇文章主要介绍了android源码探索之定制android关机界面的方法,较为详细的分析了Android关机界面的相关原理与代码实现技巧,具有一定参考借鉴价值,需要的朋友可以参考下

本文实例讲述了android源码探索之定制android关机界面的方法。分享给大家供大家参考。具体如下:

在Android系统中,长按Power键默认会弹出对话框让你选择“飞行模式”,“静音”,“关机”等功能。如下图所示:

但这些功能都对Android-x86和其他终端产品就没什么必要了。本文就简单介绍下如何定制关机界面。

我的目标是长按Power键,将会关机,弹出“设备将要关机”选择对话框。如果可以选择“是”关机,和“否”返回系统。

按照android源码定制要点中提到的,首先你要对整个系统有全面的了解,找到弹出原来这个选择框的代码,它在这里:

<pre name="code" class="java">frameworks\policies\base\phone\com\android\internal\policy\impl\PhoneWindowManager.java 
显示对话框调用的代码如下:

Runnable mPowerLongPress = new Runnable() {
 public void run() {
  mShouldTurnOffOnKeyUp = false;
  performHapticFeedbackLw(null, HapticFeedbackConstants.LONG_PRESS, false);
  sendCloseSystemWindows(SYSTEM_DIALOG_REASON_GLOBAL_ACTIONS);
  showGlobalActionsDialog();
 }
};

调用showGlobalActionsDialog方法之后将会聚到有“飞行模式”、“静音”、“关机”等选项的对话框。

找到这里,我们就知道该做什么了!干掉它,换成我们想要的关机代码,就大功告成了!既然这样,事不宜迟,让我们赶快到showGloabalActionDialog方法中看看关机的部分在哪里!

showGlobalActionsDialog的实现部分在这里:
frameworks\policies\base\phone\com\android\internal\policy\impl\GlobalAction.java

public void showDialog(boolean keyguardShowing, boolean isDeviceProvisioned) {
 mKeyguardShowing = keyguardShowing; 
 mDeviceProvisioned = isDeviceProvisioned; 
 if (mDialog == null) {
  mStatusBar = (StatusBarManager)mContext.getSystemService(Context.STATUS_BAR_SERVICE);
  mDialog = createDialog();
 } 
 prepareDialog(); 
 mStatusBar.disable(StatusBarManager.DISABLE_EXPAND);
 mDialog.show(); 
}

我们可以很清楚的看到,这里新建了一个mDialog,然后prepare接着就show了它,那么,这个mDialog就是关键了,看看它是怎么被createDialog创建出来的吧,仍然在这个文件中:

/** 
 * Create the global actions dialog. 
 * @return A new dialog. 
 */ 
private AlertDialog createDialog() { 
 mSilentModeToggle = new ToggleAction( 
   R.drawable.ic_lock_silent_mode, 
   R.drawable.ic_lock_silent_mode_off, 
   R.string.global_action_toggle_silent_mode, 
   R.string.global_action_silent_mode_on_status, 
   R.string.global_action_silent_mode_off_status) { 
  void willCreate() { 
   // XXX: FIXME: switch to ic_lock_vibrate_mode when available 
   mEnabledIconResId = (Settings.System.getInt(mContext.getContentResolver(), 
     Settings.System.VIBRATE_IN_SILENT, 1) == 1) 
    ? R.drawable.ic_lock_silent_mode_vibrate 
    : R.drawable.ic_lock_silent_mode; 
  } 
  void onToggle(boolean on) {
   if (on) { 
    mAudioManager.setRingerMode((Settings.System.getInt(mContext.getContentResolver(), 
     Settings.System.VIBRATE_IN_SILENT, 1) == 1) 
     ? AudioManager.RINGER_MODE_VIBRATE 
     : AudioManager.RINGER_MODE_SILENT); 
   } else {
    mAudioManager.setRingerMode(AudioManager.RINGER_MODE_NORMAL);
   } 
  } 
  public boolean showDuringKeyguard() {
   return true; 
  } 
  public boolean showBeforeProvisioning() {
   return false; 
  } 
 };
 mAirplaneModeOn = new ToggleAction( 
   R.drawable.ic_lock_airplane_mode, 
   R.drawable.ic_lock_airplane_mode_off, 
   R.string.global_actions_toggle_airplane_mode, 
   R.string.global_actions_airplane_mode_on_status, 
   R.string.global_actions_airplane_mode_off_status) {
  void onToggle(boolean on) { 
   if (Boolean.parseBoolean( 
     SystemProperties.get(TelephonyProperties.PROPERTY_INECM_MODE))) {
    mIsWaitingForEcmExit = true;
    // Launch ECM exit dialog
    Intent ecmDialogIntent = 
      new Intent(TelephonyIntents.ACTION_SHOW_NOTICE_ECM_BLOCK_OTHERS, null); 
    ecmDialogIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    mContext.startActivity(ecmDialogIntent);
   } else {
    changeAirplaneModeSystemSetting(on);
   }
  }
  @Override
  protected void changeStateFromPress(boolean buttonOn) {
   // In ECM mode airplane state cannot be changed
   if (!(Boolean.parseBoolean(
     SystemProperties.get(TelephonyProperties.PROPERTY_INECM_MODE)))) {
    mState = buttonOn ? State.TurningOn : State.TurningOff;
    mAirplaneState = mState;
   }
  }
  public boolean showDuringKeyguard() {
   return true;
  }
  public boolean showBeforeProvisioning() {
   return false;
  }
 };
 <span style="color:#ff0000;">mItems = Lists.newArrayList(
   // silent mode
   mSilentModeToggle,
   // next: airplane mode
   mAirplaneModeOn,
   // last: power off
   new SinglePressAction(
     com.android.internal.R.drawable.ic_lock_power_off,
     R.string.global_action_power_off) {
    </span><span style="color:#3333ff;"><u>public void onPress() {
     // shutdown by making sure radio and power are handled accordingly. 
     ShutdownThread.shutdown(mContext, true);
    }</u></span><span style="color:#ff0000;">
    public boolean showDuringKeyguard() {
     return true;
    }
    public boolean showBeforeProvisioning() {
     return true;
    }</span>
   });
 mAdapter = new MyAdapter();
 final AlertDialog.Builder ab = new AlertDialog.Builder(mContext);
 ab.setAdapter(mAdapter, this)
   .setInverseBackgroundForced(true)
   .setTitle(R.string.global_actions);
 final AlertDialog dialog = ab.create();
 dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_DIALOG);
 if (!mContext.getResources().getBoolean(
   com.android.internal.R.bool.config_sf_slowBlur)) {
  dialog.getWindow().setFlags(WindowManager.LayoutParams.FLAG_BLUR_BEHIND, 
    WindowManager.LayoutParams.FLAG_BLUR_BEHIND);
 }
 dialog.setOnDismissListener(this);
 return dialog;
}

看看我们发现了什么!!蓝色的部分就是关机调用的函数了!!shutdown方法的第二个参数标识是否弹出询问对话框。你可以选择需要(true)或者不需要(false)。这里我保守一点,还是选个true吧,万一不小心按到关机键呢,呵呵。。。

也就是说,只要我们用

复制代码 代码如下:
ShutdownThread.shutdown(mContext, true);

替换掉前面的
复制代码 代码如下:
showGlobalActionsDialog();

就可以大功告成了!还等什么!我们修改
frameworks\policies\base\phone\com\android\internal\policy\impl\PhoneWindowManager.java 
的源代码如下:

Runnable mPowerLongPress = new Runnable() { 
 public void run() { 
  mShouldTurnOffOnKeyUp = false; 
  performHapticFeedbackLw(null, HapticFeedbackConstants.LONG_PRESS, false); 
  sendCloseSystemWindows(SYSTEM_DIALOG_REASON_GLOBAL_ACTIONS); 
  //showGlobalActionsDialog(); 
  ShutdownThread.shutdown(mContext, true); 
 } 
};

好了,大功告成了!!

是不是就这样完了呢?发现编译不过。。。

细节很重要!!

原来ShutdownThread.shutdown(mContext, true)的引用包没加进来!!幸好有gcc。。。

复制代码 代码如下:
import com.android.internal.app.ShutdownThread;

将上面这个包加到

frameworks\policies\base\phone\com\android\internal\policy\impl\PhoneWindowManager.java

中,再次编译,通过,YES!

看看我们的战果吧:

是不是感觉到源码定制的快感和成就感了呢?

这仅仅只是个开始,好戏还在后头呢!!哈哈

希望本文所述对大家的Android程序设计有所帮助。

相关文章

  • 详解Android应用层制作LED指示灯

    详解Android应用层制作LED指示灯

    这篇文章主要介绍了详解Android应用层制作LED指示灯的相关资料,需要的朋友可以参考下
    2017-06-06
  • Android实现下载zip压缩文件并解压的方法(附源码)

    Android实现下载zip压缩文件并解压的方法(附源码)

    这篇文章主要给大家介绍了利用Android实现下载zip压缩文件并解压的方法,文中给出了示例代码并提供了源码下载,需要的朋友可以参考借鉴,下面来一起看看吧。
    2017-02-02
  • Android Widget 桌面组件开发介绍

    Android Widget 桌面组件开发介绍

    本篇文章主要介绍了Android中的 widget 桌面组件开发教程,对AppWidget 框架以及AppWidgetManger类进行详细讲解,开发Android widget 开发的朋友可以参考下
    2016-07-07
  • Android实现ViewFlipper图片动画滑动

    Android实现ViewFlipper图片动画滑动

    这篇文章主要为大家详细介绍了Android实现ViewFlipper图片动画滑动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-05-05
  • Android中的dumpsys命令详解

    Android中的dumpsys命令详解

    本文详细讲解了Android中的dumpsys命令,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-04-04
  • Android Loop机制中Looper与handler详细分析

    Android Loop机制中Looper与handler详细分析

    Handler是Android线程之间的消息机制,主要的作用是将一个任务切换到指定的线程中去执行,准确的说是切换到构成Handler的looper所在的线程中去出处理。本文将详细介绍Android Handler机制和Looper Handler Message关系
    2022-11-11
  • Android开发中WebView的简单使用小结

    Android开发中WebView的简单使用小结

    WebView(网络视图)能加载显示网页,可以将其视为一个浏览器。它使用了WebKit渲染引擎加载显示网页。下面这篇文章给大家总结了Android中WebView的简单使用,有需要的可以参考借鉴。
    2016-09-09
  • android scrollview顶部渐渐消失实现实例详解

    android scrollview顶部渐渐消失实现实例详解

    这篇文章主要为大家介绍了android scrollview顶部渐渐消失实现实例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-11-11
  • Android常用定时器的实现方式

    Android常用定时器的实现方式

    我们在开发中时常需要写一些定时的任务,比如每5秒执行一次,下面这篇文章主要给大家介绍了关于Android常用定时器的实现方式,文中通过示例代码介绍的非常详细,需要的朋友可以参考下
    2022-09-09
  • 退出Android程序时清除所有activity的实现方法

    退出Android程序时清除所有activity的实现方法

    这篇文章主要介绍了退出Android程序时清除所有activity的实现方法,详细分析了Android退出时清除activity的原理与实现技巧,需要的朋友可以参考下
    2016-04-04

最新评论