关闭

Android中添加自定义按键 ---- 非标准做法

标签: androidbuttonuserstructstringsearch
4109人阅读 评论(3) 收藏 举报
分类:

这里我来介绍一下怎么在Android中添加自定义按键,首先要打通kernel中的驱动,然后是Android层,我这里介绍一种不是很标准的方法,项目需要。

首先是Linux中,我这里只是把已有的search 按钮替换成我们的用户按钮,并提供长按和短按功能。

修改板级文件,我这边使用的事Freescale imx53的开发板

kernel/arch/arm/mach-mx5/mx53_smd.c 

定义按键与哪个GPIO连接

//+++++add our own user button
#define MX53_MDK_KEY_USER			(1*32 + 23)	/* GPIO_2_23 */
#define MX53_MDK_KEY_SEARCH			(1*32 + 23)	/* GPIO_2_23 */

然后定义这个按键的信息

static struct gpio_keys_button smd_buttons[] = {
#ifdef CONFIG_MACH_MX53_MIPAD
	GPIO_BUTTON(MX53_MIPAD_KEY_VOL_UP, KEY_VOLUMEUP, 1, "volume-up", 0),
	GPIO_BUTTON(MX53_MIPAD_KEY_VOL_DOWN, KEY_VOLUMEDOWN, 1, "volume-down", 0),
#elif defined(CONFIG_MACH_MX53_MDK)
	//+++++modify for user button by Jay
//	GPIO_BUTTON(MX53_MDK_KEY_SEARCH, KEY_SEARCH, 1, "search", 0),
	GPIO_BUTTON(MX53_MDK_KEY_USER, KEY_F1, 1, "user", 0),
	GPIO_BUTTON(MX53_MDK_KEY_HOME, KEY_HOME, 1, "home", 0),
	GPIO_BUTTON(MX53_MDK_KEY_BACK, KEY_BACK, 1, "back", 0),
	GPIO_BUTTON(MX53_MDK_KEY_MENU, KEY_MENU, 1, "menu", 0),
	GPIO_BUTTON(MX53_SMD_KEY_VOL_UP, KEY_VOLUMEUP, 1, "volume-up", 0),
	GPIO_BUTTON(MX53_SMD_KEY_VOL_DOWN, KEY_VOLUMEDOWN, 1, "volume-down", 0),
#else
	GPIO_BUTTON(MX53_SMD_KEY_VOL_UP, KEY_VOLUMEUP, 1, "volume-up", 0),
	GPIO_BUTTON(MX53_SMD_KEY_VOL_DOWN, KEY_VOLUMEDOWN, 1, "volume-down", 0),
#endif
};

这里我把键值定义为KEY_F1

然后我们在按键驱动中添加新的按键和功能,

/kernel/drivers/input/keyboard/gpio_keys.c

static void gpio_keys_report_event(struct gpio_button_data *bdata)
{
	struct gpio_keys_button *button = bdata->button;
	struct input_dev *input = bdata->input;
	unsigned int type = button->type ?: EV_KEY;
	int state = (gpio_get_value(button->gpio) ? 1 : 0) ^ button->active_low;

	//+++add by Jay
	struct device *dev = &bdata->input->dev;
	char event_string[20];
	char *envp[] = { event_string, NULL };
	//----

#ifdef CONFIG_MACH_MX53_MIPAD
	input_event(input, type, button->code, !state);
#else
	//++++++add user button by Jay
	if(button->code == KEY_F1) {
		if(state) {
			startKeyTime = (jiffies*1000)/HZ;
			user_button_state = 1;
			user_thread = kthread_run(user_thread_handler,
					dev, "user-button");
		} else {
			user_button_state = 0;
			if((jiffies*1000)/HZ - startKeyTime < 1500)
			{
				sprintf(event_string, "EVENT=shortpress");
				kobject_uevent_env(&dev->kobj, KOBJ_CHANGE, envp);
				printk("short press\r\n");
			}
		}
	}
	//-----
	else
		input_event(input, type, button->code, !!state);
#endif
	input_sync(input);
}


首先在按键中断到来的时候我们要判断出是我们定义的user button,当按下的时候产生一个下降沿中断,按下的时候state应该为1,先记录按下时的时间,然后开启一个线程,看下线程中做了什么

//++++add by Jay
#include <linux/kthread.h>
static int startKeyTime = 0;
static int user_button_state = 0;
static struct task_struct *user_thread;

static int user_thread_handler(void *arg)
{
	struct device *dev = arg;

	while(user_button_state == 1)
	{
		char event_string[20];
		char *envp[] = { event_string, NULL };

		msleep(100);
		if(((jiffies*1000)/HZ) - startKeyTime > 1500)
		{
			sprintf(event_string, "EVENT=longpress");
			kobject_uevent_env(&dev->kobj, KOBJ_CHANGE, envp);
			printk(KERN_INFO "user button long press.\n");
			break;
		}
	}
	return 0;
}
//-----

在线程中我们不断判断state的值,如果一直按下,就判断按下了多久,当达到我们定义的长按时间的话,就会像上层发送一个长按的uevent事件,否则返回。

在按键抬起的时候也会产生中断,这样的话在else语句中会做判断,然后向user space发送一个uevent事件。

-------------------------------------------------

接下来我们看下在android中我们如何捕获

我们使用ueventObserver的方式去捕获uevent事件,详细的不多说了,可以参考我之前的博文有介绍uevent在android中的用法,直接贴代码,本人对java不是很熟悉,简单的代码就不来卖弄了,呵呵。

我们把代码加载ActivityManagerService.java中

    //+++++add for our uevent by Jay
    private Handler UeventHandler=new Handler()
    {
	    @Override
	    public void handleMessage(Message msg)
	    {
		    super.handleMessage(msg);
		    if(msg.what==0)
		    	Toast.makeText(mContext,"short press!" ,Toast.LENGTH_SHORT).show();
		    if(msg.what==1)
			Toast.makeText(mContext,"long press!" ,Toast.LENGTH_SHORT).show();
	    }
    };
    private UEventObserver mShortUEventObserver = new UEventObserver()
    {
	    @Override
	    public void onUEvent(UEventObserver.UEvent event)
	    {
		    UeventHandler.sendEmptyMessage(0);
	    }
    };

    private UEventObserver mLongUEventObserver = new UEventObserver()
    {
	    @Override
	    public void onUEvent(UEventObserver.UEvent event)
	    {
		    UeventHandler.sendEmptyMessage(1);
	    }
    };
    //------end add
    final void finishBooting() {
	//+++++add by Jay for user button
	mShortUEventObserver.startObserving("EVENT=shortpress");
	mLongUEventObserver.startObserving("EVENT=longpress");
	//-------
        IntentFilter pkgFilter = new IntentFilter();
        pkgFilter.addAction(Intent.ACTION_QUERY_PACKAGE_RESTART);
        pkgFilter.addDataScheme("package");
        mContext.registerReceiver(new BroadcastReceiver() {

当bootFinish的时候我们就开始侦测这2个uevent时候被发生,然后用toast来显示。

最后重新编译,烧到我们的开发板中,当短按的时候会有信息吐出来,长按超过1.5s也会有信息吐出来。


分享工作经验,结束。

=========================================================

mail & MSN :zhangjie201412@live.com

=========================================================




1
0

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:335343次
    • 积分:4604
    • 等级:
    • 排名:第6614名
    • 原创:77篇
    • 转载:68篇
    • 译文:0篇
    • 评论:194条
    博客专栏
    最新评论