Android自定义系列——8.Path之贝塞尔曲线

2024-06-24 05:18

本文主要是介绍Android自定义系列——8.Path之贝塞尔曲线,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

贝塞尔曲线能干什么

贝塞尔曲线作用十分广泛,简单举几个的栗子:

QQ小红点拖拽效果
一些炫酷的下拉刷新控件
阅读软件的翻书效果
一些平滑的折线图的制作
很多炫酷的动画效果

理解贝塞尔曲线的原理

一阶曲线原理:
一阶曲线是没有控制点的,仅有两个数据点(A 和 B),最终动态过程如下:
在这里插入图片描述(本文中贝塞尔曲线相关的动态演示图片来自维基百科)。一阶曲线其实就是前面讲解过的lineTo。

二阶曲线原理:
二阶曲线由两个数据点(A 和 C),一个控制点(B)来描述曲线状态,大致如下:
在这里插入图片描述连接AB BC,并在AB上取点D,BC上取点E,使其满足条件:
在这里插入图片描述
在这里插入图片描述连接DE,取点F,使得:

在这里插入图片描述
这样获取到的点F就是贝塞尔曲线上的一个点,动态过程如下:
在这里插入图片描述二阶曲线对应的方法是quadTo。

三阶曲线原理:
三阶曲线由两个数据点(A 和 D),两个控制点(B 和 C)来描述曲线状态,如下:
在这里插入图片描述三阶曲线计算过程与二阶类似,具体可以见下图动态效果:
在这里插入图片描述三阶曲线对应的方法是cubicTo。

了解贝塞尔曲线相关函数使用方法

一阶曲线:
一阶曲线是一条线段,可以参见上一篇Android自定义系列——7.Path之基本操作 。

二阶曲线:
二阶曲线是由两个数据点,一个控制点构成。
在这里插入图片描述上图中绘制出了辅助点和辅助线,从上面的动态图可以看出,贝塞尔曲线在动态变化过程中有类似于橡皮筋一样的弹性效果,因此在制作一些弹性效果的时候很常用。

