使用的是一个第三方库android-pathview主要是一个自己定义View——PathView。跟全部自己定义View一样,重写了三个构造方法。

而且终于调用三个參数的构造方法,在里面获取自己定义属性。

  1. /**
  2. * Default constructor.
  3. *
  4. * @param context The Context of the application.
  5. */
  6. public PathView(Context context) {
  7. this(context, null);
  8. }
  9. /**
  10. * Default constructor.
  11. *
  12. * @param context The Context of the application.
  13. * @param attrs attributes provided from the resources.
  14. */
  15. public PathView(Context context, AttributeSet attrs) {
  16. this(context, attrs, 0);
  17. }
  18. /**
  19. * Default constructor.
  20. *
  21. * @param context The Context of the application.
  22. * @param attrs attributes provided from the resources.
  23. * @param defStyle Default style.
  24. */
  25. public PathView(Context context, AttributeSet attrs, int defStyle) {
  26. super(context, attrs, defStyle);
  27. paint.setStyle(Paint.Style.STROKE);
  28. getFromAttributes(context, attrs);
  29. }

查看自己定义属性的xml,里面共三个自己定义属性,pathColor为路径颜色,默认值为绿色0xff00ff00,pathWidth为路径宽度,默认值为8.0f,svg为路径文件

  1.  
  2. " data-snippet-id="ext.8372b925ab36c18fbcb3da24aa7ee54f" data-snippet-saved="false" data-codota-status="done"><?xml version="1.0" encoding="utf-8"?>
  3. <resources>
  4. <declare-styleable name="PathView">
  5. <attr name="pathColor" format="color"/>
  6. <attr name="pathWidth" format="float"/>
  7. <attr name="svg" format="reference"/>
  8. </declare-styleable>
  9. </resources>

通过getFromAttributes函数获得这些自定属性,并赋值给成员变量,最后通过finally块回收资源

  1. /**
  2. * Get all the fields from the attributes .
  3. *
  4. * @param context The Context of the application.
  5. * @param attrs attributes provided from the resources.
  6. */
  7. private void getFromAttributes(Context context, AttributeSet attrs) {
  8. final TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.PathView);
  9. try {
  10. if (a != null) {
  11. paint.setColor(a.getColor(R.styleable.PathView_pathColor, 0xff00ff00));
  12. paint.setStrokeWidth(a.getFloat(R.styleable.PathView_pathWidth, 8.0f));
  13. svgResourceId = a.getResourceId(R.styleable.PathView_svg, 0);
  14. }
  15. } finally {
  16. if (a != null) {
  17. a.recycle();
  18. }
  19. }
  20. }

