博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
谈谈Android中的Divider
阅读量:6278 次
发布时间:2019-06-22

本文共 10413 字,大约阅读时间需要 34 分钟。

hot3.png

在Android应用开发中会经常碰到一个叫divider的东西,就是两个View之间的分割线。最近工作中注意到这个divider并分析了一下,竟然发现内有乾坤,惊为天人…

ListView的divider

1. 定制divider的边距

ListView的divider默认是左右两头到底的,如何简单的设置一个边距呢?

利用inset或者layer-list都可以简单的实现,代码如下:

    
        
    
    
        
            
            

其中inset除了左边距insetLeft, 还有insetTop、insetRight、insetBottom, 效果图:

谈谈Android中的Divider

2. 最后一项的divider

很多同学可能发现了,ListView最后一项的divider有时候有,有时候又没有。

我画个图大家就都能理解了:

谈谈Android中的Divider

上面是数据不足的显示效果,如果数据满屏的话,都是看不多最后的divider的。

真相是,当ListView高度是不算最后一项divider的,所以只有在match_parent的情况下,ListView的高度是有余的,才能画出最后的那个divider。

ps:网上很多资料,把最后一项的divider和footerDividersEnabled混在一起了,这个是不对的,两个从逻辑上是独立的,类似的还有一个headerDividersEnabled,headerDividersEnabled和footerDividersEnabled不会影响到默认情况下最后的divider的绘制,他们是给header和footer专用的,特此说明。

的Divider

RecyclerView的Divider叫做ItemDecoration,RecyclerView.ItemDecoration本身是一个抽象类,官方没有提供默认实现。

官方的Support7Demos例子中有个DividerItemDecoration, 我们可以直接参考一下,位置在sdk的这里:

extras/android/support/samples/Support7Demos/src/…/…/decorator/DividerItemDecoration.java

但是这个DividerItemDecoration有三个问题:

  1. 只支持系统默认样式,不支持自定义Drawable类型的divider

  2. 里面的算法对于无高宽的Drawable(比如上面用到的InsetDrawable)是画不出东西的

  3. 水平列表的Divider绘制方法drawHorizontal()的right计算有误,导致垂直Divider会绘制不出来,应该改为:final int right = left + mDivider.getIntrinsicWidth();;

针对这几个问题,我修复并增强了一下:

import android.content.Context;import android.content.res.TypedArray;import android.graphics.Canvas;import android.graphics.Rect;import android.graphics.drawable.Drawable;import android.support.v4.view.ViewCompat;import android.support.v7.widget.LinearLayoutManager;import android.support.v7.widget.RecyclerView;import android.view.View;/** * RecyclerView的ItemDecoration的默认实现 * 1. 默认使用系统的分割线 * 2. 支持自定义Drawable类型 * 3. 支持水平和垂直方向 * 4. 修复了官方垂直Divider显示的bug * 扩展自官方android sdk下的Support7Demos下的DividerItemDecoration */public class DividerItemDecoration extends RecyclerView.ItemDecoration {    private static final int[] ATTRS = new int[]{        android.R.attr.listDivider    };    public static final int HORIZONTAL_LIST = LinearLayoutManager.HORIZONTAL;    public static final int VERTICAL_LIST = LinearLayoutManager.VERTICAL;    private Drawable mDivider;    private int mWidth;    private int mHeight;    private int mOrientation;    public DividerItemDecoration(Context context, int orientation) {        final TypedArray a = context.obtainStyledAttributes(ATTRS);        mDivider = a.getDrawable(0);        a.recycle();        setOrientation(orientation);    }    /** * 新增:支持自定义dividerDrawable * * @param context * @param orientation * @param dividerDrawable */    public DividerItemDecoration(Context context, int orientation, Drawable dividerDrawable) {        mDivider = dividerDrawable;        setOrientation(orientation);    }    public void setOrientation(int orientation) {        if (orientation != HORIZONTAL_LIST && orientation != VERTICAL_LIST) {            throw new IllegalArgumentException("invalid orientation");        }        mOrientation = orientation;    }    /** * 新增:支持手动为无高宽的drawable制定宽度 * @param width */    public void setWidth(int width) {        this.mWidth = width;    }    /** * 新增:支持手动为无高宽的drawable制定高度 * @param height */    public void setHeight(int height) {        this.mHeight = height;    }    @Override        public void onDraw(Canvas c, RecyclerView parent) {            if (mOrientation == VERTICAL_LIST) {                drawVertical(c, parent);            } else {                drawHorizontal(c, parent);            }        }    public void drawVertical(Canvas c, RecyclerView parent) {        final int left = parent.getPaddingLeft();        final int right = parent.getWidth() - parent.getPaddingRight();        final int childCount = parent.getChildCount();        for (int i = 0; i < childCount; i++) {            final View child = parent.getChildAt(i);            final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child                .getLayoutParams();            final int top = child.getBottom() + params.bottomMargin +                Math.round(ViewCompat.getTranslationY(child));            final int bottom = top + getDividerHeight();            mDivider.setBounds(left, top, right, bottom);            mDivider.draw(c);        }    }    public void drawHorizontal(Canvas c, RecyclerView parent) {        final int top = parent.getPaddingTop();        final int bottom = parent.getHeight() - parent.getPaddingBottom();        final int childCount = parent.getChildCount();        for (int i = 0; i < childCount; i++) {            final View child = parent.getChildAt(i);            final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child                .getLayoutParams();            final int left = child.getRight() + params.rightMargin +                Math.round(ViewCompat.getTranslationX(child));            final int right = left + getDividerWidth();            mDivider.setBounds(left, top, right, bottom);            mDivider.draw(c);        }    }    @Override        public void getItemOffsets(Rect outRect, int itemPosition, RecyclerView parent) {            if (mOrientation == VERTICAL_LIST) {                outRect.set(0, 0, 0, getDividerHeight());            } else {                outRect.set(0, 0, getDividerWidth(), 0);            }        }    private int getDividerWidth() {        return mWidth > 0 ? mWidth : mDivider.getIntrinsicWidth();    }    private int getDividerHeight() {        return mHeight > 0 ? mHeight : mDivider.getIntrinsicHeight();    }}

