实战演练!二阶贝塞尔仿微信扔炸弹动画

抛物线在这里是通过二阶贝塞尔曲线来完成,所以先来了解下什么是二阶贝塞尔曲线,从下图中可以发现,二阶贝塞尔曲线有三个关键点,我们可以称作起点坐标、终点坐标,还有控制点。

录屏_选择区域_20210615170032.gif

起点和终点坐标好理解,控制点可以理解成开始下降的转折点,而古老的数学大神早就提供好了公式,我们只需要向这个公式提供这几个参数即可得到x、y,当然还有个参数是时间,有了时间控制,我们可以在指定秒内把他平滑的绘制完成。

公式如下:

x = (1 - t)^2 * 0 + 2 t (1 - t) * 1 + t^2 * 1 = 2 t (1 - t) + t^2

y= (1 - t)^2 * 1 + 2 t (1 - t) * 1 + t^2 * 0 = (1 - t)^2 + 2 t (1 - t)

自定义二阶贝塞尔曲线计算器

提到动画,首先可能会想到ObjectAnimator类,没错,抛物线也是通过ObjectAnimator来完成的,只不过我们需要自定义一个TypeEvaluator,用来提供二阶贝塞尔曲线的x和y。

TypeEvaluator只有一个方法,定义如下:

public abstract T evaluate (float fraction,

T startValue,

T endValue)

复制代码

fraction表示开始值和结束值之间的比例,startValue、endValue分别是开始值和结束值,这个比例也可以当作是时间,可能官方一点叫比例,他会自动计算,值的范围是0-1,比如取值0.5的时候就是动画完成了一半,1的时候动画完成。

所以套入二阶贝塞尔曲线公式得到如下代码:

class PointFTypeEvaluator(var control: PointF) : TypeEvaluator {

override fun evaluate(fraction: Float, startValue: PointF, endValue: PointF): PointF {

return getPointF(startValue, endValue, control, fraction)

}

private fun getPointF(start: PointF, end: PointF, control: PointF, t: Float): PointF {

val pointF = PointF()

pointF.x = (1 - t) * (1 - t) * start.x + 2 * t * (1 - t) * control.x + t * t * end.x

pointF.y = (1 - t) * (1 - t) * start.y + 2 * t * (1 - t) * control.y + t * t * end.y

return pointF

}

}

复制代码

播放动画

然后使用ObjectAnimator进行播放。

val animator = ObjectAnimator.ofObject(activityMainBinding.boom, “mPointF”,

PointFTypeEvaluator(controlP), startP, endP)

复制代码

注意的是这个View需要有point方法,参数是PointF,方法内主要完成x和y的设置。

public void setPoint(PointF pointF) {

setX(pointF.x);

setY(pointF.y);

}

复制代码

当然微信炸弹落地的位置是随机的,我们也加个随机。

class MainActivity : AppCompatActivity() {

lateinit var binding: ActivityMainBinding;

private fun getRandom(max: Int, min: Int): Int {

val random = java.util.Random()

return random.nextInt(max - min + 1) + min

}

private fun getRandomPointF():PointF{

val outMetrics = DisplayMetrics()

val offset = 100

windowManager.defaultDisplay.getMetrics(outMetrics)

val width = outMetrics.widthPixels

val height = outMetrics.heightPixels

return PointF(getRandom(width / 2 + offset, width / 2 - offset).toFloat(), getRandom(height / 2 + offset, height / 2 - offset).toFloat())

}

override fun onCreate(savedInstanceState: Bundle?) {

super.onCreate(savedInstanceState)

binding = DataBindingUtil.setContentView(this, R.layout.activity_main);

binding.button.setOnClickListener {

binding!!.boom.visibility = View.VISIBLE

val startP = PointF()

val endP = PointF()

val controlP = PointF()

val randomPointF = getRandomPointF()

startP.x = 916f

startP.y = 1353f

endP.x = randomPointF.x

endP.y = randomPointF.y

controlP.x = randomPointF.x + getRandom(200, 50)

controlP.y = randomPointF.y - getRandom(200, 50)

val animator = ObjectAnimator.ofObject(binding.boom, “point”,

PointFTypeEvaluator(controlP), startP, endP)

animator.start()

}

}

}

复制代码

<?xml version="1.0" encoding="utf-8"?>

<layout xmlns:android=“http://schemas.android.com/apk/res/android”

xmlns:app=“http://schemas.android.com/apk/res-auto”

xmlns:tools=“http://schemas.android.com/tools”>

<RelativeLayout

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:orientation=“vertical”

tools:context=“.MainActivity”>

<com.airbnb.lottie.LottieAnimationView

android:visibility=“gone”

android:id=“@+id/lottie”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

app:lottie_fileName=“boom.json”></com.airbnb.lottie.LottieAnimationView>

<Button

android:id=“@+id/button”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:text=“开始”>

<com.example.kotlindemo.widget.MyImageView

android:id=“@+id/boom”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:background=“@drawable/ic_boom”

android:visibility=“gone”></com.example.kotlindemo.widget.MyImageView>

复制代码

效果如下:

录屏_选择区域_20210615174149.gif

爆炸效果

爆炸效果是使用的动画,用的lottie框架,这里提供爆炸文件的下载地址。

https://lottiefiles.com/download/public/9990-explosion

复制代码

有了结束的坐标点,只需要吧LottieAnimationView移动到对应位置进行播放即可,播放后隐藏,完整代码如下:

package com.example.kotlindemo

import android.animation.Animator

import android.animation.ObjectAnimator

import android.content.Context

import android.content.Intent

import android.graphics.BitmapFactory

import android.graphics.PointF

import android.media.projection.MediaProjectionManager

import android.os.Build

import android.os.Bundle

import android.util.DisplayMetrics

import android.view.Menu

import android.view.MenuItem

import android.view.View

import android.widget.Toast

import androidx.appcompat.app.AppCompatActivity

import androidx.databinding.DataBindingUtil

import com.example.kotlindemo.databinding.ActivityMainBinding

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
mg-l9e3jEIk-1715897768044)]

[外链图片转存中…(img-SRWX1rts-1715897768045)]

[外链图片转存中…(img-gHDUnQMG-1715897768046)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

  • 16
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值