Android利用二阶贝塞尔曲线实现添加购物车动画详解

一、引入

  1. 其实之前一直以为像饿了么或者是美团外卖那种把商品添加到购物车的动画会很难做,但是实际做起来好像并没有想象中的那么难哈哈。
  2. 布局主要使用CoordinatorLayout+AppBarLayout+CollapsingToolbarLayout+TabLayout+ViewPager
  3. 动画主要使用二阶贝塞尔曲线与属性动画
  4. 消息传递使用EventBus普通事件

二、大致思路

1、如图所示主要有三个点,起点、终点、以及贝塞尔曲线的控制点

2、起点即点击的View的位置,一般来说用如下方式即可取得。startPosition[0]为x轴开始坐标,startPosition[1]为Y轴终点坐标,两点可以看作对角线上面的两个端点(左上角x坐标,右下角y坐标)

//贝塞尔起始数据点
int[] startPosition = new int[2];
view.getLocationOnScreen(startPosition);

3、终点即购物车篮子的位置,与起点类似

mShoppingCart.getLocationInWindow(endPosition);

4、控制点,我选的控制点为上图的C点,即A点的y坐标,B点的X坐标

controlPosition[0] = endPosition[0];
controlPosition[1] = startPosition[1];

5、需要注意的地方,我不清楚是不是因为我的布局的问题,获取到的点击的A点总是会有一个偏移,后来经同事提醒,减去了TabLayout的坐标的y轴坐标即位置才可以。

// 起点
int[] startPosition;
// 终点
int[] endPosition = new int[2];
// 贝塞尔控制点
int[] controlPosition = new int[2];
// tablayout位置
int[] tablayoutPosition = new int[2];

startPosition = data.getStartPosition();
mShoppingCart.getLocationInWindow(endPosition);
mTabLayout.getLocationInWindow(tablayoutPosition);
// 处理起点y坐标偏移的问题
startPosition[1] = startPosition[1] - tablayoutPosition[1] - mTabLayout.getHeight();
// 终点进行一下居中处理
endPosition[0] = endPosition[0] + (mShoppingCart.getWidth() / 2);
controlPosition[0] = endPosition[0];
controlPosition[1] = startPosition[1];

6、通过Path的quadTo方法绘制贝塞尔曲线,使用PathMeasure获取点的坐标(借助ValueAnimator.ofFloat()配合getPosTan()来获取坐标)

Path path = new Path();
path.moveTo(startPosition[0], startPosition[1]);
path.quadTo(controlPosition[0], controlPosition[1], endPosition[0], endPosition[1]);
PathMeasure pathMeasure = new PathMeasure();
// false表示path路径不闭合
pathMeasure.setPath(path, false);

// ofFloat是一个生成器
ValueAnimator valueAnimator = ValueAnimator.ofFloat(0, pathMeasure.getLength());
// 匀速线性插值器
valueAnimator.setInterpolator(new LinearInterpolator());
valueAnimator.setDuration(800);
valueAnimator.addUpdateListener(animation -> {
 float value = (Float) animation.getAnimatedValue();
 pathMeasure.getPosTan(value, currentPosition, null);
 imageView.setX(currentPosition[0]);
 imageView.setY(currentPosition[1]);
});
valueAnimator.start();

7、下面是用属性动画给购物车篮子做了一个放大缩小的动画效果

// mShoppingCart是View
ObjectAnimator shoppingCartX = ObjectAnimator.ofFloat(mShoppingCart, "scaleX", 1.0f, 1.3f, 1.0f);
ObjectAnimator shoppingCartY = ObjectAnimator.ofFloat(mShoppingCart, "scaleY", 1.0f, 1.3f, 1.0f);
shoppingCartX.setInterpolator(new AccelerateInterpolator());
shoppingCartY.setInterpolator(new AccelerateInterpolator());
AnimatorSet shoppingCart = new AnimatorSet();
shoppingCart
 .play(shoppingCartX)
 .with(shoppingCartY);
shoppingCart.setDuration(800);
shoppingCart.start();

三、稍完整的大部分代码

