import * as React from "react"; import { View, TextInput as NativeTextInput, StyleSheet, I18nManager, Platform, TextStyle, } from "react-native"; import color from "color"; import TextInputAdornment, { getAdornmentConfig, getAdornmentStyleAdjustmentForNativeInput, TextInputAdornmentProps, } from "./Adornment/TextInputAdornment"; import InputLabel from "./Label/InputLabel"; import LabelBackground from "./Label/LabelBackground"; import type { RenderProps, ChildTextInputProps } from "./types"; import { MAXIMIZED_LABEL_FONT_SIZE, MINIMIZED_LABEL_FONT_SIZE, LABEL_WIGGLE_X_OFFSET, ADORNMENT_SIZE, ADORNMENT_OFFSET, } from "./constants"; import { calculateLabelTopPosition, calculateInputHeight, calculatePadding, adjustPaddingOut, Padding, interpolatePlaceholder, calculateOutlinedIconAndAffixTopPosition, } from "./helpers"; import { AdornmentType, AdornmentSide } from "./Adornment/enums"; import { DefaultTheme, ThemeContext } from "styled-components"; const OUTLINE_MINIMIZED_LABEL_Y_OFFSET = -6; const LABEL_PADDING_TOP = 8; const MIN_HEIGHT = 64; const MIN_DENSE_HEIGHT = 48; const INPUT_PADDING_HORIZONTAL = 14; class TextInputOutlined extends React.Component { static contextType = ThemeContext as any; static defaultProps = { disabled: false, error: false, multiline: false, editable: true, render: (props: RenderProps) => , }; render() { const { disabled, editable, label, error, selectionColor, // eslint-disable-next-line @typescript-eslint/no-unused-vars underlineColor, dense, style, render, multiline, parentState, innerRef, onFocus, forceFocus, onBlur, onChangeText, onLayoutAnimatedText, onLeftAffixLayoutChange, onRightAffixLayoutChange, left, right, placeholderTextColor, ...rest } = this.props; const theme = this.context; const adornmentConfig = getAdornmentConfig({ left, right }); const { colors, fonts } = theme; const font = fonts.regular; const hasActiveOutline = parentState.focused || error; const { fontSize: fontSizeStyle, fontWeight, height, backgroundColor = colors.background, textAlign, ...viewStyle } = (StyleSheet.flatten(style) || {}) as TextStyle; const fontSize = fontSizeStyle || MAXIMIZED_LABEL_FONT_SIZE; let inputTextColor, activeColor, outlineColor, placeholderColor, errorColor; if (disabled) { inputTextColor = activeColor = color(colors.text) .alpha(0.54) .rgb() .string(); placeholderColor = outlineColor = colors.disabled; } else { inputTextColor = colors.text; activeColor = error ? colors.error : colors.primary; placeholderColor = outlineColor = colors.placeholder; errorColor = colors.error; } const labelScale = MINIMIZED_LABEL_FONT_SIZE / fontSize; const fontScale = MAXIMIZED_LABEL_FONT_SIZE / fontSize; const labelWidth = parentState.labelLayout.width; const labelHeight = parentState.labelLayout.height; const labelHalfWidth = labelWidth / 2; const labelHalfHeight = labelHeight / 2; const baseLabelTranslateX = (I18nManager.isRTL ? 1 : -1) * (labelHalfWidth - (labelScale * labelWidth) / 2 - (fontSize - MINIMIZED_LABEL_FONT_SIZE) * labelScale); let labelTranslationXOffset = 0; const isAdornmentLeftIcon = adornmentConfig.some( ({ side, type }) => side === AdornmentSide.Left && type === AdornmentType.Icon ); if (isAdornmentLeftIcon) { labelTranslationXOffset = (I18nManager.isRTL ? -1 : 1) * (ADORNMENT_SIZE + ADORNMENT_OFFSET - 8); } const minInputHeight = (dense ? MIN_DENSE_HEIGHT : MIN_HEIGHT) - LABEL_PADDING_TOP; const inputHeight = calculateInputHeight( labelHeight, height, minInputHeight ); const topPosition = calculateLabelTopPosition( labelHeight, inputHeight, LABEL_PADDING_TOP ); if (height && typeof height !== "number") { // eslint-disable-next-line console.warn('Currently we support only numbers in height prop'); } const paddingSettings = { height: height ? +height : null, labelHalfHeight, offset: LABEL_PADDING_TOP, multiline: multiline ? multiline : null, dense: dense ? dense : null, topPosition, fontSize, label, scale: fontScale, isAndroid: Platform.OS === "android", styles: StyleSheet.flatten( dense ? styles.inputOutlinedDense : styles.inputOutlined ) as Padding, }; const pad = calculatePadding(paddingSettings); const paddingOut = adjustPaddingOut({ ...paddingSettings, pad }); const baseLabelTranslateY = -labelHalfHeight - (topPosition + OUTLINE_MINIMIZED_LABEL_Y_OFFSET); const placeholderOpacity = hasActiveOutline ? interpolatePlaceholder(parentState.labeled, hasActiveOutline) : parentState.labelLayout.measured ? 1 : 0; const labelProps = { label, onLayoutAnimatedText, placeholderOpacity, error, placeholderStyle: styles.placeholder, baseLabelTranslateY, baseLabelTranslateX, font, fontSize, fontWeight, labelScale, wiggleOffsetX: LABEL_WIGGLE_X_OFFSET, topPosition, hasActiveOutline, activeColor, placeholderColor, backgroundColor, errorColor, labelTranslationXOffset, }; const minHeight = (height || (dense ? MIN_DENSE_HEIGHT : MIN_HEIGHT)) as number; const { leftLayout, rightLayout } = parentState; const leftAffixTopPosition = calculateOutlinedIconAndAffixTopPosition({ height: minHeight, affixHeight: leftLayout.height || 0, labelYOffset: -OUTLINE_MINIMIZED_LABEL_Y_OFFSET, }); const rightAffixTopPosition = calculateOutlinedIconAndAffixTopPosition({ height: minHeight, affixHeight: rightLayout.height || 0, labelYOffset: -OUTLINE_MINIMIZED_LABEL_Y_OFFSET, }); const iconTopPosition = calculateOutlinedIconAndAffixTopPosition({ height: minHeight, affixHeight: ADORNMENT_SIZE, labelYOffset: -OUTLINE_MINIMIZED_LABEL_Y_OFFSET, }); const rightAffixWidth = right ? rightLayout.width || ADORNMENT_SIZE : ADORNMENT_SIZE; const leftAffixWidth = left ? leftLayout.width || ADORNMENT_SIZE : ADORNMENT_SIZE; const adornmentStyleAdjustmentForNativeInput = getAdornmentStyleAdjustmentForNativeInput( { adornmentConfig, rightAffixWidth, leftAffixWidth, mode: "outlined", } ); const affixTopPosition = { [AdornmentSide.Left]: leftAffixTopPosition, [AdornmentSide.Right]: rightAffixTopPosition, }; const onAffixChange = { [AdornmentSide.Left]: onLeftAffixLayoutChange, [AdornmentSide.Right]: onRightAffixLayoutChange, }; let adornmentProps: TextInputAdornmentProps = { adornmentConfig, forceFocus, topPosition: { [AdornmentType.Icon]: iconTopPosition, [AdornmentType.Affix]: affixTopPosition, }, onAffixChange, isTextInputFocused: parentState.focused, }; if (adornmentConfig.length) { adornmentProps = { ...adornmentProps, left, right, textStyle: { ...font, fontSize, fontWeight }, visible: this.props.parentState.labeled, }; } return ( {/* Render the outline separately from the container This is so that the label can overlap the outline Otherwise the border will cut off the label on Android */} {render?.({ ...rest, ref: innerRef, onChangeText, placeholder: label ? parentState.placeholder : this.props.placeholder, placeholderTextColor: placeholderTextColor || placeholderColor, editable: !disabled && editable, selectionColor: typeof selectionColor === "undefined" ? activeColor : selectionColor, onFocus, onBlur, underlineColorAndroid: "transparent", multiline, style: [ styles.input, !multiline || (multiline && height) ? { height: inputHeight } : {}, paddingOut, { ...font, fontSize, fontWeight, color: inputTextColor, textAlignVertical: multiline ? "top" : "center", textAlign: textAlign ? textAlign : I18nManager.isRTL ? "right" : "left", }, adornmentStyleAdjustmentForNativeInput, ], } as RenderProps)} ); } } export default TextInputOutlined; type OutlineType = { activeColor: string; hasActiveOutline?: boolean; outlineColor?: string; backgroundColor?: string; theme: DefaultTheme; }; const Outline = ({ theme, hasActiveOutline, activeColor, outlineColor, backgroundColor, }: OutlineType) => ( ); const styles = StyleSheet.create({ placeholder: { position: "absolute", left: 0, paddingHorizontal: INPUT_PADDING_HORIZONTAL, }, outline: { position: "absolute", left: 0, right: 0, top: 6, bottom: 0, }, input: { flexGrow: 1, paddingHorizontal: INPUT_PADDING_HORIZONTAL, margin: 0, zIndex: 1, }, inputOutlined: { paddingTop: 8, paddingBottom: 8, }, inputOutlinedDense: { paddingTop: 4, paddingBottom: 4, }, });