Android自定义View之点赞撒花(三阶贝塞尔曲线应用)

2024-03-06 16:30

本文主要是介绍Android自定义View之点赞撒花(三阶贝塞尔曲线应用),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

本文参考辉哥的贝塞尔曲线 - 花束直播点赞效果,是对三阶贝塞尔曲线和对属性动画的运用,十分适合学习自定义View。

最终效果

点赞撒花

实现思路

  • 刚开始,爱心位于整个View的最底部中间位置,我们可以继承RelativeLayout并在其底部中间位置添加ImageView,设置ImageView为爱心图片即可;
  • 爱心生成时伴随着透明度和放大动画,这个比较简单,通过属性动画实现即可;
  • 然后爱心上升的运动轨迹整体是一个三阶的贝塞尔曲线,对于三阶贝塞尔曲线我们知道存在起始位置p0终止位置p3以及两个控制点p1以及p2,如下图:
    三阶贝塞尔曲线
    我们只要计算出p0p1p2p3的坐标即可绘制出爱心上升轨迹;各个点坐标如图上,其中layoutWidthlayoutHeight为整体布局的宽高,ivWidthivHeight为爱心背景图片的宽高;
    1. 对于p0点,取爱心背景对应的左上角坐标,横坐标为layoutWidth / 2 - ivWidth / 2,纵坐标为layoutHeight - ivHeight;
    2. 对于p1点,横坐标为可以取layoutWidth任意值 - ivWidth,纵坐标对应区间应该为【layoutHeight/2,layoutHeight】;
    3. 对于p2点,横坐标为可以取layoutWidth任意值 - ivWidth,纵坐标对应区间应该为【0,layoutHeight/2】;
    4. 对于p3点,横坐标为可以取layoutWidth任意值 - ivWidth,纵坐标对应区间应该为0;
  • 当我们计算出爱心上升轨迹后,不断的去更新爱心的x,y坐标,同时伴随着透明度的变化,当动画执行结束,移除此爱心,自此,整个效果便可以实现。

相关源码

自定义鲜花点赞效果FlowersLayout

