Shape的使用(自定义绘制)

说起shape,大家应该或多或少都知道并用过,有什么用,我们平常在开发当中,通常会遇到这样的情况,就是会给控件增加一个背景,比如button,textview等
可以说shape就是一个形状定义工具。是xml绘图当中非常重要的一个工具

代码也是很简单的,这里使用到了shape的三个特性。好了到这里我已经简单介绍shape的作用以及用法,但是shape能做的可远远不止这些,下面就让我们正式开始学习吧!
首先你需要明白:使用shape一般是用来定义形状的,可以在xml上绘图,意思就是shape的表现形式就是一个xml文件,这个xml文件一般是放在drawable文件目录下,然后可以直接引用作为控件的背景。那么如何定义这样的一个xml文件呢?
在你项目的drawable目录上右击新建Drawable resource file,这代表新建一个可绘制图形资源,你会得到如下代码文件。

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

</selector>

切换成shape

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

</shape>

这样我们就得到了一个shape的xml文件了,我们之前说过了,shape是用来定义形状的,那么它可以定义哪些形状呢?分别有如下几种形状

形状

1·rectangle--矩形,总觉得这个用的比较多
2·line---线,这个只能是水平的,不能定义竖直的。
3·oval---椭圆,我们一般是定义一个正圆。
4·ring---圆环,可以定义加载控件。

这是shape可以定义的几种形状,而且他们都离不开如下几种特性(我用导图整理了一下,方便查看)


形状图

以上就是shape的六种特性了,可以说,学习shape就是学习以上六种特性。下面我将一步一步的讲解以上六种特性的使用以及实现效果,我开篇也说了,这篇文章不仅教大家如何使用shape,更会告诉初学者如何学习shape,那学习shape的关键在哪呢?那就是你一定要动手实践,自己写代码看看效果。

首先,我们以最常用的rectangle为例来讲解以上特性的使用,这个时候你就需要在shape的根元素指定你要绘制的shape类型,如下:

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

</shape>

我们这里就以开篇提到的那个button按钮的效果来实例操作一下,首先你要明确我们是利用shape来给button按钮绘制一个背景,所以我们先要定义这个背景的宽和高,这用到shape的哪个特性呢?(如果忘记都有哪些特性,可以浏览之前给的那张导图,建议下载下来,对照查看),我们可以看到在shape的特性当中有一个size,对这就是用来定义我们所要绘制图形的大小了,那我们来实际操作一下吧!

<!--大小-->
<size android:width="600dp" android:height="100dp"></size>

这行代码意思是定义一个宽高比例为6:1的矩形(宽高比例视需求而定),这个时候你可能看不到任何效果,因为背景色是没有的,这个时候就需要使用另外一个特性了—solid,我们使用这个特性来为我们绘制的图形填充颜色,代码如下:

<!--填充-->
<solid android:color="@color/colorPrimary"></solid>

如此一来我们就实现了这样的效果:

image

我们看最终效果发现矩形的四个角都是圆形的,这就要使用shape的coeners了,我们设置代码如下:

<!--圆角-->
<corners android:radius="50dp"></corners>

以上代码是给我们要绘制的图形的四个角设置一个统一的圆角半径,我们看效果

image

好了,这里我们要实现的最终效果越来越近了,现在就是颜色的问题,我们使用solid只能给图形填充纯色背景,要想达到最终效果的渐变,我们需要使用到shape的gradient特性了,这个特性可以设定的属性值较多,我们先从简单的设置渐变色来开始学习。

<!--渐变-->
<gradient android:startColor="#6fd99d" android:centerColor="#63cfc3" android:endColor="#57c5e9"></gradient>

以上代码用到了gradient的三个属性值:startColor,centerColor,endColor。这三个属性很好理解,分别是渐变开始,中间和结束的颜色,我们来看效果。

image

为了让你更加清楚startColor,centerColor,endColor这三个属性的作用,我们采取开始颜色为黑色,中间颜色为红色以及结束颜色为蓝色的渐变。

<!--渐变-->
<gradient android:startColor="#0b0b0b" android:centerColor="#f10909" android:endColor="#067fe2"></gradient>

image

如此一来你对这个渐变三种夜色的过度应该很清楚了吧!接下来我们看看如何引用我们绘制好的shape吧!

<Button android:layout_width="match_parent" android:layout_height="50dp"
android:layout_marginTop="20dp"
android:layout_marginLeft="12dp"
android:layout_marginRight="12dp"
android:background="@drawable/tv_bg"
android:text="一个自学的程序员"
android:textColor="#efebeb"
/>

此处的tv_bg就是我们绘制的shape文件的名称。效果如下

image

shape六种特性的详解