至于成员变量,这里贴出全部成员变量,详细作用见凝视

  1. paths = new ArrayList(0);
  2. /**
  3. * This is a lock before the view is redrawn
  4. * or resided it must be synchronized with this object.
  5. */
  6. private final Object mSvgLock = new Object();
  7. /**
  8. * Thread for working with the object above.
  9. */
  10. private Thread mLoader;
  11.  
  12. /**
  13. * The svg image from the raw directory.
  14. */
  15. private int svgResourceId;
  16. /**
  17. * Object that build the animation for the path.
  18. */
  19. private AnimatorBuilder animatorBuilder;
  20. /**
  21. * The progress of the drawing.
  22. */
  23. private float progress = 0f;
  24.  
  25. /**
  26. * If the used colors are from the svg or from the set color.
  27. */
  28. private boolean naturalColors;
  29. /**
  30. * If the view is filled with its natural colors after path drawing.
  31. */
  32. private boolean fillAfter;
  33. /**
  34. * The width of the view.
  35. */
  36. private int width;
  37. /**
  38. * The height of the view.
  39. */
  40. private int height;" data-snippet-id="ext.6985d4eda27e815a561fa0bc34510cee" data-snippet-saved="false" data-csrftoken="us74Km0s-ioacrN5aDh_lQUMQoPh1qeiAfqM" data-codota-status="done">/**
  41. * Logging tag.
  42. */
  43. public static final String LOG_TAG = "PathView";
  44. /**
  45. * The paint for the path.
  46. */
  47. private Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
  48. /**
  49. * Utils to catch the paths from the svg.
  50. */
  51. private final SvgUtils svgUtils = new SvgUtils(paint);
  52. /**
  53. * All the paths provided to the view. Both from Path and Svg.
  54. */
  55. private List<SvgUtils.SvgPath> paths = new ArrayList<SvgUtils.SvgPath>(0);
  56. /**
  57. * This is a lock before the view is redrawn
  58. * or resided it must be synchronized with this object.
  59. */
  60. private final Object mSvgLock = new Object();
  61. /**
  62. * Thread for working with the object above.
  63. */
  64. private Thread mLoader;
  65. /**
  66. * The svg image from the raw directory.
  67. */
  68. private int svgResourceId;
  69. /**
  70. * Object that build the animation for the path.
  71. */
  72. private AnimatorBuilder animatorBuilder;
  73. /**
  74. * The progress of the drawing.
  75. */
  76. private float progress = 0f;
  77. /**
  78. * If the used colors are from the svg or from the set color.
  79. */
  80. private boolean naturalColors;
  81. /**
  82. * If the view is filled with its natural colors after path drawing.
  83. */
  84. private boolean fillAfter;
  85. /**
  86. * The width of the view.
  87. */
  88. private int width;
  89. /**
  90. * The height of the view.
  91. */
  92. private int height;

