Android svg动画-svg动画基础

  Google在Android 5.x中增加了对svg矢量图形的支持,这对于创建新的高效率动画具有非常重大的意义。首先,我们来看看什么是svg。

1.什么是svg图形

A.可伸缩矢量图形(Scalable Vector Graphics)
B.定义用于网络的基于矢量的图形
C.使用XML格式定义图形
D.图像在放大或改变尺寸的情况下其图形质量不会有所损失
E.万维网联盟的标准
F.与诸如DOM和XML之类的W3C标准是一个整体

  在Android 5.x之前,我们使用svg必须通过第三方的库来实现;而在Android 5.x之后,Android中添加了对<path>标签的支持,从而能够使开发者使用svg来创建更加丰富的动画效果。

2.svg与Bitmap的区别

  Bitmap(位图)是通过在每个像素点上存储色彩信息来表达图像,而svg是一个绘图标准。与Bitmap相比,svg最大的优点就是放大不会失真。而且Bitmap需要为不同分辨率设计多套图标,而矢量图则不需要。

3.<path>标签

  使用<path>标签来创建svg,就像用指令的方式来控制一只画笔,例如移动画笔到某一坐标位置,画一条直线,画一条曲线,结束。<pah>标签所支持的指令有以下几种:

A. M = moveto(x, y):将画笔移动到指定的坐标位置上,但是并没有发生绘制。
B. L = lineto(x, y):画一条直线到指定的坐标位置上。
C. H = horizontal lineto(x):画水平线在指定的x坐标位置。
D. V = vertical lineto(y):画一条垂直线在指定的y坐标位置。
E. C = curveto(x1, y1, x2, y2, endX, endY):三次贝赛曲线
F. S = smooth curveto(x2, y2, endX, endY):三次贝赛曲线
G. G = quadratic Belzier curveto(x, y, endX, endY):二次贝赛曲线
H. T = smooth quadratic Belizer curveto(endX, endY):映射前面路径后的终点
I. A = elliptical Arc(rx, ry, xRotation, flag1, flag2, x, y):弧线
J. Z = colsepath():关闭路径

  在使用上面的指令的时候,我们还需要注意几点:

A. 坐标轴是以(0,0)为中心的, x轴水平向右,y轴水平向下。
B. 所有指令大小写均可。大写表示绝对坐标,参照全局坐标系;小写表示相对坐标,参照父容器的坐标系。
C. 指令和数据间的空格可以省略。
D. 同一个指令可以使用多次。

4.svg常用指令

(1).L指令

绘制直线的指令时“L”,代表从当前点绘制直线到指定点。“L”之后的参数是一个点坐标,例如:“L 200 400”绘制直线。同时还可以使用“H”和“V”指令来绘制水平、竖直线,后面的参数是X坐标(H指令) 或者y坐标(V指令)。

(2).M指令

M指令类似于Android绘图中Path类中的moveto方法,即代表将画笔移动到某一点,但并不发生绘制的动作

(3).A指令

A指令是用来绘制一段弧线的,且允许弧线不闭合。可以把A指令绘制的弧线想象成是椭圆的某一段,A指令一下有七个参数:

  A. rx, ry 表示的是椭圆的半轴大小
  B.xRotation 指椭圆的X轴与水平方向顺时针方向的夹角,可以想象成一个水平的椭圆绕中心点顺时针旋转xRotation的角度。
  C.flag1 只有两个值,1表示绘制大弧度弧线,0表示绘制小弧度弧线
  D.flag2 只有两个值,确定从起点到终点的方向,1为顺时针,0为逆时针。

5.Android中使用svg

  Google在Android 5.x中提供了下面两个新的API来帮助支持svg:

    (1).VectorDrawable
    (2).AnimatedVectorDrawable

  其中,VectorDrawable让你可以创建基于XML的svg图形,并且结合AnimatedVectorDrawable来实现动画效果。

6.VectorDrawable的相关属性和作用

  在Android中,我们可以使用XMl文件来创建SVG图形,在XML文件中,必须将其根元素设置为<vector>才行。
  下面介绍一下在XML文件中的相关属性的作用
  <vector>:定义一个vector类型的可绘制的对象。

属性名 解释
Android:name 定义该对象的名字
Android:width 定义该对象的宽,该属性支持所有的像素单位,通常使用dp
Android:height 定义该对象的高,同上
Android:viewportWidth 视口的宽,视口通常是设备上用于绘制图形的画布
Android:tint 该对象的着色色彩,默认是没有色彩的
Android:tintMode 该对象的着色模式,默认为src_in
android:autoMirrored 当该对象的布局是RTL(right-to-left),是否需要一个镜像。默认是false
android:alpha 该对象的透明度。默认为1,表示不透明

  <group>:定义一个路径的组或者是一个子组,包含图形的转换信息。这些转换信息定义在与视口相同的坐标系中,同时转换信息支持缩放、旋转以及移动。

属性名 解释
Android:name 定义group的名字
Android:rotation 定义group的旋转角度。默认为0
Android:pivotX 定义group旋转和缩放的中心点X坐标,该坐标在视口(viewport)的坐标系中。默认为0
Android:pivotY 同上
Android:scaleX X坐标上的缩放度。默认为1
Android:scaleY 同上
Android:translateX X坐标上的移动,该坐标在视口的坐标系中。默认为0
Android:translateY 同上

  <path>:定义绘制的路径

