有没有办法阻止用户通过刷卡来解雇零食吧?
我有一个应用程序,在网络登录时显示一个小吃店,我想避免它被解雇。
根据Nikola Despotoski的建议,我已经尝试了两种解决方案:
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类:
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;
}
}
但我的小吃店在刷卡时仍然消失,这是一个典型的日志:
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 android.support.design.widget.CoordinatorLayout$LayoutParams@808c0e9
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
答案 0 :(得分:16)
这对我有用:
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);
}
}
});
DisableSwipeBehavior的位置是:
public class DisableSwipeBehavior extends SwipeDismissBehavior<Snackbar.SnackbarLayout> {
@Override
public boolean canSwipeDismissView(@NonNull View view) {
return false;
}
}
答案 1 :(得分:5)
这对我有用:
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;
}
});
祝你好运! :)
答案 2 :(得分:3)
Snackbar现在通过使用setBehavior方法对此有实际支持。这里最棒的是,在你总是失去一些现在保留的行为之前。
请注意,程序包已移动,因此您必须导入&#34; new&#34; Snackbar in the snackbar package
Snackbar.make(view, stringId, Snackbar.LENGTH_LONG)
.setBehavior(new NoSwipeBehavior())
.show();
class NoSwipeBehavior extends BaseTransientBottomBar.Behavior {
@Override
public boolean canSwipeDismissView(View child) {
return false;
}
}
答案 3 :(得分: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;
}
});
或者您可以将Snackbar
行为替换为空CoordinatorLayout.Behavior
:
public CouchPotatoBehavior extends CoordinatorLayout.Behavior<View>{
//override all methods and don't call super methods.
}
这是空行为,什么都不做。默认SwipeToDismissBehavior
使用ViewDragHelper
来处理触摸事件,触发事件触发解雇。
ViewGroup.LayoutParams lp = mSnackbar.getView().getLayoutParams();
if (lp instanceof CoordinatorLayout.LayoutParams) {
((CoordinatorLayout.LayoutParams)lp).setBehavior(new CouchPotatoBehavior());
mSnackbar.getView().setLayoutParams(lp);
}
答案 4 :(得分:1)
这里有更好的解决方案....
不要在快餐栏中提供CoordinatorLayout
或其任何子项作为视图。
Snackbar.make(ROOT_LAYOUT , "No internet connection", Snackbar.LENGTH_INDEFINITE).show();
其中,ROOT_LAYOUT
应该是除了coordinatorlayout或其子节点之外的任何布局。
答案 5 :(得分:0)
这是一个不需要你弄乱ViewTreeObserver
的解决方案。请注意,以下解决方案是基于SDK 26在Kotlin中编写的。
<强> 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);
}
...
}
如果您查看方法BaseTransientBottomBar
中showView
的源代码,则会在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 }
}
})
答案 6 :(得分:0)
只需覆盖 CoordinatorLayout.LayoutParams.getBehaviour() 并返回 null 即可禁用滑动。
例如:
Snackbar snackBar = Snackbar.make(view, "Enjoy!", Snackbar.LENGTH_INDEFINITE);
CoordinatorLayout.LayoutParams snackBarLayoutParams = new CoordinatorLayout.LayoutParams(CoordinatorLayout.LayoutParams.MATCH_PARENT, CoordinatorLayout.LayoutParams.WRAP_CONTENT){
@Override
public CoordinatorLayout.Behavior getBehavior() {
return null;
}
};
snackBarLayoutParams.gravity = Gravity.BOTTOM;
snackBar.getView().setLayoutParams(snackBarLayoutParams);
snackBar.show();
享受吧!