android购物车曲线,贝塞尔曲线之购物车动画效果

question

贝塞尔曲线是什么?

贝塞尔曲线可以做什么?

怎么做?

what is it?

贝塞尔曲线在百度定义是贝塞尔曲线(Bézier curve),又称贝兹曲线或贝济埃曲线,是应用于二维图形应用程序的数学曲线。

0688db4f1bcf

1132780-253900d649118aa0.jpg

do what?

贝塞尔曲线根据不同点实现不同动态效果:

一阶贝塞尔曲线(两点),绘制成一条直线

0688db4f1bcf

1359207897_7842.gif

二阶贝塞尔曲线(三点)

0688db4f1bcf

1359208080_9518.gif

三阶贝塞尔曲线(四点)

0688db4f1bcf

1359208177_9516.gif

四阶贝塞尔曲线(五点)

0688db4f1bcf

20160402092908480.gif

五阶贝塞尔曲线(六点)

0688db4f1bcf

20160405093717253.gif

看了上面贝塞尔曲线不同点不同效果后,相信大家都清楚贝塞尔曲线能干什么?没错,贝塞尔曲线能造高逼格动画~~

就笔者目前了解的采用贝塞尔曲线实现的知名开源项目有:

QQ拖拽清除效果

纸飞机刷新动画

滴油刷新动画

波浪动画

到此大家是不是很兴奋,想更多了解如何造一个高逼格贝塞尔曲线动画。接下来我就给大家讲述如何造一个基于贝塞尔曲线实现的购物车动画,大家擦亮眼睛啦~~

0688db4f1bcf

1132780-2d7d1306d1b86b65.jpg

how to do it

思路

确定动画起终点

在起终点之间使用二次贝塞尔曲线填充起终点之间点的轨迹

设置属性动画,ValueAnimator插值器,获取中间点的坐标

将执行动画控件的x、y坐标设为上面得到的中间点坐标

开启属性动画

当动画结束时的操作

知识点

Android中提供了绘制一阶、二阶、三阶的接口:

一阶接口:

public void lineTo(float x,float y)

二阶接口:

public void quadTo(float x1, float y1, float x2, float y2)

三阶接口:

public void cubicTo(float x1, float y1, float x2, float y2, float x3, float y3)

PathMeasure使用

getLength()

理解 boolean getPosTan(float distance, float[] pos, float[] tan)

如何获取控件在屏幕中的绝对坐标

int[] location = new int[2]; view.getLocationInWindow(location); 得到view在屏幕中的绝对坐标。

理解属性动画插值器ValueAnimator

code

首先写购物车布局xml,代码如下:

android:id="@+id/rly_bezier_curve_shopping_cart"

android:layout_width="match_parent"

android:layout_height="match_parent"

android:paddingBottom="@dimen/activity_vertical_margin"

android:paddingLeft="@dimen/activity_horizontal_margin"

android:paddingRight="@dimen/activity_horizontal_margin"

android:paddingTop="@dimen/activity_vertical_margin">

android:id="@+id/fly_bezier_curve_shopping_cart"

android:layout_width="match_parent"

android:layout_height="wrap_content"

android:layout_alignParentBottom="true"

android:layout_alignParentLeft="true"

android:paddingRight="30dp"

android:layout_alignParentStart="true">

android:id="@+id/iv_bezier_curve_shopping_cart"

android:layout_width="40dp"

android:layout_height="40dp"

android:layout_gravity="right"

android:src="@drawable/menu_shop_car_selected" />

android:id="@+id/tv_bezier_curve_shopping_cart_count"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:textColor="@color/white"

android:background="@drawable/corner_view"

android:text="0"

android:layout_gravity="right"/>

android:id="@+id/lv_bezier_curve_shopping_cart"

android:layout_width="match_parent"

android:layout_height="match_parent"

android:layout_above="@+id/fly_bezier_curve_shopping_cart"/>

