android - 禁用其他子 fragment 的 View 寻呼机滑动

标签 android kotlin android-fragments instagram android-viewpager2

我正在使用 ViewPager2我在 viewpager 中有 2 个 fragment 。它工作得很好,但有一个问题,我希望它只是从 FragmentTwo 刷到 FragmentOne fragment ,因为它现在正在工作。但是当我将它从 FragmentOne 滑动到 FragmentTwo 时,它将只能从 FragmentOne fragment 的 FirstFragment 子 fragment 中滑动。
我的项目图片,
the image
在此图像中,MainActivity 是放置 viewpager2 的基本 Activity 。它包含 2 个 fragment ,FragmentOne 和 FragmentTwo。 FragmentOne 有一个定制的 viewpager2 用于 BottomNavigationView .它有 3 个 fragment ,第一个是 HomeFragment,这个 fragment 有一个按钮,第二个交易到 FragmentTwo。所以我只想让刷卡交易只能从 HomeFragment 发生,而不是从它的子 fragment 或其他两个 fragment 发生。 就像 instagram 一样,我们只能从 HomeFragment 滑动到 ChatFragment。
已编辑ViewPager 的代码在 MainActivity

  override fun onCreate(savedInstanceState: Bundle?) {
    val mViewPager = findViewById<ViewPager2>(R.id.view_pager_main_activity)
        mViewPager?.offscreenPageLimit = 2
        mViewPager?.apply { // to disable overScrollAnimation
           (getChildAt(0) as? RecyclerView)?.overScrollMode = RecyclerView.OVER_SCROLL_NEVER
        }
  }


internal class MyViewPagerAdapter(fragmentActivity: FragmentActivity)
    :FragmentStateAdapter(fragmentActivity)
{

    private val fragments: ArrayList<Fragment> = ArrayList()

    fun addFragment(fragment: Fragment)
    {
        fragments.add(fragment)
    }

    override fun getItemCount(): Int {
        return 2
    }

    override fun createFragment(position: Int): Fragment {
        when (position) {
            0 -> {
                return MainFragment()
            }
            1 ->{
                return MessengerFragment()
            }
        }
        return fragments[position]
    }

}
我已经通过在 MainActivity 中获取当前 fragment 和最后一个 fragment 和 .isVisible 或 .add fragment 来尝试这种方式,但没有任何效果。
例子:
if (!HomeFragment().isVisible){
            mViewPager?.isUserInputEnabled = false
        }

最佳答案

这是一个可怕的问题。在ios上就容易多了。
对我来说,这很有效:
如果你附上背景,那么寻呼机,刷卡是完全正常的。但是如果你附加了当前fragment的一个child,那么滑动就会注册到child的onTouch方法中。这是非常基本的权利。
您不能从子项的 onTouch 界面返回 false,因为您必须垂直触摸它...
解决方案:您创建一个 SwipeListener对于每个可以附加的 subview 。
SwipeListener 可以检测您是否在滑动。
但这不是计划。您必须检查用户是否垂直滑动。如果是这样,您可以获得 offsetX。它通常是一个非常小的数量,这取决于滚动速度。您将 SwipeListener 中的 offsetX 与 fakeDragBy(float) 连接起来。 viewPager 的。
来自 child 的滚动将被放在 viewPager 滚动上。
您将 onSwipe(...) 与 fakeDragBy(...) 连接起来。
当然,您必须计算 fakeDragBy 的数量,但这很简单。查纪录片。 docs
提示:在拖动之前调用 beginFakeDrag。
例子:

package com.exsent.app.act;

import android.os.Bundle;
import android.util.Log;
import android.view.GestureDetector;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.widget.FrameLayout;
import android.widget.LinearLayout;

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.appcompat.app.AppCompatActivity;
import androidx.recyclerview.widget.RecyclerView;
import androidx.viewpager2.widget.ViewPager2;

import com.exsent.app.R;

