@amboss/design-system 3.30.0 → 3.31.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build/cjs/components/Form/PasswordInput/PasswordInput.d.ts +48 -6
- package/build/cjs/components/Form/PasswordInput/PasswordInput.js +1 -1
- package/build/cjs/components/Form/PasswordInput/PasswordInputButton.d.ts +13 -0
- package/build/cjs/components/Form/PasswordInput/PasswordInputButton.js +1 -0
- package/build/esm/components/Form/PasswordInput/PasswordInput.d.ts +48 -6
- package/build/esm/components/Form/PasswordInput/PasswordInput.js +1 -1
- package/build/esm/components/Form/PasswordInput/PasswordInputButton.d.ts +13 -0
- package/build/esm/components/Form/PasswordInput/PasswordInputButton.js +1 -0
- package/package.json +1 -1
|
@@ -2,7 +2,21 @@ import type { InputHTMLAttributes } from "react";
|
|
|
2
2
|
import React from "react";
|
|
3
3
|
import type { FormFieldProps } from "../FormField/FormField";
|
|
4
4
|
export type PasswordInputProps = {
|
|
5
|
-
|
|
5
|
+
/**
|
|
6
|
+
* Props for sub-components.
|
|
7
|
+
*
|
|
8
|
+
* - `toggleVisibility.ariaLabelShow` — Label for screen readers when the icon can be toggled to show the password
|
|
9
|
+
*
|
|
10
|
+
* - `toggleVisibility.ariaLabelHide` — Label for screen readers when the icon can be toggled to hide the password
|
|
11
|
+
*/
|
|
12
|
+
slotProps: {
|
|
13
|
+
toggleVisibility: {
|
|
14
|
+
/** Label for screen readers when icon can be toggled to show password */
|
|
15
|
+
ariaLabelShow: string;
|
|
16
|
+
/** Label for screen readers when icon can be toggled to hide password */
|
|
17
|
+
ariaLabelHide: string;
|
|
18
|
+
};
|
|
19
|
+
};
|
|
6
20
|
} & InputProps;
|
|
7
21
|
type InputProps = {
|
|
8
22
|
/**
|
|
@@ -10,31 +24,59 @@ type InputProps = {
|
|
|
10
24
|
*/
|
|
11
25
|
hasError?: boolean;
|
|
12
26
|
/**
|
|
13
|
-
*
|
|
27
|
+
* Function for on icon click behavior
|
|
14
28
|
*/
|
|
15
29
|
onIconClick?: (e: React.MouseEvent) => void;
|
|
16
30
|
} & FormFieldProps & Omit<InputHTMLAttributes<HTMLInputElement>, "type">;
|
|
17
31
|
export declare const PasswordInputRaw: React.ForwardRefExoticComponent<{
|
|
18
|
-
|
|
32
|
+
/**
|
|
33
|
+
* Props for sub-components.
|
|
34
|
+
*
|
|
35
|
+
* - `toggleVisibility.ariaLabelShow` — Label for screen readers when the icon can be toggled to show the password
|
|
36
|
+
*
|
|
37
|
+
* - `toggleVisibility.ariaLabelHide` — Label for screen readers when the icon can be toggled to hide the password
|
|
38
|
+
*/
|
|
39
|
+
slotProps: {
|
|
40
|
+
toggleVisibility: {
|
|
41
|
+
/** Label for screen readers when icon can be toggled to show password */
|
|
42
|
+
ariaLabelShow: string;
|
|
43
|
+
/** Label for screen readers when icon can be toggled to hide password */
|
|
44
|
+
ariaLabelHide: string;
|
|
45
|
+
};
|
|
46
|
+
};
|
|
19
47
|
} & {
|
|
20
48
|
/**
|
|
21
49
|
* Set to true to toggle error state
|
|
22
50
|
*/
|
|
23
51
|
hasError?: boolean;
|
|
24
52
|
/**
|
|
25
|
-
*
|
|
53
|
+
* Function for on icon click behavior
|
|
26
54
|
*/
|
|
27
55
|
onIconClick?: (e: React.MouseEvent) => void;
|
|
28
56
|
} & FormFieldProps & Omit<InputHTMLAttributes<HTMLInputElement>, "type"> & React.RefAttributes<HTMLInputElement>>;
|
|
29
57
|
export declare const PasswordInput: React.ForwardRefExoticComponent<{
|
|
30
|
-
|
|
58
|
+
/**
|
|
59
|
+
* Props for sub-components.
|
|
60
|
+
*
|
|
61
|
+
* - `toggleVisibility.ariaLabelShow` — Label for screen readers when the icon can be toggled to show the password
|
|
62
|
+
*
|
|
63
|
+
* - `toggleVisibility.ariaLabelHide` — Label for screen readers when the icon can be toggled to hide the password
|
|
64
|
+
*/
|
|
65
|
+
slotProps: {
|
|
66
|
+
toggleVisibility: {
|
|
67
|
+
/** Label for screen readers when icon can be toggled to show password */
|
|
68
|
+
ariaLabelShow: string;
|
|
69
|
+
/** Label for screen readers when icon can be toggled to hide password */
|
|
70
|
+
ariaLabelHide: string;
|
|
71
|
+
};
|
|
72
|
+
};
|
|
31
73
|
} & {
|
|
32
74
|
/**
|
|
33
75
|
* Set to true to toggle error state
|
|
34
76
|
*/
|
|
35
77
|
hasError?: boolean;
|
|
36
78
|
/**
|
|
37
|
-
*
|
|
79
|
+
* Function for on icon click behavior
|
|
38
80
|
*/
|
|
39
81
|
onIconClick?: (e: React.MouseEvent) => void;
|
|
40
82
|
} & FormFieldProps & Omit<InputHTMLAttributes<HTMLInputElement>, "type"> & React.RefAttributes<HTMLInputElement>>;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports,"__esModule",{value:!0});var target=exports,all={get PasswordInput(){return PasswordInput},get PasswordInputRaw(){return PasswordInputRaw}};for(var name in all)Object.defineProperty(target,name,{enumerable:!0,get:Object.getOwnPropertyDescriptor(all,name).get});const _interop_require_default=require("@swc/helpers/_/_interop_require_default"),_react=/*#__PURE__*/require("@swc/helpers/_/_interop_require_wildcard")._(require("react")),_styled=/*#__PURE__*/_interop_require_default._(require("@emotion/styled")),_Icon=require("../../Icon/Icon"),_FormField=require("../FormField/FormField"),_useFormFieldMessageId=require("../FormField/useFormFieldMessageId"),handleBorderColor=(theme,hasError,readOnly)=>hasError?theme.values.color.border.error.default:readOnly?theme.values.color.border.primary.disabled:theme.values.color.border.primary.default,handleIconBorderColor=(theme,hasError)=>hasError?theme.values.color.icon.error.default:"transparent",handleIconLeftBorderColor=(theme,hasError)=>hasError?theme.values.color.border.error.default:theme.values.color.border.primary.default,StyledContainer=(0,_styled.default)("div",{target:"e1ro5src0",label:"StyledContainer"})(()=>({width:"100%",display:"flex",alignItems:"center",flexDirection:"row",position:"relative"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/PasswordInput/PasswordInput.tsx","sources":["src/components/Form/PasswordInput/PasswordInput.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport type { RefObject, InputHTMLAttributes } from \"react\";\nimport React, { useState, forwardRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { CSSObject, Theme } from \"@emotion/react\";\nimport { Icon } from \"../../Icon/Icon\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { useFormFieldMessageId } from \"../FormField/useFormFieldMessageId\";\n\nexport type PasswordInputProps = {\n  iconAriaLabel: string;\n} & InputProps;\n\ntype InputProps = {\n  /**\n   * Set to true to toggle error state\n   */\n  hasError?: boolean;\n  /**\n   * Set to true to disabled input\n   */\n  onIconClick?: (e: React.MouseEvent) => void;\n} & FormFieldProps &\n  Omit<InputHTMLAttributes<HTMLInputElement>, \"type\">;\n\ntype IconProps = {\n  hasError?: boolean;\n  disabled?: boolean;\n  onChange?: (e: React.FormEvent<HTMLInputElement>) => void;\n  tabIndex?: number;\n};\n\nconst handleBorderColor = (\n  theme: Theme,\n  hasError: boolean,\n  readOnly: boolean\n) => {\n  if (hasError) return theme.values.color.border.error.default;\n  if (readOnly) return theme.values.color.border.primary.disabled;\n\n  return theme.values.color.border.primary.default;\n};\n\nconst handleIconBorderColor = (theme: Theme, hasError: boolean) => {\n  if (hasError) return theme.values.color.icon.error.default;\n  return \"transparent\";\n};\n\nconst handleIconLeftBorderColor = (theme: Theme, hasError: boolean) => {\n  if (hasError) return theme.values.color.border.error.default;\n  return theme.values.color.border.primary.default;\n};\n\nconst StyledContainer = styled.div(\n  () =>\n    ({\n      width: \"100%\",\n      display: \"flex\",\n      alignItems: \"center\",\n      flexDirection: \"row\",\n      position: \"relative\",\n    } as CSSObject)\n);\n\nconst StyledIconBtn = styled.button<IconProps>(({ theme, hasError }) => ({\n  \"&[type='button']\": {\n    appearance: \"none\",\n    MozAppearance: \"none\",\n    WebkitAppearance: \"none\",\n  },\n  position: \"absolute\",\n  top: 0,\n  right: 0,\n  border: \"1px solid\",\n  borderColor: handleIconBorderColor(theme, hasError),\n  borderLeftColor: handleIconLeftBorderColor(theme, hasError),\n  padding: `calc(${theme.variables.size.spacing.s} - 1px) ${theme.variables.size.spacing.s}`,\n  borderRadius: theme.variables.size.borderRadius.xs,\n  borderBottomLeftRadius: \"0\",\n  borderTopLeftRadius: \"0\",\n  color: theme.values.color.text.secondary.default,\n  backgroundColor: \"inherit\",\n  cursor: \"pointer\",\n  \"&:hover\": {\n    backgroundColor: theme.values.color.background.transparent.hover,\n    borderColor: theme.values.color.border.primary.hover,\n  },\n  \"&:active\": {\n    color: theme.values.color.text.primary.default,\n    backgroundColor: theme.values.color.background.transparent.active,\n    borderColor: theme.values.color.border.primary.active,\n  },\n  \"&:disabled\": {\n    backgroundColor: theme.values.color.background.transparent.default,\n    borderLeftColor: theme.values.color.border.primary.disabled,\n    pointerEvents: \"none\",\n  },\n  ...(hasError && {\n    boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n  }),\n  \"& svg\": {\n    pointerEvents: \"none\",\n  },\n}));\n\nconst StyledPasswordInput = styled.input<InputProps>(\n  ({ theme, hasError, readOnly }) => ({\n    fontFamily: theme.variables.fontFamily.lato,\n    fontSize: theme.variables.size.font.m,\n    lineHeight: theme.variables.size.lineHeight.xl,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    borderWidth: \"1px\",\n    borderStyle: \"solid\",\n    width: \"100%\",\n    padding: `calc(${theme.variables.size.spacing.xs} - 1px) ${theme.variables.size.spacing.s}`,\n    paddingRight: theme.variables.size.spacing.xxl,\n    boxSizing: \"border-box\",\n    color: theme.values.color.text.secondary.default,\n    borderColor: handleBorderColor(theme, hasError, readOnly),\n    backgroundColor: theme.values.color.background.primary.default,\n    \"&.error\": {\n      borderColor: theme.values.color.border.error.default,\n    },\n    \"&.has-icon\": {\n      paddingRight:\n        theme.variables.size.spacing.xs + theme.variables.size.dimension.icon.m,\n    },\n    \"&::placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n      opacity: theme.variables.opacity.visible,\n    },\n    \"&:-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    \"&::-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    ...(hasError && {\n      boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n    }),\n  })\n);\n\nexport const PasswordInputRaw = forwardRef(\n  (\n    {\n      placeholder,\n      hasError = false,\n      onIconClick,\n      iconAriaLabel,\n      readOnly,\n      disabled,\n      ...rest\n    }: PasswordInputProps,\n    ref: RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const [inputType, setInputType] = useState(\"password\");\n    const currentType = inputType === \"password\" ? \"text\" : \"password\";\n    const eyeIcon = inputType === \"password\" ? \"eye-off\" : \"eye\";\n\n    return (\n      <StyledContainer>\n        <StyledPasswordInput\n          {...rest}\n          type={inputType}\n          placeholder={!readOnly && placeholder}\n          ref={ref}\n          hasError={hasError}\n          readOnly={readOnly}\n          disabled={disabled}\n          aria-invalid={hasError}\n        />\n        <StyledIconBtn\n          onClick={(e: React.MouseEvent) => {\n            if (disabled) {\n              e.preventDefault();\n            }\n            if (onIconClick) {\n              onIconClick(e);\n            }\n            setInputType(currentType);\n          }}\n          hasError={hasError}\n          disabled={readOnly || disabled}\n          aria-label={iconAriaLabel}\n          type=\"button\"\n        >\n          <Icon\n            name={eyeIcon}\n            size=\"s\"\n            color={readOnly ? \"quaternary\" : \"tertiary\"}\n          />\n        </StyledIconBtn>\n      </StyledContainer>\n    );\n  }\n);\n\nexport const PasswordInput = React.forwardRef(\n  (\n    {\n      placeholder = \"••••••••\",\n      hasError = false,\n      onIconClick,\n      autoComplete = \"on\",\n      iconAriaLabel,\n      disabled,\n      label,\n      labelHint,\n      hideLabel,\n      hint,\n      errorMessages,\n      \"aria-describedby\": ariaDescribedBy,\n      \"data-e2e-test-id\": dataE2eTestId,\n      ...rest\n    }: PasswordInputProps,\n    ref: React.RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const { id, ariaDescribedByProp } = useFormFieldMessageId({\n      ariaDescribedBy,\n      errorMessages,\n      hint,\n    });\n\n    return (\n      <FormField\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"PasswordInput\"\n        disabled={disabled}\n        label={label}\n        labelHint={labelHint}\n        hideLabel={hideLabel}\n        hint={hint}\n        errorMessages={errorMessages}\n        messageId={id}\n      >\n        <PasswordInputRaw\n          {...rest}\n          ref={ref}\n          placeholder={placeholder}\n          hasError={hasError}\n          disabled={disabled}\n          onIconClick={onIconClick}\n          autoComplete={autoComplete}\n          iconAriaLabel={iconAriaLabel}\n          {...ariaDescribedByProp}\n        />\n      </FormField>\n    );\n  }\n);\n"],"names":[],"mappings":"AAuDwB"} */"),StyledIconBtn=(0,_styled.default)("button",{target:"e1ro5src1",label:"StyledIconBtn"})(({theme,hasError})=>({"&[type='button']":{appearance:"none",MozAppearance:"none",WebkitAppearance:"none"},position:"absolute",top:0,right:0,border:"1px solid",borderColor:handleIconBorderColor(theme,hasError),borderLeftColor:handleIconLeftBorderColor(theme,hasError),padding:`calc(${theme.variables.size.spacing.s} - 1px) ${theme.variables.size.spacing.s}`,borderRadius:theme.variables.size.borderRadius.xs,borderBottomLeftRadius:"0",borderTopLeftRadius:"0",color:theme.values.color.text.secondary.default,backgroundColor:"inherit",cursor:"pointer","&:hover":{backgroundColor:theme.values.color.background.transparent.hover,borderColor:theme.values.color.border.primary.hover},"&:active":{color:theme.values.color.text.primary.default,backgroundColor:theme.values.color.background.transparent.active,borderColor:theme.values.color.border.primary.active},"&:disabled":{backgroundColor:theme.values.color.background.transparent.default,borderLeftColor:theme.values.color.border.primary.disabled,pointerEvents:"none"},...hasError&&{boxShadow:`0 0 0 1px inset ${theme.values.color.border.error.default}`},"& svg":{pointerEvents:"none"}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/PasswordInput/PasswordInput.tsx","sources":["src/components/Form/PasswordInput/PasswordInput.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport type { RefObject, InputHTMLAttributes } from \"react\";\nimport React, { useState, forwardRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { CSSObject, Theme } from \"@emotion/react\";\nimport { Icon } from \"../../Icon/Icon\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { useFormFieldMessageId } from \"../FormField/useFormFieldMessageId\";\n\nexport type PasswordInputProps = {\n  iconAriaLabel: string;\n} & InputProps;\n\ntype InputProps = {\n  /**\n   * Set to true to toggle error state\n   */\n  hasError?: boolean;\n  /**\n   * Set to true to disabled input\n   */\n  onIconClick?: (e: React.MouseEvent) => void;\n} & FormFieldProps &\n  Omit<InputHTMLAttributes<HTMLInputElement>, \"type\">;\n\ntype IconProps = {\n  hasError?: boolean;\n  disabled?: boolean;\n  onChange?: (e: React.FormEvent<HTMLInputElement>) => void;\n  tabIndex?: number;\n};\n\nconst handleBorderColor = (\n  theme: Theme,\n  hasError: boolean,\n  readOnly: boolean\n) => {\n  if (hasError) return theme.values.color.border.error.default;\n  if (readOnly) return theme.values.color.border.primary.disabled;\n\n  return theme.values.color.border.primary.default;\n};\n\nconst handleIconBorderColor = (theme: Theme, hasError: boolean) => {\n  if (hasError) return theme.values.color.icon.error.default;\n  return \"transparent\";\n};\n\nconst handleIconLeftBorderColor = (theme: Theme, hasError: boolean) => {\n  if (hasError) return theme.values.color.border.error.default;\n  return theme.values.color.border.primary.default;\n};\n\nconst StyledContainer = styled.div(\n  () =>\n    ({\n      width: \"100%\",\n      display: \"flex\",\n      alignItems: \"center\",\n      flexDirection: \"row\",\n      position: \"relative\",\n    } as CSSObject)\n);\n\nconst StyledIconBtn = styled.button<IconProps>(({ theme, hasError }) => ({\n  \"&[type='button']\": {\n    appearance: \"none\",\n    MozAppearance: \"none\",\n    WebkitAppearance: \"none\",\n  },\n  position: \"absolute\",\n  top: 0,\n  right: 0,\n  border: \"1px solid\",\n  borderColor: handleIconBorderColor(theme, hasError),\n  borderLeftColor: handleIconLeftBorderColor(theme, hasError),\n  padding: `calc(${theme.variables.size.spacing.s} - 1px) ${theme.variables.size.spacing.s}`,\n  borderRadius: theme.variables.size.borderRadius.xs,\n  borderBottomLeftRadius: \"0\",\n  borderTopLeftRadius: \"0\",\n  color: theme.values.color.text.secondary.default,\n  backgroundColor: \"inherit\",\n  cursor: \"pointer\",\n  \"&:hover\": {\n    backgroundColor: theme.values.color.background.transparent.hover,\n    borderColor: theme.values.color.border.primary.hover,\n  },\n  \"&:active\": {\n    color: theme.values.color.text.primary.default,\n    backgroundColor: theme.values.color.background.transparent.active,\n    borderColor: theme.values.color.border.primary.active,\n  },\n  \"&:disabled\": {\n    backgroundColor: theme.values.color.background.transparent.default,\n    borderLeftColor: theme.values.color.border.primary.disabled,\n    pointerEvents: \"none\",\n  },\n  ...(hasError && {\n    boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n  }),\n  \"& svg\": {\n    pointerEvents: \"none\",\n  },\n}));\n\nconst StyledPasswordInput = styled.input<InputProps>(\n  ({ theme, hasError, readOnly }) => ({\n    fontFamily: theme.variables.fontFamily.lato,\n    fontSize: theme.variables.size.font.m,\n    lineHeight: theme.variables.size.lineHeight.xl,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    borderWidth: \"1px\",\n    borderStyle: \"solid\",\n    width: \"100%\",\n    padding: `calc(${theme.variables.size.spacing.xs} - 1px) ${theme.variables.size.spacing.s}`,\n    paddingRight: theme.variables.size.spacing.xxl,\n    boxSizing: \"border-box\",\n    color: theme.values.color.text.secondary.default,\n    borderColor: handleBorderColor(theme, hasError, readOnly),\n    backgroundColor: theme.values.color.background.primary.default,\n    \"&.error\": {\n      borderColor: theme.values.color.border.error.default,\n    },\n    \"&.has-icon\": {\n      paddingRight:\n        theme.variables.size.spacing.xs + theme.variables.size.dimension.icon.m,\n    },\n    \"&::placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n      opacity: theme.variables.opacity.visible,\n    },\n    \"&:-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    \"&::-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    ...(hasError && {\n      boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n    }),\n  })\n);\n\nexport const PasswordInputRaw = forwardRef(\n  (\n    {\n      placeholder,\n      hasError = false,\n      onIconClick,\n      iconAriaLabel,\n      readOnly,\n      disabled,\n      ...rest\n    }: PasswordInputProps,\n    ref: RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const [inputType, setInputType] = useState(\"password\");\n    const currentType = inputType === \"password\" ? \"text\" : \"password\";\n    const eyeIcon = inputType === \"password\" ? \"eye-off\" : \"eye\";\n\n    return (\n      <StyledContainer>\n        <StyledPasswordInput\n          {...rest}\n          type={inputType}\n          placeholder={!readOnly && placeholder}\n          ref={ref}\n          hasError={hasError}\n          readOnly={readOnly}\n          disabled={disabled}\n          aria-invalid={hasError}\n        />\n        <StyledIconBtn\n          onClick={(e: React.MouseEvent) => {\n            if (disabled) {\n              e.preventDefault();\n            }\n            if (onIconClick) {\n              onIconClick(e);\n            }\n            setInputType(currentType);\n          }}\n          hasError={hasError}\n          disabled={readOnly || disabled}\n          aria-label={iconAriaLabel}\n          type=\"button\"\n        >\n          <Icon\n            name={eyeIcon}\n            size=\"s\"\n            color={readOnly ? \"quaternary\" : \"tertiary\"}\n          />\n        </StyledIconBtn>\n      </StyledContainer>\n    );\n  }\n);\n\nexport const PasswordInput = React.forwardRef(\n  (\n    {\n      placeholder = \"••••••••\",\n      hasError = false,\n      onIconClick,\n      autoComplete = \"on\",\n      iconAriaLabel,\n      disabled,\n      label,\n      labelHint,\n      hideLabel,\n      hint,\n      errorMessages,\n      \"aria-describedby\": ariaDescribedBy,\n      \"data-e2e-test-id\": dataE2eTestId,\n      ...rest\n    }: PasswordInputProps,\n    ref: React.RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const { id, ariaDescribedByProp } = useFormFieldMessageId({\n      ariaDescribedBy,\n      errorMessages,\n      hint,\n    });\n\n    return (\n      <FormField\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"PasswordInput\"\n        disabled={disabled}\n        label={label}\n        labelHint={labelHint}\n        hideLabel={hideLabel}\n        hint={hint}\n        errorMessages={errorMessages}\n        messageId={id}\n      >\n        <PasswordInputRaw\n          {...rest}\n          ref={ref}\n          placeholder={placeholder}\n          hasError={hasError}\n          disabled={disabled}\n          onIconClick={onIconClick}\n          autoComplete={autoComplete}\n          iconAriaLabel={iconAriaLabel}\n          {...ariaDescribedByProp}\n        />\n      </FormField>\n    );\n  }\n);\n"],"names":[],"mappings":"AAkEsB"} */"),StyledPasswordInput=(0,_styled.default)("input",{target:"e1ro5src2",label:"StyledPasswordInput"})(({theme,hasError,readOnly})=>({fontFamily:theme.variables.fontFamily.lato,fontSize:theme.variables.size.font.m,lineHeight:theme.variables.size.lineHeight.xl,borderRadius:theme.variables.size.borderRadius.xs,borderWidth:"1px",borderStyle:"solid",width:"100%",padding:`calc(${theme.variables.size.spacing.xs} - 1px) ${theme.variables.size.spacing.s}`,paddingRight:theme.variables.size.spacing.xxl,boxSizing:"border-box",color:theme.values.color.text.secondary.default,borderColor:handleBorderColor(theme,hasError,readOnly),backgroundColor:theme.values.color.background.primary.default,"&.error":{borderColor:theme.values.color.border.error.default},"&.has-icon":{paddingRight:theme.variables.size.spacing.xs+theme.variables.size.dimension.icon.m},"&::placeholder":{fontStyle:"italic",color:theme.values.color.text.tertiary.default,opacity:theme.variables.opacity.visible},"&:-ms-input-placeholder":{fontStyle:"italic",color:theme.values.color.text.tertiary.default},"&::-ms-input-placeholder":{fontStyle:"italic",color:theme.values.color.text.tertiary.default},...hasError&&{boxShadow:`0 0 0 1px inset ${theme.values.color.border.error.default}`}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/PasswordInput/PasswordInput.tsx","sources":["src/components/Form/PasswordInput/PasswordInput.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport type { RefObject, InputHTMLAttributes } from \"react\";\nimport React, { useState, forwardRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { CSSObject, Theme } from \"@emotion/react\";\nimport { Icon } from \"../../Icon/Icon\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { useFormFieldMessageId } from \"../FormField/useFormFieldMessageId\";\n\nexport type PasswordInputProps = {\n  iconAriaLabel: string;\n} & InputProps;\n\ntype InputProps = {\n  /**\n   * Set to true to toggle error state\n   */\n  hasError?: boolean;\n  /**\n   * Set to true to disabled input\n   */\n  onIconClick?: (e: React.MouseEvent) => void;\n} & FormFieldProps &\n  Omit<InputHTMLAttributes<HTMLInputElement>, \"type\">;\n\ntype IconProps = {\n  hasError?: boolean;\n  disabled?: boolean;\n  onChange?: (e: React.FormEvent<HTMLInputElement>) => void;\n  tabIndex?: number;\n};\n\nconst handleBorderColor = (\n  theme: Theme,\n  hasError: boolean,\n  readOnly: boolean\n) => {\n  if (hasError) return theme.values.color.border.error.default;\n  if (readOnly) return theme.values.color.border.primary.disabled;\n\n  return theme.values.color.border.primary.default;\n};\n\nconst handleIconBorderColor = (theme: Theme, hasError: boolean) => {\n  if (hasError) return theme.values.color.icon.error.default;\n  return \"transparent\";\n};\n\nconst handleIconLeftBorderColor = (theme: Theme, hasError: boolean) => {\n  if (hasError) return theme.values.color.border.error.default;\n  return theme.values.color.border.primary.default;\n};\n\nconst StyledContainer = styled.div(\n  () =>\n    ({\n      width: \"100%\",\n      display: \"flex\",\n      alignItems: \"center\",\n      flexDirection: \"row\",\n      position: \"relative\",\n    } as CSSObject)\n);\n\nconst StyledIconBtn = styled.button<IconProps>(({ theme, hasError }) => ({\n  \"&[type='button']\": {\n    appearance: \"none\",\n    MozAppearance: \"none\",\n    WebkitAppearance: \"none\",\n  },\n  position: \"absolute\",\n  top: 0,\n  right: 0,\n  border: \"1px solid\",\n  borderColor: handleIconBorderColor(theme, hasError),\n  borderLeftColor: handleIconLeftBorderColor(theme, hasError),\n  padding: `calc(${theme.variables.size.spacing.s} - 1px) ${theme.variables.size.spacing.s}`,\n  borderRadius: theme.variables.size.borderRadius.xs,\n  borderBottomLeftRadius: \"0\",\n  borderTopLeftRadius: \"0\",\n  color: theme.values.color.text.secondary.default,\n  backgroundColor: \"inherit\",\n  cursor: \"pointer\",\n  \"&:hover\": {\n    backgroundColor: theme.values.color.background.transparent.hover,\n    borderColor: theme.values.color.border.primary.hover,\n  },\n  \"&:active\": {\n    color: theme.values.color.text.primary.default,\n    backgroundColor: theme.values.color.background.transparent.active,\n    borderColor: theme.values.color.border.primary.active,\n  },\n  \"&:disabled\": {\n    backgroundColor: theme.values.color.background.transparent.default,\n    borderLeftColor: theme.values.color.border.primary.disabled,\n    pointerEvents: \"none\",\n  },\n  ...(hasError && {\n    boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n  }),\n  \"& svg\": {\n    pointerEvents: \"none\",\n  },\n}));\n\nconst StyledPasswordInput = styled.input<InputProps>(\n  ({ theme, hasError, readOnly }) => ({\n    fontFamily: theme.variables.fontFamily.lato,\n    fontSize: theme.variables.size.font.m,\n    lineHeight: theme.variables.size.lineHeight.xl,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    borderWidth: \"1px\",\n    borderStyle: \"solid\",\n    width: \"100%\",\n    padding: `calc(${theme.variables.size.spacing.xs} - 1px) ${theme.variables.size.spacing.s}`,\n    paddingRight: theme.variables.size.spacing.xxl,\n    boxSizing: \"border-box\",\n    color: theme.values.color.text.secondary.default,\n    borderColor: handleBorderColor(theme, hasError, readOnly),\n    backgroundColor: theme.values.color.background.primary.default,\n    \"&.error\": {\n      borderColor: theme.values.color.border.error.default,\n    },\n    \"&.has-icon\": {\n      paddingRight:\n        theme.variables.size.spacing.xs + theme.variables.size.dimension.icon.m,\n    },\n    \"&::placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n      opacity: theme.variables.opacity.visible,\n    },\n    \"&:-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    \"&::-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    ...(hasError && {\n      boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n    }),\n  })\n);\n\nexport const PasswordInputRaw = forwardRef(\n  (\n    {\n      placeholder,\n      hasError = false,\n      onIconClick,\n      iconAriaLabel,\n      readOnly,\n      disabled,\n      ...rest\n    }: PasswordInputProps,\n    ref: RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const [inputType, setInputType] = useState(\"password\");\n    const currentType = inputType === \"password\" ? \"text\" : \"password\";\n    const eyeIcon = inputType === \"password\" ? \"eye-off\" : \"eye\";\n\n    return (\n      <StyledContainer>\n        <StyledPasswordInput\n          {...rest}\n          type={inputType}\n          placeholder={!readOnly && placeholder}\n          ref={ref}\n          hasError={hasError}\n          readOnly={readOnly}\n          disabled={disabled}\n          aria-invalid={hasError}\n        />\n        <StyledIconBtn\n          onClick={(e: React.MouseEvent) => {\n            if (disabled) {\n              e.preventDefault();\n            }\n            if (onIconClick) {\n              onIconClick(e);\n            }\n            setInputType(currentType);\n          }}\n          hasError={hasError}\n          disabled={readOnly || disabled}\n          aria-label={iconAriaLabel}\n          type=\"button\"\n        >\n          <Icon\n            name={eyeIcon}\n            size=\"s\"\n            color={readOnly ? \"quaternary\" : \"tertiary\"}\n          />\n        </StyledIconBtn>\n      </StyledContainer>\n    );\n  }\n);\n\nexport const PasswordInput = React.forwardRef(\n  (\n    {\n      placeholder = \"••••••••\",\n      hasError = false,\n      onIconClick,\n      autoComplete = \"on\",\n      iconAriaLabel,\n      disabled,\n      label,\n      labelHint,\n      hideLabel,\n      hint,\n      errorMessages,\n      \"aria-describedby\": ariaDescribedBy,\n      \"data-e2e-test-id\": dataE2eTestId,\n      ...rest\n    }: PasswordInputProps,\n    ref: React.RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const { id, ariaDescribedByProp } = useFormFieldMessageId({\n      ariaDescribedBy,\n      errorMessages,\n      hint,\n    });\n\n    return (\n      <FormField\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"PasswordInput\"\n        disabled={disabled}\n        label={label}\n        labelHint={labelHint}\n        hideLabel={hideLabel}\n        hint={hint}\n        errorMessages={errorMessages}\n        messageId={id}\n      >\n        <PasswordInputRaw\n          {...rest}\n          ref={ref}\n          placeholder={placeholder}\n          hasError={hasError}\n          disabled={disabled}\n          onIconClick={onIconClick}\n          autoComplete={autoComplete}\n          iconAriaLabel={iconAriaLabel}\n          {...ariaDescribedByProp}\n        />\n      </FormField>\n    );\n  }\n);\n"],"names":[],"mappings":"AA2G4B"} */"),PasswordInputRaw=(0,_react.forwardRef)(({placeholder,hasError=!1,onIconClick,iconAriaLabel,readOnly,disabled,...rest},ref)=>{let[inputType,setInputType]=(0,_react.useState)("password"),currentType="password"===inputType?"text":"password";return _react.default.createElement(StyledContainer,null,_react.default.createElement(StyledPasswordInput,{...rest,type:inputType,placeholder:!readOnly&&placeholder,ref:ref,hasError:hasError,readOnly:readOnly,disabled:disabled,"aria-invalid":hasError}),_react.default.createElement(StyledIconBtn,{onClick:e=>{disabled&&e.preventDefault(),onIconClick&&onIconClick(e),setInputType(currentType)},hasError:hasError,disabled:readOnly||disabled,"aria-label":iconAriaLabel,type:"button"},_react.default.createElement(_Icon.Icon,{name:"password"===inputType?"eye-off":"eye",size:"s",color:readOnly?"quaternary":"tertiary"})))}),PasswordInput=_react.default.forwardRef(({placeholder="••••••••",hasError=!1,onIconClick,autoComplete="on",iconAriaLabel,disabled,label,labelHint,hideLabel,hint,errorMessages,"aria-describedby":ariaDescribedBy,"data-e2e-test-id":dataE2eTestId,...rest},ref)=>{let{id,ariaDescribedByProp}=(0,_useFormFieldMessageId.useFormFieldMessageId)({ariaDescribedBy,errorMessages,hint});return _react.default.createElement(_FormField.FormField,{"data-e2e-test-id":dataE2eTestId,"data-ds-id":"PasswordInput",disabled:disabled,label:label,labelHint:labelHint,hideLabel:hideLabel,hint:hint,errorMessages:errorMessages,messageId:id},_react.default.createElement(PasswordInputRaw,{...rest,ref:ref,placeholder:placeholder,hasError:hasError,disabled:disabled,onIconClick:onIconClick,autoComplete:autoComplete,iconAriaLabel:iconAriaLabel,...ariaDescribedByProp}))});
|
|
1
|
+
"use strict";Object.defineProperty(exports,"__esModule",{value:!0});var target=exports,all={get PasswordInput(){return PasswordInput},get PasswordInputRaw(){return PasswordInputRaw}};for(var name in all)Object.defineProperty(target,name,{enumerable:!0,get:Object.getOwnPropertyDescriptor(all,name).get});const _interop_require_default=require("@swc/helpers/_/_interop_require_default"),_react=/*#__PURE__*/require("@swc/helpers/_/_interop_require_wildcard")._(require("react")),_styled=/*#__PURE__*/_interop_require_default._(require("@emotion/styled")),_FormField=require("../FormField/FormField"),_useFormFieldMessageId=require("../FormField/useFormFieldMessageId"),_PasswordInputButton=require("./PasswordInputButton"),handleBorderColor=(theme,hasError,readOnly)=>hasError?theme.values.color.border.error.default:readOnly?theme.values.color.border.primary.disabled:theme.values.color.border.primary.default,StyledContainer=(0,_styled.default)("div",{target:"e3t34vm0",label:"StyledContainer"})(()=>({width:"100%",display:"flex",alignItems:"center",flexDirection:"row",position:"relative"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/PasswordInput/PasswordInput.tsx","sources":["src/components/Form/PasswordInput/PasswordInput.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport type { RefObject, InputHTMLAttributes } from \"react\";\nimport React, { useState, forwardRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { CSSObject, Theme } from \"@emotion/react\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { useFormFieldMessageId } from \"../FormField/useFormFieldMessageId\";\nimport { PasswordInputButton } from \"./PasswordInputButton\";\n\nexport type PasswordInputProps = {\n  /**\n   * Props for sub-components.\n   *\n   * - `toggleVisibility.ariaLabelShow` — Label for screen readers when the icon can be toggled to show the password\n   *\n   * - `toggleVisibility.ariaLabelHide` — Label for screen readers when the icon can be toggled to hide the password\n   */\n  slotProps: {\n    toggleVisibility: {\n      /** Label for screen readers when icon can be toggled to show password */\n      ariaLabelShow: string;\n      /** Label for screen readers when icon can be toggled to hide password */\n      ariaLabelHide: string;\n    };\n  };\n} & InputProps;\n\ntype InputProps = {\n  /**\n   * Set to true to toggle error state\n   */\n  hasError?: boolean;\n  /**\n   * Function for on icon click behavior\n   */\n  onIconClick?: (e: React.MouseEvent) => void;\n} & FormFieldProps &\n  Omit<InputHTMLAttributes<HTMLInputElement>, \"type\">;\n\nconst handleBorderColor = (\n  theme: Theme,\n  hasError: boolean,\n  readOnly: boolean\n) => {\n  if (hasError) return theme.values.color.border.error.default;\n  if (readOnly) return theme.values.color.border.primary.disabled;\n\n  return theme.values.color.border.primary.default;\n};\n\nconst StyledContainer = styled.div(\n  () =>\n    ({\n      width: \"100%\",\n      display: \"flex\",\n      alignItems: \"center\",\n      flexDirection: \"row\",\n      position: \"relative\",\n    } as CSSObject)\n);\n\nconst StyledPasswordInput = styled.input<InputProps>(\n  ({ theme, hasError, readOnly }) => ({\n    fontFamily: theme.variables.fontFamily.lato,\n    fontSize: theme.variables.size.font.m,\n    lineHeight: theme.variables.size.lineHeight.xl,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    borderWidth: \"1px\",\n    borderStyle: \"solid\",\n    width: \"100%\",\n    padding: `calc(${theme.variables.size.spacing.xs} - 1px) ${theme.variables.size.spacing.s}`,\n    paddingRight: theme.variables.size.spacing.xxl,\n    boxSizing: \"border-box\",\n    color: theme.values.color.text.secondary.default,\n    borderColor: handleBorderColor(theme, hasError, readOnly),\n    backgroundColor: theme.values.color.background.primary.default,\n    \"&.error\": {\n      borderColor: theme.values.color.border.error.default,\n    },\n    \"&.has-icon\": {\n      paddingRight:\n        theme.variables.size.spacing.xs + theme.variables.size.dimension.icon.m,\n    },\n    \"&::placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n      opacity: theme.variables.opacity.visible,\n    },\n    \"&:-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    \"&::-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    ...(hasError && {\n      boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n    }),\n  })\n);\n\nexport const PasswordInputRaw = forwardRef(\n  (\n    {\n      placeholder,\n      hasError = false,\n      onIconClick,\n      readOnly,\n      disabled,\n      slotProps: {\n        toggleVisibility: { ariaLabelShow, ariaLabelHide },\n      },\n      ...rest\n    }: PasswordInputProps,\n    ref: RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false);\n    const inputType = isPasswordVisible ? \"text\" : \"password\";\n    const ariaLabel = isPasswordVisible ? ariaLabelHide : ariaLabelShow;\n\n    return (\n      <StyledContainer>\n        <StyledPasswordInput\n          {...rest}\n          type={inputType}\n          placeholder={!readOnly && placeholder}\n          ref={ref}\n          hasError={hasError}\n          readOnly={readOnly}\n          disabled={disabled}\n          aria-invalid={hasError}\n        />\n        <PasswordInputButton\n          hasError={hasError}\n          disabled={disabled}\n          onIconClick={onIconClick}\n          isPasswordVisible={isPasswordVisible}\n          readOnly={readOnly}\n          ariaLabel={ariaLabel}\n          onToggle={() => setIsPasswordVisible((prev) => !prev)}\n        />\n      </StyledContainer>\n    );\n  }\n);\n\nexport const PasswordInput = React.forwardRef(\n  (\n    {\n      placeholder = \"••••••••\",\n      hasError = false,\n      onIconClick,\n      autoComplete = \"on\",\n      slotProps,\n      disabled,\n      label,\n      labelHint,\n      hideLabel,\n      hint,\n      errorMessages,\n      \"aria-describedby\": ariaDescribedBy,\n      \"data-e2e-test-id\": dataE2eTestId,\n      ...rest\n    }: PasswordInputProps,\n    ref: React.RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const { id, ariaDescribedByProp } = useFormFieldMessageId({\n      ariaDescribedBy,\n      errorMessages,\n      hint,\n    });\n\n    return (\n      <FormField\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"PasswordInput\"\n        disabled={disabled}\n        label={label}\n        labelHint={labelHint}\n        hideLabel={hideLabel}\n        hint={hint}\n        errorMessages={errorMessages}\n        messageId={id}\n      >\n        <PasswordInputRaw\n          {...rest}\n          ref={ref}\n          placeholder={placeholder}\n          hasError={hasError}\n          disabled={disabled}\n          onIconClick={onIconClick}\n          autoComplete={autoComplete}\n          slotProps={slotProps}\n          {...ariaDescribedByProp}\n        />\n      </FormField>\n    );\n  }\n);\n"],"names":[],"mappings":"AAoDwB"} */"),StyledPasswordInput=(0,_styled.default)("input",{target:"e3t34vm1",label:"StyledPasswordInput"})(({theme,hasError,readOnly})=>({fontFamily:theme.variables.fontFamily.lato,fontSize:theme.variables.size.font.m,lineHeight:theme.variables.size.lineHeight.xl,borderRadius:theme.variables.size.borderRadius.xs,borderWidth:"1px",borderStyle:"solid",width:"100%",padding:`calc(${theme.variables.size.spacing.xs} - 1px) ${theme.variables.size.spacing.s}`,paddingRight:theme.variables.size.spacing.xxl,boxSizing:"border-box",color:theme.values.color.text.secondary.default,borderColor:handleBorderColor(theme,hasError,readOnly),backgroundColor:theme.values.color.background.primary.default,"&.error":{borderColor:theme.values.color.border.error.default},"&.has-icon":{paddingRight:theme.variables.size.spacing.xs+theme.variables.size.dimension.icon.m},"&::placeholder":{fontStyle:"italic",color:theme.values.color.text.tertiary.default,opacity:theme.variables.opacity.visible},"&:-ms-input-placeholder":{fontStyle:"italic",color:theme.values.color.text.tertiary.default},"&::-ms-input-placeholder":{fontStyle:"italic",color:theme.values.color.text.tertiary.default},...hasError&&{boxShadow:`0 0 0 1px inset ${theme.values.color.border.error.default}`}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/PasswordInput/PasswordInput.tsx","sources":["src/components/Form/PasswordInput/PasswordInput.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport type { RefObject, InputHTMLAttributes } from \"react\";\nimport React, { useState, forwardRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { CSSObject, Theme } from \"@emotion/react\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { useFormFieldMessageId } from \"../FormField/useFormFieldMessageId\";\nimport { PasswordInputButton } from \"./PasswordInputButton\";\n\nexport type PasswordInputProps = {\n  /**\n   * Props for sub-components.\n   *\n   * - `toggleVisibility.ariaLabelShow` — Label for screen readers when the icon can be toggled to show the password\n   *\n   * - `toggleVisibility.ariaLabelHide` — Label for screen readers when the icon can be toggled to hide the password\n   */\n  slotProps: {\n    toggleVisibility: {\n      /** Label for screen readers when icon can be toggled to show password */\n      ariaLabelShow: string;\n      /** Label for screen readers when icon can be toggled to hide password */\n      ariaLabelHide: string;\n    };\n  };\n} & InputProps;\n\ntype InputProps = {\n  /**\n   * Set to true to toggle error state\n   */\n  hasError?: boolean;\n  /**\n   * Function for on icon click behavior\n   */\n  onIconClick?: (e: React.MouseEvent) => void;\n} & FormFieldProps &\n  Omit<InputHTMLAttributes<HTMLInputElement>, \"type\">;\n\nconst handleBorderColor = (\n  theme: Theme,\n  hasError: boolean,\n  readOnly: boolean\n) => {\n  if (hasError) return theme.values.color.border.error.default;\n  if (readOnly) return theme.values.color.border.primary.disabled;\n\n  return theme.values.color.border.primary.default;\n};\n\nconst StyledContainer = styled.div(\n  () =>\n    ({\n      width: \"100%\",\n      display: \"flex\",\n      alignItems: \"center\",\n      flexDirection: \"row\",\n      position: \"relative\",\n    } as CSSObject)\n);\n\nconst StyledPasswordInput = styled.input<InputProps>(\n  ({ theme, hasError, readOnly }) => ({\n    fontFamily: theme.variables.fontFamily.lato,\n    fontSize: theme.variables.size.font.m,\n    lineHeight: theme.variables.size.lineHeight.xl,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    borderWidth: \"1px\",\n    borderStyle: \"solid\",\n    width: \"100%\",\n    padding: `calc(${theme.variables.size.spacing.xs} - 1px) ${theme.variables.size.spacing.s}`,\n    paddingRight: theme.variables.size.spacing.xxl,\n    boxSizing: \"border-box\",\n    color: theme.values.color.text.secondary.default,\n    borderColor: handleBorderColor(theme, hasError, readOnly),\n    backgroundColor: theme.values.color.background.primary.default,\n    \"&.error\": {\n      borderColor: theme.values.color.border.error.default,\n    },\n    \"&.has-icon\": {\n      paddingRight:\n        theme.variables.size.spacing.xs + theme.variables.size.dimension.icon.m,\n    },\n    \"&::placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n      opacity: theme.variables.opacity.visible,\n    },\n    \"&:-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    \"&::-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    ...(hasError && {\n      boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n    }),\n  })\n);\n\nexport const PasswordInputRaw = forwardRef(\n  (\n    {\n      placeholder,\n      hasError = false,\n      onIconClick,\n      readOnly,\n      disabled,\n      slotProps: {\n        toggleVisibility: { ariaLabelShow, ariaLabelHide },\n      },\n      ...rest\n    }: PasswordInputProps,\n    ref: RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false);\n    const inputType = isPasswordVisible ? \"text\" : \"password\";\n    const ariaLabel = isPasswordVisible ? ariaLabelHide : ariaLabelShow;\n\n    return (\n      <StyledContainer>\n        <StyledPasswordInput\n          {...rest}\n          type={inputType}\n          placeholder={!readOnly && placeholder}\n          ref={ref}\n          hasError={hasError}\n          readOnly={readOnly}\n          disabled={disabled}\n          aria-invalid={hasError}\n        />\n        <PasswordInputButton\n          hasError={hasError}\n          disabled={disabled}\n          onIconClick={onIconClick}\n          isPasswordVisible={isPasswordVisible}\n          readOnly={readOnly}\n          ariaLabel={ariaLabel}\n          onToggle={() => setIsPasswordVisible((prev) => !prev)}\n        />\n      </StyledContainer>\n    );\n  }\n);\n\nexport const PasswordInput = React.forwardRef(\n  (\n    {\n      placeholder = \"••••••••\",\n      hasError = false,\n      onIconClick,\n      autoComplete = \"on\",\n      slotProps,\n      disabled,\n      label,\n      labelHint,\n      hideLabel,\n      hint,\n      errorMessages,\n      \"aria-describedby\": ariaDescribedBy,\n      \"data-e2e-test-id\": dataE2eTestId,\n      ...rest\n    }: PasswordInputProps,\n    ref: React.RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const { id, ariaDescribedByProp } = useFormFieldMessageId({\n      ariaDescribedBy,\n      errorMessages,\n      hint,\n    });\n\n    return (\n      <FormField\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"PasswordInput\"\n        disabled={disabled}\n        label={label}\n        labelHint={labelHint}\n        hideLabel={hideLabel}\n        hint={hint}\n        errorMessages={errorMessages}\n        messageId={id}\n      >\n        <PasswordInputRaw\n          {...rest}\n          ref={ref}\n          placeholder={placeholder}\n          hasError={hasError}\n          disabled={disabled}\n          onIconClick={onIconClick}\n          autoComplete={autoComplete}\n          slotProps={slotProps}\n          {...ariaDescribedByProp}\n        />\n      </FormField>\n    );\n  }\n);\n"],"names":[],"mappings":"AA+D4B"} */"),PasswordInputRaw=(0,_react.forwardRef)(({placeholder,hasError=!1,onIconClick,readOnly,disabled,slotProps:{toggleVisibility:{ariaLabelShow,ariaLabelHide}},...rest},ref)=>{let[isPasswordVisible,setIsPasswordVisible]=(0,_react.useState)(!1);return _react.default.createElement(StyledContainer,null,_react.default.createElement(StyledPasswordInput,{...rest,type:isPasswordVisible?"text":"password",placeholder:!readOnly&&placeholder,ref:ref,hasError:hasError,readOnly:readOnly,disabled:disabled,"aria-invalid":hasError}),_react.default.createElement(_PasswordInputButton.PasswordInputButton,{hasError:hasError,disabled:disabled,onIconClick:onIconClick,isPasswordVisible:isPasswordVisible,readOnly:readOnly,ariaLabel:isPasswordVisible?ariaLabelHide:ariaLabelShow,onToggle:()=>setIsPasswordVisible(prev=>!prev)}))}),PasswordInput=_react.default.forwardRef(({placeholder="••••••••",hasError=!1,onIconClick,autoComplete="on",slotProps,disabled,label,labelHint,hideLabel,hint,errorMessages,"aria-describedby":ariaDescribedBy,"data-e2e-test-id":dataE2eTestId,...rest},ref)=>{let{id,ariaDescribedByProp}=(0,_useFormFieldMessageId.useFormFieldMessageId)({ariaDescribedBy,errorMessages,hint});return _react.default.createElement(_FormField.FormField,{"data-e2e-test-id":dataE2eTestId,"data-ds-id":"PasswordInput",disabled:disabled,label:label,labelHint:labelHint,hideLabel:hideLabel,hint:hint,errorMessages:errorMessages,messageId:id},_react.default.createElement(PasswordInputRaw,{...rest,ref:ref,placeholder:placeholder,hasError:hasError,disabled:disabled,onIconClick:onIconClick,autoComplete:autoComplete,slotProps:slotProps,...ariaDescribedByProp}))});
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import type { ReactElement } from "react";
|
|
2
|
+
import React from "react";
|
|
3
|
+
type PasswordInputButtonProps = {
|
|
4
|
+
onIconClick?: (e: React.MouseEvent) => void;
|
|
5
|
+
hasError: boolean;
|
|
6
|
+
disabled: boolean;
|
|
7
|
+
readOnly: boolean;
|
|
8
|
+
ariaLabel: string;
|
|
9
|
+
isPasswordVisible: boolean;
|
|
10
|
+
onToggle: () => void;
|
|
11
|
+
};
|
|
12
|
+
export declare const PasswordInputButton: ({ hasError, disabled, onIconClick, readOnly, ariaLabel, isPasswordVisible, onToggle, }: PasswordInputButtonProps) => ReactElement;
|
|
13
|
+
export {};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
"use strict";Object.defineProperty(exports,"__esModule",{value:!0}),Object.defineProperty(exports,"PasswordInputButton",{enumerable:!0,get:function(){return PasswordInputButton}});const _interop_require_default=require("@swc/helpers/_/_interop_require_default"),_react=/*#__PURE__*/require("@swc/helpers/_/_interop_require_wildcard")._(require("react")),_styled=/*#__PURE__*/_interop_require_default._(require("@emotion/styled")),_Icon=require("../../Icon/Icon"),_ScreenReaderText=require("../../Utilities/ScreenReaderText/ScreenReaderText"),handleIconBorderColor=(theme,hasError)=>hasError?theme.values.color.icon.error.default:"transparent",handleIconLeftBorderColor=(theme,hasError)=>hasError?theme.values.color.border.error.default:theme.values.color.border.primary.default,handleClick=({e,disabled,onIconClick,onToggle})=>{disabled&&e.preventDefault(),onIconClick?.(e),onToggle()},StyledIconBtn=(0,_styled.default)("button",{target:"e94fqmq0",label:"StyledIconBtn"})(({theme,hasError})=>({"&[type='button']":{appearance:"none",MozAppearance:"none",WebkitAppearance:"none"},position:"absolute",top:0,right:0,border:"1px solid",borderColor:handleIconBorderColor(theme,hasError),borderLeftColor:handleIconLeftBorderColor(theme,hasError),padding:`calc(${theme.variables.size.spacing.s} - 1px) ${theme.variables.size.spacing.s}`,borderRadius:theme.variables.size.borderRadius.xs,borderBottomLeftRadius:"0",borderTopLeftRadius:"0",color:theme.values.color.text.secondary.default,backgroundColor:"inherit",cursor:"pointer","&:hover":{backgroundColor:theme.values.color.background.transparent.hover,borderColor:theme.values.color.border.primary.hover},"&:active":{color:theme.values.color.text.primary.default,backgroundColor:theme.values.color.background.transparent.active,borderColor:theme.values.color.border.primary.active},"&:disabled":{backgroundColor:theme.values.color.background.transparent.default,borderLeftColor:theme.values.color.border.primary.disabled,pointerEvents:"none"},...hasError&&{boxShadow:`0 0 0 1px inset ${theme.values.color.border.error.default}`},"& svg":{pointerEvents:"none"}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3JjL2NvbXBvbmVudHMvRm9ybS9QYXNzd29yZElucHV0L1Bhc3N3b3JkSW5wdXRCdXR0b24udHN4Iiwic291cmNlcyI6WyJzcmMvY29tcG9uZW50cy9Gb3JtL1Bhc3N3b3JkSW5wdXQvUGFzc3dvcmRJbnB1dEJ1dHRvbi50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBSZWFjdEVsZW1lbnQgfSBmcm9tIFwicmVhY3RcIjtcbmltcG9ydCBSZWFjdCwgeyB1c2VJZCwgdXNlU3RhdGUgfSBmcm9tIFwicmVhY3RcIjtcbmltcG9ydCB0eXBlIHsgVGhlbWUgfSBmcm9tIFwiQGVtb3Rpb24vcmVhY3RcIjtcbmltcG9ydCBzdHlsZWQgZnJvbSBcIkBlbW90aW9uL3N0eWxlZFwiO1xuaW1wb3J0IHsgSWNvbiB9IGZyb20gXCIuLi8uLi9JY29uL0ljb25cIjtcbmltcG9ydCB7IFNjcmVlblJlYWRlclRleHQgfSBmcm9tIFwiLi4vLi4vVXRpbGl0aWVzL1NjcmVlblJlYWRlclRleHQvU2NyZWVuUmVhZGVyVGV4dFwiO1xuXG50eXBlIFBhc3N3b3JkSW5wdXRCdXR0b25Qcm9wcyA9IHtcbiAgb25JY29uQ2xpY2s/OiAoZTogUmVhY3QuTW91c2VFdmVudCkgPT4gdm9pZDtcbiAgaGFzRXJyb3I6IGJvb2xlYW47XG4gIGRpc2FibGVkOiBib29sZWFuO1xuICByZWFkT25seTogYm9vbGVhbjtcbiAgYXJpYUxhYmVsOiBzdHJpbmc7XG4gIGlzUGFzc3dvcmRWaXNpYmxlOiBib29sZWFuO1xuICBvblRvZ2dsZTogKCkgPT4gdm9pZDtcbn07XG5cbnR5cGUgSWNvblByb3BzID0ge1xuICBoYXNFcnJvcj86IGJvb2xlYW47XG4gIGRpc2FibGVkPzogYm9vbGVhbjtcbiAgb25DaGFuZ2U/OiAoZTogUmVhY3QuRm9ybUV2ZW50PEhUTUxJbnB1dEVsZW1lbnQ+KSA9PiB2b2lkO1xuICB0YWJJbmRleD86IG51bWJlcjtcbn07XG5cbmNvbnN0IGhhbmRsZUljb25Cb3JkZXJDb2xvciA9ICh0aGVtZTogVGhlbWUsIGhhc0Vycm9yOiBib29sZWFuKSA9PiB7XG4gIGlmIChoYXNFcnJvcikgcmV0dXJuIHRoZW1lLnZhbHVlcy5jb2xvci5pY29uLmVycm9yLmRlZmF1bHQ7XG4gIHJldHVybiBcInRyYW5zcGFyZW50XCI7XG59O1xuXG5jb25zdCBoYW5kbGVJY29uTGVmdEJvcmRlckNvbG9yID0gKHRoZW1lOiBUaGVtZSwgaGFzRXJyb3I6IGJvb2xlYW4pID0+IHtcbiAgaWYgKGhhc0Vycm9yKSByZXR1cm4gdGhlbWUudmFsdWVzLmNvbG9yLmJvcmRlci5lcnJvci5kZWZhdWx0O1xuICByZXR1cm4gdGhlbWUudmFsdWVzLmNvbG9yLmJvcmRlci5wcmltYXJ5LmRlZmF1bHQ7XG59O1xuXG5jb25zdCBoYW5kbGVDbGljayA9ICh7XG4gIGUsXG4gIGRpc2FibGVkLFxuICBvbkljb25DbGljayxcbiAgb25Ub2dnbGUsXG59OiB7XG4gIGU6IFJlYWN0Lk1vdXNlRXZlbnQ7XG4gIGRpc2FibGVkOiBQYXNzd29yZElucHV0QnV0dG9uUHJvcHNbXCJkaXNhYmxlZFwiXTtcbiAgb25JY29uQ2xpY2s6IFBhc3N3b3JkSW5wdXRCdXR0b25Qcm9wc1tcIm9uSWNvbkNsaWNrXCJdO1xuICBvblRvZ2dsZTogUGFzc3dvcmRJbnB1dEJ1dHRvblByb3BzW1wib25Ub2dnbGVcIl07XG59KSA9PiB7XG4gIGlmIChkaXNhYmxlZCkge1xuICAgIGUucHJldmVudERlZmF1bHQoKTtcbiAgfVxuICBvbkljb25DbGljaz8uKGUpO1xuICBvblRvZ2dsZSgpO1xufTtcblxuY29uc3QgU3R5bGVkSWNvbkJ0biA9IHN0eWxlZC5idXR0b248SWNvblByb3BzPigoeyB0aGVtZSwgaGFzRXJyb3IgfSkgPT4gKHtcbiAgXCImW3R5cGU9J2J1dHRvbiddXCI6IHtcbiAgICBhcHBlYXJhbmNlOiBcIm5vbmVcIixcbiAgICBNb3pBcHBlYXJhbmNlOiBcIm5vbmVcIixcbiAgICBXZWJraXRBcHBlYXJhbmNlOiBcIm5vbmVcIixcbiAgfSxcbiAgcG9zaXRpb246IFwiYWJzb2x1dGVcIixcbiAgdG9wOiAwLFxuICByaWdodDogMCxcbiAgYm9yZGVyOiBcIjFweCBzb2xpZFwiLFxuICBib3JkZXJDb2xvcjogaGFuZGxlSWNvbkJvcmRlckNvbG9yKHRoZW1lLCBoYXNFcnJvciksXG4gIGJvcmRlckxlZnRDb2xvcjogaGFuZGxlSWNvbkxlZnRCb3JkZXJDb2xvcih0aGVtZSwgaGFzRXJyb3IpLFxuICBwYWRkaW5nOiBgY2FsYygke3RoZW1lLnZhcmlhYmxlcy5zaXplLnNwYWNpbmcuc30gLSAxcHgpICR7dGhlbWUudmFyaWFibGVzLnNpemUuc3BhY2luZy5zfWAsXG4gIGJvcmRlclJhZGl1czogdGhlbWUudmFyaWFibGVzLnNpemUuYm9yZGVyUmFkaXVzLnhzLFxuICBib3JkZXJCb3R0b21MZWZ0UmFkaXVzOiBcIjBcIixcbiAgYm9yZGVyVG9wTGVmdFJhZGl1czogXCIwXCIsXG4gIGNvbG9yOiB0aGVtZS52YWx1ZXMuY29sb3IudGV4dC5zZWNvbmRhcnkuZGVmYXVsdCxcbiAgYmFja2dyb3VuZENvbG9yOiBcImluaGVyaXRcIixcbiAgY3Vyc29yOiBcInBvaW50ZXJcIixcbiAgXCImOmhvdmVyXCI6IHtcbiAgICBiYWNrZ3JvdW5kQ29sb3I6IHRoZW1lLnZhbHVlcy5jb2xvci5iYWNrZ3JvdW5kLnRyYW5zcGFyZW50LmhvdmVyLFxuICAgIGJvcmRlckNvbG9yOiB0aGVtZS52YWx1ZXMuY29sb3IuYm9yZGVyLnByaW1hcnkuaG92ZXIsXG4gIH0sXG4gIFwiJjphY3RpdmVcIjoge1xuICAgIGNvbG9yOiB0aGVtZS52YWx1ZXMuY29sb3IudGV4dC5wcmltYXJ5LmRlZmF1bHQsXG4gICAgYmFja2dyb3VuZENvbG9yOiB0aGVtZS52YWx1ZXMuY29sb3IuYmFja2dyb3VuZC50cmFuc3BhcmVudC5hY3RpdmUsXG4gICAgYm9yZGVyQ29sb3I6IHRoZW1lLnZhbHVlcy5jb2xvci5ib3JkZXIucHJpbWFyeS5hY3RpdmUsXG4gIH0sXG4gIFwiJjpkaXNhYmxlZFwiOiB7XG4gICAgYmFja2dyb3VuZENvbG9yOiB0aGVtZS52YWx1ZXMuY29sb3IuYmFja2dyb3VuZC50cmFuc3BhcmVudC5kZWZhdWx0LFxuICAgIGJvcmRlckxlZnRDb2xvcjogdGhlbWUudmFsdWVzLmNvbG9yLmJvcmRlci5wcmltYXJ5LmRpc2FibGVkLFxuICAgIHBvaW50ZXJFdmVudHM6IFwibm9uZVwiLFxuICB9LFxuICAuLi4oaGFzRXJyb3IgJiYge1xuICAgIGJveFNoYWRvdzogYDAgMCAwIDFweCBpbnNldCAke3RoZW1lLnZhbHVlcy5jb2xvci5ib3JkZXIuZXJyb3IuZGVmYXVsdH1gLFxuICB9KSxcbiAgXCImIHN2Z1wiOiB7XG4gICAgcG9pbnRlckV2ZW50czogXCJub25lXCIsXG4gIH0sXG59KSk7XG5cbmV4cG9ydCBjb25zdCBQYXNzd29yZElucHV0QnV0dG9uID0gKHtcbiAgaGFzRXJyb3IsXG4gIGRpc2FibGVkLFxuICBvbkljb25DbGljayxcbiAgcmVhZE9ubHksXG4gIGFyaWFMYWJlbCxcbiAgaXNQYXNzd29yZFZpc2libGUsXG4gIG9uVG9nZ2xlLFxufTogUGFzc3dvcmRJbnB1dEJ1dHRvblByb3BzKTogUmVhY3RFbGVtZW50ID0+IHtcbiAgY29uc3QgW2lzQnV0dG9uRm9jdXNlZCwgc2V0SXNCdXR0b25Gb2N1c2VkXSA9IHVzZVN0YXRlKGZhbHNlKTtcbiAgY29uc3Qgc2NyZWVuUmVhZGVyVGV4dElkID0gdXNlSWQoKTtcbiAgY29uc3Qgc2NyZWVuUmVhZGVyVGV4dCA9IGlzQnV0dG9uRm9jdXNlZCA/IGFyaWFMYWJlbCA6IFwiXCI7XG4gIGNvbnN0IGV5ZUljb24gPSBpc1Bhc3N3b3JkVmlzaWJsZSA/IFwiZXllXCIgOiBcImV5ZS1vZmZcIjtcblxuICByZXR1cm4gKFxuICAgIDw+XG4gICAgICA8U2NyZWVuUmVhZGVyVGV4dCBpZD17c2NyZWVuUmVhZGVyVGV4dElkfT5cbiAgICAgICAge3NjcmVlblJlYWRlclRleHR9XG4gICAgICA8L1NjcmVlblJlYWRlclRleHQ+XG4gICAgICA8U3R5bGVkSWNvbkJ0blxuICAgICAgICBvbkNsaWNrPXsoZTogUmVhY3QuTW91c2VFdmVudCkgPT4ge1xuICAgICAgICAgIGhhbmRsZUNsaWNrKHsgZSwgZGlzYWJsZWQsIG9uSWNvbkNsaWNrLCBvblRvZ2dsZSB9KTtcbiAgICAgICAgfX1cbiAgICAgICAgaGFzRXJyb3I9e2hhc0Vycm9yfVxuICAgICAgICBvbkZvY3VzPXsoKSA9PiBzZXRJc0J1dHRvbkZvY3VzZWQodHJ1ZSl9XG4gICAgICAgIG9uQmx1cj17KCkgPT4gc2V0SXNCdXR0b25Gb2N1c2VkKGZhbHNlKX1cbiAgICAgICAgZGlzYWJsZWQ9e3JlYWRPbmx5IHx8IGRpc2FibGVkfVxuICAgICAgICB0eXBlPVwiYnV0dG9uXCJcbiAgICAgICAgYXJpYS1sYWJlbGxlZGJ5PXtzY3JlZW5SZWFkZXJUZXh0SWR9XG4gICAgICA+XG4gICAgICAgIDxJY29uXG4gICAgICAgICAgbmFtZT17ZXllSWNvbn1cbiAgICAgICAgICBzaXplPVwic1wiXG4gICAgICAgICAgY29sb3I9e3JlYWRPbmx5ID8gXCJxdWF0ZXJuYXJ5XCIgOiBcInRlcnRpYXJ5XCJ9XG4gICAgICAgIC8+XG4gICAgICA8L1N0eWxlZEljb25CdG4+XG4gICAgPC8+XG4gICk7XG59O1xuIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQW9Ec0IifQ== */"),PasswordInputButton=({hasError,disabled,onIconClick,readOnly,ariaLabel,isPasswordVisible,onToggle})=>{let[isButtonFocused,setIsButtonFocused]=(0,_react.useState)(!1),screenReaderTextId=(0,_react.useId)();return _react.default.createElement(_react.default.Fragment,null,_react.default.createElement(_ScreenReaderText.ScreenReaderText,{id:screenReaderTextId},isButtonFocused?ariaLabel:""),_react.default.createElement(StyledIconBtn,{onClick:e=>{handleClick({e,disabled,onIconClick,onToggle})},hasError:hasError,onFocus:()=>setIsButtonFocused(!0),onBlur:()=>setIsButtonFocused(!1),disabled:readOnly||disabled,type:"button","aria-labelledby":screenReaderTextId},_react.default.createElement(_Icon.Icon,{name:isPasswordVisible?"eye":"eye-off",size:"s",color:readOnly?"quaternary":"tertiary"})))};
|
|
@@ -2,7 +2,21 @@ import type { InputHTMLAttributes } from "react";
|
|
|
2
2
|
import React from "react";
|
|
3
3
|
import type { FormFieldProps } from "../FormField/FormField";
|
|
4
4
|
export type PasswordInputProps = {
|
|
5
|
-
|
|
5
|
+
/**
|
|
6
|
+
* Props for sub-components.
|
|
7
|
+
*
|
|
8
|
+
* - `toggleVisibility.ariaLabelShow` — Label for screen readers when the icon can be toggled to show the password
|
|
9
|
+
*
|
|
10
|
+
* - `toggleVisibility.ariaLabelHide` — Label for screen readers when the icon can be toggled to hide the password
|
|
11
|
+
*/
|
|
12
|
+
slotProps: {
|
|
13
|
+
toggleVisibility: {
|
|
14
|
+
/** Label for screen readers when icon can be toggled to show password */
|
|
15
|
+
ariaLabelShow: string;
|
|
16
|
+
/** Label for screen readers when icon can be toggled to hide password */
|
|
17
|
+
ariaLabelHide: string;
|
|
18
|
+
};
|
|
19
|
+
};
|
|
6
20
|
} & InputProps;
|
|
7
21
|
type InputProps = {
|
|
8
22
|
/**
|
|
@@ -10,31 +24,59 @@ type InputProps = {
|
|
|
10
24
|
*/
|
|
11
25
|
hasError?: boolean;
|
|
12
26
|
/**
|
|
13
|
-
*
|
|
27
|
+
* Function for on icon click behavior
|
|
14
28
|
*/
|
|
15
29
|
onIconClick?: (e: React.MouseEvent) => void;
|
|
16
30
|
} & FormFieldProps & Omit<InputHTMLAttributes<HTMLInputElement>, "type">;
|
|
17
31
|
export declare const PasswordInputRaw: React.ForwardRefExoticComponent<{
|
|
18
|
-
|
|
32
|
+
/**
|
|
33
|
+
* Props for sub-components.
|
|
34
|
+
*
|
|
35
|
+
* - `toggleVisibility.ariaLabelShow` — Label for screen readers when the icon can be toggled to show the password
|
|
36
|
+
*
|
|
37
|
+
* - `toggleVisibility.ariaLabelHide` — Label for screen readers when the icon can be toggled to hide the password
|
|
38
|
+
*/
|
|
39
|
+
slotProps: {
|
|
40
|
+
toggleVisibility: {
|
|
41
|
+
/** Label for screen readers when icon can be toggled to show password */
|
|
42
|
+
ariaLabelShow: string;
|
|
43
|
+
/** Label for screen readers when icon can be toggled to hide password */
|
|
44
|
+
ariaLabelHide: string;
|
|
45
|
+
};
|
|
46
|
+
};
|
|
19
47
|
} & {
|
|
20
48
|
/**
|
|
21
49
|
* Set to true to toggle error state
|
|
22
50
|
*/
|
|
23
51
|
hasError?: boolean;
|
|
24
52
|
/**
|
|
25
|
-
*
|
|
53
|
+
* Function for on icon click behavior
|
|
26
54
|
*/
|
|
27
55
|
onIconClick?: (e: React.MouseEvent) => void;
|
|
28
56
|
} & FormFieldProps & Omit<InputHTMLAttributes<HTMLInputElement>, "type"> & React.RefAttributes<HTMLInputElement>>;
|
|
29
57
|
export declare const PasswordInput: React.ForwardRefExoticComponent<{
|
|
30
|
-
|
|
58
|
+
/**
|
|
59
|
+
* Props for sub-components.
|
|
60
|
+
*
|
|
61
|
+
* - `toggleVisibility.ariaLabelShow` — Label for screen readers when the icon can be toggled to show the password
|
|
62
|
+
*
|
|
63
|
+
* - `toggleVisibility.ariaLabelHide` — Label for screen readers when the icon can be toggled to hide the password
|
|
64
|
+
*/
|
|
65
|
+
slotProps: {
|
|
66
|
+
toggleVisibility: {
|
|
67
|
+
/** Label for screen readers when icon can be toggled to show password */
|
|
68
|
+
ariaLabelShow: string;
|
|
69
|
+
/** Label for screen readers when icon can be toggled to hide password */
|
|
70
|
+
ariaLabelHide: string;
|
|
71
|
+
};
|
|
72
|
+
};
|
|
31
73
|
} & {
|
|
32
74
|
/**
|
|
33
75
|
* Set to true to toggle error state
|
|
34
76
|
*/
|
|
35
77
|
hasError?: boolean;
|
|
36
78
|
/**
|
|
37
|
-
*
|
|
79
|
+
* Function for on icon click behavior
|
|
38
80
|
*/
|
|
39
81
|
onIconClick?: (e: React.MouseEvent) => void;
|
|
40
82
|
} & FormFieldProps & Omit<InputHTMLAttributes<HTMLInputElement>, "type"> & React.RefAttributes<HTMLInputElement>>;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import React,{useState,forwardRef}from"react";import styled from"@emotion/styled";import{Icon}from"../../Icon/Icon";import{FormField}from"../FormField/FormField";import{useFormFieldMessageId}from"../FormField/useFormFieldMessageId";let handleBorderColor=(theme,hasError,readOnly)=>hasError?theme.values.color.border.error.default:readOnly?theme.values.color.border.primary.disabled:theme.values.color.border.primary.default,handleIconBorderColor=(theme,hasError)=>hasError?theme.values.color.icon.error.default:"transparent",handleIconLeftBorderColor=(theme,hasError)=>hasError?theme.values.color.border.error.default:theme.values.color.border.primary.default,StyledContainer=styled("div",{target:"e1ro5src0",label:"StyledContainer"})(()=>({width:"100%",display:"flex",alignItems:"center",flexDirection:"row",position:"relative"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/PasswordInput/PasswordInput.tsx","sources":["src/components/Form/PasswordInput/PasswordInput.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport type { RefObject, InputHTMLAttributes } from \"react\";\nimport React, { useState, forwardRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { CSSObject, Theme } from \"@emotion/react\";\nimport { Icon } from \"../../Icon/Icon\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { useFormFieldMessageId } from \"../FormField/useFormFieldMessageId\";\n\nexport type PasswordInputProps = {\n  iconAriaLabel: string;\n} & InputProps;\n\ntype InputProps = {\n  /**\n   * Set to true to toggle error state\n   */\n  hasError?: boolean;\n  /**\n   * Set to true to disabled input\n   */\n  onIconClick?: (e: React.MouseEvent) => void;\n} & FormFieldProps &\n  Omit<InputHTMLAttributes<HTMLInputElement>, \"type\">;\n\ntype IconProps = {\n  hasError?: boolean;\n  disabled?: boolean;\n  onChange?: (e: React.FormEvent<HTMLInputElement>) => void;\n  tabIndex?: number;\n};\n\nconst handleBorderColor = (\n  theme: Theme,\n  hasError: boolean,\n  readOnly: boolean\n) => {\n  if (hasError) return theme.values.color.border.error.default;\n  if (readOnly) return theme.values.color.border.primary.disabled;\n\n  return theme.values.color.border.primary.default;\n};\n\nconst handleIconBorderColor = (theme: Theme, hasError: boolean) => {\n  if (hasError) return theme.values.color.icon.error.default;\n  return \"transparent\";\n};\n\nconst handleIconLeftBorderColor = (theme: Theme, hasError: boolean) => {\n  if (hasError) return theme.values.color.border.error.default;\n  return theme.values.color.border.primary.default;\n};\n\nconst StyledContainer = styled.div(\n  () =>\n    ({\n      width: \"100%\",\n      display: \"flex\",\n      alignItems: \"center\",\n      flexDirection: \"row\",\n      position: \"relative\",\n    } as CSSObject)\n);\n\nconst StyledIconBtn = styled.button<IconProps>(({ theme, hasError }) => ({\n  \"&[type='button']\": {\n    appearance: \"none\",\n    MozAppearance: \"none\",\n    WebkitAppearance: \"none\",\n  },\n  position: \"absolute\",\n  top: 0,\n  right: 0,\n  border: \"1px solid\",\n  borderColor: handleIconBorderColor(theme, hasError),\n  borderLeftColor: handleIconLeftBorderColor(theme, hasError),\n  padding: `calc(${theme.variables.size.spacing.s} - 1px) ${theme.variables.size.spacing.s}`,\n  borderRadius: theme.variables.size.borderRadius.xs,\n  borderBottomLeftRadius: \"0\",\n  borderTopLeftRadius: \"0\",\n  color: theme.values.color.text.secondary.default,\n  backgroundColor: \"inherit\",\n  cursor: \"pointer\",\n  \"&:hover\": {\n    backgroundColor: theme.values.color.background.transparent.hover,\n    borderColor: theme.values.color.border.primary.hover,\n  },\n  \"&:active\": {\n    color: theme.values.color.text.primary.default,\n    backgroundColor: theme.values.color.background.transparent.active,\n    borderColor: theme.values.color.border.primary.active,\n  },\n  \"&:disabled\": {\n    backgroundColor: theme.values.color.background.transparent.default,\n    borderLeftColor: theme.values.color.border.primary.disabled,\n    pointerEvents: \"none\",\n  },\n  ...(hasError && {\n    boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n  }),\n  \"& svg\": {\n    pointerEvents: \"none\",\n  },\n}));\n\nconst StyledPasswordInput = styled.input<InputProps>(\n  ({ theme, hasError, readOnly }) => ({\n    fontFamily: theme.variables.fontFamily.lato,\n    fontSize: theme.variables.size.font.m,\n    lineHeight: theme.variables.size.lineHeight.xl,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    borderWidth: \"1px\",\n    borderStyle: \"solid\",\n    width: \"100%\",\n    padding: `calc(${theme.variables.size.spacing.xs} - 1px) ${theme.variables.size.spacing.s}`,\n    paddingRight: theme.variables.size.spacing.xxl,\n    boxSizing: \"border-box\",\n    color: theme.values.color.text.secondary.default,\n    borderColor: handleBorderColor(theme, hasError, readOnly),\n    backgroundColor: theme.values.color.background.primary.default,\n    \"&.error\": {\n      borderColor: theme.values.color.border.error.default,\n    },\n    \"&.has-icon\": {\n      paddingRight:\n        theme.variables.size.spacing.xs + theme.variables.size.dimension.icon.m,\n    },\n    \"&::placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n      opacity: theme.variables.opacity.visible,\n    },\n    \"&:-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    \"&::-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    ...(hasError && {\n      boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n    }),\n  })\n);\n\nexport const PasswordInputRaw = forwardRef(\n  (\n    {\n      placeholder,\n      hasError = false,\n      onIconClick,\n      iconAriaLabel,\n      readOnly,\n      disabled,\n      ...rest\n    }: PasswordInputProps,\n    ref: RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const [inputType, setInputType] = useState(\"password\");\n    const currentType = inputType === \"password\" ? \"text\" : \"password\";\n    const eyeIcon = inputType === \"password\" ? \"eye-off\" : \"eye\";\n\n    return (\n      <StyledContainer>\n        <StyledPasswordInput\n          {...rest}\n          type={inputType}\n          placeholder={!readOnly && placeholder}\n          ref={ref}\n          hasError={hasError}\n          readOnly={readOnly}\n          disabled={disabled}\n          aria-invalid={hasError}\n        />\n        <StyledIconBtn\n          onClick={(e: React.MouseEvent) => {\n            if (disabled) {\n              e.preventDefault();\n            }\n            if (onIconClick) {\n              onIconClick(e);\n            }\n            setInputType(currentType);\n          }}\n          hasError={hasError}\n          disabled={readOnly || disabled}\n          aria-label={iconAriaLabel}\n          type=\"button\"\n        >\n          <Icon\n            name={eyeIcon}\n            size=\"s\"\n            color={readOnly ? \"quaternary\" : \"tertiary\"}\n          />\n        </StyledIconBtn>\n      </StyledContainer>\n    );\n  }\n);\n\nexport const PasswordInput = React.forwardRef(\n  (\n    {\n      placeholder = \"••••••••\",\n      hasError = false,\n      onIconClick,\n      autoComplete = \"on\",\n      iconAriaLabel,\n      disabled,\n      label,\n      labelHint,\n      hideLabel,\n      hint,\n      errorMessages,\n      \"aria-describedby\": ariaDescribedBy,\n      \"data-e2e-test-id\": dataE2eTestId,\n      ...rest\n    }: PasswordInputProps,\n    ref: React.RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const { id, ariaDescribedByProp } = useFormFieldMessageId({\n      ariaDescribedBy,\n      errorMessages,\n      hint,\n    });\n\n    return (\n      <FormField\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"PasswordInput\"\n        disabled={disabled}\n        label={label}\n        labelHint={labelHint}\n        hideLabel={hideLabel}\n        hint={hint}\n        errorMessages={errorMessages}\n        messageId={id}\n      >\n        <PasswordInputRaw\n          {...rest}\n          ref={ref}\n          placeholder={placeholder}\n          hasError={hasError}\n          disabled={disabled}\n          onIconClick={onIconClick}\n          autoComplete={autoComplete}\n          iconAriaLabel={iconAriaLabel}\n          {...ariaDescribedByProp}\n        />\n      </FormField>\n    );\n  }\n);\n"],"names":[],"mappings":"AAuDwB"} */"),StyledIconBtn=styled("button",{target:"e1ro5src1",label:"StyledIconBtn"})(({theme,hasError})=>({"&[type='button']":{appearance:"none",MozAppearance:"none",WebkitAppearance:"none"},position:"absolute",top:0,right:0,border:"1px solid",borderColor:handleIconBorderColor(theme,hasError),borderLeftColor:handleIconLeftBorderColor(theme,hasError),padding:`calc(${theme.variables.size.spacing.s} - 1px) ${theme.variables.size.spacing.s}`,borderRadius:theme.variables.size.borderRadius.xs,borderBottomLeftRadius:"0",borderTopLeftRadius:"0",color:theme.values.color.text.secondary.default,backgroundColor:"inherit",cursor:"pointer","&:hover":{backgroundColor:theme.values.color.background.transparent.hover,borderColor:theme.values.color.border.primary.hover},"&:active":{color:theme.values.color.text.primary.default,backgroundColor:theme.values.color.background.transparent.active,borderColor:theme.values.color.border.primary.active},"&:disabled":{backgroundColor:theme.values.color.background.transparent.default,borderLeftColor:theme.values.color.border.primary.disabled,pointerEvents:"none"},...hasError&&{boxShadow:`0 0 0 1px inset ${theme.values.color.border.error.default}`},"& svg":{pointerEvents:"none"}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/PasswordInput/PasswordInput.tsx","sources":["src/components/Form/PasswordInput/PasswordInput.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport type { RefObject, InputHTMLAttributes } from \"react\";\nimport React, { useState, forwardRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { CSSObject, Theme } from \"@emotion/react\";\nimport { Icon } from \"../../Icon/Icon\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { useFormFieldMessageId } from \"../FormField/useFormFieldMessageId\";\n\nexport type PasswordInputProps = {\n  iconAriaLabel: string;\n} & InputProps;\n\ntype InputProps = {\n  /**\n   * Set to true to toggle error state\n   */\n  hasError?: boolean;\n  /**\n   * Set to true to disabled input\n   */\n  onIconClick?: (e: React.MouseEvent) => void;\n} & FormFieldProps &\n  Omit<InputHTMLAttributes<HTMLInputElement>, \"type\">;\n\ntype IconProps = {\n  hasError?: boolean;\n  disabled?: boolean;\n  onChange?: (e: React.FormEvent<HTMLInputElement>) => void;\n  tabIndex?: number;\n};\n\nconst handleBorderColor = (\n  theme: Theme,\n  hasError: boolean,\n  readOnly: boolean\n) => {\n  if (hasError) return theme.values.color.border.error.default;\n  if (readOnly) return theme.values.color.border.primary.disabled;\n\n  return theme.values.color.border.primary.default;\n};\n\nconst handleIconBorderColor = (theme: Theme, hasError: boolean) => {\n  if (hasError) return theme.values.color.icon.error.default;\n  return \"transparent\";\n};\n\nconst handleIconLeftBorderColor = (theme: Theme, hasError: boolean) => {\n  if (hasError) return theme.values.color.border.error.default;\n  return theme.values.color.border.primary.default;\n};\n\nconst StyledContainer = styled.div(\n  () =>\n    ({\n      width: \"100%\",\n      display: \"flex\",\n      alignItems: \"center\",\n      flexDirection: \"row\",\n      position: \"relative\",\n    } as CSSObject)\n);\n\nconst StyledIconBtn = styled.button<IconProps>(({ theme, hasError }) => ({\n  \"&[type='button']\": {\n    appearance: \"none\",\n    MozAppearance: \"none\",\n    WebkitAppearance: \"none\",\n  },\n  position: \"absolute\",\n  top: 0,\n  right: 0,\n  border: \"1px solid\",\n  borderColor: handleIconBorderColor(theme, hasError),\n  borderLeftColor: handleIconLeftBorderColor(theme, hasError),\n  padding: `calc(${theme.variables.size.spacing.s} - 1px) ${theme.variables.size.spacing.s}`,\n  borderRadius: theme.variables.size.borderRadius.xs,\n  borderBottomLeftRadius: \"0\",\n  borderTopLeftRadius: \"0\",\n  color: theme.values.color.text.secondary.default,\n  backgroundColor: \"inherit\",\n  cursor: \"pointer\",\n  \"&:hover\": {\n    backgroundColor: theme.values.color.background.transparent.hover,\n    borderColor: theme.values.color.border.primary.hover,\n  },\n  \"&:active\": {\n    color: theme.values.color.text.primary.default,\n    backgroundColor: theme.values.color.background.transparent.active,\n    borderColor: theme.values.color.border.primary.active,\n  },\n  \"&:disabled\": {\n    backgroundColor: theme.values.color.background.transparent.default,\n    borderLeftColor: theme.values.color.border.primary.disabled,\n    pointerEvents: \"none\",\n  },\n  ...(hasError && {\n    boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n  }),\n  \"& svg\": {\n    pointerEvents: \"none\",\n  },\n}));\n\nconst StyledPasswordInput = styled.input<InputProps>(\n  ({ theme, hasError, readOnly }) => ({\n    fontFamily: theme.variables.fontFamily.lato,\n    fontSize: theme.variables.size.font.m,\n    lineHeight: theme.variables.size.lineHeight.xl,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    borderWidth: \"1px\",\n    borderStyle: \"solid\",\n    width: \"100%\",\n    padding: `calc(${theme.variables.size.spacing.xs} - 1px) ${theme.variables.size.spacing.s}`,\n    paddingRight: theme.variables.size.spacing.xxl,\n    boxSizing: \"border-box\",\n    color: theme.values.color.text.secondary.default,\n    borderColor: handleBorderColor(theme, hasError, readOnly),\n    backgroundColor: theme.values.color.background.primary.default,\n    \"&.error\": {\n      borderColor: theme.values.color.border.error.default,\n    },\n    \"&.has-icon\": {\n      paddingRight:\n        theme.variables.size.spacing.xs + theme.variables.size.dimension.icon.m,\n    },\n    \"&::placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n      opacity: theme.variables.opacity.visible,\n    },\n    \"&:-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    \"&::-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    ...(hasError && {\n      boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n    }),\n  })\n);\n\nexport const PasswordInputRaw = forwardRef(\n  (\n    {\n      placeholder,\n      hasError = false,\n      onIconClick,\n      iconAriaLabel,\n      readOnly,\n      disabled,\n      ...rest\n    }: PasswordInputProps,\n    ref: RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const [inputType, setInputType] = useState(\"password\");\n    const currentType = inputType === \"password\" ? \"text\" : \"password\";\n    const eyeIcon = inputType === \"password\" ? \"eye-off\" : \"eye\";\n\n    return (\n      <StyledContainer>\n        <StyledPasswordInput\n          {...rest}\n          type={inputType}\n          placeholder={!readOnly && placeholder}\n          ref={ref}\n          hasError={hasError}\n          readOnly={readOnly}\n          disabled={disabled}\n          aria-invalid={hasError}\n        />\n        <StyledIconBtn\n          onClick={(e: React.MouseEvent) => {\n            if (disabled) {\n              e.preventDefault();\n            }\n            if (onIconClick) {\n              onIconClick(e);\n            }\n            setInputType(currentType);\n          }}\n          hasError={hasError}\n          disabled={readOnly || disabled}\n          aria-label={iconAriaLabel}\n          type=\"button\"\n        >\n          <Icon\n            name={eyeIcon}\n            size=\"s\"\n            color={readOnly ? \"quaternary\" : \"tertiary\"}\n          />\n        </StyledIconBtn>\n      </StyledContainer>\n    );\n  }\n);\n\nexport const PasswordInput = React.forwardRef(\n  (\n    {\n      placeholder = \"••••••••\",\n      hasError = false,\n      onIconClick,\n      autoComplete = \"on\",\n      iconAriaLabel,\n      disabled,\n      label,\n      labelHint,\n      hideLabel,\n      hint,\n      errorMessages,\n      \"aria-describedby\": ariaDescribedBy,\n      \"data-e2e-test-id\": dataE2eTestId,\n      ...rest\n    }: PasswordInputProps,\n    ref: React.RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const { id, ariaDescribedByProp } = useFormFieldMessageId({\n      ariaDescribedBy,\n      errorMessages,\n      hint,\n    });\n\n    return (\n      <FormField\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"PasswordInput\"\n        disabled={disabled}\n        label={label}\n        labelHint={labelHint}\n        hideLabel={hideLabel}\n        hint={hint}\n        errorMessages={errorMessages}\n        messageId={id}\n      >\n        <PasswordInputRaw\n          {...rest}\n          ref={ref}\n          placeholder={placeholder}\n          hasError={hasError}\n          disabled={disabled}\n          onIconClick={onIconClick}\n          autoComplete={autoComplete}\n          iconAriaLabel={iconAriaLabel}\n          {...ariaDescribedByProp}\n        />\n      </FormField>\n    );\n  }\n);\n"],"names":[],"mappings":"AAkEsB"} */"),StyledPasswordInput=styled("input",{target:"e1ro5src2",label:"StyledPasswordInput"})(({theme,hasError,readOnly})=>({fontFamily:theme.variables.fontFamily.lato,fontSize:theme.variables.size.font.m,lineHeight:theme.variables.size.lineHeight.xl,borderRadius:theme.variables.size.borderRadius.xs,borderWidth:"1px",borderStyle:"solid",width:"100%",padding:`calc(${theme.variables.size.spacing.xs} - 1px) ${theme.variables.size.spacing.s}`,paddingRight:theme.variables.size.spacing.xxl,boxSizing:"border-box",color:theme.values.color.text.secondary.default,borderColor:handleBorderColor(theme,hasError,readOnly),backgroundColor:theme.values.color.background.primary.default,"&.error":{borderColor:theme.values.color.border.error.default},"&.has-icon":{paddingRight:theme.variables.size.spacing.xs+theme.variables.size.dimension.icon.m},"&::placeholder":{fontStyle:"italic",color:theme.values.color.text.tertiary.default,opacity:theme.variables.opacity.visible},"&:-ms-input-placeholder":{fontStyle:"italic",color:theme.values.color.text.tertiary.default},"&::-ms-input-placeholder":{fontStyle:"italic",color:theme.values.color.text.tertiary.default},...hasError&&{boxShadow:`0 0 0 1px inset ${theme.values.color.border.error.default}`}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/PasswordInput/PasswordInput.tsx","sources":["src/components/Form/PasswordInput/PasswordInput.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport type { RefObject, InputHTMLAttributes } from \"react\";\nimport React, { useState, forwardRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { CSSObject, Theme } from \"@emotion/react\";\nimport { Icon } from \"../../Icon/Icon\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { useFormFieldMessageId } from \"../FormField/useFormFieldMessageId\";\n\nexport type PasswordInputProps = {\n  iconAriaLabel: string;\n} & InputProps;\n\ntype InputProps = {\n  /**\n   * Set to true to toggle error state\n   */\n  hasError?: boolean;\n  /**\n   * Set to true to disabled input\n   */\n  onIconClick?: (e: React.MouseEvent) => void;\n} & FormFieldProps &\n  Omit<InputHTMLAttributes<HTMLInputElement>, \"type\">;\n\ntype IconProps = {\n  hasError?: boolean;\n  disabled?: boolean;\n  onChange?: (e: React.FormEvent<HTMLInputElement>) => void;\n  tabIndex?: number;\n};\n\nconst handleBorderColor = (\n  theme: Theme,\n  hasError: boolean,\n  readOnly: boolean\n) => {\n  if (hasError) return theme.values.color.border.error.default;\n  if (readOnly) return theme.values.color.border.primary.disabled;\n\n  return theme.values.color.border.primary.default;\n};\n\nconst handleIconBorderColor = (theme: Theme, hasError: boolean) => {\n  if (hasError) return theme.values.color.icon.error.default;\n  return \"transparent\";\n};\n\nconst handleIconLeftBorderColor = (theme: Theme, hasError: boolean) => {\n  if (hasError) return theme.values.color.border.error.default;\n  return theme.values.color.border.primary.default;\n};\n\nconst StyledContainer = styled.div(\n  () =>\n    ({\n      width: \"100%\",\n      display: \"flex\",\n      alignItems: \"center\",\n      flexDirection: \"row\",\n      position: \"relative\",\n    } as CSSObject)\n);\n\nconst StyledIconBtn = styled.button<IconProps>(({ theme, hasError }) => ({\n  \"&[type='button']\": {\n    appearance: \"none\",\n    MozAppearance: \"none\",\n    WebkitAppearance: \"none\",\n  },\n  position: \"absolute\",\n  top: 0,\n  right: 0,\n  border: \"1px solid\",\n  borderColor: handleIconBorderColor(theme, hasError),\n  borderLeftColor: handleIconLeftBorderColor(theme, hasError),\n  padding: `calc(${theme.variables.size.spacing.s} - 1px) ${theme.variables.size.spacing.s}`,\n  borderRadius: theme.variables.size.borderRadius.xs,\n  borderBottomLeftRadius: \"0\",\n  borderTopLeftRadius: \"0\",\n  color: theme.values.color.text.secondary.default,\n  backgroundColor: \"inherit\",\n  cursor: \"pointer\",\n  \"&:hover\": {\n    backgroundColor: theme.values.color.background.transparent.hover,\n    borderColor: theme.values.color.border.primary.hover,\n  },\n  \"&:active\": {\n    color: theme.values.color.text.primary.default,\n    backgroundColor: theme.values.color.background.transparent.active,\n    borderColor: theme.values.color.border.primary.active,\n  },\n  \"&:disabled\": {\n    backgroundColor: theme.values.color.background.transparent.default,\n    borderLeftColor: theme.values.color.border.primary.disabled,\n    pointerEvents: \"none\",\n  },\n  ...(hasError && {\n    boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n  }),\n  \"& svg\": {\n    pointerEvents: \"none\",\n  },\n}));\n\nconst StyledPasswordInput = styled.input<InputProps>(\n  ({ theme, hasError, readOnly }) => ({\n    fontFamily: theme.variables.fontFamily.lato,\n    fontSize: theme.variables.size.font.m,\n    lineHeight: theme.variables.size.lineHeight.xl,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    borderWidth: \"1px\",\n    borderStyle: \"solid\",\n    width: \"100%\",\n    padding: `calc(${theme.variables.size.spacing.xs} - 1px) ${theme.variables.size.spacing.s}`,\n    paddingRight: theme.variables.size.spacing.xxl,\n    boxSizing: \"border-box\",\n    color: theme.values.color.text.secondary.default,\n    borderColor: handleBorderColor(theme, hasError, readOnly),\n    backgroundColor: theme.values.color.background.primary.default,\n    \"&.error\": {\n      borderColor: theme.values.color.border.error.default,\n    },\n    \"&.has-icon\": {\n      paddingRight:\n        theme.variables.size.spacing.xs + theme.variables.size.dimension.icon.m,\n    },\n    \"&::placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n      opacity: theme.variables.opacity.visible,\n    },\n    \"&:-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    \"&::-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    ...(hasError && {\n      boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n    }),\n  })\n);\n\nexport const PasswordInputRaw = forwardRef(\n  (\n    {\n      placeholder,\n      hasError = false,\n      onIconClick,\n      iconAriaLabel,\n      readOnly,\n      disabled,\n      ...rest\n    }: PasswordInputProps,\n    ref: RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const [inputType, setInputType] = useState(\"password\");\n    const currentType = inputType === \"password\" ? \"text\" : \"password\";\n    const eyeIcon = inputType === \"password\" ? \"eye-off\" : \"eye\";\n\n    return (\n      <StyledContainer>\n        <StyledPasswordInput\n          {...rest}\n          type={inputType}\n          placeholder={!readOnly && placeholder}\n          ref={ref}\n          hasError={hasError}\n          readOnly={readOnly}\n          disabled={disabled}\n          aria-invalid={hasError}\n        />\n        <StyledIconBtn\n          onClick={(e: React.MouseEvent) => {\n            if (disabled) {\n              e.preventDefault();\n            }\n            if (onIconClick) {\n              onIconClick(e);\n            }\n            setInputType(currentType);\n          }}\n          hasError={hasError}\n          disabled={readOnly || disabled}\n          aria-label={iconAriaLabel}\n          type=\"button\"\n        >\n          <Icon\n            name={eyeIcon}\n            size=\"s\"\n            color={readOnly ? \"quaternary\" : \"tertiary\"}\n          />\n        </StyledIconBtn>\n      </StyledContainer>\n    );\n  }\n);\n\nexport const PasswordInput = React.forwardRef(\n  (\n    {\n      placeholder = \"••••••••\",\n      hasError = false,\n      onIconClick,\n      autoComplete = \"on\",\n      iconAriaLabel,\n      disabled,\n      label,\n      labelHint,\n      hideLabel,\n      hint,\n      errorMessages,\n      \"aria-describedby\": ariaDescribedBy,\n      \"data-e2e-test-id\": dataE2eTestId,\n      ...rest\n    }: PasswordInputProps,\n    ref: React.RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const { id, ariaDescribedByProp } = useFormFieldMessageId({\n      ariaDescribedBy,\n      errorMessages,\n      hint,\n    });\n\n    return (\n      <FormField\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"PasswordInput\"\n        disabled={disabled}\n        label={label}\n        labelHint={labelHint}\n        hideLabel={hideLabel}\n        hint={hint}\n        errorMessages={errorMessages}\n        messageId={id}\n      >\n        <PasswordInputRaw\n          {...rest}\n          ref={ref}\n          placeholder={placeholder}\n          hasError={hasError}\n          disabled={disabled}\n          onIconClick={onIconClick}\n          autoComplete={autoComplete}\n          iconAriaLabel={iconAriaLabel}\n          {...ariaDescribedByProp}\n        />\n      </FormField>\n    );\n  }\n);\n"],"names":[],"mappings":"AA2G4B"} */");export const PasswordInputRaw=forwardRef(({placeholder,hasError=!1,onIconClick,iconAriaLabel,readOnly,disabled,...rest},ref)=>{let[inputType,setInputType]=useState("password"),currentType="password"===inputType?"text":"password";return React.createElement(StyledContainer,null,React.createElement(StyledPasswordInput,{...rest,type:inputType,placeholder:!readOnly&&placeholder,ref:ref,hasError:hasError,readOnly:readOnly,disabled:disabled,"aria-invalid":hasError}),React.createElement(StyledIconBtn,{onClick:e=>{disabled&&e.preventDefault(),onIconClick&&onIconClick(e),setInputType(currentType)},hasError:hasError,disabled:readOnly||disabled,"aria-label":iconAriaLabel,type:"button"},React.createElement(Icon,{name:"password"===inputType?"eye-off":"eye",size:"s",color:readOnly?"quaternary":"tertiary"})))});export const PasswordInput=React.forwardRef(({placeholder="••••••••",hasError=!1,onIconClick,autoComplete="on",iconAriaLabel,disabled,label,labelHint,hideLabel,hint,errorMessages,"aria-describedby":ariaDescribedBy,"data-e2e-test-id":dataE2eTestId,...rest},ref)=>{let{id,ariaDescribedByProp}=useFormFieldMessageId({ariaDescribedBy,errorMessages,hint});return React.createElement(FormField,{"data-e2e-test-id":dataE2eTestId,"data-ds-id":"PasswordInput",disabled:disabled,label:label,labelHint:labelHint,hideLabel:hideLabel,hint:hint,errorMessages:errorMessages,messageId:id},React.createElement(PasswordInputRaw,{...rest,ref:ref,placeholder:placeholder,hasError:hasError,disabled:disabled,onIconClick:onIconClick,autoComplete:autoComplete,iconAriaLabel:iconAriaLabel,...ariaDescribedByProp}))});
|
|
1
|
+
import React,{useState,forwardRef}from"react";import styled from"@emotion/styled";import{FormField}from"../FormField/FormField";import{useFormFieldMessageId}from"../FormField/useFormFieldMessageId";import{PasswordInputButton}from"./PasswordInputButton";let handleBorderColor=(theme,hasError,readOnly)=>hasError?theme.values.color.border.error.default:readOnly?theme.values.color.border.primary.disabled:theme.values.color.border.primary.default,StyledContainer=styled("div",{target:"e3t34vm0",label:"StyledContainer"})(()=>({width:"100%",display:"flex",alignItems:"center",flexDirection:"row",position:"relative"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/PasswordInput/PasswordInput.tsx","sources":["src/components/Form/PasswordInput/PasswordInput.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport type { RefObject, InputHTMLAttributes } from \"react\";\nimport React, { useState, forwardRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { CSSObject, Theme } from \"@emotion/react\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { useFormFieldMessageId } from \"../FormField/useFormFieldMessageId\";\nimport { PasswordInputButton } from \"./PasswordInputButton\";\n\nexport type PasswordInputProps = {\n  /**\n   * Props for sub-components.\n   *\n   * - `toggleVisibility.ariaLabelShow` — Label for screen readers when the icon can be toggled to show the password\n   *\n   * - `toggleVisibility.ariaLabelHide` — Label for screen readers when the icon can be toggled to hide the password\n   */\n  slotProps: {\n    toggleVisibility: {\n      /** Label for screen readers when icon can be toggled to show password */\n      ariaLabelShow: string;\n      /** Label for screen readers when icon can be toggled to hide password */\n      ariaLabelHide: string;\n    };\n  };\n} & InputProps;\n\ntype InputProps = {\n  /**\n   * Set to true to toggle error state\n   */\n  hasError?: boolean;\n  /**\n   * Function for on icon click behavior\n   */\n  onIconClick?: (e: React.MouseEvent) => void;\n} & FormFieldProps &\n  Omit<InputHTMLAttributes<HTMLInputElement>, \"type\">;\n\nconst handleBorderColor = (\n  theme: Theme,\n  hasError: boolean,\n  readOnly: boolean\n) => {\n  if (hasError) return theme.values.color.border.error.default;\n  if (readOnly) return theme.values.color.border.primary.disabled;\n\n  return theme.values.color.border.primary.default;\n};\n\nconst StyledContainer = styled.div(\n  () =>\n    ({\n      width: \"100%\",\n      display: \"flex\",\n      alignItems: \"center\",\n      flexDirection: \"row\",\n      position: \"relative\",\n    } as CSSObject)\n);\n\nconst StyledPasswordInput = styled.input<InputProps>(\n  ({ theme, hasError, readOnly }) => ({\n    fontFamily: theme.variables.fontFamily.lato,\n    fontSize: theme.variables.size.font.m,\n    lineHeight: theme.variables.size.lineHeight.xl,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    borderWidth: \"1px\",\n    borderStyle: \"solid\",\n    width: \"100%\",\n    padding: `calc(${theme.variables.size.spacing.xs} - 1px) ${theme.variables.size.spacing.s}`,\n    paddingRight: theme.variables.size.spacing.xxl,\n    boxSizing: \"border-box\",\n    color: theme.values.color.text.secondary.default,\n    borderColor: handleBorderColor(theme, hasError, readOnly),\n    backgroundColor: theme.values.color.background.primary.default,\n    \"&.error\": {\n      borderColor: theme.values.color.border.error.default,\n    },\n    \"&.has-icon\": {\n      paddingRight:\n        theme.variables.size.spacing.xs + theme.variables.size.dimension.icon.m,\n    },\n    \"&::placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n      opacity: theme.variables.opacity.visible,\n    },\n    \"&:-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    \"&::-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    ...(hasError && {\n      boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n    }),\n  })\n);\n\nexport const PasswordInputRaw = forwardRef(\n  (\n    {\n      placeholder,\n      hasError = false,\n      onIconClick,\n      readOnly,\n      disabled,\n      slotProps: {\n        toggleVisibility: { ariaLabelShow, ariaLabelHide },\n      },\n      ...rest\n    }: PasswordInputProps,\n    ref: RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false);\n    const inputType = isPasswordVisible ? \"text\" : \"password\";\n    const ariaLabel = isPasswordVisible ? ariaLabelHide : ariaLabelShow;\n\n    return (\n      <StyledContainer>\n        <StyledPasswordInput\n          {...rest}\n          type={inputType}\n          placeholder={!readOnly && placeholder}\n          ref={ref}\n          hasError={hasError}\n          readOnly={readOnly}\n          disabled={disabled}\n          aria-invalid={hasError}\n        />\n        <PasswordInputButton\n          hasError={hasError}\n          disabled={disabled}\n          onIconClick={onIconClick}\n          isPasswordVisible={isPasswordVisible}\n          readOnly={readOnly}\n          ariaLabel={ariaLabel}\n          onToggle={() => setIsPasswordVisible((prev) => !prev)}\n        />\n      </StyledContainer>\n    );\n  }\n);\n\nexport const PasswordInput = React.forwardRef(\n  (\n    {\n      placeholder = \"••••••••\",\n      hasError = false,\n      onIconClick,\n      autoComplete = \"on\",\n      slotProps,\n      disabled,\n      label,\n      labelHint,\n      hideLabel,\n      hint,\n      errorMessages,\n      \"aria-describedby\": ariaDescribedBy,\n      \"data-e2e-test-id\": dataE2eTestId,\n      ...rest\n    }: PasswordInputProps,\n    ref: React.RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const { id, ariaDescribedByProp } = useFormFieldMessageId({\n      ariaDescribedBy,\n      errorMessages,\n      hint,\n    });\n\n    return (\n      <FormField\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"PasswordInput\"\n        disabled={disabled}\n        label={label}\n        labelHint={labelHint}\n        hideLabel={hideLabel}\n        hint={hint}\n        errorMessages={errorMessages}\n        messageId={id}\n      >\n        <PasswordInputRaw\n          {...rest}\n          ref={ref}\n          placeholder={placeholder}\n          hasError={hasError}\n          disabled={disabled}\n          onIconClick={onIconClick}\n          autoComplete={autoComplete}\n          slotProps={slotProps}\n          {...ariaDescribedByProp}\n        />\n      </FormField>\n    );\n  }\n);\n"],"names":[],"mappings":"AAoDwB"} */"),StyledPasswordInput=styled("input",{target:"e3t34vm1",label:"StyledPasswordInput"})(({theme,hasError,readOnly})=>({fontFamily:theme.variables.fontFamily.lato,fontSize:theme.variables.size.font.m,lineHeight:theme.variables.size.lineHeight.xl,borderRadius:theme.variables.size.borderRadius.xs,borderWidth:"1px",borderStyle:"solid",width:"100%",padding:`calc(${theme.variables.size.spacing.xs} - 1px) ${theme.variables.size.spacing.s}`,paddingRight:theme.variables.size.spacing.xxl,boxSizing:"border-box",color:theme.values.color.text.secondary.default,borderColor:handleBorderColor(theme,hasError,readOnly),backgroundColor:theme.values.color.background.primary.default,"&.error":{borderColor:theme.values.color.border.error.default},"&.has-icon":{paddingRight:theme.variables.size.spacing.xs+theme.variables.size.dimension.icon.m},"&::placeholder":{fontStyle:"italic",color:theme.values.color.text.tertiary.default,opacity:theme.variables.opacity.visible},"&:-ms-input-placeholder":{fontStyle:"italic",color:theme.values.color.text.tertiary.default},"&::-ms-input-placeholder":{fontStyle:"italic",color:theme.values.color.text.tertiary.default},...hasError&&{boxShadow:`0 0 0 1px inset ${theme.values.color.border.error.default}`}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/PasswordInput/PasswordInput.tsx","sources":["src/components/Form/PasswordInput/PasswordInput.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport type { RefObject, InputHTMLAttributes } from \"react\";\nimport React, { useState, forwardRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { CSSObject, Theme } from \"@emotion/react\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { useFormFieldMessageId } from \"../FormField/useFormFieldMessageId\";\nimport { PasswordInputButton } from \"./PasswordInputButton\";\n\nexport type PasswordInputProps = {\n  /**\n   * Props for sub-components.\n   *\n   * - `toggleVisibility.ariaLabelShow` — Label for screen readers when the icon can be toggled to show the password\n   *\n   * - `toggleVisibility.ariaLabelHide` — Label for screen readers when the icon can be toggled to hide the password\n   */\n  slotProps: {\n    toggleVisibility: {\n      /** Label for screen readers when icon can be toggled to show password */\n      ariaLabelShow: string;\n      /** Label for screen readers when icon can be toggled to hide password */\n      ariaLabelHide: string;\n    };\n  };\n} & InputProps;\n\ntype InputProps = {\n  /**\n   * Set to true to toggle error state\n   */\n  hasError?: boolean;\n  /**\n   * Function for on icon click behavior\n   */\n  onIconClick?: (e: React.MouseEvent) => void;\n} & FormFieldProps &\n  Omit<InputHTMLAttributes<HTMLInputElement>, \"type\">;\n\nconst handleBorderColor = (\n  theme: Theme,\n  hasError: boolean,\n  readOnly: boolean\n) => {\n  if (hasError) return theme.values.color.border.error.default;\n  if (readOnly) return theme.values.color.border.primary.disabled;\n\n  return theme.values.color.border.primary.default;\n};\n\nconst StyledContainer = styled.div(\n  () =>\n    ({\n      width: \"100%\",\n      display: \"flex\",\n      alignItems: \"center\",\n      flexDirection: \"row\",\n      position: \"relative\",\n    } as CSSObject)\n);\n\nconst StyledPasswordInput = styled.input<InputProps>(\n  ({ theme, hasError, readOnly }) => ({\n    fontFamily: theme.variables.fontFamily.lato,\n    fontSize: theme.variables.size.font.m,\n    lineHeight: theme.variables.size.lineHeight.xl,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    borderWidth: \"1px\",\n    borderStyle: \"solid\",\n    width: \"100%\",\n    padding: `calc(${theme.variables.size.spacing.xs} - 1px) ${theme.variables.size.spacing.s}`,\n    paddingRight: theme.variables.size.spacing.xxl,\n    boxSizing: \"border-box\",\n    color: theme.values.color.text.secondary.default,\n    borderColor: handleBorderColor(theme, hasError, readOnly),\n    backgroundColor: theme.values.color.background.primary.default,\n    \"&.error\": {\n      borderColor: theme.values.color.border.error.default,\n    },\n    \"&.has-icon\": {\n      paddingRight:\n        theme.variables.size.spacing.xs + theme.variables.size.dimension.icon.m,\n    },\n    \"&::placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n      opacity: theme.variables.opacity.visible,\n    },\n    \"&:-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    \"&::-ms-input-placeholder\": {\n      fontStyle: \"italic\",\n      color: theme.values.color.text.tertiary.default,\n    },\n    ...(hasError && {\n      boxShadow: `0 0 0 1px inset ${theme.values.color.border.error.default}`,\n    }),\n  })\n);\n\nexport const PasswordInputRaw = forwardRef(\n  (\n    {\n      placeholder,\n      hasError = false,\n      onIconClick,\n      readOnly,\n      disabled,\n      slotProps: {\n        toggleVisibility: { ariaLabelShow, ariaLabelHide },\n      },\n      ...rest\n    }: PasswordInputProps,\n    ref: RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false);\n    const inputType = isPasswordVisible ? \"text\" : \"password\";\n    const ariaLabel = isPasswordVisible ? ariaLabelHide : ariaLabelShow;\n\n    return (\n      <StyledContainer>\n        <StyledPasswordInput\n          {...rest}\n          type={inputType}\n          placeholder={!readOnly && placeholder}\n          ref={ref}\n          hasError={hasError}\n          readOnly={readOnly}\n          disabled={disabled}\n          aria-invalid={hasError}\n        />\n        <PasswordInputButton\n          hasError={hasError}\n          disabled={disabled}\n          onIconClick={onIconClick}\n          isPasswordVisible={isPasswordVisible}\n          readOnly={readOnly}\n          ariaLabel={ariaLabel}\n          onToggle={() => setIsPasswordVisible((prev) => !prev)}\n        />\n      </StyledContainer>\n    );\n  }\n);\n\nexport const PasswordInput = React.forwardRef(\n  (\n    {\n      placeholder = \"••••••••\",\n      hasError = false,\n      onIconClick,\n      autoComplete = \"on\",\n      slotProps,\n      disabled,\n      label,\n      labelHint,\n      hideLabel,\n      hint,\n      errorMessages,\n      \"aria-describedby\": ariaDescribedBy,\n      \"data-e2e-test-id\": dataE2eTestId,\n      ...rest\n    }: PasswordInputProps,\n    ref: React.RefObject<HTMLInputElement>\n  ): React.ReactElement => {\n    const { id, ariaDescribedByProp } = useFormFieldMessageId({\n      ariaDescribedBy,\n      errorMessages,\n      hint,\n    });\n\n    return (\n      <FormField\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"PasswordInput\"\n        disabled={disabled}\n        label={label}\n        labelHint={labelHint}\n        hideLabel={hideLabel}\n        hint={hint}\n        errorMessages={errorMessages}\n        messageId={id}\n      >\n        <PasswordInputRaw\n          {...rest}\n          ref={ref}\n          placeholder={placeholder}\n          hasError={hasError}\n          disabled={disabled}\n          onIconClick={onIconClick}\n          autoComplete={autoComplete}\n          slotProps={slotProps}\n          {...ariaDescribedByProp}\n        />\n      </FormField>\n    );\n  }\n);\n"],"names":[],"mappings":"AA+D4B"} */");export const PasswordInputRaw=forwardRef(({placeholder,hasError=!1,onIconClick,readOnly,disabled,slotProps:{toggleVisibility:{ariaLabelShow,ariaLabelHide}},...rest},ref)=>{let[isPasswordVisible,setIsPasswordVisible]=useState(!1);return React.createElement(StyledContainer,null,React.createElement(StyledPasswordInput,{...rest,type:isPasswordVisible?"text":"password",placeholder:!readOnly&&placeholder,ref:ref,hasError:hasError,readOnly:readOnly,disabled:disabled,"aria-invalid":hasError}),React.createElement(PasswordInputButton,{hasError:hasError,disabled:disabled,onIconClick:onIconClick,isPasswordVisible:isPasswordVisible,readOnly:readOnly,ariaLabel:isPasswordVisible?ariaLabelHide:ariaLabelShow,onToggle:()=>setIsPasswordVisible(prev=>!prev)}))});export const PasswordInput=React.forwardRef(({placeholder="••••••••",hasError=!1,onIconClick,autoComplete="on",slotProps,disabled,label,labelHint,hideLabel,hint,errorMessages,"aria-describedby":ariaDescribedBy,"data-e2e-test-id":dataE2eTestId,...rest},ref)=>{let{id,ariaDescribedByProp}=useFormFieldMessageId({ariaDescribedBy,errorMessages,hint});return React.createElement(FormField,{"data-e2e-test-id":dataE2eTestId,"data-ds-id":"PasswordInput",disabled:disabled,label:label,labelHint:labelHint,hideLabel:hideLabel,hint:hint,errorMessages:errorMessages,messageId:id},React.createElement(PasswordInputRaw,{...rest,ref:ref,placeholder:placeholder,hasError:hasError,disabled:disabled,onIconClick:onIconClick,autoComplete:autoComplete,slotProps:slotProps,...ariaDescribedByProp}))});
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import type { ReactElement } from "react";
|
|
2
|
+
import React from "react";
|
|
3
|
+
type PasswordInputButtonProps = {
|
|
4
|
+
onIconClick?: (e: React.MouseEvent) => void;
|
|
5
|
+
hasError: boolean;
|
|
6
|
+
disabled: boolean;
|
|
7
|
+
readOnly: boolean;
|
|
8
|
+
ariaLabel: string;
|
|
9
|
+
isPasswordVisible: boolean;
|
|
10
|
+
onToggle: () => void;
|
|
11
|
+
};
|
|
12
|
+
export declare const PasswordInputButton: ({ hasError, disabled, onIconClick, readOnly, ariaLabel, isPasswordVisible, onToggle, }: PasswordInputButtonProps) => ReactElement;
|
|
13
|
+
export {};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
import React,{useId,useState}from"react";import styled from"@emotion/styled";import{Icon}from"../../Icon/Icon";import{ScreenReaderText}from"../../Utilities/ScreenReaderText/ScreenReaderText";let handleIconBorderColor=(theme,hasError)=>hasError?theme.values.color.icon.error.default:"transparent",handleIconLeftBorderColor=(theme,hasError)=>hasError?theme.values.color.border.error.default:theme.values.color.border.primary.default,handleClick=({e,disabled,onIconClick,onToggle})=>{disabled&&e.preventDefault(),onIconClick?.(e),onToggle()},StyledIconBtn=styled("button",{target:"e94fqmq0",label:"StyledIconBtn"})(({theme,hasError})=>({"&[type='button']":{appearance:"none",MozAppearance:"none",WebkitAppearance:"none"},position:"absolute",top:0,right:0,border:"1px solid",borderColor:handleIconBorderColor(theme,hasError),borderLeftColor:handleIconLeftBorderColor(theme,hasError),padding:`calc(${theme.variables.size.spacing.s} - 1px) ${theme.variables.size.spacing.s}`,borderRadius:theme.variables.size.borderRadius.xs,borderBottomLeftRadius:"0",borderTopLeftRadius:"0",color:theme.values.color.text.secondary.default,backgroundColor:"inherit",cursor:"pointer","&:hover":{backgroundColor:theme.values.color.background.transparent.hover,borderColor:theme.values.color.border.primary.hover},"&:active":{color:theme.values.color.text.primary.default,backgroundColor:theme.values.color.background.transparent.active,borderColor:theme.values.color.border.primary.active},"&:disabled":{backgroundColor:theme.values.color.background.transparent.default,borderLeftColor:theme.values.color.border.primary.disabled,pointerEvents:"none"},...hasError&&{boxShadow:`0 0 0 1px inset ${theme.values.color.border.error.default}`},"& svg":{pointerEvents:"none"}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3JjL2NvbXBvbmVudHMvRm9ybS9QYXNzd29yZElucHV0L1Bhc3N3b3JkSW5wdXRCdXR0b24udHN4Iiwic291cmNlcyI6WyJzcmMvY29tcG9uZW50cy9Gb3JtL1Bhc3N3b3JkSW5wdXQvUGFzc3dvcmRJbnB1dEJ1dHRvbi50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBSZWFjdEVsZW1lbnQgfSBmcm9tIFwicmVhY3RcIjtcbmltcG9ydCBSZWFjdCwgeyB1c2VJZCwgdXNlU3RhdGUgfSBmcm9tIFwicmVhY3RcIjtcbmltcG9ydCB0eXBlIHsgVGhlbWUgfSBmcm9tIFwiQGVtb3Rpb24vcmVhY3RcIjtcbmltcG9ydCBzdHlsZWQgZnJvbSBcIkBlbW90aW9uL3N0eWxlZFwiO1xuaW1wb3J0IHsgSWNvbiB9IGZyb20gXCIuLi8uLi9JY29uL0ljb25cIjtcbmltcG9ydCB7IFNjcmVlblJlYWRlclRleHQgfSBmcm9tIFwiLi4vLi4vVXRpbGl0aWVzL1NjcmVlblJlYWRlclRleHQvU2NyZWVuUmVhZGVyVGV4dFwiO1xuXG50eXBlIFBhc3N3b3JkSW5wdXRCdXR0b25Qcm9wcyA9IHtcbiAgb25JY29uQ2xpY2s/OiAoZTogUmVhY3QuTW91c2VFdmVudCkgPT4gdm9pZDtcbiAgaGFzRXJyb3I6IGJvb2xlYW47XG4gIGRpc2FibGVkOiBib29sZWFuO1xuICByZWFkT25seTogYm9vbGVhbjtcbiAgYXJpYUxhYmVsOiBzdHJpbmc7XG4gIGlzUGFzc3dvcmRWaXNpYmxlOiBib29sZWFuO1xuICBvblRvZ2dsZTogKCkgPT4gdm9pZDtcbn07XG5cbnR5cGUgSWNvblByb3BzID0ge1xuICBoYXNFcnJvcj86IGJvb2xlYW47XG4gIGRpc2FibGVkPzogYm9vbGVhbjtcbiAgb25DaGFuZ2U/OiAoZTogUmVhY3QuRm9ybUV2ZW50PEhUTUxJbnB1dEVsZW1lbnQ+KSA9PiB2b2lkO1xuICB0YWJJbmRleD86IG51bWJlcjtcbn07XG5cbmNvbnN0IGhhbmRsZUljb25Cb3JkZXJDb2xvciA9ICh0aGVtZTogVGhlbWUsIGhhc0Vycm9yOiBib29sZWFuKSA9PiB7XG4gIGlmIChoYXNFcnJvcikgcmV0dXJuIHRoZW1lLnZhbHVlcy5jb2xvci5pY29uLmVycm9yLmRlZmF1bHQ7XG4gIHJldHVybiBcInRyYW5zcGFyZW50XCI7XG59O1xuXG5jb25zdCBoYW5kbGVJY29uTGVmdEJvcmRlckNvbG9yID0gKHRoZW1lOiBUaGVtZSwgaGFzRXJyb3I6IGJvb2xlYW4pID0+IHtcbiAgaWYgKGhhc0Vycm9yKSByZXR1cm4gdGhlbWUudmFsdWVzLmNvbG9yLmJvcmRlci5lcnJvci5kZWZhdWx0O1xuICByZXR1cm4gdGhlbWUudmFsdWVzLmNvbG9yLmJvcmRlci5wcmltYXJ5LmRlZmF1bHQ7XG59O1xuXG5jb25zdCBoYW5kbGVDbGljayA9ICh7XG4gIGUsXG4gIGRpc2FibGVkLFxuICBvbkljb25DbGljayxcbiAgb25Ub2dnbGUsXG59OiB7XG4gIGU6IFJlYWN0Lk1vdXNlRXZlbnQ7XG4gIGRpc2FibGVkOiBQYXNzd29yZElucHV0QnV0dG9uUHJvcHNbXCJkaXNhYmxlZFwiXTtcbiAgb25JY29uQ2xpY2s6IFBhc3N3b3JkSW5wdXRCdXR0b25Qcm9wc1tcIm9uSWNvbkNsaWNrXCJdO1xuICBvblRvZ2dsZTogUGFzc3dvcmRJbnB1dEJ1dHRvblByb3BzW1wib25Ub2dnbGVcIl07XG59KSA9PiB7XG4gIGlmIChkaXNhYmxlZCkge1xuICAgIGUucHJldmVudERlZmF1bHQoKTtcbiAgfVxuICBvbkljb25DbGljaz8uKGUpO1xuICBvblRvZ2dsZSgpO1xufTtcblxuY29uc3QgU3R5bGVkSWNvbkJ0biA9IHN0eWxlZC5idXR0b248SWNvblByb3BzPigoeyB0aGVtZSwgaGFzRXJyb3IgfSkgPT4gKHtcbiAgXCImW3R5cGU9J2J1dHRvbiddXCI6IHtcbiAgICBhcHBlYXJhbmNlOiBcIm5vbmVcIixcbiAgICBNb3pBcHBlYXJhbmNlOiBcIm5vbmVcIixcbiAgICBXZWJraXRBcHBlYXJhbmNlOiBcIm5vbmVcIixcbiAgfSxcbiAgcG9zaXRpb246IFwiYWJzb2x1dGVcIixcbiAgdG9wOiAwLFxuICByaWdodDogMCxcbiAgYm9yZGVyOiBcIjFweCBzb2xpZFwiLFxuICBib3JkZXJDb2xvcjogaGFuZGxlSWNvbkJvcmRlckNvbG9yKHRoZW1lLCBoYXNFcnJvciksXG4gIGJvcmRlckxlZnRDb2xvcjogaGFuZGxlSWNvbkxlZnRCb3JkZXJDb2xvcih0aGVtZSwgaGFzRXJyb3IpLFxuICBwYWRkaW5nOiBgY2FsYygke3RoZW1lLnZhcmlhYmxlcy5zaXplLnNwYWNpbmcuc30gLSAxcHgpICR7dGhlbWUudmFyaWFibGVzLnNpemUuc3BhY2luZy5zfWAsXG4gIGJvcmRlclJhZGl1czogdGhlbWUudmFyaWFibGVzLnNpemUuYm9yZGVyUmFkaXVzLnhzLFxuICBib3JkZXJCb3R0b21MZWZ0UmFkaXVzOiBcIjBcIixcbiAgYm9yZGVyVG9wTGVmdFJhZGl1czogXCIwXCIsXG4gIGNvbG9yOiB0aGVtZS52YWx1ZXMuY29sb3IudGV4dC5zZWNvbmRhcnkuZGVmYXVsdCxcbiAgYmFja2dyb3VuZENvbG9yOiBcImluaGVyaXRcIixcbiAgY3Vyc29yOiBcInBvaW50ZXJcIixcbiAgXCImOmhvdmVyXCI6IHtcbiAgICBiYWNrZ3JvdW5kQ29sb3I6IHRoZW1lLnZhbHVlcy5jb2xvci5iYWNrZ3JvdW5kLnRyYW5zcGFyZW50LmhvdmVyLFxuICAgIGJvcmRlckNvbG9yOiB0aGVtZS52YWx1ZXMuY29sb3IuYm9yZGVyLnByaW1hcnkuaG92ZXIsXG4gIH0sXG4gIFwiJjphY3RpdmVcIjoge1xuICAgIGNvbG9yOiB0aGVtZS52YWx1ZXMuY29sb3IudGV4dC5wcmltYXJ5LmRlZmF1bHQsXG4gICAgYmFja2dyb3VuZENvbG9yOiB0aGVtZS52YWx1ZXMuY29sb3IuYmFja2dyb3VuZC50cmFuc3BhcmVudC5hY3RpdmUsXG4gICAgYm9yZGVyQ29sb3I6IHRoZW1lLnZhbHVlcy5jb2xvci5ib3JkZXIucHJpbWFyeS5hY3RpdmUsXG4gIH0sXG4gIFwiJjpkaXNhYmxlZFwiOiB7XG4gICAgYmFja2dyb3VuZENvbG9yOiB0aGVtZS52YWx1ZXMuY29sb3IuYmFja2dyb3VuZC50cmFuc3BhcmVudC5kZWZhdWx0LFxuICAgIGJvcmRlckxlZnRDb2xvcjogdGhlbWUudmFsdWVzLmNvbG9yLmJvcmRlci5wcmltYXJ5LmRpc2FibGVkLFxuICAgIHBvaW50ZXJFdmVudHM6IFwibm9uZVwiLFxuICB9LFxuICAuLi4oaGFzRXJyb3IgJiYge1xuICAgIGJveFNoYWRvdzogYDAgMCAwIDFweCBpbnNldCAke3RoZW1lLnZhbHVlcy5jb2xvci5ib3JkZXIuZXJyb3IuZGVmYXVsdH1gLFxuICB9KSxcbiAgXCImIHN2Z1wiOiB7XG4gICAgcG9pbnRlckV2ZW50czogXCJub25lXCIsXG4gIH0sXG59KSk7XG5cbmV4cG9ydCBjb25zdCBQYXNzd29yZElucHV0QnV0dG9uID0gKHtcbiAgaGFzRXJyb3IsXG4gIGRpc2FibGVkLFxuICBvbkljb25DbGljayxcbiAgcmVhZE9ubHksXG4gIGFyaWFMYWJlbCxcbiAgaXNQYXNzd29yZFZpc2libGUsXG4gIG9uVG9nZ2xlLFxufTogUGFzc3dvcmRJbnB1dEJ1dHRvblByb3BzKTogUmVhY3RFbGVtZW50ID0+IHtcbiAgY29uc3QgW2lzQnV0dG9uRm9jdXNlZCwgc2V0SXNCdXR0b25Gb2N1c2VkXSA9IHVzZVN0YXRlKGZhbHNlKTtcbiAgY29uc3Qgc2NyZWVuUmVhZGVyVGV4dElkID0gdXNlSWQoKTtcbiAgY29uc3Qgc2NyZWVuUmVhZGVyVGV4dCA9IGlzQnV0dG9uRm9jdXNlZCA/IGFyaWFMYWJlbCA6IFwiXCI7XG4gIGNvbnN0IGV5ZUljb24gPSBpc1Bhc3N3b3JkVmlzaWJsZSA/IFwiZXllXCIgOiBcImV5ZS1vZmZcIjtcblxuICByZXR1cm4gKFxuICAgIDw+XG4gICAgICA8U2NyZWVuUmVhZGVyVGV4dCBpZD17c2NyZWVuUmVhZGVyVGV4dElkfT5cbiAgICAgICAge3NjcmVlblJlYWRlclRleHR9XG4gICAgICA8L1NjcmVlblJlYWRlclRleHQ+XG4gICAgICA8U3R5bGVkSWNvbkJ0blxuICAgICAgICBvbkNsaWNrPXsoZTogUmVhY3QuTW91c2VFdmVudCkgPT4ge1xuICAgICAgICAgIGhhbmRsZUNsaWNrKHsgZSwgZGlzYWJsZWQsIG9uSWNvbkNsaWNrLCBvblRvZ2dsZSB9KTtcbiAgICAgICAgfX1cbiAgICAgICAgaGFzRXJyb3I9e2hhc0Vycm9yfVxuICAgICAgICBvbkZvY3VzPXsoKSA9PiBzZXRJc0J1dHRvbkZvY3VzZWQodHJ1ZSl9XG4gICAgICAgIG9uQmx1cj17KCkgPT4gc2V0SXNCdXR0b25Gb2N1c2VkKGZhbHNlKX1cbiAgICAgICAgZGlzYWJsZWQ9e3JlYWRPbmx5IHx8IGRpc2FibGVkfVxuICAgICAgICB0eXBlPVwiYnV0dG9uXCJcbiAgICAgICAgYXJpYS1sYWJlbGxlZGJ5PXtzY3JlZW5SZWFkZXJUZXh0SWR9XG4gICAgICA+XG4gICAgICAgIDxJY29uXG4gICAgICAgICAgbmFtZT17ZXllSWNvbn1cbiAgICAgICAgICBzaXplPVwic1wiXG4gICAgICAgICAgY29sb3I9e3JlYWRPbmx5ID8gXCJxdWF0ZXJuYXJ5XCIgOiBcInRlcnRpYXJ5XCJ9XG4gICAgICAgIC8+XG4gICAgICA8L1N0eWxlZEljb25CdG4+XG4gICAgPC8+XG4gICk7XG59O1xuIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQW9Ec0IifQ== */");export const PasswordInputButton=({hasError,disabled,onIconClick,readOnly,ariaLabel,isPasswordVisible,onToggle})=>{let[isButtonFocused,setIsButtonFocused]=useState(!1),screenReaderTextId=useId();return React.createElement(React.Fragment,null,React.createElement(ScreenReaderText,{id:screenReaderTextId},isButtonFocused?ariaLabel:""),React.createElement(StyledIconBtn,{onClick:e=>{handleClick({e,disabled,onIconClick,onToggle})},hasError:hasError,onFocus:()=>setIsButtonFocused(!0),onBlur:()=>setIsButtonFocused(!1),disabled:readOnly||disabled,type:"button","aria-labelledby":screenReaderTextId},React.createElement(Icon,{name:isPasswordVisible?"eye":"eye-off",size:"s",color:readOnly?"quaternary":"tertiary"})))};
|