前面两篇文章主要讲解了 Path 的概念和基本使用,今天我们一起利用 Path 做个比较实用的小例子;

上一篇我们使用 Path 绘制了一个小桃心,我们这一篇继续围绕着这个小桃心进行展开:

--------------------------------------------------

如果你想看 GAStudio
Github主页
,请戳这里; 
如果你想看 GAStudio更多技术文章,请戳这里; 
QQ技术交流群:277582728;

--------------------------------------------------

如果对这个桃心绘制有问题或有兴趣的同学,可以链接到 Path相关方法讲解(二),此时我们的需求是这样的:

假定我们现在是一个婚恋产品,有一个“心动”的功能,用户点击“心动”按钮的时候,有一个光点快速的沿着桃心转一圈,然后整个桃心泛起光晕!

针对这个需求,很多人可能会想到以下方案:

不就一个光点沿着桃心跑一圈么,既然桃心是使用贝塞尔曲线画出来的,那么我们就可以用对应的函数模拟出这条曲线,然后算出对应位置上的点,不断将光点绘制到对应的位置上!

这个思路当然没有问题,但我们还有相对简单的方式,那就是使用 PathMeasure:

我们主要使用它两个方法:

1.getLength() - 获取路径的长度

2.getPosTan(float
distance, float pos[],float
tan[]) - path 为 null ,返回 false

distance 为一个 0 - getLength() 之间的值,根据这个值 PathMeasure 会计算出当前点的坐标封装到 pos 中;

上面这句话我们可以这么来理解,不管实际
Path 多么的复杂,PathMeasure 都相当于做了一个事情,就是把 Path
“拉直”,然后给了我们一个接口(getLength)告诉我们path的总长度,然后我们想要知道具体某一点的坐标,只需要用相对的distance去取即可,这样就省去了自己用函数模拟path,然后计算获取点坐标的过程;

接下来,我们用代码实现这一效果:

我们先创建一个 PathMeasure ,并将创建好的 path 作为参数传入

  1. mPath = new Path();
  2. mPath.moveTo(START_POINT[0], START_POINT[1]);
  3. mPath.quadTo(RIGHT_CONTROL_POINT[0], RIGHT_CONTROL_POINT[1], BOTTOM_POINT[0],
  4. BOTTOM_POINT[1]);
  5. mPath.quadTo(LEFT_CONTROL_POINT[0], LEFT_CONTROL_POINT[1], START_POINT[0], START_POINT[1]);
  6. mPathMeasure = new PathMeasure(mPath, true);

然后用一个数组纪录点的坐标:

  1. private float[] mCurrentPosition = new float[2];

向外暴露一个开启动效的接口:

  1. // 开启路径动画
  2. public void startPathAnim(long duration) {
  3. // 0 - getLength()
  4. ValueAnimator valueAnimator = ValueAnimator.ofFloat(0, mPathMeasure.getLength());
  5. Log.i(TAG, "measure length = " + mPathMeasure.getLength());
  6. valueAnimator.setDuration(duration);
  7. // 减速插值器
  8. valueAnimator.setInterpolator(new DecelerateInterpolator());
  9. valueAnimator.addUpdateListener(new AnimatorUpdateListener() {
  10. @Override
  11. public void onAnimationUpdate(ValueAnimator animation) {
  12. float value = (Float) animation.getAnimatedValue();
  13. // 获取当前点坐标封装到mCurrentPosition
  14. mPathMeasure.getPosTan(value, mCurrentPosition, null);
  15. postInvalidate();
  16. }
  17. });
  18. valueAnimator.start();
  19. }

实时获取到当前点之后,将目标绘制到对应位置:

  1. protected void onDraw(Canvas canvas) {
  2. super.onDraw(canvas);
  3. canvas.drawColor(Color.WHITE);
  4. canvas.drawPath(mPath, mPaint);
  5. canvas.drawCircle(RIGHT_CONTROL_POINT[0], RIGHT_CONTROL_POINT[1], 5, mPaint);
  6. canvas.drawCircle(LEFT_CONTROL_POINT[0], LEFT_CONTROL_POINT[1], 5, mPaint);
  7. // 绘制对应目标
  8. canvas.drawCircle(mCurrentPosition[0], mCurrentPosition[1], 10, mPaint);
  9. }

到这里目标环绕 path 的效果就ok了,不管这条路径简单也好,复杂也罢,我们都可以如此简单的完成对应的效果,而不需要自己用简单或复杂函数模拟求解了;

