Jetpack Compose Text文本控件讲解

2,171 阅读8分钟

Text是用于显示文本的控件,它的底层实现是通过Cavans的drawText去实现。接下来我们来讲解下它的用法。

一:Text的构造源码

@Composable
fun Text(
    text: String,
    modifier: Modifier = Modifier,
    color: Color = Color.Unspecified,
    fontSize: TextUnit = TextUnit.Unspecified,
    fontStyle: FontStyle? = null,
    fontWeight: FontWeight? = null,
    fontFamily: FontFamily? = null,
    letterSpacing: TextUnit = TextUnit.Unspecified,
    textDecoration: TextDecoration? = null,
    textAlign: TextAlign? = null,
    lineHeight: TextUnit = TextUnit.Unspecified,
    overflow: TextOverflow = TextOverflow.Clip,
    softWrap: Boolean = true,
    maxLines: Int = Int.MAX_VALUE,
    onTextLayout: (TextLayoutResult) -> Unit = {},
    style: TextStyle = AmbientTextStyle.current
) {
    ...
}
  • text是显示的文本
  • modifier 修饰符,详细可以点击链接查看文章 Modifier详解
  • color 是颜色
  • fontSize 字体的大小
  • fontStyle 字体的样式 (比如斜体)
  • fontWeight 字体的样式 (比如粗体)
  • fontFamily 处理字体 (比如设置宋体等)
  • letterSpacing 每个字符之间的间距
  • textDecoration 文本的装饰,比如TextDecoration.Underline 添加下划线,比如TextDecoration.LineThrough 中线划线
  • textAlign 设置文字对齐 对齐方式
  • lineHeight 设置行高
  • overflow 当文字溢出的时候,比如...设置显示在最后
  • softWrap 声明是否应在换行符处断开文本,默认是true
  • maxLines 最大行数
  • onTextLayout
  • style 是TextStyle,文本样式的支持

二:显示文字

显示文字的最基本方法是使用以 String 作为参数的 Text 可组合项:

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

1.2 显示资源中的文字

我们建议您使用字符串资源,而不是对 Text 值进行硬编码,因为使用字符串资源时您可以与 Android 视图共享相同的字符串,并为您的应用国际化做好准备

@Composable
fun StringResourceText() {
  Text(stringResource(R.string.hello_world))
}

三:设置文字样式

3.1 更改文字颜色

@Composable
fun BlueText() {
  Text("Hello World", color = Color.Blue)
}

3.2 更改字号

@Composable
fun BigText() {
  Text("Hello World", fontSize = 30.sp)
}

3.3 设置斜体

@Composable
fun ItalicText() {
  // 设置斜体
  Text("Hello World", fontStyle = FontStyle.Italic)
}

3.4 设置粗体

@Composable
fun ItalicText() {
  // 设置粗体
  Text("Hello World2", fontWeight = FontWeight.Bold)
}

3.5 文字对齐

@Preview(showBackground = true)
@Composable
fun CenterText() {
    Text("Hello World", textAlign = TextAlign.Center,
                modifier = Modifier.width(150.dp))
}

bloack
默认情况下,Text 会根据其内容值选择自然的文字对齐方式:

  • 对于从左到右书写的文字,如拉丁语、西里尔文或朝鲜文,向 Text 容器的左边缘对齐
  • 对于从右到左书写的文字,如阿拉伯语或希伯来语,向 Text 容器的右边缘对齐 如果您想手动设置 Text 可组合项的文字对齐方式,最好分别使用 TextAlign.Start 和 TextAlign.End(而不要使用 TextAlign.Left 和 TextAlign.Right),这样系统就可以根据具体语言的首选文字方向,将您的设置解析为向 Text 的右边缘对齐

3.6 处理字体

Text 有一个 fontFamily 参数,用于设置可组合项中使用的字体。默认情况下,系统会添加 Serif、Sans Serif、等宽和 Cursive 字体系列:

@Composable
fun DifferentFonts() {
    Column {
        Text("Hello World", fontFamily = FontFamily.Serif)
        Text("Hello World", fontFamily = FontFamily.SansSerif)
    }
}

bloack
当然你依然可以使用自己的字体文件,他依旧放在res-font下 bloack 然后你需要定义一个字体簇,这是Compose推荐的用法,因为这样可以包含不同的字重

val firaSansFamily = FontFamily(
        Font(R.font.firasans_light, FontWeight.Light),
        Font(R.font.firasans_regular, FontWeight.Normal),
        Font(R.font.firasans_italic, FontWeight.Normal, FontStyle.Italic),
        Font(R.font.firasans_medium, FontWeight.Medium),
        Font(R.font.firasans_bold, FontWeight.Bold)
)

最后,您可以将此 fontFamily 传递给 Text 可组合项。由于 fontFamily 可以包含不同的粗细度,因此您可以手动设置 fontWeight 来为您的文字选择合适的粗细度:

Column {
    Text(..., fontFamily = firaSansFamily, fontWeight = FontWeight.Light)
    Text(..., fontFamily = firaSansFamily, fontWeight = FontWeight.Normal)
    Text(..., fontFamily = firaSansFamily, fontWeight = FontWeight.Normal,
                    fontStyle = FontStyle.Italic)
    Text(..., fontFamily = firaSansFamily, fontWeight = FontWeight.Medium)
    Text(..., fontFamily = firaSansFamily, fontWeight = FontWeight.Bold)
}

bloack

3.7 文本装饰

@Preview()
@Composable
fun textTest(){
    Column() {
        Text(text = stringResource(id = R.string.app_name),textDecoration = TextDecoration.LineThrough)
    }
}

3.8 行高

@Preview()
@Composable
fun textTest(){
    Column() {
        Text(text = stringResource(id = R.string.app_name),lineHeight = 15.sp)
    }
}

3.9 字符间距

@Preview()
@Composable
fun textTest(){
    Column() {
        Text(text = stringResource(id = R.string.app_name),letterSpacing = 3.sp)
    }
}

3.10 maxline是设置最大行数

@Composable
@Composable
fun LongText() {
    Text("hello ".repeat(50), maxLines = 2)
}

3.11 overflow设置文字溢出

在限制长文字时,您可能需要指定文字溢出,这些内容只有在显示的文字被截断时才会显示。如需指定文字溢出,请按如下方式设置 textOverflow 参数:

@Composable
fun OverflowedText() {
    // TextOverflow.Ellipsis是末尾...   TextOverflow.Clip是截取
    Text("Hello Compose ".repeat(50), maxLines = 2, overflow = TextOverflow.Ellipsis)
}

3.12 TextStyle样式

我们来看一下, Text的最后一个参数style,实际上是TextStyle, TextStyle主要也是对text样式的支持,我们来看下TextStyle的源码

@Immutable
data class TextStyle(
    val color: Color = Color.Unspecified,
    val fontSize: TextUnit = TextUnit.Unspecified,
    val fontWeight: FontWeight? = null,
    val fontStyle: FontStyle? = null,
    val fontSynthesis: FontSynthesis? = null,
    val fontFamily: FontFamily? = null,
    val fontFeatureSettings: String? = null,
    val letterSpacing: TextUnit = TextUnit.Unspecified,
    val baselineShift: BaselineShift? = null,
    val textGeometricTransform: TextGeometricTransform? = null,
    val localeList: LocaleList? = null,
    val background: Color = Color.Unspecified,
    val textDecoration: TextDecoration? = null,
    val shadow: Shadow? = null,
    val textAlign: TextAlign? = null,
    val textDirection: TextDirection? = null,
    val lineHeight: TextUnit = TextUnit.Unspecified,
    val textIndent: TextIndent? = null
){
    ...
}
  • color 一样是设置文字颜色
  • fontSize 字体大小
  • fontWeight 设置字体样式比如粗体等
  • fontStyle 设置字体样式比如斜体
  • fontFamily 设置字体。比如宋体
  • letterSpacing 设置字符间距
  • background 设置背景颜色
  • textDecoration 设置装饰,比如下划线,中划线
  • textAlign 设置对齐方式
  • textDirection 设置文本方向,比如从左往右还是从右往左
  • lineHeight 设置行高
