Android使用ViewPager2实现页面滑动切换

Android使用ViewPager2实现页面滑动切换 

作者:QiShare
转载地址:https://juejin.cn/post/7065566099223347213

1.引言

在很多应用中,我们经常会看到多个页面之间滑动切换的场景,ViewPager2是ViewPager的升级版,本文将简要介绍如何使用ViewPager2、FragmentStateAdapter和Fragment来实现页面之间的滑动切换。

2.实现页面滑动切换

2.1 引入ViewPager2库

要使用ViewPager2,需要引入ViewPager2库,引入方法如下:

implementation "androidx.viewpager2:viewpager2:1.0.0" 

2.2 使用ViewPager2

在布局中使用ViewPager2,示例如下:

<androidx.viewpager2.widget.ViewPager2
    android:id="@+id/viewPager2"
    android:layout_width="match_parent"
    android:layout_height="match_parent"/>

2.3 构建Fragment

本Fragment只为简单演示使用,其布局如下:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:gravity="center"
    android:orientation="vertical">

    <TextView
        android:id="@+id/tv_content"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:textSize="18sp"
        android:textColor="@color/black"/>

</LinearLayout>

ContentFragment类的实现如下:

public class ContentFragment extends Fragment {
    private String content;

    public ContentFragment(String content) {
        this.content = content;
    }

    private TextView tv_content;

    @Nullable
    @Override
    public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.fragment_content, container, false);
        tv_content = view.findViewById(R.id.tv_content);
        tv_content.setText(content);
        return view;
    }

    public void setContent(String content) {
        this.content = content;
        tv_content.setText(content);
    }
}

2.4 继承FragmentStateAdapter

创建自定义的类ContentPagerAdapter,让它继承FragmentStateAdapter,并实现createFragment(int position)和getItemCount()方法,示例如下:

public class ContentPagerAdapter extends FragmentStateAdapter {
    private List<ContentFragment> datas;

    public ContentPagerAdapter(@NonNull FragmentActivity fragmentActivity,List<ContentFragment> datas) {
        super(fragmentActivity);
        this.datas = datas;
    }

    @NonNull
    @Override
    public Fragment createFragment(int position) {
        return datas.get(position);
    }

    @Override
    public int getItemCount() {
        return datas.size();
    }
}

2.5 将ViewPager2与适配器绑定

将ViewPager2与适配器绑定后,便可实现页面滑动切换,示例如下:

datas = new ArrayList<>();
datas.add(new ContentFragment("页面1"));
datas.add(new ContentFragment("页面2"));
datas.add(new ContentFragment("页面3"));
datas.add(new ContentFragment("页面4"));
datas.add(new ContentFragment("页面5"));
contentPagerAdapter = new ContentPagerAdapter(this, datas);
viewPager2.setAdapter(contentPagerAdapter);

2.6 垂直方向滑动切换

ViewPager2不仅支持水平方向的滑动,还支持垂直方向的滑动,实现垂直滑动也是相当简单,在布局文件中添加android:orientation="vertical"属性即可,如下所示:

<androidx.viewpager2.widget.ViewPager2
    android:id="@+id/viewPager2"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"/>

或者在代码中调用ViewPager2的setOrientation(ViewPager2.ORIENTATION_VERTICAL)方法也可以让ViewPager2实现垂直方向的滑动。

2.7 Fragment更新

当Fragment集合发生变化需要更新时,使用FragmentStateAdapter进行更新也很便捷,由于ViewPager2是基于RecyclerView实现的,所以更新数据的时候可以调用notifyItemChanged(int position)、notifyItemInserted(int position)等方法进行更新。

3.总结

使用ViewPager2、FragmentStateAdapter和Fragment可以便捷地实现页面之间的滑动切换,它不仅支持水平方向的滑动,还能通过简单的设置就能实现垂直方向的滑动,灵活地运用ViewPager2能实现实际的需求。

发布于 2022-02-19 20:46

android 垂直画廊效果,Android利用PageView打造垂直的画廊效果-Go语言中文社区

首先放一下效果图

 

 

这里是采用FragmentPageView所打造的效果

打造这样效果第一步就是先让ViewPager垂直滑动

这里使ViewPager垂直滑动的代码参考这位大神的帖子

https://www.jianshu.com/p/d3065bbc1167

其核心思想就是拦截触摸事件并反转横向和垂直滑动。

