贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)
贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)
2024-06-27 04:19:29  作者:绿亦歌  网址:https://m.xinb2b.cn/life/plc248741.html

贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(1)

贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(2)

作者 | 威威喵

责编 | 屠敏

出品 | CSDN 博客

直接步入正题,我们要实现的是一个 Android 贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(3)客户端应用里面的一种点赞效果,比如你点一下那个爱心型的图片,就会产生一个小爱心,而且会以曲线的方式进行上升,直到它消失为止。

文字描述只能是这样的了,我们直接来看动态图吧,效果更直观。

贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(4)

本案例是由我自己写的,因为之前对这个贝塞尔曲线有一点点了解,还有无意间看到了这个效果,觉得挺赞的,就顺便写了一下demo,并且学习了一些关于贝塞尔曲线的相关知识。

首先,要看懂本案例的代码,你需要具备 Android 自定义 View 的基本知识,并且你还有了解一些关于贝塞尔曲线的公式和算法。不过没关系,我们并不需要对贝塞尔深刻了解,只要会基本的根据公式,套用代码就好了。

来看一下贝塞尔曲线的一些相关知识,我也是从大佬的博客中学习得来的。我们来看看什么是贝塞尔曲线?

贝塞尔曲线(Bézier curve),又称贝兹曲线或贝济埃曲线,是应用于二维图形应用程序的数学曲线。一般的矢量图形软件通过它来精确画出曲线,贝兹曲线由线段与节点组成,节点是可拖动的支点,线段像可伸缩的皮筋,我们在绘图工具上看到的钢笔工具就是来做这种矢量曲线的。

更形象的就直接来看动态图吧。

一阶贝塞尔曲线公式:由 P0 至 P1 的连续点, 描述的一条线段

贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(5)

贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(6)

二阶贝塞尔曲线公式:曲线的切线 P0-P1、P1-P2 组成的运动轨迹

贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(7)

贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(8)

三阶贝塞尔曲线公式:

贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(9)

贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(10)

从上面的动态图,可以很直观的看到曲线的计算公式和它的路径形成的规律。而我们要实现的效果,运用的就是三阶贝塞尔曲线的公式。首先,需要确定曲线的路径的话,就必须先确定它的点位置。我以是这样的方式来确定点位置的,如下图:

贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(11)

我使用的就是这三个点,两边都可以,随机的选择一边。这样的话,我们的曲线就在屏幕内,它的形成大致和我们上面的动态图有点类似。那么看代码:

private Point setPoint1 {

Point points = new Point{

new Point(mLoveX, mLoveY),

new Point(0, mcanvasHeight / 2),

new Point(mCanvasWidth 20, -mLoveWidth - 10),

};

return points;

}

private Point setPoint2 {

Point points = new Point{

new Point(mLoveX, mLoveY),

new Point(mCanvasWidth, mCanvasHeight / 2),

new Point(-mLoveWidth - 20, -mLoveWidth - 10),

};

return points;

}

上面代码是初始化两种点的坐标,mLoveX,mLoveY 表示我们的爱心起始的位置。第一个集合点,对应图中的蓝线,第二个集合点,就对应橙色了。

接下来是重点部分,也就是把贝塞尔曲线公式转化为代码的形式,根据动态图中有一个 t 值,它的区间是 [0,1] 的,这个也很形象,t 从 0 变到 1 时,意味着曲线已经绘制完了。看代码:

private Point deCasteljau(Point[] points, float k) {

final int n = points.length;

for (int i = 1; i <= n; i )

for (int j = 0; j < n - i; j ) {

points[j].x = (int) ((1 - k) * points[j].x k * points[j 1].x);

points[j].y = (int) ((1 - k) * points[j].y k * points[j 1].y);

}

return points[0];

}

刚刚我们定义的两种点的集合,就可以将它传入了,这样根据 k 值的变化,就可以得到对应位置曲线上的点坐标。接下来,我们的任务就是开启一个子线程去跟新 k 值,将 k 值有 0 加到 1,然后返回的每个 point 对象,就是整条曲线的坐标散点。执行子线程获取点的代码:

mLoveThread = new Thread(new Runnable {

@Override

public void run {

while (k < 1) {

k = 0.01;

Point point = deCasteljau(mPoints, k);

mLoveX = point.x;

mLoveY = point.y;

if (mLoveY <= -mLoveWidth || mLoveY >= mCanvasHeight) {

k = 1;

}

if (mLoveX <= -mLoveWidth || mLoveX >= mCanvasWidth) {

k = 1;

}

postInvalidate;//异步刷新

try {

Thread.sleep(80);

} catch (InterruptedException e) {

e.printStackTrace;

}

}

}

});

通过上面代码,我们就可以获取爱心图片的 x,y 坐标值了,然后再通过 onDraw 里面将它进行绘制就搞定啦。

@Override

protected void onDraw(Canvas canvas) {

super.onDraw(canvas);

mCanvasWidth = canvas.getWidth;

mCanvasHeight = canvas.getHeight;

mLoveBitmapX = mCanvasWidth / 2 - mLoveBitmapWidth / 2;

mLoveBitmapY = mCanvasHeight - 2 * mLoveBitmapHeight;

drawLoveBitmap(canvas);

canvas.drawBitmap(mDefLove, mLoveX, mLoveY, mPaint);

//随便画的

canvas.drawText("点赞", mCanvasWidth / 2 - mPaint.getTextSize, mLoveBitmapY mLoveBitmapHeight 100, mPaint);

canvas.drawLine(0, mLoveBitmapY mLoveBitmapHeight 20, mCanvasWidth, mLoveBitmapY mLoveBitmapHeight 20, mPaint);

}

这里的爱心,我使用的是六张不同的图片,我之前想尝试使用爱心函数公式来绘制的,不过也放弃了,计算太慢了,每个爱心算出来都要停顿一下,只好换图片的形式。

贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(12)

最后提一下就是点击这个图片才绘制的功能,我是在 onTouchEvent 中拿到点击的坐标位置,然后去判断它的点击位置是不是在那个爱心图片里面,代码如下:

private boolean isTouchLoveArea(int touchX, int touchY) {

return touchX >= mLoveBitmapX && touchX <= mLoveBitmapX mLoveBitmapWidth

&& touchY > mLoveBitmapY && touchY <= mLoveBitmapY mLoveBitmapHeight;

}

好了,最后也没什么好介绍的了,剩下的基本都是自定义 View 的知识,我们主要是贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(13)关注这个贝塞尔曲线是如何绘制的就好,那么完整代码如下:

package com.example.xww.my贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(14)application;

import android.content.Context;

import android.graphics.Bitmap;

import android.graphics.BitmapFactory;

import android.graphics.Canvas;

import android.graphics.Paint;

import android.graphics.Point;

import android.os.Build;

import android.support.annotation.able;

import android.support.annotation.RequiresApi;

import android.util.AttributeSet;

import android.view.MotionEvent;

import android.view.View;

import java.util.Random;

import java.util.concurrent.ExecutorService;

import java.util.concurrent.Executors;

@RequiresApi(api = Build.VERSION_CODES.N)

public class LoveView extends View {

private Paint mPaint;

//爱心图片

private Bitmap mLoveBitmap;

private Bitmap mLove1;

private Bitmap mLove2;

private Bitmap mLove3;

private Bitmap mLove4;

private Bitmap mLove5;

private Bitmap mLove6;

private Bitmap mDefLove;

private int mLoveWidth;

private int mLoveX;

private int mLoveY;

//图片绘制的 x,y 坐标

private int mLoveBitmapX;

private int mLoveBitmapY;

//图片的宽、高

private int mLoveBitmapWidth;

private int mLoveBitmapHeight;

// 画布宽、高

private int mCanvasWidth;

private int mCanvasHeight;

//触摸点

private int mTouchX;

private int mTouchY;

private ExecutorService mExecutorService;

private Thread mLoveThread;

//随机数

private Random mRandom;

private float k;//曲线斜率 k:[0,1]

private Point mPoints;//构成曲线随机点集合

@Override

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

setMeasuredDimension(measureSpecWidth(widthMeasureSpec), measureSpecHeigth(heightMeasureSpec));

}