public class Bezier extends View {private Paint mPaint;private int centerX, centerY;private PointF start, end, control;public Bessel1(Context context) {super(context);mPaint = new Paint();mPaint.setColor(Color.BLACK);mPaint.setStrokeWidth(8);mPaint.setStyle(Paint.Style.STROKE);mPaint.setTextSize(60);start = new PointF(0,0);end = new PointF(0,0);control = new PointF(0,0);}@Overrideprotected void onSizeChanged(int w, int h, int oldw, int oldh) {super.onSizeChanged(w, h, oldw, oldh);centerX = w/2;centerY = h/2;// 初始化数据点和控制点的位置start.x = centerX-200;start.y = centerY;end.x = centerX+200;end.y = centerY;control.x = centerX;control.y = centerY-100;}@Overridepublic boolean onTouchEvent(MotionEvent event) {// 根据触摸位置更新控制点,并提示重绘control.x = event.getX();control.y = event.getY();invalidate();return true;}@Overrideprotected void onDraw(Canvas canvas) {super.onDraw(canvas);// 绘制数据点和控制点mPaint.setColor(Color.GRAY);mPaint.setStrokeWidth(20);canvas.drawPoint(start.x,start.y,mPaint);canvas.drawPoint(end.x,end.y,mPaint);canvas.drawPoint(control.x,control.y,mPaint);// 绘制辅助线mPaint.setStrokeWidth(4);canvas.drawLine(start.x,start.y,control.x,control.y,mPaint);canvas.drawLine(end.x,end.y,control.x,control.y,mPaint);// 绘制贝塞尔曲线mPaint.setColor(Color.RED);mPaint.setStrokeWidth(8);Path path = new Path();path.moveTo(start.x,start.y);path.quadTo(control.x,control.y,end.x,end.y);canvas.drawPath(path, mPaint);}
}

三阶曲线:
三阶曲线由两个数据点和两个控制点来控制曲线状态。
在这里插入图片描述

public class Bezier2 extends View {private Paint mPaint;private int centerX, centerY;private PointF start, end, control1, control2;private boolean mode = true;public Bezier2(Context context) {this(context, null);}public Bezier2(Context context, AttributeSet attrs) {super(context, attrs);mPaint = new Paint();mPaint.setColor(Color.BLACK);mPaint.setStrokeWidth(8);mPaint.setStyle(Paint.Style.STROKE);mPaint.setTextSize(60);start = new PointF(0, 0);end = new PointF(0, 0);control1 = new PointF(0, 0);control2 = new PointF(0, 0);}public void setMode(boolean mode) {this.mode = mode;}@Overrideprotected void onSizeChanged(int w, int h, int oldw, int oldh) {super.onSizeChanged(w, h, oldw, oldh);centerX = w / 2;centerY = h / 2;// 初始化数据点和控制点的位置start.x = centerX - 200;start.y = centerY;end.x = centerX + 200;end.y = centerY;control1.x = centerX;control1.y = centerY - 100;control2.x = centerX;control2.y = centerY - 100;}@Overridepublic boolean onTouchEvent(MotionEvent event) {// 根据触摸位置更新控制点,并提示重绘if (mode) {control1.x = event.getX();control1.y = event.getY();} else {control2.x = event.getX();control2.y = event.getY();}invalidate();return true;}@Overrideprotected void onDraw(Canvas canvas) {super.onDraw(canvas);//drawCoordinateSystem(canvas);// 绘制数据点和控制点mPaint.setColor(Color.GRAY);mPaint.setStrokeWidth(20);canvas.drawPoint(start.x, start.y, mPaint);canvas.drawPoint(end.x, end.y, mPaint);canvas.drawPoint(control1.x, control1.y, mPaint);canvas.drawPoint(control2.x, control2.y, mPaint);// 绘制辅助线mPaint.setStrokeWidth(4);canvas.drawLine(start.x, start.y, control1.x, control1.y, mPaint);canvas.drawLine(control1.x, control1.y,control2.x, control2.y, mPaint);canvas.drawLine(control2.x, control2.y,end.x, end.y, mPaint);// 绘制贝塞尔曲线mPaint.setColor(Color.RED);mPaint.setStrokeWidth(8);Path path = new Path();path.moveTo(start.x, start.y);path.cubicTo(control1.x, control1.y, control2.x,control2.y, end.x, end.y);canvas.drawPath(path, mPaint);}
}

三阶曲线相比于二阶曲线可以制作更加复杂的形状,但是对于高阶的曲线,用低阶的曲线组合也可达到相同的效果,就是传说中的降阶。因此我们对贝塞尔曲线的封装方法一般最高只到三阶曲线。

贝塞尔曲线使用实例

首先要明确一个内容,就是在什么情况下需要使用贝塞尔曲线?可能会有如下几个方面:

序号内容用例
1事先不知道曲线状态,需要实时计算时天气预报气温变化的平滑折线图
2显示状态会根据用户操作改变时QQ小红点,仿真翻书效果
3一些比较复杂的运动状态(配合PathMeasure使用)复杂运动状态的动画效果

贝塞尔曲线的主要优点是可以实时控制曲线状态,并可以通过改变控制点的状态实时让曲线进行平滑的状态变化。

在这里插入图片描述
思路分析:
我们最终的需要的效果是将一个圆转变成一个心形,通过分析可知,圆可以由四段三阶贝塞尔曲线组合而成,如下:
在这里插入图片描述心形也可以由四段的三阶的贝塞尔曲线组成,如下:
在这里插入图片描述两者的差别仅仅在于数据点和控制点位置不同,因此只需要调整数据点和控制点的位置,就能将圆形变为心形。

核心难点:
1.如何得到数据点和控制点的位置?
关于使用绘制圆形的数据点与控制点早就已经有人详细的计算好了,可以参考https://stackoverflow.com/questions/1734745/how-to-create-circle-with-b%C3%A9zier-curves

而对于心形的数据点和控制点,可以由圆形的部分数据点和控制点平移后得到

2.如何达到渐变效果?
渐变其实就是每次对数据点和控制点稍微移动一点,然后重绘界面,在短时间多次的调整数据点与控制点,使其逐渐接近目标值,通过不断的重绘界面达到一种渐变的效果。
在这里插入图片描述`public class Bezier3 extends View {
private static final float C = 0.551915024494f; // 一个常量,用来计算绘制圆形贝塞尔曲线控制点的位置

private Paint mPaint;
private int mCenterX, mCenterY;private PointF mCenter = new PointF(0,0);
private float mCircleRadius = 200;                  // 圆的半径
private float mDifference = mCircleRadius*C;        // 圆形的控制点与数据点的差值private float[] mData = new float[8];               // 顺时针记录绘制圆形的四个数据点
private float[] mCtrl = new float[16];              // 顺时针记录绘制圆形的八个控制点private float mDuration = 1000;                     // 变化总时长
private float mCurrent = 0;                         // 当前已进行时长
private float mCount = 100;                         // 将时长总共划分多少份
private float mPiece = mDuration/mCount;            // 每一份的时长public Bezier3(Context context) {this(context, null);}public Bezier3(Context context, AttributeSet attrs) {super(context, attrs);mPaint = new Paint();mPaint.setColor(Color.BLACK);mPaint.setStrokeWidth(8);mPaint.setStyle(Paint.Style.STROKE);mPaint.setTextSize(60);// 初始化数据点mData[0] = 0;mData[1] = mCircleRadius;mData[2] = mCircleRadius;mData[3] = 0;mData[4] = 0;mData[5] = -mCircleRadius;mData[6] = -mCircleRadius;mData[7] = 0;// 初始化控制点mCtrl[0]  = mData[0]+mDifference;mCtrl[1]  = mData[1];mCtrl[2]  = mData[2];mCtrl[3]  = mData[3]+mDifference;mCtrl[4]  = mData[2];mCtrl[5]  = mData[3]-mDifference;mCtrl[6]  = mData[4]+mDifference;mCtrl[7]  = mData[5];mCtrl[8]  = mData[4]-mDifference;mCtrl[9]  = mData[5];mCtrl[10] = mData[6];mCtrl[11] = mData[7]-mDifference;mCtrl[12] = mData[6];mCtrl[13] = mData[7]+mDifference;mCtrl[14] = mData[0]-mDifference;mCtrl[15] = mData[1];
}@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {super.onSizeChanged(w, h, oldw, oldh);mCenterX = w / 2;mCenterY = h / 2;
}@Override
protected void onDraw(Canvas canvas) {super.onDraw(canvas);drawCoordinateSystem(canvas);       // 绘制坐标系canvas.translate(mCenterX, mCenterY); // 将坐标系移动到画布中央canvas.scale(1,-1);                 // 翻转Y轴drawAuxiliaryLine(canvas);// 绘制贝塞尔曲线mPaint.setColor(Color.RED);mPaint.setStrokeWidth(8);Path path = new Path();path.moveTo(mData[0],mData[1]);path.cubicTo(mCtrl[0],  mCtrl[1],  mCtrl[2],  mCtrl[3],     mData[2], mData[3]);path.cubicTo(mCtrl[4],  mCtrl[5],  mCtrl[6],  mCtrl[7],     mData[4], mData[5]);path.cubicTo(mCtrl[8],  mCtrl[9],  mCtrl[10], mCtrl[11],    mData[6], mData[7]);path.cubicTo(mCtrl[12], mCtrl[13], mCtrl[14], mCtrl[15],    mData[0], mData[1]);canvas.drawPath(path, mPaint);mCurrent += mPiece;if (mCurrent < mDuration){mData[1] -= 120/mCount;mCtrl[7] += 80/mCount;mCtrl[9] += 80/mCount;mCtrl[4] -= 20/mCount;mCtrl[10] += 20/mCount;postInvalidateDelayed((long) mPiece);}
}// 绘制辅助线
private void drawAuxiliaryLine(Canvas canvas) {// 绘制数据点和控制点mPaint.setColor(Color.GRAY);mPaint.setStrokeWidth(20);for (int i=0; i<8; i+=2){canvas.drawPoint(mData[i],mData[i+1], mPaint);}for (int i=0; i<16; i+=2){canvas.drawPoint(mCtrl[i], mCtrl[i+1], mPaint);}// 绘制辅助线mPaint.setStrokeWidth(4);for (int i=2, j=2; i<8; i+=2, j+=4){canvas.drawLine(mData[i],mData[i+1],mCtrl[j],mCtrl[j+1],mPaint);canvas.drawLine(mData[i],mData[i+1],mCtrl[j+2],mCtrl[j+3],mPaint);}canvas.drawLine(mData[0],mData[1],mCtrl[0],mCtrl[1],mPaint);canvas.drawLine(mData[0],mData[1],mCtrl[14],mCtrl[15],mPaint);
}// 绘制坐标系
private void drawCoordinateSystem(Canvas canvas) {canvas.save();                      // 绘制做坐标系canvas.translate(mCenterX, mCenterY); // 将坐标系移动到画布中央canvas.scale(1,-1);                 // 翻转Y轴Paint fuzhuPaint = new Paint();fuzhuPaint.setColor(Color.RED);fuzhuPaint.setStrokeWidth(5);fuzhuPaint.setStyle(Paint.Style.STROKE);canvas.drawLine(0, -2000, 0, 2000, fuzhuPaint);canvas.drawLine(-2000, 0, 2000, 0, fuzhuPaint);canvas.restore();
}

}
`

这篇关于Android自定义系列——8.Path之贝塞尔曲线的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1089262

相关文章

ROS话题通信流程自定义数据格式

ROS话题通信流程自定义数据格式 需求流程实现步骤定义msg文件编辑配置文件编译 在 ROS 通信协议中,数据载体是一个较为重要组成部分,ROS 中通过 std_msgs 封装了一些原生的数据类型,比如:String、Int32、Int64、Char、Bool、Empty… 但是,这些数据一般只包含一个 data 字段,结构的单一意味着功能上的局限性,当传输一些复杂的数据,比如:

Eclipse+ADT与Android Studio开发的区别

下文的EA指Eclipse+ADT,AS就是指Android Studio。 就编写界面布局来说AS可以边开发边预览(所见即所得,以及多个屏幕预览),这个优势比较大。AS运行时占的内存比EA的要小。AS创建项目时要创建gradle项目框架,so,创建项目时AS比较慢。android studio基于gradle构建项目,你无法同时集中管理和维护多个项目的源码,而eclipse ADT可以同时打开

android 免费短信验证功能

没有太复杂的使用的话,功能实现比较简单粗暴。 在www.mob.com网站中可以申请使用免费短信验证功能。 步骤: 1.注册登录。 2.选择“短信验证码SDK” 3.下载对应的sdk包,我这是选studio的。 4.从头像那进入后台并创建短信验证应用,获取到key跟secret 5.根据技术文档操作(initSDK方法写在setContentView上面) 6.关键:在有用到的Mo

android一键分享功能部分实现

为什么叫做部分实现呢,其实是我只实现一部分的分享。如新浪微博,那还有没去实现的是微信分享。还有一部分奇怪的问题:我QQ分享跟QQ空间的分享功能,我都没配置key那些都是原本集成就有的key也可以实现分享,谁清楚的麻烦详解下。 实现分享功能我们可以去www.mob.com这个网站集成。免费的,而且还有短信验证功能。等这分享研究完后就研究下短信验证功能。 开始实现步骤(新浪分享,以下是本人自己实现

Android我的二维码扫描功能发展史(完整)

最近在研究下二维码扫描功能,跟据从网上查阅的资料到自己勉强已实现扫描功能来一一介绍我的二维码扫描功能实现的发展历程: 首页通过网络搜索发现做android二维码扫描功能看去都是基于google的ZXing项目开发。 2、搜索怎么使用ZXing实现自己的二维码扫描:从网上下载ZXing-2.2.zip以及core-2.2-source.jar文件,分别解压两个文件。然后把.jar解压出来的整个c

android 带与不带logo的二维码生成

该代码基于ZXing项目,这个网上能下载得到。 定义的控件以及属性: public static final int SCAN_CODE = 1;private ImageView iv;private EditText et;private Button qr_btn,add_logo;private Bitmap logo,bitmap,bmp; //logo图标private st

Android多线程下载见解

通过for循环开启N个线程,这是多线程,但每次循环都new一个线程肯定很耗内存的。那可以改用线程池来。 就以我个人对多线程下载的理解是开启一个线程后: 1.通过HttpUrlConnection对象获取要下载文件的总长度 2.通过RandomAccessFile流对象在本地创建一个跟远程文件长度一样大小的空文件。 3.通过文件总长度/线程个数=得到每个线程大概要下载的量(线程块大小)。

时间服务器中,适用于国内的 NTP 服务器地址,可用于时间同步或 Android 加速 GPS 定位

NTP 是什么?   NTP 是网络时间协议(Network Time Protocol),它用来同步网络设备【如计算机、手机】的时间的协议。 NTP 实现什么目的?   目的很简单,就是为了提供准确时间。因为我们的手表、设备等,经常会时间跑着跑着就有误差,或快或慢的少几秒,时间长了甚至误差过分钟。 NTP 服务器列表 最常见、熟知的就是 www.pool.ntp.org/zo

JavaWeb系列二十: jQuery的DOM操作 下

jQuery的DOM操作 CSS-DOM操作多选框案例页面加载完毕触发方法作业布置jQuery获取选中复选框的值jQuery控制checkbox被选中jQuery控制(全选/全不选/反选)jQuery动态添加删除用户 CSS-DOM操作 获取和设置元素的样式属性: css()获取和设置元素透明度: opacity属性获取和设置元素高度, 宽度: height(), widt

高仿精仿愤怒的小鸟android版游戏源码

这是一款很完美的高仿精仿愤怒的小鸟android版游戏源码,大家可以研究一下吧、 为了报复偷走鸟蛋的肥猪们,鸟儿以自己的身体为武器,仿佛炮弹一样去攻击肥猪们的堡垒。游戏是十分卡通的2D画面,看着愤怒的红色小鸟,奋不顾身的往绿色的肥猪的堡垒砸去,那种奇妙的感觉还真是令人感到很欢乐。而游戏的配乐同样充满了欢乐的感觉,轻松的节奏,欢快的风格。 源码下载