属性名 解释
Android:name 定义路径的名字
Android:pathData 使用svg属性来绘制路径。路径定在视口(viewport)里面
Android:fillColor 定义填充路径的颜色。可以是颜色,或者在API 23+中,颜色状态列表或者渐变颜色
Android:strokeColor 定义填充路径轮廓的颜色。其余的同上
Android:strokeWidth 定义路径线条的宽度。默认为0
Android:strokeAlpha 定义路径线条的透明度。默认为1
Android:fillAlpha 定义填充路径的透明度。默认为1
Android:trimPathStart 从路径开始的地方,截取路径一部分,范围是0到1。默认为0
Android:trimPathEnd 从路径结束的地方,截取路径的一部分,范围是0到1。默认为1
Android:trimPathOffset 移动修剪的区域(显示区域包括开始和结束),范围是0到1。默认为0
Android:strokeLineCap 定义线的两端的样式,有 butt, round, square三种模式。默认为butt
Android:strokeLineJin 定义线段连接处的样式,有miter,round,bevel三种模式。默认为miter
Android:strokeMiterLimit 设置斜角的上限。默认为4
Android:fillType 定义填充模式,有evenOdd、nonZero两种模式。默认为nonZero

  <clip-path>:给一个绘制的路径定义一个特定的截取路线,也就是说在已经存在的路线上定义它的截取路线,不绘制截取的部分。注意:截取路径只能定义在特定的group中或者是group的子组中。

属性名 解释
Android:name 定义截取路径的名字
Android:pathData 同path中的pathData

7.在Android使用相关的属性来绘制图形

  Android中使用svg绘制的图形也是Drawable对象,不过是VectorDrawable对象。因此,我们可以在Drawable文件夹中定义相关的XML文件来绘制我们的图形,跟平常的Drawable对象绘制方式是一样的,只是相关的属性是不同的。

<?xml version="1.0" encoding="utf-8"?>
<vector xmlns:android="http://schemas.android.com/apk/res/android"
    android:height="200dp"
    android:width="200dp"
    android:viewportWidth="110"
    android:viewportHeight="110"

    >
    <group
        android:name="test"
        android:pivotX="55"
        android:pivotY="55"
        >
        <path android:strokeColor="@android:color/holo_blue_bright"
            android:strokeWidth="2"
            android:pathData="M 26,50
                              a 25,25, 0,0,0,25,25"
            />
    </group>
</vector>

  如上面的例子,我们就绘制了一条弧线。
  经过上面的定义,就绘制了一个svg图形。关于它的引用,就像使用平常的Drawable一样。

<ImageView

        android:id="@+id/id_imageView"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:src="@drawable/demo"
        />

8.给svg图形增加动画

  通过上面的步骤,我们定义了一个svg图形,将其设置给一个ImageView。但是有没有发现,这并没有什么特点啊?好像跟我们平常使用的Drawable是一样的。其实并不是这样的,我们还可以将一个动态的svg图形设置给一个控件,也就是要说的,给一个svg图形增加动画,也就是之前说的AnimatedVectorDrawable。
  AnimatedVectorDrawable的作用就是给一个VectorDrawable增加动画效果。Google将AnimatedVectorDrawable比喻成胶水,通过AnimatedVectorDrawable来连接静态的VectorDrawable和动态的ObjectDrawable。


(1). 静态的VectorDrawable:

<?xml version="1.0" encoding="utf-8"?>
<vector xmlns:android="http://schemas.android.com/apk/res/android"
    android:height="200dp"
    android:width="200dp"
    android:viewportWidth="110"
    android:viewportHeight="110"

    >
    <group
        android:name="test"
        android:pivotX="55"
        android:pivotY="55"
        >
        <path android:strokeColor="@android:color/holo_blue_bright"
            android:strokeWidth="2"
            android:pathData="M 26,50
                              a 25,25, 0,0,0,25,25"
            />
    </group>
</vector>

(2). 动态的ObjectorAnimator:

<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android">
    <objectAnimator
        android:duration="500"
        android:propertyName="rotation"
        android:valueFrom="0"
        android:valueTo="360"
        android:repeatCount="100"
        android:startOffset="-1"
        android:interpolator="@android:anim/overshoot_interpolator"
        />
</set>

(3). 粘合起来的AnimatedVectorDrawable:

<?xml version="1.0" encoding="utf-8"?>
<animated-vector xmlns:android="http://schemas.android.com/apk/res/android"
    android:drawable="@drawable/demo"
    >
    <target
        android:animation="@animator/anim"
        android:name="test"/>
</animated-vector>

  需要的注意的是:AnimatedVectorDrawable中指定的target的name属性的内容必须与VectorDrawable中需要作用的name属性必须保持一致,这样系统才能找到要实现动画效果的元素。最后,通过AnimatedVectorDrawable中target的animation属性,将一个动画作用到对应name的元素上。
  在<group>标签和<path>标签中,添加了rotation、fillColor、pathData等属性,那么在ObjectAnimator中,就可以通过指定Android:propertyName="XXXX"属性来选择控制哪一种属性,通过Android:valueFrom="XXXX"和Android:valueTo="XXXX"属性,可以控制动画的起始值。唯一需要注意的是:如果指定控制的属性为pathData,那么需要添加一个属性--Android:valueType="pathType"来告诉系统进行pathData变换。
  在控件引用Drawable的时候,记住引用的是AnimatedVectorDrawable,而不是VectorDrawable。

<?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"
    android:gravity="center"
    android:orientation="vertical">
    <ImageView

        android:id="@+id/id_imageView"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:src="@drawable/demo1"
        />
</LinearLayout>

  当我们引用一个动态的AnimatedVectorDrawable后,就可以在Java代码中使用动画效果了。

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

推荐阅读更多精彩内容