// 由于前面这些属性我们都介绍过,所以同意简单写个样例
@Preview()
@Composable
fun textStyleTest(){
    Column {
        Text(text = stringResource(id = R.string.app_name),style = TextStyle(
                color = Color.Blue,fontSize = 13.sp,fontWeight=FontWeight.Bold
                ,fontStyle = FontStyle.Italic,fontFamily= FontFamily.Cursive
                ,letterSpacing= TextUnit.Companion.Em(1f),background = Color.Yellow
                ,textDecoration = TextDecoration.Underline,textAlign = TextAlign.Center,
                textDirection = TextDirection.ContentOrRtl,lineHeight = TextUnit.Companion.Em(13f)
        ))
    }
}
  • shadow 设置阴影 (给文字设置阴影)
@Preview()
@Composable
fun textStyleTest(){
    Column {
        Text(text = stringResource(id = R.string.app_name),style = TextStyle(
               shadow = Shadow(
                    color = Color.Red,blurRadius = 3f,offset = Offset(3)
               )
        ))
    }
}
  • fontSynthesis 字体的合成,主要有四种情况
enum class FontSynthesis {
    /**
     * 关闭字体合成。如果[FontFamily]中不存在粗体或倾斜的面,则不会合成它们
     */
    None,

    /**
     * 如果[FontFamily]中没有粗体字体,则只合成粗体字体。不会合成倾斜字体。
     */
    Weight,

    /**
     * 如果[FontFamily]中没有粗体字体,则只合成粗体字体。不会合成倾斜字体。
     */
    Style,

    /**
     * 如果[FontFamily]中没有粗体和倾斜字体,则系统会综合这两种字体
     */
    All;

    internal val isWeightOn: Boolean
        get() = this == All || this == Weight

    internal val isStyleOn: Boolean
        get() = this == All || this == Style
}
  • textGeometricTransform 几何变换 两格参数scaleX是表示文本在水平方向上的缩放比例,skewX是表示文本在水平方向上的裁剪倾斜。
@Immutable
data class TextGeometricTransform(
    val scaleX: Float = 1.0f,
    val skewX: Float = 0f
) {
   ... 
}

举个例子

// 举个例子
@Preview()
@Composable
fun textStyleTest(){
    Column {
        Text(text = "你好 Android",style = TextStyle(
                textGeometricTransform = TextGeometricTransform(2f,2f)
        ))
    }
}
  • fontFeatureSettings
  • baselineShift 基准线的设置,我们文本都会有一个基准线,而这里可以设置三种值。BaselineShift.BaselineShift,BaselineShift.Subscript,BaselineShift.NONE
// 举个例子
@Preview()
@Composable
fun textStyleTest(){
    Column {
        Text(text = "你好 Android",style = TextStyle(
                baselineShift = BaselineShift.Subscript
        ))
    }
}
  • localeList
  • textIndent 指定段落的缩进。firstLine是第一行的缩进多少,restLine除第一行以外的每行的缩进量
data class TextIndent(
    val firstLine: TextUnit = 0.sp,
    val restLine: TextUnit = 0.sp
) {
   ...
}

@Preview()
@Composable
fun textStyleTest(){
    Column {
        Text(text = stringResource(id = R.string.app_name),style = TextStyle(
                textIndent = TextIndent(40.sp, 20.sp)
        ))
    }
}

bloack

3.13 文字中包含多种样式

当同一个文本里有存在多种样式,传统的设置不同的样式用Span,Compose里面使用 AnnotatedString,该字符串可使用任意注解样式加以注解。AnnotatedString 是一个数据类,其中包含:

  • 一个 Text 值
  • 一个 SpanStyleRange 的 List,等同于位置范围在文字值内的内嵌样式
  • 一个 ParagraphStyleRange 的 List,用于指定文字对齐、文字方向、行高和文字缩进样式 TextStyle 用于 Text 可组合项,而 SpanStyle 和 ParagraphStyle 用于 AnnotatedString。

