Android: изменение вкладок в событии Fling Gesture — не работает для ListActivity внутри TabActivity

У меня есть вкладка с тремя вкладками. Одна из вкладок содержит ListActivity, а две другие содержат простую активность. И я реализовал OnGestureListener, чтобы изменить вкладку в событии броска.

События onTouchEvent и onFling не вызываются для вкладки с ListActivity. Но отлично работает для Simple Activity (без списков).

Вот мой код:

Основной класс активности:

public class GestureTest extends TabActivity implements OnGestureListener {
    private TabHost tabHost;
    private GestureDetector gestureScanner;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.testmainlayoutforgesture);
        gestureScanner = new GestureDetector(this);
        addTabs();
    }

    private void addTabs() {
        Intent i1 = new Intent().setClass(this, SimpleListActivity.class);
        Intent i2 = new Intent().setClass(this, SimpleActivity.class);
        Intent i3 = new Intent().setClass(this, SimpleActivity.class);

        tabHost = getTabHost();
        tabHost.addTab(tabHost.newTabSpec("First").setIndicator("First").setContent(i1));
        tabHost.addTab(tabHost.newTabSpec("Second").setIndicator("Second").setContent(i2));
        tabHost.addTab(tabHost.newTabSpec("Third").setIndicator("Third").setContent(i3));

        tabHost.setCurrentTab(0);
    }

    @Override
    public boolean onTouchEvent(MotionEvent me) {
        return gestureScanner.onTouchEvent(me);
    }

    @Override
    public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) {
        float dispX = e2.getX() - e1.getX();
        float dispY = e2.getY() - e1.getY();
        if (Math.abs(dispX) >= 200 && Math.abs(dispY) <= 100) {
            // swipe ok
            if (dispX > 0) {
                // L-R swipe
                System.out.println("L-R swipe");
                changeLtoR();
            } else {
                // R-L swipe
                System.out.println("R-L swipe");
            }
        }
        return true;
    }

    private void changeLtoR() {
        int curTab = tabHost.getCurrentTab();
        int nextTab = ((curTab + 1) % 3);
        tabHost.setCurrentTab(nextTab);

    }
//other interface methods    
}

SimpleListActivity.java:

public class SimpleListActivity extends ListActivity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        List<String> list1Strings = new ArrayList<String>();
        list1Strings.add("List 11");
        list1Strings.add("List 12");
        list1Strings.add("List 13");
        list1Strings.add("List 14");
        setListAdapter(new ArrayAdapter(this, android.R.layout.simple_list_item_1, list1Strings));
    }
}

SimpleActivity.java:

public class SimpleActivity extends Activity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.simplelayouttest);
    }
}

макет testmainlayoutforgesture.xml:

<?xml version="1.0" encoding="utf-8"?>
<TabHost
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@android:id/tabhost"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent">
    <LinearLayout
        android:orientation="vertical"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent">
        <TabWidget
            android:id="@android:id/tabs"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content" />
        <FrameLayout
            android:id="@android:id/tabcontent"
            android:layout_width="fill_parent"
            android:layout_height="fill_parent">
        </FrameLayout>
    </LinearLayout>
</TabHost>

макет simplelayouttest.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent">
    <TextView
        android:id="@+id/helloTv"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="Hello Hello" />
</LinearLayout>

Я не мог понять, что здесь не так.


person gtiwari333    schedule 26.09.2011    source источник


Ответы (1)


Попробуйте добавить это в свою TabActivity:

@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
    if (gestureScanner != null) {
        if (gestureScanner.onTouchEvent(ev))
            return true;
    }
    return super.dispatchTouchEvent(ev);
}

Реализация этого сообщит вашему устройству, что жестовый сканер использовал событие onFling.

РЕДАКТИРОВАТЬ еще раз:

Вам также нужно будет немного изменить свой метод onFling на это:

    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;

    @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) {
            // do stuff
            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) {
            // do stuff
            return true;
        }

        return false;
    }
}    

Другая возможность может заключаться в том, что ваш ListView не заполняет экран, а «пустое» пространство недоступно для щелчка, и вы бросаете это «пустое» пространство, событие касания не будет зарегистрировано и отправлено. Работает ли, если кинуть поверх ListView?

person kaspermoerch    schedule 26.09.2011
comment
Спасибо, это сработало. Я не мог понять твоего «другого варианта». Да, мой исходный код работал с событием броска, если я кидаю заголовок своего приложения (раздел, где отображается имя приложения) - person gtiwari333; 26.09.2011
comment
Но я обнаружил еще одну проблему с этим подходом. Ни один из предметов нельзя трогать. Я имею в виду, что я не могу выбирать (касаться) элементы списка и вкладки. - person gtiwari333; 26.09.2011
comment
Не могли бы вы опубликовать решение здесь тогда? - person kaspermoerch; 26.09.2011
comment
я только что добавил ваш код. жест броска работал нормально. но возникла другая проблема. - person gtiwari333; 26.09.2011
comment
пожалуйста, обратитесь к моему предыдущему комментарию: ни один из предметов не является сенсорным. Я имею в виду, что я не могу выбирать (касаться) элементы списка и вкладки. Нет проблем с событием onFling. он работает нормально. - person gtiwari333; 26.09.2011
comment
Я понимаю это и считаю, что причина, по которой ни один из элементов не является сенсорным, заключается в том, что метод onFling может потреблять все сенсорные события. - person kaspermoerch; 26.09.2011