然后写购物车适配器、实体类,代码如下:

/**

* @className: GoodsAdapter

* @classDescription: 购物车商品适配器

* @author: leibing

* @createTime: 2016/09/28

*/

public class GoodsAdapter extends BaseAdapter{

// 数据源(购物车商品图片)

private ArrayList mData;

// 布局

private LayoutInflater mLayoutInflater;

// 回调监听

private CallBackListener mCallBackListener;

/**

* 构造函数

* @author leibing

* @createTime 2016/09/28

* @lastModify 2016/09/28

* @param context 上下文

* @param mData 数据源(购物车商品图片)

* @return

*/

public GoodsAdapter(Context context, ArrayList mData){

mLayoutInflater = LayoutInflater.from(context);

this.mData = mData;

}

@Override

public int getCount() {

return mData != null ? mData.size(): 0;

}

@Override

public Object getItem(int i) {

return mData != null ? mData.get(i): null;

}

@Override

public long getItemId(int i) {

return i;

}

@Override

public View getView(int i, View view, ViewGroup viewGroup) {

ViewHolder viewHolder;

if (view == null){

view = mLayoutInflater.inflate(R.layout.adapter_shopping_cart_item, null);

viewHolder = new ViewHolder(view);

view.setTag(viewHolder);

}else {

// 复用ViewHolder

viewHolder = (ViewHolder) view.getTag();

}

// 更新UI

if (i < mData.size())

viewHolder.updateUI(mData.get(i));

return view;

}

/**

* @className: ViewHolder

* @classDescription: 商品ViewHolder

* @author: leibing

* @createTime: 2016/09/28

*/

class ViewHolder{

// 显示商品图片

private ImageView mShoppingCartItemIv;

/**

* 构造函数

* @author leibing

* @createTime 2016/09/28

* @lastModify 2016/09/28

* @param view 视图

* @return

*/

public ViewHolder(View view){

// findView

mShoppingCartItemIv = (ImageView) view.findViewById(R.id.iv_shopping_cart_item);

// onClick

view.findViewById(R.id.tv_shopping_cart_item).setOnClickListener(

new View.OnClickListener() {

@Override

public void onClick(View view) {

if (mShoppingCartItemIv != null && mCallBackListener != null)

mCallBackListener.callBackImg(mShoppingCartItemIv);

}

});

}

/**

* 更新UI

* @author leibing

* @createTime 2016/09/28

* @lastModify 2016/09/28

* @param goods 商品实体对象

* @return

*/

public void updateUI(GoodsModel goods){

if (goods != null

&& goods.getmGoodsBitmap() != null

&& mShoppingCartItemIv != null)

mShoppingCartItemIv.setImageBitmap(goods.getmGoodsBitmap());

}

}

/**

* 设置回调监听

* @author leibing

* @createTime 2016/09/28

* @lastModify 2016/09/28

* @param mCallBackListener 回调监听

* @return

*/

public void setCallBackListener(CallBackListener mCallBackListener){

this.mCallBackListener = mCallBackListener;

}

/**

* @interfaceName: CallBackListener

* @interfaceDescription: 回调监听

* @author: leibing

* @createTime: 2016/09/28

*/

public interface CallBackListener{

void callBackImg(ImageView goodsImg);

}

}

然后写添加数据源以及设置适配器,代码如下:

// 购物车父布局

private RelativeLayout mShoppingCartRly;

// 购物车列表显示

private ListView mShoppingCartLv;

// 购物数目显示

private TextView mShoppingCartCountTv;

// 购物车图片显示

private ImageView mShoppingCartIv;

// 购物车适配器

private GoodsAdapter mGoodsAdapter;

// 数据源(购物车商品图片)

private ArrayList mData;

// 贝塞尔曲线中间过程点坐标

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

// 路径测量

private PathMeasure mPathMeasure;

// 购物车商品数目

private int goodsCount = 0;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

// findView

