粘性头部 + 流式布局

1、依赖

// 侧滑菜单、TabLayout
implementation 'com.android.support:design:28.0.0'
// 粘性头部/悬浮头部
implementation 'com.github.qdxxxx:StickyHeaderDecoration:1.0.1'

2、在 Project 的 build.gradle 文件里

allprojects {
    repositories {
        ...
        maven { url 'https://jitpack.io' }
    }
}

3、Activity 中

private void initData() {
    mCars = new ArrayList<>();
    mCars.add(new Car("奥迪",  "A"));
    mCars.add(new Car("阿尔法罗密欧",  "A"));
    mCars.add(new Car("阿斯顿马丁",  "A"));
    mCars.add(new Car("ALPINA",  "A"));
    mCars.add(new Car("安凯客车",  "A"));

    mCars.add(new Car("本田", "B"));
    mCars.add(new Car("别克", "B"));
    mCars.add(new Car("奔驰",  "B"));
    mCars.add(new Car("宝马", "B"));
    mCars.add(new Car("保时捷",  "B"));
    mCars.add(new Car("比亚迪", "B"));
    mCars.add(new Car("北京", "B"));
    mCars.add(new Car("宾利",  "B"));
    mCars.add(new Car("巴博斯",  "B"));
    mCars.add(new Car("布加迪威龙", "B"));

    mCars.add(new Car("长安", "C"));
    mCars.add(new Car("长城",  "C"));

    mCars.add(new Car("大众", "D"));
    mCars.add(new Car("东南",  "D"));
    mCars.add(new Car("东风", "D"));
    mCars.add(new Car("DS", "D"));
    mCars.add(new Car("道奇", "D"));
    mCars.add(new Car("东风小康", "D"));
}

private void initView() {
    final LayoutInflater inflater = LayoutInflater.from(this);
    mRlv = (RecyclerView) findViewById(R.id.rlv);
    mRlv.setLayoutManager(new LinearLayoutManager(this));
    RlvAdapter rlvAdapter = new RlvAdapter(mCars);
    // 返回头布局的内容
    final NormalDecoration decoration = new NormalDecoration() {
        @Override
        public String getHeaderName(int i) {
            return mCars.get(i).headerName;
        }
    };
    // 自定义头布局,可不设置
    decoration.setOnDecorationHeadDraw(new NormalDecoration.OnDecorationHeadDraw() {
        @Override
        public View getHeaderView(final int i) {
            View inflate = inflater.inflate(R.layout.item_header, null);
            TextView tv = inflate.findViewById(R.id.tv);
            tv.setText(mCars.get(i).headerName);

            return inflate;
        }
    });
    mRlv.addItemDecoration(decoration);
    // 头布局的点击事件
    decoration.setOnHeaderClickListener(new NormalDecoration.OnHeaderClickListener() {
        @Override
        public void headerClick(int i) {
            Toast.makeText(MainActivity.this, mCars.get(i).headerName, Toast.LENGTH_SHORT).show();
            startActivity(new Intent(MainActivity.this,FlowActivity.class));
        }
    });
    mRlv.setAdapter(rlvAdapter);
}
粘性头部相当于是 RecyclerView 中两个条目之间的分割线,在数据如下情况下就可以使用粘性头部
import java.util.ArrayList;

public class NodeBean {
    String header;
    ArrayList<String> list;

    public NodeBean() {
    }

    public NodeBean(String header, ArrayList<String> list) {
        this.header = header;
        this.list = list;
    }

    public String getHeader() {

        return header;
    }

    public void setHeader(String header) {
        this.header = header;
    }

    public ArrayList<String> getList() {
        return list;
    }

    public void setList(ArrayList<String> list) {
        this.list = list;
    }
}

如果 item 条目中是简短的 String 类型的数据的时候还可以使用流式布局

流式布局.gif

流式布局里的内容就是每一个粘性头部下的数据放到了一起

1、FlowLayout(自定义 View,直接复制就可以,在 RecyclerView 的 item 布局里引用就可以)

import android.content.Context;
import android.util.AttributeSet;
import android.view.View;
import android.view.ViewGroup;

import java.util.ArrayList;
import java.util.List;

public class FlowLayout extends ViewGroup {

    private List<Line> mLines = new ArrayList<Line>(); // 用来记录描述有多少行 View
    private Line mCurrrenLine;  // 用来记录当前已经添加到了哪一行
    private int mHorizontalSpace    = 40;
    private int mVerticalSpace = mHorizontalSpace;
    private int mMaxLines = -1;

    public int getMaxLines() {
        return mMaxLines;
    }

