android 添加按(power键)电源键结束通话(挂断电话)

时间:2022-04-17 07:57:09

首先我们发现现在我们所用的android智能手机大部分都有当你在打电话时按power键来挂断电话,一般都是在设置中。
我主要是在原生源码中添加这一功能,主要用于学习。。。。先看一张图:
android 添加按(power键)电源键结束通话(挂断电话) 
看到那个按电源键挂断电话吧,那就是我所添加的,本来原生源码中是没有这一栏的。。。。。

大概思路
首先我先添加这一个checkboxpreference,然后将是否选择这一功能的值(0和1)存到data/data/com.android.providers.settings
/databases/settings.db数据库的system表中
,然后再根据数据库表中的值在phonewindownmanager.java中去处理。

具体过程
首先找到setting的源码,在源码下我们要找到通话设置,在seting.xml中我们能找到

复制代码 代码如下:


<span style="font-size: 14px"> <com.android.settings.iconpreferencescreen
android:key="call_settings"
settings:icon="@drawable/ic_settings_call"
android:title="@string/call_settings_title">
<intent
android:action="android.intent.action.main"
android:targetpackage="com.android.phone"
android:targetclass="com.android.phone.callfeaturessetting" />
</com.android.settings.iconpreferencescreen></span>


这个call_settings就是我们在setting(设置)中看到的通话设置,但是我们却不能在settings中的源码中找到关于call_settings的布局文件, 因此我们需要找到它,其实这个布局文件是在package/app/phone中,也就是在phone这个app源码的资源文件中。

因此我们在phone的资源文件下能找到call_feature_setting.xml文件如下:

复制代码 代码如下:


<span style="font-size: 14px"><preferencescreen xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:phone="http://schemas.android.com/apk/res/com.android.phone"
android:title="@string/call_settings">
<preferencescreen
android:key="button_fdn_key"
android:title="@string/fdn"
android:summary="@string/sum_fdn"
android:persistent="false">
<intent android:action="android.intent.action.main"
android:targetpackage="com.android.phone"
android:targetclass="com.android.phone.fdnsetting" />
</preferencescreen>
<preferencecategory
android:key="button_voicemail_category_key"
android:title="@string/voicemail"
android:persistent="false">
<listpreference
android:key="button_voicemail_provider_key"
android:title="@string/voicemail_provider"
android:summary="@string/sum_voicemail_choose_provider"
android:defaultvalue=""
android:persistent="true"
/>
<preferencescreen android:key="button_voicemail_setting_key"
android:title="@string/voicemail_settings"
android:persistent="false">
<!-- note for all com.android.phone.editphonenumberpreference objects
the last several attributes are for use with the edittext field
in the dialog. these attributes are forwarded to that field
when the edittext is created. the attributes include:
1. android:singleline
2. android:autotext
3. android:background -->
<com.android.phone.editphonenumberpreference
android:key="button_voicemail_key"
android:title="@string/voicemail_settings_number_label"
android:persistent="false"
android:dialogtitle="@string/voicemail"
phone:confirmmode="confirm"
android:singleline="true"
android:autotext="false" />
</preferencescreen>
</preferencecategory>
。。。。。。。。。。。。。。。。。。
。。。。。。。。。。。。。。。。。
</span>


因此我们可以在最前面添加一个checkboxpreference

复制代码 代码如下:


<span style="font-size: 14px"><checkboxpreference
android:key="press_power_end_call_key"
android:title="@string/press_power_end_call"
android:persistent="false"/></span>


变成

复制代码 代码如下:


<span style="font-size: 14px"><preferencescreen xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:phone="http://schemas.android.com/apk/res/com.android.phone"
android:title="@string/call_settings">
<checkboxpreference
android:key="press_power_end_call_key"
android:title="@string/press_power_end_call"
android:persistent="false"/>
<preferencescreen
android:key="button_fdn_key"
android:title="@string/fdn"
android:summary="@string/sum_fdn"
android:persistent="false">
<intent android:action="android.intent.action.main"
android:targetpackage="com.android.phone"
android:targetclass="com.android.phone.fdnsetting" />
</preferencescreen>
。。。。。。。
。。。。。。。
。。。。。。。</span>


在这里有自己定义的
android:title="@string/press_power_end_call"
所以我们要在资源的string.xml文件中添加相关的信息:
在package/app/phone/res/values/string.xml中添加:
<string name="press_power_end_call">press_power_end_call</string>
在package/app/phone/res/values-zh-rcn/string.xml中添加:
<string name="press_power_end_call" msgid="4676390750360727396">按电源键挂断电话</string>
到这里就算添加好了ui上的东西,接下来就是代码了:
在package/app/phone/src/com/android/phone下找到callfeaturesetting.java文件,
在 public boolean onpreferencechange(preference preference, object objvalue) 方法中要增加一个如果选择了按power键挂电话的事件:

复制代码 代码如下:


<span style="font-size: 14px">//add by xxnan
else if (preference == press_power_end_call) {
//如果勾选就将1存到system表的press_power_end_call中
settings.system.putint(getcontentresolver(),
"press_power_end_call",
press_power_end_call.ischecked() ? 1 : 0);
//end by xxnan </span>


在oncreate添加如下代码之后

复制代码 代码如下:


protected void oncreate(bundle icicle) {
super.oncreate(icicle);
if (dbg) log("creating activity");
mphone = phonefactory.getdefaultphone();
addpreferencesfromresource(r.xml.call_feature_setting);
//add by xxnan
contentresolver resolver = getcontentresolver();
press_power_end_call= (checkboxpreference)findpreference(press_power_end_call_key);
press_power_end_call.setonpreferencechangelistener(this);
// 获的数据库system表里press_power_end_call的值,也就是是否选择了checkboxpreference
int press_power_end_call_key=settings.system.getint(getcontentresolver(),
"press_power_end_call",0);
//如果得到的值是1,则下次打开setting的话,选项框要勾选
if(press_power_end_call_key==1)
press_power_end_call.setchecked(true);
//end by xxnan
maudiomanager = (audiomanager) getsystemservice(context.audio_service);
// get buttons
preferencescreen prefset = getpreferencescreen();
msubmenuvoicemailsettings = (editphonenumberpreference)findpreference(button_voicemail_key);
。。。。。。。
。。。。。。。


这样就算差不多完成了到获取是否开启这一功能存放和取出到系统数据库中,接下来就是到framework/base/policy/src/com/android
/internal/policy/impl下的
phonewindowmanager.java中去处理了,之前我们就有分析到phonewindowmanager.java中的
public int interceptkeybeforequeueing(long whennanos, int action, int flags, int keycode, int scancode, int policyflags,
boolean isscreenon)方法来接受按power键的事件,在这个方法里我们只需要添加很少代码:
原来代码是

复制代码 代码如下:


case keyevent.keycode_power: {
result &= ~action_pass_to_user;
if (down) {
log.i("xxnan","xxnan"+"xiaxiangnan");
itelephony telephonyservice = gettelephonyservice();
boolean hungup = false;
if (telephonyservice != null) {
try {
if (telephonyservice.isringing()) {
// pressing power while there's a ringing incoming
// call should silence the ringer.
telephonyservice.silenceringer();
} else if ((mincallpowerbehavior
& settings.secure.incall_power_button_behavior_hangup) != 0
&& telephonyservice.isoffhook()) {
// otherwise, if "power button ends call" is enabled,
// the power button will hang up any current active call.
hungup = telephonyservice.endcall();
}
} catch (remoteexception ex) {
log.w(tag, "itelephony threw remoteexception", ex);
}
}
interceptpowerkeydown(!isscreenon || hungup);
。。。。。。。。。。。。
。。。。。。。。。。。。


修改后

复制代码 代码如下:


case keyevent.keycode_power: {
result &= ~action_pass_to_user;
if (down) {
log.i("xxnan","xxnan"+"xiaxiangnan");
int end_call_key=settings.system.getint(mcontext.getcontentresolver(),
"press_power_end_call",0); //取出数据库中是否打开这一功能的值
log.i("end_call_key","end_call_key="+end_call_key);
itelephony telephonyservice = gettelephonyservice();
boolean hungup = false;
if (telephonyservice != null) {
try {
//如果是电话正在打且开启了这一功能,当按power键就挂掉电话
if (telephonyservice.isringing()&&end_call_key==1) {
// pressing power while there's a ringing incoming
// call should silence the ringer.
// telephonyservice.silenceringer();
hungup=telephonyservice.endcall();
} else if ((mincallpowerbehavior
& settings.secure.incall_power_button_behavior_hangup) != 0
&& telephonyservice.isoffhook()) {
// otherwise, if "power button ends call" is enabled,
// the power button will hang up any current active call.
hungup = telephonyservice.endcall();
}
} catch (remoteexception ex) {
log.w(tag, "itelephony threw remoteexception", ex);
}
}
interceptpowerkeydown(!isscreenon || hungup);
。。。。。。。。。。。
。。。。。。。。。。。


由于我这个开发板上是不能插电话卡的也就没能实验成功,但是原理应该就这样的!
最后修改过的地方都要重新编译,那么我们要在源码下编译app下的phone以及framework下的policy
最后生成的out/。。。/system/app/phone.apk和out/。。。。/system/framework/android.policy.jar都要替换
手机里的相同(adb shell 进入你的手机,要有root权限)文件应该就可以了。