Drawable及其子类

在Android系统中,Drawable是一个抽象类,它是所有Drawable对象的基类,它有很多子类,比较常见的有ShapeDrawable、BitmapDrawable、ColorDrawable等。

Drawable的内部宽高尺寸是通过getIntrinsicWidth和getIntrinsicHeight方法来获取的,但并不是所有Drawable都有内部宽高尺寸这个参数的,比如ColorDrawable和ShapeDrawable就没有,而图片形成的Drawable就有,它的内部宽高就是图片的宽高。

  • BitmapDrawable表示的就是一张图片,使用时我们除了可以直接引用原始的图片,还可以通过xml描述BitmapDrawable,比如:
<?xml version="1.0" encoding="utf-8"?>
<bitmap xmlns:android="http://schemas.android.com/apk/res/android"
    android:antialias="true"
    android:dither="true"
    android:filter="true"
    android:gravity="center"
    android:mipMap="false"
    android:src="@drawable/rn_finish"
    android:tileMode="clamp" />

其中的属性是不是好像在哪见过?没错,在自定义View时,我们会使用Paint,Paint就可以设置这些属性,基本上这些属性的含义都是一样的。antialias表示是否抗锯齿,抗锯齿的效果是在图片边缘会表现的更圆润,而实际上加了抗锯齿效果边缘会变的模糊,所以看起来会圆滑。dither表示是否开启抖动效果,当图片的像素配置和手机屏幕的像素配置不一致时,开启抖动效果可以让高质量的图片在低质量的屏幕上还能保持较好的显示效果,比如图片的色彩模式为ARGB8888,但手机屏幕支持的色彩模式为RGB555,如果开启抖动效果就可以让图片显示不会过于失真。filter表示是否开启过滤效果,当图片尺寸被拉伸或者压缩时,开启过滤效果可以保持较好的显示效果。src表示图片的资源id。gravity表示图片在容器中的位置。tileMode表示图片的平铺模式,默认是disbaled,另外还有三个选项mirror、repeat、clamp,repeat是重复,mirror是对称翻转,clamp是延伸。

另外NinePatchDrawable表示的是一张.9格式的图片,.9图片可以自动根据所需的宽高进行相应的缩放来保证不失真。同样它也可以使用xml来描述,在xml中的节点名是nine-patch,其他属性和bitmap是一样的,而且在bitmap标签中也可以使用.9图片,所以BitampDrawable也可以表示一个.9格式的图片。

  • ShapeDrawable可以通过颜色来构造图形,比如当我们要为一个按钮加上一个圆角的纯色背景时,找UI切图太麻烦了,我们完全可以使用ShapeDrawable来实现。
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
    android:shape="rectangle">

    <corners
        android:bottomLeftRadius="5dp"
        android:bottomRightRadius="5dp"
        android:radius="20dp"
        android:topLeftRadius="5dp"
        android:topRightRadius="5dp" />

    <gradient
        android:angle="90"
        android:centerColor="@color/colorPrimaryDark"
        android:centerX="100dp"
        android:centerY="100dp"
        android:endColor="@color/colorAccent"
        android:gradientRadius="10dp"
        android:startColor="@color/colorPrimary"
        android:type="radial"
        android:useLevel="false" />

    <padding
        android:bottom="10dp"
        android:left="10dp"
        android:right="10dp"
        android:top="10dp" />


    <size
        android:width="200dp"
        android:height="200dp" />


    <solid android:color="@color/colorPrimary" />


    <stroke
        android:width="10dp"
        android:color="@color/colorAccent"
        android:dashGap="1dp"
        android:dashWidth="2dp" />

</shape>

