Android输入系统(6)——多点触摸处理

1. 多触摸和单触摸的Mapper不同

InputReader::addDeviceLocked(nsecs_t when, int32_t deviceId)
    InputDevice* device = createDeviceLocked(deviceId, controllerNumber, identifier, classes);
        //键盘的Mapper
        if (classes & (INPUT_DEVICE_CLASS_KEYBOARD | INPUT_DEVICE_CLASS_DPAD | INPUT_DEVICE_CLASS_GAMEPAD))
        device->addMapper(new KeyboardInputMapper(device, keyboardSource, keyboardType));
        //多触摸和单触摸的Mapper不同。
        if (classes & INPUT_DEVICE_CLASS_TOUCH_MT) {
            device->addMapper(new MultiTouchInputMapper(device));
        } else if (classes & INPUT_DEVICE_CLASS_TOUCH) {
            device->addMapper(new SingleTouchInputMapper(device));
        }
        //这些Mapper是处理输入事件的核心

InputReader::loopOnce() //InputReader.cpp
    mEventHub->getEvents(timeoutMillis, mEventBuffer, EVENT_BUFFER_SIZE);
        processEventsLocked(mEventBuffer, count); //获取到事件后处理
            InputDevice device->process(rawEvents, count); //调用InputDevice的process()来处理
                InputMapper mapper->process(rawEvent); //调用InputMapper的process()来处理。它是个纯虚函数,需要子类必须实现的。
                //这些Mapper就是上面添加的Mapper

因此处理按键事件的核心是KeyboardInputMapper.process(),单点触摸的是SingleTouchInputMapper.process(),多点触摸的是MultiTouchInputMapper.process()。

2. 一般来说触摸屏的坐标不等于LCD屏的坐标,之间可能需要转换。

3. 手势识别是由App决定的,输入系统只是上报坐标点。

4. 对于多点触摸屏处理函数是MultiTouchInputMapper.process()

MultiTouchInputMapper.process    //InputReader.cpp
    TouchInputMapper::process(rawEvent);
        //由于多点触摸屏中没有Button,Scroll,因此这里不做处理
        mCursorButtonAccumulator.process(rawEvent); //type==EV_KEY才处理
        mCursorScrollAccumulator.process(rawEvent); //type==EV_REL才处理
        mTouchButtonAccumulator.process(rawEvent);  //type==EV_KEY才处理
        //多点触摸的前几个点是不会执行sync的,所有点上报完后input_sync()上报的
        是这个键对,
        if (rawEvent->type == EV_SYN && rawEvent->code == SYN_REPORT) {
            sync(rawEvent->when);
        }
    mMultiTouchMotionAccumulator.process(rawEvent);
        //记录上报的type code pressure等值。

5. 驱动上报一个点和上报完所有点

/*上报完所有的触摸点后上报这个type=EV_SYN,code=SYN_REPORT*/
static inline void input_sync(struct input_dev *dev)
{
    input_event(dev, EV_SYN, SYN_REPORT, 0);
}
/*上报完一个触摸点后上报这个type=EV_SYN,code=SYN_MT_REPORT*/
static inline void input_mt_sync(struct input_dev *dev)
{
    input_event(dev, EV_SYN, SYN_MT_REPORT, 0);
}

6. InputReader读取完所有的触摸点后执行下面函数:
上面的sync(rawEvent->when)为:

TouchInputMapper::sync(nsecs_t when) //InputReader.cpp, 注意这个是TouchInputMapper::sync,MultiTouchInputMapper没有重写它。
    MultiTouchInputMapper::syncTouch(nsecs_t when, RawState* outState)
    /*
     * 若驱动不能上报ABS_MT_TRACKING_ID的话,就由软件分配一个id值,
     * 算法就是认为最近的点的id(id是用来判断2点是不是在同一个划痕上)
     * 相同。
     */
    assignPointerIds(last, next)
    processRawTouches(false /*timeout*/);
    TouchInputMapper::processRawTouches
        cookAndDispatch(when);
            cookPointerData();
                dispatchTouches(when, policyFlags);
                    dispatchMotion
                        NotifyMotionArgs args
                        getListener()->notifyMotion(&args);
                            mArgsQueue.push(new NotifyMotionArgs(*args)); //放入队列中

7. 若驱动不能上报ABS_MT_TRACKING_ID的话,就由软件分配一个id值,算法就是认为最近的点的id(id是用来判断2点是不是在同一个划痕上)
相同。

8. 对应多点触摸我们只关心ViewPostIme,也就是在输入法之后的处理,对输入法之前的处理我们根本不关心,因为对于多点触摸屏,它没有
做任何事情。

9. 触摸坐标与焦点的对应关系
根据触点(x,y)坐标,递归地找到目标控件

