Compose的动画Api用起来很简单,效果看起来很神奇,那么它内部到底是如何运转的呢?
使用动画的代码示例:
- var isOffset by remember { mutableStateOf(false) }
- val offsetAnimation by animateDpAsState(targetValue = if (isOffset) 100.dp else 0.dp)
- Button(
- onClick = { isOffset = !isOffset },
- modifier = Modifier.offset(0.dp, offsetAnimation)
- ) {
- Text(text = "点我进行位移")
- }
可以看到有一个Boolean类型的isOffset状态,控制着offsetAnimation动画,然后动画又控制着Button的offset,最终实现了动画效果
我们主要就看一下animateDpAsState(animate*AsState)做了什么
跟一下animateDpAsState最后会走进animateValueAsState方法中:
方法内部创建了一个Animatable动画类
然后我们跟着上图的箭头看,在targetValue发生变化后,会通过channel来发送targetValue值的变化,然后launch启动一个协程,并在其中调用了animatable的animateTo方法
接着我们就看看Animatable类是如何做动画的:
主要就是内部持有一个AnimationState类型(State)的internalState 变量,然后我们接着上面的代码跟一下animateTo方法:
没什么好说的,包了一下targetValue,接着就调用了runAnimation方法,接着往下跟(截不下分成两张图):
主要逻辑就是endState.animate()
endState就是copy的Animatable中的AnimationState对象internalState,也就是动画的初始值
然后animation就是在animateTo方法中包装了动画目标值的对象
接着跟animate方法:
这个方法主要分为两部分,第一部分就是构建了一个AnimationScope,一个数据结构,用来存放动画需要的一些信息
第二个部分就是真正动画计算和生效的地方,doAnimationFrame
首先会在第一次创建AnimationScope的时候执行一次(或再一下帧执行一次,通过callWithFrameNanos方法)
然后会判断如果动画还未执行完毕,就一直循环(while),一帧一帧执行doAnimationFrame计算动画的值
doAnimationFrame方法代码:
其中通过时间等参数计算当前动画应该设置的值,包括lastFrameTimeNanos和value等属性,然后再最后调用updateState方法去将AnimationScope的值更新到AnimationState中
updatState方法代码:
这个state对象其实也就是animateDpAsState中的Animatable动画类中的AnimationState对象internalState
所以上面其实就是Compose中动画的简化流程
由于AnimationState是一个State,在Compose中使用会自动监听其变化,只要其value变化了,就会导致相应位置重组,然后Composable就会使用新的值来展示不同的效果,比如最开始的示例代码:
- var isOffset by remember { mutableStateOf(false) }
- val offsetAnimation by animateDpAsState(targetValue = if (isOffset) 100.dp else 0.dp)
- Button(
- onClick = { isOffset = !isOffset },
- modifier = Modifier.offset(0.dp, offsetAnimation)
- ) {
- Text(text = "点我进行位移")
- }
在修改了isOffset后,animateDpAsState中的值就会因为动画的计算和修改内部state的value,导致Button的offset函数一直被重新调用,使Button不停的向下移动
其实Compose中的动画如果不考虑那么多东西的话,可以简化为如下代码:
- /**
- * creator: lt
- * effect : 自定义的动画播放器,逻辑更简单
- * warning:
- * [initialValueWithState]动画要改变的状态,起始动画值为其value值
- * [targetValue]要通过动画转化到的目标值
- * [duration]动画的持续时间
- */
- @OptIn(ExperimentalComposeApi::class)
- suspend fun animateWithFloat(
- initialValueWithState: MutableState<Float>,
- targetValue: Float,
- duration: Int = AnimationConstants.DefaultDurationMillis,
- ) {
- //动画起始值,目标差值
- val startValue = initialValueWithState.value
- val valueToBeTransformed = targetValue - startValue
- //动画起始时间,持续时间
- val startTime = System.nanoTime()
- val duration = duration * 1000000L
- //通过循环在下一帧计算动画的值
- val frameClock = coroutineContext.monotonicFrameClock
- while (System.nanoTime() <= startTime + duration) {
- frameClock.withFrameNanos {
- //计算动画的值,并设置值给状态
- val progress = minOf(it - startTime, duration).toFloat() / duration
- val increase = progress * valueToBeTransformed
- initialValueWithState.value = startValue + increase
- }
- }
- }
使用方式如下,效果跟示例差不多:
ps:不建议线上项目用这个api,还是用系统的比较好,如果想使用也可以参考(欢迎star): ComposeViews/MAnimator.kt at main · ltttttttttttt/ComposeViews (github.com)
如果有分析的不对的地方请大佬们指出
end