Android CoordinatorLayout(六) 加入下拉功能

简介: 上章讲了CoordinatorLayout的卡顿BUG,既然有BUG又没解决,说实话没必要讲下去,但是做事总要有始有终,既然写了就把它写完吧,顶着BUG去写。

上章讲了CoordinatorLayout的卡顿BUG,既然有BUG又没解决,说实话没必要讲下去,但是做事总要有始有终,既然写了就把它写完吧,顶着BUG去写。

四、CoordinatorLayout + 下拉刷新

之前用过,但是在google找不到我心目中的刷新方式,google上写的都是在滑动控件上加一层刷新,而且都是SwipeRefreshLayout。如果你是想实现这样的效果,你直接在google上找代码就是,一大堆。我要做的是在整个CoordinatorLayout 外加一层刷新,入下图的效果。

img_44dcb5f0d681786172604199548e1c9c.png
image.png

我们在CoordinatorLayout+ AppBarLayout + ViewPager这个例子上加下拉刷新框架。

1. 准备

我使用的是PullToRefresh框架,但是我会告诉你,你要用SwipeRefreshLayout的做法也是一样的,下拉框架的原理都是那样。

既然是使用PullToRefresh,就要导入PullToRefresh的Module,用过的都知道,没用过的希望先去了解一下。不懂导入Module的赶紧去学。

2.思路

在实现这样一个框架,你想想,就需要在原来的布局中加入一层上拉下拉框架。简单来说,我们要把CoordinatorLayout+ AppBarLayout + ViewPager放到PullToRefresh的中间部分,也就是PullToRefresh的PullToRefreshBase。那我们就需要先写个PullToRefreshBase代表CoordinatorLayout+ AppBarLayout + ViewPager。

注意:

如果你不会用PullToRefresh框架,你肯定看懵逼我接下来的代码,所以不了解这个框架的一定要先学会这个框架的简单用法,比如你写个demo让PullToRefresh装RecyclerView,等你大概会用了再看我接下去的步骤,我也不想写太多基础上的细节。
3.定义PullToRefreshBase

xml布局:

<com.example.kylinarm.coordinatorlayouttest.PullRefreshCoordTest
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:id="@+id/pull"
        >
    </com.example.kylinarm.coordinatorlayouttest.PullRefreshCoordTest>

布局没什么好解释的。

自定义PullToRefreshBase:

public class PullRefreshCoordTest extends PullToRefreshBase<CoordinatorView> {

    private CoordinatorView coordinatorView;

    public PullRefreshCoordTest(Context context) {
        super(context);
    }

    public PullRefreshCoordTest(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public PullRefreshCoordTest(Context context, Mode mode) {
        super(context, mode);
    }

    public PullRefreshCoordTest(Context context, Mode mode, AnimationStyle animStyle) {
        super(context, mode, animStyle);
    }

    @Override
    public Orientation getPullToRefreshScrollDirection() {
        return Orientation.VERTICAL;
    }

    @Override
    protected CoordinatorView createRefreshableView(Context context, AttributeSet attrs) {
        coordinatorView = new CoordinatorView(context);
        return coordinatorView;
    }

    @Override
    protected boolean isReadyForPullEnd() {
        return false;
    }

    @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
    @Override
    protected boolean isReadyForPullStart() {
        // 或者判断CoordinatorView是否全部展开

        return coordinatorView.getScrollY() == 0 && coordinatorView.getmVerticalOffset() >= 0;
    }

    public CoordinatorView getCoordinatorView() {
        return coordinatorView;
    }
}

介绍:
(1)CoordinatorView就是我们的CoordinatorLayout+ AppBarLayout + ViewPager,等下会放这个view具体处理的代码。
(2)createRefreshableView()方法就是创建嵌套在PullToRefresh里面的view,这是基础。
(3)isReadyForPullStart(),这个方法很重要,是判断什么时候可以下拉刷新,里面是一个算法,这个算法我放到最后讲。

4. 定义CoordinatorView(CoordinatorLayout+ AppBarLayout + ViewPager)
public class CoordinatorView extends FrameLayout {