    public void setMaxLines(int maxLines) {
        mMaxLines = maxLines;
    }

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

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

    public void setSpace(int horizontalSpace, int verticalSpace) {
        this.mHorizontalSpace = horizontalSpace;
        this.mVerticalSpace = verticalSpace;
    }

    public void clearAll(){
        mLines.clear();
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        // 清空
        mLines.clear();
        mCurrrenLine = null;

        int layoutWidth = MeasureSpec.getSize(widthMeasureSpec);

        // 获取行最大的宽度
        int maxLineWidth = layoutWidth - getPaddingLeft() - getPaddingRight();

        // 测量孩子
        int count = getChildCount();
        for (int i = 0; i < count; i++)
        {
            View view = getChildAt(i);

            // 如果孩子不可见
            if (view.getVisibility() == View.GONE)
            {
                continue;
            }

            // 测量孩子
            measureChild(view, widthMeasureSpec, heightMeasureSpec);

            // 往 lines 添加孩子
            if (mCurrrenLine == null)
            {
                // 说明还没有开始添加孩子
                mCurrrenLine = new Line(maxLineWidth, mHorizontalSpace);

                // 添加到 Lines 中
                mLines.add(mCurrrenLine);

                // 行中一个孩子都没有
                mCurrrenLine.addView(view);
            }
            else
            {
                // 行不为空,行中有孩子了
                boolean canAdd = mCurrrenLine.canAdd(view);
                if (canAdd) {
                    // 可以添加
                    mCurrrenLine.addView(view);
                }
                else {
                    // 不可以添加,装不下去
                    // 换行
                    if (mMaxLines >0){
                        if (mLines.size()<mMaxLines){
                            // 新建行
                            mCurrrenLine = new Line(maxLineWidth, mHorizontalSpace);
                            // 添加到 lines 中
                            mLines.add(mCurrrenLine);
                            // 将 view 添加到 line
                            mCurrrenLine.addView(view);
                        }
                    } else {
                        // 新建行
                        mCurrrenLine = new Line(maxLineWidth, mHorizontalSpace);
                        // 添加到 lines 中
                        mLines.add(mCurrrenLine);
                        // 将 view 添加到 line
                        mCurrrenLine.addView(view);
                    }
                }
            }
        }

        // 设置自己的宽度和高度
        int measuredWidth = layoutWidth;
        // paddingTop + paddingBottom + 所有的行间距 + 所有的行的高度

        float allHeight = 0;
        for (int i = 0; i < mLines.size(); i++)
        {
            float mHeigth = mLines.get(i).mHeigth;

            // 加行高
            allHeight += mHeigth;
            // 加间距
            if (i != 0)
            {
                allHeight += mVerticalSpace;
            }
        }

        int measuredHeight = (int) (allHeight + getPaddingTop() + getPaddingBottom() + 0.5f);
        setMeasuredDimension(measuredWidth, measuredHeight);
    }

    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b)
    {
        // 给 Child 布局 ---> 给 Line 布局

        int paddingLeft = getPaddingLeft();
        int offsetTop = getPaddingTop();
        for (int i = 0; i < mLines.size(); i++)
        {
            Line line = mLines.get(i);

            // 给行布局
            line.layout(paddingLeft, offsetTop);

            offsetTop += line.mHeigth + mVerticalSpace;
        }
    }

    class Line
    {
        // 属性
        private List<View> mViews = new ArrayList<View>(); // 用来记录每一行有几个 View
        private float mMaxWidth; // 行最大的宽度
        private float mUsedWidth; // 已经使用了多少宽度
        private float mHeigth; // 行的高度
        private float mMarginLeft;
        private float mMarginRight;
        private float mMarginTop;
        private float mMarginBottom;
        private float mHorizontalSpace; // View 和 view 之间的水平间距

        // 构造
        public Line(int maxWidth, int horizontalSpace) {
            this.mMaxWidth = maxWidth;
            this.mHorizontalSpace = horizontalSpace;
        }

        // 方法
        /**
         * 添加 view,记录属性的变化
         * 
         * @param view
         */
        public void addView(View view)
        {
            // 加载 View 的方法

            int size = mViews.size();
            int viewWidth = view.getMeasuredWidth();
            int viewHeight = view.getMeasuredHeight();
            // 计算宽和高
            if (size == 0)
            {
                // 说明还没有添加 View
                if (viewWidth > mMaxWidth)
                {
                    mUsedWidth = mMaxWidth;
                }
                else
                {
                    mUsedWidth = viewWidth;
                }
                mHeigth = viewHeight;
            }
            else
            {
                // 多个 view 的情况
                mUsedWidth += viewWidth + mHorizontalSpace;
                mHeigth = mHeigth < viewHeight ? viewHeight : mHeigth;
            }

            // 将 View 记录到集合中
            mViews.add(view);
        }

        /**
         * 用来判断是否可以将 View 添加到 line 中
         * 
         * @param view
         * @return
         */
        public boolean canAdd(View view)
        {
            // 判断是否能添加 View

            int size = mViews.size();

            if (size == 0) { return true; }

            int viewWidth = view.getMeasuredWidth();

            // 预计使用的宽度
            float planWidth = mUsedWidth + mHorizontalSpace + viewWidth;

            if (planWidth > mMaxWidth)
            {
                // 加不进去
                return false;
            }

            return true;
        }

        /**
         * 给孩子布局
         * 
         * @param offsetLeft
         * @param offsetTop
         */
        public void layout(int offsetLeft, int offsetTop)
        {
            // 给孩子布局

            int currentLeft = offsetLeft;

            int size = mViews.size();
            // 判断已经使用的宽度是否小于最大的宽度
            float extra = 0;
            float widthAvg = 0;
            if (mMaxWidth > mUsedWidth)
            {
                extra = mMaxWidth - mUsedWidth;
                widthAvg = extra / size;
            }

            for (int i = 0; i < size; i++)
            {
                View view = mViews.get(i);
                int viewWidth = view.getMeasuredWidth();
                int viewHeight = view.getMeasuredHeight();

                // 判断是否有富余
                if (widthAvg != 0)
                {
                    // 改变宽度,变为不改变,避免最后一行因 label 不足,单个 label 变宽
                    // int newWidth = (int) (viewWidth + widthAvg + 0.5f);
                    int newWidth = viewWidth;
                    int widthMeasureSpec = MeasureSpec.makeMeasureSpec(newWidth, MeasureSpec.EXACTLY);
                    int heightMeasureSpec = MeasureSpec.makeMeasureSpec(viewHeight, MeasureSpec.EXACTLY);
                    view.measure(widthMeasureSpec, heightMeasureSpec);

                    viewWidth = view.getMeasuredWidth();
                    viewHeight = view.getMeasuredHeight();
                }

                // 布局
                int left = currentLeft;
                int top = (int) (offsetTop + (mHeigth - viewHeight) / 2 +
                            0.5f);
                // int top = offsetTop;
                int right = left + viewWidth;
                int bottom = top + viewHeight;
                view.layout(left, top, right, bottom);

                currentLeft += viewWidth + mHorizontalSpace;
            }
        }
    }

}