mShoppingCartLv = (ListView) findViewById(R.id.lv_bezier_curve_shopping_cart);

mShoppingCartCountTv = (TextView) findViewById(R.id.tv_bezier_curve_shopping_cart_count);

mShoppingCartRly = (RelativeLayout) findViewById(R.id.rly_bezier_curve_shopping_cart);

mShoppingCartIv = (ImageView) findViewById(R.id.iv_bezier_curve_shopping_cart);

// 是否显示购物车商品数目

isShowCartGoodsCount();

// 添加数据源

addData();

// 设置适配器

setAdapter();

}

/**

* 设置适配器

* @author leibing

* @createTime 2016/09/28

* @lastModify 2016/09/28

* @param

* @return

*/

private void setAdapter() {

// 初始化适配器

mGoodsAdapter = new GoodsAdapter(this, mData);

// 设置适配器监听

mGoodsAdapter.setCallBackListener(new GoodsAdapter.CallBackListener() {

@Override

public void callBackImg(ImageView goodsImg) {

// 添加商品到购物车

addGoodsToCart(goodsImg);

}

});

// 设置适配器

mShoppingCartLv.setAdapter(mGoodsAdapter);

}

接下来写最重要的一块,添加商品到购物车,代码如下:

/**

* 添加商品到购物车

* @author leibing

* @createTime 2016/09/28

* @lastModify 2016/09/28

* @param goodsImg 商品图标

* @return

*/

private void addGoodsToCart(ImageView goodsImg) {

// 创造出执行动画的主题goodsImg(这个图片就是执行动画的图片,从开始位置出发,经过一个抛物线(贝塞尔曲线),移动到购物车里)

final ImageView goods = new ImageView(this);

goods.setImageDrawable(goodsImg.getDrawable());

RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(100, 100);

mShoppingCartRly.addView(goods, params);

// 得到父布局的起始点坐标(用于辅助计算动画开始/结束时的点的坐标)

int[] parentLocation = new int[2];

mShoppingCartRly.getLocationInWindow(parentLocation);

// 得到商品图片的坐标(用于计算动画开始的坐标)

int startLoc[] = new int[2];

goodsImg.getLocationInWindow(startLoc);

// 得到购物车图片的坐标(用于计算动画结束后的坐标)

int endLoc[] = new int[2];

mShoppingCartIv.getLocationInWindow(endLoc);

// 开始掉落的商品的起始点:商品起始点-父布局起始点+该商品图片的一半

float startX = startLoc[0] - parentLocation[0] + goodsImg.getWidth() / 2;

float startY = startLoc[1] - parentLocation[1] + goodsImg.getHeight() / 2;

// 商品掉落后的终点坐标:购物车起始点-父布局起始点+购物车图片的1/5

float toX = endLoc[0] - parentLocation[0] + mShoppingCartIv.getWidth() / 5;

float toY = endLoc[1] - parentLocation[1];

// 开始绘制贝塞尔曲线

Path path = new Path();

// 移动到起始点(贝塞尔曲线的起点)

path.moveTo(startX, startY);

// 使用二阶贝塞尔曲线:注意第一个起始坐标越大,贝塞尔曲线的横向距离就会越大,一般按照下面的式子取即可

path.quadTo((startX + toX) / 2, startY, toX, toY);

// mPathMeasure用来计算贝塞尔曲线的曲线长度和贝塞尔曲线中间插值的坐标,如果是true,path会形成一个闭环

mPathMeasure = new PathMeasure(path, false);

// 属性动画实现(从0到贝塞尔曲线的长度之间进行插值计算,获取中间过程的距离值)

ValueAnimator valueAnimator = ValueAnimator.ofFloat(0, mPathMeasure.getLength());

valueAnimator.setDuration(500);

// 匀速线性插值器

valueAnimator.setInterpolator(new LinearInterpolator());

valueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {

@Override

public void onAnimationUpdate(ValueAnimator animation) {

// 当插值计算进行时,获取中间的每个值,

// 这里这个值是中间过程中的曲线长度(下面根据这个值来得出中间点的坐标值)

float value = (Float) animation.getAnimatedValue();

// 获取当前点坐标封装到mCurrentPosition

// boolean getPosTan(float distance, float[] pos, float[] tan) :

// 传入一个距离distance(0<=distance<=getLength()),然后会计算当前距离的坐标点和切线,pos会自动填充上坐标,这个方法很重要。

// mCurrentPosition此时就是中间距离点的坐标值

mPathMeasure.getPosTan(value, mCurrentPosition, null);

// 移动的商品图片(动画图片)的坐标设置为该中间点的坐标

goods.setTranslationX(mCurrentPosition[0]);

goods.setTranslationY(mCurrentPosition[1]);

}

});