    @InjectView(R.id.tl_tab)
    TabLayout tlTab;
    @InjectView(R.id.vp_content)
    MyViewPager vpContent;
    @InjectView(R.id.appbar)
    AppBarLayout appbar;

    private View contentView;
    //记录是否处于展开状态
    public int mVerticalOffset = 0;
    private String[] titles = {"tab1","tab2"};
    private List<Fragment> fragments = new ArrayList<>();
    private Context context;
    private FragmentActivity fActivity;

    public CoordinatorView(Context context) {
        super(context);
        this.context = context;

        if (context instanceof FragmentActivity){
            fActivity = (FragmentActivity) context;
        }

        init();
    }

    private void init(){
        contentView = LayoutInflater.from(getContext()).inflate(R.layout.activity_cv,null);
        contentView.setLayoutParams(new LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT));
        addView(contentView);
        ButterKnife.inject(this,contentView);
        appbar.addOnOffsetChangedListener(new AppBarLayout.OnOffsetChangedListener() {
            @Override
            public void onOffsetChanged(AppBarLayout appBarLayout, int verticalOffset) {
                mVerticalOffset = verticalOffset;
            }
        });
    }

    public void initView(){
        for (int i = 0; i < titles.length; i++) {
            tlTab.addTab(tlTab.newTab().setText(titles[i]));
        }
        fragments = setFragmentList();
        vpContent.setAdapter(new FragmentPagerAdapter(fActivity.getSupportFragmentManager()) {

            @Override
            public int getCount() {
                return titles.length;
            }

            @Override
            public Fragment getItem(int position) {
                return fragments.get(position);
            }

            @Override
            public CharSequence getPageTitle(int position) {
                if (null != titles) {
                    return titles[position];
                }
                return super.getPageTitle(position);
            }
        });
        tlTab.setupWithViewPager(vpContent);
    }

    private List<Fragment> setFragmentList(){
        List<Fragment> fragmentList = new ArrayList<>();
        fragmentList.add(new ListFragment());
        fragmentList.add(new ImageFragment());
        return fragmentList;
    }

    public int getmVerticalOffset() {
        return mVerticalOffset;
    }

}

其实这里的代码很容易看到,就是CoordinatorLayout+ AppBarLayout + ViewPager的正常操作,不过加了一步很重要的方法:

在init()中

appbar.addOnOffsetChangedListener(new AppBarLayout.OnOffsetChangedListener() {
            @Override
            public void onOffsetChanged(AppBarLayout appBarLayout, int verticalOffset) {
                mVerticalOffset = verticalOffset;
            }
        });

这里是给AppBarLayout加了一个监听事件addOnOffsetChangedListener,这个监听是什么意思呢?就是监听折叠。
这个方法中的verticalOffset表示的是...我也不懂怎么解释好,我就告诉你,当AppBarLayout处于完全展开状态,verticalOffset为0,而当你折叠的时候verticalOffset就变成负数,完全折叠时verticalOffset就变成负的AppBarLayout内折叠控件的高度。这个属性和判断什么时候能下拉有关,一定要先了解。

5.判断下拉

我们在PullRefreshCoordTest 的isReadyForPullStart方法中写了一个算法来判断什么时候能下拉,现在我们在了解完上面的代码之后来讲讲这个算法。

protected boolean isReadyForPullStart() {
        // 或者判断CoordinatorView是否全部展开

        return coordinatorView.getScrollY() == 0 && coordinatorView.getmVerticalOffset() >= 0;
    }

(1)coordinatorView.getScrollY()这个不用讲了,很容易懂
(2)coordinatorView.getmVerticalOffset() >= 0;
什么意思呢?coordinatorView.getmVerticalOffset() 能返回我们第4步中讲的verticalOffset,上面我们说过完全展开的时候是0,折叠之后变成负数。也就是说这句话的意思是完全展开时才能让这个view刷新。

其实也不是很难理解,多看两次代码就能看懂了。记住,要想让CoordinatorLayout刷新,只有在它处于完全展开的时候才能刷新。

