diff --git a/packages/web/src/scss/components/TextField/README.md b/packages/web/src/scss/components/TextField/README.md index 1a289d6dfa..1b9a5a3e15 100644 --- a/packages/web/src/scss/components/TextField/README.md +++ b/packages/web/src/scss/components/TextField/README.md @@ -93,7 +93,7 @@ The `size` attribute is supported on inputs of the following types: `email`, This option is generally recommended for inputs with a limited value length (e.g. numeric representation of day, month, year). Supported values are `2`, `3` and `4` (characters). If you need any other value or prefer using `em` unit -instead of default `ch`, define a `--width` CSS custom property on the `` +instead of default `ch`, define a `--textfield-input-width` CSS custom property on the `` element: ```html @@ -110,7 +110,7 @@ element: class="TextField__input" name="sizeEm" placeholder="Placeholder" - style="--width: 4em;" + style="--textfield-input-width: 4em;" /> ``` diff --git a/packages/web/src/scss/components/TextField/_TextField.scss b/packages/web/src/scss/components/TextField/_TextField.scss index f598c6cf11..af116117d4 100644 --- a/packages/web/src/scss/components/TextField/_TextField.scss +++ b/packages/web/src/scss/components/TextField/_TextField.scss @@ -1,3 +1,20 @@ +// 1. We pass the input arrows width if the input type is number. +// 2. If the input has a size attribute, we calculate the width based on the input width, padding, border width and arrows width. +// 3. We generate the input width based on the number of characters. This is useful if you want to set the width based on the number of characters. +// 4. We position the password toggle button next to the input and use pseudo-element for border styling. +// We do this because password managers place their icon buttons inside the password input. +// 5. Because the button is next to the input, we need to use pseudo element to style the border around both of them. +// 6. We need to set the same background color for the button when the input is hovered, active or focused. +// 7. We toggle icons of the password toggle and we disable the no-descending-specificity to make the code more readable. +// 8. To make the 5. work, we need to style the input and the button when the input is focused. +// 9. The password toggle button should have its own focus-visible style. +// 10. Disabled input and its optional password toggle should have the same disabled style. +// The code is more comprehensible when styling elements in this order: +// 1. input (with a pseudo-element), +// 2. password toggle button, +// 3. validation text. +// Also, high-specificity selectors are necessary to achieve the desired result. + @use '../../settings/cursors'; @use '../../theme/form-fields' as form-fields-theme; @use '../../tools/form-fields' as form-fields-tools; @@ -35,20 +52,23 @@ $_field-name: 'TextField'; appearance: none; } + // 1. &[type='number'] { - --arrows-width: #{theme.$input-number-arrows-width}; + --textfield-arrows-width: #{theme.$input-number-arrows-width}; } + // 2. &[size] { width: calc( - var(--width) + 2 * #{form-fields-theme.$box-field-input-padding-x} + 2 * #{form-fields-theme.$box-field-input-border-width} + - var(--arrows-width, 0px) + var(--textfield-input-width) + 2 * #{form-fields-theme.$box-field-input-padding-x} + 2 * #{form-fields-theme.$box-field-input-border-width} + + var(--textfield-arrows-width, 0px) ); } + // 3. @each $size in theme.$input-size-characters { &[size='#{$size}'] { - --width: #{$size}ch; + --textfield-input-width: #{$size}ch; } } } @@ -70,8 +90,7 @@ $_field-name: 'TextField'; border-bottom-right-radius: 0; } -// We position the button next to the input and use pseudo-element for border styling. -// We do this because password managers place their icon buttons inside the password input. +// 4. .TextField__passwordToggle__button { @include reset.button(); @@ -81,19 +100,30 @@ $_field-name: 'TextField'; padding: theme.$input-password-toggle-padding; color: form-fields-theme.$box-field-input-color-default; border-radius: 0 form-fields-theme.$box-field-input-border-radius form-fields-theme.$box-field-input-border-radius 0; - background-color: form-fields-theme.$box-field-input-background; + background-color: form-fields-theme.$input-background-color; + // 5. &::before { content: ''; position: absolute; inset: 0; border: form-fields-theme.$box-field-input-border-width form-fields-theme.$box-field-input-border-style - form-fields-theme.$box-field-input-border-color-default; + form-fields-theme.$input-border-color; border-radius: form-fields-theme.$box-field-input-border-radius; pointer-events: none; } } +// 6. +.TextField__input:hover + .TextField__passwordToggle__button { + background-color: form-fields-theme.$input-background-color-hover; +} + +.TextField__input:active + .TextField__passwordToggle__button, +.TextField__input:focus-within + .TextField__passwordToggle__button { + background-color: form-fields-theme.$input-background-color-active; +} + .TextField__passwordToggle__icon { pointer-events: none; } @@ -103,13 +133,13 @@ $_field-name: 'TextField'; display: block; } -// stylelint-disable-next-line no-descending-specificity -- In this case ↕️, the code is more readable this way. +// stylelint-disable-next-line no-descending-specificity -- 7. .TextField__passwordToggle__icon--shown, .TextField__passwordToggle__button[aria-pressed='true'] > .TextField__passwordToggle__icon--hidden { display: none; } -// stylelint-disable selector-max-specificity, selector-max-class, selector-max-compound-selectors -- We need high-specificity selectors here. +// stylelint-disable selector-max-specificity, selector-max-class, selector-max-compound-selectors -- 8. .TextField > .TextField__input:focus-visible, .TextField > .TextField__passwordToggle > .TextField__input:focus-visible ~ .TextField__passwordToggle__button::before { @include form-fields-tools.box-field-focus-visible(); @@ -121,6 +151,7 @@ $_field-name: 'TextField'; outline: 0; } +// 9. .TextField__passwordToggle__button:focus-visible::after { content: ''; position: absolute; @@ -153,36 +184,31 @@ $_field-name: 'TextField'; @include form-fields-tools.box-field-disabled-input(); } -// stylelint-disable no-descending-specificity, selector-max-specificity, selector-max-class, selector-max-compound-selectors -// -- The code is more comprehensible when styling elements in this order: -// 1. input (with a pseudo-element), -// 2. password toggle button, -// 3. validation text. -// Also, high-specificity selectors are necessary to achieve the desired result. +// stylelint-disable no-descending-specificity, selector-max-specificity, selector-max-class, selector-max-compound-selectors -- 10. .TextField > .TextField__input:disabled, :is(.TextField--disabled, .TextField.is-disabled) > .TextField__input, .TextField .TextField__passwordToggle .TextField__input:disabled ~ .TextField__passwordToggle__button::before, :is(.TextField--disabled, .TextField.is-disabled) .TextField__passwordToggle .TextField__passwordToggle__button::before { - border-color: theme.$input-border-color-disabled; + border-color: form-fields-theme.$input-border-color-disabled; } .TextField .TextField__input:disabled ~ .TextField__passwordToggle__button, :is(.TextField--disabled, .TextField.is-disabled) .TextField__passwordToggle__button { @include form-fields-tools.input-disabled(); - background-color: form-fields-theme.$box-field-input-background-disabled; + background-color: form-fields-theme.$input-background-color-disabled; pointer-events: none; cursor: cursors.$disabled; } // stylelint-enable +:is(.TextField--disabled, .TextField.is-disabled) > :is(.TextField__helperText) { + @include form-fields-tools.helper-text-disabled(); +} + :is(.TextField--disabled, .TextField.is-disabled) > :is(.TextField__validationText, [data-spirit-element='validation_text']) { @include form-fields-tools.validation-text-disabled(); } - -:is(.TextField--disabled, .TextField.is-disabled) > :is(.TextField__helperText) { - @include form-fields-tools.helper-text-disabled(); -} diff --git a/packages/web/src/scss/components/TextField/_theme.scss b/packages/web/src/scss/components/TextField/_theme.scss index cf733cfc77..1686280c7e 100644 --- a/packages/web/src/scss/components/TextField/_theme.scss +++ b/packages/web/src/scss/components/TextField/_theme.scss @@ -1,7 +1,7 @@ -@use '@tokens' as tokens; +@use '@global' as global-tokens; +@use '../../settings/globals'; -$input-border-color-disabled: tokens.$border-primary-disabled; $input-number-arrows-width: 1.25rem; -$input-password-toggle-padding: tokens.$space-400; -$input-password-toggle-icon-size: tokens.$space-700; +$input-password-toggle-padding: global-tokens.$space-500; +$input-password-toggle-icon-size: global-tokens.$space-900; $input-size-characters: (2, 3, 4); diff --git a/packages/web/src/scss/components/index.scss b/packages/web/src/scss/components/index.scss index 4dcbe6460b..91c3233cfb 100644 --- a/packages/web/src/scss/components/index.scss +++ b/packages/web/src/scss/components/index.scss @@ -32,7 +32,8 @@ @forward 'Tag'; // @forward 'TextArea'; -// @forward 'TextField'; +@forward 'TextField'; + // @forward 'Toast'; // @forward 'Tooltip'; @forward 'UNSTABLE_ActionLayout'; diff --git a/packages/web/src/scss/theme/_form-fields.scss b/packages/web/src/scss/theme/_form-fields.scss index b79611ad63..755f41b09a 100644 --- a/packages/web/src/scss/theme/_form-fields.scss +++ b/packages/web/src/scss/theme/_form-fields.scss @@ -40,20 +40,20 @@ $inline-field-input-background-color-checked: var(--#{globals.$prefix}color-sele // Box field form components – TextField, TextArea, etc. // Commented out to allow nice diff -// $box-field-input-color-default: var(--#{globals.$prefix}color-form-field-content); -// $box-field-input-border-width: global-tokens.$border-width-100; -// $box-field-input-border-style: solid; -// $box-field-input-border-color-focus: var(--#{globals.$prefix}color-border-focus); -// $box-field-input-focus-shadow: global-tokens.$focus; -// -// $box-field-input-border-radius: global-tokens.$radius-300; -// $box-field-input-placeholder-color-default: var(--#{globals.$prefix}color-form-field-placeholder); -// $box-field-input-placeholder-color-disabled: var(--#{globals.$prefix}color-disabled-content); -// $box-field-input-padding-x: global-tokens.$space-600; -// $box-field-input-padding-y: calc(#{global-tokens.$space-500} - #{global-tokens.$border-width-100}); -// $box-field-input-width: 18rem; -// $box-field-label-typography: global-tokens.$body-small-semibold; -// $box-field-label-margin-bottom: global-tokens.$space-400; +$box-field-input-color-default: var(--#{globals.$prefix}color-form-field-content); +$box-field-input-border-width: global-tokens.$border-width-100; +$box-field-input-border-style: solid; +$box-field-input-border-color-focus: var(--#{globals.$prefix}color-border-focus); +$box-field-input-focus-shadow: global-tokens.$focus; + +$box-field-input-border-radius: global-tokens.$radius-300; +$box-field-input-placeholder-color-default: var(--#{globals.$prefix}color-form-field-placeholder); +$box-field-input-placeholder-color-disabled: var(--#{globals.$prefix}color-disabled-content); +$box-field-input-padding-x: global-tokens.$space-600; +$box-field-input-padding-y: calc(#{global-tokens.$space-500} - #{global-tokens.$border-width-100}); +$box-field-input-width: 18rem; +$box-field-label-typography: global-tokens.$body-small-semibold; +$box-field-label-margin-bottom: global-tokens.$space-400; $validation-states: ( success: ( diff --git a/packages/web/src/scss/tools/_form-fields.scss b/packages/web/src/scss/tools/_form-fields.scss index 472d88ff14..ecaa4684b0 100644 --- a/packages/web/src/scss/tools/_form-fields.scss +++ b/packages/web/src/scss/tools/_form-fields.scss @@ -101,14 +101,25 @@ padding: form-fields-theme.$box-field-input-padding-y form-fields-theme.$box-field-input-padding-x; color: form-fields-theme.$box-field-input-color-default; border: form-fields-theme.$box-field-input-border-width form-fields-theme.$box-field-input-border-style - form-fields-theme.$box-field-input-border-color-default; + form-fields-theme.$input-border-color; border-radius: form-fields-theme.$box-field-input-border-radius; - background: form-fields-theme.$box-field-input-background; + background: form-fields-theme.$input-background-color; &::placeholder { color: form-fields-theme.$box-field-input-placeholder-color-default; opacity: 1; } + + @media (hover: hover) { + &:hover { + background-color: form-fields-theme.$input-background-color-hover; + } + } + + &:active, + &:focus-within { + background-color: form-fields-theme.$input-background-color-active; + } } @mixin input-field-validation-states($field-name, $shadow-x: null, $shadow-y: null) { @@ -183,8 +194,8 @@ @mixin box-field-disabled-input() { @include input-disabled(); - border-color: form-fields-theme.$box-field-input-border-color-disabled; - background-color: form-fields-theme.$box-field-input-background-disabled; + border-color: form-fields-theme.$input-border-color-disabled; + background-color: form-fields-theme.$input-background-color-disabled; &::placeholder { color: form-fields-theme.$box-field-input-placeholder-color-disabled; diff --git a/tests/e2e/demo-components-compare.spec.ts b/tests/e2e/demo-components-compare.spec.ts index 1a4ad3410a..cdc4c1faa7 100644 --- a/tests/e2e/demo-components-compare.spec.ts +++ b/tests/e2e/demo-components-compare.spec.ts @@ -22,7 +22,6 @@ const IGNORED_TESTS: string[] = [ 'Stack', 'Text', 'TextArea', - 'TextField', 'Toast', 'Tooltip', 'UNSTABLE_Slider', diff --git a/tests/e2e/demo-components-compare.spec.ts-snapshots/textfield-chromium-linux.png b/tests/e2e/demo-components-compare.spec.ts-snapshots/textfield-chromium-linux.png index 582822c32a..0941b6f3c3 100644 Binary files a/tests/e2e/demo-components-compare.spec.ts-snapshots/textfield-chromium-linux.png and b/tests/e2e/demo-components-compare.spec.ts-snapshots/textfield-chromium-linux.png differ