// 开始执行动画

valueAnimator.start();

// 动画结束后的处理

valueAnimator.addListener(new Animator.AnimatorListener() {

@Override

public void onAnimationStart(Animator animation) {

}

@Override

public void onAnimationEnd(Animator animation) {

// 购物车商品数量加1

goodsCount ++;

isShowCartGoodsCount();

mShoppingCartCountTv.setText(String.valueOf(goodsCount));

// 把执行动画的商品图片从父布局中移除

mShoppingCartRly.removeView(goods);

}

@Override

public void onAnimationCancel(Animator animation) {

}

@Override

public void onAnimationRepeat(Animator animation) {

}

});

}

0688db4f1bcf

1132780-f58fe3a5c5a7c281.jpg

代码分析完毕,一个高逼格贝塞尔曲线实现的购物车添加商品动画效果实现分析完毕~~

效果图如下:

0688db4f1bcf

BezierCurveShoppingCart.gif

笔者文笔太糟,欢迎吐槽,如有不对之处,请留言指点~~

呼吁大家动手实践,一切将会变得很容易~~~

关于作者

js加入购物车抛物线动画购物车效果特效,亲测可用, 当您在电商购物网站浏览中意的商品时,您可以点击页面中的“加入购物车”按钮即可将商品加入的购物车中。本文介绍借助一款基于jQuery的动画插件,点击加入购物车按钮时,实现商品将飞入到右侧的购物车中的效果。 HTML 首先载入jQuery库文件和jquery.fly.min.js插件。 复制代码 代码如下: 接着,将商品信息html结构布置好,本例中,我们用四个商品并排布置,每个商品box中包括有商品图片、价格、名称以及加入购物车按钮等信息。 复制代码 代码如下: ¥3499.00 LG 49LF5400-CA 49寸IPS硬屏富贵招财铜钱设计 加入购物车 ¥3799.00 Hisense/海信 LED50T1A 海信电视官方旗舰店 加入购物车 ¥¥3999.00 Skyworth/创维 50E8EUS 8核4Kj极清酷开系统智能液晶电视 加入购物车 ¥6969.00 乐视TV Letv X60S 4核1080P高清3D安卓智能超级电视 加入购物车 然后,我们还需要在页面的右侧加上购物车以及提示信息。 复制代码 代码如下: 购物车 已成功加入购物车! CSS 我们使用CSS先将商品排列美化,然后设置右侧购物车样式,具体请看代码: 复制代码 代码如下: .box{float:left; width:198px; height:320px; margin-left:5px; border:1px solid #e0e0e0; text-align:center} .box p{line-height:20px; padding:4px 4px 10px 4px; text-align:left} .box:hover{border:1px solid #f90} .box h4{line-height:32px; font-size:14px; color:#f30;font-weight:500} .box h4 span{font-size:20px} .u-flyer{display: block;width: 50px;height: 50px;border-radius: 50px;position: fixed;z-index: 9999;} .m-sidebar{position: fixed;top: 0;right: 0;background: #000;z-index: 2000;width: 35px;height: 100%;font-size: 12px;color: #fff;} .cart{color: #fff;t
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值