项目要求:该项目因为没有使用android5.0,导致启动bluetooth的蓝牙audio slave功能必须使用第三方模组,该第三方模组,启动是通过android主板通过GPIO控制。UI界面是通过图形选择或者一个kpd组合按键来打开关闭或者是启动蓝牙搜索功能。
1,用户按键的侦測:
标准的行为,用户的组合按键,kernel里面向上层发送scancode,然后framework把scancode转换成keycode的keyevent,该keyevent会被PhoneWindows接受并处理。
对于mtk,从按键到keycode,已经被封装起来。这个封装的工具就是dct.
比如。我们的项目组合按键是:KCOL2+KROW1,我们能够在图形工具中如选择我们的配置
在样例中,我们设置该案件的keycode是SYM
当然,也能够自定义新的keycode。我嫌麻烦,就直接使用现成的,怎样定义新的按键,请看附录:
2,侦測用户按键行为,发送对应的intent
上面的keycode在发送到各个window之前。是被PhoneWindowManager.java接收到的,在当中有一个方法:
@Override
public intinterceptKeyBeforeQueueing(KeyEvent event, int policyFlags, booleanisScreenOn) {
望文生义。看名字我们就知道这个函数的作用。
我们在代码中例如以下拦截:
if (keyCode== KeyEvent.KEYCODE_SYM) {
Log.e("zcfdebug","nowwe catch the bluetooth button action !");
if (down){
Log.e("zcfdebug","nowwe catch the bluetooth button down !");
}
if (up){
Log.e("zcfdebug","nowwe catch the bluetooth button up !");
}
return 1;
}
当然,KEYCODE_SYM是原来系统中已经定义的keycode,事实上就是输入法的选择,我们须要屏蔽这个功能。在alps/frameworks/base/core/java/android/view/inputmethod/InputMethodManager.java:
/**
* @hide
*/
public voiddispatchKeyEvent(Context context, int seq, KeyEvent key,
FinishedEventCallback callback) {
booleanhandled = false;
/*zcfdebug--<<
synchronized(mH) {
if(DEBUG) Log.d(TAG, "dispatchKeyEvent");
if(mCurMethod != null) {
if(key.getAction() == KeyEvent.ACTION_DOWN
&& key.getKeyCode() == KeyEvent.KEYCODE_SYM) {
showInputMethodPickerLocked();
handled = true;
}else {
try {
if (DEBUG) Log.v(TAG, "DISPATCH KEY: " + mCurMethod);
final long startTime = SystemClock.uptimeMillis();
enqueuePendingEventLocked(startTime, seq, mCurId, callback);
mCurMethod.dispatchKeyEvent(seq, key, mInputMethodCallback);
return;
} catch (RemoteException e) {
Log.w(TAG, "IME died: " + mCurId + " dropping: "+ key, e);
}
}
}
}
zcfdebug-->>*/
callback.finishedEvent(seq, handled);
}
凝视掉原有功能就好了。
依照我们的项目定义,按键3-5秒为蓝牙状态电源打开搜索功能,长按8秒以上为关闭。
关闭的时候,长按8秒没有不论什么动作。
if (keyCode== KeyEvent.KEYCODE_SYM) {
Log.e("zcfdebug","nowwe catch the bluetooth button action !");
if (down){
Log.e("zcfdebug","nowwe catch the bluetooth button down !");
//lastSymKeyDownTime=System.currentTimeMillis();
lastSymKeyDownTime=event.getEventTime();
}
if (up){
Log.e("zcfdebug","nowwe catch the bluetooth button up !");
SymKeyPressTime=event.getEventTime()-lastSymKeyDownTime;
//SymKeyPressTime=System.currentTimeMillis()-lastSymKeyDownTime;
Log.e("zcfdebug","theSymKeyPressTime is "+SymKeyPressTime);
if(SymKeyPressTime>2000&& SymKeyPressTime<5000){
Log.e("zcfdebug","nowwe catch the bluetooth press 3 second!");
}
if(SymKeyPressTime>7000){
Log.e("zcfdebug","nowwe catch the bluetooth press 8 second!");
}
}
return 1;
}
蓝牙开关等动作,我们在外边的apk程序中实现。我们仅仅要发送intent实现就OK了,代码例如以下:
if (keyCode== KeyEvent.KEYCODE_SYM) {
Log.e("zcfdebug","nowwe catch the bluetooth button action !");
if (down){
Log.e("zcfdebug","nowwe catch the bluetooth button down !");
//lastSymKeyDownTime=System.currentTimeMillis();
lastSymKeyDownTime=event.getEventTime();
}
if (up){
Log.e("zcfdebug","nowwe catch the bluetooth button up !");
SymKeyPressTime=event.getEventTime()-lastSymKeyDownTime;
//SymKeyPressTime=System.currentTimeMillis()-lastSymKeyDownTime;
Log.e("zcfdebug","theSymKeyPressTime is "+SymKeyPressTime);
if(SymKeyPressTime>2000&& SymKeyPressTime<5000){
Log.e("zcfdebug","nowwe catch the bluetooth press 3 second!");
Intent btStartStopIntent = new Intent(BT_STARTSTOP);
mContext.sendBroadcast(btStartStopIntent);
}
if(SymKeyPressTime>7000){
Log.e("zcfdebug","nowwe catch the bluetooth press 8 second!");
Intent btPairIntent = new Intent(BT_PAIR);
mContext.sendBroadcast(btPairIntent);
}
}
return 1;
}
应用程序怎样接收处理intent,将在下一篇文章描写叙述
附录:Howto add a new key on android ICSICS2:
[Description]
How to add a new keyon android ICS/ICS2
[Solution]
1.在DCTtool keypad list 文件添加新按键的选项
alpsmediateksourcedctKeypad_YuSu.cmp中加入新键,如SMS快捷键
KEY_SYM
KEY_SMS
KEY_0
2.打开DCTtool 在keypad矩阵中在对应定义的按键位中加入新按键,如SMS,然后Save
3.改动linux键盘码文件input.h
因为preloaderubootkernelfactory等情况分开使用,对应的文件路径下的input.h都应该改动
为新按键添加键码值
kernelincludelinuxinput.h
bioniclibckernelcommonlinuxinput.h
externalkernel-headersoriginallinuxinput.h
externalqemulinux_keycodes.h
mediatekplaformmt6575preloadersrcdriversinclinuxinput.h
Mediatekplaformmt6575ubootinclinuxinput.h
如KEY_SMS
#define KEY_SMS 252
4.添加keypadlayout文件键盘映射,linux和androidkey映射
mediatekconfig<projectname>mt6575-kpd.kl
如:
key 252 SMS
当中252是linux键码,SMS是android识别key值假设是须要唤醒系统,还须要添加WAKE
假设新按键是全键盘的一些生僻字符,改动:mediatekconfig<project name>mt6575-kpd.kcm
5.改动Java识别keycode
framework/base/include/ui/KeyCodelabels.h
KEYCODES数据结构后面添加
{"SMS",220}
framework/base/native/include/android/KeyCodes.h
在按键定义项添加AKEYCODE_SMS = 220;
6.改动Java键盘事件
framework/base/core/java/android/view/keyevent.java
/**
*@hide
*/
public static finalint KEYCODE_SMS =220;
最后的按键为新增的
private static finalint LAST_KEYCODE ==KEYCODE_SMS;
以上/**/凝视的code是android非开放API或变量定义的时候,须要加入JavaDoc的识别,否则要运
行makeupdate-api才干build通过
假设是系统按键,改动framework/base/libs/ui/input.cpp
isSystemKey()添加caseAKEYCODE_SMS:
7.改动XML文件描写叙述符framework/base/core/res/res/values/attr.xml
<enumname="KEYCODE_SMS" value="220" />
8.添加測试验证log在android
frameworkasepolicysrccomandroidinternalpolidyimplphoneWindowManager.java
在interceptKeyBeforeDispatching()添加
if(keycode==KeyEvent.KEYCODE_SMS){
log.d(TAG,"interceptKeyTi KEYCODE_SMS keyCode="+ keyCode + "down=" + down + "
repeatCount=" +repeatCount + “ keyguardOn=” + keyguardOn + “ mHomePressed=”+
9.能够抓log确认,或者添加測试APK检測
Kernal log:
<4>[253.828234]kpd:register = fffe ffff ffff ffff ff
<4>[253.828825]kpd:(pressed) HW keycode = 0
<4>[253.829348]kpd:report Linux keycode = 252
<4>[253.829857]kpd:save new keymap state
<4>[254.030814]kpd:register = ffff ffff ffff ffff ff
<4>[254.031405]kpd:(released) HW keycode = 0
<4>[254.031936]kpd:report Linux keycode = 252
<4>[254.032445]kpd:save new keymap state
Android log:
WindowManager:interceptKeyTq keycode=220 screenIsOn=true keyguardActive=false
policyFlags =#2000000 down =false canceled = false
D WindowManager:interceptKeyTi keyCode=220 down=false repeatCount=0 keyguardOn=false
mHomePressed=falsecanceled=false
D WindowManager:interceptKeyTi KEYCODE_SMS keyCode=220 down=false repeatCount=0
keyguardOn=falsemHomePressed=false canceled=false
版权声明:本文博主原创文章。博客,未经同意不得转载。