然后是一些getter和setter方法。

  1. paths) {
  2. for (Path path : paths) {
  3. this.paths.add(new SvgUtils.SvgPath(path, paint));
  4. }
  5. synchronized (mSvgLock) {
  6. updatePathsPhaseLocked();
  7. }
  8. }
  9.  
  10. /**
  11. * Set path to be drawn and animated.
  12. *
  13. * @param path - Paths that can be drawn.
  14. */
  15. public void setPath(final Path path) {
  16. paths.add(new SvgUtils.SvgPath(path, paint));
  17. synchronized (mSvgLock) {
  18. updatePathsPhaseLocked();
  19. }
  20. }
  21. /**
  22. * If the real svg need to be drawn after the path animation.
  23. *
  24. * @param fillAfter - boolean if the view needs to be filled after path animation.
  25. */
  26. public void setFillAfter(final boolean fillAfter) {
  27. this.fillAfter = fillAfter;
  28. }
  29. /**
  30. * Animator for the paths of the view.
  31. *
  32. * @return The AnimatorBuilder to build the animation.
  33. */
  34. public AnimatorBuilder getPathAnimator() {
  35. if (animatorBuilder == null) {
  36. animatorBuilder = new AnimatorBuilder(this);
  37. }
  38. return animatorBuilder;
  39. }
  40. /**
  41. * Get the path color.
  42. *
  43. * @return The color of the paint.
  44. */
  45. public int getPathColor() {
  46. return paint.getColor();
  47. }
  48.  
  49. /**
  50. * Set the path color.
  51. *
  52. * @param color -The color to set to the paint.
  53. */
  54. public void setPathColor(final int color) {
  55. paint.setColor(color);
  56. }
  57.  
  58. /**
  59. * Get the path width.
  60. *
  61. * @return The width of the paint.
  62. */
  63. public float getPathWidth() {
  64. return paint.getStrokeWidth();
  65. }
  66.  
  67. /**
  68. * Set the path width.
  69. *
  70. * @param width - The width of the path.
  71. */
  72. public void setPathWidth(final float width) {
  73. paint.setStrokeWidth(width);
  74. }
  75.  
  76. /**
  77. * Get the svg resource id.
  78. *
  79. * @return The svg raw resource id.
  80. */
  81. public int getSvgResource() {
  82. return svgResourceId;
  83. }
  84.  
  85. /**
  86. * Set the svg resource id.
  87. *
  88. * @param svgResource - The resource id of the raw svg.
  89. */
  90. public void setSvgResource(int svgResource) {
  91. svgResourceId = svgResource;
  92. }
  93. /**
  94. * If the real svg need to be drawn after the path animation.
  95. *
  96. * @param fillAfter - boolean if the view needs to be filled after path animation.
  97. */
  98. public void setFillAfter(final boolean fillAfter) {
  99. this.fillAfter = fillAfter;
  100. }
  101. /**
  102. * Animate this property. It is the percentage of the path that is drawn.
  103. * It must be [0,1].
  104. *
  105. * @param percentage float the percentage of the path.
  106. */
  107. public void setPercentage(float percentage) {
  108. if (percentage 1.0f) {
  109. throw new IllegalArgumentException("setPercentage not between 0.0f and 1.0f");
  110. }
  111. progress = percentage;
  112. synchronized (mSvgLock) {
  113. updatePathsPhaseLocked();
  114. }
  115. invalidate();
  116. }" data-snippet-id="ext.1be3027e509316e9e2378a76aa3aa94e" data-snippet-saved="false" data-csrftoken="SeP2anif-STH2gPOCQWUpbGDdCh2FrxssGrM" data-codota-status="done"> /**
  117. * Set paths to be drawn and animated.
  118. *
  119. * @param paths - Paths that can be drawn.
  120. */
  121. public void setPaths(final List<Path> paths) {
  122. for (Path path : paths) {
  123. this.paths.add(new SvgUtils.SvgPath(path, paint));
  124. }
  125. synchronized (mSvgLock) {
  126. updatePathsPhaseLocked();
  127. }
  128. }
  129. /**
  130. * Set path to be drawn and animated.
  131. *
  132. * @param path - Paths that can be drawn.
  133. */
  134. public void setPath(final Path path) {
  135. paths.add(new SvgUtils.SvgPath(path, paint));
  136. synchronized (mSvgLock) {
  137. updatePathsPhaseLocked();
  138. }
  139. }
  140. /**
  141. * If the real svg need to be drawn after the path animation.
  142. *
  143. * @param fillAfter - boolean if the view needs to be filled after path animation.
  144. */
  145. public void setFillAfter(final boolean fillAfter) {
  146. this.fillAfter = fillAfter;
  147. }
  148. /**
  149. * Animator for the paths of the view.
  150. *
  151. * @return The AnimatorBuilder to build the animation.
  152. */
  153. public AnimatorBuilder getPathAnimator() {
  154. if (animatorBuilder == null) {
  155. animatorBuilder = new AnimatorBuilder(this);
  156. }
  157. return animatorBuilder;
  158. }
  159. /**
  160. * Get the path color.
  161. *
  162. * @return The color of the paint.
  163. */
  164. public int getPathColor() {
  165. return paint.getColor();
  166. }
  167. /**
  168. * Set the path color.
  169. *
  170. * @param color -The color to set to the paint.
  171. */
  172. public void setPathColor(final int color) {
  173. paint.setColor(color);
  174. }
  175. /**
  176. * Get the path width.
  177. *
  178. * @return The width of the paint.
  179. */
  180. public float getPathWidth() {
  181. return paint.getStrokeWidth();
  182. }
  183. /**
  184. * Set the path width.
  185. *
  186. * @param width - The width of the path.
  187. */
  188. public void setPathWidth(final float width) {
  189. paint.setStrokeWidth(width);
  190. }
  191. /**
  192. * Get the svg resource id.
  193. *
  194. * @return The svg raw resource id.
  195. */
  196. public int getSvgResource() {
  197. return svgResourceId;
  198. }
  199. /**
  200. * Set the svg resource id.
  201. *
  202. * @param svgResource - The resource id of the raw svg.
  203. */
  204. public void setSvgResource(int svgResource) {
  205. svgResourceId = svgResource;
  206. }
  207. /**
  208. * If the real svg need to be drawn after the path animation.
  209. *
  210. * @param fillAfter - boolean if the view needs to be filled after path animation.
  211. */
  212. public void setFillAfter(final boolean fillAfter) {
  213. this.fillAfter = fillAfter;
  214. }
  215. /**
  216. * Animate this property. It is the percentage of the path that is drawn.
  217. * It must be [0,1].
  218. *
  219. * @param percentage float the percentage of the path.
  220. */
  221. public void setPercentage(float percentage) {
  222. if (percentage < 0.0f || percentage > 1.0f) {
  223. throw new IllegalArgumentException("setPercentage not between 0.0f and 1.0f");
  224. }
  225. progress = percentage;
  226. synchronized (mSvgLock) {
  227. updatePathsPhaseLocked();
  228. }
  229. invalidate();
  230. }