以上是shape标签下所有的属性和子标签了,分别解释下其意思。

  • 属性shape表示图形的形状,共有四个可选值,line是直线,rectangle是矩形,oval是椭圆形,ring是圆环,默认值是矩形,其中line和ring这两个选项必须通过stroke标签来确定形状的宽度和颜色等信息,否则可能会无法显示正常。(而且ring这个形状还有5个额外的属性,innerRaius表示圆环的内半径,thickness表示圆环的厚度,innerRadiusRatio表示内半径占整个Drawable宽度的比例,默认是9,thicknessRatio表示厚度占整个Drawable宽度的比例,默认是3,useLevel表示是否使用等级,一般都是false。)

  • corners表示圆角半径,这个标签只适合rectangle形状,radius表示四个角设置为相同的圆角半径,还有四个分别表示四个角的圆角半径的属性topLeftRaduis、topRightRadius、bottomLeftRadius、bottomRightRadius,如果设置了这四个中的某一个,那么它会覆盖前面的radius属性所设置的值。

  • solid是使用纯色填充图形,只有一个属性color,表示要填充图形的颜色。

  • gradient表示图形的渐变效果,它与solid标签是不能同时使用的,solid是使用纯色填充图形,gradient则是渐变,它几个属性的含义如angle表示渐变的角度,默认是0,其值必须是45的倍数,0表示从左到右,90表示从下到上。centerX表示渐变中心点的横坐标,centerY表示渐变中心点的纵坐标,startColor表示渐变的起始色,endColor表示渐变的结束色,centerColor表示渐变的中间色,gradientRadius表示渐变半径,这个属性只有当type属性值为radial时才有效,type有三个选项,linear表示线性渐变,radial表示辐射渐变,也就是以圆环的形状渐变,sweep表示扫描渐变。

  • stroke表示图形的描边,width表示描边的宽度,color表示描边的颜色,这俩属性可以决定我们背景shape的边框颜色和边框粗细,dashWidth表示组成虚线的线段的宽度,dashGap表示虚线线段之间的间隔,

  • size表示shape的大小,其实就是给shape设置固有的宽高,我们知道ShapeDrawable本身是没有固有宽高的,但是如果设置了size,那么它就有了,不过当把它设置给View的背景时,还是会被拉伸或缩小来适应View的宽高的。

LayerDrawable对应的xml标签是<layer-list>,看字面意思我们就知道它是一种层次化的Drawable集合,通过将不同的Drawable放在不同的层上来达到一种叠加效果。

<?xml version="1.0" encoding="utf-8"?>
<layer-list xmlns:android="http://schemas.android.com/apk/res/android">

    <item>
        <shape android:shape="rectangle">
            <solid android:color="#0ac39e" />
        </shape>
    </item>


    <item android:bottom="6dp">
        <shape android:shape="rectangle">
            <solid android:color="#ffffff" />
        </shape>
    </item>

    <item
        android:bottom="1dp"
        android:left="1dp"
        android:right="1dp">
        <shape android:shape="rectangle">
            <solid android:color="#ffffff" />
        </shape>
    </item>

</layer-list>

其中item的left、right、top和bottom属性表示相对于View的上下左右的偏移量,item也可以直接使用drawable属性来引用一个已有的Drawable资源。另外下面的item会覆盖上面的item,达到叠加效果。

StateListDrawable对应于<selector>标签,它和LayerDrawable一样也是表示Drawable的集合,它每个Drawable都对应着一种状态,系统会根据View不同的状态来选择合适的Drawable。

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

    <item android:drawable="@color/colorAccent" android:state_checked="true" />
    <item android:drawable="@color/colorAccent" android:state_focused="true" />
    <item android:drawable="@color/colorAccent" android:state_enabled="true" />
    <item android:drawable="@color/colorAccent" android:state_hovered="true" />
    <item android:drawable="@color/colorAccent" android:state_window_focused="true" />
    <item android:drawable="@color/colorPrimary" />

</selector>

selector标签本身几个属性的含义:

  • constantSize表示StateListDrawable的固有大小是否随着状态的改变而改变,true表示不改变,默认为false。
  • dither表示是否开启抖动效果,这个和BitmapDrawable中的一样,默认是true。
  • variablePadding表示StateListDrawable的padding是否随着状态的改变而改变,true表示随着状态的改变而改变,false表示StateListDrawable的padding是内部所有Drawable的padding的最大值,默认为false。

对于常使用的几种状态,state_pressed表示按压,state_enabled表示当前可用,state_checked表示选中,state_focused表示获取到焦点,state_selected表示已选择。其他几个基本不会使用到。

LevelListDrawable对应于<level-list>标签,它也表示一个Drawable集合,集合中的Drawable都有一个level,根据不同的level,LevelListDrawable会切换到对应的Drawable。

<?xml version="1.0" encoding="utf-8"?>
<level-list xmlns:android="http://schemas.android.com/apk/res/android">

    <item
        android:drawable="@color/colorAccent"
        android:maxLevel="10"
        android:minLevel="1" />

    <item
        android:drawable="@color/colorPrimary"
        android:maxLevel="100"
        android:minLevel="11" />

    <item
        android:drawable="@color/colorPrimaryDark"
        android:maxLevel="1000"
        android:minLevel="101" />

</level-list>