打造好垂直滑动效果之后剩下就变得简单,首先想达到这样一个效果需要将ViewPager及其父元素添加属性android:clipChildren=“false”,使得其可以超出自身回追范围

这里上一下这个效果的布局

d6bde5cf06b894880d5c6f969fae907b.png

因为PageViewr和外面的ViewGroup大小相同,因此一开始看到的就是一个Fragment

之后为PageViewr设置

//预加载Fragment数量

verticalPageView.setOffscreenPageLimit(3);

//两个Fragment之间的间距

verticalPageView.setPageMargin(60);

如果再加上放大缩小就可以达到画廊效果了…但是我这里不需要,有需要的可以自己添加

之后再为自定义的ViewPager重写onInterceptTouchEvent,在里面加上这样一段代码:

switch (event.getAction()){

case MotionEvent.ACTION_DOWN:

this.animate().scaleX((float) 0.8).scaleY((float) 0.8).setDuration(300).start();

break;

case MotionEvent.ACTION_MOVE:

break;

case MotionEvent.ACTION_UP:

this.animate().scaleX((float) 1.0).scaleY((float) 1.0).setDuration(300).start();

break;

}

注意这里有一个坑,就是只能触发ACTION_DOWN事件,因此上面这些代码应该在onInterceptTouchEvent一开始就调用

写到现在应该就可以达到上面的效果了,但还有一个非常严重的问题,就是Fragment上面的点击事件,这里以上面的中止按钮举例

由于Android的触摸事件是由顶层向下传递的,看上去没什么问题,但是这里最顶层是ViewPager,因此无论如何都会先触发ViewPager里面的onInterceptTouchEvent,并执行缩放动画。就算为其中的按钮执行onInterceptTouchEvent也只会让ViewPager缩小并无法恢复,因为只有ACTION_UP被阻拦

那么就没有办法了么?NO,这里在ViewGroup之上的还有一个Activity,只要获取按钮在屏幕中的位置并根据触摸点的位置判断是否点击到了按钮即可。

这里创建一个自定义接口

public interface IndexInterfaceForFragment {

void sendWidget(View view);

}

Activity实现这个接口

@Override

public void sendWidget(View view) {

interceptViewArray.add(view);

}

这里我将控件都存入了一个数组中

之后Fragment在onCreateView中执行以下代码

@Nullable

@Override

public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {

View view = inflater.inflate(R.layout.fragment_index_hint, container, false);

//注意:这个functionButton发送到了Activity

functionButton =view.findViewById(R.id.function_button);

Random random = new Random();

final int num = random.nextInt(100);

functionButton.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

Toast.makeText(getActivity(),"functionbutton点击了!"+num,Toast.LENGTH_SHORT).show();

}

});

IndexInterfaceForFragment indexInterfaceForFragment = (IndexInterfaceForFragment)getActivity();

indexInterfaceForFragment.sendWidget(functionButton);

return view;

}

这里可以看到,functionButton就是我们想阻拦的控件,通过

IndexInterfaceForFragment indexInterfaceForFragment = (IndexInterfaceForFragment)getActivity();

indexInterfaceForFragment.sendWidget(functionButton);

这种方式将控件发送到Activity中

在Activity中编写以下代码

public boolean dispatchTouchEvent(MotionEvent ev) {

if (ev.getAction() == MotionEvent.ACTION_DOWN){

if (!interceptEventControl(interceptViewArray,ev)){

return false;

}

}

return super.dispatchTouchEvent(ev);

}

private boolean interceptEventControl(ArrayList arrayList, MotionEvent event){

for (View v : arrayList){

int[] location = {0,0};

v.getLocationInWindow(location);

Log.d("座标Button","X:"+location[0]+"Y:"+location[1]);

int left = location[0],top = location[1],bottom = top + v.getHeight(),right = left+v.getWidth();

if (event.getX() > left && event.getX() < right && event.getY() > top && event.getY() < bottom){

//点击到了要阻拦的控件

v.callOnClick();

return false;

}

}

return true;

}

这里重写了dispatchTouchEvent并调用interceptEventControl做判断,在interceptEventControl中编写了判断点击区域的逻辑,如果发现点击了控件就阻拦点击事件的传递并调用控件的callOnClick()方法以触发其点击事件。

嗯,到这里就完成了最开始Gif的效果

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章