Android 常见横向滚动GridView效果实现及封装

目录

1.需求分析

2.功能实现

3.使用说明

3.1 使用前准备

3.2 布局引用

3.3 适配器生成、绑定及数据更新

4.注意事项

5.最后


1.需求分析

京东首页分类栏滑动效果图

像直播间礼物列表和电商首页类别列表,常见出现这种需求:当前页展示效果为GridView样式,同时具有ViewPager换页效果

一般采用 ViewPager + RecyclerView 组合实现,本文 HorizontalGridView 只是在此基础上进行封装,以达到能更方便快捷实现该效果的目的

2.功能实现

HorizontalGridView展示效果

设计思路:

自定义 HorizontalGridView 继承 LinearLayout ,添加 ViewPager 和 TabLayout ,根据需要展示的 总数据数目 、 每行最大展示数目 及 每页最大展示数目 生成把多个 RecyclerView ,并填充到 ViewPager 中

需要实现带 圆角 指示器 指示器栏背景

为了提高该 View 适用性,该View需要支持设定 每行最大展示数目每页最大展示数目

为了提高该 View 的使用便捷性,采用 适配器模式观察者模式 进行封装设计

存在问题:当 HorizontalGridView 高度设置为 wrap_content 时,会出现占满全屏的现象。该现象因为 ViewPager 高度默认占满全屏造成。解决办法如下

   /**
     * 重写 onMeasure 方法,将 ViewPager 展示的所有 RecyclerView 中高度最大的一个作为 ViewPager 的高度
     */
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        int heightMode = MeasureSpec.getMode(heightMeasureSpec);
        //固定viewpager高度
        int vpHeight = 0;
        for (int i = 0; i < mViewPager.getChildCount(); i++) {
            View child = mViewPager.getChildAt(i);
            child.measure(widthMeasureSpec,
                    MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED));
            int h = child.getMeasuredHeight();
            if (h > vpHeight) {
                vpHeight = h;
            }
        }
        mViewPager.getLayoutParams().height = vpHeight;
        mViewPager.measure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(vpHeight, heightMode));

        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    }

核心方法:根据展示数据数目更新 ViewPager 页数及展示内容

   /**
     * 更新数据
     * mAdapter 为当前View的适配器
     * mAdapter.mData 为需要展示总数据lsit
     * mPageAdapter 为ViewPager的适配器
     * mDataViewList 为mPageAdapter填充展示的View(RecyclerView)集合
     */
    void notifyDataSetChange() {
        if (mAdapter == null) {
            return;
        }
        if (mAdapter.mData == null || mAdapter.mData.isEmpty()) {
            mDataViewList.clear();
            mPagerAdapter.notifyDataSetChanged();
            mTabLayout.setVisibility(GONE);
            return;
        } else {
            mAdapter.mData.size();
        }
        //总的页数向上取整
        List totalData = mAdapter.mData;
        int totalPage = (int) Math.ceil(totalData.size() * 1.0 / mPageDisplaysCount);
        int oldPage = mDataViewList.size();
        for (int i = 0; i < totalPage; i++) {
            //获取子列表
            List subList = new ArrayList<>();
            if ((i + 1) * mPageDisplaysCount > totalData.size()) {
                subList.addAll(totalData.subList(i * mPageDisplaysCount, totalData.size()));
            } else {
                subList.addAll(totalData.subList(i * mPageDisplaysCount, (i + 1) * mPageDisplaysCount));
            }
            if (mDataViewList.size() < i + 1) {
                GridLayoutManager manager = new GridLayoutManager(getContext(), mSpanCount);
                RecyclerView recyclerView = new RecyclerView(getContext());
                LayoutParams layoutParams = new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT);
                recyclerView.setLayoutParams(layoutParams);
                recyclerView.setOverScrollMode(OVER_SCROLL_NEVER);
                recyclerView.setLayoutManager(manager);
                ChildAdapter childAdapter = new ChildAdapter(subList, i);
                recyclerView.setAdapter(childAdapter);
                mDataViewList.add(recyclerView);
            } else {
                ChildAdapter childAdapter = (ChildAdapter) mDataViewList.get(i).getAdapter();
                childAdapter.mData.clear();
                childAdapter.mData.addAll(subList);
                childAdapter.notifyDataSetChanged();
            }
        }
        if (totalPage < mDataViewList.size()) {
            mDataViewList = mDataViewList.subList(0, totalPage);
        }
        int current;
        if (totalPage >= oldPage) {
            current = mViewPager.getCurrentItem();
        } else {
            current = Math.min(mViewPager.getCurrentItem(), totalPage);
        }
        //不采用 ViewPagerAdapter.notifyDataSetChange 方法更新数据,防止页数减少时,还能看到已经被销毁的页面
        mViewPager.setAdapter(mPagerAdapter);
        mTabLayout.setupWithViewPager(mViewPager);
        //mViewPager.setOffscreenPageLimit(mDataViewList.size());
        mViewPager.setCurrentItem(current);
        mTabLayout.setVisibility(VISIBLE);
    }

完整源码

3.使用说明

3.1 使用前准备