2、RecyclerView 的 item 布局

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <com.example.lenovo.day04.widget.FlowLayout
        android:id="@+id/fl"
        android:layout_width="match_parent"
        android:layout_height="wrap_content">

    </com.example.lenovo.day04.widget.FlowLayout>

</LinearLayout>

3、在 RecyclerView 的适配器的 onBindViewHolder() 方法里

NodeBean 就是上面提到的数据的 bean 类

    @Override
    public void onBindViewHolder(@NonNull ViewHolder viewHolder, int i) {

        NodeBean nodeBean = recyclerviewList.get(i);

        ArrayList<String> list = nodeBean.getList();
        // 清空 fl 里的数据,不然会把其他集合里的数据添加进来
        viewHolder.fl.removeAllViews();
        for (int j = 0; j < list.size(); j++) {
            // 获取视图,视图可以自定义,可以添加自己想要的效果
            TextView label = (TextView) View.inflate(context, R.layout.item_label, null);
            // 获取数据
            final String data = list.get(j);
            // 绑定数据
            label.setText(data);
            
            // 监听方法
            label.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    
                }
            });

            // 加到容器中,parent 是 FlowLayout
            viewHolder.fl.addView(label);
        }

    }
item_label.xml
<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text=""
    android:maxLines="1"
    android:ellipsize="end"
    android:gravity="center"
    android:paddingLeft="12dp"
    android:paddingRight="12dp"
    android:background="@drawable/bg_e7e7e7_r30"
    android:paddingTop="2dp"
    android:paddingBottom="2dp"
    android:textColor="@color/c_636363"
    android:textSize="14sp"/>
流式布局相当于一个容器,把一堆数据放到一个容器中,它会自动把数据隔开

Activtiy 所有代码

public class FadeActivity extends AppCompatActivity {

    private static final String TAG = "FadeActivity";

    private String mUrl="https://www.v2ex.com/?tab=creative";

