2
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

AndroidAdvent Calendar 2023

Day 17

Composeで縁取り文字を描画しよう

Posted at

縁取り文字とは

文字の色とは違う色などで縁取られた文字列のこと。

実装

縁取り文字を実現するためには、なるべくならComposeで提供されているTextをそのまま使えると嬉しい。ということで、ここではComposeのTextをベースとして実装する。そのためまずはTextを拡張するためのComposableとしてOutlinedTextを定義する。TextをラップするようなComposableを定義する際には、TextStyleを受け取るようにしておこう。

@Composable
fun OutlinedText(
    text: String,
    modifier: Modifier = Modifier,
    textStyle: TextStyle = TextStyle.Default,
) {
    Text(
        text = text,
        style = textStyle,
        modifier = modifier
    )
}

この時点では、普通に文字列が描画されるだけなのでここから縁取り部分(Stroke)を描画するようにしていく。Text自体にそれを実現できるようなパラメータはない。なので Modifier.drawBehind() を使って通常の文字の描画の後ろに縁取り部分を描画していくことにする。

@Composable
fun OutlinedText(...) {
    Text(
        ...,
        modifier = modifier
            .drawBehind {
                drawText(
                    textLayoutResult = TODO(),
                    drawStyle = Stroke(),
                    color = Color.Magenta,
                )
            }
    )
}

このときdrawTextでTextと同様のTextLayoutResultを渡す必要がある。そのため、Textのレイアウトが計算された際にそれを保持するようにし利用する。

@Composable
fun OutlinedText(...) {
    var textLayoutResult: TextLayoutResult? by remember {
        mutableStateOf(null)
    }
    Text(
        ...,
        onTextLayout = { textLayoutResult = it },
        modifier = modifier
            .drawBehind {
                textLayoutResult?.let {
                    drawText(
                        textLayoutResult = it,
                        drawStyle = Stroke(),
                        color = Color.Magenta,
                    )
                }
            }
    )
}

あとはdrawTextに渡すStrokeに width を設定する。このとき必要になる値が画面上のPixelにあたるFloatなので、DpからPxに変換してから使うようにする。

@Composable
fun OutlinedText(...) {
    ...
    val strokeWidth = with(LocalDensity.current) { 4.dp.toPx() }
    Text(
        ...,
        modifier = modifier
            .drawBehind {
                textLayoutResult?.let {
                    drawText(
                        textLayoutResult = it,
                        drawStyle = Stroke(width = strokeWidth),
                        color = Color.Magenta,
                    )
                }
            }
    )
}

この時点でPreviewしてあげると次のようになる。

outlined_text_preview.png

@Preview
@Composable
private fun OutlinedTextPreview() {
    val text = "OutlinedText"
    val textStyle = TextStyle(
        color = Color.White,
        fontSize = 40.sp,
        fontWeight = FontWeight.ExtraBold,
    )
    MyTheme {
        OutlinedTextQiita(
            text = text,
            textStyle = textStyle,
        )
    }
}

完成

あとはコードを整理して最終的には次のようなComposableが完成する。

@Composable
fun OutlinedText(
    text: String,
    modifier: Modifier = Modifier,
    textStyle: TextStyle = TextStyle.Default,
    stroke: Stroke = Stroke(),
    strokeColor: Color = Color.Transparent,
) {
    var textLayoutResult: TextLayoutResult? by remember {
        mutableStateOf(null)
    }
    Text(
        text = text,
        style = textStyle,
        onTextLayout = {
            textLayoutResult = it
        },
        modifier = modifier
            .drawBehind {
                textLayoutResult?.let {
                    drawText(
                        textLayoutResult = it,
                        drawStyle = stroke,
                        color = strokeColor,
                    )
                }
            }
    )
}

まとめ

今回はComposeから提供されているTextを使って、縁取り文字を描画するComposableを作成することができた。この実装で重要な点は次のようなものがある。

  • TextをラップするComposableではTextStyleを受け取ると楽
  • Textのレイアウトの情報は onTextLayout で取得できる
  • drawTextの drawStyle で描画のスタイルを指定できる

これらの点や仕組みを知ることで今回のような文字列を描画することができるようになった。実際にプロダクトで利用できるかどうかはまだわからないが仕組みとしては面白いと思ったし、やってみた楽しかった。

2
4
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
2
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?