确认已经导入 google material 包 : implementation 'com.google.android.material:material:1.1.0'

复制 HorizontalGridView 类到项目中

在 Module 的\src\main\res\values\attrs.xml中声明 自定义属性

3.2 布局引用

布局内引用

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <com.ziwenl.horizontalgridview.widgets.HorizontalGridView
        android:id="@+id/hgv"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        app:indicatorBackground="@drawable/shape_indicator_background"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        app:pageDisplaysCount="10"
        app:pageSpanCount="5"
        app:tabBackground="@drawable/shape_tab_background"
        app:tabLayoutHeight="5dp"
        app:tabLayoutInterval="5dp"
        app:tabLayoutWidth="50dp" />

</androidx.constraintlayout.widget.ConstraintLayout>

自定义属性说明

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <declare-styleable name="HorizontalGridView">
        <!--指示器背景-->
        <attr name="indicatorBackground" format="reference" />
        <!--指示器颜色 只有 "未设置indicatorBackground" 或 "设置了indicatorBackground但SDK低于24" 时才生效-->
        <attr name="indicatorBackgroundColor" format="color" />
        <!--TabLayout宽度-->
        <attr name="tabLayoutWidth" format="dimension" />
        <!--TabLayout高度-->
        <attr name="tabLayoutHeight" format="dimension" />
        <!--TabLayout背景-->
        <attr name="tabBackground" format="reference" />
        <!--TabLayout和ViewPager的间隔-->
        <attr name="tabLayoutInterval" format="dimension" />
        <!--ViewPager每页展示总数量-->
        <attr name="pageDisplaysCount" format="integer" />
        <!--ViewPager每行展示数目-->
        <attr name="pageSpanCount" format="integer" />
    </declare-styleable>
</resources>

3.3 适配器生成、绑定及数据更新

/**
 * 自定义适配器继承 HorizontalGridView.Adapter<T> 
 * T为业务数据bean
 */
class MainAdapter(data: List<String>) : HorizontalGridView.Adapter<String>(data) {

    /**
     * 绑定 itemLayout.xml
     */
    override fun onBindItemLayout(): Int {
        return R.layout.item_test
    }

    /**
     * 根据 ViewHolder 对象操作view
     */
    override fun onBindViewHolder(
        holder: HorizontalGridView.ViewHolder,
        dto: String,
        currentPosition: Int,
        realPosition: Int
    ) {
        Glide.with(holder.itemView.context)
            .load(dto)
            .into(holder.itemView.iv_picture)
        holder.itemView.tv_content.text = "当前$realPosition"

    }
}

class MainActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        //绑定适配器
        val adapter = MainAdapter(mutableListOf())
        hgv.setAdapter(adapter)

        //变更及更新内容
        adapter.data.addAll(NetImageUtil.getUrls(25))
        adapter.notifyDataSetChange()

        //点击事件监听
        hgv.setOnClickItemListener { realPosition ->
            Toast.makeText(this, realPosition.toString(), Toast.LENGTH_SHORT).show()
        }
        
        //页面滑动事件监听
        hgv.addOnPageChangeListener(object: ViewPager.OnPageChangeListener{
            override fun onPageScrollStateChanged(state: Int) {
            
            }
            
            override fun onPageScrolled(
                position: Int,
                positionOffset: Float,
                positionOffsetPixels: Int
            ) {
            
            }
            
            override fun onPageSelected(position: Int) {
            
            }
        })
    }
}

4.注意事项

单独通过 indicatorBackground 属性引用shape资源设置 指示器圆角及背景色 功能只在 SDK >= 24 时生效,当 SDK < 24 时需要配合 indicatorBackgroundColor 设置 指示器背景色

当 tabLayoutHeight 值为 0未赋值 时表示隐藏 TabLayout