现在我们来稍微小结一下,在实现上述效果的过程中,我们使用到了shape的size,solid,corners以及gradient特性,我们在之前也说过,shape一共有六个特性需要我们学习,这里只使用到了其中的四个,为了方便大家学习,我将六种特性的含义解释分类如下:

size—-定义形状的宽高

image

solid—-定义形状的背景填充色

image

corners—-设置形状四个角的圆角大小

image

注意:如果使用radius属性会统一设置四个角的圆角大小,单独设置会覆盖radius的效果。

gradient—-设置形状背景色的渐变(一共有九个属性)

在学习这个特性之前,我们要清楚这个渐变包含三种类型,一种是linear表示线性渐变,一种是放射性渐变还一种是扫描性渐变,由于这块比较复杂,我将gradient特性的各个渐变类型对应的属性值用导图整理了一下。

image

下面看各个渐变类型的效果图(我们画的依然是矩形)

线性渐变

image

放射性渐变

image

stroke—-给形状设置描边

image

padding—-设置以此形状为背景的内容距形状四周的内边距。

image

以上以画矩形为例借助图例的形式详细的将shape的六种特性的使用演示了一遍,现在我们来做个总结:

首先shape可以画三种类型的图形,分别是rectangle–矩形,oval–椭圆(一般用来画圆),ring–环形.

无论是画哪一种形状,都是要使用到我们上诉说的六种特性,分别是size,solid,corners,gradient,stroke,padding。只不过画某一种形状可能只用到六种特性中的部分特性,只要我们掌握了这六种特性,无论画哪一种形状应该都是毫无压力的。

椭圆

以上我们都是以画矩形为例详细介绍了六种特性的使用方式,对于椭圆和环形都是大同小异,非要说不同的话,环形有一些特殊的属性值需要我们学习,接下来,我们以画椭圆为例,介绍如何使用shape。

首先更改shape的根元素将shape的值改为oval。

android:shape="oval"

我们之前也说了,使用shape画oval我们一般就是用来画正圆比较多,所以我们可以将size特性的属性值宽高比例设为1:1即可。

<!--大小-->
<size android:width="1dp" android:height="1dp"></size>

我们看下效果

image

在绘制圆形的时候渐变用到的是比较多的,这里我们就主要介绍下在画圆中关于渐变的使用。

首先我们看第一种渐变类型–线性渐变且倾斜角度为0

我们首先看下效果。

image

代码如下

<!--渐变-->
<gradient android:startColor="#0b0b0b" android:centerColor="#f10918" android:endColor="#067fe2"
android:angle="0" android:type="linear"
></gradient>

这里如果你不指定渐变的类型,则其默认的渐变方式就是线性渐变且倾斜角度为0.

我们再来看下倾斜角度为90的线性渐变

image

你们要与倾斜角度为0的进行对比以属性angle属性的使用,下面我们再来看下另外一种渐变类型,那就是放射性渐变。

image

这里需要注意一点,有的人可能得不到这样的效果,为什么呢?我们先来看下我的代码

<!--渐变-->
<gradient android:startColor="#0b0b0b" android:centerColor="#f10918" android:endColor="#067fe2"
android:angle="0" android:type="radial" android:gradientRadius="0.5dp" android:centerX="0.5"
android:centerY="0.5"
></gradient>

知道你们注意到没我这里的gradientRadius设置的是0.5dp,你们一不小心设置成大于1的话就会出现这样的效果。

image

这是为什么呢?其实我这里为了说明一个问题给大家挖了一个坑,我们在之前设置圆形的size的时候还记得是怎么设置的吗?我当时说只要宽高比例为1:1就会得到一个正圆,于是我是这样写的代码。

<!--大小-->
<size android:width="1dp" android:height="1dp"></size>

看到没,我将宽和高都设置成了1dp,这样导致什么结果呢?我们要知道gradientRadius是用来设置放射的半径的,如果我们设置的参数大于1dp(要注意我们的正圆宽高才1dp)就会超过我们的正圆大小,这样就得不到我们想要的放射性渐变的效果了,所以这点要注意gradientRadius的属性值要根据设置的size的大小来设定。

我们子啊之前介绍画矩形的时候说到过,渐变的类型是有三种的,除了以上说的还有一种扫描型渐变,为什么在画矩形的时候没有介绍呢?因为我觉得在画圆形的时候看扫描型渐变才会更加直观,我们来做个测试,在矩形中的扫描型渐变是这个样子的。

image

第一次看可能你对什么是扫描型渐变可能还不太理解,下面我们看在圆形中的扫描型渐变。

image

怎么样,是不是有点明白了,其实扫描型渐变就是以图形中心为点从右侧水平线开始逆时针旋转依次历经开始的颜色,中间的颜色和结束的颜色。代码如下。