它的每个item都表示一个Drawable,通过maxLevel和minLevel来控制level,level数值在0~10000之间,最小是0,最大10000,默认是0。

TransitionDrawable对应xml中的<transition>标签,它用于两个Drawable之间实现淡入淡出的效果。

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

    <item android:drawable="@color/colorAccent" />
    <item android:drawable="@color/colorPrimaryDark" />

</transition>

实际使用中是将上面的TransitionDrawable设置为View的背景,然后在代码中通过View获取其background,强转成TransitionDrawable,然后调用其startTransition(int duration)方法来实现Drawable切换的效果,duration是完成切换的时间。也可以通过startTransition结合reverseTransition来实现其反向切换过程。

InsetDrawable对应xml中的<inset>标签,它可以将其他Drawable内嵌到自己中,并可以在周围留出一定的间距。当一个View需要自己的背景比自己的实际区域小时就可以采用InsetDrawable来实现。

<?xml version="1.0" encoding="utf-8"?>
<inset xmlns:android="http://schemas.android.com/apk/res/android"
    android:insetBottom="10dp"
    android:insetLeft="10dp"
    android:insetRight="10dp"
    android:insetTop="10dp">

    <shape android:shape="rectangle">
        <solid android:color="@color/colorAccent" />
    </shape>

</inset>

ScaleDrawable对应xml中的<scale>标签,它可以根据自己的level将指定的Drawable缩放到一定比例。

<?xml version="1.0" encoding="utf-8"?>
<scale xmlns:android="http://schemas.android.com/apk/res/android"
    android:drawable="@drawable/rn_finish"
    android:scaleGravity="center"
    android:scaleHeight="25%"
    android:scaleWidth="30%">

</scale>

其中android:scaleHeight和android:scaleWidth分别表示对指定Drawable宽高的缩放比例,以百分比的形式表示。ScaleDrawable的level为0时不可见,0也是默认值,只有当等级不等于0时ScaleDrawable才会可见。

另外ScaleDrawable的level最大值为10000,最大时就没有缩放的效果,level越大,内部的Drawable看起来就越大,如果ScaleDrawable的xml中所定义的缩放比越大,那么内部的Drawable看起来就越小。

TextView view = (TextView) findViewById(R.id.view);
ScaleDrawable scaleDrawable = (ScaleDrawable) view.getBackground();
scaleDrawable.setLevel(9999);

这里要注意的是必须将level设置为大于0,少了这一步,ScaleDrawable将无法显示。

ClipDrawable对应xml中的<clip>标签,它根据自己当前的level来裁剪另一个Drawable,裁剪方向可以通过clipOrientation和gravity共同控制,clipOrientation是裁剪方向,gravity表示Drawable在容器内的位置以及影响开始裁剪的位置。

<?xml version="1.0" encoding="utf-8"?>
<clip xmlns:android="http://schemas.android.com/apk/res/android"
    android:clipOrientation="vertical"
    android:drawable="@drawable/rn_finish"
    android:gravity="top">

</clip>
<ImageView
    android:id="@+id/imageview"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:src="@drawable/clip" />

clipOrientation="vertical"表示竖直方向裁剪,gravity=“top”表示从底部开始裁剪,接着在代码中设置ClipDrawable的level:

ImageView imageView = (ImageView) findViewById(R.id.imageview);
ClipDrawable clipDrawable = (ClipDrawable) imageView.getDrawable();
clipDrawable.setLevel(10000);

ClipDrawable的level为0时表示完全裁剪,为10000时表示不裁剪,范围也是0~10000,设置8000表示裁剪了20%,设置3000表示裁剪了70%。等级越大裁剪的区域越小。

推荐阅读更多精彩内容

  • 概述 今天我们来探究一下android的样式。其实,几乎所有的控件都可以使用 background属性去引用自定义...
    CokeNello阅读 3,075评论 1 17
  • 转载自Keegan小钢并标明原文链接:http://keeganlee.me/post/android/20150...
    坚持编程_lyz阅读 329评论 0 1
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 141,825评论 18 609
  • 滴滴刚刚拿下Uber中国就出了国泰君安的罗生门事件,当然事实真相到底如何也只有当事人心里最清楚。从滴滴目前的状态来...
    耿彪阅读 53评论 0 1
  • 突然脑袋很沉,,,,只想回到寝室睡觉,没有其他想法,,,,hhhhhh,最近兴奋的不正常,12点半睡,6点起。
    瓜宁阅读 23评论 0 0