检测视图上的滑动手势和触摸事件

Detect swipe gestures and touch events on a view

Android工作室,Java

有没有办法在单视图上检测触摸、触摸释放和滑动手势?我有一个按钮,我需要能够在上面收听所有这些内容。

我正在寻找 listeners/detectors 最终调用这些方法的解决方案,以便我可以轻松地使用它们:

public void buttonPressedDown() {

}

public void buttonReleased() {

}

public void swipeUp() {

}

public void swipeRight() {

}

public void swipeDown() {

}

public void swipeLeft() {

}

在开发 ios 应用程序时,这在 xcode 中很容易实现,但我似乎找不到 android 的方法。

额外信息: 这是我试过的:

    timerButton.setOnTouchListener(new View.OnTouchListener() { //Detecting a press down
        @Override
        public boolean onTouch(View v, MotionEvent event) {

            timerDown();

            return false;
        }
    });

    timerButton.setOnTouchListener(new OnSwipeTouchListener(timerActivity.this) {//Detecting swipes

        public void onSwipeTop() {

            showStats();

        }

        public void onSwipeBottom() {

            showType();

        }

        public void onSwipeRight() {

            showMenu();

        }

        public void onSwipeLeft() {

            showTimes();

        }

    });


    .
    .
    .


    public void timerButtonPressed(View view) {//Detecting press releases
        timerUp();
    }

这似乎是我要找的东西,只是它不起作用。在同一个视图上设置两个 onTouchListeners 时,只有一个侦听器被调用。当我测试它时,要么只有滑动检测有效,要么只有按压检测有效。

求救!?

编辑: 我试过这个:

OnSwipeTouchListener.java

public class OnSwipeTouchListener implements View.OnTouchListener {

    private final GestureDetector gestureDetector;

    public OnSwipeTouchListener(Context context) {
        gestureDetector = new GestureDetector(context, new GestureListener());
    }

    public void onSwipeLeft() {
    }

    public void onSwipeRight() {
    }

    public boolean onTouch(View v, MotionEvent event) {
        return gestureDetector.onTouchEvent(event);
    }

    private final class GestureListener extends GestureDetector.SimpleOnGestureListener {

        private static final int SWIPE_DISTANCE_THRESHOLD = 100;
        private static final int SWIPE_VELOCITY_THRESHOLD = 100;

        @Override
        public boolean onDown(MotionEvent e) {
            return true;
        }

        @Override
        public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) {
            float distanceX = e2.getX() - e1.getX();
            float distanceY = e2.getY() - e1.getY();
            if (Math.abs(distanceX) > Math.abs(distanceY) && Math.abs(distanceX) > SWIPE_DISTANCE_THRESHOLD && Math.abs(velocityX) > SWIPE_VELOCITY_THRESHOLD) {
                if (distanceX > 0)
                    onSwipeRight();
                else
                    onSwipeLeft();
                return true;
            }
            return false;
        }
    }
}

在我的 activity 中:

timerButton.setOnTouchListener(new View.OnTouchListener() {
        @Override
        public boolean onTouch(View v, MotionEvent event) {

            OnSwipeTouchListener onSwipeTouchListener = new OnSwipeTouchListener(timerActivity.this);
            boolean result = onSwipeTouchListener.onTouch(v, event);
            // check if swipe was detected:
            if (result) return true;

            // Otherwise, detect other types of motion event
            switch (event.getAction()) {
                case MotionEvent.ACTION_DOWN:
                    // touch down

                    timerDown();

                    break;
                case MotionEvent.ACTION_UP:
                    // touch released

                    timerUp();

                    break;
            }

            return false;
        }
    });

但是我如何从这里检测到滑动?此外,当我 运行 应用程序时,它甚至不再检测到触摸。

我也在我的activity中尝试过这个:

timerButton.setOnTouchListener(new OnSwipeTouchListener(timerActivity.this) {

        public void onSwipeLeft() {
            Log.i("LEFT", "LEFT");
        }

        public void onSwipeRight() {
            Log.i("RIGHT", "RIGHT");
        }

        @Override
        public boolean onTouch(View v, MotionEvent event) {

            switch (event.getAction()) {
                case MotionEvent.ACTION_DOWN:
                    // touch down

                    timerDown();

                    break;
                case MotionEvent.ACTION_UP:
                    // touch released

                    timerUp();

                    break;
            }

            return false;
        }
    });

现在它可以检测触摸但不会检测滑动。如果我删除了上面的 @override onTouch 方法,但仍然有 onSwipe 方法,则会检测到滑动。这怎么这么复杂?为什么它不能同时检测到滑动和触摸,我该如何解决?

第二次调用 setOnTouchListener 会覆盖您之前的侦听器。您可以通过以下方式检测触摸和手势:

timerButton.setOnTouchListener(new View.OnTouchListener() { 
        @Override
        public boolean onTouch(View v, MotionEvent event) {
            // deliver motion event to your swipe listener
            boolean result = mySwipeListener.onTouch(event);
            // check if swipe was detected:
            if (result) return true;

            // Otherwise, detect other types of motion event
            switch (event.getAction()) {
                case MotionEvent.ACTION_DOWN:
                    // touch down
                    break;
                case MotionEvent.ACTION_UP:
                    // touch released
                    break;
                // etc... see this link for more motion events https://developer.android.com/reference/android/view/MotionEvent
            }

            return false;
        }
    });

这里有一个实现滑动监听器的例子:Android: How to handle right to left swipe gestures

我找到了解决方案。

在OnSwipeTouchListener.java中:

public class OnSwipeTouchListener implements View.OnTouchListener {

private final GestureDetector gestureDetector;

public OnSwipeTouchListener(Context context) {
    gestureDetector = new GestureDetector(context, new GestureListener());
}

public void onSwipeLeft() {
}

public void onSwipeRight() {
}

public void onSwipeBottom() {
}

public void onSwipeTop() {
}

public void onDownTouch() {

}

public boolean onTouch(View v, MotionEvent event) {
    return gestureDetector.onTouchEvent(event);
}

private final class GestureListener extends GestureDetector.SimpleOnGestureListener {

    private static final int SWIPE_DISTANCE_THRESHOLD = 100;
    private static final int SWIPE_VELOCITY_THRESHOLD = 100;

    @Override
    public boolean onDown(MotionEvent e) {
        onDownTouch();
        return false;
    }

    @Override
    public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) {
        boolean result = false;
        try {
            float diffY = e2.getY() - e1.getY();
            float diffX = e2.getX() - e1.getX();
            if (Math.abs(diffX) > Math.abs(diffY)) {
                if (Math.abs(diffX) > SWIPE_DISTANCE_THRESHOLD && Math.abs(velocityX) > SWIPE_VELOCITY_THRESHOLD) {
                    if (diffX > 0) {
                        onSwipeRight();
                    } else {
                        onSwipeLeft();
                    }
                    result = true;
                }
            }
            else if (Math.abs(diffY) > SWIPE_DISTANCE_THRESHOLD && Math.abs(velocityY) > SWIPE_VELOCITY_THRESHOLD) {
                if (diffY > 0) {
                    onSwipeBottom();
                } else {
                    onSwipeTop();
                }
                result = true;
            }
        } catch (Exception exception) {
            exception.printStackTrace();
        }
        return result;
    }
}

}

在activity中:

检测按钮释放按下:(点击)

public void timerButtonPressed(View view) {

}

要检测滑动和按下:

timerButton.setOnTouchListener(new OnSwipeTouchListener(timerActivity.this) {

        public void onSwipeLeft() {

        }

        public void onSwipeRight() {

        }

        public void onSwipeBottom() {

        }

        public void onSwipeTop() {

        }

        public void onDownTouch() {

        }
    });