<!--渐变-->
<gradient android:startColor="#0b0b0b" android:centerColor="#f10918" android:endColor="#067fe2"
android:type="sweep" android:centerX="0.5"
android:centerY="0.5"
></gradient>

好了,以上介绍了画椭圆的一些使用方法,这里你可能会发现,我并没有想之前介绍画矩形的时候那么详细把每一个特性的属性值什么意思怎么用都说了一遍,因为无论画什么这些属性表达的都是一个意思,关键在于适不适合在当前你绘制的图形中使用。

line–线

我们说了shape可以绘制四种形状,我们以上讲了矩形和椭圆,接下来讲解下一个line–线型。

画线的时候主要用到了size和stroke,也就是靠这两个特性才会显示线,而且要注意使用size的时候只能画高,不要画宽。

我们看下代码。

<!--大小-->
<size android:height="20dp"></size>

<!--描边-->
<stroke android:color="#ef0b0b" android:width="1dp" android:dashGap="20dp" android:dashWidth="32dp"

></stroke>

以上代码实现的效果如下。

image

ring–环

画线的时候没有像画矩形或者圆的时候那么麻烦,用到的属性也没那么多,相对而言还是比较简单的,下面让我们继续,继续学习最后一个形状ring–环型。

为什么要最后一个来将ring呢?因为它还是有点特殊的,我们在之前写代码的时候对于shape的根元素我们只写了这么一个属性。

<shape xmlns:android="http://schemas.android.com/apk/res/android";
android:shape="ring"
>

就是指定我们要画的图形的类型的,但是对于shape的根元素还是可以设置一些特殊元素的,而这些元素只适用于ring,下面一起来看下吧!

* android:innerRadius 设置内环的半径
* android:innerRadiusRatio 以环的宽度比率来表示内环的半径,这个值默认为3,表示内环半径为环的宽度除以3,但是这个值会被android:innerRadius给覆盖
* android:thickness 环的厚度
* android:thicknessRatio 以环的宽度比率来表示环的厚度,这个值默认为9,表示环的厚度为环的宽度除以9,同样这个值会被android:thickness覆盖
* android:useLevel 一般为false,否则可能环形无法显示,只有作为LevelListDrawable使用时才设为true

我们先来看下效果图吧。

image

代码如下。

<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android";
android:shape="ring"
android:innerRadiusRatio="3"
android:thicknessRatio="9"
android:useLevel="false"
>

<!--一共有六个属性-->

<!--大小-->
<size android:height="10dp"></size>

<!--渐变-->
<gradient android:startColor="#0b0b0b" android:centerColor="#f10918" android:endColor="#067fe2"
android:type="sweep" android:centerX="0.5"
android:centerY="0.5"
></gradient>

<!--描边-->
<stroke android:color="#151515" android:width="1dp" android:dashGap="20dp" android:dashWidth="32dp"

></stroke>

</shape>

大家在使用ring画圆环的时候可能对那几个特有的元素不是太清楚,这里重点说下这几个属性。

* android:innerRadius 设置内环的半径
* android:innerRadiusRatio 以环的宽度比率来表示内环的半径,这个值默认为3,表示内环半径为环的宽度除以3,但是这个值会被android:innerRadius给覆盖
* android:thickness 环的厚度
* android:thicknessRatio 以环的宽度比率来表示环的厚度,这个值默认为9,表示环的厚度为环的宽度除以9,同样这个值会被android:thickness覆盖
* android:useLevel 一般为false,否则可能环形无法显示,只有作为LevelListDrawable使用时才设为true

image

大家在使用ring的这些特殊属性的时候参照我给的这个图(虽然有点丑)可能有助于你理解,尤其是innerRadiusRatio和thicknessRatio这两个属性。

到此我们已经将Android中关于shape的使用全部讲解完成了,现在来个大总结吧!首先shape是用来进行xml绘制图形的,可以给一些控件设置背景来达到我们想要的效果,shape其实就是一个xml文件,一般放在我们项目中的drawable文件目录下,使用shape可以画四种图形分别是矩形(常用),椭圆(一般用来画正圆),直线和环形。无论是画哪一种图形都离不开shape的六种特性,只要掌握这六种特性的使用方法就可以掌握各种图形的绘制,区别就在于不同的特性适不适合当前所绘制的图形(这句话是重点,圈起来要考),关于ring-画环形这块它有一些特殊的属性,结合文中我给出的图掌握起来难度也不大,好了,再放出这非常重要的六种特性吧!

image.png

文章很短,路还漫长,大家好,我是玖玖君,一个帅气与才华并存的男人,我们下期再见

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

推荐阅读更多精彩内容