然后通过一个函数进行路径的更新

  1. /**
  2. * This refreshes the paths before draw and resize.
  3. */
  4. private void updatePathsPhaseLocked() {
  5. final int count = paths.size();
  6. for (int i = 0; i < count; i++) {
  7. SvgUtils.SvgPath svgPath = paths.get(i);
  8. svgPath.path.reset();
  9. svgPath.measure.getSegment(0.0f, svgPath.length * progress, svgPath.path, true);
  10. // Required only for Android 4.4 and earlier
  11. svgPath.path.rLineTo(0.0f, 0.0f);
  12. }
  13. }

接着就一大堆measure啊,draw啊

  1. @Override
  2. protected void onSizeChanged(final int w, final int h, int oldw, int oldh) {
  3. super.onSizeChanged(w, h, oldw, oldh);
  4. if (mLoader != null) {
  5. try {
  6. mLoader.join();
  7. } catch (InterruptedException e) {
  8. Log.e(LOG_TAG, "Unexpected error", e);
  9. }
  10. }
  11. if (svgResourceId != 0) {
  12. mLoader = new Thread(new Runnable() {
  13. @Override
  14. public void run() {
  15. svgUtils.load(getContext(), svgResourceId);
  16. synchronized (mSvgLock) {
  17. width = w - getPaddingLeft() - getPaddingRight();
  18. height = h - getPaddingTop() - getPaddingBottom();
  19. paths = svgUtils.getPathsForViewport(width, height);
  20. updatePathsPhaseLocked();
  21. }
  22. }
  23. }, "SVG Loader");
  24. mLoader.start();
  25. }
  26. }
  27. @Override
  28. protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
  29. super.onMeasure(widthMeasureSpec, heightMeasureSpec);
  30. if (svgResourceId != 0) {
  31. int widthSize = MeasureSpec.getSize(widthMeasureSpec);
  32. int heightSize = MeasureSpec.getSize(heightMeasureSpec);
  33. setMeasuredDimension(widthSize, heightSize);
  34. return;
  35. }
  36. int desiredWidth = 0;
  37. int desiredHeight = 0;
  38. final float strokeWidth = paint.getStrokeWidth() / 2;
  39. for (SvgUtils.SvgPath path : paths) {
  40. desiredWidth += path.bounds.left + path.bounds.width() + strokeWidth;
  41. desiredHeight += path.bounds.top + path.bounds.height() + strokeWidth;
  42. }
  43. int widthSize = MeasureSpec.getSize(widthMeasureSpec);
  44. int heightSize = MeasureSpec.getSize(heightMeasureSpec);
  45. int widthMode = MeasureSpec.getMode(widthMeasureSpec);
  46. int heightMode = MeasureSpec.getMode(widthMeasureSpec);
  47. int measuredWidth, measuredHeight;
  48. if (widthMode == MeasureSpec.AT_MOST) {
  49. measuredWidth = desiredWidth;
  50. } else {
  51. measuredWidth = widthSize;
  52. }
  53. if (heightMode == MeasureSpec.AT_MOST) {
  54. measuredHeight = desiredHeight;
  55. } else {
  56. measuredHeight = heightSize;
  57. }
  58. setMeasuredDimension(measuredWidth, measuredHeight);
  59. }
  60. @Override
  61. protected void onDraw(Canvas canvas) {
  62. super.onDraw(canvas);
  63. synchronized (mSvgLock) {
  64. canvas.save();
  65. canvas.translate(getPaddingLeft(), getPaddingTop());
  66. final int count = paths.size();
  67. for (int i = 0; i < count; i++) {
  68. final SvgUtils.SvgPath svgPath = paths.get(i);
  69. final Path path = svgPath.path;
  70. final Paint paint1 = naturalColors ? svgPath.paint : paint;
  71. canvas.drawPath(path, paint1);
  72. }
  73. fillAfter(canvas);
  74. canvas.restore();
  75. }
  76. }
  77. /**
  78. * If there is svg , the user called setFillAfter(true) and the progress is finished.
  79. *
  80. * @param canvas Draw to this canvas.
  81. */
  82. private void fillAfter(final Canvas canvas) {
  83. if (svgResourceId != 0 && fillAfter && progress == 1f) {
  84. svgUtils.drawSvgAfter(canvas, width, height);
  85. }
  86. }

