使用可组合项预览来预览界面

可组合项由函数定义,并使用 @Composable 进行注解:

@Composable
fun SimpleComposable() {
    Text("Hello World")
}

包含“Hello World”字样的简单文本元素

如需实现此可组合项的预览,请再创建一个使用 @Composable@Preview 进行注解的可组合项。这个带注解的新可组合项现在包含您最初创建的可组合项 SimpleComposable

@Preview
@Composable
fun SimpleComposablePreview() {
    SimpleComposable()
}

@Preview 注解会告知 Android Studio 应在此文件的设计视图中显示此可组合项。在您进行修改时,可实时查看可组合项预览的更新。

一张显示了使用 Compose 预览进行实时更新的 GIF

您可以在代码中手动添加参数,以自定义 Android Studio 呈现 @Preview 的方式。您甚至可以多次向同一函数添加 @Preview 注解,以预览具有不同属性的可组合项。

使用 @Preview 可组合项的一个主要好处是可以避免依赖于 Android Studio 中的模拟器。您可以省下占用大量内存的模拟器启动过程,以便进行更多最终的外观和风格更改,并使 @Preview 能够轻松进行和测试细微的代码更改。

如需最有效地利用 @Preview 注解,请务必根据屏幕接收的输入状态和输出的事件来定义屏幕。

明确您的@Preview

Android Studio 提供了一些功能来扩展可组合项预览。您可以更改其容器设计,与之互动,或直接将其部署到模拟器或设备。

尺寸

默认情况下,系统会自动选择 @Preview 尺寸来封装其内容。如需手动设置尺寸,请添加 heightDpwidthDp 参数。这些值已解释为 dp,因此您无需向其中添加 .dp

@Preview(widthDp = 50, heightDp = 50)
@Composable
fun SquareComposablePreview() {
    Box(Modifier.background(Color.Yellow)) {
        Text("Hello World")
    }
}

包含“Hello World”字样的黄色方形

动态配色预览

如果您已在应用中启用了动态配色,请使用 wallpaper 属性切换壁纸,并查看界面对不同用户选择的壁纸有何反应。从 Wallpaper 类提供的不同壁纸主题中进行选择。此功能需要 Compose 1.4.0 或更高版本。

在不同设备上使用

在 Android Studio Flamingo 中,您可以修改预览注解的 device 参数,以定义不同设备中可组合项的配置。

可组合函数示例

当设备参数为空字符串 (@Preview(device = "")) 时,您可以按 Ctrl + Space 来调用自动补全功能。然后,您可以设置每个参数的值。

修改示例函数

在自动补全功能中,您可以从列表中选择任何设备选项,例如 @Preview(device = "id:pixel_4")。或者,您也可以选择 spec:width=px,height=px,dpi=int… 以输入自定义设备,从而设置每个参数的各个值。

规范列表

如需应用,请按 Enter;如需取消,请按 Esc

如果您设置的值无效,系统会以红色下划声明,并且可能提供修复程序(Alt + Enter [在 macOS 上为 ⌥ + ⏎] > Replace with …。检查会尝试提供与您的输入最相似的修复程序。

无效值示例

语言区域

如需测试不同的用户语言区域,请添加 locale 参数:

@Preview(locale = "fr-rFR")
@Composable
fun DifferentLocaleComposablePreview() {
    Text(text = stringResource(R.string.greeting))
}

一个简单的文本元素,其中包含“Bonjour”一词和一面法国国旗

设置背景颜色

默认情况下,可组合项会以透明背景显示。如需添加背景,请添加 showBackgroundbackgroundColor 参数。请注意,backgroundColor 是 ARGB Long,而不是 Color 值:

@Preview(showBackground = true, backgroundColor = 0xFF00FF00)
@Composable
fun WithGreenBackground() {
    Text("Hello World")
}

包含“Hello World”字样的绿色矩形

系统界面

如需在预览中显示状态栏和操作栏,请添加 showSystemUi 参数:

@Preview(showSystemUi = true)
@Composable
fun DecoratedComposablePreview() {
    Text("Hello World")
}

一个预览窗口,其中显示了一个具有状态栏和操作栏的 activity。

界面模式

参数 uiMode 可接受任何 Configuration.UI_* 常量,并允许您相应地更改预览的行为。例如,您可以将预览设置为夜间模式,看看主题如何响应。

Compose 预览界面

LocalInspectionMode

您可以从 LocalInspectionMode CompositionLocal 中读取数据,以查看可组合项是否在预览中(可检查的组件内)呈现。如果可组合项在预览中呈现,LocalInspectionMode.current 的结果为 true。借助这些信息,您可以自定义预览;例如,您可以在预览窗口中显示占位符图片,而不是显示实际数据。

这样,您也可以规避限制。例如,显示示例数据,而不是调用网络请求。

@Composable
fun GreetingScreen(name: String) {
    if (LocalInspectionMode.current) {
        // Show this text in a preview window:
        Text("Hello preview user!")
    } else {
        // Show this text in the app:
        Text("Hello $name!")
    }
}

@Preview 交互

Android Studio 提供了一些功能,可让您与定义的预览进行互动。这种互动有助于您了解预览的运行时行为,并让您能够通过预览更好地浏览界面。

互动模式

在互动模式下,您可以与预览互动,就像在运行您的程序的设备(例如手机或平板电脑)上互动一样。互动模式会被隔离在沙盒环境中(也就是说,与其他预览隔离开),在该模式下,您可以点击元素并在预览中输入用户输入。这是一种测试可组合项的不同状态、手势甚至动画的快捷方法。

用户点击预览的“互动”按钮

用户与预览互动的视频

代码导航和可组合项大纲

您可以将鼠标悬停在预览上,以查看其中包含的可组合项的大纲。点击可组合项大纲会触发您的编辑器视图,从而导航到其定义。

用户将鼠标悬停在预览上,使 Studio 显示其可组合项的大纲

运行预览

您可以在模拟器或实体设备上运行特定的 @Preview。预览将作为新的 Activity 部署在同一项目应用中,因此它具有相同的上下文和权限。您无需编写样板代码来请求权限(如果已授予权限)。

点击 @Preview 注解旁边或预览顶部的 Run Preview 图标 “Run Preview”图标,Android Studio 会将该 @Preview 部署到连接的设备或模拟器上。

用户点击预览的“运行预览”按钮

用户将预览部署到设备的视频

复制 @Preview 呈现

通过右键点击呈现的每个预览,即可将其作为图像来复制。

用户点击预览以将其作为图片来复制。

同一 @Preview 注释的多个预览

您可以展示同一 @Preview 可组合项的多个版本,这些版本采用不同的规范,也可以展示传递给可组合项的不同参数。这样,您就可以减少原本需要编写的样板代码。

Multipreview 模板

androidx.compose.ui:ui-tooling-preview 1.6.0-alpha01+ 引入了 Multipreview API 模板:@PreviewScreenSizes@PreviewFontScales@PreviewLightDark@PreviewDynamicColors,因此,您只需一个注解,便可在常见场景中预览 Compose 界面。

使用模板预览不同的字体和屏幕尺寸

创建自定义多预览注解

使用 Multipreview 时,您可以定义一个注解类,该类本身可具有多个采用不同配置的 @Preview 注解。将此注解添加到可组合函数后,系统会自动同时呈现所有不同的预览。例如,您可以使用此注解同时预览多个设备、字体大小或主题,而无需为每一个可组合项重复这些定义。

首先,创建您自己的自定义注解类:

@Preview(
    name = "small font",
    group = "font scales",
    fontScale = 0.5f
)
@Preview(
    name = "large font",
    group = "font scales",
    fontScale = 1.5f
)
annotation class FontScalePreviews

您可以对预览可组合项使用此自定义注解:

@FontScalePreviews
@Composable
fun HelloWorldPreview() {
    Text("Hello World")
}

Android Studio 的 Design 标签页,显示了分别采用小号和大号字体的可组合项

您可以将多个 MultiPreview 注解和普通 preview 注解结合使用,从而创建一个更完整的预览集。结合使用 MultiPreview 注解并不意味着所有不同的组合都会得以呈现。实际上,每个 MultiPreview 注解会独立运行,并且仅会呈现自己的变体。

@Preview(
    name = "Spanish",
    group = "locale",
    locale = "es"
)
@FontScalePreviews
annotation class CombinedPreviews

@CombinedPreviews
@Composable
fun HelloWorldPreview2() {
    MaterialTheme { Surface { Text(stringResource(R.string.hello_world)) } }
}

Android Studio 的 Design 标签页,显示了采用所有配置的可组合项

多预览和普通预览的混合使用让您可以更全面地测试大型项目的许多属性。

@Preview 和大型数据集

在很多情况下,您需要向可组合项预览传递大型数据集。为此,只需使用 @PreviewParameter 注解来添加参数,即可将示例数据传递给某个可组合项预览函数。

@Preview
@Composable
fun UserProfilePreview(
    @PreviewParameter(UserPreviewParameterProvider::class) user: User
) {
    UserProfile(user)
}

如需提供示例数据,请创建一个可实现 PreviewParameterProvider 并以序列形式返回示例数据的类。

class UserPreviewParameterProvider : PreviewParameterProvider<User> {
    override val values = sequenceOf(
        User("Elise"),
        User("Frank"),
        User("Julia")
    )
}

这样,序列中的每个数据元素都会呈现一个预览:

Elise、Frank 和 Julia 可组合项的预览

您可以为多个预览使用相同的提供程序类。如有必要,可通过设置 limit 参数来限制预览次数。

@Preview
@Composable
fun UserProfilePreview2(
    @PreviewParameter(UserPreviewParameterProvider::class, limit = 2) user: User
) {
    UserProfile(user)
}

限制和最佳做法

Android Studio 会直接在预览区域中执行预览代码。它不需要运行模拟器或实体设备,因为它利用了 Android 框架的一部分(称为 Layoutlib)进行移植。Layoutlib 是 Android 框架的自定义版本,旨在在 Android 设备之外运行。该库的目标是在 Android Studio 中提供与设备上的渲染非常接近的布局预览。

预览版限制

由于 Android Studio 中预览的呈现方式,预览非常轻量,不需要整个 Android 框架即可呈现。不过,这种方法存在以下限制:

  • 无法访问网络
  • 无法访问文件
  • 有些 Context API 不一定完全可用

预览和ViewModels

在可组合项中使用 ViewModel 时,预览功能受到限制。预览系统无法构建传递给 ViewModel 的所有参数,例如代码库、用例、管理器或类似内容。此外,如果您的 ViewModel 参与了依赖项注入(例如使用 Hilt),预览系统将无法构建整个依赖关系图来构造 ViewModel

当您尝试使用 ViewModel 预览可组合项时,Android Studio 在渲染特定可组合项时会显示错误:

Android Studio 问题窗格,显示“未能实例化“ViewModel”消息

如果您想预览使用 ViewModel 的可组合项,则应创建另一个可组合项,并将 ViewModel 中的参数作为可组合项的参数传递给该可组合项。这样,您就无需预览使用 ViewModel 的可组合项。

@Composable
fun AuthorColumn(viewModel: AuthorViewModel = viewModel()) {
  AuthorColumn(
    name = viewModel.authorName,
    // ViewModel sends the network requests and makes posts available as a state
    posts = viewModel.posts
  )
}

@Preview
@Composable
fun AuthorScreenPreview(
  // You can use some sample data to preview your composable without the need to construct the ViewModel
  name: String = sampleAuthor.name,
  posts: List<Post> = samplePosts[sampleAuthor]
) {
  AuthorColumn(...) {
    name = NameLabel(name),
    posts = PostsList(posts)
  }
}

注解类 @Preview

您可以随时在 Android Studio 中通过按住 Ctrl 键或 ⌘ 键的同时点击 @Preview 注解来获取可在自定义预览时调整的完整参数列表。

annotation class Preview(
    val name: String = "",
    val group: String = "",
    @IntRange(from = 1) val apiLevel: Int = -1,
    val widthDp: Int = -1,
    val heightDp: Int = -1,
    val locale: String = "",
    @FloatRange(from = 0.01) val fontScale: Float = 1f,
    val showSystemUi: Boolean = false,
    val showBackground: Boolean = false,
    val backgroundColor: Long = 0,
    @UiMode val uiMode: Int = 0,
    @Device val device: String = Devices.DEFAULT,
    @Wallpaper val wallpaper: Int = Wallpapers.NONE,
)

其他资源

如需详细了解 Android Studio 如何提高 @Preview 的易用性,并了解更多工具提示,请参阅 Compose 工具博客。