You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
196 lines
5.0 KiB
TypeScript
196 lines
5.0 KiB
TypeScript
import { ChangeEvent, ReactNode, useEffect, useState } from 'react';
|
|
|
|
import { motion } from 'framer-motion';
|
|
import { isEmpty, isEqual } from 'lodash';
|
|
import styled from 'styled-components';
|
|
import { THEME_GLOBALS } from '../../themes/globals';
|
|
import { Noop } from '../../types/Util';
|
|
import { useHTMLDirection } from '../../util/i18n';
|
|
import { Flex } from '../basic/Flex';
|
|
import { SpacerMD } from '../basic/Text';
|
|
import { SessionIconButton } from '../icon';
|
|
|
|
const StyledInputContainer = styled(Flex)<{ error: boolean }>`
|
|
position: relative;
|
|
width: 100%;
|
|
|
|
label {
|
|
color: var(--text-primary-color);
|
|
opacity: 0;
|
|
transition: opacity var(--default-duration);
|
|
|
|
&.filled {
|
|
opacity: 1;
|
|
}
|
|
|
|
&.error {
|
|
color: var(--danger-color);
|
|
font-weight: 700;
|
|
}
|
|
}
|
|
|
|
input::placeholder {
|
|
transition: opacity var(--default-duration) color var(--default-duration);
|
|
${props => props.error && `color: var(--danger-color); opacity: 1;`}
|
|
}
|
|
`;
|
|
|
|
const StyledInput = styled(motion.input)`
|
|
border: 1px solid var(--input-border-color);
|
|
border-radius: 13px;
|
|
outline: 0;
|
|
width: 100%;
|
|
background: transparent;
|
|
color: var(--input-text-color);
|
|
|
|
font-family: var(--font-default);
|
|
font-size: 12px;
|
|
line-height: 14px;
|
|
padding: var(--margins-lg);
|
|
|
|
&::placeholder {
|
|
color: var(--input-text-placeholder-color);
|
|
}
|
|
`;
|
|
|
|
const ErrorItem = (props: { id: string; error: string }) => {
|
|
return (
|
|
<motion.label
|
|
htmlFor={props.id}
|
|
className={'filled error'}
|
|
initial={{ opacity: 0 }}
|
|
animate={{ opacity: 1 }}
|
|
transition={{ duration: THEME_GLOBALS['--default-duration-seconds'] }}
|
|
>
|
|
{props.error}
|
|
</motion.label>
|
|
);
|
|
};
|
|
|
|
const ShowHideButton = (props: { toggleForceShow: Noop }) => {
|
|
const htmlDirection = useHTMLDirection();
|
|
const position =
|
|
htmlDirection === 'ltr' ? { right: 'var(--margins-md)' } : { left: 'var(--margins-md)' };
|
|
|
|
return (
|
|
<SessionIconButton
|
|
iconType="eye"
|
|
iconSize="medium"
|
|
onClick={props.toggleForceShow}
|
|
style={{ position: 'absolute', top: '50%', transform: 'translateY(-50%)', ...position }}
|
|
/>
|
|
);
|
|
};
|
|
|
|
const StyledCtaContainer = styled(motion.div)`
|
|
width: 100%;
|
|
`;
|
|
|
|
type Props = {
|
|
error?: string;
|
|
type?: string;
|
|
value?: string;
|
|
placeholder: string;
|
|
maxLength?: number;
|
|
enableShowHide?: boolean;
|
|
onValueChanged?: (value: string) => any;
|
|
onEnterPressed?: (value: string) => any;
|
|
autoFocus?: boolean;
|
|
ref?: any;
|
|
inputDataTestId?: string;
|
|
id?: string;
|
|
ctaButton?: ReactNode;
|
|
};
|
|
|
|
export const SessionInput = (props: Props) => {
|
|
const {
|
|
autoFocus,
|
|
placeholder,
|
|
type = 'text',
|
|
value,
|
|
maxLength,
|
|
enableShowHide,
|
|
error,
|
|
onValueChanged,
|
|
inputDataTestId,
|
|
id = 'session-input-floating-label',
|
|
ctaButton,
|
|
} = props;
|
|
const [inputValue, setInputValue] = useState('');
|
|
const [errorString, setErrorString] = useState('');
|
|
const [forceShow, setForceShow] = useState(false);
|
|
|
|
const correctType = forceShow ? 'text' : type;
|
|
|
|
const updateInputValue = (e: ChangeEvent<HTMLInputElement>) => {
|
|
e.preventDefault();
|
|
const val = e.target.value;
|
|
setInputValue(val);
|
|
if (onValueChanged) {
|
|
onValueChanged(val);
|
|
}
|
|
};
|
|
|
|
// if we have an error, we want to show it even if the input changes to a valid value
|
|
useEffect(() => {
|
|
if (error && !isEmpty(error) && !isEqual(error, errorString)) {
|
|
setErrorString(error);
|
|
}
|
|
}, [error, errorString]);
|
|
|
|
return (
|
|
<StyledInputContainer
|
|
container={true}
|
|
flexDirection="column"
|
|
justifyContent="center"
|
|
alignItems="center"
|
|
error={Boolean(errorString)}
|
|
>
|
|
<StyledInput
|
|
id={id}
|
|
type={correctType}
|
|
placeholder={placeholder}
|
|
value={value}
|
|
maxLength={maxLength}
|
|
autoFocus={autoFocus}
|
|
data-testid={inputDataTestId}
|
|
onChange={updateInputValue}
|
|
style={{ paddingInlineEnd: enableShowHide ? '30px' : undefined }}
|
|
// just in case onChange isn't triggered
|
|
onBlur={updateInputValue}
|
|
onKeyDown={event => {
|
|
if (event.key === 'Enter' && props.onEnterPressed) {
|
|
props.onEnterPressed(inputValue);
|
|
setErrorString('');
|
|
}
|
|
}}
|
|
initial={{
|
|
borderColor: errorString ? 'var(--input-border-color)' : undefined,
|
|
}}
|
|
animate={{
|
|
borderColor: errorString ? 'var(--danger-color)' : undefined,
|
|
}}
|
|
transition={{ duration: THEME_GLOBALS['--default-duration-seconds'] }}
|
|
/>
|
|
|
|
{enableShowHide && (
|
|
<ShowHideButton
|
|
toggleForceShow={() => {
|
|
setForceShow(!forceShow);
|
|
}}
|
|
/>
|
|
)}
|
|
|
|
{ctaButton || errorString ? <SpacerMD /> : null}
|
|
{errorString ? <ErrorItem id={id} error={errorString} /> : null}
|
|
|
|
<StyledCtaContainer
|
|
initial={{ y: errorString && ctaButton ? 0 : undefined }}
|
|
animate={{ y: errorString && ctaButton ? 'var(--margins-md)' : undefined }}
|
|
>
|
|
{ctaButton}
|
|
</StyledCtaContainer>
|
|
</StyledInputContainer>
|
|
);
|
|
};
|