private void AddAnimation(AddEventBean data) {
 // 起点
 int[] startPosition;
 // 终点
 int[] endPosition = new int[2];
 // 贝塞尔控制点
 int[] controlPosition = new int[2];
 // 当前位置
 float[] currentPosition = new float[2];
 // tablayout位置
 int[] tablayoutPosition = new int[2];

 startPosition = data.getStartPosition();
 mShoppingCart.getLocationInWindow(endPosition);
 mTabLayout.getLocationInWindow(tablayoutPosition);
 // 处理起点y坐标偏移的问题
 startPosition[1] = startPosition[1] - tablayoutPosition[1] - mTabLayout.getHeight();
 // 终点进行一下居中处理
 endPosition[0] = endPosition[0] + (mShoppingCart.getWidth() / 2);
 controlPosition[0] = endPosition[0];
 controlPosition[1] = startPosition[1];

 final ImageView imageView = new ImageView(this);
 mConView.addView(imageView);
 imageView.setImageResource(R.drawable.specialadd);
 imageView.getLayoutParams().width = getResources().getDimensionPixelSize(R.dimen.dp_px_30);
 imageView.getLayoutParams().height = getResources().getDimensionPixelSize(R.dimen.dp_px_30);
 imageView.setScaleType(ImageView.ScaleType.CENTER_CROP);
 imageView.setVisibility(View.VISIBLE);
 imageView.setX(startPosition[0]);
 imageView.setY(startPosition[1]);

 Path path = new Path();
 path.moveTo(startPosition[0], startPosition[1]);
 path.quadTo(controlPosition[0], controlPosition[1], endPosition[0], endPosition[1]);
 PathMeasure pathMeasure = new PathMeasure();
 // false表示path路径不闭合
 pathMeasure.setPath(path, false);

 // ofFloat是一个生成器
 ValueAnimator valueAnimator = ValueAnimator.ofFloat(0, pathMeasure.getLength());
 // 匀速线性插值器
 valueAnimator.setInterpolator(new LinearInterpolator());
 valueAnimator.setDuration(800);
 valueAnimator.addUpdateListener(animation -> {
 float value = (Float) animation.getAnimatedValue();
 pathMeasure.getPosTan(value, currentPosition, null);
 imageView.setX(currentPosition[0]);
 imageView.setY(currentPosition[1]);
 });
 valueAnimator.start();

 ObjectAnimator shoppingCartX = ObjectAnimator.ofFloat(mShoppingCart, "scaleX", 1.0f, 1.3f, 1.0f);
 ObjectAnimator shoppingCartY = ObjectAnimator.ofFloat(mShoppingCart, "scaleY", 1.0f, 1.3f, 1.0f);
 shoppingCartX.setInterpolator(new AccelerateInterpolator());
 shoppingCartY.setInterpolator(new AccelerateInterpolator());
 AnimatorSet shoppingCart = new AnimatorSet();
 shoppingCart
 .play(shoppingCartX)
 .with(shoppingCartY);
 shoppingCart.setDuration(800);
 shoppingCart.start();

 valueAnimator.addListener(new Animator.AnimatorListener() {
 @Override
 public void onAnimationStart(Animator animation) {

 }

 //当动画结束后:
 @Override
 public void onAnimationEnd(Animator animation) {
 goodsChange(data);
 }

 @Override
 public void onAnimationCancel(Animator animation) {

 }

 @Override
 public void onAnimationRepeat(Animator animation) {

 }
 });
}

四、大致写下布局(同时也算留做备份)

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

 <RelativeLayout
 ... ...>

 顶部常驻的toolbar

 </RelativeLayout>

 <android.support.design.widget.CoordinatorLayout
 android:layout_width="match_parent"
 android:layout_height="0dp"
 android:layout_weight="1">

 <android.support.design.widget.AppBarLayout
 ... ...>

 <android.support.design.widget.CollapsingToolbarLayout
 ... ...
 app:layout_scrollFlags="scroll|exitUntilCollapsed">

 <LinearLayout
  ... ...>

  TabLayout上面的View

 </LinearLayout>

 </android.support.design.widget.CollapsingToolbarLayout>

 <android.support.design.widget.TabLayout
 ... ... />

 </android.support.design.widget.AppBarLayout>

 <RelativeLayout
 ... ...
 android:fillViewport="true"
 app:layout_behavior="@string/appbar_scrolling_view_behavior">

 <android.support.v4.view.ViewPager
 android:id="@+id/view_pager"
 android:layout_width="match_parent"
 android:layout_height="match_parent" />
 </RelativeLayout>

 </android.support.design.widget.CoordinatorLayout>

 <LinearLayout
 ... ...>

 最下面的购物车一栏

 </LinearLayout>
</LinearLayout>

五、推荐资源

