五分钟搞定 Compose 的打字机效果

这周一天中午吃饭,我同事问我,能不能用 Compose 模仿一下 AI 的那种打字效果。

我跟他说那个不是模仿出来的,一般大模型接口使用的是 SSE 的协议,这个数据一帧一帧的发送,效果上就是几个字,几个字那么吐出来的,你不是大模型的接口没必要搞这个效果啊!

他说,那如果我想假装我是大模型接口呢?

斯......

我懂你意思!

那我们,Let's go!

核心思路

首先,我们需要创建一个可复用的 Composable 函数 TextTypeWriter,该函数将负责承载打字机动画的核心逻辑。为了保证灵活性,我们设计以下参数:

  • text:需要展示的完整文本字符串;
  • modifier:修饰符,用于调整文本组件的布局、间距等属性,在 Compose 中,这个参数默认在第一位;
  • onDone:动画完成后的回调函数,可用于触发后续操作。

Let's Go

思路已经清晰,那么让我们开始编码吧!

1. 状态管理:存储文本

首先需要创建一个可变状态变量,用于存储当前已显示的文本内容。借助 Jetpack Compose 的 remembermutableStateOf,可以确保状态在重组时保持持久化,且状态变化时会自动触发 UI 更新:

kotlin 复制代码
var textToDisplay by remember { mutableStateOf("") }

2. 动画触发:逐字追加

使用 LaunchedEffect 来启动动画逻辑------该函数的核心作用是在 Compose 生命周期内执行挂起函数,且仅在依赖参数变化时重新执行。这里我们将 text 作为依赖项,确保文本变化时动画会重新触发。

LaunchedEffect 内部,通过循环遍历完整文本的每个字符,逐字追加到 textToDisplay 中,并在每次追加后添加延迟,以此模拟打字机的逐字输出效果:

kotlin 复制代码
LaunchedEffect(key1 = text) {
    for (char in text.toCharArray()) {
        textToDisplay += char.toString()
        delay(delay) // 控制每个字符的显示间隔
    }
}

3. 文本展示:动态更新

textToDisplay 绑定到 Text 组件,当该状态变量更新时,Text 会自动重组并显示最新的文本内容,从而实现逐字显示的视觉效果:

kotlin 复制代码
Text(
    modifier = modifier,
    text = textToDisplay,
)

4. 回调触发:后续操作

为了增强扩展性,我们添加动画完成后的回调逻辑。即在携程执行完成之后,回调 onDone

完整代码实现

将上述步骤整合,最终的完整代码如下:

kotlin 复制代码
@Composable
fun TextTypeWriter(
    modifier: Modifier = Modifier,
    text: String,
    delay: Long = 160L,
    onDone: () -> Unit,
) {
    var textToDisplay by remember { mutableStateOf("") }

    LaunchedEffect(key1 = text) {
        for (char in text.toCharArray()) {
            textToDisplay += char.toString()
            delay(delay)
        }
        onDone.invoke()
    }

    Text(
        modifier = modifier,
        text = textToDisplay,
    )
}

使用示例

只需在你的 Composable 布局中直接调用该函数,即可快速实现打字机动画:

kotlin 复制代码
// 示例:在屏幕中央显示打字机动画文本
Box(
    modifier = Modifier.fillMaxSize(),
    contentAlignment = Alignment.Center,
) {
    TextTypeWriter(text = "I Love Android Development") {
        Log.d("TyW", "Done")
    }
}

我们看下效果:

还不错!

小优化一下

实际上这个效果还有个小优化空间,AI 的 SSE 接口一般返回来的数据,并不是匀速的,所以我们可以让打字机的效果带有随机的延迟,让其打印的延迟在一个区间,这样会更加真实。

优化后代码如下:

Kotlin 复制代码
@Composable
fun TextTypeWriter(
    modifier: Modifier = Modifier,
    text: String,
    delayRange: LongRange = 10L..400L, // 默认的随机延迟
    onDone: () -> Unit,
) {
    var textToDisplay by remember { mutableStateOf("") }

    LaunchedEffect(key1 = text) {
        for (char in text.toCharArray()) {
            textToDisplay += char.toString()
            delay(delayRange.random()) // LongRange.random() 随机选择
        }
        onDone.invoke()
    }

    Text(
        modifier = modifier,
        text = textToDisplay,
    )
}

看看效果:

嗯,就这样!

总结

通过本文,你已掌握了 Jetpack Compose 中打字机动画的核心实现逻辑。该方案代码简洁、复用性强,可通过参数灵活调整动画效果,满足不同场景的需求。

相关推荐
草莓熊Lotso23 分钟前
【Linux系统加餐】从原理到封装:基于建造者模式实现System V信号量工业级C++封装
android·linux·运维·服务器·网络·c++·建造者模式
程序员煊子5 小时前
用 Cursor 从零搭一个 Compose 本地记账 App:实战记录与源码解析
android·kotlin·compose·cursor
alexhilton7 小时前
面向Android开发者的Google I/O 2026
android·kotlin·android jetpack
私人珍藏库8 小时前
【Android】豆图助手-永久HY-模拟微X~zfb各种截图
android·app·工具·软件·多功能
程序员陆业聪9 小时前
Shadow实战接入与生产落地:从零搭建到稳定运行
android
程序员陆业聪9 小时前
Shadow Transform:编译期的魔法——字节码替换实战
android
imuliuliang13 小时前
Laravel6.x核心特性全解析
android·php·laravel
idingzhi14 小时前
A股量化策略日报(2026年05月22日)
android·开发语言·python·kotlin
测试员周周15 小时前
【Appium 系列】第14节-断言与验证 — Validator 的设计
android·人工智能·python·功能测试·ios·单元测试·appium
赏金术士15 小时前
Android 动画对比指南:View 系统 vs Jetpack Compose
android·kotlin·compose