diff --git a/packages/console/src/ds-components/TextInput/NumericInput.module.scss b/packages/console/src/ds-components/TextInput/NumericInput.module.scss new file mode 100644 index 000000000..92f83f569 --- /dev/null +++ b/packages/console/src/ds-components/TextInput/NumericInput.module.scss @@ -0,0 +1,39 @@ +@use '@/scss/underscore' as _; + +.container { + position: absolute; + // The design is 2px off from the edge of the input, while the input has 1px border. + right: 1px; + top: 1px; + bottom: 1px; + + .button { + width: 32px; + height: 16px; + margin: 0; + padding-inline: 0; + padding-block: 0; + padding: 0 _.unit(2); + + &.disabled > svg { + color: var(--color-disabled); + } + + &:not(.disabled):hover { + cursor: pointer; + background: var(--color-hover); + } + + &:not(.disabled):active { + background: var(--color-pressed); + } + + &.up { + border-radius: 6px 6px 2px 2px; + } + + &.down { + border-radius: 2px 2px 6px 6px; + } + } +} diff --git a/packages/console/src/ds-components/TextInput/NumericInput.tsx b/packages/console/src/ds-components/TextInput/NumericInput.tsx new file mode 100644 index 000000000..00dd6ff2b --- /dev/null +++ b/packages/console/src/ds-components/TextInput/NumericInput.tsx @@ -0,0 +1,89 @@ +import classNames from 'classnames'; +import { type ComponentProps } from 'react'; + +import CaretDown from '@/assets/icons/caret-down.svg'; +import CaretUp from '@/assets/icons/caret-up.svg'; +import { onKeyDownHandler } from '@/utils/a11y'; + +import * as styles from './NumericInput.module.scss'; +import TextInput from './index'; + +type ButtonProps = { + className?: string; + onTrigger?: ( + event: React.MouseEvent | React.KeyboardEvent + ) => void; + children: React.ReactNode; + isDisabled?: boolean; +}; + +function Button({ className, onTrigger, children, isDisabled }: ButtonProps) { + return ( +
{ + event.preventDefault(); + if (isDisabled) { + return; + } + onTrigger?.(event); + }} + > + {children} +
+ ); +} + +type Props = Omit, 'type' | 'suffix'> & { + /** The event handler for when the value is incremented by the up button. */ + onValueUp: ButtonProps['onTrigger']; + /** The event handler for when the value is decremented by the down button. */ + onValueDown: ButtonProps['onTrigger']; +}; + +/** A numeric text input with up and down buttons for incrementing and decrementing the value. */ +function NumericInput({ onValueUp, onValueDown, ...props }: Props) { + const isDisabled = Boolean(props.disabled) || Boolean(props.readOnly); + + return ( + + + + + } + /> + ); +} + +export default NumericInput; diff --git a/packages/console/src/ds-components/TextInput/index.module.scss b/packages/console/src/ds-components/TextInput/index.module.scss index 593ce95e9..45ca07550 100644 --- a/packages/console/src/ds-components/TextInput/index.module.scss +++ b/packages/console/src/ds-components/TextInput/index.module.scss @@ -4,6 +4,10 @@ width: _.unit(8); height: _.unit(8); display: none; + + &.visible { + display: block; + } } .hideTextContainerContent { @@ -13,6 +17,7 @@ } .container { + position: relative; display: flex; align-items: center; border-radius: 6px; diff --git a/packages/console/src/ds-components/TextInput/index.tsx b/packages/console/src/ds-components/TextInput/index.tsx index f534516a0..6603b697b 100644 --- a/packages/console/src/ds-components/TextInput/index.tsx +++ b/packages/console/src/ds-components/TextInput/index.tsx @@ -21,7 +21,14 @@ import * as styles from './index.module.scss'; type Props = Omit, 'size'> & { error?: string | boolean; icon?: ReactElement; - suffix?: ReactElement; + /** + * An element to be rendered on the right side of the input. + * By default, the suffix is only visible when the input is focused. + */ + suffix?: ReactElement>; + /** Whether to always show the suffix. */ + // eslint-disable-next-line react/boolean-prop-naming + alwaysShowSuffix?: boolean; isConfidential?: boolean; }; @@ -30,6 +37,7 @@ function TextInput( error, icon, suffix, + alwaysShowSuffix = false, disabled, className, readOnly, @@ -95,7 +103,13 @@ function TextInput( {suffixIcon && cloneElement(suffixIcon, { - className: classNames([suffixIcon.props.className, styles.suffix]), + className: classNames( + // Handle by classNames + // eslint-disable-next-line @typescript-eslint/no-unsafe-argument + suffixIcon.props.className, + styles.suffix, + alwaysShowSuffix && styles.visible + ), })} {Boolean(error) && typeof error === 'string' && ( diff --git a/packages/console/src/pages/SignInExperience/index.tsx b/packages/console/src/pages/SignInExperience/index.tsx index 7dc0151cc..0782ed269 100644 --- a/packages/console/src/pages/SignInExperience/index.tsx +++ b/packages/console/src/pages/SignInExperience/index.tsx @@ -214,7 +214,10 @@ function SignInExperience() { {t('sign_in_exp.tabs.content')} - {t('sign_in_exp.tabs.password_policy')} + {/* Uncomment until all the changes are merged */} + {isCloud && ( + {t('sign_in_exp.tabs.password_policy')} + )} {data && defaultFormData && (
diff --git a/packages/console/src/pages/SignInExperience/tabs/PasswordPolicy/index.module.scss b/packages/console/src/pages/SignInExperience/tabs/PasswordPolicy/index.module.scss index ea51f1d14..809c58fc0 100644 --- a/packages/console/src/pages/SignInExperience/tabs/PasswordPolicy/index.module.scss +++ b/packages/console/src/pages/SignInExperience/tabs/PasswordPolicy/index.module.scss @@ -1,5 +1,10 @@ @use '@/scss/underscore' as _; +.minLength > div[class*='container'] { + // From Figma design + max-width: 156px; +} + .characterTypes { display: flex; gap: _.unit(6); diff --git a/packages/console/src/pages/SignInExperience/tabs/PasswordPolicy/index.tsx b/packages/console/src/pages/SignInExperience/tabs/PasswordPolicy/index.tsx index 9978f9974..9cf208955 100644 --- a/packages/console/src/pages/SignInExperience/tabs/PasswordPolicy/index.tsx +++ b/packages/console/src/pages/SignInExperience/tabs/PasswordPolicy/index.tsx @@ -10,7 +10,7 @@ import Checkbox from '@/ds-components/Checkbox/Checkbox'; import FormField from '@/ds-components/FormField'; import RadioGroup, { Radio } from '@/ds-components/RadioGroup'; import TabWrapper from '@/ds-components/TabWrapper'; -import TextInput from '@/ds-components/TextInput'; +import NumericInput from '@/ds-components/TextInput/NumericInput'; import Textarea from '@/ds-components/Textarea'; import { type SignInExperienceForm } from '../../types'; @@ -62,7 +62,7 @@ function PasswordPolicy({ isActive }: Props) { getValues, formState: { errors }, } = useFormContext(); - const maxPasswordLength = getValues('passwordPolicy.length.max'); + const { max } = getValues('passwordPolicy.length'); const { t } = useTranslation(undefined, { keyPrefix: 'admin_console.sign_in_exp.password_policy', }); @@ -76,27 +76,41 @@ function PasswordPolicy({ isActive }: Props) {
{t('password_requirements')}
- {t('minimum_length_description', { max: maxPasswordLength })} + {t('minimum_length_description', { max })}
( - ) => { onChange(Number(event.target.value)); }} + onValueUp={() => { + onChange(value + 1); + }} + onValueDown={() => { + onChange(value - 1); + }} + onBlur={() => { + if (value < 1) { + onChange(1); + } else if (value > max) { + onChange(max); + } + }} /> )} /> diff --git a/packages/console/src/utils/a11y.ts b/packages/console/src/utils/a11y.ts index 6f65b62c1..96aceb2e5 100644 --- a/packages/console/src/utils/a11y.ts +++ b/packages/console/src/utils/a11y.ts @@ -1,11 +1,11 @@ import type { KeyboardEventHandler, KeyboardEvent } from 'react'; -type callbackHandler = ((event: KeyboardEvent) => void) | undefined; +type CallbackHandler = ((event: KeyboardEvent) => void) | undefined; -type callbackHandlerMap = Record>; +type CallbackHandlerMap = Record>; export const onKeyDownHandler = - (callback?: callbackHandler | callbackHandlerMap): KeyboardEventHandler => + (callback?: CallbackHandler | CallbackHandlerMap): KeyboardEventHandler => (event) => { const { key } = event;