5.最后

  • 3
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
PageGridViewViewPager GridView组合控件实现网格布局分页效果大多数app首页界面一般貌似可以滑动切换上下页的网格布局,这种效果用得挺多,可惜Android原生是没有这样的控件的。在工作中,我也用到过几次。但每次实现都是按ctrl A C V,代码写得也不优雅,就是再次重新复制粘贴,也避免不了看上几眼,修改修改,特别不爽快。每次吐槽的同时在想,下次我要再用到这种效果,我就将这些代码关进小黑屋。拒绝再写重复代码,虽然大多数只是按ctrl A C V。这次,我终于要将自己吐槽时的想法付之行动的。再写之前,为了避免重复造轮子。在网上搜了好多次,嗯,没有意外,我还是没有发现到惊喜。那就开始行动了。这种效果用ViewPager GridView实现,那我就叫它PageGridView吧(其实名字我也想了好久)有用的话,star一下也是不错的!!!PageGridView 特点可自由定制Item布局无需写adapter适配器,直接调用setData()方法填充数据只需实体类继承ItemModel即可可显示本地图、网络图、资源ID图标,由开发者实现动态计算PageGridView高度,不需要在布局里面写固定高度可重复setData()方法刷新数据PageGridView自定义属性属性名说明默认值pageSize每页大小8numColumns列数4isShowIndicator是否显示指示器trueselectedIndicator选中指示点资源IDR.mipmap.ic_dot_selectedunSelectedIndicator未选中指示点资源IDR.mipmap.ic_dot_normalindicatorGravity指示器位置centerindicatorPaddingLeft指示器左内边距0pxindicatorPaddingRight指示器右内边距0pxindicatorPaddingTop指示器上内边距0pxindicatorPaddingBottom指示器下内边距0pxindicatorPadding指示器内边距0pxindicatorBackground指示器背景颜色Color.WHITEitemViewItem布局R.layout.item_view约定规则Item布局必须给定具体高度,默认itemView布局的高度为@dimen/item_height=80dpItem布局 TextView的id 为R.id.tv_item_nameItem布局 ImageView的id 为R.id.iv_item_iconItem默认点击效果R.drawable.selector_item_view_bgItem点击颜色值:R.color.item_view_selected_colorItem背景颜色值:R.color.item_view_normal_color一行四列效果图两行四列效果图自定义Item布局效果图PageGridView使用Gradle 依赖库implementation 'com.wihaohao:PageGridView:1.0.1'默认布局一般默认参数即可满足需求<com.wihaohao.PageGridView           android:id="@ id/vp_grid_view"           android:layout_width="match_parent"           android:layout_height="wrap_content"           />一行4列的布局  <com.wihaohao.PageGridView           android:id="@ id/vp_grid_view"           android:layout_width="match_parent"           android:layout_height="wrap_content"           app:numColumns="4"           app:pageSize="4"           app:selectedIndicator="@mipmap/ic_dot_selected"           app:unSelectedIndicator="@mipmap/ic_dot_normal"           app:isShowIndicator="true"           />两行4列的布局    <com.wihaohao.PageGridView         android:id="@ id/vp_grid_view"         android:layout_width="match_parent"         android:layout_height="wrap_content"         app:indicatorPadding="10dp"         app:isShowIndicator="true"         app:itemView="@layout/my_item_view"         app:numColumns="4"         app:pageSize="8"         app:selectedIndicator="@mipmap/ic_dot_selected"         app:unSelectedIndicator="@mipmap/ic_dot_normal" />自定义Item布局 <com.wihaohao.PageGridView                 android:id="@ id/p_grid_view2"         android:layout_width="match_parent"         android:layout_height="wrap_content"         app:indicatorGravity="right"         app:indicatorPaddingRight="@dimen/padding_10"         app:isShowIndicator="true"         app:itemView="@layout/item_custom"         app:indicatorBackground="@color/colorPrimary"         app:numColumns="5"         app:pageSize="5"/>ItemModelpublic abstract static class ItemModel {                 /**                  * 返回item名字                  *                  * @return                  */         protected abstract String getItemName();         /**                  * 设置图标                  *                  * @param imageView         */         protected abstract void setIcon(ImageView imageView);          /**                  * 特殊需求,重写该方法,设置item                  *                  * @param itemView                  */         protected void setItemView(View itemView) {         }     }Model继承VpGridView.ItemModel 为item赋值和设置图标public abstract static class ItemModel {                 /**                  * 返回item名字                  *                  * @return         */         protected abstract String getItemName();                 /**                  * 设置图标                  *                  * @param imageView                  */         protected abstract void setIcon(ImageView imageView);                 /**                  * 特殊需求,重写该方法,设置item                  *                  * @param itemView                  */         protected void setItemView(View itemView) {         }     }继承VpGridView.ItemModel 设置ItemViewpublic class MyIconModel extends PageGridView.ItemModel {     private String name;         private int iconId;         public String getName() {             return name;     }         public void setName(String name) {             this.name = name;     }         public int getIconId() {             return iconId;     }         public void setIconId(int iconId) {             this.iconId = iconId;     }         public MyIconModel(String name, int iconId) {             this.name = name;             this.iconId = iconId;     }         @Override     protected String getItemName() {             return name;     }         @Override     protected void setIcon(ImageView imageView) {         imageView.setImageResource(iconId);     } }PageGridView的使用public class MainActivity extends AppCompatActivity {     List<MyIconModel> mList;         List<CustomModel> mList2;         private PageGridView<MyIconModel> mPageGridView;         private PageGridView<CustomModel> mPageGridView2;         @Override     protected void onCreate(Bundle savedInstanceState) {                 super.onCreate(savedInstanceState);         setContentView(R.layout.activity_main);         mPageGridView =findViewById(R.id.vp_grid_view);         initData();         mPageGridView.setData(mList);         mPageGridView.setOnItemClickListener(new PageGridView.OnItemClickListener() {            @Override             public void onItemClick(int position) {                Toast.makeText(MainActivity.this,position "",Toast.LENGTH_SHORT).show();             }         });        //自定义item         mPageGridView2=findViewById(R.id.p_grid_view2);         mPageGridView2.setData(mList2);     }         private void initData() {         mList=new ArrayList<>();         mList2=new ArrayList<>();                 for(int i=0;i<30;i ){             mList.add(new MyIconModel("测试" i,R.mipmap.ic_launcher));             mList2.add(new CustomModel("标题" i));         }     } }

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值