All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.peanuuutz.fork.ui.foundation.text.ClickableText.kt Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2020 The Android Open Source Project
 * Modifications Copyright 2022 Peanuuutz
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package net.peanuuutz.fork.ui.foundation.text

import androidx.compose.runtime.Composable
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember
import androidx.compose.runtime.rememberUpdatedState
import net.peanuuutz.fork.ui.foundation.input.detectPressRelease
import net.peanuuutz.fork.ui.ui.draw.text.Paragraph
import net.peanuuutz.fork.ui.ui.draw.text.TextStyle
import net.peanuuutz.fork.ui.ui.modifier.Modifier
import net.peanuuutz.fork.ui.ui.modifier.input.PointerInputScope
import net.peanuuutz.fork.ui.ui.modifier.input.pointerInput

@Composable
fun ClickableText(
    paragraph: Paragraph,
    onClick: (charIndex: Int) -> Unit,
    modifier: Modifier = Modifier,
    ignoreClickOnGap: Boolean = true,
    textStyle: TextStyle = LocalTextStyle.current,
    overflow: TextOverflow = TextOverflow.Clip,
    softWrap: Boolean = true,
    maxLines: Int = Int.MAX_VALUE,
    onTextLayout: ((TextLayoutResult) -> Unit)? = null,
    selector: TextSelector? = null
) {
    val layoutResultState = remember { mutableStateOf(null) }
    val ignoreClickOnGapState = rememberUpdatedState(ignoreClickOnGap)
    val onClickState = rememberUpdatedState(onClick)

    BasicText(
        paragraph = paragraph,
        modifier = modifier.pointerInput {
            detectClickOnText(
                layoutResultProvider = layoutResultState::value,
                ignoreClickOnGapProvider = ignoreClickOnGapState::value,
                onClickProvider = onClickState::value
            )
        },
        textStyle = textStyle,
        overflow = overflow,
        softWrap = softWrap,
        maxLines = maxLines,
        onTextLayout = { result ->
            layoutResultState.value = result
            onTextLayout?.invoke(result)
        },
        selector = selector
    )
}

// Internal

internal suspend fun PointerInputScope.detectClickOnText(
    layoutResultProvider: () -> TextLayoutResult?,
    ignoreClickOnGapProvider: () -> Boolean,
    onClickProvider: () -> ((Int) -> Unit)
) {
    detectPressRelease { tapEvent ->
        val result = layoutResultProvider() ?: return@detectPressRelease
        val pressedCharIndex = if (ignoreClickOnGapProvider()) {
            result.measuredParagraph.getCharIndexFromOffsetExactly(tapEvent.position)
        } else {
            result.measuredParagraph.getCharIndexFromOffsetApproximately(tapEvent.position)
        }
        onClickProvider()(pressedCharIndex) // Do not filter NullIndex here
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy