Jetpack (七)Android Compose 基础<3>

Jetpack Compose 是用于构建原生 Android 界面的新工具包。Jetpack Compose 使用更少的代码、强大的工具和直观的 Kotlin API 简化并加快了 Android 上的界面开发。

在本教程中,您将使用声明性的函数构建一个简单的界面组件。您无需修改任何 XML 布局,也不需要直接创建界面微件,而只需要调用 Jetpack Compose 函数来声明您想要的元素,Compose 编译器即会完成后面的所有工作。

1.0 API Surface 功能完整,但可能包含错误

之前,简单用两篇文章认识了下 Compose 使用,接下来拆分知识点学习。

1. 可组合函数

Jetpack Compose 是围绕可组合函数构建的。这些函数可让您以编程方式定义应用界面,只需描述应用界面的形状和数据依赖关系,而不必关注界面的构建过程。如需创建可组合函数,只需将 @Composable 注释添加到函数名称中即可。

添加文本元素 Text("Hello world!")

首先,按照 Jetpack Compose 设置说明操作,使用 Empty Compose Activity 模板创建一个应用。默认模板已包含一些 Compose 元素,但我们下面要逐步进行构建。首先,删除“Greeting”和“Default Preview”函数,然后从 MainActivity 中删除 setContent 块,将该 Activity 留空。编译并运行您的空白应用。

现在,向空白的 Activity 中添加文本元素。可以通过定义内容块并调用 Text() 函数来实现此目的。

setContent 块定义了 Activity 的布局。我们不使用 XML 文件来定义布局内容,而是调用可组合函数。Jetpack Compose 使用自定义 Kotlin 编译器插件将这些可组合函数转换为应用的界面元素。例如,Compose 界面库定义了 Text() 函数;您可以调用该函数在应用中声明文本元素。

class MainActivity : ComponentActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContent {
            Text("Hello world!")
        }
    }
}
  

定义可组合函数

可组合函数只能在其他可组合函数的范围内调用。要使函数成为可组合函数,请添加 @Composable 注释。如需尝试此操作,请定义一个 Greeting() 函数并向其传递一个名称,然后该函数就会使用该名称配置文本元素。

在 Android Studio 中预览函数

当前的 Canary 版 Android Studio 允许您在 IDE 中预览可组合函数,而无需将应用下载到 Android 设备或模拟器中。主要限制在于,可组合函数不能接受任何参数。因此,您无法直接预览 Greeting() 函数,而是需要创建另一个名为 PreviewGreeting() 的函数,由该函数使用适当的参数调用 Greeting()。请在 @Composable 上方添加 @Preview 注释。

class MainActivity : ComponentActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContent {
            Greeting("Android")
        }
    }
}

@Composable
fun Greeting(name: String) {
    Text (text = "Hello $name!")
}
  

@Preview
@Composable
fun PreviewGreeting() {
    Greeting("Android")
}
  

重新构建您的项目。由于新的 previewGreeting() 函数未在任何位置受到调用,因此应用本身不会更改,但 Android Studio 会添加一个预览窗口。此窗口会显示由标有 @Preview 注释的可组合函数创建的界面元素的预览。任何时候,如需更新预览,请点击预览窗口顶部的刷新按钮。

2. 布局

从一些文本开始, 会出现重叠现象

返回到您的 Activity,用新的 NewsStory() 函数替换 Greeting() 函数。在本教程的其余部分,您将修改该 NewsStory() 函数,并且不会再更改 Activity 代码。

最佳做法是单独创建不会被应用调用的预览函数;专门的预览函数可以提高性能,并且有利于以后更轻松地设置多个预览。因此,请创建一个默认预览函数,该函数的唯一用途就是调用 NewsStory() 函数。随着您按照本教程对 NewsStory() 进行更改,预览内容会反映您所做的更改。

这段代码会在内容视图中创建三个文本元素。但是,由于我们未提供有关如何排列这三个文本元素的信息,因此它们会相互重叠,使文本无法阅读。

class MainActivity : ComponentActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContent {
            NewsStory()
        }
    }
}

@Composable
fun NewsStory() {
    Text("A day in Shark Fin Cove")
    Text("Davenport, California")
    Text("December 2018")
}

@Preview
@Composable
fun DefaultPreview() {
    NewsStory()
}
  

使用 Column,解决布局问题

Column 函数可让您垂直堆叠元素。向 NewsStory() 函数中添加一个 Column。

默认设置会直接将所有子项逐个堆叠起来,中间不留间距。Column 本身位于内容视图的左上角。

@Composable
fun NewsStory() {
    Column {
        Text("A day in Shark Fin Cove")
        Text("Davenport, California")
        Text("December 2018")
    }
}
  

