import React, { forwardRef, ChangeEvent } from 'react' import classnames from 'classnames' import { globalProps, GlobalProps, domSafeProps } from '../utilities/globalProps' import { buildAriaProps, buildDataProps, buildHtmlProps } from '../utilities/props' import Flex from '../pb_flex/_flex' import Card from '../pb_card/_card' import Caption from '../pb_caption/_caption' import Body from '../pb_body/_body' import Icon from '../pb_icon/_icon' import { INPUTMASKS } from './inputMask' type TextInputProps = { aria?: { [key: string]: string }, className?: string, data?: { [key: string]: string }, dark?: boolean, disabled?: boolean, error?: string, htmlOptions?: {[key: string]: string | number | boolean | (() => void)}, id?: string, inline?: boolean, name: string, label: string, mask?: 'currency' | 'zipCode' | 'postalCode' | 'ssn', onChange: (e: React.FormEvent) => void, placeholder: string, required?: boolean, type: string, value: string | number, children: React.ReactElement, addOn?: { icon?: string, alignment?: "right" | "left", border?: boolean, }, } & GlobalProps const TextInput = (props: TextInputProps, ref: React.LegacyRef) => { const { addOn = { icon: null, alignment: 'right', border: true }, aria = {}, className, dark = false, data = {}, disabled, error, htmlOptions = {}, id, inline = false, mask = null, name, label, onChange = () => { void 0 }, placeholder, required, type = 'text', value = '', children = null, } = props const ariaProps = buildAriaProps(aria) const dataProps = buildDataProps(data) const htmlProps = buildHtmlProps(htmlOptions) const filteredProps = {...props} if (filteredProps.marginBottom === undefined) { filteredProps.marginBottom = "sm" } const { alignment, border, icon } = addOn const addOnAlignment = alignment === 'left' ? 'left' : 'right' const borderToChange = addOnAlignment === 'left' ? 'right' : 'left' const borderToggle = border === false ? 'off' : 'on' const borderCss = `border_${borderToChange}_${borderToggle}` const shouldShowAddOn = icon !== null const addOnCss = shouldShowAddOn ? 'text_input_wrapper_add_on' : "" const addOnDarkModeCardCss = (shouldShowAddOn && dark) ? 'add-on-card-dark' : "" const css = classnames([ 'pb_text_input_kit', inline ? 'inline' : "", error ? 'error' : "", globalProps(filteredProps), className, ]) const addOnIcon = ( ) const isMaskedInput = mask && mask in INPUTMASKS const handleChange = (e: ChangeEvent) => { if (isMaskedInput) { const inputValue = e.target.value let cursorPosition = e.target.selectionStart; const isAtEnd = cursorPosition === inputValue.length; const formattedValue = INPUTMASKS[mask].format(inputValue) e.target.value = formattedValue // Keep cursor position if (!isAtEnd) { // Account for extra characters (e.g., commas added/removed in currency) if (formattedValue.length - inputValue.length === 1) { cursorPosition = cursorPosition + 1 } else if (mask === "currency" && formattedValue.length - inputValue.length === -1) { cursorPosition = cursorPosition - 1 } e.target.selectionStart = e.target.selectionEnd = cursorPosition } } onChange(e) } const childInput = children ? children.type === "input" : undefined let formattedValue; if (isMaskedInput) { formattedValue = INPUTMASKS[mask].format(value.toString()) } else { formattedValue = value } const textInput = ( childInput ? React.cloneElement(children, { className: "text_input" }) : () ) const addOnInput = ( {addOnAlignment == 'left' && <> {addOnIcon} {textInput} } {addOnAlignment != 'left' && <> {textInput} {addOnIcon} } ) const render = (() => { if (children && !childInput) return children if (shouldShowAddOn) return addOnInput return textInput })() return (
{label && }
{render} {error && }
) } export default forwardRef(TextInput)