import color from 'color';
import * as React from 'react';
import {
ColorValue,
I18nManager,
Platform,
StyleSheet,
TextInput as NativeTextInput,
TextStyle,
View,
} from 'react-native';
import { AdornmentSide, AdornmentType } from './Adornment/enums';
import TextInputAdornment, {
getAdornmentConfig,
getAdornmentStyleAdjustmentForNativeInput,
TextInputAdornmentProps,
} from './Adornment/TextInputAdornment';
import {
ADORNMENT_OFFSET,
ADORNMENT_SIZE,
LABEL_WIGGLE_X_OFFSET,
MAXIMIZED_LABEL_FONT_SIZE,
MINIMIZED_LABEL_FONT_SIZE,
} from './constants';
import {
adjustPaddingOut,
calculateInputHeight,
calculateLabelTopPosition,
calculateOutlinedIconAndAffixTopPosition,
calculatePadding,
interpolatePlaceholder,
Padding,
} from './helpers';
import InputLabel from './Label/InputLabel';
import LabelBackground from './Label/LabelBackground';
import type { ChildTextInputProps, RenderProps } from './types';
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;
const TextInputOutlined = ({
disabled = false,
editable = true,
label,
error = false,
selectionColor,
underlineColor: _underlineColor,
outlineColor: customOutlineColor,
activeOutlineColor,
dense,
style,
theme,
render = (props: RenderProps) => ,
multiline = false,
parentState,
innerRef,
onFocus,
forceFocus,
onBlur,
onChangeText,
onLayoutAnimatedText,
onLeftAffixLayoutChange,
onRightAffixLayoutChange,
left,
right,
placeholderTextColor,
...rest
}: ChildTextInputProps) => {
const adornmentConfig = getAdornmentConfig({ left, right });
const { colors, fonts } = theme;
const font = fonts.regular;
const hasActiveOutline = parentState.focused || error;
const {
fontSize: fontSizeStyle,
fontWeight,
lineHeight,
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) {
const isTransparent = color(customOutlineColor).alpha() === 0;
inputTextColor = activeColor = color(colors.text)
.alpha(0.54)
.rgb()
.string();
placeholderColor = colors.disabled;
outlineColor = isTransparent ? customOutlineColor : colors.disabled;
} else {
inputTextColor = colors.text;
activeColor = error ? colors.error : activeOutlineColor || colors.primary;
placeholderColor = colors.placeholder;
outlineColor = customOutlineColor || 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,
lineHeight,
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: backgroundColor as ColorValue,
errorColor,
labelTranslationXOffset,
roundness: theme.roundness,
maxFontSizeMultiplier: rest.maxFontSizeMultiplier,
};
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,
maxFontSizeMultiplier: rest.maxFontSizeMultiplier,
};
if (adornmentConfig.length) {
adornmentProps = {
...adornmentProps,
left,
right,
textStyle: { ...font, fontSize, fontWeight },
visible: 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?.({
testID: 'text-input-outlined',
...rest,
ref: innerRef,
onChangeText,
placeholder: label ? parentState.placeholder : rest.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',
},
Platform.OS === 'web' && { outline: 'none' },
adornmentStyleAdjustmentForNativeInput,
],
} as RenderProps)}
);
};
export default TextInputOutlined;
type OutlineProps = {
activeColor: string;
hasActiveOutline?: boolean;
focused?: boolean;
outlineColor?: string;
backgroundColor: ColorValue;
theme: ReactNativePaper.Theme;
};
const Outline = ({
theme,
hasActiveOutline,
activeColor,
outlineColor,
focused,
backgroundColor,
}: OutlineProps) => (
);
const styles = StyleSheet.create({
placeholder: {
position: 'absolute',
left: 0,
paddingHorizontal: INPUT_PADDING_HORIZONTAL,
},
outline: {
position: 'absolute',
left: 0,
right: 0,
top: 6,
bottom: 0,
},
labelContainer: {
paddingBottom: 0,
},
input: {
flexGrow: 1,
paddingHorizontal: INPUT_PADDING_HORIZONTAL,
margin: 0,
zIndex: 1,
},
inputOutlined: {
paddingTop: 8,
paddingBottom: 8,
},
inputOutlinedDense: {
paddingTop: 4,
paddingBottom: 4,
},
});