Горизонтальный салфетка не обнаружена в родительском элементе ScrollView

Возможный дубликат:
Обнаружение жестов и проблема с прокруткой

EDIT: вопрос с полным кодом спросил здесь.


У меня есть макет с ребенком. Я установил прослушиватель жестов для определения горизонтального прокрутки на макете. Когда макет LinearLayout правильно распознается, но когда это ScrollView, это не так. Я предполагаю, что жест сначала обнаруживается ScrollView и не распространяется на его потомков, но я не знаю, как его решить.

Здесь мой макет:

<?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="fill_parent"
    android:orientation="vertical">
    <ImageView android:layout_width="320dp" android:layout_height="30dp"
            android:src="@drawable/header"/>
    <ScrollView android:layout_width="fill_parent" android:layout_height="wrap_content">
        <!-- stuff -->
    </ScrollView>
</LinearLayout>

Я установил для своего макета следующий слушатель:

class ProductGestureListener extends SimpleOnGestureListener {

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

    @Override
    public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) {

        final int SWIPE_MIN_DISTANCE = 120;
        final int SWIPE_MAX_OFF_PATH = 250;
        final int SWIPE_THRESHOLD_VELOCITY = 200;            

        try {
            if (Math.abs(e1.getY() - e2.getY()) > SWIPE_MAX_OFF_PATH)
                return false;
            if(e2.getX() - e1.getX() > SWIPE_MIN_DISTANCE && Math.abs(velocityX) > SWIPE_THRESHOLD_VELOCITY) {                   
                // show previous item
            }  else if (e1.getX() - e2.getX() > SWIPE_MIN_DISTANCE && Math.abs(velocityX) > SWIPE_THRESHOLD_VELOCITY) {
               // show next item
            }
        } catch (Exception e) {
        }           
        return false;
    }
}

Ответ 1

Мне пришлось добавить

@Override
public boolean dispatchTouchEvent(MotionEvent ev){
    super.dispatchTouchEvent(ev);    
    return productGestureDetector.onTouchEvent(ev); 
}

Добавьте этот метод в свой класс активности, который использует swiper так же, как метод onCreate.

Ответ 2

Если вы хотите, чтобы вся Activity быть Перелистываемыми по горизонтали вы можете использовать следующий как супер класс для вашей Activity:

public abstract class SwipeActivity extends Activity {
   private static final int SWIPE_MIN_DISTANCE = 120;
   private static final int SWIPE_MAX_OFF_PATH = 250;
   private static final int SWIPE_THRESHOLD_VELOCITY = 200;
   private GestureDetector gestureDetector;

   @Override
   protected void onCreate( Bundle savedInstanceState ) {
      super.onCreate( savedInstanceState );
      gestureDetector = new GestureDetector( new SwipeDetector() );
   }

   private class SwipeDetector extends SimpleOnGestureListener {
      @Override
      public boolean onFling( MotionEvent e1, MotionEvent e2, float velocityX, float velocityY ) {

         // Check movement along the Y-axis. If it exceeds SWIPE_MAX_OFF_PATH,
         // then dismiss the swipe.
         if( Math.abs( e1.getY() - e2.getY() ) > SWIPE_MAX_OFF_PATH )
            return false;

         // Swipe from right to left.
         // The swipe needs to exceed a certain distance (SWIPE_MIN_DISTANCE)
         // and a certain velocity (SWIPE_THRESHOLD_VELOCITY).
         if( e1.getX() - e2.getX() > SWIPE_MIN_DISTANCE && Math.abs( velocityX ) > SWIPE_THRESHOLD_VELOCITY ) {
            next();
            return true;
         }

         // Swipe from left to right.
         // The swipe needs to exceed a certain distance (SWIPE_MIN_DISTANCE)
         // and a certain velocity (SWIPE_THRESHOLD_VELOCITY).
         if( e2.getX() - e1.getX() > SWIPE_MIN_DISTANCE && Math.abs( velocityX ) > SWIPE_THRESHOLD_VELOCITY ) {
            previous();
            return true;
         }

         return false;
      }
   }

   @Override
   public boolean dispatchTouchEvent( MotionEvent ev ) {
      // TouchEvent dispatcher.
      if( gestureDetector != null ) {
         if( gestureDetector.onTouchEvent( ev ) )
            // If the gestureDetector handles the event, a swipe has been
            // executed and no more needs to be done.
            return true;
      }
      return super.dispatchTouchEvent( ev );
   }

   @Override
   public boolean onTouchEvent( MotionEvent event ) {
      return gestureDetector.onTouchEvent( event );
   }

   protected abstract void previous();

   protected abstract void next();
}

Все, что вам нужно сделать, это реализовать next() методы next() и previous() после расширения SwipeActivity.

Ответ 3

Последовательность событий для android - это

пользователь выполняет какое-то действие → действие передается родительскому → если родитель обрабатывает действие, то действие уничтожается → else действие передается дочернему → если ребенок обрабатывает действие, то действие уничтожается → else действие передается на

этот процесс продолжается до тех пор, пока действие не будет уничтожено или все дети не получат действие, и никто из них не обработал его.

Чтобы обнаружить горизонтальное прокручивание в виде прокрутки и передать его дочернему элементу, а не просматривать прокрутку, оно должно быть перехвачено.

т.е. пользователь выполняет какое-либо действие → действие передается родительскому → если горизонтальное прокручивание переходит к дочернему → else, вид прокрутки обрабатывает действие

чтобы сделать это (как выровнено в верхнем ответе здесь: HorizontalScrollView в области управления окнами ScrollView) детектор жестов используется в режиме прокрутки с единственной целью - определить, является ли этот жест горизонтальным или вертикальным.

Если жест горизонтален, мы хотим перехватить событие и передать его ребенку.

Вам нужно создать собственное прокручивание, а затем реализовать там же детектор жестов и называть его onInterceptTouch(). Возвращаясь к истине или false, мы можем сказать, нужно ли здесь принимать это событие, все, что вам нужно, находится в этой ссылке выше.