禁用其他子片段的视图寻呼机滑动

mr.*_*oot 5 android android-fragments instagram kotlin android-viewpager2

我正在处理ViewPager2我在 viewpager 中有 2 个片段的地方。它工作得很好,但有一个问题,我希望它只是从FragmentTwo滑动到FragmentOne片段,因为它现在正在工作。但是当我将它从FragmentOne滑动到FragmentTwo 时,它将只能从FragmentOne片段的 FirstFragment 子片段滑动。

我的项目图片, 图片

在此图像中 MainActivity 是放置 viewpager2 的基本活动。它包含 2 个片段,FragmentOne 和 FragmentTwo。FragmentOne 有一个自定义的 viewpager2 用于BottomNavigationView. 它有 3 个片段,第一个是 HomeFragment,这个片段有一个按钮,两个交易到 FragmentTwo。所以我只希望刷卡交易只能从 HomeFragment 发生,而不是从它的孩子或其他两个片段发生。就像 Instagram 一样,我们只能从 HomeFragment 滑动到 ChatFragment。

已编辑

ViewPagerMainActivity 中的代码

  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]
    }

}
Run Code Online (Sandbox Code Playgroud)

我已经通过在 MainActivity 中获取当前片段和最后一个片段和 .isVisible 或 .add 片段来尝试这种方式,但没有任何效果。

例子:

if (!HomeFragment().isVisible){
            mViewPager?.isUserInputEnabled = false
        }
Run Code Online (Sandbox Code Playgroud)

Val*_*ing 1

这是一个可怕的问题。在ios上就容易多了。对我来说这有效:

如果附加背景,那么寻呼机、滑动就完全正常了。但是,如果您附加当前片段的子级,则滑动将在子级的 onTouch 方法中注册。这是非常基本的权利。

你不能从孩子的 onTouch 接口返回 false,因为你必须垂直触摸它......

解决方案:您为每个可以附加的子视图创建一个SwipeListener 。SwipeListener 可以检测您是否在滑动。

但这不是计划。您必须检查用户是否垂直滑动。如果是这样你就可以获得offsetX。它通常是一个非常小的量,这取决于滚动速度。您将 SwipeListener 中的 offsetX 与viewPager 的fakeDragBy(float)连接起来。

子项的滚动将放在 viewPager 滚动上。您将 onSwipe(...) 与 fakeDragBy(...) 连接起来。

当然,你必须计算 fakeDragBy 的数量,但这很简单。查一下纪录片吧 文档

提示:在拖动之前调用 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;
        }
    }
   }
}
Run Code Online (Sandbox Code Playgroud)