77

以下のスキームを検討してください(私の問題をよりよく理解するために)。 ここに画像の説明を入力

ご覧のとおり、パディングで囲まれたリスト ビューを検討しています。ここで、ユーザーがリストビュー項目を押した場合、アクションとして明るい青色の背景色を提供しました。現在、私のアプリケーションはonTouchイベント自体を処理して、次のようなアクションを決定しています

  • クリック
  • 左から右へスワイプ
  • 右から左へスワイプ

これが私のコードです。

public boolean onTouch(View v, MotionEvent event) {
        if(v == null)
        {
            mSwipeDetected = Action.None;
            return false;
        }
        switch (event.getActionMasked()) {
        case MotionEvent.ACTION_DOWN: {
            downX = event.getRawX();
            downY = event.getRawY();
            mSwipeDetected = Action.Start;

         // Find the child view that was touched (perform a hit test)
            Rect rect = new Rect();
            int childCount = listView.getChildCount();
            int[] listViewCoords = new int[2];
            listView.getLocationOnScreen(listViewCoords);
            int x = (int) event.getRawX() - listViewCoords[0];
            int y = (int) event.getRawY() - listViewCoords[1];
            View child;
            for (int i = 0; i < childCount; i++) {
                child = listView.getChildAt(i);
                child.getHitRect(rect);
                if (rect.contains(x, y)) {
                    mDownView = child;
                    break;
                }
            }


            return false; // allow other events like Click to be processed
        }
        case MotionEvent.ACTION_MOVE: {
            upX = event.getRawX();
            upY = event.getRawY();
            float deltaX=0,deltaY=0;
             deltaX = downX - upX;
             deltaY = downY - upY;

                if(deltaY < VERTICAL_MIN_DISTANCE)
                {
                            setTranslationX(mDownView, -(deltaX));
                            setAlpha(mDownView, Math.max(0f, Math.min(1f, 1f - 2f * Math.abs(deltaX) / listView.getWidth())));
                            return false;
                }
                else
                {
                    forceBringBack(v);
                }

                          return false;              

        }
        case MotionEvent.ACTION_UP:
        {

             stopX = event.getX();
             float stopValueY = event.getRawY() - downY;             
             float stopValue = stopX - downX;

             if(!mDownView.isPressed())
             {
                 forceBringBack(mDownView);
                 return false;
             }             

             boolean dismiss = false;
             boolean dismissRight = false;


             if(Math.abs(stopValue)<10)
             {
                 mSwipeDetected = Action.Start;
             }
             else
             {
                 mSwipeDetected = Action.None;

             }
             String log = "";
             Log.d(log, "Here is Y" + Math.abs(stopValueY));
             Log.d(log, "First Comparison of Stop Value > with/4" + (Math.abs(stopValue) > (listView.getWidth() /4)));
             Log.d(log, "Second Comparison " + (Math.abs(stopValueY)<VERTICAL_MIN_DISTANCE));
             Log.d(log, "Action Detected is " + mSwipeDetected + " with Stop Value  " + stopValue);

             if((Math.abs(stopValue) > (listView.getWidth() /4))&&(Math.abs(stopValueY)<VERTICAL_MIN_DISTANCE))
             {
                 dismiss = true;
                 dismissRight = stopValue > 0;

                 if(stopValue>0)
                 {
                 mSwipeDetected = Action.LR;

                 }
                 else
                     mSwipeDetected = Action.RL;
             }
             Log.d(log, "Action Detected is " + mSwipeDetected + " with Stop Value after dissmiss" + stopValue);

             if(dismiss)
             {
                 if(dismissRight)
                     mSwipeDetected = Action.LR;
                 else
                     mSwipeDetected = Action.RL;
                 animate(mDownView)
                 .translationX(dismissRight ? listView.getWidth() : - listView.getWidth())
                 .alpha(0)
                 .setDuration(mAnimationTime)
                 .setListener(new AnimatorListenerAdapter() {
                     public void onAnimationEnd(Animator animation)
                     {

                     }
                });
             }
             else
             {
                 animate(mDownView)
                 .translationX(0)
                 .alpha(1)
                 .setDuration(mAnimationTime)
                 .setListener(null);
             }


             break;           

        }
        }
        return false;
    }

ご覧のとおり、MotionEvent.ACTION_UP で実行されたアクションを特定し、それに応じて Enum Action の値を設定します。ユーザーがリスト ビューの境界を越えていない場合、このロジックは魅力的に機能します。

ここで、ユーザーがスライドしている間 (または具体的には)、リスト項目に沿って指を青からオレンジに移動すると、MotionEvent.ACTION_UP がリストビューに渡されないため、コードが決定を下さず、translationX が原因で発生します。 () メソッドと setAlpha() の場合、この場合は Action が決定されないため、その特定のリスト項目は空白になります。

毎回ビューを膨張させていないため、問題はここで止まりません。同じ translateX() 行が毎回膨張し、空白/ホワイトリスト項目が複数発生するためです。

MotionEvent.ACTION_UP に遭遇しなかったとしても、何か決定を下すことができるようにするためにできることはありますか?

4

4 に答える 4

31

また、特定の状況 (画面の回転など) ではジェスチャがキャンセルされる場合があり、その場合は aMotionEvent.ACTION_UPが送信されないことにも注意してください。代わりに aMotionEvent.ACTION_CANCELが送信されます。したがって、通常のアクション switch ステートメントは次のようになります。

switch (event.getActionMasked()) {
    case MotionEvent.ACTION_DOWN:
        // check if we want to handle touch events, return true
        // else don't handle further touch events, return false
    break;

    // ... handle other cases

    case MotionEvent.ACTION_UP:
    case MotionEvent.ACTION_CANCEL:
        // finish handling touch events
        // note that these methods won't be called if 'false' was returned
        // from any previous events related to the gesture
    break;
}
于 2016-04-18T05:09:35.237 に答える
8

return true;ケースMotionEvent.ACTION_DOWN:に追加しても最終的に問題が解決するとは思いません。return false魔法のように仕事をすることができた状況を複雑にしました。

注目すべきMotionEvent.ACTION_DOWN: /*something*/ return true;点は、onClickListenerm を含め、ビューで使用可能な他のリスナー コールバックをすべてブロックする一方で、適切にreturn falseinMotionEvent.ACTION_UP:を指定すると、MotionEvent が適切な宛先に伝播されるのに役立ちます。

彼の元のコード ソースへの参照: https://github.com/romannurik/android-swipetodismiss

于 2014-06-19T01:48:17.670 に答える