我们看到他重写了onSizeChanged方法。首先推断线程是否是null,假设不为null,则调用join方法,让该thread运行完run方法里面的东西后。再运行join()方法后面的代码。

后面的代码就是起了一个线程获取宽度,高度。路径并更新路径等信息。

onMeasure()方法里就是一些測量内容了。详细測量逻辑见源代码。

onDraw()方法主要完毕绘制,事实上非常easy,就是在Canvas上绘图。遍历全部Path,画在画布上,画完后调用了fillAfter方法,假设设置了svg资源,以及fillAfter 成员变量为true。动画已完毕,则会保持动画

最后这个自己定义View类里使用了创建者模式进行创建路径动画,比方动画的时间,插值器。延时,监听器等等。

  1. /**
  2. * Object for building the animation of the path of this view.
  3. */
  4. public static class AnimatorBuilder {
  5. /**
  6. * Duration of the animation.
  7. */
  8. private int duration = 350;
  9. /**
  10. * Interpolator for the time of the animation.
  11. */
  12. private Interpolator interpolator;
  13. /**
  14. * The delay before the animation.
  15. */
  16. private int delay = 0;
  17. /**
  18. * ObjectAnimator that constructs the animation.
  19. */
  20. private final ObjectAnimator anim;
  21. /**
  22. * Listener called before the animation.
  23. */
  24. private ListenerStart listenerStart;
  25. /**
  26. * Listener after the animation.
  27. */
  28. private ListenerEnd animationEnd;
  29. /**
  30. * Animation listener.
  31. */
  32. private PathViewAnimatorListener pathViewAnimatorListener;
  33. /**
  34. * Default constructor.
  35. *
  36. * @param pathView The view that must be animated.
  37. */
  38. public AnimatorBuilder(final PathView pathView) {
  39. anim = ObjectAnimator.ofFloat(pathView, "percentage", 0.0f, 1.0f);
  40. }
  41. /**
  42. * Set the duration of the animation.
  43. *
  44. * @param duration - The duration of the animation.
  45. * @return AnimatorBuilder.
  46. */
  47. public AnimatorBuilder duration(final int duration) {
  48. this.duration = duration;
  49. return this;
  50. }
  51. /**
  52. * Set the Interpolator.
  53. *
  54. * @param interpolator - Interpolator.
  55. * @return AnimatorBuilder.
  56. */
  57. public AnimatorBuilder interpolator(final Interpolator interpolator) {
  58. this.interpolator = interpolator;
  59. return this;
  60. }
  61. /**
  62. * The delay before the animation.
  63. *
  64. * @param delay - int the delay
  65. * @return AnimatorBuilder.
  66. */
  67. public AnimatorBuilder delay(final int delay) {
  68. this.delay = delay;
  69. return this;
  70. }
  71. /**
  72. * Set a listener before the start of the animation.
  73. *
  74. * @param listenerStart an interface called before the animation
  75. * @return AnimatorBuilder.
  76. */
  77. public AnimatorBuilder listenerStart(final ListenerStart listenerStart) {
  78. this.listenerStart = listenerStart;
  79. if (pathViewAnimatorListener == null) {
  80. pathViewAnimatorListener = new PathViewAnimatorListener();
  81. anim.addListener(pathViewAnimatorListener);
  82. }
  83. return this;
  84. }
  85. /**
  86. * Set a listener after of the animation.
  87. *
  88. * @param animationEnd an interface called after the animation
  89. * @return AnimatorBuilder.
  90. */
  91. public AnimatorBuilder listenerEnd(final ListenerEnd animationEnd) {
  92. this.animationEnd = animationEnd;
  93. if (pathViewAnimatorListener == null) {
  94. pathViewAnimatorListener = new PathViewAnimatorListener();
  95. anim.addListener(pathViewAnimatorListener);
  96. }
  97. return this;
  98. }
  99. /**
  100. * Starts the animation.
  101. */
  102. public void start() {
  103. anim.setDuration(duration);
  104. anim.setInterpolator(interpolator);
  105. anim.setStartDelay(delay);
  106. anim.start();
  107. }
  108. /**
  109. * Animation listener to be able to provide callbacks for the caller.
  110. */
  111. private class PathViewAnimatorListener implements Animator.AnimatorListener {
  112. @Override
  113. public void onAnimationStart(Animator animation) {
  114. if (listenerStart != null) listenerStart.onAnimationStart();
  115. }
  116. @Override
  117. public void onAnimationEnd(Animator animation) {
  118. if (animationEnd != null) animationEnd.onAnimationEnd();
  119. }
  120. @Override
  121. public void onAnimationCancel(Animator animation) {
  122. }
  123. @Override
  124. public void onAnimationRepeat(Animator animation) {
  125. }
  126. }
  127. /**
  128. * Called when the animation start.
  129. */
  130. public interface ListenerStart {
  131. /**
  132. * Called when the path animation start.
  133. */
  134. void onAnimationStart();
  135. }
  136. /**
  137. * Called when the animation end.
  138. */
  139. public interface ListenerEnd {
  140. /**
  141. * Called when the path animation end.
  142. */
  143. void onAnimationEnd();
  144. }
  145. }