完成了一步,自己提的需求还有一点就是光晕的问题,这个东西如何是好呢?切图?! 不需要,Android 已经给我们提供了一个好用的东西 MaskFilter ,后面我就不做了,大家有兴趣自己做的玩玩,只需要注意一点,MaskFilter
不支持硬件加速,记得关掉!

好了,PathMeasure 看似很简单,但着实很有用,有了它,再结合上 Path 、Shader、ColorMatrix  等利器,我们已经可以做出很多酷炫的效果了!

最后,完整的代码献上,请笑纳:

  1. public class DynamicHeartView extends View {
  2. private static final String TAG = "DynamicHeartView";
  3. private static final int PATH_WIDTH = 2;
  4. // 起始点
  5. private static final int[] START_POINT = new int[] {
  6. 300, 270
  7. };
  8. // 爱心下端点
  9. private static final int[] BOTTOM_POINT = new int[] {
  10. 300, 400
  11. };
  12. // 左侧控制点
  13. private static final int[] LEFT_CONTROL_POINT = new int[] {
  14. 450, 200
  15. };
  16. // 右侧控制点
  17. private static final int[] RIGHT_CONTROL_POINT = new int[] {
  18. 150, 200
  19. };
  20. private PathMeasure mPathMeasure;
  21. private Paint mPaint;
  22. private Path mPath;
  23. private float[] mCurrentPosition = new float[2];
  24. public DynamicHeartView(Context context) {
  25. super(context);
  26. init();
  27. }
  28. private void init() {
  29. mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
  30. mPaint.setStyle(Style.STROKE);
  31. mPaint.setStrokeWidth(PATH_WIDTH);
  32. mPaint.setColor(Color.RED);
  33. mPath = new Path();
  34. mPath.moveTo(START_POINT[0], START_POINT[1]);
  35. mPath.quadTo(RIGHT_CONTROL_POINT[0], RIGHT_CONTROL_POINT[1], BOTTOM_POINT[0],
  36. BOTTOM_POINT[1]);
  37. mPath.quadTo(LEFT_CONTROL_POINT[0], LEFT_CONTROL_POINT[1], START_POINT[0], START_POINT[1]);
  38. mPathMeasure = new PathMeasure(mPath, true);
  39. mCurrentPosition = new float[2];
  40. }
  41. @Override
  42. protected void onDraw(Canvas canvas) {
  43. super.onDraw(canvas);
  44. canvas.drawColor(Color.WHITE);
  45. canvas.drawPath(mPath, mPaint);
  46. canvas.drawCircle(RIGHT_CONTROL_POINT[0], RIGHT_CONTROL_POINT[1], 5, mPaint);
  47. canvas.drawCircle(LEFT_CONTROL_POINT[0], LEFT_CONTROL_POINT[1], 5, mPaint);
  48. // 绘制对应目标
  49. canvas.drawCircle(mCurrentPosition[0], mCurrentPosition[1], 10, mPaint);
  50. }
  51. // 开启路径动画
  52. public void startPathAnim(long duration) {
  53. // 0 - getLength()
  54. ValueAnimator valueAnimator = ValueAnimator.ofFloat(0, mPathMeasure.getLength());
  55. Log.i(TAG, "measure length = " + mPathMeasure.getLength());
  56. valueAnimator.setDuration(duration);
  57. // 减速插值器
  58. valueAnimator.setInterpolator(new DecelerateInterpolator());
  59. valueAnimator.addUpdateListener(new AnimatorUpdateListener() {
  60. @Override
  61. public void onAnimationUpdate(ValueAnimator animation) {
  62. float value = (Float) animation.getAnimatedValue();
  63. // 获取当前点坐标封装到mCurrentPosition
  64. mPathMeasure.getPosTan(value, mCurrentPosition, null);
  65. postInvalidate();
  66. }
  67. });
  68. valueAnimator.start();
  69. }
  70. }

以上代码的效果如下,其余效果大家自行补充:

PathMeasure打造万能路径特效 - 源码下载

