2015-12-01 35 views
9

スワイ・バーをスワイプすることによってユーザがスワイクを解除できないようにする方法はありますか?snackbarのスワイプ・スワイプ・ディス・ミスの動作を無効にする方法

私は、ネットワークログイン中にスナックバーを表示するアプリを持っていますが、私はそれを却下しないようにしたいと思います。

ニコラDespotoskiの提案によると、私は両方のソリューションを試してきました:

public class DummyBehavior extends CoordinatorLayout.Behavior<View>{ 

    /** 
    * Debugging tag used by the Android logger. 
    */ 
    protected final static String TAG = 
      DummyBehavior.class.getSimpleName(); 



    public DummyBehavior() { 
     Log.i(TAG, "Dummy behavior created"); 
     StackTraceElement[] stackTrace = Thread.currentThread().getStackTrace(); 
     Log.i(TAG, "Method " + stackTrace[2].getMethodName()); 

    } 

    public DummyBehavior(Context context, AttributeSet attrs) { 
     super(context, attrs); 
     Log.i(TAG, "Dummy behavior created"); 

    } 

    @Override 
    public boolean onInterceptTouchEvent(CoordinatorLayout parent, View child, MotionEvent ev) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
     return false; 
    } 

    @Override 
    public boolean onTouchEvent(CoordinatorLayout parent, View child, MotionEvent ev) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
     return false; 
    } 

    @Override 
    public boolean blocksInteractionBelow(CoordinatorLayout parent, View child) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
     return false; 
    } 

    @Override 
    public boolean layoutDependsOn(CoordinatorLayout parent, View child, View dependency) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
     return false; 
    } 

    @Override 
    public boolean onDependentViewChanged(CoordinatorLayout parent, View child, View dependency) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
     return false; 
    } 

    @Override 
    public void onDependentViewRemoved(CoordinatorLayout parent, View child, View dependency) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
    } 

    @Override 
    public boolean isDirty(CoordinatorLayout parent, View child) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
     return false; 
    } 

    @Override 
    public boolean onMeasureChild(CoordinatorLayout parent, View child, int parentWidthMeasureSpec, int widthUsed, int parentHeightMeasureSpec, int heightUsed) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
     return false; 
    } 

    @Override 
    public boolean onLayoutChild(CoordinatorLayout parent, View child, int layoutDirection) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
     return false; 
    } 

    @Override 
    public boolean onStartNestedScroll(CoordinatorLayout coordinatorLayout, View child, View directTargetChild, View target, int nestedScrollAxes) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
     return false; 
    } 

    @Override 
    public void onNestedScrollAccepted(CoordinatorLayout coordinatorLayout, View child, View directTargetChild, View target, int nestedScrollAxes) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
    } 

    @Override 
    public void onStopNestedScroll(CoordinatorLayout coordinatorLayout, View child, View target) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
    } 

    @Override 
    public void onNestedScroll(CoordinatorLayout coordinatorLayout, View child, View target, int dxConsumed, int dyConsumed, int dxUnconsumed, int dyUnconsumed) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
    } 

    @Override 
    public void onNestedPreScroll(CoordinatorLayout coordinatorLayout, View child, View target, int dx, int dy, int[] consumed) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
    } 

    @Override 
    public boolean onNestedFling(CoordinatorLayout coordinatorLayout, View child, View target, float velocityX, float velocityY, boolean consumed) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
     return false; 
    } 

    @Override 
    public boolean onNestedPreFling(CoordinatorLayout coordinatorLayout, View child, View target, float velocityX, float velocityY) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
     return false; 
    } 

    @Override 
    public WindowInsetsCompat onApplyWindowInsets(CoordinatorLayout coordinatorLayout, View child, WindowInsetsCompat insets) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
     return null; 
    } 

    @Override 
    public void onRestoreInstanceState(CoordinatorLayout parent, View child, Parcelable state) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
    } 

    @Override 
    public Parcelable onSaveInstanceState(CoordinatorLayout parent, View child) { 
     Log.i(TAG, "Method " + Thread.currentThread().getStackTrace()[2].getMethodName()); 
     return null; 
    } 
} 

しかし、盗んだとき、私のスナックバーは、まだ消えて、これは典型的なログです:

private void startSnack(){ 

    loadingSnack = Snackbar.make(findViewById(R.id.email_login_form), getString(R.string.logging_in), Snackbar.LENGTH_INDEFINITE) 
      .setAction("CANCEL", new OnClickListener() { 
       @Override 
       public void onClick(View view) { 
        getOps().cancelLogin(); 
        enableControls(); 
       } 
      }); 

    loadingSnack.getView().setOnTouchListener(new View.OnTouchListener() { 
     public long mInitialTime; 
     @Override 
     public boolean onTouch(View v, MotionEvent event) { 
      if (v instanceof Button) return false; //Action view was touched, proceed normally. 
      else { 
       switch (MotionEventCompat.getActionMasked(event)) { 
        case MotionEvent.ACTION_DOWN: { 
         Log.i(TAG, "ACTION_DOWN"); 
         mInitialTime = System.currentTimeMillis(); 
         break; 
        } 
        case MotionEvent.ACTION_UP: { 
         Log.i(TAG, "ACTION_UP"); 
         long clickDuration = System.currentTimeMillis() - mInitialTime; 
         if (clickDuration <= ViewConfiguration.getTapTimeout()) { 
          return false;// click event, proceed normally 
         } 
        } 
        case MotionEvent.ACTION_MOVE: { 
         Log.i(TAG, "ACTION_MOVE"); 
         return true; 
        } 
       } 
       return true; 
      } 
     } 
    }); 

    ViewGroup.LayoutParams lp = loadingSnack.getView().getLayoutParams(); 
    if (lp != null && lp instanceof CoordinatorLayout.LayoutParams) { 
     ((CoordinatorLayout.LayoutParams)lp).setBehavior(new DummyBehavior()); 
     loadingSnack.getView().setLayoutParams(lp); 
     Log.i(TAG, "Dummy behavior assigned to " + lp.toString()); 

    } 

    loadingSnack.show(); 

} 

これはDummyBehaviorクラスです

12-02 22:26:43.864 19598-19598/ I/DummyBehavior: Dummy behavior created 
12-02 22:26:43.866 19598-19598/ I/DummyBehavior: Method <init> 
12-02 22:26:43.866 19598-19598/ I/LifecycleLoggingActivity: Dummy behavior assigned to [email protected]c0e9 
12-02 22:26:44.755 19598-19598/ I/LifecycleLoggingActivity: ACTION_DOWN 
12-02 22:26:44.798 19598-19598/ I/LifecycleLoggingActivity: ACTION_MOVE 
12-02 22:26:44.815 19598-19598/ I/LifecycleLoggingActivity: ACTION_MOVE 
12-02 22:26:44.832 19598-19598/ I/LifecycleLoggingActivity: ACTION_MOVE 
12-02 22:26:44.849 19598-19598/ I/LifecycleLoggingActivity: ACTION_MOVE 
12-02 22:26:44.866 19598-19598/ I/LifecycleLoggingActivity: ACTION_MOVE 
12-02 22:26:44.883 19598-19598/ I/LifecycleLoggingActivity: ACTION_MOVE 

答えて

1

クリック数ではなくストリーミングタッチイベントを無効にすることができますSnackbarビューに移動します。

mSnackBar.getView().setOnTouchListener(new View.OnTouchListener() { 
      public long mInitialTime; 
      @Override 
      public boolean onTouch(View v, MotionEvent event) { 
       if (v instanceof Button) return false; //Action view was touched, proceed normally. 
       else { 
        switch (MotionEventCompat.getActionMasked(event)) { 
         case MotionEvent.ACTION_DOWN: { 
          mInitialTime = System.currentTimeMillis(); 
          break; 
         } 
         case MotionEvent.ACTION_UP: { 
          long clickDuration = System.currentTimeMillis() - mInitialTime; 
          if (clickDuration <= ViewConfiguration.getTapTimeout()) { 
           return false;// click event, proceed normally 
          } 
         } 
        } 
        return true; 
       } 
      }); 

それとも、単にいくつかの空のCoordinatorLayout.BehaviorSnackbar行動を置き換えることができます:

public CouchPotatoBehavior extends CoordinatorLayout.Behavior<View>{ 

    //override all methods and don't call super methods. 

} 

これは何もしない空の行動、です。デフォルトのSwipeToDismissBehaviorViewDragHelperを使用してタッチイベントを処理し、その際に解雇がトリガーされます。ここ

ViewGroup.LayoutParams lp = mSnackbar.getView().getLayoutParams(); 
if (lp instanceof CoordinatorLayout.LayoutParams) { 
    ((CoordinatorLayout.LayoutParams)lp).setBehavior(new CouchPotatoBehavior()); 
     mSnackbar.getView().setLayoutParams(lp);    
} 
+0

_Orあなたは空のCoordinatorLayout.Behaviorでスナップバーの動作を置き換えることができます。私はちょっとこのアイデアが好きです。拡張できますか? 'ViewConfiguration.getTapTimeout()'の使用が大好きです。 – petey

+0

@peteyあなたは行き​​ます。 getView()がnullでないことを確認してください。 :) –

+0

提案ありがとうございますが、私はそれらを働かせることができません。あなたの提案されたソリューションに従って開発されたコードを追加する質問を修正しました。 – Paolone

1

より良いソリューション.... はスナックバーでのビューとしてCoordinatorLayoutまたはその子のいずれかを提供しないでください。 、ROOT_LAYOUTはcoordinatorlayoutまたはその子以外の任意のレイアウトである必要があり

Snackbar.make(ROOT_LAYOUT , "No internet connection", Snackbar.LENGTH_INDEFINITE).show(); 

+1