SVG文件的载入等工作都在SvgUtils类中完毕。

依赖了一个jar包,该工具类主要负责资源文件的载入和初始化

构造方法是传入一个画笔

  1. /**
  2. * Init the SVGUtils with a paint for coloring.
  3. *
  4. * @param sourcePaint - the paint for the coloring.
  5. */
  6. public SvgUtils(final Paint sourcePaint) {
  7. mSourcePaint = sourcePaint;
  8. }

资源文件的载入

  1. /**
  2. * Loading the svg from the resources.
  3. *
  4. * @param context Context object to get the resources.
  5. * @param svgResource int resource id of the svg.
  6. */
  7. public void load(Context context, int svgResource) {
  8. if (mSvg != null) return;
  9. try {
  10. mSvg = SVG.getFromResource(context, svgResource);
  11. mSvg.setDocumentPreserveAspectRatio(PreserveAspectRatio.UNSCALED);
  12. } catch (SVGParseException e) {
  13. Log.e(LOG_TAG, "Could not load specified SVG resource", e);
  14. }
  15. }

其余的方法就是跟绘制有关了。

有兴趣自己去下源代码看看。这里不做展开了。

源代码解析完毕后,我们使用一下这个库。我们首先使用github上的svg文件做測试。

在布局文件中加入代码

" data-snippet-id="ext.1e55dba08b347498987ce9f5e8814add" data-snippet-saved="false" data-csrftoken="vzOQfhuW-PjhzjgF9jblnzZGyN0JgbLTGUgE" data-codota-status="done">    <com.eftimoff.androipathview.PathView
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/pathView"
android:layout_width="150dp"
android:layout_height="150dp"
app:svg="@raw/monitor"
app:pathWidth="2"
app:pathColor="#ff0000"
/>