private int measureSpecWidth(int widthMeasureSpec) {

int mode = MeasureSpec.getMode(widthMeasureSpec);

int size = MeasureSpec.getSize(widthMeasureSpec);

return mode == MeasureSpec.EXACTLY ? size : Math.min(200, size);

}

private int measureSpecHeigth(int heightMeasureSpec) {

int mode = MeasureSpec.getMode(heightMeasureSpec);

int size = MeasureSpec.getSize(heightMeasureSpec);

return mode == MeasureSpec.EXACTLY ? size : Math.min(200, size);

}

private void init {

initPaint;

initBitmap;

mRandom = new Random;

mExecutorService = Executors.newWorkStealingPool(6);

}

private void initBitmap {

mLoveBitmap = BitmapFactory.decodeResource(getResources, R.drawable.loveclick);

mLoveBitmap = Bitmap.createScaledBitmap(mLoveBitmap, 180, 180, false);

mLoveBitmapWidth = mLoveBitmap.getWidth;

mLoveBitmapHeight = mLoveBitmap.getHeight;

mLove1 = BitmapFactory.decodeResource(getResources, R.drawable.love1);

mLove2 = BitmapFactory.decodeResource(getResources, R.drawable.love2);

mLove3 = BitmapFactory.decodeResource(getResources, R.drawable.love3);

mLove4 = BitmapFactory.decodeResource(getResources, R.drawable.love4);

mLove5 = BitmapFactory.decodeResource(getResources, R.drawable.love5);

mLove6 = BitmapFactory.decodeResource(getResources, R.drawable.love6);

mLove1 = reSizeLove(mLove1);

mLove2 = reSizeLove(mLove2);

mLove3 = reSizeLove(mLove3);

mLove4 = reSizeLove(mLove4);

mLove5 = reSizeLove(mLove5);

mLove6 = reSizeLove(mLove6);

mDefLove = mLove1;

mLoveWidth = mLove1.getWidth;

setDefPosition;

}

private Bitmap reSizeLove(Bitmap src) {

return Bitmap.createScaledBitmap(src, 160, 160, false);

}

private void initPaint {

mPaint = new Paint;

mPaint.setColor(getResources.getColor(android.R.color.holo_purple));

mPaint.setStrokeWidth(8f);

mPaint.setStyle(Paint.Style.FILL);

mPaint.setDither(true);

mPaint.setAntiAlias(true);

mPaint.setTextSize(45f);

}

public LoveView(Context context) {

this(context, );

}

public LoveView(Context context, @able AttributeSet attrs) {

this(context, attrs, 0);

}

public LoveView(Context context, @able AttributeSet attrs, int defStyleAttr) {

super(context, attrs, defStyleAttr);

init;

}

@Override

protected void onDraw(Canvas canvas) {

super.onDraw(canvas);

mCanvasWidth = canvas.getWidth;

mCanvasHeight = canvas.getHeight;

mLoveBitmapX = mCanvasWidth / 2 - mLoveBitmapWidth / 2;

mLoveBitmapY = mCanvasHeight - 2 * mLoveBitmapHeight;

drawLoveBitmap(canvas);

canvas.drawBitmap(mDefLove, mLoveX, mLoveY, mPaint);

//随便画的

canvas.drawText("点赞", mCanvasWidth / 2 - mPaint.getTextSize, mLoveBitmapY mLoveBitmapHeight 100, mPaint);

canvas.drawLine(0, mLoveBitmapY mLoveBitmapHeight 20, mCanvasWidth, mLoveBitmapY mLoveBitmapHeight 20, mPaint);

}

private Point setPoint1 {

Point points = new Point{

new Point(mLoveX, mLoveY),

new Point(0, mCanvasHeight / 2),

new Point(mCanvasWidth 20, -mLoveWidth - 10),

};

return points;

}

