拦截单击/触摸事件而不覆盖ViewGroup或View方法

时间:2017-09-05 14:02:24

标签: java android android-event

有没有什么方法可以拦截/装饰视图的触摸事件而不扩展View或包装一些ViewGroup(可以拦截子事件)?

假设我有ExpandableListView来处理项目点击事件。如果我在适配器返回的膨胀项视图上设置适配器OnClickListenerOnTouchListener,则ExpandableListView不执行相应的操作(组扩展),因为项目的侦听器使用了事件。

我不想使用ExpandableListView#setOnItemClickListener的原因是,我想在不使用ExpandableListView依赖项的情况下在适配器中修饰click事件。

1 个答案:

答案 0 :(得分:1)

我找到了解决此问题的有效方案。

解决方案:在OnTouchListener中收集事件克隆,然后将它们分派到父视图。

private final Queue<MotionEntry> consumedEvents = new LinkedList<>();
private final AtomicBoolean isDispatching = new AtomicBoolean(false);
...
    groupView.setOnTouchListener(new OnTouchListener() {
        @Override 
        public boolean onTouch(View v, MotionEvent e) {
            // we don't want to handle re-dispatched event...
            if (isDispatching.get()) {
                return false; 
            }
            // create clone as event might be changed by parent
            MotionEvent clone = MotionEvent.obtain(e);
            MotionEntry entry = new MotionEntry(v, clone);
            consumedEvents.add(entry);

            // consume ACTION_DOWN in order to receive subsequent motion events 
            // like ACTION_MOVE, ACTION_CANCEL/ACTION_UP...
            if (event.getActionMasked() == MotionEvent.ACTION_DOWN) {
                return true;
            }
            // we do not want to handle canceled motion...
            if (event.getActionMasked() == MotionEvent.ACTION_CANCEL) {
                consumedEvents.clear();
                return false;
            }
            // at this moment we have intercepted whole motion 
            // = re-dispatch to parent in order to apply default handling...
            if (event.getActionMasked() == MotionEvent.ACTION_UP) {
                dispatchEvents();
            }
            return true;
        }
    });
...

派遣方法:

private void dispatchEvents() {
    isDispatching.set(true);
    while (!consumedEvents.isEmpty()) {
        MotionEntry entry = consumedEvents.poll();

        ViewGroup parent = (ViewGroup) entry.view.getParent();
        if (parent == null || entry.view.getVisibility() != View.VISIBLE) {
            continue; // skip dispatching to detached/invisible view
        }
        // make position relative to parent...
        entry.event.offsetLocation(entry.view.getLeft(), entry.view.getTop());
        entry.event.setSource(PARENT_DISPATCHER);
        parent.dispatchTouchEvent(entry.event);

        if (event.getActionMasked() == MotionEvent.ACTION_UP) {
            clickListener.onClick(entry.view);
        }
    }
    isDispatching.set(false);
}

助手班

private class MotionEntry {
    private final View view;
    private final MotionEvent event;

    public MotionEntry(View view, MotionEvent event) {
        this.view = view;
        this.event = event;
    }
}