View的位置参数

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • Android贝塞尔曲线初步学习第二课 仿QQ未读消息气泡拖拽黏连效果

    上一节初步了解了Android端的贝塞尔曲线,这一节就举个栗子练习一下,仿QQ未读消息气泡,是最经典的练习贝塞尔曲线的东东,效果如下 附上github源码地址:https://github.com/MonkeyMushroom/DragBubbleView 欢迎star~ 大体思路就是画两个圆,一个黏连小球固定在一个点上,一个气泡小球跟随手指的滑动改变坐标.随着两个圆间距越来越大,黏连小球半径越来越小.当间距小于一定值,松开手指气泡小球会恢复原来位置:当间距超过一定值之后,黏连小球消失,气泡小球

  • Android把商品添加到购物车的动画效果(贝塞尔曲线)

    当我们写商城类的项目的时候,一般都会有加入购物车的功能,加入购物车的时候会有一些抛物线动画,具体代码如下: 实现效果如图: 思路: 确定动画的起终点 在起终点之间使用二次贝塞尔曲线填充起终点之间的点的轨迹 设置属性动画,ValueAnimator插值器,获取中间点的坐标 将执行动画的控件的x.y坐标设为上面得到的中间点坐标 开启属性动画 当动画结束时的操作 难点: PathMeasure的使用 - getLength() - boolean getPosTan(float distance, f

  • android中贝塞尔曲线的应用示例

    前言: 贝塞尔曲线又称贝兹曲线,它的主要意义在于无论是直线或曲线都能在数学上予以描述.最初由保罗·德卡斯特里奥(Paul de Casteljau)于1959年运用德卡斯特里奥演算法开发(de Casteljau Algorithm),在1962,由法国工程师皮埃尔·贝塞尔(Pierre Bézier)所广泛发表.目前广泛应用于图形绘制领域来模拟光滑曲线,为计算机矢量图形学奠定了基础.在一些图形处理软件中都能见到贝塞尔曲线,比如CorelDraw中翻译成"贝赛尔工具":而在Firewo

  • Android贝塞尔曲线初步学习第三课 Android实现添加至购物车的运动轨迹

    不知上一节高仿QQ未读消息气泡大家还喜欢么,今天继续练习贝赛尔曲线,这一节我们通过贝赛尔曲线和属性动画估值器实现添加至购物车的运动轨迹,效果如下: 1.新建自定义View,重写构造方法,初始化Paint.Path: 2.确定起始点.终止点.控制点坐标,这里我们直接固定: @Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(w, h, oldw, oldh);

  • Android贝塞尔曲线实现填充不规则图形并随手指运动

    贝塞尔曲线: 贝塞尔曲线于1962,由法国工程师皮埃尔·贝塞尔所广泛发表,他运用贝塞尔曲线来为汽车的主体进行设计.贝塞尔曲线最初由 Paul de Casteljau 于 1959 年运用 de Casteljau 演算法开发,以稳定数值的方法求出贝兹曲线.贝塞尔曲线主要用于二维图形应用程序中的数学曲线,曲线由起始点,终止点(也称锚点)和控制点组成,通过调整控制点,贝塞尔曲线的形状会发生变化. 在此举一个例子,实现贝塞尔曲线,基于以下场景: 上面的图片,我们可以见到一个白色的区域,边缘为弧形,这

  • Android贝塞尔曲线实现手指轨迹

    本文实例为大家分享了Android贝塞尔曲线实现手指轨迹的具体代码,供大家参考,具体内容如下 1.使用贝塞尔曲线前 MyView.java public class MyView extends View { // 实例一个路径对象 private Path mPath = new Path(); public MyView(Context context) { super(context); // TODO Auto-generated constructor stub } public My

  • Android中贝塞尔曲线的绘制方法示例代码

    贝塞尔曲线,很多人可能不太了解,什么叫做贝塞尔曲线呢?这里先做一下简单介绍:贝塞尔曲线也可以叫做贝济埃曲线或者贝兹曲线,它由线段与节点组成,节点是可拖动的支点,线段像可伸缩的皮筋.一般的矢量图形软件常利用贝塞尔曲线来精确画出曲线. 上面的介绍中,"线段像可伸缩的皮筋"这句话非常关键,但也特别好理解.至于贝塞尔曲线的详细内容大家可以查阅相关资料.        Android提供的贝塞尔曲线绘制接口 在Android开发中,要实现贝塞尔曲线其实还是很简单的,因为Android已经给我们提

  • Android贝塞尔曲线初步学习第一课

    贝塞尔曲线有一阶.二阶.三阶.N阶 一阶就是一条直线,有起点终点,没有控制点,对应方法就是 canvas.drawLine(float startX, float startY, float stopX, float stopY, @NonNull Paint paint) ; 二阶为曲线,有起点终点,一个控制点,对应方法就是 path.quadTo(float x1, float y1, float x2, float y2); 其中x1.y1为控制点坐标, x2.y2为终点坐标,效果如下:

  • Android Path绘制贝塞尔曲线实现QQ拖拽泡泡

    这两天学习了使用Path绘制贝塞尔曲线相关,然后自己动手做了一个类似QQ未读消息可拖拽的小气泡,效果图如下: 最终效果图 接下来一步一步的实现整个过程. 基本原理 其实就是使用Path绘制三点的二次方贝塞尔曲线来完成那个妖娆的曲线的.然后根据触摸点不断绘制对应的圆形,根据距离的改变改变原始固定圆形的半径大小.最后就是松手后返回或者爆裂的实现. Path介绍: 顾名思义,就是一个路径的意思,Path里面有很多的方法,本次设计主要用到的相关方法有 moveTo() 移动Path到一个指定的点 qua

  • Android 利用三阶贝塞尔曲线绘制运动轨迹的示例

    本篇文章主要介绍了Android 利用三阶贝塞尔曲线绘制运动轨迹的示例,分享给大家,具体如下: 实现点赞效果,自定义起始点以及运动轨迹 效果图: xml布局: <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/rl_root&

随机推荐