private Point setPoint2 {

Point points = new Point{

new Point(mLoveX, mLoveY),

new Point(mCanvasWidth, mCanvasHeight / 2),

new Point(-mLoveWidth - 20, -mLoveWidth - 10),

};

return points;

}

private void setDefPosition {

mLoveX = mCanvasWidth / 2 - mLoveWidth / 2;

mLoveY = mLoveBitmapY - 80;

}

private void drawDynamicLove {

setDefPosition;

//设置爱心的样式和位置

int color = mRandom.nextInt(6) 1;

mDefLove = getBitmap(color);

k = 0;//开始

//添加贝塞尔路径的点

if (mRandom.nextInt(2) == 0) {

mPoints = setPoint1;

} else {

mPoints = setPoint2;

}

mLoveThread = new Thread(new Runnable {

@Override

public void run {

while (k < 1) {

k = 0.01;

Point point = deCasteljau(mPoints, k);

mLoveX = point.x;

mLoveY = point.y;

if (mLoveY <= -mLoveWidth || mLoveY >= mCanvasHeight) {

k = 1;

}

if (mLoveX <= -mLoveWidth || mLoveX >= mCanvasWidth) {

k = 1;

}

postInvalidate;//异步刷新

try {

Thread.sleep(80);

} catch (InterruptedException e) {

e.printStackTrace;

}

}

}

});

mExecutorService.execute(mLoveThread);

}

private Bitmap getBitmap(int color) {

switch (color) {

case 1:

return mLove1;

case 2:

return mLove2;

case 3:

return mLove3;

case 4:

return mLove4;

case 5:

return mLove5;

case 6:

return mLove6;

}

return ;

}

private void drawLoveBitmap(Canvas canvas) {

canvas.drawBitmap(mLoveBitmap, mLoveBitmapX, mLoveBitmapY, mPaint);

}

private Point deCasteljau(Point[] points, float k) {

final int n = points.length;

for (int i = 1; i <= n; i )

for (int j = 0; j < n - i; j ) {

points[j].x = (int) ((1 - k) * points[j].x k * points[j 1].x);

points[j].y = (int) ((1 - k) * points[j].y k * points[j 1].y);

}

return points[0];

}

@Override

public boolean onTouchEvent(MotionEvent event) {

mTouchX = (int) event.getX;

mTouchY = (int) event.getY;

switch (event.getAction) {

case MotionEvent.ACTION_DOWN:

if (isTouchLoveArea(mTouchX, mTouchY)) {

drawDynamicLove;

}

break;

case MotionEvent.ACTION_UP:

break;

}

return super.onTouchEvent(event);

}

private boolean isTouchLoveArea(int touchX, int touchY) {

return touchX >= mLoveBitmapX && touchX <= mLoveBitmapX mLoveBitmapWidth

&& touchY > mLoveBitmapY && touchY <= mLoveBitmapY mLoveBitmapHeight;

}

}

这就是整个效果的代码图了,将它放到 activity_main 里面,运行一下就可以看到效果了。

声明:本文为 CSDN 博客精选文章,版权归作者所有。作者:威威喵

原文:https://blog.csdn.net/smile_Running/article/details/98170645

【END】

