Blob Blame History Raw
/*
   2 finger gesture detector

   Copyright 2013 Thincast Technologies GmbH, Author: Martin Fleisz

   This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0. 
   If a copy of the MPL was not distributed with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
*/

package com.freerdp.freerdpcore.utils;

import android.content.Context;
import android.os.Handler;
import android.view.MotionEvent;
import android.view.ScaleGestureDetector;

import com.freerdp.freerdpcore.utils.GestureDetector.OnGestureListener;

public class DoubleGestureDetector {
    // timeout during that the second finger has to touch the screen before the double finger detection is cancelled
    private static final long DOUBLE_TOUCH_TIMEOUT = 100;
    // timeout during that an UP event will trigger a single double touch event
    private static final long SINGLE_DOUBLE_TOUCH_TIMEOUT = 1000;
    // constants for Message.what used by GestureHandler below
    private static final int TAP = 1;
    // different detection modes
    private static final int MODE_UNKNOWN = 0;
    private static final int MODE_PINCH_ZOOM = 1;
    private static final int MODE_SCROLL = 2;
    private static final int SCROLL_SCORE_TO_REACH = 20;
    private final OnDoubleGestureListener mListener;
    private int mPointerDistanceSquare;
    private int mCurrentMode;
    private int mScrollDetectionScore;
    private ScaleGestureDetector scaleGestureDetector;
    private boolean mCancelDetection;
    private boolean mDoubleInProgress;
    private GestureHandler mHandler;
    private MotionEvent mCurrentDownEvent;
    private MotionEvent mCurrentDoubleDownEvent;
    private MotionEvent mPreviousUpEvent;
    private MotionEvent mPreviousPointerUpEvent;
    /**
     * Creates a GestureDetector with the supplied listener.
     * You may only use this constructor from a UI thread (this is the usual situation).
     *
     * @param context  the application's context
     * @param listener the listener invoked for all the callbacks, this must
     *                 not be null.
     * @throws NullPointerException if {@code listener} is null.
     * @see android.os.Handler#Handler()
     */
    public DoubleGestureDetector(Context context, Handler handler, OnDoubleGestureListener listener) {
        mListener = listener;
        init(context, handler);
    }

    private void init(Context context, Handler handler) {
        if (mListener == null) {
            throw new NullPointerException("OnGestureListener must not be null");
        }

        if (handler != null)
            mHandler = new GestureHandler(handler);
        else
            mHandler = new GestureHandler();

        // we use 1cm distance to decide between scroll and pinch zoom
        //  - first convert cm to inches
        //  - then multiply inches by dots per inch
        float distInches = 0.5f / 2.54f;
        float distPixelsX = distInches * context.getResources().getDisplayMetrics().xdpi;
        float distPixelsY = distInches * context.getResources().getDisplayMetrics().ydpi;

        mPointerDistanceSquare = (int) (distPixelsX * distPixelsX + distPixelsY * distPixelsY);
    }

    /**
     * Set scale gesture detector
     *
     * @param scaleGestureDetector
     */
    public void setScaleGestureDetector(ScaleGestureDetector scaleGestureDetector) {
        this.scaleGestureDetector = scaleGestureDetector;
    }

    /**
     * Analyzes the given motion event and if applicable triggers the
     * appropriate callbacks on the {@link OnGestureListener} supplied.
     *
     * @param ev The current motion event.
     * @return true if the {@link OnGestureListener} consumed the event,
     * else false.
     */
    public boolean onTouchEvent(MotionEvent ev) {
        boolean handled = false;
        final int action = ev.getAction();
        //dumpEvent(ev);

        switch (action & MotionEvent.ACTION_MASK) {
            case MotionEvent.ACTION_DOWN:
                if (mCurrentDownEvent != null)
                    mCurrentDownEvent.recycle();

                mCurrentMode = MODE_UNKNOWN;
                mCurrentDownEvent = MotionEvent.obtain(ev);
                mCancelDetection = false;
                mDoubleInProgress = false;
                mScrollDetectionScore = 0;
                handled = true;
                break;

            case MotionEvent.ACTION_POINTER_UP:
                if (mPreviousPointerUpEvent != null)
                    mPreviousPointerUpEvent.recycle();
                mPreviousPointerUpEvent = MotionEvent.obtain(ev);
                break;

            case MotionEvent.ACTION_POINTER_DOWN:
                // more than 2 fingers down? cancel
                // 2nd finger touched too late? cancel
                if (ev.getPointerCount() > 2 || (ev.getEventTime() - mCurrentDownEvent.getEventTime()) > DOUBLE_TOUCH_TIMEOUT) {
                    cancel();
                    break;
                }

                // detection cancelled?
                if (mCancelDetection)
                    break;

                // double touch gesture in progress
                mDoubleInProgress = true;
                if (mCurrentDoubleDownEvent != null)
                    mCurrentDoubleDownEvent.recycle();
                mCurrentDoubleDownEvent = MotionEvent.obtain(ev);

                // set detection mode to unkown and send a TOUCH timeout event to detect single taps
                mCurrentMode = MODE_UNKNOWN;
                mHandler.sendEmptyMessageDelayed(TAP, SINGLE_DOUBLE_TOUCH_TIMEOUT);

                handled |= mListener.onDoubleTouchDown(ev);
                break;

            case MotionEvent.ACTION_MOVE:

                // detection cancelled or not active?
                if (mCancelDetection || !mDoubleInProgress || ev.getPointerCount() != 2)
                    break;

                // determine mode
                if (mCurrentMode == MODE_UNKNOWN) {
                    // did the pointer distance change?
                    if (pointerDistanceChanged(mCurrentDoubleDownEvent, ev)) {
                        handled |= scaleGestureDetector.onTouchEvent(mCurrentDownEvent);
                        MotionEvent e = MotionEvent.obtain(ev);
                        e.setAction(mCurrentDoubleDownEvent.getAction());
                        handled |= scaleGestureDetector.onTouchEvent(e);
                        mCurrentMode = MODE_PINCH_ZOOM;
                        break;
                    } else {
                        mScrollDetectionScore++;
                        if (mScrollDetectionScore >= SCROLL_SCORE_TO_REACH)
                            mCurrentMode = MODE_SCROLL;
                    }
                }

                switch (mCurrentMode) {
                    case MODE_PINCH_ZOOM:
                        if (scaleGestureDetector != null)
                            handled |= scaleGestureDetector.onTouchEvent(ev);
                        break;

                    case MODE_SCROLL:
                        handled = mListener.onDoubleTouchScroll(mCurrentDownEvent, ev);
                        break;

                    default:
                        handled = true;
                        break;
                }

                break;

            case MotionEvent.ACTION_UP:
                // fingers were not removed equally? cancel
                if (mPreviousPointerUpEvent != null && (ev.getEventTime() - mPreviousPointerUpEvent.getEventTime()) > DOUBLE_TOUCH_TIMEOUT) {
                    mPreviousPointerUpEvent.recycle();
                    mPreviousPointerUpEvent = null;
                    cancel();
                    break;
                }

                // detection cancelled or not active?
                if (mCancelDetection || !mDoubleInProgress)
                    break;

                boolean hasTapEvent = mHandler.hasMessages(TAP);
                MotionEvent currentUpEvent = MotionEvent.obtain(ev);
                if (mCurrentMode == MODE_UNKNOWN && hasTapEvent)
                    handled = mListener.onDoubleTouchSingleTap(mCurrentDoubleDownEvent);
                else if (mCurrentMode == MODE_PINCH_ZOOM)
                    handled = scaleGestureDetector.onTouchEvent(ev);

                if (mPreviousUpEvent != null)
                    mPreviousUpEvent.recycle();

                // Hold the event we obtained above - listeners may have changed the original.
                mPreviousUpEvent = currentUpEvent;
                handled |= mListener.onDoubleTouchUp(ev);
                break;

            case MotionEvent.ACTION_CANCEL:
                cancel();
                break;
        }

        if ((action == MotionEvent.ACTION_MOVE) && handled == false)
            handled = true;

        return handled;
    }