SpanStyle 和 ParagraphStyle 之间的区别在于,ParagraphStyle 可应用于整个段落,而 SpanStyle 可以在字符级别应用。一旦用 ParagraphStyle 标记了一部分文字,该部分就会与其余部分隔开,就像在开头和末尾有换行符一样

@Preview()
@Composable
fun textTest(){
    Column() {
        androidx.compose.material.Text(text = AnnotatedString(){
            withStyle(style = SpanStyle(color = Color.Blue)){
                append("H")
            }
            append("ello ")
            withStyle(style = SpanStyle(fontWeight = FontWeight.Bold)){
                append("W")
            }
            append("orld")
        })
    }
}

bloack

我们可以按相同的方式设置段落样式:

@Preview()
@Composable
fun textTest2(){
    Column() {
        androidx.compose.material.Text(text = AnnotatedString(){
            withStyle(style = ParagraphStyle(lineHeight = 30.sp)) {
                withStyle(style = SpanStyle(color = Color.Blue)) {
                    append("Hello\n")
                }
                withStyle(style = SpanStyle(fontWeight = FontWeight.Bold,
                        color = Color.Red)) {
                    append("World\n")
                }
                append("Compose")
            }
        })
    }
}

bloack

三:用户互动

3.1 选择文字

默认情况下,可组合项不可选择,这意味着在默认情况下用户无法从您的应用中选择和复制文字。要启用文字选择,需要使用 SelectionContainer 可组合项封装文字元素:

@Composable
fun SelectableText() {
    SelectionContainer {
        Text("This text is selectable")
    }
}

bloack 您可能想为可选择区域的特定部分停用选择功能。如果要执行此操作,您需要使用 DisableSelection 可组合项来封装不可选择的部分:

@Composable
fun PartiallySelectableText() {
    SelectionContainer {
        Column {
            Text("This text is selectable")
            Text("This one too")
            Text("This one as well")
            DisableSelection {
                Text("But not this one")
                Text("Neither this one")
            }
            Text("But again, you can select this one")
            Text("And this one too")
        }
    }
}

bloack

3.2 获取点击文字的位置

如需监听 Text 的点击次数,您可以添加 clickable 修饰符。不过,如果您想在 Text 可组合项内获取点击位置,在对文字的不同部分执行了不同操作的情况下,您需要改用 ClickableText。

@Composable
fun SimpleClickableText() {
    ClickableText(
        text = AnnotatedString("Click Me"),
        onClick = { offset ->
            Log.d("ClickableText", "$offset -th character is clicked.")
        }
    )
}

3.3 点击注解

当用户点击 Text 可组合项时,您可能想向 Text 值的某一部分附加额外信息,例如向特定字词附加可在浏览器中打开的网址。如果要执行此操作,您需要附加一个注解,用于获取一个标记 (String)、一个项 (String) 和一个文字范围作为参数。在 AnnotatedString 中,这些注解可以按照其标记或文字范围进行过滤。示例如下:

@Preview()
@Composable
fun textTest3(){
    val annotatedText = AnnotatedString {
        append("Click ")

        // We attach this *URL* annotation to the following content
        // until `pop()` is called
        pushStringAnnotation(tag = "URL",
                annotation = "https://developer.android.com")
        withStyle(style = SpanStyle(color = Color.Blue,
                fontWeight = FontWeight.Bold)) {
            append("here")
        }

        pop()
    }

    ClickableText(
            text = annotatedText,
            onClick = { offset ->
                // We check if there is an *URL* annotation attached to the text
                // at the clicked position
                annotatedText.getStringAnnotations(tag = "URL", start = offset,
                        end = offset)
                        .firstOrNull()?.let { annotation ->
                            // If yes, we log its value
                            Log.d("Clicked URL", annotation.item)
                        }
            }
    )
}