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程序设计有所帮助。

(0)

相关推荐

  • Android仿苹果关机界面实现代码

    本文实例为大家分享了Android仿苹果关机界面的具体代码,供大家参考,具体内容如下 主class 用来控制viewdialog的显示 package com.android.server.policy; import android.app.AlertDialog; import android.app.StatusBarManager; import android.os.Handler; import android.os.Message; import android.os.Remote

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

    本文实例讲述了android源码探索之定制android关机界面的方法.分享给大家供大家参考.具体如下: 在Android系统中,长按Power键默认会弹出对话框让你选择"飞行模式","静音","关机"等功能.如下图所示: 但这些功能都对Android-x86和其他终端产品就没什么必要了.本文就简单介绍下如何定制关机界面. 我的目标是长按Power键,将会关机,弹出"设备将要关机"选择对话框.如果可以选择"是&quo

  • Android 源码如何编译调试

    android提供的工具链和开发工具比较完善,因此它的开发环境的搭建比较简单,相信许多朋友都已经搭建好环境,并编写了HelloActivity入门程序了.这里先看几个问题: 1.android的文件系统结构是怎样的,我们安装的程序放在那里? 编译android源码之后,在out/target/product/generic一些文件: ramdisk.img.system.img.userdata.img. system. data.root 其中, system.img是由 system打包压缩

  • Android源码解析之截屏事件流程

    今天这篇文章我们主要讲一下Android系统中的截屏事件处理流程.用过android系统手机的同学应该都知道,一般的android手机按下音量减少键和电源按键就会触发截屏事件(国内定制机做个修改的这里就不做考虑了).那么这里的截屏事件是如何触发的呢?触发之后android系统是如何实现截屏操作的呢?带着这两个问题,开始我们的源码阅读流程. 我们知道这里的截屏事件是通过我们的按键操作触发的,所以这里就需要我们从android系统的按键触发模块开始看起,由于我们在不同的App页面,操作音量减少键和电

  • Android源码学习之工厂方法模式应用及优势介绍

    工厂方法模式定义: Define an interface for creating an object, but let subclasses decide which class to instantiate. Factory Method lets a class defer instantiation to subclasses. 定义一个用于创建对象的接口,让子类决定实例化哪一个类.工厂方法使一个类的实例化延迟到其子类. 常用的工厂方法模式结构: 如上图所示(截取自<Head Firs

  • 从Android源码剖析Intent查询匹配的实现

    前言     这篇文章主要是介绍一下Android Intent,并且从Android源码的角度对Intent查询匹配过程进行分析. Intent介绍     Intent的中文是"意图"的意思,而意图是一个非常抽象的概念,那么在Android的编码设计中,如何实例化意图呢?因此Android系统明确指定一个Intent可由两方面属性来衡量. 主要属性:包括Action和Data.其中Action用于表示该Intent所表达的动作意图,Data用于表示该Action所操作的数据.   

  • Windows下获取Android 源码方法的详解

    前言:略!获取源码的原因千千万~~~ 1.安装GIT工具.GIT是林纳斯·托瓦兹大神为了管理器Linux内核开发而创立的分布式版本控制软件.下载地址:http://code.google.com/p/msysgit/一路next将安装进行到底. 2.在磁盘剩余空间较大的磁盘下新建一个文件夹,用于存放源码.我在F盘下:新建了androidsourcecode文件夹. 3.访问Android源码网站,获取你所需要的源码"下载链接".网站地址:http://android.git.kerne

  • Ubuntu Android源码以及内核下载与编译

    本教程是基于Ubuntu下Android6.0.1源码以及内核的下载和编译,记录一下,以后也就不用自己去找资料,一遍一遍的尝试了.可以翻墙的,英语好的,直接去AndroidSource. 系统环境:Ubuntu14.04LTS Android版本:6.0.1 重要网址 清华大学镜像 AndroidSource 下载前的准备 安装OpenJdk sudo add-apt-repository ppa:openjdk-r/ppa sudo apt-get update sudo apt-get in

  • Android源码学习之组合模式定义及应用

    组合模式定义: Compose objects into tree structures to represent part-whole hierarchies. Composite lets clients treat individual objects and compositions of objects uniformly. 将对象组合成树形结构以表示"部分-整体"的层次结构,使得用户对单个对象和组合对象的使用具有一致性. 如上图所示(截取自<Head First De

  • Android源码学习之观察者模式应用及优点介绍

    观察者模式定义: Define a one-to-many dependency between objects so that when one object changes state, all its dependents aer notified and updated automatically. 定义对象间一种一对多的依赖关系,使得当一个对象改变状态,则所有依赖于它的对象都会得到通知并被自动更新.  如上图所示(截取自<Head First Design Patterns>一书),

  • Android源码学习之单例模式应用及优点介绍

    单例模式定义: Ensure a class has only one instance, and provide a global point of access to it. 动态确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例. 如上图所示(截取自<Head First Design Patterns>一书). 通过使用private的构造函数确保了在一个应用中产生一个实例,并且是自行实例化(在Singleton中自己使用new Singleton()). 具体单例模式有

随机推荐