美文网首页高级UIAndroid
WanAndroid 导航页面 erticalTabLayout

WanAndroid 导航页面 erticalTabLayout

作者: 龙_君 | 来源:发表于2019-08-08 14:46 被阅读0次

请看效果图:

导航.png

依赖:

粘性头部局:
在Module的build.gradle里面添加依赖:
implementation 'com.github.qdxxxx:StickyHeaderDecoration:1.0.1'
然后在项目的build.gradle中添加:
allprojects {
    repositories {
        google()
        jcenter()
        maven { url 'https://jitpack.io' }//添加这个
    }
}
VerticalTabLayout垂直tab的依赖:
 //垂直 VerticalTabLayout
    implementation 'q.rorbin:VerticalTabLayout:1.2.5'

/RecyclerView的使用 依 赖 :/

    implementation 'com.android.support:recyclerview-v7:29.0.1'

接下来看官请看XML文件:

导航页面xml布局:

<?xml version="1.0" encoding="utf-8"?>
<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:orientation="horizontal"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:background="#FFFFFF"
    tools:context=".ui.wan.fragment.fragment.Wan_navigationFragment">



    <q.rorbin.verticaltablayout.VerticalTabLayout
        android:id="@+id/wan_navigation_vtab"
        android:layout_width="@dimen/x200"
        android:layout_height="match_parent"
        android:background="#EDEDED"
        app:indicator_color="#EDEDED"
        app:indicator_width="@dimen/x10"
        app:indicator_gravity="left"
        app:tab_height="@dimen/x100"
        android:visibility="visible"
        app:tab_mode="scrollable"/>




    <androidx.recyclerview.widget.RecyclerView
        android:id="@+id/wan_navigation_rlv"
        android:background="@color/cardview_light_background"
        android:layout_width="match_parent"
        android:layout_height="match_parent"/>





    <ProgressBar
        android:id="@+id/wan_navigation_pb"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerVertical="true"
        android:layout_centerHorizontal="true"/>

</LinearLayout>


这是一个自定义view,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:background="#FFFFFF"
    android:layout_height="match_parent">

    <com.long.wanandroids.ui.wan.Wan_navigation.view.FlowLayout
        android:id="@+id/fl"
        android:background="#FFFFFF"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:paddingTop="12dp"
        android:paddingBottom="12dp" />

</LinearLayout>

自定义viewFlowLayout:

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

创建一个TextView的xml文件 item_laber:

<?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="123456"
    android:maxLines="1"
    android:ellipsize="end"
    android:gravity="center"
    android:paddingLeft="12dp"
    android:paddingRight="12dp"
    android:background="@drawable/shape"
    android:paddingBottom="2dp"
    android:paddingTop="2dp"
    android:textColor="#0F0E0E"
    android:textSize="14sp"/>

导航页面的Fragment:

/**
 * A simple {@link Fragment} subclass.
 */
public class Wan_navigationFragment extends BaseFragment implements IView {


    @BindView(R.id.wan_navigation_vtab)
    VerticalTabLayout wanNavigationVtab;
    @BindView(R.id.wan_navigation_pb)
    ProgressBar wanNavigationPb;
    @BindView(R.id.wan_navigation_rlv)
    RecyclerView wanNavigationRlv;
    private Wan_navigationPresenter wan_navigationPresenter;
    private ArrayList<String> titles;
    private List<WanNavigationinfo.DataBean> list;
    private WanNavigationAdapter wanNavigationAdapter;
    private LinearLayoutManager manager;