package com.crystal.view.animationimport android.animation.*
import android.content.Context
import android.graphics.PointF
import android.util.AttributeSet
import android.view.ViewGroup
import android.view.animation.*
import android.widget.ImageView
import android.widget.RelativeLayout
import androidx.appcompat.content.res.AppCompatResources
import com.crystal.view.R/*** 自定义鲜花点赞效果【三阶贝塞尔曲线使用】* on 2022/11/11*/
class FlowersLayout : RelativeLayout {/*** 资源文件*/private val imageRes = intArrayOf(R.drawable.pl_blue, R.drawable.pl_red, R.drawable.pl_yellow)/*** 差值器集合,用于贝塞尔随机设置差值器*/private val interpolator = arrayListOf<Interpolator>(AccelerateDecelerateInterpolator(),AccelerateInterpolator(), DecelerateInterpolator(), LinearInterpolator())/*** 布局宽高*/private var layoutWidth = 0private var layoutHeight = 0/*** 鲜花宽高*/private var ivWidth = 0fprivate var ivHeight = 0fprivate var random = java.util.Random()private var layoutParams: LayoutParamsconstructor(context: Context) : this(context, null)constructor(context: Context, attrs: AttributeSet?) : this(context, attrs, 0)constructor(context: Context, attrs: AttributeSet?, defStyleAttr: Int) : super(context, attrs, defStyleAttr) {val drawable = AppCompatResources.getDrawable(context, R.drawable.pl_blue)!!ivWidth = drawable.intrinsicWidth.toFloat()ivHeight = drawable.intrinsicHeight.toFloat()layoutParams =LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT)layoutParams.addRule(ALIGN_PARENT_BOTTOM)layoutParams.addRule(CENTER_HORIZONTAL)}override fun onMeasure(widthMeasureSpec: Int, heightMeasureSpec: Int) {super.onMeasure(widthMeasureSpec, heightMeasureSpec)layoutWidth = MeasureSpec.getSize(widthMeasureSpec)layoutHeight = MeasureSpec.getSize(heightMeasureSpec)}/*** 添加鲜花到布局中*/fun addFlower() {val ivFlower = ImageView(context)ivFlower.setImageResource(imageRes[random.nextInt(imageRes.size - 1)])ivFlower.layoutParams = layoutParamsaddView(ivFlower)//执行相关动画executeAnimations(ivFlower)}private fun executeAnimations(ivFlower: ImageView) {//所有动画集合val allAnimator = AnimatorSet()//刚添加进来的时候伴随着透明度和放大效果val initAnimator = AnimatorSet()val alphaAnimator = ObjectAnimator.ofFloat(ivFlower, "alpha", 0.3f, 1f)val scaleXAnimator = ObjectAnimator.ofFloat(ivFlower, "scaleX", 0.3f, 1f)val scaleYAnimator = ObjectAnimator.ofFloat(ivFlower, "scaleY", 0.3f, 1f)initAnimator.playTogether(alphaAnimator, scaleXAnimator, scaleYAnimator)initAnimator.duration = 300allAnimator.playSequentially(initAnimator, constructBezierAnimator(ivFlower))allAnimator.addListener(object : AnimatorListenerAdapter() {override fun onAnimationEnd(animation: Animator?) {//动画执行完毕,移除鲜花viewremoveView(ivFlower)}})allAnimator.start()}/*** 构造三阶贝塞尔曲线动画*/private fun constructBezierAnimator(ivFlower: ImageView): Animator {//P0点为起始点,坐标应为(width/2 - iv.width/2,height - iv.height)val p0 = PointF(layoutWidth / 2 - ivWidth / 2,layoutHeight - ivHeight)//P1点为控制点 x坐标在屏幕范围内即可,y坐标范围应该在【height/2 ~ height】之间,这里我们均选随机数val p1 = PointF(random.nextInt(layoutWidth) - ivWidth,(random.nextInt(layoutHeight / 2) + layoutHeight / 2).toFloat())//P2点为控制点 x坐标在屏幕范围内即可,y坐标范围应该在【 0 ~ height/2 】之间,这里我们均选随机数val p2 = PointF(random.nextInt(layoutWidth) - ivWidth,random.nextInt(layoutHeight / 2).toFloat())//P3点为终点,x坐标在屏幕范围内,y坐标应该为0点val p3 = PointF(random.nextInt(layoutWidth) - ivWidth, 0f)val typeEvaluator = FlowersTypeEvaluator(p1, p2)val bezierAnimator = ObjectAnimator.ofObject(typeEvaluator, p0, p3)//随机选取差值器,效果更佳bezierAnimator.interpolator = interpolator[random.nextInt(interpolator.size - 1)]bezierAnimator.duration = 3000bezierAnimator.addUpdateListener {val point = it.animatedValue as PointF//设置三阶贝塞尔曲线获取的数据,不断移动鲜花的位置ivFlower.x = point.xivFlower.y = point.y//改变鲜花的透明度ivFlower.alpha = (1 - it.animatedFraction + 0.2f)}return bezierAnimator}private class FlowersTypeEvaluator(val p1: PointF, val p2: PointF) : TypeEvaluator<PointF> {override fun evaluate(t: Float, p0: PointF, p3: PointF): PointF {//三阶贝塞尔曲线公式:B(t) = P0 * (1-t)^3 + 3 * P1 * t * (1-t)^2 + 3 * P2 * t^2 * (1-t) + P3 * t^3, t ∈ [0,1]val point = PointF()point.x =p0.x * (1 - t) * (1 - t) * (1 - t) + 3 * p1.x * t * (1 - t) * (1 - t) + 3 * p2.x * t * t * (1 - t) + p3.x * t * t * tpoint.y =p0.y * (1 - t) * (1 - t) * (1 - t) + 3 * p1.y * t * (1 - t) * (1 - t) + 3 * p2.y * t * t * (1 - t) + p3.y * t * t * treturn point}}
}