然后就能够通过getPathAnimator方法获得创建者对象,通过一系列的链式调用配置动画,比方延时,动画持续时间。插值器,监听器,最后调用start開始动画。


  1. PathView mPathView= (PathView) findViewById(R.id.pathView);
  2. mPathView.getPathAnimator()
  3. .delay(1000)
  4. .duration(3000)
  5. .interpolator(new BounceInterpolator())
  6. .listenerStart(new PathView.AnimatorBuilder.ListenerStart() {
  7. @Override
  8. public void onAnimationStart() {
  9. Log.e("TAG", "start");
  10. }
  11. })
  12. .listenerEnd(new PathView.AnimatorBuilder.ListenerEnd() {
  13. @Override
  14. public void onAnimationEnd() {
  15. Log.e("TAG","end");
  16. }
  17. }).start();

终于效果就是这样子

当然我们也能够使用svg文件中的颜色,让它绘制完毕后绘制终于真正的svg,能够加入下面代码

  1. mPathView.useNaturalColors();
  2. mPathView.setFillAfter(true);

终于效果例如以下

当然你可能想让它停留在某个状态点,你能够使用下面函数进行设置,參数值范围为0到1

  1. mPathView.setPercentage(0.8f);

甚至你能够不用svg文件,你能够使用Path类

  1. Path path=new Path();
  2. path.moveTo(0.0f, 0.0f);
  3. path.lineTo(length / 4f, 0.0f);
  4. path.lineTo(length, height / 2.0f);
  5. path.lineTo(length / 4f, height);
  6. path.lineTo(0.0f, height);
  7. path.lineTo(length * 3f / 4f, height / 2f);
  8. path.lineTo(0.0f, 0.0f);
  9. path.close();
  10. mPathView.setPath(path);

还有List类型

  1. paths = new ArrayList();
  2. //to do paths.add(path)
  3. mPathView.setPaths(paths);" data-snippet-id="ext.bbf20c1365e0d132851cfceab79afdd5" data-snippet-saved="false" data-csrftoken="4Ep2pzXU-TCk-4L9GymtrqhT8iG_nnud_1Do" data-codota-status="done">List<Path> paths = new ArrayList<Path>();
  4. //to do paths.add(path)
  5. mPathView.setPaths(paths);

当然对于自己定义属性,也能够通过代码设置

  1. mPathView.setPathColor(0x00ff00);
  2. mPathView.setPathWidth(5f);
  3. mPathView.setSvgResource(R.raw.logout);

最后贴一下svg的一些命令,每一个命令都有大写和小写形式,大写代表后面的參数是绝对坐标,小写表示相对坐标。參数之间用空格或逗号隔开

  • M: move to 移动绘制点
  • L:line to 直线
  • Z:close 闭合
  • C:cubic bezier 三次贝塞尔曲线
  • Q:quatratic bezier 二次贝塞尔曲线
  • A:ellipse 圆弧

详细解释

  • M (x y) 移动到x,y
  • L (x y) 直线连到x,y,还有简化命令H(x) 水平连接、V(y)垂直连接
  • Z。没有參数。连接起点和终点
  • C(x1 y1 x2 y2 x y),控制点x1,y1 x2,y2,终点x,y
  • Q(x1 y1 x y),控制点x1,y1,终点x,y
  • A(rx ry x-axis-rotation large-arc-flag sweep-flag x y)

    rx ry 椭圆半径

    x-axis-rotation x轴旋转角度

    large-arc-flag 为0时表示取小弧度。1时取大弧度

    sweep-flag 0取逆时针方向,1取顺时针方向

很多其它详情请參考w3c

最后上源代码http://download.csdn.net/detail/sbsujjbcy/8989661