    @Override
    protected void initData(Bundle savedInstanceState) {
        wan_navigationPresenter = new Wan_navigationPresenter(this);
        wan_navigationPresenter.startLoadData();
        titles = new ArrayList<>();

        manager = new LinearLayoutManager(BaseApp.getContext());
        wanNavigationRlv.setLayoutManager(manager);


        list = new ArrayList<>();
        wanNavigationAdapter = new WanNavigationAdapter(BaseApp.getContext(), list);

        wanNavigationRlv.setAdapter(wanNavigationAdapter);


        //RecyclerView和tab栏联动       
        wanNavigationRlv.addOnScrollListener(new RecyclerView.OnScrollListener() {
            @Override
            public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newState) {
                super.onScrollStateChanged(recyclerView, newState);
                wanNavigationVtab.setTabSelected(manager.findFirstVisibleItemPosition());
            }

        });
        //tab栏和RecyclerView联动 
        wanNavigationVtab.addOnTabSelectedListener(new VerticalTabLayout.OnTabSelectedListener() {
             @Override
             public void onTabSelected(TabView tab, int position) {
                 manager.scrollToPositionWithOffset(position,0);
             }
             @Override
             public void onTabReselected(TabView tab, int position) {
             }
         });


    }

    //布局ID
    @Override
    protected int getFragmentResLayoutId() {
        return R.layout.fragment_wan_navigation;
    }

    //显示加载中动画
    @Override
    public void showLoading() {
        wanNavigationPb.setVisibility(View.VISIBLE);
    }

    //关闭加载中动画
    @Override
    public void hideLoading() {
        wanNavigationPb.setVisibility(View.GONE);
    }

    //返回解析结果
    @Override
    public void loadDataHttpSucess(Object o) {
        if (o instanceof WanNavigationinfo) {
            WanNavigationinfo wn = (WanNavigationinfo) o;
            final List<WanNavigationinfo.DataBean> data = wn.getData();

            for (int i = 0; i < data.size(); i++) {
                titles.add(data.get(i).getName());
            }


            //设置粘性头部局,返回对应的头布局
            NormalDecoration normalDecoration = new NormalDecoration() {

                @Override
                public String getHeaderName(int i) {
                    return data.get(i).getName();
                }
            };

            //设置item分割
            wanNavigationRlv.addItemDecoration(normalDecoration);
            //刷新数据
            wanNavigationAdapter.addData(data);
        }


        //垂直Tab
        wanNavigationVtab.setTabAdapter(new TabAdapter() {
            @Override
            public int getCount() {
                return titles.size();
            }

            @Override
            public ITabView.TabBadge getBadge(int position) {
                return null;
            }

            @Override
            public ITabView.TabIcon getIcon(int position) {
                return null;
            }

            @Override
            public ITabView.TabTitle getTitle(int position) {
                return new ITabView.TabTitle.Builder()
                        .setContent(titles.get(position))//从集合中获取标题
                        .setTextColor(Color.RED, Color.BLACK)
                        .build();
            }

            @Override
            public int getBackground(int position) {
                return 0;
            }
        });
    }

    @Override
    public void loadDataFaile(String errorMsg) {
        showToast(errorMsg);
    }

    @Override
    public void showToast(String msg) {
        Toast.makeText(BaseApp.getContext(), msg, Toast.LENGTH_LONG).show();
    }
}

导航页面的适配器Adapter:


public class WanNavigationAdapter extends RecyclerView.Adapter<WanNavigationAdapter.ViewHolder> {
    private Context context;
    private List<WanNavigationinfo.DataBean> list;

    public WanNavigationAdapter(Context context, List<WanNavigationinfo.DataBean> list) {
        this.context = context;
        this.list = list;
    }

    public void addData(List<WanNavigationinfo.DataBean> dataBeans) {
        list.addAll(dataBeans);
        notifyDataSetChanged();
    }


    @NonNull
    @Override
    public ViewHolder onCreateViewHolder(@NonNull ViewGroup viewGroup, int i) {
        View inflate = LayoutInflater.from(context).inflate(R.layout.wannavigationitem, null);
        return new ViewHolder(inflate);
    }

    @Override
    public void onBindViewHolder(@NonNull ViewHolder viewHolder, int i) {
        WanNavigationinfo.DataBean dataBean = list.get(i);

        final List<WanNavigationinfo.DataBean.ArticlesBean> articles = dataBean.getArticles();

        if (articles != null && articles.size() > 0) {
            for (int j = 0; j < articles.size(); j++) {

                TextView label = (TextView) LayoutInflater.from(context).inflate(R.layout.item_laber, null);
                label.setText(articles.get(j).getTitle());


                final int finalI = j;

                //标签点击事件
                label.setOnClickListener(new View.OnClickListener() {
                    @Override
                    public void onClick(View v) {
//                        ToastUtil.showLong(articles.get(finalI).getTitle());
                    }
                });
                //将TextView标签添加到flowlayout
                viewHolder.fl.addView(label);
            }
        }

    }

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


    public class ViewHolder extends RecyclerView.ViewHolder {
         public FlowLayout fl;

        public ViewHolder(@NonNull View itemView) {
            super(itemView);
             fl = itemView.findViewById(R.id.fl);
        }
    }
}

写的不好,各位看官请多多包含!!! > - <

休息一下吧!


图1.jpg

相关文章

网友评论

    本文标题:WanAndroid 导航页面 erticalTabLayout

    本文链接:https://www.haomeiwen.com/subject/mflxjctx.html