commonMain.com.lt.compose_views.text_field.GoodTextField.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of ComposeViews-desktop Show documentation
Show all versions of ComposeViews-desktop Show documentation
Jatpack(JetBrains) Compose views
The newest version!
/*
* Copyright lt 2023
*
* 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 com.lt.compose_views.text_field
import androidx.compose.foundation.interaction.MutableInteractionSource
import androidx.compose.foundation.layout.*
import androidx.compose.foundation.text.BasicTextField
import androidx.compose.foundation.text.KeyboardActions
import androidx.compose.foundation.text.KeyboardOptions
import androidx.compose.material.LocalTextStyle
import androidx.compose.material.MaterialTheme
import androidx.compose.runtime.Composable
import androidx.compose.runtime.remember
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.graphics.Brush
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.graphics.SolidColor
import androidx.compose.ui.text.TextLayoutResult
import androidx.compose.ui.text.TextStyle
import androidx.compose.ui.text.input.TextFieldValue
import androidx.compose.ui.text.input.VisualTransformation
import androidx.compose.ui.unit.Dp
import androidx.compose.ui.unit.TextUnit
import androidx.compose.ui.unit.dp
import androidx.compose.ui.unit.sp
import com.lt.compose_views.other.HorizontalSpace
import com.lt.compose_views.util.Color333
/**
* creator: lt 2022/7/16 [email protected]
* effect : 更方便易用的TextField(文本输入框)
* More convenient and easy to use the [TextField]
* warning:
* @param value 输入框中的文字
* Text of the [TextField]
* @param onValueChange 输入框中文字的变化回调
* Text change of the [TextField]
* @param modifier 修饰
* @param hint 输入框没有文字时展示的内容
* Content of the [TextField] with if value is Empty
* @param maxLines 最多能展示多少行文字
* How many lines of text can be displayed
* @param fontSize text和hint的字体大小
* Font size of text and hint
* @param fontColor text的字体颜色
* Color of text
* @param maxLength 最多能展示多少个文字,ps:由于会截断文字,会导致截断时重置键盘状态(TextField特性)
* How many texts can be displayed at most
* @param contentAlignment text和hint对其方式
* Text and hint to the way
* @param leading 展示在左边的组件
* Components displayed on the start
* @param trailing 展示在右边的组件
* Components displayed on the end
* @param background 背景
* The background
* @param horizontalPadding 横向的内间距
* Horizontal inner spacing
* @param enabled 是否可输入,false无法输入和复制
* Is it possible to enter
* @param readOnly 是否可输入,true无法输入,但可复制,获取焦点,移动光标
* Read-only
* @param textStyle 字体样式
* The [TextStyle]
* @param keyboardOptions 键盘配置
* Reference the [BasicTextField]
* @param keyboardActions 键盘回调
* Reference the [BasicTextField]
* @param visualTransformation 文本展示的转换
* Reference the [BasicTextField]
* @param onTextLayout 计算新文本布局时执行的回调
* Reference the [BasicTextField]
* @param interactionSource 状态属性
* Reference the [BasicTextField]
* @param cursorBrush 光标绘制
* Reference the [BasicTextField]
*/
@Composable
fun GoodTextField(
value: String,
onValueChange: (String) -> Unit,
modifier: Modifier = Modifier,
hint: HintComposeWithTextField? = null,
maxLines: Int = 1,
fontSize: TextUnit = 16.sp,
fontColor: Color = Color333,
maxLength: Int = Int.MAX_VALUE,
contentAlignment: Alignment.Vertical = Alignment.CenterVertically,
leading: (@Composable RowScope.() -> Unit)? = null,
trailing: (@Composable RowScope.() -> Unit)? = null,
background: BackgroundComposeWithTextField? = BackgroundComposeWithTextField.DEFAULT,
horizontalPadding: Dp = 16.dp,
enabled: Boolean = true,
readOnly: Boolean = false,
textStyle: TextStyle = LocalTextStyle.current,
keyboardOptions: KeyboardOptions = KeyboardOptions.Default,
keyboardActions: KeyboardActions = KeyboardActions.Default,
visualTransformation: VisualTransformation = VisualTransformation.None,
onTextLayout: (TextLayoutResult) -> Unit = {},
interactionSource: MutableInteractionSource = remember { MutableInteractionSource() },
cursorBrush: Brush = SolidColor(MaterialTheme.colors.primary),
) {
//使用基础输入框
BasicTextField(
value = value,
onValueChange = {
val text = if (it.length > maxLength) {
//使value不超过maxLength
it.substring(0, maxLength)
} else if (maxLines == 1 && it.contains('\n')) {
//处理特殊情况下单行输入框能输入换行符
it.replace("\n", "")
} else {
it
}
onValueChange(text)
},
textStyle = textStyle.copy(color = fontColor, fontSize = fontSize),
singleLine = maxLines == 1,
maxLines = maxLines,
modifier = modifier.height(IntrinsicSize.Min),
enabled = enabled,
readOnly = readOnly,
keyboardOptions = keyboardOptions,
keyboardActions = keyboardActions,
visualTransformation = visualTransformation,
onTextLayout = onTextLayout,
interactionSource = interactionSource,
cursorBrush = cursorBrush,
decorationBox = {
GoodTextFieldContent(
background,
horizontalPadding,
leading,
contentAlignment,
{ value.isEmpty() },
hint,
fontSize,
it,
trailing
)
}
)
}
/**
* 更方便易用的TextField(文本输入框)
* More convenient and easy to use the [TextField]
*
* @param value 输入框中的文字
* Text of the [TextField]
* @param onValueChange 输入框中文字的变化回调
* Text change of the [TextField]
* @param modifier 修饰
* @param hint 输入框没有文字时展示的内容
* Content of the [TextField] with if value is Empty
* @param maxLines 最多能展示多少行文字
* How many lines of text can be displayed
* @param fontSize text和hint的字体大小
* Font size of text and hint
* @param fontColor text的字体颜色
* Color of text
* @param maxLength 最多能展示多少个文字,ps:由于会截断文字,会导致截断时重置键盘状态(TextField特性)
* How many texts can be displayed at most
* @param contentAlignment text和hint对其方式
* Text and hint to the way
* @param leading 展示在左边的组件
* Components displayed on the start
* @param trailing 展示在右边的组件
* Components displayed on the end
* @param background 背景
* The background
* @param horizontalPadding 横向的内间距
* Horizontal inner spacing
* @param enabled 是否可输入,false无法输入和复制
* Is it possible to enter
* @param readOnly 是否可输入,true无法输入,但可复制,获取焦点,移动光标
* Read-only
* @param textStyle 字体样式
* The [TextStyle]
* @param keyboardOptions 键盘配置
* Reference the [BasicTextField]
* @param keyboardActions 键盘回调
* Reference the [BasicTextField]
* @param visualTransformation 文本展示的转换
* Reference the [BasicTextField]
* @param onTextLayout 计算新文本布局时执行的回调
* Reference the [BasicTextField]
* @param interactionSource 状态属性
* Reference the [BasicTextField]
* @param cursorBrush 光标绘制
* Reference the [BasicTextField]
*/
@Composable
fun GoodTextField(
value: TextFieldValue,
onValueChange: (TextFieldValue) -> Unit,
modifier: Modifier = Modifier,
hint: HintComposeWithTextField? = null,
maxLines: Int = 1,
fontSize: TextUnit = 16.sp,
fontColor: Color = Color333,
maxLength: Int = Int.MAX_VALUE,
contentAlignment: Alignment.Vertical = Alignment.CenterVertically,
leading: (@Composable RowScope.() -> Unit)? = null,
trailing: (@Composable RowScope.() -> Unit)? = null,
background: BackgroundComposeWithTextField? = BackgroundComposeWithTextField.DEFAULT,
horizontalPadding: Dp = 16.dp,
enabled: Boolean = true,
readOnly: Boolean = false,
textStyle: TextStyle = LocalTextStyle.current,
keyboardOptions: KeyboardOptions = KeyboardOptions.Default,
keyboardActions: KeyboardActions = KeyboardActions.Default,
visualTransformation: VisualTransformation = VisualTransformation.None,
onTextLayout: (TextLayoutResult) -> Unit = {},
interactionSource: MutableInteractionSource = remember { MutableInteractionSource() },
cursorBrush: Brush = SolidColor(MaterialTheme.colors.primary),
) {
//使用基础输入框
BasicTextField(
value = value,
onValueChange = {
val text: TextFieldValue = if (it.annotatedString.length > maxLength) {
//使value不超过maxLength
it.copy(
it.annotatedString.substring(0, maxLength)
)
} else if (maxLines == 1 && it.annotatedString.contains('\n')) {
//处理特殊情况下单行输入框能输入换行符
it.copy(
it.annotatedString.replace(Regex("\n"), "")
)
} else {
it
}
onValueChange(text)
},
textStyle = textStyle.copy(color = fontColor, fontSize = fontSize),
singleLine = maxLines == 1,
maxLines = maxLines,
modifier = modifier.height(IntrinsicSize.Min),
enabled = enabled,
readOnly = readOnly,
keyboardOptions = keyboardOptions,
keyboardActions = keyboardActions,
visualTransformation = visualTransformation,
onTextLayout = onTextLayout,
interactionSource = interactionSource,
cursorBrush = cursorBrush,
decorationBox = {
GoodTextFieldContent(
background,
horizontalPadding,
leading,
contentAlignment,
{ value.annotatedString.isEmpty() },
hint,
fontSize,
it,
trailing
)
}
)
}
//放置背景等布局,并放置基础输入框
@Composable
private inline fun GoodTextFieldContent(
background: BackgroundComposeWithTextField?,
horizontalPadding: Dp,
noinline leading: @Composable (RowScope.() -> Unit)?,
contentAlignment: Alignment.Vertical,
valueIsEmpty: () -> Boolean,
hint: HintComposeWithTextField?,
fontSize: TextUnit,
it: @Composable () -> Unit,
noinline trailing: @Composable (RowScope.() -> Unit)?
) {
Box(
Modifier
.let {
background?.setBackground(it) ?: it
}
.padding(horizontal = horizontalPadding),
) {
Row(Modifier.fillMaxSize(), verticalAlignment = Alignment.CenterVertically) {
if (leading != null) {
leading()
HorizontalSpace(horizontalPadding)
}
Box(
Modifier
.weight(1f)
.align(contentAlignment)
.let {
if (contentAlignment != Alignment.CenterVertically)
it.padding(vertical = horizontalPadding / 2)
else
it
}
) {
if (valueIsEmpty() && hint != null)
hint.Hint(fontSize = fontSize)
it()
}
if (trailing != null) {
HorizontalSpace(horizontalPadding)
trailing()
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy