检测ListView上的手势有问题

时间:2011-01-13 20:58:47

标签: android listview gesture viewflipper gesturedetector

我有一个包含ViewFlipper的Activity。 ViewFlipper包含2个布局,这两个布局基本上只是ListViews。

所以这里的想法是我有两个列表并且从一个列表导航到另一个列表我将使用水平滑动。

我有那个工作。但是,当滑动开始执行时,您的手指所在的列表项目,也会长按该项目。

以下是我的相关代码:

public class MyActivity extends Activity implements OnItemClickListener, OnClickListener {

    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 mGestureDetector;
    View.OnTouchListener mGestureListener;

    class MyGestureDetector extends SimpleOnGestureListener {
        @Override
        public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) {
            try {
                if (Math.abs(e1.getY() - e2.getY()) > SWIPE_MAX_OFF_PATH)
                    return false;
                // right to left swipe
                if(e1.getX() - e2.getX() > SWIPE_MIN_DISTANCE && Math.abs(velocityX) > SWIPE_THRESHOLD_VELOCITY) {
                    if (mCurrentScreen != SCREEN_SECONDLIST) {
                        mCurrentScreen = SCREEN_SECONDLIST;
                        mFlipper.setInAnimation(inFromRightAnimation());
                        mFlipper.setOutAnimation(outToLeftAnimation());
                        mFlipper.showNext();
                        updateNavigationBar();
                    }
                }
                else if (e2.getX() - e1.getX() > SWIPE_MIN_DISTANCE && Math.abs(velocityX) > SWIPE_THRESHOLD_VELOCITY) {
                    if (mCurrentScreen != SCREEN_FIRSTLIST) {
                        mCurrentScreen = SCREEN_FIRSTLIST;
                        mFlipper.setInAnimation(inFromLeftAnimation());
                        mFlipper.setOutAnimation(outToRightAnimation());
                        mFlipper.showPrevious();
                        updateNavigationBar();
                    }
                }
            } catch (Exception e) {
                // nothing
            }
            return true;
        }
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        if (mGestureDetector.onTouchEvent(event))
            return true;
        else
            return false;
    }





    ViewFlipper mFlipper;

    private int mCurrentScreen = SCREEN_FIRSTLIST;

    private ListView mList1;
    private ListView mList2;

    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        requestWindowFeature(Window.FEATURE_NO_TITLE);
        setContentView(R.layout.layout_flipper);

        mFlipper = (ViewFlipper) findViewById(R.id.flipper);

        mGestureDetector = new GestureDetector(new MyGestureDetector());
        mGestureListener = new View.OnTouchListener() {
            public boolean onTouch(View v, MotionEvent event) {
                if (mGestureDetector.onTouchEvent(event)) {
                    return true;
                }
                return false;
            }
        };

        // set up List1 screen

        mList1List = (ListView)findViewById(R.id.list1);
        mList1List.setOnItemClickListener(this);
        mList1List.setOnTouchListener(mGestureListener);

        // set up List2 screen
        mList2List = (ListView)findViewById(R.id.list2);
        mList2List.setOnItemClickListener(this);
        mList2List.setOnTouchListener(mGestureListener);

    }

    …
}

如果我改变“return true;”从GestureDetector声明“返回false;”,我没有得到长按。不幸的是,我经常点击。

有谁知道如何解决这个问题?

6 个答案:

答案 0 :(得分:7)

只是用完全不同的方法提出一个完全不同的答案......

我做了一个名为SwipeView的自定义视图,它是开源等等等等等等等等等等。它应该让你做你想做的事情就像去:

mSwipeView.addChild(myList1);
mSwipeView.addChild(myList2);

您不必乱用手势探测器或任何东西,滑动应该按照您的手指操作...

(这听起来像一个可怕的可怕广告,请原谅。这是漫长的一天;)

有一些很棒的链接:

http://jasonfry.co.uk/?id=23

http://jasonfry.co.uk/?id=24

http://jasonfry.co.uk/?id=28

答案 1 :(得分:1)

由于您使用的是GestureDetector,因此需要实施SimpleGestureDetector.onLongPress(MotionEvent e)来处理长时间点击。但是,您无法知道从那里被长按的列表项,因此您需要从正常的onItemLongClick()中记住它。

class MyGestureDetector extends SimpleOnGestureListener {
    ...
    @Override public void onLongPress(MotionEvent e) {
        if (longClickedItem != -1) {
           Toast.makeText(MainActivity.this, "Long click!", Toast.LENGTH_LONG).show();
        }           
    }
}

int longClickedItem = -1;

@Override
public boolean onItemLongClick(AdapterView<?> list, View view, int position, long id) {
    longClickedItem = position;
    return true;
}

@Override
public boolean onTouchEvent(MotionEvent event) {

    // Reset longclick item if new touch is starting
    if (event.getAction()==MotionEvent.ACTION_DOWN) {
        longClickedItem = -1;
    }

    if (mGestureDetector.onTouchEvent(event))
        return true;
    else
        return false;
}

我已对此进行了测试,似乎工作正常。

答案 2 :(得分:1)

找你的gestureDetector并设置

 mGestureDetector.setIsLongpressEnabled(false);

这样可以防止检测长按事件。

答案 3 :(得分:0)

[编辑]划痕,完全错了。

我认为您必须更新此部分,并检测您是向左还是向右滚动,如果您处于滑动状态,则返回true。或者至少那是我先看的地方。

 mGestureListener = new View.OnTouchListener() {
            public boolean onTouch(View v, MotionEvent event) {
                if (mGestureDetector.onTouchEvent(event)) {
                    return true;
                }
                return false;
            }
        };

继续我糟糕的例子,抱歉。

在onTouch中我相信你可以测试event.getAction()并确定是否发生了长时间的点击。如果它有,并且你的投掷动作,则返回true以捕获长按。

害怕这更像是一个建议,而不是一个明确的答案。

还可以在[方法]上检查可以从SimpleOnGestureListener覆盖的另一个。刚检查过并

@Override
public void onLongPress(MotionEvent e) {
    // TODO Auto-generated method stub
    super.onLongPress(e);
}

可能是您可以尝试的东西。

答案 4 :(得分:0)

我开发了一个派生自horizo​​ntalScrollView的自定义组件。我有listViews作为子视图和滑动没有长按。它由listAdapter支持,它可以是你想要的任何东西(ArrayAdapter,CursorAdapter ......)。它最多可以加载三个子视图,然后在翻转时,只需在两侧交换它们。其中大部分都来自ListView,需要重构。它太长了,不能直接在这里发布。

http://pastie.org/1480091

答案 5 :(得分:0)

删除getListView().setOnItemLongClickListener

在课堂上添加 MyGestureDetector

public void onLongPress(MotionEvent e) {
    final int position = getListView().pointToPosition((int) e.getX(),
            (int) e.getY());
    // what you want do
    super.onLongPress(e);
}