Android SVG动画PathView源代码解析与使用教程(API 14)的更多相关文章

  1. Android 属性动画 源码解析 深入了解其内部实现

    转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/42056859,本文出自:[张鸿洋的博客] 我参加了博客之星评选,如果你喜欢我的博 ...

  2. 使用具体解释及源代码解析Android中的Adapter、BaseAdapter、ArrayAdapter、SimpleAdapter和SimpleCursorAdapter

    Adapter相当于一个数据源,能够给AdapterView提供数据.并依据数据创建相应的UI.能够通过调用AdapterView的setAdapter方法使得AdapterView将Adapter作 ...

  3. Android实现炫酷SVG动画效果

    svg是眼下十分流行的图像文件格式了,svg严格来说应该是一种开放标准的矢量图形语言,使用svg格式我们能够直接用代码来描画图像,能够用不论什么文字处理工具打开svg图像.通过改变部分代码来使图像具有 ...

  4. Android 属性动画(Property Animation) 全然解析 (下)

    转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/38092093 上一篇Android 属性动画(Property Animatio ...

  5. Android View体系(八)从源代码解析View的layout和draw流程

    相关文章 Android View体系(一)视图坐标系 Android View体系(二)实现View滑动的六种方法 Android View体系(三)属性动画 Android View体系(四)从源 ...

  6. Android属性动画完全解析(下)

    转载:http://blog.csdn.net/guolin_blog/article/details/44171115 大家好,欢迎继续回到Android属性动画完全解析.在上一篇文章当中我们学习了 ...

  7. Android属性动画完全解析(上),初识属性动画的基本用法

    转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/43536355 在手机上去实现一些动画效果算是件比较炫酷的事情,因此Android系 ...

  8. Android属性动画完全解析(中)

    转载:http://blog.csdn.net/guolin_blog/article/details/43536355 大家好,在上一篇文章当中,我们学习了Android属性动画的基本用法,当然也是 ...

  9. Android属性动画完全解析(上)

    Android属性动画完全解析(上) 转载:http://blog.csdn.net/guolin_blog/article/details/43536355 在手机上去实现一些动画效果算是件比较炫酷 ...

随机推荐

  1. 1.flume概述

    我们的web服务器等等每天会产生大量的日志,我们要把这些日志收集起来,移动到hadoop平台上进行分析. 那么如何移动这些数据呢?一种方法是通过shell cp到hadoop集群上,然后通过hdfs ...

  2. jquery 的combobox 处理级联

    随笔---jquery 的combobox 处理级联 ------------------------html------------- <select id="groupId&quo ...

  3. AC日记——逃离僵尸岛 洛谷 P3393

    逃离僵尸岛 思路: spfa: 代码: #include <cstdio> #include <cstring> #include <iostream> #incl ...

  4. 关于存储权限WRITE_EXTERNAL_STORAGE和READ_EXTERNAL_STORAGE

    WRITE_EXTERNAL_STORAGE:写入权限 READ_EXTERNAL_STORAGE:读取权限 最近申请存储权限时,在纠结是否需要同时申明WRITE_EXTERNAL_STORAGE 和 ...

  5. MVC模型绑定

    在项目中使用MVC的时候,我们无需像WebForm那样手动获取值再赋值到Model上,这得益于MVC的模型绑定,下面就介绍下复杂类型的模型绑定 Controller: public class Hom ...

  6. HDU 2824.The Euler function-筛选法求欧拉函数

    欧拉函数: φ(n)=n*(1-1/p1)(1-1/p2)....(1-1/pk),其中p1.p2…pk为n的所有素因子.比如:φ(12)=12*(1-1/2)(1-1/3)=4.可以用类似求素数的筛 ...

  7. unused import statement android studio 解决方法

    解决方法:“file”-->“invalidate caches/restart” 解决 感谢大神的解答

  8. POJ 1164 城堡问题【DFS/位运算/种子填充法/染色法】

    1 2 3 4 5 6 7 ############################# 1 # | # | # | | # #####---#####---#---#####---# 2 # # | ...

  9. 基于JS的event-manage事件管理库(一步一步实现)

    关于文章 最近在提升个人技能的同时,决定把自己为数不多的沉淀记录下来,让自己理解的更加深刻,同时也欢迎各位看官指出不足之处. 随着node.js的盛行,引领着Javascript上天下地无所不能啊,本 ...

  10. USACO1.3.2修理牛棚

    在学习一段时间贪心并写了一些贪心题之后,又一次看到了农夫和牛幸福美满的生活故事(雾).嘛,闲话少说,上题目 在一个暴风雨的夜晚,农民约翰的牛棚的屋顶.门被吹飞了. 好在许多牛正在度假,所以牛棚没有住满 ...