向 Column 中添加样式设置

通过将参数传递给 Column 调用,可以配置 Column 的尺寸和位置,以及 Column 的子项的排列方式。

该设置具有以下含义:

modifier:可供您配置布局。本例中使用了一个 Modifier.padding 修饰符,将 Column 内嵌在周围的视图中。

@Composable
fun NewsStory() {
    Column(
        modifier = Modifier.padding(16.dp)
    ) {
        Text("A day in Shark Fin Cove")
        Text("Davenport, California")
        Text("December 2018")
    }
}
  

添加图片

我们想在文本上面添加图片。使用资源管理器将这张名为 header 的照片添加到应用的可绘制资源。

现在,修改您的 NewsStory() 函数。您将添加对 Image() 的调用,以将图片放入 Column。“foundation”软件包中提供了这些可组合项,您可能需要添加该软件包。请参阅 Jetpack Compose 设置说明。图片的比例会有问题,但没关系,您可以在下一步中纠正此问题。

@Composable
fun NewsStory() {
    Column(
        modifier = Modifier.padding(16.dp)
    ) {
        Image(
            painter = painterResource(R.drawable.header),
            contentDescription = null
        )

        Text("A day in Shark Fin Cove")
        Text("Davenport, California")
        Text("December 2018")
    }
}
  

图片已添加到布局中,但其尺寸尚未调整为适当大小。如需设置图片样式,请将尺寸 Modifier 传递给对 Image() 的调用。

height(180.dp):指定图片的高度。
fillMaxWidth():指定图片的宽度应足以填充所属布局。
您还需要向 Image() 传递一个 contentScale 参数:

contentScale = ContentScale.Crop:指定图片应填充 Column 的整个宽度,并根据需要剪裁为适当的高度。

@Composable
fun NewsStory() {
    Column(
        modifier = Modifier.padding(16.dp)
    ) {
        Image(
            painter = painterResource(R.drawable.header),
            contentDescription = null,
            modifier = Modifier
                .height(180.dp)
                .fillMaxWidth(),
            contentScale = ContentScale.Crop
        )

        Text("A day in Shark Fin Cove")
        Text("Davenport, California")
        Text("December 2018")
    }
}
  

添加 Spacer,将图片与标题分开。

@Composable
fun NewsStory() {
    Column(
        modifier = Modifier.padding(16.dp)
    ) {
        Image(
            painter = painterResource(R.drawable.header),
            contentDescription = null,
            modifier = Modifier
                .height(180.dp)
                .fillMaxWidth(),
            contentScale = ContentScale.Crop
        )
        Spacer(Modifier.height(16.dp))

        Text("A day in Shark Fin Cove")
        Text("Davenport, California")
        Text("December 2018")
    }
}
  

3. Material Design

Compose 旨在支持 Material Design 原则。它的许多界面元素都原生支持 Material Design。使用 Material 微件来设置应用的样式。

采用形状

Material Design 系统的关键要素之一就是 Shape。使用 clip() 函数对图片的四角进行圆角化处理。

Shape 不可见,但图片已被剪裁以匹配 Shape,因此现在呈现轻微的圆角。

设置文本样式

借助 Compose,您可以轻松遵循 Material Design 原则。将 MaterialTheme 应用到您创建的组件。

差别可能不太明显,但文本现在采用了 MaterialTheme 的默认文本样式。接下来,对每个文本元素应用特定的段落样式。

但有时,一篇文章的标题很长,我们不希望过长的标题影响应用的外观。尝试更改第一个文本元素。

配置文本元素,将长度上限设置为 2 行。如果文本很短,不超过此限制,则此设置没有影响;但如果文本过长,显示的文本就会被自动截短。

@Composable
fun NewsStory() {
    MaterialTheme {
        val typography = MaterialTheme.typography
        Column(
            modifier = Modifier.padding(16.dp)
        ) {
            Image(
                painter = painterResource(R.drawable.header),
                contentDescription = null,
                modifier = Modifier
                    .height(180.dp)
                    .fillMaxWidth()
                    .clip(shape = RoundedCornerShape(4.dp)),
                contentScale = ContentScale.Crop
            )
            Spacer(Modifier.height(16.dp))

            Text(
                "A day wandering through the sandhills " +
                     "in Shark Fin Cove, and a few of the " +
                     "sights I saw",
                style = typography.h6,
                maxLines = 2,
                overflow = TextOverflow.Ellipsis)
            Text("Davenport, California",
                style = typography.body2)
            Text("December 2018",
                style = typography.body2)
        }
    }
}
  
©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容