    private RecyclerView recyclerview;
    private ArrayList<NodeBean> recyclerviewList;
    private V2exNodeRecyclerViewAdapter v2exNodeRecyclerViewAdapter;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_fade);
        initView();
        initData();
    }

    private void initView() {
        // 系统自带的返回箭头
        getSupportActionBar().setDisplayHomeAsUpEnabled(true);
        getSupportActionBar().setHomeButtonEnabled(true);
        getSupportActionBar().setTitle("节点导航");

        recyclerview = (RecyclerView) findViewById(R.id.recyclerview);
        recyclerviewList = new ArrayList<>();

        setData();
    }

    private void setData() {
        recyclerview.setLayoutManager(new LinearLayoutManager(FadeActivity.this));
        v2exNodeRecyclerViewAdapter = new V2exNodeRecyclerViewAdapter(recyclerviewList, FadeActivity.this);
        // 返回头布局的内容
        final NormalDecoration decoration = new NormalDecoration() {
            @Override
            public String getHeaderName(int i) {
                return recyclerviewList.get(i).getHeader();
            }
        };
        // 自定义头布局,可不设置
//        decoration.setOnDecorationHeadDraw(new NormalDecoration.OnDecorationHeadDraw() {
//            @Override
//            public View getHeaderView(final int i) {
//                View inflate = LayoutInflater.from(getContext()).inflate(R.layout.item_header, null);
//                TextView tv = inflate.findViewById(R.id.tv);
//                tv.setText(mCars.get(i).header);
//
//                return inflate;
//            }
//        });

        recyclerview.addItemDecoration(decoration);
        // 头布局的点击事件
//        decoration.setOnHeaderClickListener(new NormalDecoration.OnHeaderClickListener() {
//            @Override
//            public void headerClick(int i) {
//                Toast.makeText(getContext(), mCars.get(i).header, Toast.LENGTH_SHORT).show();
//                //startActivity(new Intent(getContext(), FlowActivity.class));
//                startActivity(new Intent(getContext(), MaterialActivity.class));
//            }
//        });
        recyclerview.setAdapter(v2exNodeRecyclerViewAdapter);
    }

    private void initData() {
        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    Document doc = Jsoup.connect(mUrl).get();
                    Elements elements = doc.select("div#Main div.box");
                    Element element1 = elements.get(1);

                    final ArrayList<String> titles = new ArrayList<>();
                    final ArrayList<ArrayList<String>> arrayLists = new ArrayList<>();
                    Elements cells = element1.select("div.cell");
                    for (int i = 1; i < cells.size(); i++) {

                        // 分享与探索、V2EX、iOS、Geek、游戏、Apple、生活、Internet、城市
                        String title = cells.get(i).select("table tr td span").text();
                        titles.add(title);

                        ArrayList<String> arrayList = new ArrayList<>();
                        Elements articles = cells.get(i).select("table tr td a");
                        for (int j = 0; j < articles.size(); j++) {
                            String text = articles.get(j).text();
                            arrayList.add(text);
                        }
                        arrayLists.add(arrayList);
                    }

                    // 品牌
                    String title = element1.select("div.inner table tr td span").text();
                    titles.add(title);

                    ArrayList<String> arrayList = new ArrayList<>();
                    Elements select = element1.select("div.inner table tr td a");
                    for (int i = 0; i < select.size(); i++) {
                        String text = select.get(i).text();
                        arrayList.add(text);
                    }
                    arrayLists.add(arrayList);

                    runOnUiThread(new Runnable() {

                        private NodeBean nodeBean;
                        private ArrayList<String> list;
                        private String title;

                        @Override
                        public void run() {
                            ArrayList<NodeBean> nodeBeans = new ArrayList<>();
                            for (int i = 0; i < titles.size(); i++) {
                                title = titles.get(i);
                                nodeBean = new NodeBean();
                                nodeBean.setHeader(title);
                                nodeBeans.add(nodeBean);
                            }
                            for (int j = 0; j < arrayLists.size(); j++) {
                                list = arrayLists.get(j);
                                nodeBean.setList(list);
                                NodeBean nodeBean = nodeBeans.get(j);
                                nodeBean.setList(list);
                            }
                            Log.e(TAG, "run: " +arrayLists.size());
                            recyclerviewList.addAll(nodeBeans);
                            v2exNodeRecyclerViewAdapter.notifyDataSetChanged();
                        }
                    });

                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }).start();
    }
    
    // 返回箭头的监听
    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        switch (item.getItemId()) {
            case android.R.id.home:
                finish();
                break;
        }
        return true;
    }
}

一步一步使用自定义 ViewGroup 实现流式布局

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352