良い解決方法がありますが、自己責任で使用してください。 [Android docs](https://developer.android.com/reference/android/support/design/widget/Snackbar.html)から:_ "ビュー階層にCoordinatorLayoutを設定すると、Snackbarはスワイプ - Dismiss **とFloatingActionButton **のようなウィジェットを自動的に動かします。 "_ –

14
これは私のために働い

:これは私のために働いた

public class DisableSwipeBehavior extends SwipeDismissBehavior<Snackbar.SnackbarLayout> { 
    @Override 
    public boolean canSwipeDismissView(@NonNull View view) { 
     return false; 
    } 
} 
+0

これは素晴らしい機能を果たしました。 –

+0

これは受け入れられる回答です。たとえスワイプジェスチャーを無効にしたい場合は、スナックバーを使用しないでください。 – Tobliug

2

Snackbar snackbar = Snackbar.make(findViewById(container), R.string.offers_refreshed, Snackbar.LENGTH_LONG); 
    final View snackbarView = snackbar.getView(); 
    snackbar.show(); 

    snackbarView.getViewTreeObserver().addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() { 
     @Override 
     public boolean onPreDraw() { 
      snackbarView.getViewTreeObserver().removeOnPreDrawListener(this); 
      ((CoordinatorLayout.LayoutParams) snackbarView.getLayoutParams()).setBehavior(null); 
      return true; 
     } 
    }); 

幸運DisableSwipeBehaviorがある

Snackbar.SnackbarLayout layout = (Snackbar.SnackbarLayout) snackbar.getView(); 
    snackbar.setDuration(Snackbar.LENGTH_INDEFINITE); 
    snackbar.show(); 
    layout.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() { 
     @Override 
     public void onGlobalLayout() { 
      ViewGroup.LayoutParams lp = layout.getLayoutParams(); 
      if (lp instanceof CoordinatorLayout.LayoutParams) { 
       ((CoordinatorLayout.LayoutParams) lp).setBehavior(new DisableSwipeBehavior()); 
       layout.setLayoutParams(lp); 
      } 
      if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) { 
       layout.getViewTreeObserver().removeOnGlobalLayoutListener(this); 
      } else { 
       //noinspection deprecation 
       layout.getViewTreeObserver().removeGlobalOnLayoutListener(this); 
      } 
     } 
    }); 

! :)

0

ここでは、ViewTreeObserverを使いこなす必要はありません。以下の溶液をSDKに基づいてKotlinで書かれていることに注意してください26

BaseTransientBottomBar

final void showView() { 
    if (mView.getParent() == null) { 
     final ViewGroup.LayoutParams lp = mView.getLayoutParams(); 

     if (lp instanceof CoordinatorLayout.LayoutParams) { 
      // If our LayoutParams are from a CoordinatorLayout, we'll setup our Behavior 
      final CoordinatorLayout.LayoutParams clp = (CoordinatorLayout.LayoutParams) lp; 

      final Behavior behavior = new Behavior(); 
      behavior.setStartAlphaSwipeDistance(0.1f); 
      behavior.setEndAlphaSwipeDistance(0.6f); 
      behavior.setSwipeDirection(SwipeDismissBehavior.SWIPE_DIRECTION_START_TO_END); 
      behavior.setListener(new SwipeDismissBehavior.OnDismissListener() { 
       @Override 
       public void onDismiss(View view) { 
        view.setVisibility(View.GONE); 
        dispatchDismiss(BaseCallback.DISMISS_EVENT_SWIPE); 
       } 

       @Override 
       public void onDragStateChanged(int state) { 
        switch (state) { 
         case SwipeDismissBehavior.STATE_DRAGGING: 
         case SwipeDismissBehavior.STATE_SETTLING: 
          // If the view is being dragged or settling, pause the timeout 
          SnackbarManager.getInstance().pauseTimeout(mManagerCallback); 
          break; 
         case SwipeDismissBehavior.STATE_IDLE: 
          // If the view has been released and is idle, restore the timeout 
          SnackbarManager.getInstance() 
            .restoreTimeoutIfPaused(mManagerCallback); 
          break; 
        } 
       } 
      }); 
      clp.setBehavior(behavior); 
      // Also set the inset edge so that views can dodge the bar correctly 
      clp.insetEdge = Gravity.BOTTOM; 
     } 

     mTargetParent.addView(mView); 
    } 

    ... 
} 

あなたが方法showViewBaseTransientBottomBarのソースコードに見ればlayoutParamsがある場合、それは動作を追加CoordinatorLayout.LayoutParams。この動作をnullに戻して元に戻すことができます。

ビューが表示される直前にビヘイビアが追加されるので、ビューが表示された後で元に戻す必要があります。

val snackbar = Snackbar.make(coordinatorLayout, "Hello World!", Snackbar.LENGTH_INDEFINITE) 
snackbar.show() 
snackbar.addCallback(object : BaseTransientBottomBar.BaseCallback<Snackbar>() { 
    override fun onShown(transientBottomBar: Snackbar) { 
     val layoutParams = transientBottomBar.view.layoutParams as? CoordinatorLayout.LayoutParams 
     layoutParams?.let { it.behavior = null } 
    } 
}) 
関連する問題