Path特效之PathMeasure打造万能路径动效的更多相关文章

  1. android打造万能的适配器(转)

    荒废了两天,今天与大家分享一个ListView的适配器 前段时间在学习慕课网的视频,觉得这种实现方式较好,便记录了下来,最近的项目中也使用了多次,节省了大量的代码,特此拿来与大家分享一下. 还是先看图 ...

  2. Android 快速开发系列 打造万能的ListView GridView 适配器

    转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/38902805 ,本文出自[张鸿洋的博客] 1.概述 相信做Android开发的写 ...

  3. 安卓开发笔记——打造万能适配器(Adapter)

    为什么要打造万能适配器? 在安卓开发中,用到ListView和GridView的地方实在是太多了,系统默认给我们提供的适配器(ArrayAdapter,SimpleAdapter)经常不能满足我们的需 ...

  4. 解决 U2000 R017 安装报错: 检查SQL server数据库环境变量信息 ( 异常 ) [ 详细信息 ] PATH环境变量中缺少数据库路径的信息

    U2000 R017 安装报错: 检查SQL server数据库环境变量信息 ( 异常 ) [ 详细信息 ] PATH环境变量中缺少数据库路径的信息 管理员模式打开注册表位置: HKEY_LOCAL_ ...

  5. H5动效的常见制作手法

    众所周知,一个元素,动往往比静更吸引眼球: 一套操作界面,合适的动态交互反馈能给用户带来更好的操作体验: 一个H5运营宣传页,炫酷的动画特效定能助力传播和品牌打造. 近两年,小到loading动画,表 ...

  6. 动效解析工厂:Mask 动画

    转载自:http://www.cocoachina.com/ios/20160214/15250.html 前言:很多动效都是多种动画的组合,有时候你可能只是需要其中某个动画,但面对庞杂的代码库或是教 ...

  7. android动效开篇

    大神博客:http://blog.csdn.net/tianjian4592/article/details/44155147 在现在的Android App开发中,动效越来越受到产品和设计师同学的重 ...

  8. 玩转HTML5移动页面(动效篇)

    原文:http://www.grycheng.com/?p=458 作为一名前端,在拿到设计稿时你有两种选择: 1.快速输出静态页面 2.加上高级大气上档次狂拽炫酷屌炸天的动画让页面动起来 作为一个有 ...

  9. 基于clip-path的任意元素的碎片拼接动效(源自鑫空间)

    一.实现原理. 效果本质上是CSS3动画,就是旋转transform:rotate和位移:transform:translate,只是旋转和位移的部件是三角碎片而已.三角是使用CSS3 clip-pa ...

随机推荐

  1. Python框架之Django学习笔记(二)

    安装Django 我是在windows下安装的python以及django,下面的链接可以下载Django: http://www.djangoproject.com/download/ 1.下载 D ...

  2. Windows核心编程小结3

    内存映射和堆栈 内存映射文件 内存映射文件可以用于3个不同的目的: 系统使用内存映射文件,以便加载和执行.exe和DLL文件.这可以大大节省页文件空间和应用程序启动运行所需的时间. 可以使用内存映射文 ...

  3. Linux之ubuntu系统操作学习笔记

    1,swp分区:当内存不够时用swp分区顶替内存 2,语言环境检查  locale –a:可以明白系统支持什么语言 3,安装软件: apt-cache search(软件):搜索软件 apt-cach ...

  4. hnust 心电图

    问题 A: 心电图 时间限制: 1 Sec  内存限制: 128 MB提交: 621  解决: 250[提交][状态][讨论版] 题目描述 众所周知,ACM/ICPC实验室聚集了一堆学霸Orz 有学霸 ...

  5. 【Luogu】P2468粟粟的书架(主席树+前缀和)

    题目链接 我仿佛中了个爆零debuff 本题分成两部分,五十分用前缀和,f[i][j][k]表示(1,1)到(i,j)的矩形大于等于k的有多少个数(再记录页数和),查询时二分,另外的用主席树,类似方法 ...

  6. 【Luogu】P3750分手是祝愿(期望DP)

    题目链接 这题好喵啊…… 设f[i]是最少用i次才能全关上转移到最少用i-1次才能全关上灯的期望值,那么n个灯里有i个是正确的,剩下的都是不正确的 因此期望是$f[i]=frac{n}{i}+frac ...

  7. jQuery动画的hover连续触发动画bug处理

    一.问题 为元素设置hover上实现动画的效果,当鼠标反复快速进入元素时,动画会在鼠标停止后依然执行,导致动画和鼠标的动作不一致. 二.解决方法 要解决这种问题,可以使用jquery的stop()方法 ...

  8. Codeforces Round #323 (Div. 2) B 贪心,暴力

    B. Robot's Task time limit per test 1 second memory limit per test 256 megabytes input standard inpu ...

  9. docker介绍与安装

    .docker是什么 Docker是一个开源的应用容器引擎,使用Go语言开发,基于Linux内核的cgroup,namespace,Union FS等技术,对应用进程进行封装隔离,并且独立于宿主机与其 ...

  10. Java语法糖(一)

    概述 语法糖(Syntactic Sugar):主要作用是提高编码效率,减少编码出错的机会. 解语法糖发生在Java源码被编译成Class字节码的过程中,还原回简单的基础语法结构. 语法糖之一:泛型( ...