目录
相关文章
|
8月前
|
XML 缓存 Android开发
Android开发,使用kotlin学习多媒体功能(详细)
Android开发,使用kotlin学习多媒体功能(详细)
168 0
|
3月前
|
Android开发
Android开发表情emoji功能开发
本文介绍了一种在Android应用中实现emoji表情功能的方法,通过将图片与表情字符对应,实现在`TextView`中的正常显示。示例代码展示了如何使用自定义适配器加载emoji表情,并在编辑框中输入或删除表情。项目包含完整的源码结构,可作为开发参考。视频演示和源码详情见文章内链接。
82 4
Android开发表情emoji功能开发
|
3月前
|
安全 Android开发 iOS开发
Android vs iOS:探索移动操作系统的设计与功能差异###
【10月更文挑战第20天】 本文深入分析了Android和iOS两个主流移动操作系统在设计哲学、用户体验、技术架构等方面的显著差异。通过对比,揭示了这两种系统各自的独特优势与局限性,并探讨了它们如何塑造了我们的数字生活方式。无论你是开发者还是普通用户,理解这些差异都有助于更好地选择和使用你的移动设备。 ###
63 3
|
7月前
|
前端开发 Android开发 Windows
27. 【Android教程】下拉选择框 Spinner
27. 【Android教程】下拉选择框 Spinner
252 2
|
5月前
|
编解码 测试技术 Android开发
Android经典实战之用 CameraX 库实现高质量的照片和视频拍摄功能
本文详细介绍了如何利用CameraX库实现高质量的照片及视频拍摄功能,包括添加依赖、初始化、权限请求、配置预览与捕获等关键步骤。此外,还特别针对不同分辨率和帧率的视频拍摄提供了性能优化策略,确保应用既高效又稳定。
473 1
Android经典实战之用 CameraX 库实现高质量的照片和视频拍摄功能
|
4月前
|
Android开发 开发者
Android平台无纸化同屏如何实现实时录像功能
Android平台无纸化同屏,如果需要本地录像的话,实现难度不大,只要复用之前开发的录像模块的就可以,对我们来说,同屏采集这块,只是数据源不同而已,如果是自采集的其他数据,我们一样可以编码录像。
|
5月前
|
图形学 Android开发
小功能⭐️Unity调用Android常用事件
小功能⭐️Unity调用Android常用事件
|
7月前
|
数据库 Android开发 数据安全/隐私保护
在 Android Studio 中结合使用 SQLite 数据库实现简单的注册和登录功能
在 Android Studio 中结合使用 SQLite 数据库实现简单的注册和登录功能
296 2
|
7月前
|
Android开发
Android中如何快速的实现RecycleView的拖动重排序功能
使用`ItemTouchHelper`和自定义`Callback`,在`RecyclerView`中实现拖动排序功能。定义`ItemTouchHelperAdapter`接口,`Adapter`实现它以处理`onItemMove`方法。`SimpleItemTouchHelperCallback`设置拖动标志,如`LEFT`或`RIGHT`(水平拖动),并绑定到`RecyclerView`以启用拖动。完成这些步骤后,即可实现拖放排序。关注公众号“AntDream”获取更多内容。
129 3
|
8月前
|
移动开发 监控 Android开发
构建高效Android应用:从内存优化到电池寿命代码之美:从功能实现到艺术创作
【5月更文挑战第28天】 在移动开发领域,特别是针对Android系统,性能优化始终是关键议题之一。本文深入探讨了如何通过细致的内存管理和电池使用策略,提升Android应用的运行效率和用户体验。文章不仅涵盖了现代Android设备上常见的内存泄漏问题,还提出了有效的解决方案,包括代码级优化和使用工具进行诊断。同时,文中也详细阐述了如何通过减少不必要的后台服务、合理管理设备唤醒锁以及优化网络调用等手段延长应用的电池续航时间。这些方法和技术旨在帮助开发者构建更加健壮、高效的Android应用程序。