使用如下:

// 默认系统的dividerdividerItemDecoration = new DividerItemDecoration(this, DividerItemDecoration.VERTICAL_LIST);// 自定义图片drawable分的dividerdividerItemDecoration = new DividerItemDecoration(this, DividerItemDecoration.VERTICAL_LIST, getResources().getDrawable(R.drawable.ic_launcher));// 自定义无高宽的drawable的divider - 垂直列表dividerItemDecoration = new DividerItemDecoration(this, DividerItemDecoration.VERTICAL_LIST, new ColorDrawable(Color.parseColor("#ff00ff")));dividerItemDecoration.setHeight(1);// 自定义无高宽的drawable的divider - 水平列表dividerItemDecoration = new DividerItemDecoration(this, DividerItemDecoration.HORIZONTAL_LIST, new ColorDrawable(Color.parseColor("#ff00ff")));dividerItemDecoration.setWidth(1);// 自定义带边距且无高宽的drawable的divider(以上面InsetDrawable为例子)// 这个地方也可以在drawable的xml文件设置size指定宽高,效果一样dividerItemDecoration = new DividerItemDecoration(this, DividerItemDecoration.HORIZONTAL_LIST, getResources().getDrawable(R.drawable.list_divider));dividerItemDecoration.setWidth(DisplayLess.$dp2px(16) + 1);

手动的Divider

有的时候没有系统控件的原生支持,只能手动在两个view加一个divider,比如,设置界面每项之间的divider,水平平均分隔的几个view之间加一个竖的divider等等。

无论横的竖的,都非常简单,定一个View,设置一个background就可以了,正常情况下没什么好说的。

下面我们来考虑一种常见设置界面,这种设置界面的分割线是有左边距的,比如微信的设置界面,我相信绝大部分人的布局代码都是这样实现的:

    
    
        
            
                
        
            
            

效果图如下,顺便我们也看看它的Overdraw状态:

谈谈Android中的Divider

通过分析Overdraw的层次,我们发现为了一个小小的边距,设置了整个groud_container的背景,从而导致了一次Overdraw。

能不能优化掉这个Overdraw?答案是肯定的。

背景肯定要去掉,但是这个左边距的View就不能这么简单的写了,需要自定义一个View,它要支持能把左边距的空出的16dp的线用list_item_normal的颜色值绘制一遍,这样才能看的出左边距。

这个View具体代码如下:

import android.content.Context;import android.content.res.TypedArray;import android.graphics.Canvas;import android.graphics.Color;import android.graphics.Paint;import android.util.AttributeSet;import android.util.TypedValue;import android.view.View;import com.jayfeng.lesscode.core.R;public class SpaceDividerView extends View {    private int mSpaceLeft = 0;    private int mSpaceTop = 0;    private int mSpaceRight = 0;    private int mSpaceBottom = 0;    private int mSpaceColor = Color.TRANSPARENT;    private Paint mPaint = new Paint();    public SpaceDividerView(Context context) {        this(context, null);    }    public SpaceDividerView(Context context, AttributeSet attrs) {        this(context, attrs, 0);    }    public SpaceDividerView(Context context, AttributeSet attrs, int defStyleAttr) {        super(context, attrs, defStyleAttr);        TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.SpaceDividerView, defStyleAttr, 0);        mSpaceLeft = a.getDimensionPixelSize(R.styleable.SpaceDividerView_spaceLeft,                (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 0, getResources().getDisplayMetrics()));        mSpaceTop = a.getDimensionPixelSize(R.styleable.SpaceDividerView_spaceTop,                (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 0, getResources().getDisplayMetrics()));        mSpaceRight = a.getDimensionPixelSize(R.styleable.SpaceDividerView_spaceRight,                (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 0, getResources().getDisplayMetrics()));        mSpaceBottom = a.getDimensionPixelSize(R.styleable.SpaceDividerView_spaceBottom,                (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 0, getResources().getDisplayMetrics()));        mSpaceColor = a.getColor(R.styleable.SpaceDividerView_spaceColor, Color.TRANSPARENT);        a.recycle();        mPaint.setColor(mSpaceColor);    }    @Override    protected void onDraw(Canvas canvas) {        super.onDraw(canvas);        if (mSpaceLeft > 0) {            canvas.drawRect(0, 0, mSpaceLeft, getMeasuredHeight(), mPaint);        }        if (mSpaceTop > 0) {            canvas.drawRect(0, 0, getMeasuredWidth(), mSpaceTop, mPaint);        }        if (mSpaceRight > 0) {            canvas.drawRect(getMeasuredWidth() - mSpaceRight, 0, getMeasuredWidth(), getMeasuredHeight(), mPaint);        }        if (mSpaceBottom > 0) {            canvas.drawRect(0, getMeasuredHeight() - mSpaceBottom, getMeasuredWidth(), getMeasuredHeight(), mPaint);        }    }}

用这个SpaceDividerView我们重写一下上面的布局代码:

    
        
            
                
        
            
            

效果图和Overdraw状态如下:

谈谈Android中的Divider

界面中group_container那块由之前的绿色变成了蓝色,说明减少了一次Overdraw。

上述情况下,SpaceDividerView解耦了背景色,优化了Overdraw,而且这个SpaceDividerView也是支持4个方向的,使用起来特别方便。

阴影divider

阴影分割线的特点是重叠在下面的view之上的,它的目的是一种分割线的立体效果。

谈谈Android中的Divider

使用RelativeLayout并控制上边距离可以实现:

    
    
        
    

虽然再简单不过了,还是稍微分析一下,header包括内容48dp和阴影8dp,那么marginTop就是48dp了。

问啊-定制化IT,服务,,开发编程社交头条 官方网站:

QQ群290551701 聚集很多互联网精英,技术总监,架构师,项目经理!开源技术研究,欢迎业内人士,大牛及新手有志于从事IT行业人员进入!

转载于:https://my.oschina.net/u/2394328/blog/630420

你可能感兴趣的文章
【转】利用mybatis-generator自动生成代码
查看>>
架构师应该了解的知识1
查看>>
在Flex (Flash)中嵌入HTML 代码或页面—Flex IFrame
查看>>
防止Direct Input获取多次输入
查看>>
Interspeech 2017 | Self-adaptive Speech Recognition Technology
查看>>
Linux中MySQL数据库max_allowed_packet的调整
查看>>
MySQL 学习笔记 二
查看>>
Host prepare for your automation work
查看>>
Thinkphp中field和getField
查看>>
AngularJS之初级Route【一】(六)
查看>>
QTP的那些事--采用DOM,描述性编程获取指定的对象
查看>>
linux异步通信之epoll【转】
查看>>
前端自学路线之js篇
查看>>
C++:运算符重载函数之友元运算符重载
查看>>
ANT task之Junit、JunitReport
查看>>
selenium的那些事--运行报错
查看>>
谋求职业发展,是“走”还是“留”
查看>>
SpreadJS 在 Angular2 中支持绑定哪些属性?
查看>>
Lucene 定义
查看>>
硅谷返乡大潮,AI海归大调查:薪酬、发展、故乡情
查看>>