Android系统中自定义按键的短按、双击、长按事件
在项目中碰到这样的问题:
由于系统中的按键在底层做了重新定义或者新增了按键,此时需要在APP层对按键事件(keyevent)做分解处理,模拟Android系统做法,把keyevent分解成:
1、单击事件:就是普通key的单击;
2、双击事件:500ms内同一按键单击两次;
3、长按事件:同一按键长按超过1000ms(系统中长按事件为500ms);
4、组合按键:两个以上按键同时按住;
其中的keyevent可以来自Activity、View子类的dispatchKeyEvent方法,也可以是我们自定义的接口,也可以是我们发广播送上来的,根据项目需求;
关于各事件的原理:
1、双击事件:每次点击的up事件中启动一个定时(500ms)线程消息,用Handler.postDelayed()方法。
2、长按事件:每次点击的down事件中启动一个定时(1000ms)线程消息,用Handler.postDelayed()方法,注意:在RepeatCount==0时启动;
3、组合按键:用变量记录每个按键的状态,再进行判断;
具体代码如下: Java代码
package com.jerome.util; import android.content.Context;
import android.os.Handler;
import android.util.Log;
import android.view.KeyEvent; public class KeyUtil {
private boolean isVolumeDown = false;
private boolean isVolumeUp = false;
private boolean isMenu = false;
private int currentKeyCode = 0; private static Boolean isDoubleClick = false;
private static Boolean isLongClick = false; CheckForLongPress mPendingCheckForLongPress = null;
CheckForDoublePress mPendingCheckForDoublePress = null;
Handler mHandler = new Handler(); Context mContext = null;
private String TAG = ""; public KeyUtil(Context context, String tag) {
mContext = context;
TAG = tag;
} public void dispatchKeyEvent(KeyEvent event) {
int keycode = event.getKeyCode(); // 有不同按键按下,取消长按、短按的判断
if (currentKeyCode != keycode) {
removeLongPressCallback();
isDoubleClick = false;
} // 处理长按、单击、双击按键
if (event.getAction() == KeyEvent.ACTION_DOWN) {
checkForLongClick(event);
} else if (event.getAction() == KeyEvent.ACTION_UP) {
checkForDoubleClick(event);
} if (keycode == KeyEvent.KEYCODE_VOLUME_DOWN) {
if (event.getAction() == KeyEvent.ACTION_DOWN) {
isVolumeDown = true;
} else if (event.getAction() == KeyEvent.ACTION_UP) {
isVolumeDown = false;
}
} else if (keycode == KeyEvent.KEYCODE_VOLUME_UP) {
if (event.getAction() == KeyEvent.ACTION_DOWN) {
isVolumeUp = true;
} else if (event.getAction() == KeyEvent.ACTION_UP) {
isVolumeUp = false;
}
} else if (keycode == KeyEvent.KEYCODE_MENU) {
if (event.getAction() == KeyEvent.ACTION_DOWN) {
isMenu = true;
} else if (event.getAction() == KeyEvent.ACTION_UP) {
isMenu = true;
}
} // 判断组合按键
if (isVolumeDown
&& isVolumeUp
&& isMenu
&& (keycode == KeyEvent.KEYCODE_VOLUME_UP
|| keycode == KeyEvent.KEYCODE_VOLUME_DOWN || keycode == KeyEvent.KEYCODE_MENU)
&& event.getAction() == KeyEvent.ACTION_DOWN) {
//组合按键事件处理;
isVolumeDown = false;
isVolumeUp = false;
isMenu = false;
}
} private void removeLongPressCallback() {
if (mPendingCheckForLongPress != null) {
mHandler.removeCallbacks(mPendingCheckForLongPress);
}
} private void checkForLongClick(KeyEvent event) {
int count = event.getRepeatCount();
int keycode = event.getKeyCode();
if (count == 0) {
currentKeyCode = keycode;
} else {
return;
}
if (mPendingCheckForLongPress == null) {
mPendingCheckForLongPress = new CheckForLongPress();
}
mPendingCheckForLongPress.setKeycode(event.getKeyCode());
mHandler.postDelayed(mPendingCheckForLongPress, 1000);
} class CheckForLongPress implements Runnable { int currentKeycode = 0; public void run() {
isLongClick = true;
longPress(currentKeycode);
} public void setKeycode(int keycode) {
currentKeycode = keycode;
}
} private void longPress(int keycode) {
Log.i(TAG, "--longPress 长按事件--" + keycode);
} private void singleClick(int keycode) {
Log.i(TAG, "--singleClick 单击事件--" + keycode);
} private void doublePress(int keycode) {
Log.i(TAG, "---doublePress 双击事件--" + keycode);
} private void checkForDoubleClick(KeyEvent event) {
// 有长按时间发生,则不处理单击、双击事件
removeLongPressCallback();
if (isLongClick) {
isLongClick = false;
return;
} if (!isDoubleClick) {
isDoubleClick = true;
if (mPendingCheckForDoublePress == null) {
mPendingCheckForDoublePress = new CheckForDoublePress();
}
mPendingCheckForDoublePress.setKeycode(event.getKeyCode());
mHandler.postDelayed(mPendingCheckForDoublePress, 500);
} else {
// 500ms内两次单击,触发双击
isDoubleClick = false;
doublePress(event.getKeyCode());
}
} class CheckForDoublePress implements Runnable { int currentKeycode = 0; public void run() {
if (isDoubleClick) {
singleClick(currentKeycode);
}
isDoubleClick = false;
} public void setKeycode(int keycode) {
currentKeycode = keycode;
}
} private void removeDoublePressCallback() {
if (mPendingCheckForDoublePress != null) {
mHandler.removeCallbacks(mPendingCheckForDoublePress);
}
}
}
注意:
只有Action Down状态下RepeatCount才会>0,避免长按和单击事件混淆;
Android系统中自定义按键的短按、双击、长按事件的更多相关文章
- Android Tv 中的按键事件 KeyEvent 分发处理流程
这次打算来梳理一下 Android Tv 中的按键点击事件 KeyEvent 的分发处理流程.一谈到点击事件机制,网上资料已经非常齐全了,像什么分发.拦截.处理三大流程啊:或者 dispatchTou ...
- Android系统中的广播(Broadcast)机制简要介绍和学习计划
在Android系统中,广播(Broadcast)是在组件之间传播数据(Intent)的一种机制:这些组件甚至是可以位于不同的进程中,这样它就像Binder机制一样,起到进程间通信的作用:本文通过一个 ...
- Android系统中设置TextView的行间距(非行高)
Android系统中TextView默认显示中文时会比较紧凑,不是很美观.为了让每行保持一定的行间距,可以设置属性android:lineSpacingExtra或android:lineSpacin ...
- Android系统对话框——自定义关闭
Android系统对话框--自定义关闭 Dialog是我们在项目中经常用到的,5.x以后的Dialog也很好看,很安卓风,Android也给我们提供了新的包,低版本可以显示一样的效果.我们在使用的导入 ...
- AIDL在android系统中的作用
AIDL,Android Interface definition language的缩写,它是一种android内部进程通信接口的描述语言,通过它我们可以定义进程间的通信接口.最近看了下AIDL在A ...
- Android系统中的6种模式
Android系统中的6种模式 1:一般启动模式(normal mode): 功能是正常启动手机,方法为关机状态下按电源键启动. 2:安全模式(safe mode): 此模式和正常启动一样 ...
- [原创]Android系统中常用JAVA类源码浅析之HashMap
由于是浅析,所以我只分析常用的接口,注意是Android系统中的JAVA类,可能和JDK的源码有区别. 首先从构造函数开始, /** * Min capacity (other than zero) ...
- Android系统中 setprop,getprop,watchprops命令的使用
如:在frameworks/opt/net/ims/src/java/com/android/ims/ImsManager.java if (SystemProperties.get("pe ...
- 用adb pull命令从android系统中读取文件失败的原因及解决办法
问题:使用adb pull命令从android系统中读取文件失败.显示:Permission denied 原因:是由于文件权限原因引起. 使用ls -l命令查看android系统中的 ...
随机推荐
- 如何使用Google Map API开发Android地图应用
两年前开发过的GoogleMap已经大变样,最近有项目要用到GoogleMap,重新来配置Android GoogleMap开发环境,还真是踩了不少坑. 一.下载Android SDK Manager ...
- Linq分组
1.lin语句 int[] nums = { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0, 3 }; DataTable table = new DataTable("Numb ...
- Android Touch事件传递机制详解
Android开发的朋友经常处理各种触摸事件,然而在触摸事件的传递过程中主要用到三个方法:dispatchTouchEvent().onInterceptTouchEvent()和onTouchEve ...
- poi获取合并单元格内的第一行第一列的值
当读取如图所示的excel时,显示为第1行 第1列 的内容是:合并单元格 其它在合并单元格区域内的单元格不显示 示例代码如下: import java.io.FileInputStream; impo ...
- STMFD 和LDMFD指令
http://blog.163.com/oy_mcu/blog/static/16864297220120193458892/ LDM/STM指令主要用于现场保护,数据复制,参数传送等. STMFD指 ...
- Unity Mono IDE Setting
Mac: 修改快捷键 删除一整行 Delete Entire Line Command+D Format Document Shift+Command+F 在文件中查找 Fin in Fil ...
- C++动态加载DLL调用方法
一.构建DLL路径 char szTmp[_MAX_PATH]; char* szPath = getcwd(szTmp, _MAX_PATH);//获取当前工作目录 //构建dll路径 strc ...
- 精华 对express中next函数的一些理解
关于next主要从三点来进行说明: next的作用是什么? 我们应该在何时使用next? next的内部实现机制是什么? Next的作用 我们在定义express中间件函数的时候都会将第三个参 ...
- [转] vim自定义配置 和 在ubnetu中安装vim
Ubuntu 12.04安装vim和配置 问题: ubuntu默认没有安装vim,出现: jyg@ubuntu:~$ vim test.cThe program 'vim' can be foun ...
- activity 四种启动模式
前言 Activity的启动模式决定了Activity的启动运行方式 四种模式 Activity启动模式设置: <activity android:name=".MainActivit ...