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

commonMain.androidx.compose.foundation.text.TextFieldScroll.kt Maven / Gradle / Ivy

/*
 * Copyright 2023 The Android Open Source Project
 *
 * 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 androidx.compose.foundation.text

import androidx.compose.foundation.ExperimentalFoundationApi
import androidx.compose.foundation.OverscrollEffect
import androidx.compose.foundation.gestures.Orientation
import androidx.compose.foundation.gestures.ScrollableState
import androidx.compose.foundation.gestures.rememberScrollableState
import androidx.compose.foundation.gestures.scrollable
import androidx.compose.foundation.interaction.MutableInteractionSource
import androidx.compose.foundation.layout.offset
import androidx.compose.runtime.Composable
import androidx.compose.runtime.Stable
import androidx.compose.runtime.derivedStateOf
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableFloatStateOf
import androidx.compose.runtime.mutableIntStateOf
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember
import androidx.compose.runtime.saveable.listSaver
import androidx.compose.runtime.setValue
import androidx.compose.runtime.structuralEqualityPolicy
import androidx.compose.ui.Modifier
import androidx.compose.ui.composed
import androidx.compose.ui.draw.clipToBounds
import androidx.compose.ui.geometry.Rect
import androidx.compose.ui.layout.LayoutModifier
import androidx.compose.ui.layout.Measurable
import androidx.compose.ui.layout.MeasureResult
import androidx.compose.ui.layout.MeasureScope
import androidx.compose.ui.platform.LocalLayoutDirection
import androidx.compose.ui.platform.debugInspectorInfo
import androidx.compose.ui.text.TextLayoutResult
import androidx.compose.ui.text.TextRange
import androidx.compose.ui.text.input.TextFieldValue
import androidx.compose.ui.text.input.TransformedText
import androidx.compose.ui.text.input.VisualTransformation
import androidx.compose.ui.unit.Constraints
import androidx.compose.ui.unit.Density
import androidx.compose.ui.unit.LayoutDirection
import androidx.compose.ui.util.fastRoundToInt
import kotlin.math.min

@ExperimentalFoundationApi
@Composable
internal expect fun rememberTextFieldOverscrollEffect(): OverscrollEffect?

// Scrollable
internal fun Modifier.textFieldScrollable(
    scrollerPosition: TextFieldScrollerPosition,
    interactionSource: MutableInteractionSource? = null,
    enabled: Boolean = true,
    overscrollEffect: OverscrollEffect? = null
) = composed(
    inspectorInfo =
        debugInspectorInfo {
            name = "textFieldScrollable"
            properties["scrollerPosition"] = scrollerPosition
            properties["interactionSource"] = interactionSource
            properties["enabled"] = enabled
        }
) {
    // do not reverse direction only in case of RTL in horizontal orientation
    val rtl = LocalLayoutDirection.current == LayoutDirection.Rtl
    val reverseDirection = scrollerPosition.orientation == Orientation.Vertical || !rtl
    val scrollableState = rememberScrollableState { delta ->
        val newOffset = scrollerPosition.offset + delta
        val consumedDelta =
            when {
                newOffset > scrollerPosition.maximum ->
                    scrollerPosition.maximum - scrollerPosition.offset
                newOffset < 0f -> -scrollerPosition.offset
                else -> delta
            }
        scrollerPosition.offset += consumedDelta
        consumedDelta
    }
    // TODO: b/255557085 remove when / if rememberScrollableState exposes lambda parameters for
    //  setting these
    val wrappedScrollableState =
        remember(scrollableState, scrollerPosition) {
            object : ScrollableState by scrollableState {
                override val canScrollForward by derivedStateOf {
                    scrollerPosition.offset < scrollerPosition.maximum
                }
                override val canScrollBackward by derivedStateOf {
                    scrollerPosition.offset > 0f
                }
            }
        }
    val scroll = Modifier.scrollable(
        orientation = scrollerPosition.orientation,
        reverseDirection = reverseDirection,
        overscrollEffect = overscrollEffect,
        state = wrappedScrollableState,
        interactionSource = interactionSource,
        enabled = enabled && scrollerPosition.maximum != 0f
    )
    scroll
}

// Layout
// Expect/actual is needed due to a different implementation in uikit
internal expect fun Modifier.textFieldScroll(
    scrollerPosition: TextFieldScrollerPosition,
    textFieldValue: TextFieldValue,
    visualTransformation: VisualTransformation,
    textLayoutResultProvider: () -> TextLayoutResultProxy?
): Modifier

internal fun Modifier.defaultTextFieldScroll(
    scrollerPosition: TextFieldScrollerPosition,
    textFieldValue: TextFieldValue,
    visualTransformation: VisualTransformation,
    textLayoutResultProvider: () -> TextLayoutResultProxy?
): Modifier {
    val orientation = scrollerPosition.orientation
    val cursorOffset = scrollerPosition.getOffsetToFollow(textFieldValue.selection)
    scrollerPosition.previousSelection = textFieldValue.selection

    val transformedText = visualTransformation.filterWithValidation(textFieldValue.annotatedString)

    val layout =
        when (orientation) {
            Orientation.Vertical ->
                VerticalScrollLayoutModifier(
                    scrollerPosition,
                    cursorOffset,
                    transformedText,
                    textLayoutResultProvider
                )
            Orientation.Horizontal ->
                HorizontalScrollLayoutModifier(
                    scrollerPosition,
                    cursorOffset,
                    transformedText,
                    textLayoutResultProvider
                )
        }
    return this.clipToBounds().then(layout)
}

private data class VerticalScrollLayoutModifier(
    val scrollerPosition: TextFieldScrollerPosition,
    val cursorOffset: Int,
    val transformedText: TransformedText,
    val textLayoutResultProvider: () -> TextLayoutResultProxy?
) : LayoutModifier {
    override fun MeasureScope.measure(
        measurable: Measurable,
        constraints: Constraints
    ): MeasureResult {
        val childConstraints = constraints.copy(maxHeight = Constraints.Infinity)
        val placeable = measurable.measure(childConstraints)
        val height = min(placeable.height, constraints.maxHeight)

        return layout(placeable.width, height) {
            val cursorRect =
                getCursorRectInScroller(
                    cursorOffset = cursorOffset,
                    transformedText = transformedText,
                    textLayoutResult = textLayoutResultProvider()?.value,
                    rtl = false,
                    textFieldWidth = placeable.width
                )

            scrollerPosition.update(
                orientation = Orientation.Vertical,
                cursorRect = cursorRect,
                containerSize = height,
                textFieldSize = placeable.height
            )

            val offset = -scrollerPosition.offset
            placeable.placeRelative(0, offset.fastRoundToInt())
        }
    }
}

private data class HorizontalScrollLayoutModifier(
    val scrollerPosition: TextFieldScrollerPosition,
    val cursorOffset: Int,
    val transformedText: TransformedText,
    val textLayoutResultProvider: () -> TextLayoutResultProxy?
) : LayoutModifier {
    override fun MeasureScope.measure(
        measurable: Measurable,
        constraints: Constraints
    ): MeasureResult {
        // If the maxIntrinsicWidth of the children is already smaller than the constraint, pass
        // the original constraints so that the children has more information to  determine its
        // size.
        val maxIntrinsicWidth = measurable.maxIntrinsicWidth(constraints.maxHeight)
        val childConstraints =
            if (maxIntrinsicWidth < constraints.maxWidth) {
                constraints
            } else {
                constraints.copy(maxWidth = Constraints.Infinity)
            }
        val placeable = measurable.measure(childConstraints)
        val width = min(placeable.width, constraints.maxWidth)

        return layout(width, placeable.height) {
            val cursorRect =
                getCursorRectInScroller(
                    cursorOffset = cursorOffset,
                    transformedText = transformedText,
                    textLayoutResult = textLayoutResultProvider()?.value,
                    rtl = layoutDirection == LayoutDirection.Rtl,
                    textFieldWidth = placeable.width
                )

            scrollerPosition.update(
                orientation = Orientation.Horizontal,
                cursorRect = cursorRect,
                containerSize = width,
                textFieldSize = placeable.width
            )

            val offset = -scrollerPosition.offset
            placeable.placeRelative(offset.fastRoundToInt(), 0)
        }
    }
}

private fun Density.getCursorRectInScroller(
    cursorOffset: Int,
    transformedText: TransformedText,
    textLayoutResult: TextLayoutResult?,
    rtl: Boolean,
    textFieldWidth: Int
): Rect {
    val cursorRect =
        textLayoutResult?.getCursorRect(
            transformedText.offsetMapping.originalToTransformed(cursorOffset)
        ) ?: Rect.Zero
    val thickness = DefaultCursorThickness.roundToPx()

    val cursorLeft =
        if (rtl) {
            textFieldWidth - cursorRect.left - thickness
        } else {
            cursorRect.left
        }

    val cursorRight =
        if (rtl) {
            textFieldWidth - cursorRect.left
        } else {
            cursorRect.left + thickness
        }
    return cursorRect.copy(left = cursorLeft, right = cursorRight)
}

@Stable
internal class TextFieldScrollerPosition(
    initialOrientation: Orientation,
    initial: Float = 0f,
) {

    /*@VisibleForTesting*/
    constructor() : this(Orientation.Vertical)

    /**
     * Left or top offset. Takes values from 0 to [maximum]. Taken with the opposite sign defines
     * the x or y position of the text field in the horizontal or vertical scroller container
     * correspondingly.
     */
    var offset by mutableFloatStateOf(initial)

    /**
     * Maximum length by which the text field can be scrolled. Defined as a difference in size
     * between the scroller container and the text field.
     */
    var maximum by mutableFloatStateOf(0f)
        private set

    /** Size of the visible part, on the scrollable axis, in pixels. */
    var viewportSize by mutableIntStateOf(0)
        private set

    /**
     * Keeps the cursor position before a new symbol has been typed or the text field has been
     * dragged. We check it to understand if the [offset] needs to be updated.
     */
    private var previousCursorRect: Rect = Rect.Zero

    /**
     * Keeps the previous selection data in TextFieldValue in order to identify what has changed in
     * the new selection, and decide which selection offset (start, end) to follow.
     */
    var previousSelection: TextRange = TextRange.Zero

    var orientation by mutableStateOf(initialOrientation, structuralEqualityPolicy())

    fun update(orientation: Orientation, cursorRect: Rect, containerSize: Int, textFieldSize: Int) {
        val difference = (textFieldSize - containerSize).toFloat()
        maximum = difference

        if (
            cursorRect.left != previousCursorRect.left || cursorRect.top != previousCursorRect.top
        ) {
            val vertical = orientation == Orientation.Vertical
            val cursorStart = if (vertical) cursorRect.top else cursorRect.left
            val cursorEnd = if (vertical) cursorRect.bottom else cursorRect.right
            coerceOffset(cursorStart, cursorEnd, containerSize)
            previousCursorRect = cursorRect
        }
        offset = offset.coerceIn(0f, difference)
        viewportSize = containerSize
    }

    /*@VisibleForTesting*/
    internal fun coerceOffset(cursorStart: Float, cursorEnd: Float, containerSize: Int) {
        val startVisibleBound = offset
        val endVisibleBound = startVisibleBound + containerSize
        val offsetDifference =
            when {
                // make bottom/end of the cursor visible
                //
                // text box
                // +----------------------+
                // |                      |
                // |                      |
                // |          cursor      |
                // |             |        |
                // +-------------|--------+
                //               |
                //
                cursorEnd > endVisibleBound -> cursorEnd - endVisibleBound

                // in rare cases when there's not enough space to fit the whole cursor, prioritise
                // the bottom/end of the cursor
                //
                //             cursor
                // text box      |
                // +-------------|--------+
                // |             |        |
                // +-------------|--------+
                //               |
                //
                cursorStart < startVisibleBound && cursorEnd - cursorStart > containerSize ->
                    cursorEnd - endVisibleBound

                // make top/start of the cursor visible if there's enough space to fit the whole
                // cursor
                //
                //               cursor
                // text box       |
                // +--------------|-------+
                // |              |       |
                // |                      |
                // |                      |
                // |                      |
                // +----------------------+
                //
                cursorStart < startVisibleBound && cursorEnd - cursorStart <= containerSize ->
                    cursorStart - startVisibleBound

                // otherwise keep current offset
                else -> 0f
            }
        offset += offsetDifference
    }

    fun getOffsetToFollow(selection: TextRange): Int {
        return when {
            selection.start != previousSelection.start -> selection.start
            selection.end != previousSelection.end -> selection.end
            else -> selection.min
        }
    }

    companion object {
        val Saver =
            listSaver(
                save = { listOf(it.offset, it.orientation == Orientation.Vertical) },
                restore = { restored ->
                    TextFieldScrollerPosition(
                        if (restored[1] as Boolean) Orientation.Vertical
                        else Orientation.Horizontal,
                        restored[0] as Float
                    )
                }
            )
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy