diff --git a/playwright/core/expectScreenshotFixture.ts b/playwright/core/expectScreenshotFixture.ts index 49e9afba96..6e769247dd 100644 --- a/playwright/core/expectScreenshotFixture.ts +++ b/playwright/core/expectScreenshotFixture.ts @@ -29,6 +29,18 @@ export const expectScreenshotFixture: PlaywrightFixture const themes = paramsThemes || defaultParams.themes; + // Wait for loading of all the images + const locators = await page.locator('//img').all(); + await Promise.all( + locators.map((locator) => + locator.evaluate( + (image: HTMLImageElement) => + image.complete || + new Promise((resolve) => image.addEventListener('load', resolve)), + ), + ), + ); + if (themes?.includes('light')) { await page.emulateMedia({colorScheme: 'light'}); diff --git a/playwright/playwright.config.ts b/playwright/playwright.config.ts index 802d1b97cb..7c423e4aa2 100644 --- a/playwright/playwright.config.ts +++ b/playwright/playwright.config.ts @@ -33,7 +33,6 @@ const config: PlaywrightTestConfig = { updateSnapshots: process.env.UPDATE_REQUEST ? 'all' : 'missing', snapshotPathTemplate: '{testDir}/{testFileDir}/../__snapshots__/{testFileName}-snapshots/{arg}{-projectName}-linux{ext}', - /* The base directory, relative to the config file, for snapshot files created with toMatchSnapshot and toHaveScreenshot. */ /* Maximum time one test can run for. */ timeout: 10 * 1000, /* Run tests in files in parallel */ @@ -52,7 +51,6 @@ const config: PlaywrightTestConfig = { /* Collect trace when retrying the failed test. See https://playwright.dev/docs/trace-viewer */ trace: 'on-first-retry', headless: true, - /* Port to use for Playwright component endpoint. */ screenshot: 'only-on-failure', timezoneId: 'UTC', ctCacheDir: process.env.IS_DOCKER ? '.cache-docker' : '.cache', diff --git a/src/components/Avatar/Avatar.scss b/src/components/Avatar/Avatar.scss index 0ea1866bf3..6a1943d5d8 100644 --- a/src/components/Avatar/Avatar.scss +++ b/src/components/Avatar/Avatar.scss @@ -5,12 +5,14 @@ $block: '.#{variables.$ns}avatar'; #{$block} { --_--size: #{avatar-variables.$default-size}; - --_--background-color: var(--g-color-base-misc-light); + --_--border-width: 2px; + --_--inner-border-width: 3px; --_--border-color: currentColor; + --_--background-color: var(--g-color-base-misc-light); --_--color: var(--g-color-text-misc); + --_--font-weight: var(--g-text-body-font-weight); --_--font-size: var(--g-text-body-1-font-size); --_--line-height: var(--g-text-body-1-line-height); - --_--font-weight: var(--g-text-body-font-weight); overflow: hidden; display: inline-flex; @@ -21,28 +23,6 @@ $block: '.#{variables.$ns}avatar'; border-radius: 50%; background-color: var(--g-avatar-background-color, var(--_--background-color)); - &__image { - display: block; - width: 100%; - height: 100%; - object-fit: cover; - } - - &__icon { - color: var(--g-avatar-color, var(--_--color)); - - & > svg { - display: block; - } - } - - &__text { - color: var(--g-avatar-color, var(--_--color)); - font-size: var(--g-avatar-font-size, var(--_--font-size)); - line-height: var(--g-avatar-line-height, var(--_--line-height)); - font-weight: var(--_--font-weight); - } - &_with-border, &_view_outlined { position: relative; @@ -57,11 +37,13 @@ $block: '.#{variables.$ns}avatar'; } &::before { - border: 3px solid var(--g-color-base-background); + border: var(--g-avatar-inner-border-width, var(--_--inner-border-width)) solid + var(--g-color-base-background); } &::after { - border: 2px solid var(--g-avatar-border-color, var(--_--border-color)); + border: var(--g-avatar-border-width, var(--_--border-width)) solid + var(--g-avatar-border-color, var(--_--border-color)); } } @@ -72,30 +54,46 @@ $block: '.#{variables.$ns}avatar'; } } - &_2xs { + &_3xs, + &_2xs, + &_xs { + --_--font-weight: var(--g-text-caption-font-weight); --_--font-size: var(--g-text-caption-1-font-size); --_--line-height: var(--g-text-caption-1-line-height); - --_--font-weight: var(--g-text-caption-font-weight); } - &_xs, &_s { - --_--font-size: var(--g-text-caption-1-font-size); - --_--line-height: var(--g-text-caption-1-line-height); --_--font-weight: var(--g-text-caption-font-weight); + --_--font-size: var(--g-text-caption-2-font-size); + --_--line-height: var(--g-text-caption-2-line-height); } &_m, &_l { + --_--font-weight: var(--g-text-subheader-font-weight); --_--font-size: var(--g-text-subheader-1-font-size); --_--line-height: var(--g-text-subheader-1-line-height); - --_--font-weight: var(--g-text-subheader-font-weight); } &_xl { + --_--font-weight: var(--g-text-subheader-font-weight); --_--font-size: var(--g-text-subheader-2-font-size); --_--line-height: var(--g-text-subheader-2-line-height); - --_--font-weight: var(--g-text-subheader-font-weight); + } + + &_3xs, + &_2xs { + --_--border-width: 1.5px; + --_--inner-border-width: 2.5px; + } + + &_xs, + &_s, + &_m, + &_l, + &_xl { + --_--border-width: 2px; + --_--inner-border-width: 3px; } } @@ -130,4 +128,26 @@ $block: '.#{variables.$ns}avatar'; } } } + + &__image { + display: block; + width: 100%; + height: 100%; + object-fit: cover; + } + + &__icon { + color: var(--g-avatar-color, var(--_--color)); + + & > svg { + display: block; + } + } + + &__text { + color: var(--g-avatar-color, var(--_--color)); + font-weight: var(--g-avatar-font-weight, var(--_--font-weight)); + font-size: var(--g-avatar-font-size, var(--_--font-size)); + line-height: var(--g-avatar-line-height, var(--_--line-height)); + } } diff --git a/src/components/Avatar/Avatar.tsx b/src/components/Avatar/Avatar.tsx index 61b5cef88c..aa9c86f071 100644 --- a/src/components/Avatar/Avatar.tsx +++ b/src/components/Avatar/Avatar.tsx @@ -1,6 +1,7 @@ import React from 'react'; import {block} from '../utils/cn'; +import {filterDOMProps} from '../utils/filterDOMProps'; import {AvatarIcon} from './AvatarIcon'; import {AvatarImage} from './AvatarImage'; @@ -20,8 +21,6 @@ export const Avatar = React.forwardRef((props, ref) backgroundColor, borderColor, title, - 'aria-label': ariaLabel, - 'aria-labelledby': ariaLabelledby, className, style: styleProp, qa, @@ -75,11 +74,10 @@ export const Avatar = React.forwardRef((props, ref) className={b({size, theme, view, 'with-border': Boolean(borderColor)}, className)} title={title} role="img" - aria-label={ariaLabel} - aria-labelledby={ariaLabelledby} style={style} data-qa={qa} ref={ref} + {...filterDOMProps(props, {labelable: true})} > {renderContent()} diff --git a/src/components/Avatar/AvatarIcon/AvatarIcon.tsx b/src/components/Avatar/AvatarIcon/AvatarIcon.tsx index b34625c0ff..62ae916292 100644 --- a/src/components/Avatar/AvatarIcon/AvatarIcon.tsx +++ b/src/components/Avatar/AvatarIcon/AvatarIcon.tsx @@ -6,6 +6,7 @@ import type {AvatarSize} from '../types/common'; import type {AvatarIconProps} from './types'; const avatarSizeToIconSize: Record = { + '3xs': 10, '2xs': 12, xs: 14, s: 16, diff --git a/src/components/Avatar/AvatarText/AvatarText.tsx b/src/components/Avatar/AvatarText/AvatarText.tsx index 05f3c9cb92..bd4cb5d38f 100644 --- a/src/components/Avatar/AvatarText/AvatarText.tsx +++ b/src/components/Avatar/AvatarText/AvatarText.tsx @@ -3,9 +3,9 @@ import React from 'react'; import type {AvatarTextProps} from './types'; import {getAvatarDisplayText} from './utils'; -export const AvatarText = ({text, color, className}: AvatarTextProps) => { +export const AvatarText = ({text, color, size, className}: AvatarTextProps) => { const style = {color}; - const displayText = getAvatarDisplayText(text); + const displayText = getAvatarDisplayText(text, size); return (
diff --git a/src/components/Avatar/AvatarText/utils.ts b/src/components/Avatar/AvatarText/utils.ts index 1ab9966299..de72a1f20a 100644 --- a/src/components/Avatar/AvatarText/utils.ts +++ b/src/components/Avatar/AvatarText/utils.ts @@ -1,7 +1,15 @@ -export const getAvatarDisplayText = (text: string) => { - const words = text.split(/\s+/); - const result = - words.length > 1 ? [words[0][0], words[1][0]].filter(Boolean).join('') : text.slice(0, 2); +import type {AvatarSize} from '../types/common'; - return result.toUpperCase(); +export const getAvatarDisplayText = (text: string, size: AvatarSize) => { + if (size === '3xs') { + return text[0].toUpperCase(); + } + + const words = text.split(/[^a-zA-Z]+/); + + if (words.length <= 1) { + return text.slice(0, 2).toUpperCase(); + } + + return [words[0][0], words[1][0]].filter(Boolean).join('').toUpperCase(); }; diff --git a/src/components/Avatar/README.md b/src/components/Avatar/README.md index 63f2edbcce..63d0c8e2bf 100644 --- a/src/components/Avatar/README.md +++ b/src/components/Avatar/README.md @@ -141,12 +141,13 @@ LANDING_BLOCK--> ### Size -To control the size of the `Avatar` use the `size` property. The default size is `m`. Possible values: `2xs`, `xs`, `s`, `m`, `l`, `xl`. +To control the size of the `Avatar` use the `size` property. The default size is `m`. Possible values: `3xs`, `2xs`, `xs`, `s`, `m`, `l`, `xl`. ### Common -| Name | Description | Type | Default | -| :-------------- | :-------------------------------------- | :-------------------------------------: | :------: | -| size | Avatar size | `'2xs'` `'xs'` `'s'` `'m'` `'l'` `'xl'` | `m` | -| theme | Avatar theme | `'normal'` `'brand'` | `normal` | -| view | Avatar view | `'filled'` `'outlined'` | `filled` | -| backgroundColor | Custom background color | `string` | | -| borderColor | Custom border color | `string` | | -| title | HTML `title` attributes | `string` | | -| aria-label | `aria-label` for avatar block | `string` | | -| aria-labelledby | `aria-labelledby` for avatar block | `string` | | -| className | Custom CSS class for root element | `string` | | -| style | HTML style attribute | `React.CSSProperties` | | -| qa | HTML `data-qa` attribute, used in tests | `string` | | +| Name | Description | Type | Default | +| :-------------- | :-------------------------------------- | :---------------------------------------------: | :------: | +| size | Avatar size | `'3xs'` `'2xs'` `'xs'` `'s'` `'m'` `'l'` `'xl'` | `m` | +| theme | Avatar theme | `'normal'` `'brand'` | `normal` | +| view | Avatar view | `'filled'` `'outlined'` | `filled` | +| backgroundColor | Custom background color | `string` | | +| borderColor | Custom border color | `string` | | +| title | HTML `title` attributes | `string` | | +| aria-label | `aria-label` for avatar block | `string` | | +| aria-labelledby | `aria-labelledby` for avatar block | `string` | | +| className | Custom CSS class for root element | `string` | | +| style | HTML style attribute | `React.CSSProperties` | | +| qa | HTML `data-qa` attribute, used in tests | `string` | | ### Image-specific @@ -210,11 +212,14 @@ LANDING_BLOCK--> ## CSS API -| Name | Description | -| :---------------------------- | :---------------------- | -| `--g-avatar-size` | Size (width and height) | -| `--g-avatar-background-color` | Background color | -| `--g-avatar-border-color` | Border color | -| `--g-avatar-color` | Icon and text color | -| `--g-avatar-font-size` | Text font size | -| `--g-avatar-line-height` | Text line height | +| Name | Description | +| :------------------------------ | :---------------------- | +| `--g-avatar-size` | Size (width and height) | +| `--g-avatar-border-width` | Border width | +| `--g-avatar-inner-border-width` | Inner border width | +| `--g-avatar-border-color` | Border color | +| `--g-avatar-background-color` | Background color | +| `--g-avatar-color` | Icon and text color | +| `--g-avatar-font-weight` | Text font weight | +| `--g-avatar-font-size` | Text font size | +| `--g-avatar-line-height` | Text line height | diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-1-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-1-chromium-linux.png deleted file mode 100644 index d849f90c6b..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-1-chromium-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-1-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-1-webkit-linux.png deleted file mode 100644 index 0c343c9af8..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-1-webkit-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-dark-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-dark-chromium-linux.png new file mode 100644 index 0000000000..12b15646da Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-dark-chromium-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-dark-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-dark-webkit-linux.png new file mode 100644 index 0000000000..71fa4831a4 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-dark-webkit-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-light-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-light-chromium-linux.png new file mode 100644 index 0000000000..4ac0172d27 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-light-chromium-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-light-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-light-webkit-linux.png new file mode 100644 index 0000000000..32497e5eb6 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Icon-light-webkit-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-1-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-1-chromium-linux.png deleted file mode 100644 index 8a7a8b192f..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-1-chromium-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-1-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-1-webkit-linux.png deleted file mode 100644 index 603f4ccf56..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-1-webkit-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-dark-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-dark-chromium-linux.png new file mode 100644 index 0000000000..444fb52a1b Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-dark-chromium-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-dark-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-dark-webkit-linux.png new file mode 100644 index 0000000000..fae6b482bc Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-dark-webkit-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-light-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-light-chromium-linux.png new file mode 100644 index 0000000000..283d5c677e Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-light-chromium-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-light-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-light-webkit-linux.png new file mode 100644 index 0000000000..810c08caa1 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Image-light-webkit-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-ImageFallback-1-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-ImageFallback-1-chromium-linux.png deleted file mode 100644 index 8a7a8b192f..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-ImageFallback-1-chromium-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-ImageFallback-dark-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-ImageFallback-dark-chromium-linux.png new file mode 100644 index 0000000000..444fb52a1b Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-ImageFallback-dark-chromium-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-ImageFallback-light-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-ImageFallback-light-chromium-linux.png new file mode 100644 index 0000000000..283d5c677e Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-ImageFallback-light-chromium-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-1-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-1-chromium-linux.png deleted file mode 100644 index e8d1040607..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-1-chromium-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-1-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-1-webkit-linux.png deleted file mode 100644 index 32c6a4708c..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-1-webkit-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-dark-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-dark-chromium-linux.png new file mode 100644 index 0000000000..1813285d3f Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-dark-chromium-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-dark-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-dark-webkit-linux.png new file mode 100644 index 0000000000..a8fdcbe603 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-dark-webkit-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-light-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-light-chromium-linux.png new file mode 100644 index 0000000000..0c01dcb1f3 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-light-chromium-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-light-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-light-webkit-linux.png new file mode 100644 index 0000000000..3cbdbbd05a Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Showcase-light-webkit-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-1-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-1-chromium-linux.png deleted file mode 100644 index e662f2dc70..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-1-chromium-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-1-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-1-webkit-linux.png deleted file mode 100644 index ecef4ac4b8..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-1-webkit-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-dark-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-dark-chromium-linux.png new file mode 100644 index 0000000000..812a3c92b2 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-dark-chromium-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-dark-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-dark-webkit-linux.png new file mode 100644 index 0000000000..d31e459c07 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-dark-webkit-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-light-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-light-chromium-linux.png new file mode 100644 index 0000000000..9e90e8b9f1 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-light-chromium-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-light-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-light-webkit-linux.png new file mode 100644 index 0000000000..e76113f3b8 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-Text-light-webkit-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-1-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-1-chromium-linux.png deleted file mode 100644 index e8015e0799..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-1-chromium-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-1-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-1-webkit-linux.png deleted file mode 100644 index e9619e35bd..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-1-webkit-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-dark-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-dark-chromium-linux.png new file mode 100644 index 0000000000..f7c116f757 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-dark-chromium-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-dark-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-dark-webkit-linux.png new file mode 100644 index 0000000000..2f839e05be Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-dark-webkit-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-light-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-light-chromium-linux.png new file mode 100644 index 0000000000..af5ec55afb Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-light-chromium-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-light-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-light-webkit-linux.png new file mode 100644 index 0000000000..84dabf3b54 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-TextInitials-light-webkit-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-1-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-1-chromium-linux.png deleted file mode 100644 index d279870575..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-1-chromium-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-1-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-1-webkit-linux.png deleted file mode 100644 index f7f0b7ce61..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-1-webkit-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-dark-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-dark-chromium-linux.png new file mode 100644 index 0000000000..51dd7b4922 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-dark-chromium-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-dark-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-dark-webkit-linux.png new file mode 100644 index 0000000000..a04e1fc47c Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-dark-webkit-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-light-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-light-chromium-linux.png new file mode 100644 index 0000000000..30406f8093 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-light-chromium-linux.png differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-light-webkit-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-light-webkit-linux.png new file mode 100644 index 0000000000..ae345285b0 Binary files /dev/null and b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-render-story-WithBorder-light-webkit-linux.png differ diff --git a/src/components/Avatar/__stories__/Avatar.stories.tsx b/src/components/Avatar/__stories__/Avatar.stories.tsx index 9cf2207c4d..c1b56a4b23 100644 --- a/src/components/Avatar/__stories__/Avatar.stories.tsx +++ b/src/components/Avatar/__stories__/Avatar.stories.tsx @@ -54,12 +54,14 @@ const randomAvatars = faker.helpers const imageProps = { imgUrl, + alt: 'Sample image', }; const iconProps = { backgroundColor: 'var(--g-color-base-brand)', icon: FaceRobot, color: 'var(--g-color-text-brand-contrast)', + 'aria-label': 'Sample icon', }; const textProps = { @@ -132,9 +134,8 @@ export const TextInitials: Story = { export const WithBorder: Story = { args: { - imgUrl, + ...imageProps, borderColor: 'var(--g-color-line-misc)', - alt: 'Sample image', 'aria-label': 'Image with border', }, }; @@ -145,52 +146,33 @@ export const AvatarShowcase: Story = { return ( + + + - + - + - + - + - + + + + @@ -198,7 +180,6 @@ export const AvatarShowcase: Story = { {...imageProps} size="2xs" borderColor={BORDER_COLOR} - alt="Sample image" aria-label="Avatar with 2XS size and border" /> @@ -207,7 +188,6 @@ export const AvatarShowcase: Story = { {...imageProps} size="xs" borderColor={BORDER_COLOR} - alt="Sample image" aria-label="Avatar with XS size and border" /> @@ -216,7 +196,6 @@ export const AvatarShowcase: Story = { {...imageProps} size="s" borderColor={BORDER_COLOR} - alt="Sample image" aria-label="Avatar with S size and border" /> @@ -225,7 +204,6 @@ export const AvatarShowcase: Story = { {...imageProps} size="m" borderColor={BORDER_COLOR} - alt="Sample image" aria-label="Avatar with M size and border" /> @@ -234,7 +212,6 @@ export const AvatarShowcase: Story = { {...imageProps} size="l" borderColor={BORDER_COLOR} - alt="Sample image" aria-label="Avatar with L size and border" /> @@ -243,80 +220,58 @@ export const AvatarShowcase: Story = { {...imageProps} size="xl" borderColor={BORDER_COLOR} - alt="Sample image" aria-label="Avatar with XL size and border" /> + + + - + - + - + - + - + - + + + + - + - + - + - + - + - + + + + @@ -335,6 +290,14 @@ export const AvatarShowcase: Story = { + + + { - test('render story: ', async ({mount}) => { + test('render story: ', async ({mount, expectScreenshot}) => { const component = await mount(); - - await expect(component).toHaveScreenshot(); + await expectScreenshot({component}); }); - test('render story: ', async ({mount, browserName}) => { + test('render story: ', async ({browserName, mount, expectScreenshot}) => { test.skip(browserName === 'webkit', 'Test is flaky for webkit'); const component = await mount(); - - await expect(component).toHaveScreenshot(); + await expectScreenshot({component}); }); - test('render story: ', async ({mount}) => { + test('render story: ', async ({mount, expectScreenshot}) => { const component = await mount(); - - await expect(component).toHaveScreenshot(); + await expectScreenshot({component}); }); - test('render story: ', async ({mount}) => { + test('render story: ', async ({mount, expectScreenshot}) => { const component = await mount(); - - await expect(component).toHaveScreenshot(); + await expectScreenshot({component}); }); - test('render story: ', async ({mount}) => { + test('render story: ', async ({mount, expectScreenshot}) => { const component = await mount(); - - await expect(component).toHaveScreenshot(); + await expectScreenshot({component}); }); - test('render story: ', async ({mount}) => { + test('render story: ', async ({mount, expectScreenshot}) => { const component = await mount(); - - await expect(component).toHaveScreenshot(); + await expectScreenshot({component}); }); - test('render story: ', async ({mount}) => { + test('render story: ', async ({mount, expectScreenshot}) => { const component = await mount(); - - await expect(component).toHaveScreenshot(); + await expectScreenshot({component}); }); }); diff --git a/src/components/Avatar/_variables.scss b/src/components/Avatar/_variables.scss index 957c60eb0b..ba9acd3e19 100644 --- a/src/components/Avatar/_variables.scss +++ b/src/components/Avatar/_variables.scss @@ -1,6 +1,7 @@ @use 'sass:map'; $sizes: ( + '3xs': 16px, '2xs': 20px, 'xs': 24px, 's': 28px, diff --git a/src/components/Avatar/constants.ts b/src/components/Avatar/constants.ts index 0b470c5be9..a127e58a44 100644 --- a/src/components/Avatar/constants.ts +++ b/src/components/Avatar/constants.ts @@ -1,6 +1,7 @@ import type {AvatarSize} from './types/common'; export const AVATAR_SIZES: Record = { + '3xs': 16, '2xs': 20, xs: 24, s: 28, diff --git a/src/components/Avatar/types/common.ts b/src/components/Avatar/types/common.ts index 6d3a51a1ff..ce41d14ac6 100644 --- a/src/components/Avatar/types/common.ts +++ b/src/components/Avatar/types/common.ts @@ -1,6 +1,7 @@ -export type AvatarSize = '2xs' | 'xs' | 's' | 'm' | 'l' | 'xl'; +import type {DOMProps} from '../../types'; -export interface AvatarCommonProps { +export type AvatarSize = '3xs' | '2xs' | 'xs' | 's' | 'm' | 'l' | 'xl'; + +export interface AvatarCommonProps extends Pick { size: AvatarSize; - className?: string; } diff --git a/src/components/Avatar/types/main.ts b/src/components/Avatar/types/main.ts index 17037a6e47..f5fb575296 100644 --- a/src/components/Avatar/types/main.ts +++ b/src/components/Avatar/types/main.ts @@ -1,5 +1,5 @@ import type {DistributiveOmit} from '../../../types/utils'; -import type {DOMProps, QAProps} from '../../types'; +import type {AriaLabelingProps, DOMProps, QAProps} from '../../types'; import type {AvatarIconProps} from '../AvatarIcon'; import type {AvatarImageProps} from '../AvatarImage'; import type {AvatarTextProps} from '../AvatarText'; @@ -9,12 +9,7 @@ import type {AvatarCommonProps, AvatarSize} from './common'; export type AvatarTheme = 'normal' | 'brand'; export type AvatarView = 'filled' | 'outlined'; -interface AvatarAriaProps { - 'aria-label'?: string; - 'aria-labelledby'?: string; -} - -interface AvatarBaseProps extends DOMProps, QAProps, AvatarAriaProps { +interface AvatarBaseProps extends AriaLabelingProps, DOMProps, QAProps { size?: AvatarSize; theme?: AvatarTheme; view?: AvatarView; @@ -24,7 +19,4 @@ interface AvatarBaseProps extends DOMProps, QAProps, AvatarAriaProps { } export type AvatarProps = AvatarBaseProps & - DistributiveOmit< - AvatarImageProps | AvatarIconProps | AvatarTextProps | AvatarAriaProps, - keyof AvatarCommonProps - >; + DistributiveOmit; diff --git a/src/components/User/README.md b/src/components/User/README.md index 94f6f4060b..b67670c229 100644 --- a/src/components/User/README.md +++ b/src/components/User/README.md @@ -28,7 +28,7 @@ LANDING_BLOCK--> ## Size -To control the size of the `User` use the `size` property. The default size is `m`. Possible values: `2xs`, `xs`, `s`, `m`, `l`, `xl`. +To control the size of the `User` use the `size` property. The default size is `m`. Possible values: `3xs`, `2xs`, `xs`, `s`, `m`, `l`, `xl`. This propeperty passes to the internal `Avatar` component too. @@ -36,6 +36,7 @@ This propeperty passes to the internal `Avatar` component too. @@ -44,6 +45,7 @@ This propeperty passes to the internal `Avatar` component too. `} > + @@ -56,21 +58,26 @@ LANDING_BLOCK--> ## Properties -| Name | Description | Type | Default | -| :-------------- | :-------------------------------------- | :----------------------------------------------------------------: | :-----: | -| avatar | User avatar | [AvatarProps](../Avatar/README.md#properties) `React.ReactElement` | | -| name | User name | `React.ReactNode` | | -| description | User description | `React.ReactNode` | | -| size | User block size | `'2xs'` `'xs'` `'s'` `'m'` `'l'` `'xl'` | `m` | -| aria-label | `aria-label` for user block | `string` | | -| aria-labelledby | `aria-labelledby` for user block | `string` | | -| className | Custom CSS class for root element | `string` | | -| style | HTML style attribute | `React.CSSProperties` | | -| qa | HTML `data-qa` attribute, used in tests | `string` | | +| Name | Description | Type | Default | +| :-------------- | :-------------------------------------- | :-------------------------------------------------------------------------: | :-----: | +| avatar | User avatar | [AvatarProps](../Avatar/README.md#properties) `string` `React.ReactElement` | | +| name | User name | `React.ReactNode` | | +| description | User description | `React.ReactNode` | | +| size | User block size | `'3xs'` `'2xs'` `'xs'` `'s'` `'m'` `'l'` `'xl'` | `m` | +| aria-label | `aria-label` for user block | `string` | | +| aria-labelledby | `aria-labelledby` for user block | `string` | | +| className | Custom CSS class for root element | `string` | | +| style | HTML style attribute | `React.CSSProperties` | | +| qa | HTML `data-qa` attribute, used in tests | `string` | | ## CSS API -| Name | Description | -| :--------------------- | :------------------------------- | -| `--g-user-font-size` | Name and description font size | -| `--g-user-line-height` | Name and description line height | +| Name | Description | +| :--------------------------------- | :-------------------------------- | +| `--g-user-gap` | Gap between avatar and text block | +| `--g-user-name-font-weight` | Name font weight | +| `--g-user-name-font-size` | Name font size | +| `--g-user-name-line-height` | Name line height | +| `--g-user-description-font-weight` | Description font weight | +| `--g-user-description-font-size` | Description font size | +| `--g-user-description-line-height` | Description line height | diff --git a/src/components/User/User.scss b/src/components/User/User.scss index 962040ee19..59b6c07449 100644 --- a/src/components/User/User.scss +++ b/src/components/User/User.scss @@ -3,22 +3,51 @@ $block: '.#{variables.$ns}user'; -@mixin user-text() { - @include mixins.text-body-short(); - - font-size: var(--g-user-font-size, var(--g-text-body-short-font-size)); - line-height: var(--g-user-line-height, var(--g-text-body-short-line-height)); -} - -@mixin user-text-small() { - font-size: var(--g-user-font-size, var(--g-text-code-inline-1-font-size)); - line-height: var(--g-user-line-height, var(--g-text-code-inline-1-line-height)); - font-weight: var(--g-text-body-font-weight); -} - #{$block} { + --_--gap: 8px; + --_--name-font-weight: var(--g-text-body-font-weight); + --_--name-font-size: var(--g-text-body-short-font-size); + --_--name-line-height: var(--g-text-body-short-line-height); + --_--description-font-weight: var(--g-text-body-font-weight); + --_--description-font-size: var(--g-text-body-short-font-size); + --_--description-line-height: var(--g-text-body-short-line-height); + display: flex; align-items: center; + gap: var(--g-user-gap, var(--_--gap)); + + &_size { + &_3xs, + &_2xs, + &_xs, + &_s { + --_--gap: 6px; + } + + &_m { + --_--gap: 8px; + } + + &_l, + &_xl { + --_--gap: 12px; + } + + &_3xs, + &_2xs, + &_xs, + &_s, + &_m, + &_l { + --_--name-font-size: var(--g-text-body-short-font-size); + --_--name-line-height: var(--g-text-body-short-line-height); + } + + &_xl { + --_--name-font-size: var(--g-text-body-2-font-size); + --_--name-line-height: var(--g-text-body-2-line-height); + } + } &__avatar { display: flex; @@ -31,35 +60,22 @@ $block: '.#{variables.$ns}user'; flex-direction: column; min-width: 0; - @include user-text(); - #{$block}__name { - color: var(--g-color-text-primary); @include mixins.overflow-ellipsis(); + + color: var(--g-color-text-primary); + font-weight: var(--g-user-name-font-weight, var(--_--name-font-weight)); + font-size: var(--g-user-name-font-size, var(--_--name-font-size)); + line-height: var(--g-user-name-line-height, var(--_--name-line-height)); } #{$block}__description { - color: var(--g-color-text-secondary); @include mixins.overflow-ellipsis(); - } - } - - &__avatar + &__info { - margin-inline-start: 12px; - } - - &_size { - &_xs, - &_2xs { - #{$block}__avatar + #{$block}__info { - margin-inline-start: 6px; - } - } - &_2xs { - #{$block}__info { - @include user-text-small(); - } + color: var(--g-color-text-secondary); + font-weight: var(--g-user-description-font-weight, var(--_--description-font-weight)); + font-size: var(--g-user-description-font-size, var(--_--description-font-size)); + line-height: var(--g-user-description-line-height, var(--_--description-line-height)); } } } diff --git a/src/components/User/User.tsx b/src/components/User/User.tsx index 27494bc830..7761182a30 100644 --- a/src/components/User/User.tsx +++ b/src/components/User/User.tsx @@ -3,7 +3,7 @@ import React from 'react'; import {Avatar} from '../Avatar'; import {block} from '../utils/cn'; -import {COMPACT_SIZES, DEFAULT_SIZE} from './constants'; +import {COMPACT_SIZES, DEFAULT_USER_SIZE} from './constants'; import type {UserProps} from './types'; import './User.scss'; @@ -21,7 +21,7 @@ export const User = React.forwardRef( avatar, name, description, - size = DEFAULT_SIZE, + size = DEFAULT_USER_SIZE, 'aria-label': ariaLabel, 'aria-labelledby': ariaLabelledby, className, @@ -30,6 +30,16 @@ export const User = React.forwardRef( }, ref, ) => { + let avatarView: React.ReactNode = null; + + if (typeof avatar === 'string') { + avatarView = ; + } else if (React.isValidElement(avatar)) { + avatarView = avatar; + } else if (avatar) { + avatarView = ; + } + const showDescription = Boolean(description && !COMPACT_SIZES.includes(size)); return ( @@ -41,11 +51,7 @@ export const User = React.forwardRef( data-qa={qa} ref={ref} > - {avatar ? ( -
- {React.isValidElement(avatar) ? avatar : } -
- ) : null} + {avatarView ?
{avatarView}
: null} {name || showDescription ? (
{name ? ( diff --git a/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-1-chromium-linux.png b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-1-chromium-linux.png deleted file mode 100644 index 81de70c937..0000000000 Binary files a/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-1-chromium-linux.png and /dev/null differ diff --git a/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-1-webkit-linux.png b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-1-webkit-linux.png deleted file mode 100644 index 5af295c7b6..0000000000 Binary files a/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-1-webkit-linux.png and /dev/null differ diff --git a/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-dark-chromium-linux.png b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-dark-chromium-linux.png new file mode 100644 index 0000000000..d5277197e5 Binary files /dev/null and b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-dark-chromium-linux.png differ diff --git a/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-dark-webkit-linux.png b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-dark-webkit-linux.png new file mode 100644 index 0000000000..f0b51fbb85 Binary files /dev/null and b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-dark-webkit-linux.png differ diff --git a/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-light-chromium-linux.png b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-light-chromium-linux.png new file mode 100644 index 0000000000..6a81950a45 Binary files /dev/null and b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-light-chromium-linux.png differ diff --git a/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-light-webkit-linux.png b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-light-webkit-linux.png new file mode 100644 index 0000000000..bd48c048c3 Binary files /dev/null and b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Default-light-webkit-linux.png differ diff --git a/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Showcase-dark-chromium-linux.png b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Showcase-dark-chromium-linux.png new file mode 100644 index 0000000000..fb7ea680cb Binary files /dev/null and b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Showcase-dark-chromium-linux.png differ diff --git a/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Showcase-dark-webkit-linux.png b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Showcase-dark-webkit-linux.png new file mode 100644 index 0000000000..ad0e1926ab Binary files /dev/null and b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Showcase-dark-webkit-linux.png differ diff --git a/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Showcase-light-chromium-linux.png b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Showcase-light-chromium-linux.png new file mode 100644 index 0000000000..227a87ae09 Binary files /dev/null and b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Showcase-light-chromium-linux.png differ diff --git a/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Showcase-light-webkit-linux.png b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Showcase-light-webkit-linux.png new file mode 100644 index 0000000000..a1b419f6fb Binary files /dev/null and b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-render-story-Showcase-light-webkit-linux.png differ diff --git a/src/components/User/__stories__/User.stories.tsx b/src/components/User/__stories__/User.stories.tsx index 13efa861a4..2f1bb6adbc 100644 --- a/src/components/User/__stories__/User.stories.tsx +++ b/src/components/User/__stories__/User.stories.tsx @@ -1,5 +1,9 @@ +import React from 'react'; + import type {Meta, StoryObj} from '@storybook/react'; +import {Showcase} from '../../../demo/Showcase'; +import {ShowcaseItem} from '../../../demo/ShowcaseItem'; import {User} from '../User'; const meta: Meta = { @@ -24,14 +28,47 @@ export default meta; type Story = StoryObj; +const commonProps = { + avatar: { + imgUrl: '', + 'aria-label': "Isaac's avatar", + alt: 'Isaac', + }, + name: 'Isaac', + description: 'user@gravity-ui.com', +}; + export const Default: Story = { - args: { - avatar: { - imgUrl: '', - 'aria-label': 'Avatar of user@gravity-ui.com', - alt: 'Isaac', - }, - name: 'Isaac', - description: 'user@gravity-ui.com', + args: commonProps, +}; + +export const UserShowcase: Story = { + name: 'Showcase', + render: () => { + return ( + + + + + + + + + + + + + + + + + + + + + + + + ); }, }; diff --git a/src/components/User/__tests__/User.visual.test.tsx b/src/components/User/__tests__/User.visual.test.tsx index c94f096430..fce69e988f 100644 --- a/src/components/User/__tests__/User.visual.test.tsx +++ b/src/components/User/__tests__/User.visual.test.tsx @@ -1,15 +1,17 @@ import React from 'react'; -import {expect} from '@playwright/experimental-ct-react'; - import {test} from '~playwright/core'; -import {Default} from './stories'; +import {UserStories} from './stories'; test.describe('User', () => { - test('render story: ', async ({mount}) => { - const component = await mount(); + test('render story: ', async ({mount, expectScreenshot}) => { + const component = await mount(); + await expectScreenshot({component}); + }); - await expect(component).toHaveScreenshot(); + test('render story: ', async ({mount, expectScreenshot}) => { + const component = await mount(); + await expectScreenshot({component}); }); }); diff --git a/src/components/User/__tests__/stories.ts b/src/components/User/__tests__/stories.ts index ad381f1b5e..df1422953f 100644 --- a/src/components/User/__tests__/stories.ts +++ b/src/components/User/__tests__/stories.ts @@ -2,4 +2,4 @@ import {composeStories} from '@storybook/react'; import * as Stories from '../__stories__/User.stories'; -export const {Default} = composeStories(Stories); +export const UserStories = composeStories(Stories); diff --git a/src/components/User/constants.ts b/src/components/User/constants.ts index 91ec59451f..be628892a4 100644 --- a/src/components/User/constants.ts +++ b/src/components/User/constants.ts @@ -1,4 +1,5 @@ import type {UserSize} from './types'; -export const COMPACT_SIZES: UserSize[] = ['xs', '2xs']; -export const DEFAULT_SIZE: UserSize = 'm'; +export const DEFAULT_USER_SIZE: UserSize = 'm'; + +export const COMPACT_SIZES: UserSize[] = ['xs', '2xs', '3xs']; diff --git a/src/components/User/index.ts b/src/components/User/index.ts index b2563fe154..930f0f616b 100644 --- a/src/components/User/index.ts +++ b/src/components/User/index.ts @@ -1,2 +1,3 @@ export type {UserSize, UserProps} from './types'; +export {DEFAULT_USER_SIZE} from './constants'; export {User} from './User'; diff --git a/src/components/User/types.ts b/src/components/User/types.ts index 5d1c0f80c3..cdc488bb9e 100644 --- a/src/components/User/types.ts +++ b/src/components/User/types.ts @@ -4,10 +4,13 @@ import type {DistributiveOmit} from '../../types/utils'; import type {AvatarProps} from '../Avatar'; import type {DOMProps, QAProps} from '../types'; -export type UserSize = '2xs' | 'xs' | 's' | 'm' | 'l' | 'xl'; +export type UserSize = '3xs' | '2xs' | 'xs' | 's' | 'm' | 'l' | 'xl'; export interface UserProps extends DOMProps, QAProps { - avatar?: DistributiveOmit | React.ReactElement; + avatar?: + | DistributiveOmit + | string + | React.ReactElement; name?: React.ReactNode; description?: React.ReactNode; size?: UserSize; diff --git a/src/components/UserLabel/README.md b/src/components/UserLabel/README.md index bc3a610712..df6a9636f4 100644 --- a/src/components/UserLabel/README.md +++ b/src/components/UserLabel/README.md @@ -89,15 +89,16 @@ LANDING_BLOCK--> | Name | Description | Type | Default | | :----------- | :------------------------------------------ | :-------------------------------------------------------------------------: | :----------: | | type | Avatar appearance | `'person'` `'email'` `'empty'` | `'person'` | -| avatar | User avatar | [AvatarProps](../Avatar/README.md#properties) `string` `React.ReactElement` | | -| children | Visible text | `React.ReactNode` | | | view | UserLabel view | `'outlined'` `'clear'` | `'outlined'` | +| size | Avatar size | `'3xs'` `'2xs'` `'xs'` `'s'` `'m'` `'l'` `'xl'` | `'s'` | +| avatar | User avatar | [AvatarProps](../Avatar/README.md#properties) `string` `React.ReactElement` | | +| name | User name | `React.ReactNode` | | +| description | User description | `React.ReactNode` | | | onClick | `click` event handler for component itself | `Function` | | | onCloseClick | `click` event handler for button with cross | `Function` | | | className | Custom CSS class for root element | `string` | | | style | HTML style attribute | `React.CSSProperties` | | | qa | HTML `data-qa` attribute, used in tests | `string` | | -| size | Avatar size | `'xs'` `'s'` `'m'` `'l'` `'xl'` | `'s'` | ## CSS API diff --git a/src/components/UserLabel/UserLabel.scss b/src/components/UserLabel/UserLabel.scss index f01d582877..edcc801b67 100644 --- a/src/components/UserLabel/UserLabel.scss +++ b/src/components/UserLabel/UserLabel.scss @@ -1,3 +1,4 @@ +@use 'sass:map'; @use '../../../styles/mixins'; @use '../variables'; @use '../Avatar/variables' as avatar-variables; @@ -8,25 +9,36 @@ $block: '.#{variables.$ns}user-label'; $transitionDuration: 0.1s; $transitionTimingFunction: ease-in-out; + --_--size: #{map.get(avatar-variables.$sizes, 's')}; + --_--border-radius: 25px; + --_--outer-gap: 12px; + --_--inner-gap: 10px; + --_--text-font-weight: var(--g-text-body-font-weight); + --_--text-font-size: var(--g-text-body-short-font-size); + --_--text-line-height: var(--g-text-body-short-line-height); + --_--description-font-weight: var(--g-text-body-font-weight); + --_--description-font-size: var(--g-text-body-short-font-size); + --_--description-line-height: var(--g-text-body-short-line-height); + position: relative; z-index: 0; display: inline-flex; max-width: 100%; - height: 28px; - border-radius: 20px; + height: var(--g-user-label-size, var(--_--size)); + border-radius: var(--_--border-radius); transition-property: background-color; transition-duration: $transitionDuration; transition-timing-function: $transitionTimingFunction; &_view_outlined { - &:after { + &::after { + content: ''; position: absolute; z-index: -1; inset: 0; - content: ''; border: 1px solid var(--g-color-line-generic); - border-radius: 20px; + border-radius: var(--_--border-radius); transition-property: border-color; transition-duration: $transitionDuration; @@ -34,15 +46,71 @@ $block: '.#{variables.$ns}user-label'; } } - &_empty { - padding-inline-start: 12px; + &_size { + @each $size-name, $size-value in avatar-variables.$sizes { + &_#{$size-name} { + --_--size: #{$size-value}; + } + } + + &_3xs { + --_--outer-gap: 6px; + --_--inner-gap: 4px; + } + + &_2xs, + &_xs { + --_--outer-gap: 8px; + --_--inner-gap: 6px; + } + + &_s { + --_--outer-gap: 12px; + --_--inner-gap: 8px; + } + + &_m { + --_--outer-gap: 12px; + --_--inner-gap: 10px; + } + + &_l { + --_--outer-gap: 16px; + --_--inner-gap: 12px; + } + + &_xl { + --_--outer-gap: 20px; + --_--inner-gap: 12px; + } + + &_3xs, + &_2xs { + --_--text-font-weight: var(--g-text-caption-font-weight); + --_--text-font-size: var(--g-text-caption-2-font-size); + --_--text-line-height: var(--g-text-caption-2-line-height); + } + + &_xs, + &_s, + &_m, + &_l { + --_--text-font-weight: var(--g-text-body-font-weight); + --_--text-font-size: var(--g-text-body-short-font-size); + --_--text-line-height: var(--g-text-body-short-line-height); + } + + &_xl { + --_--text-font-weight: var(--g-text-body-font-weight); + --_--text-font-size: var(--g-text-body-2-font-size); + --_--text-line-height: var(--g-text-body-2-line-height); + } } &_clickable:hover { - cursor: pointer; background-color: var(--g-color-base-simple-hover); - &:after { + &::after { border-color: transparent; } } @@ -50,54 +118,82 @@ $block: '.#{variables.$ns}user-label'; &__main { @include mixins.button-reset(); - display: inline-flex; + display: flex; align-items: center; min-width: 0; - border-radius: inherit; - padding-inline-end: 6px; + border-radius: var(--_--border-radius); + padding-inline-end: var(--_--outer-gap); + cursor: unset; - #{$block}_closeable & { - padding-inline-end: 0; + #{$block}_empty & { + padding-inline-start: var(--_--outer-gap); } #{$block}_clickable & { outline-offset: -1px; + cursor: pointer; &:focus-visible { outline: 2px solid var(--g-color-line-focus); } } + + #{$block}_closeable & { + padding-inline-end: var(--_--inner-gap); + } } &__avatar { - --g-avatar-background-color: var(--g-color-base-generic-accent); + --g-avatar-size: var(--g-user-label-size, var(--_--size)); + --g-avatar-border-width: 1px; + --g-avatar-inner-border-width: 0; + --g-avatar-background-color: var(--g-color-base-neutral-light); --g-avatar-color: var(--g-color-text-primary); display: flex; - margin-inline-end: 6px; + margin-inline-end: var(--_--inner-gap); } - &__text { - font-size: var(--g-user-label-font-size, inherit); - line-height: var(--g-user-label-line-height, inherit); + &__info { + display: flex; + flex-direction: column; min-width: 0; - margin-inline-end: 6px; - white-space: nowrap; - text-overflow: ellipsis; - overflow: hidden; + } + + &__text { + @include mixins.overflow-ellipsis(); + + color: var(--g-color-text-primary); + font-weight: var(--g-user-label-text-font-weight, var(--_--text-font-weight)); + font-size: var(--g-user-label-text-font-size, var(--_--text-font-size)); + line-height: var(--g-user-label-text-line-height, var(--_--text-line-height)); + + #{$block}_closable & { + margin-inline-end: var(--_--inner-gap); + } + } + + &__description { + @include mixins.overflow-ellipsis(); + + color: var(--g-color-text-secondary); + font-weight: var(--g-user-label-description-font-weight, var(--_--description-font-weight)); + font-size: var(--g-user-label-description-font-size, var(--_--description-font-size)); + line-height: var(--g-user-label-description-line-height, var(--_--description-line-height)); } &__close { @include mixins.button-reset(); box-sizing: initial; - display: inline-flex; + display: flex; justify-content: center; align-items: center; width: 16px; - cursor: pointer; - padding-inline-end: 6px; + border-radius: var(--_--border-radius); + padding-inline-end: var(--_--outer-gap); color: var(--g-color-text-secondary); + cursor: pointer; transition-property: color; transition-duration: $transitionDuration; @@ -115,15 +211,4 @@ $block: '.#{variables.$ns}user-label'; outline: 2px solid var(--g-color-line-focus); } } - - &_size { - @each $size-name, $size-value in avatar-variables.$sizes { - &_#{$size-name} { - height: #{$size-value}; - } - &_xl::after { - border-radius: 150px; - } - } - } } diff --git a/src/components/UserLabel/UserLabel.tsx b/src/components/UserLabel/UserLabel.tsx index 69eb3eaeb2..40c21bcc3e 100644 --- a/src/components/UserLabel/UserLabel.tsx +++ b/src/components/UserLabel/UserLabel.tsx @@ -3,9 +3,11 @@ import React from 'react'; import {Envelope, Xmark} from '@gravity-ui/icons'; import {Avatar} from '../Avatar'; +import type {AvatarProps} from '../Avatar'; import {Icon} from '../Icon'; import {block} from '../utils/cn'; +import {COMPACT_SIZES, DEFAULT_USER_LABEL_SIZE} from './constants'; import i18n from './i18n'; import type {UserLabelProps} from './types'; @@ -14,43 +16,50 @@ import './UserLabel.scss'; const b = block('user-label'); export const UserLabel = React.forwardRef( + // eslint-disable-next-line complexity ( { type = 'person', - avatar, - children, view = 'outlined', + size = DEFAULT_USER_LABEL_SIZE, + avatar, + text, + description, onClick, onCloseClick, className, style, qa, - size = 's', }, ref, ) => { const clickable = Boolean(onClick); const closeable = Boolean(onCloseClick); + const MainComponent = clickable ? 'button' : 'div'; let avatarView: React.ReactNode = null; + let avatarProps: AvatarProps | undefined; - let avatarProps; if (typeof avatar === 'string') { - avatarProps = { - imgUrl: avatar, - }; + avatarProps = {imgUrl: avatar}; } else if (avatar && !React.isValidElement(avatar)) { - avatarProps = avatar; - } else if (!avatar && typeof children === 'string') { - avatarProps = { - text: children, - }; + if ( + ('imgUrl' in avatar && avatar.imgUrl) || + ('icon' in avatar && avatar.icon) || + ('text' in avatar && avatar.text) + ) { + avatarProps = avatar as AvatarProps; + } else if (typeof text === 'string') { + avatarProps = {text, borderColor: 'var(--g-color-line-generic)', ...avatar}; + } + } else if (!avatar && typeof text === 'string') { + avatarProps = {text, borderColor: 'var(--g-color-line-generic)'}; } switch (type) { case 'email': - avatarView = ; + avatarView = ; break; case 'empty': avatarView = null; @@ -65,15 +74,17 @@ export const UserLabel = React.forwardRef( break; } + const showDescription = Boolean(description && !COMPACT_SIZES.has(size)); + return (
( onClick={onClick} > {avatarView ?
{avatarView}
: null} -
{children}
+
+ {text} + {showDescription ? ( + {description} + ) : null} +
{onCloseClick ? (