/**
* Created by Jim-Linus Valentin Ahrend on 3/26/21.
* 
**/
public class EXAMPLE extends AppCompatActivity {

@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    ViewPager2 vp2 = findViewById(R.id.viewPager);
    RecyclerView.Adapter anyAdapter = /* your adapter */ null;
    vp2.setAdapter(anyAdapter);

    //so this is the basic background
    //now goto the Adapter


}

void drag(int dx){
    ViewPager2 vp2 = findViewById(R.id.viewPager);
    vp2.beginFakeDrag();
    vp2.fakeDragBy(dx);
    
}

static class MyExampleAdapter extends 
     RecyclerView.Adapter<MyExampleAdapter.ViewHolder>{

    private EXAMPLE x;
    MyExampleAdapter(EXAMPLE x){this.x = x;}
    private GestureDetector detector;

    @Override
    public void onBindViewHolder(@NonNull ViewHolder holder, int 
         position) {

        if(position == 0){

            //gesture detector

            this.detector = new GestureDetector(x, new 
       MyGestureListener());

            RecyclerView recyclerView = 
        holder.itemView.findViewWithTag("recycler");
            recyclerView.setOnTouchListener(new 
         View.OnTouchListener() {
                @Override
                public boolean onTouch(View v, MotionEvent event) {
                    return detector.onTouchEvent(event);
                }
            });


        }

    }

    @Override
    public int getItemCount() {
        return 2;
    }

    @NonNull
    @Override
    public ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {

        FrameLayout fragemntView = new FrameLayout(x);
        fragemntView.setLayoutParams(new LinearLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT));

        RecyclerView recyclerView = new RecyclerView(x);
        recyclerView.setLayoutParams(new LinearLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT));
        recyclerView.setTag("recycler");

        fragemntView.addView(recyclerView);

        return new ViewHolder(fragemntView);
    }

    static class ViewHolder extends RecyclerView.ViewHolder {

        public ViewHolder(@NonNull View itemView) {
            super(itemView);
        }


    }

    static class MyGestureListener extends GestureDetector.SimpleOnGestureListener {
        
        
        EXAMPLE e;
        public MyGestureListener(EXAMPLE e){
            this.e = e;
        }

        @Override
        public boolean onDown(MotionEvent event) {
            Log.d("TAG","onDown: ");

            // don't return false here or else none of the other
            // gestures will work
            return true;
        }

        @Override
        public boolean onSingleTapConfirmed(MotionEvent e) {
            Log.i("TAG", "onSingleTapConfirmed: ");
            return true;
        }

        @Override
        public void onLongPress(MotionEvent e) {
            Log.i("TAG", "onLongPress: ");
        }

        @Override
        public boolean onDoubleTap(MotionEvent e) {
            Log.i("TAG", "onDoubleTap: ");
            return true;
        }

        @Override
        public boolean onScroll(MotionEvent e1, MotionEvent e2,
                                float distanceX, float distanceY) {
            Log.i("TAG", "onScroll: ");
            
            
            //drag
            if(distanceX>distanceY){ //this condition should check if it is vertical or horizontal. you can define it better if you want.
                
                e.drag(distanceX);
                
            }
            
            return true;
        }

        @Override
        public boolean onFling(MotionEvent event1, MotionEvent 
                               event2,
                               float velocityX, float velocityY) {
            Log.d("TAG", "onFling: ");
            return true;
        }
    }
   }
}

关于android - 禁用其他子 fragment 的 View 寻呼机滑动,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/66745448/

相关文章:

android - 在 Kotlin 中无法 "findViewById"。收到错误 "Type inference failed"

java - 如何添加 fragment 以使我的应用程序不会在屏幕上显示为空白

java - 从Dialog到MainActivity的访问方法

android - 使用电话作为接入点的设备的 IP 地址

android - 为复选框设置禁用背景图像

android - 在 gridview 适配器中,getView(position == 0) 在加载器中的 setImageBitmap() 被调用太多次以测量布局

android - OpenVpn android 应用程序出错

android - 使用 Kotlin 处理错误 RXJava Android

android - 当点击后退按钮时,应用程序在一秒钟后崩溃

java - CursorLoader 不刷新数据