总结

通过实现点赞撒花效果,了解了Android中三阶贝塞尔的使用方式,同时对属性动画的使用有了进一步的认知。

结语

如果以上文章对您有一点点帮助,希望您不要吝啬的点个赞加个关注,您每一次小小的举动都是我坚持写作的不懈动力!ღ( ´・ᴗ・` )

这篇关于Android自定义View之点赞撒花(三阶贝塞尔曲线应用)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

将Python应用部署到生产环境的小技巧分享

《将Python应用部署到生产环境的小技巧分享》文章主要讲述了在将Python应用程序部署到生产环境之前,需要进行的准备工作和最佳实践,包括心态调整、代码审查、测试覆盖率提升、配置文件优化、日志记录完... 目录部署前夜:从开发到生产的心理准备与检查清单环境搭建:打造稳固的应用运行平台自动化流水线:让部署像

Linux中Curl参数详解实践应用

《Linux中Curl参数详解实践应用》在现代网络开发和运维工作中,curl命令是一个不可或缺的工具,它是一个利用URL语法在命令行下工作的文件传输工具,支持多种协议,如HTTP、HTTPS、FTP等... 目录引言一、基础请求参数1. -X 或 --request2. -d 或 --data3. -H 或

在Ubuntu上部署SpringBoot应用的操作步骤

《在Ubuntu上部署SpringBoot应用的操作步骤》随着云计算和容器化技术的普及,Linux服务器已成为部署Web应用程序的主流平台之一,Java作为一种跨平台的编程语言,具有广泛的应用场景,本... 目录一、部署准备二、安装 Java 环境1. 安装 JDK2. 验证 Java 安装三、安装 mys

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链

Node.js 中 http 模块的深度剖析与实战应用小结

《Node.js中http模块的深度剖析与实战应用小结》本文详细介绍了Node.js中的http模块,从创建HTTP服务器、处理请求与响应,到获取请求参数,每个环节都通过代码示例进行解析,旨在帮... 目录Node.js 中 http 模块的深度剖析与实战应用一、引言二、创建 HTTP 服务器:基石搭建(一

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

java中VO PO DTO POJO BO DO对象的应用场景及使用方式

《java中VOPODTOPOJOBODO对象的应用场景及使用方式》文章介绍了Java开发中常用的几种对象类型及其应用场景,包括VO、PO、DTO、POJO、BO和DO等,并通过示例说明了它... 目录Java中VO PO DTO POJO BO DO对象的应用VO (View Object) - 视图对象

Go信号处理如何优雅地关闭你的应用

《Go信号处理如何优雅地关闭你的应用》Go中的优雅关闭机制使得在应用程序接收到终止信号时,能够进行平滑的资源清理,通过使用context来管理goroutine的生命周期,结合signal... 目录1. 什么是信号处理?2. 如何优雅地关闭 Go 应用?3. 代码实现3.1 基本的信号捕获和优雅关闭3.2

SpringBoot 自定义消息转换器使用详解

《SpringBoot自定义消息转换器使用详解》本文详细介绍了SpringBoot消息转换器的知识,并通过案例操作演示了如何进行自定义消息转换器的定制开发和使用,感兴趣的朋友一起看看吧... 目录一、前言二、SpringBoot 内容协商介绍2.1 什么是内容协商2.2 内容协商机制深入理解2.2.1 内容

正则表达式高级应用与性能优化记录

《正则表达式高级应用与性能优化记录》本文介绍了正则表达式的高级应用和性能优化技巧,包括文本拆分、合并、XML/HTML解析、数据分析、以及性能优化方法,通过这些技巧,可以更高效地利用正则表达式进行复杂... 目录第6章:正则表达式的高级应用6.1 模式匹配与文本处理6.1.1 文本拆分6.1.2 文本合并6