对于触摸屏有两种触摸方式
click: 松开后的处理
touch:按下,松开,滑动

10. 画UML图取巧的办法:在上面两个函数中加打印,把函数的调用栈打印出来,这样就知道调用流程了。
怎么添加这dump stack的函数:
在APP_0009 v1版本上添加,是在AS中的代码中加打印调用栈的函数。AS中使用的那些接口就是Framework里面的,在SI工程里面可以直接看。
打印调用栈的方法参考:https://blog.csdn.net/freshui/article/details/9456889
一句代码即可:Log.d(TAG,Log.getStackTraceString(new Throwable()));

它可以打印出具体类的函数(包名+类名+函数名),可以避开由于类的继承关系混淆视听!

原文地址:https://www.cnblogs.com/hellokitty2/p/10884402.html

时间: 2024-08-01 19:25:18

Android输入系统(6)——多点触摸处理的相关文章

《深入理解Android 卷III》第五章 深入理解Android输入系统

<深入理解Android 卷III>即将公布.作者是张大伟.此书填补了深入理解Android Framework卷中的一个主要空白.即Android Framework中和UI相关的部分.在一个特别讲究颜值的时代,本书分析了Android 4.2中WindowManagerService.ViewRoot.Input系统.StatusBar.Wallpaper等重要"颜值绘制/处理"模块 第5章 深入理解Android输入系统(节选) 本章主要内容: ·  研究输入事件从设

10.1、android输入系统_必备Linux编程知识_inotify和epoll

1. inotify和epoll 怎么监测键盘接入与拔出? (1)hotplug机制:内核发现键盘接入/拔出==>启动hotplug进程==>发消息给输入系统 (2)inotify机制:输入系统使用inotify来监测目录/dev/input android使用inofity机制 当插入多个键盘时,系统怎么知道哪个键盘被按下? android下使用epoll,可以同时监控多个文件,当文件发生改变,其会知道谁变化了 参考代码:frameworks\native\services\inputfli

10.5 android输入系统_Reader线程_使用EventHub读取事件和核心类及配置文件_实验_分析

4. Reader线程_使用EventHub读取事件 使用inotify监测/dev/input下文件的创建和删除 使用epoll监测有无数据上报 细节: a.fd1 = inotify_init("/dev/input") b.假设input下已经有了event0和event1 fd2 = open("/dev/input/event0") fd3= open("/dev/input/event1") c.使用epoll_wait监测fd1.f

10.6 android输入系统_Dispatcher线程_总体框架

图解Android - Android GUI 系统 (5) - Android的Event Input System - 漫天尘沙 - 博客园.htm // 关注里面的Dispatcher处理流程http://www.cnblogs.com/samchen2009/p/3368158.html Dispatcher线程框架: 分发 问:发什么?发给谁? Dispatcher流程如下: 获得事件: (1)放入队列前先稍加处理:分类(Global输入/System输入/User输入).处理紧急事件

Android输入系统(4)——InputStage

一.两个线程启动过程 SystemService.java 启动 InputManagerService 服务 Service: InputManagerService.java JNI: com_android_server_input_InputManagerService.cpp InputManagerService(Context context) /*调用的第一个本地函数*/ mPtr = nativeInit(this, mContext, mHandler.getLooper()

Android 手势滑动,多点触摸放大缩小图片

学习安卓两点触摸滑动缩放图片 1.布局文件如下main.xml <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="

10.8 android输入系统_实战_使用GlobalKey一键启动程序

11. 实战_使用GlobalKey一键启动程序参考文章:Android 两种注册.发送广播的区别http://www.jianshu.com/p/ea5e233d9f43 [Android]动态注册广播接收器 http://blog.csdn.net/etzmico/article/details/7317528 Android初学习 - 在BroadcastReceiver中启动Activity的问题 http://blog.csdn.net/cnmilan/article/details/

Android 4.0屏蔽式多点触摸

比方这张图.我想不接或者接单,二者仅仅能点一个,不能同一时候点击,否则会造成混乱.我们仅仅要在嵌套他们俩的布局中增加这么一段话: android:splitMotionEvents="false" 上ok该. 版权声明:本文博客原创文章.博客,未经同意,不得转载.

[Android] 输入系统(三):加载按键映射

映射表基本概念 由于Android调用getEvents得到的key是linux发送过来的scan code,而Android处理的是类似于KEY_UP这种统一类型的key code,因此需要有映射表把scan code转换成key code.映射表在板子上的位置是/system/usr/keylayout/xxx.kl,先看一下映射表是什么样子的,下面截选了一段. key 2 1 key 3 2 key 4 3 key 5 4 key 6 5 key 7 6 key 8 7 key 9 8 k