    private void cancel() {
        mHandler.removeMessages(TAP);
        mCurrentMode = MODE_UNKNOWN;
        mCancelDetection = true;
        mDoubleInProgress = false;
    }

    // returns true of the distance between the two pointers changed
    private boolean pointerDistanceChanged(MotionEvent oldEvent, MotionEvent newEvent) {
        int deltaX1 = Math.abs((int) oldEvent.getX(0) - (int) oldEvent.getX(1));
        int deltaX2 = Math.abs((int) newEvent.getX(0) - (int) newEvent.getX(1));
        int distXSquare = (deltaX2 - deltaX1) * (deltaX2 - deltaX1);

        int deltaY1 = Math.abs((int) oldEvent.getY(0) - (int) oldEvent.getY(1));
        int deltaY2 = Math.abs((int) newEvent.getY(0) - (int) newEvent.getY(1));
        int distYSquare = (deltaY2 - deltaY1) * (deltaY2 - deltaY1);

        return (distXSquare + distYSquare) > mPointerDistanceSquare;
    }

    /**
     * The listener that is used to notify when gestures occur.
     * If you want to listen for all the different gestures then implement
     * this interface. If you only want to listen for a subset it might
     * be easier to extend {@link SimpleOnGestureListener}.
     */
    public interface OnDoubleGestureListener {

        /**
         * Notified when a multi tap event starts
         */
        boolean onDoubleTouchDown(MotionEvent e);

        /**
         * Notified when a multi tap event ends
         */
        boolean onDoubleTouchUp(MotionEvent e);

        /**
         * Notified when a tap occurs with the up {@link MotionEvent}
         * that triggered it.
         *
         * @param e The up motion event that completed the first tap
         * @return true if the event is consumed, else false
         */
        boolean onDoubleTouchSingleTap(MotionEvent e);

        /**
         * Notified when a scroll occurs with the initial on down {@link MotionEvent} and the
         * current move {@link MotionEvent}. The distance in x and y is also supplied for
         * convenience.
         *
         * @param e1        The first down motion event that started the scrolling.
         * @param e2        The move motion event that triggered the current onScroll.
         * @param distanceX The distance along the X axis that has been scrolled since the last
         *                  call to onScroll. This is NOT the distance between {@code e1}
         *                  and {@code e2}.
         * @param distanceY The distance along the Y axis that has been scrolled since the last
         *                  call to onScroll. This is NOT the distance between {@code e1}
         *                  and {@code e2}.
         * @return true if the event is consumed, else false
         */
        boolean onDoubleTouchScroll(MotionEvent e1, MotionEvent e2);
    }
    
    /*
    private void dumpEvent(MotionEvent event) {
		   String names[] = { "DOWN" , "UP" , "MOVE" , "CANCEL" , "OUTSIDE" ,
		      "POINTER_DOWN" , "POINTER_UP" , "7?" , "8?" , "9?" };
		   StringBuilder sb = new StringBuilder();
		   int action = event.getAction();
		   int actionCode = action & MotionEvent.ACTION_MASK;
		   sb.append("event ACTION_" ).append(names[actionCode]);
		   if (actionCode == MotionEvent.ACTION_POINTER_DOWN
		         || actionCode == MotionEvent.ACTION_POINTER_UP) {
		      sb.append("(pid " ).append(
		      action >> MotionEvent.ACTION_POINTER_ID_SHIFT);
		      sb.append(")" );
		   }
		   sb.append("[" );
		   for (int i = 0; i < event.getPointerCount(); i++) {
		      sb.append("#" ).append(i);
		      sb.append("(pid " ).append(event.getPointerId(i));
		      sb.append(")=" ).append((int) event.getX(i));
		      sb.append("," ).append((int) event.getY(i));
		      if (i + 1 < event.getPointerCount())
		         sb.append(";" );
		   }
		   sb.append("]" );
		   Log.d("DoubleDetector", sb.toString());
		}	
    */

    private class GestureHandler extends Handler {
        GestureHandler() {
            super();
        }

        GestureHandler(Handler handler) {
            super(handler.getLooper());
        }

    }

}