贝塞尔工具画曲线的原理是什么(贝塞尔曲线之爱心点赞代码全解析)(16)

  • 胎梦代表了什么(胎梦与生男生女有关系吗)
  • 2024-06-27胎梦与生男生女有关系吗周公解梦对胎梦都有不同的解释,但是胎梦究竟与生男生女有无关系呢?梦既是一种生理现象,也是一种心理现象,做什么样的梦,与做梦者的欲望、情绪状态以及所发生的刺激性事件有关而胎儿的性别并不是由父母的欲望所决。
  • 战国红玛瑙盘玩效果:战国红玛瑙赏玩
  • 2024-06-27战国红玛瑙盘玩效果:战国红玛瑙赏玩3、关于战国红玛瑙的图案(2)战国红——灵动草花前面提到切宣化战国红玛瑙原石如同开盲盒,图案千变万化,但宣化战国红玛瑙图案最鲜明的特征就是在各种颜色上面遍布草花,随着玛瑙底润度、颜色以及草花的颜色、形。
  • 赵雅芝御装照(赵雅芝穿的7套古装嫁衣)
  • 2024-06-27赵雅芝穿的7套古装嫁衣说起赵雅芝,童鞋们印入脑海的词汇是不是就是——优雅?气质优雅又高贵的她早期还是一个出众的古装美人,演了20余部古装剧的她也尝试了各种古装扮相,其中为扮新娘,她穿了7次嫁衣,每次穿嫁衣的芝姐都什么样呢?。
  • 海南清补凉的正确做法(海南美食中的代表作)
  • 2024-06-27海南美食中的代表作一说起海南,映入脑海的是酷暑和海浪,而清补凉则是这座热带岛屿中最具有代表性的消暑美食这道在海南家喻户晓的甜品,用料非常地讲究不仅仅是用料,它名字所代表的含义也让人想在入夏时就品尝一番:清代表着清除。
  • 沈阳游记张氏帅府(沈阳故事沈阳福陵的传说)
  • 2024-06-27沈阳故事沈阳福陵的传说这篇故事广泛流传于民间,其反映的内容是清初定都盛京后,努尔哈赤在攻打明宁远城(今兴城)时,不幸被大炮击中而亡由其第八个儿子皇太极即位称汗后,为老罕王寻找安葬墓地时发生的皇家墓地规格要求很高,选址尤其重。
  • 儿童快手菜谱家常菜(孩子爱吃的快手菜)
  • 2024-06-27孩子爱吃的快手菜轻轻一点,悄悄关注↑↑↑↑↑↑五彩蔬菜丁原料瘦肉100克、玉米1条、红萝卜100克、青瓜100克、生粉1茶匙、食用油1小匙、盐做法:1、热锅下油油热后,倒入瘦肉粒快速翻炒几下,倒入红萝卜、玉米粒2、加。
  • 蜘蛛侠梅姨超燃混剪(蜘蛛侠梅婶辣出新高度)
  • 2024-06-27蜘蛛侠梅婶辣出新高度众所期待的《蜘蛛侠:英雄无归》即将上映!这次除了看“小蜘蛛”汤姆·霍兰德(TomHolland)和到底有没有“三代同堂”外,可别忘了本片的女生们,可是各个都别具特色与迷人气质,像是小蜘蛛心繫的“MJ妹。
  • 阿伟乱葬场什么梗(关于阿伟乱葬场的解释)
  • 2024-06-27关于阿伟乱葬场的解释阿伟乱葬场是形容某个作品太好了,观众美死了一大片,横七竖八地躺倒在这个作品的石榴裙下阿伟来自awsl(啊我死了)中的“aw”这个拼音首字母缩写的汉字化、拟人化,就有了阿伟这么个虚拟人物乱葬岗就是传说中。
  • 脾胃虚弱的人会有什么表现(可能会出现以下几种症状)
  • 2024-06-27可能会出现以下几种症状导语:“医生,我最近疲倦乏力、食欲不振、胸闷气短,去医院检查也没什么问题,是咋回事?”高先生今年41岁,这一段时间总是觉得腹胀不舒服,同时大便也出现溏薄的状态,去医院检查,医生说没啥问题,这让他感到十。
  • 雅塑健康减肥新方法(怎么减肥效果最好)
  • 2024-06-27怎么减肥效果最好魔鬼身材是所有MM的心愿,减肥方法也是纷繁芜杂,从而减肥的产品也就强出不穷的,怎么减肥效果最?有效的减肥产品还是需要见证~~这些减肥产品看起来小巧精致,每一种产品都宣扬可以帮助人体实现快速减肥,可真正。
  • 当年洪金宝为何发火(两人几十年的老朋友)
  • 2024-06-27两人几十年的老朋友林正英是无可争议的僵尸道长,洪金宝是影坛大哥大,两人也是多年好友,今天我们就来看看两人有多好?林正英和洪金宝同龄,九叔属于粉菊花门下,洪金宝属于于占元门下,当时两个门派还在一起合作演出过,算是从小就结。