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/scripts/playwright-docker.sh b/scripts/playwright-docker.sh index 8b59d87b21..58879a3fa2 100755 --- a/scripts/playwright-docker.sh +++ b/scripts/playwright-docker.sh @@ -12,12 +12,12 @@ command_exists() { } run_command() { - $CONTAINER_TOOL run --rm --network host -it -w /work \ - -v $(pwd):/work \ - -v "$NODE_MODULES_CACHE_DIR:/work/node_modules" \ - -e IS_DOCKER=1 \ - "$IMAGE_NAME:$IMAGE_TAG" \ - /bin/bash -c "$1" + $CONTAINER_TOOL run --rm --network host -it -w /work \ + -v $(pwd):/work \ + -v "$NODE_MODULES_CACHE_DIR:/work/node_modules" \ + -e IS_DOCKER=1 \ + "$IMAGE_NAME:$IMAGE_TAG" \ + /bin/bash -c "$1" } if command_exists docker; then @@ -30,14 +30,14 @@ else fi if [[ "$1" = "clear-cache" ]]; then - rm -rf "$NODE_MODULES_CACHE_DIR" - rm -rf "./playwright/.cache-docker" - exit 0 + rm -rf "$NODE_MODULES_CACHE_DIR" + rm -rf "./playwright/.cache-docker" + exit 0 fi if [[ ! -d "$NODE_MODULES_CACHE_DIR" ]]; then - mkdir -p "$NODE_MODULES_CACHE_DIR" - run_command 'npm ci' + mkdir -p "$NODE_MODULES_CACHE_DIR" + run_command 'npm ci' fi run_command "$1" diff --git a/src/components/Avatar/Avatar.scss b/src/components/Avatar/Avatar.scss index 0ea1866bf3..2bcc78162b 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; - --_--color: var(--g-color-text-misc); + --_--background-color: var(--g-color-base-misc-light); + --_--text-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; } } @@ -104,13 +102,13 @@ $block: '.#{variables.$ns}avatar'; &#{$block}_view { &_filled { --_--background-color: var(--g-color-base-misc-light); - --_--color: var(--g-color-text-misc); + --_--text-color: var(--g-color-text-misc); } &_outlined { --_--background-color: var(--g-color-base-background); --_--border-color: var(--g-color-text-misc); - --_--color: var(--g-color-text-misc); + --_--text-color: var(--g-color-text-misc); } } } @@ -119,15 +117,37 @@ $block: '.#{variables.$ns}avatar'; &#{$block}_view { &_filled { --_--background-color: var(--g-color-base-brand); - --_--color: var(--g-color-text-brand-contrast); + --_--text-color: var(--g-color-text-brand-contrast); } &_outlined { --_--background-color: var(--g-color-base-background); --_--border-color: var(--g-color-text-brand); - --_--color: var(--g-color-text-brand); + --_--text-color: var(--g-color-text-brand); } } } } + + &__image { + display: block; + width: 100%; + height: 100%; + object-fit: cover; + } + + &__icon { + color: var(--g-avatar-text-color, var(--_--text-color)); + + & > svg { + display: block; + } + } + + &__text { + color: var(--g-avatar-text-color, var(--_--text-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 7a26dcaa97..05f6166563 100644 --- a/src/components/Avatar/Avatar.tsx +++ b/src/components/Avatar/Avatar.tsx @@ -1,6 +1,7 @@ import * as 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 770f8190d5..44088d42ab 100644 --- a/src/components/Avatar/AvatarIcon/AvatarIcon.tsx +++ b/src/components/Avatar/AvatarIcon/AvatarIcon.tsx @@ -4,6 +4,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 b078f500e9..cc75c15d12 100644 --- a/src/components/Avatar/AvatarText/AvatarText.tsx +++ b/src/components/Avatar/AvatarText/AvatarText.tsx @@ -1,9 +1,9 @@ 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 fa7fc961e3..4bd48a8d47 100644 --- a/src/components/Avatar/README.md +++ b/src/components/Avatar/README.md @@ -140,12 +140,14 @@ LANDING_BLOCK--> ### Size -Use the `size` property to manage the `Avatar` size. The default size is `m`. The possible values are `xs`, `s`, `m`, `l`, and `xl`. +Use the `size` property to manage the `Avatar` size. The default size is `m`. The possible values are `3xs`, `2xs`, `xs`, `s`, `m`, `l`, and `xl`. ### Common -| Name | Description | Type | Default | -| :-------------- | :----------------------------------------- | :-----------------------------: | :------: | -| size | Avatar size | `'xs'` `'s'` `'m'` `'l'` `'xl'` | `m` | -| theme | Avatar theme | `'normal'` `'brand'` | `normal` | -| view | Avatar filling and outlining options | `'filled'` `'outlined'` | `filled` | -| backgroundColor | Custom background color | `string` | | -| borderColor | Custom border color | `string` | | -| title | `title` HTML attribute | `string` | | -| aria-label | `aria-label` for the avatar section | `string` | | -| aria-labelledby | `aria-labelledby` for the avatar section | `string` | | -| className | Custom CSS class for the root element | `string` | | -| style | `style` HTML attribute | `React.CSSProperties` | | -| qa | `data-qa` HTML attribute, used for testing | `string` | | +| Name | Description | Type | Default | +| :--------------- | :----------------------------------------- | :---------------------------------------------: | :------: | +| size | Avatar size | `'3xs'` `'2xs'` `'xs'` `'s'` `'m'` `'l'` `'xl'` | `m` | +| theme | Avatar theme | `'normal'` `'brand'` | `normal` | +| view | Avatar filling and outlining options | `'filled'` `'outlined'` | `filled` | +| backgroundColor | Custom background color | `string` | | +| borderColor | Custom border color | `string` | | +| title | `title` HTML attribute | `string` | | +| aria-label | `aria-label` for the avatar section | `string` | | +| aria-labelledby | `aria-labelledby` for the avatar section | `string` | | +| aria-describedby | `aria-describedby` for avatar block | `string` | | +| aria-details | `aria-details` for avatar block | `string` | | +| className | Custom CSS class for the root element | `string` | | +| style | `style` HTML attribute | `React.CSSProperties` | | +| qa | `data-qa` HTML attribute, used for testing | `string` | | ### Image-specific @@ -207,11 +213,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-text-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 6c921c86f5..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-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 431c60db6e..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-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 431c60db6e..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-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 2465ea9ab7..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-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 2dc3864f4b..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-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 8cc43ca1eb..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-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 423b5227f6..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-smoke-with-icon-light-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-smoke-with-icon-light-chromium-linux.png deleted file mode 100644 index 8b9e1fcd9e..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-smoke-with-icon-light-chromium-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-smoke-with-image-light-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-smoke-with-image-light-chromium-linux.png deleted file mode 100644 index 7a08d6926a..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-smoke-with-image-light-chromium-linux.png and /dev/null differ diff --git a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-smoke-with-text-light-chromium-linux.png b/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-smoke-with-text-light-chromium-linux.png deleted file mode 100644 index 4da4f7a938..0000000000 Binary files a/src/components/Avatar/__snapshots__/Avatar.visual.test.tsx-snapshots/Avatar-smoke-with-text-light-chromium-linux.png and /dev/null differ diff --git a/src/components/Avatar/__stories__/Avatar.stories.tsx b/src/components/Avatar/__stories__/Avatar.stories.tsx index 0c2ff961d0..d9bdcb7e40 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}); }); const defaultProps: AvatarProps = {}; 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/PersonaWrap/PersonaWrap.scss b/src/components/PersonaWrap/PersonaWrap.scss deleted file mode 100644 index b3bf24ec9e..0000000000 --- a/src/components/PersonaWrap/PersonaWrap.scss +++ /dev/null @@ -1,136 +0,0 @@ -@use '../../../styles/mixins'; -@use '../variables'; - -$block: '.#{variables.$ns}persona'; - -#{$block} { - $avatarSize: 28px; - $transitionDuration: 0.1s; - $transitionTimingFunction: ease-in-out; - - position: relative; - z-index: 0; - display: inline-flex; - height: $avatarSize; - border-radius: 20px; - - transition-property: background-color; - transition-duration: $transitionDuration; - transition-timing-function: $transitionTimingFunction; - - &_theme_default { - &:after { - position: absolute; - z-index: -1; - inset: 0; - content: ''; - border: 1px solid var(--g-color-line-generic); - border-radius: 20px; - - transition-property: border-color; - transition-duration: $transitionDuration; - transition-timing-function: $transitionTimingFunction; - } - } - - &_empty { - &#{$block}_size_s { - padding-inline-start: 12px; - } - - &#{$block}_size_n { - padding-inline-start: 16px; - } - } - - &_clickable:hover { - cursor: pointer; - background-color: var(--g-color-base-simple-hover); - - &:after { - border-color: transparent; - } - } - - &__main { - @include mixins.button-reset(); - - display: inline-flex; - align-items: center; - border-radius: inherit; - padding-inline-end: 6px; - - #{$block}_closeable & { - padding-inline-end: 0; - } - #{$block}_clickable & { - outline-offset: -1px; - - &:focus { - outline: 2px solid var(--g-color-line-focus); - } - - &:focus:not(:focus-visible) { - outline: 0; - } - } - } - - &__avatar { - --g-avatar-background-color: var(--g-color-base-generic-accent); - --g-avatar-color: var(--g-color-text-primary); - - #{$block}_size_n & { - margin-inline-end: 12px; - } - - #{$block}_size_s & { - margin-inline-end: 6px; - } - } - - &__text { - #{$block}_size_n & { - @include mixins.text-body-2; - margin-inline-end: 10px; - } - - #{$block}_size_s & { - @include mixins.text-body-1; - margin-inline-end: 6px; - } - } - - &__close { - @include mixins.button-reset(); - - box-sizing: initial; - display: inline-flex; - justify-content: center; - align-items: center; - width: 16px; - cursor: pointer; - padding-inline-end: 6px; - color: var(--g-color-text-secondary); - - transition-property: color; - transition-duration: $transitionDuration; - transition-timing-function: $transitionTimingFunction; - - &:hover { - color: var(--g-color-text-primary); - } - } - - &__close-icon { - border-radius: var(--g-focus-border-radius); - - #{$block}__close:focus & { - outline: 2px solid var(--g-color-line-focus); - } - - #{$block}__close:focus:not(:focus-visible) & { - outline: 0; - } - } -} diff --git a/src/components/PersonaWrap/PersonaWrap.tsx b/src/components/PersonaWrap/PersonaWrap.tsx deleted file mode 100644 index b6feb7f7b8..0000000000 --- a/src/components/PersonaWrap/PersonaWrap.tsx +++ /dev/null @@ -1,62 +0,0 @@ -import type * as React from 'react'; - -import {Xmark} from '@gravity-ui/icons'; - -import {Icon} from '../Icon'; -import type {QAProps} from '../types'; -import {block} from '../utils/cn'; - -import './PersonaWrap.scss'; - -const b = block('persona'); - -export interface PersonaWrapProps extends QAProps { - avatar: React.ReactNode; - children?: React.ReactNode; - isEmpty?: boolean; - theme?: 'default' | 'clear'; - size?: 's' | 'n'; - onClose?: (event: React.MouseEvent) => void; - onClick?: (event: React.MouseEvent) => void; - className?: string; - style?: React.CSSProperties; - closeButtonAriaAttributes?: React.AriaAttributes; -} - -export function PersonaWrap({ - size = 's', - theme = 'default', - isEmpty, - onClick, - onClose, - className, - avatar, - children, - style, - closeButtonAriaAttributes, - qa, -}: PersonaWrapProps) { - const clickable = Boolean(onClick); - const closeable = Boolean(onClose); - const MainComponent = clickable ? 'button' : 'div'; - - return ( -
- - {avatar &&
{avatar}
} -
{children}
-
- {onClose && ( - - )} -
- ); -} - -PersonaWrap.displayName = 'PersonaWrap'; diff --git a/src/components/PersonaWrap/index.ts b/src/components/PersonaWrap/index.ts deleted file mode 100644 index 5926780a89..0000000000 --- a/src/components/PersonaWrap/index.ts +++ /dev/null @@ -1 +0,0 @@ -export * from './PersonaWrap'; diff --git a/src/components/User/README.md b/src/components/User/README.md index 3eaf5ffbfe..b571b19977 100644 --- a/src/components/User/README.md +++ b/src/components/User/README.md @@ -28,7 +28,7 @@ LANDING_BLOCK--> ## Size -Use the `size` property to manage the `User` size. The default size is `m`. The possible values are `xs`, `s`, `m`, `l`, and `xl`. +Use the `size` property to manage the `User` size. The default size is `m`. The possible values are `3xs`, `2xs`, `xs`, `s`, `m`, `l`, and `xl`. This property is also provided to the internal `Avatar` component. @@ -36,6 +36,8 @@ This property is also provided to the internal `Avatar` component. + @@ -43,6 +45,8 @@ This property is also provided to the internal `Avatar` component. `} > + + @@ -54,14 +58,28 @@ LANDING_BLOCK--> ## Properties -| Name | Description | Type | Default | -| :-------------- | :----------------------------------------- | :--------------------------------------------------------------------: | :-----: | -| avatar | User avatar | `React.ReactElement` [avatar property](../Avatar/README.md#properties) | | -| name | User name | `React.ReactNode` | | -| description | User description | `React.ReactNode` | | -| size | User section size | `'xs'` `'s'` `'m'` `'l'` `'xl'` | `m` | -| aria-label | `aria-label` for the user section | `string` | | -| aria-labelledby | `aria-labelledby` for the user section | `string` | | -| className | Custom CSS class for the root element | `string` | | -| style | HTML style attribute | `React.CSSProperties` | | -| qa | `data-qa` HTML attribute, used for testing | `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 section size | `'3xs'` `'2xs'` `'xs'` `'s'` `'m'` `'l'` `'xl'` | `m` | +| aria-label | `aria-label` for the user section | `string` | | +| aria-labelledby | `aria-labelledby` for the user section | `string` | | +| aria-describedby | `aria-describedby` for the user section | `string` | | +| aria-details | `aria-details` for the user section | `string` | | +| className | Custom CSS class for the root element | `string` | | +| style | HTML style attribute | `React.CSSProperties` | | +| qa | `data-qa` HTML attribute, used for testing | `string` | | + +## CSS API + +| Name | Description | +| :--------------------------------- | :-------------------------------- | +| `--g-user-avatar-offset` | 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..26f162d2bf 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} { + --_--avatar-offset: var(--g-spacing-2); + --_--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-avatar-offset, var(--_--avatar-offset)); + + &_size { + &_3xs, + &_2xs, + &_xs, + &_s { + --_--avatar-offset: calc(var(--g-spacing-base) * 1.5); + } + + &_m { + --_--avatar-offset: var(--g-spacing-2); + } + + &_l, + &_xl { + --_--avatar-offset: var(--g-spacing-3); + } + + &_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 81eb4dd98a..d551625a90 100644 --- a/src/components/User/User.tsx +++ b/src/components/User/User.tsx @@ -2,73 +2,62 @@ import * as React from 'react'; import {Avatar} from '../Avatar'; import {block} from '../utils/cn'; +import {filterDOMProps} from '../utils/filterDOMProps'; -import {COMPACT_SIZES, DEFAULT_SIZE, UserQa} from './constants'; +import {COMPACT_SIZES, DEFAULT_USER_SIZE, UserQa} from './constants'; import type {UserProps} from './types'; import './User.scss'; const b = block('user'); -export const User = React.forwardRef( - ( - { - avatar, - name, - description, - size = DEFAULT_SIZE, - 'aria-label': ariaLabel, - 'aria-labelledby': ariaLabelledby, - className, - style, - qa, - }, - ref, - ) => { - const showDescription = Boolean(description && !COMPACT_SIZES.has(size)); - - const nameTitle = typeof name === 'string' ? name : undefined; - const descriptionTitle = typeof description === 'string' ? description : undefined; - - return ( -
- {avatar ? ( -
- {React.isValidElement(avatar) ? ( - avatar - ) : ( - - )} -
- ) : null} - {name || showDescription ? ( -
- {name ? ( - - {name} - - ) : null} - {showDescription ? ( - - {description} - - ) : null} -
- ) : null} -
- ); - }, -); +export const User = React.forwardRef((props, ref) => { + const {avatar, name, description, size = DEFAULT_USER_SIZE, className, style, qa} = props; + + const nameTitle = typeof name === 'string' ? name : undefined; + const descriptionTitle = typeof description === 'string' ? description : undefined; + + 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.has(size)); + + return ( +
+ {avatarView ?
{avatarView}
: null} + {name || showDescription ? ( +
+ {name ? ( + + {name} + + ) : null} + {showDescription ? ( + + {description} + + ) : null} +
+ ) : null} +
+ ); +}); User.displayName = 'User'; 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 e26fcd92c0..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-smoke-light-chromium-linux.png b/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-smoke-light-chromium-linux.png deleted file mode 100644 index 8ae1f54f5b..0000000000 Binary files a/src/components/User/__snapshots__/User.visual.test.tsx-snapshots/User-smoke-light-chromium-linux.png and /dev/null 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 7f80749026..f4b2ef2ad1 100644 --- a/src/components/User/__tests__/User.visual.test.tsx +++ b/src/components/User/__tests__/User.visual.test.tsx @@ -1,5 +1,3 @@ -import {expect} from '@playwright/experimental-ct-react'; - import {smokeTest, test} from '~playwright/core'; import {createSmokeScenarios} from '../../../stories/tests-factory/create-smoke-scenarios'; @@ -7,13 +5,17 @@ import type {UserProps} from '../types'; import {avatarCases, descriptionCases, nameCases, sizeCases} from './cases'; import {TestUser} from './helpers'; -import {Default} from './stories'; +import {UserStories} from './stories'; test.describe('User', {tag: '@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}); }); smokeTest('', async ({mount, expectScreenshot}) => { 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 becc6aa2ca..6904f35dfb 100644 --- a/src/components/User/constants.ts +++ b/src/components/User/constants.ts @@ -1,7 +1,8 @@ import type {UserSize} from './types'; -export const COMPACT_SIZES: Set = new Set(['xs', '2xs']); -export const DEFAULT_SIZE: UserSize = 'm'; +export const DEFAULT_USER_SIZE: UserSize = 'm'; + +export const COMPACT_SIZES: Set = new Set(['xs', '2xs', '3xs']); export const UserQa = { NAME: 'user-name', diff --git a/src/components/User/index.ts b/src/components/User/index.ts index c4efa01a1b..7a9b5e38fb 100644 --- a/src/components/User/index.ts +++ b/src/components/User/index.ts @@ -1,3 +1,3 @@ export type {UserSize, UserProps} from './types'; -export {UserQa} from './constants'; +export {DEFAULT_USER_SIZE, UserQa} from './constants'; export {User} from './User'; diff --git a/src/components/User/types.ts b/src/components/User/types.ts index c9991da89d..8dd61aafda 100644 --- a/src/components/User/types.ts +++ b/src/components/User/types.ts @@ -2,15 +2,16 @@ import type * as React from 'react'; import type {DistributiveOmit} from '../../types/utils'; import type {AvatarProps} from '../Avatar'; -import type {DOMProps, QAProps} from '../types'; +import type {AriaLabelingProps, 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; +export interface UserProps extends AriaLabelingProps, DOMProps, QAProps { + avatar?: + | DistributiveOmit + | string + | React.ReactElement; name?: React.ReactNode; description?: React.ReactNode; size?: UserSize; - 'aria-label'?: string; - 'aria-labelledby'?: string; } diff --git a/src/components/UserLabel/README.md b/src/components/UserLabel/README.md index f5623c39ae..70cc2584e1 100644 --- a/src/components/UserLabel/README.md +++ b/src/components/UserLabel/README.md @@ -2,35 +2,35 @@ The `UserLabel` component can be used to display users or user-related information. -### Type +## Type Used to manage avatar appearance. Use `"person"` for a personalized entity and `"email"`, for an email address. If you do not need any avatar, use `"empty"`. ```tsx -Charles Darwin (person) -email@example.com (email) -Alan Turing (other) + + + ``` -### Avatar +## Avatar This component can be used with a custom avatar. It only works with `type: 'person'`. You can provide an image, a property of the [Avatar](../Avatar/README.md) component, or a custom React node. @@ -39,12 +39,12 @@ This component can be used with a custom avatar. It only works with `type: 'pers code={` import {GraduationCap} from '@gravity-ui/icons'; -Charles Darwin -Charles Darwin + + `} > - Charles Darwin - '}}>Charles Darwin + + '}} text="Charles Darwin" />
LANDING_BLOCK--> @@ -53,55 +53,64 @@ LANDING_BLOCK--> ```tsx import {GraduationCap} from '@gravity-ui/icons'; -Charles Darwin -Charles Darwin + + ``` -### Interactivity +## Interactivity This component is also interactive: it can be clickable or closable. ```tsx - alert('onClick triggered')}>Charles Darwin - alert('onCloseClick triggered')}>Charles Darwin + alert('onClick triggered')} /> + alert('onCloseClick triggered')} /> ``` ## Properties -| Name | Description | Type | Default | -| :----------- | :---------------------------------------------- | :-----------------------------------------------------------------------------------: | :----------: | -| type | Avatar appearance | `'person'` `'email'` `'empty'` | `'person'` | -| avatar | User avatar | `string` and `React.ReactElement` [avatar properties](../Avatar/README.md#properties) | | -| children | Visible text | `React.ReactNode` | | -| view | `UserLabel` view | `'outlined'` `'clear'` | `'outlined'` | -| onClick | `click` event handler for the component | `Function` | | -| onCloseClick | `click` event handler for the cross-icon button | `Function` | | -| className | Custom CSS class for the root element | `string` | | -| style | HTML style attribute | `React.CSSProperties` | | -| qa | `data-qa` HTML attribute, used for testing | `string` | | -| size | Avatar size | `'xs'` `'s'` `'m'` `'l'` `'xl'` | `'s'` | +| Name | Description | Type | Default | +| :----------- | :---------------------------------------------- | :-------------------------------------------------------------------------: | :----------: | +| type | Avatar appearance | `'person'` `'email'` `'empty'` | `'person'` | +| 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` | | +| text | Visible text | `React.ReactNode` | | +| description | User description | `React.ReactNode` | | +| onClick | `click` event handler for the component | `Function` | | +| onCloseClick | `click` event handler for the cross-icon button | `Function` | | +| className | Custom CSS class for the root element | `string` | | +| style | HTML style attribute | `React.CSSProperties` | | +| qa | `data-qa` HTML attribute, used for testing | `string` | | ## CSS API -| Name | Description | -| :--------------------------- | :--------------- | -| `--g-user-label-font-size` | Text font size | -| `--g-user-label-line-height` | Text line height | +| Name | Description | +| :--------------------------------------- | :------------------------------------------------------ | +| `--g-user-label-size` | Size for avatar (width and height) and height for label | +| `--g-user-label-border-radius` | Label border radius | +| `--g-user-label-padding` | Label horizontal padding | +| `--g-user-label-gap` | Gap between elements (avatar, text, close icon) | +| `--g-user-label-text-font-weight` | Text font weight | +| `--g-user-label-text-font-size` | Text font size | +| `--g-user-label-text-line-height` | Text line height | +| `--g-user-label-description-font-weight` | Description font weight | +| `--g-user-label-description-font-size` | Description font size | +| `--g-user-label-description-line-height` | Description line height | diff --git a/src/components/UserLabel/UserLabel.scss b/src/components/UserLabel/UserLabel.scss index f01d582877..5d2509c129 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; + --_--padding: var(--g-spacing-3); + --_--gap: calc(var(--g-spacing-base) * 1.5); + --_--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(--g-user-label-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: 1px solid var(--g-color-line-generic-solid); + border-radius: inherit; transition-property: border-color; transition-duration: $transitionDuration; @@ -34,70 +46,149 @@ $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, + &_2xs { + --_--padding: calc(var(--g-spacing-base) * 1.5); + --_--gap: var(--g-spacing-1); + } + + &_xs { + --_--padding: calc(var(--g-spacing-base) * 2.5); + --_--gap: calc(var(--g-spacing-base) * 1.5); + } + + &_s { + --_--padding: var(--g-spacing-3); + --_--gap: calc(var(--g-spacing-base) * 1.5); + } + + &_m { + --_--padding: var(--g-spacing-3); + --_--gap: var(--g-spacing-2); + } + + &_l { + --_--padding: var(--g-spacing-4); + --_--gap: var(--g-spacing-3); + } + + &_xl { + --_--padding: var(--g-spacing-5); + --_--gap: var(--g-spacing-4); + } + + &_3xs, + &_2xs, + &_xs { + --_--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); + } + + &_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; } } + &__avatar { + --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-text-color: var(--g-color-text-primary); + + display: flex; + margin-inline-end: var(--g-user-label-gap, var(--_--gap)); + } + &__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(--g-user-label-border-radius, var(--_--border-radius)); + padding-inline-end: var(--g-user-label-padding, var(--_--padding)); + cursor: unset; - #{$block}_closeable & { - padding-inline-end: 0; + #{$block}_empty & { + padding-inline-start: var(--g-user-label-padding, var(--_--padding)); } #{$block}_clickable & { outline-offset: -1px; + cursor: pointer; &:focus-visible { outline: 2px solid var(--g-color-line-focus); } } - } - &__avatar { - --g-avatar-background-color: var(--g-color-base-generic-accent); - --g-avatar-color: var(--g-color-text-primary); + #{$block}_closeable & { + padding-inline-end: var(--g-user-label-gap, var(--_--gap)); + } + } + &__info { display: flex; - margin-inline-end: 6px; + flex-direction: column; + min-width: 0; } &__text { - font-size: var(--g-user-label-font-size, inherit); - line-height: var(--g-user-label-line-height, inherit); - min-width: 0; - margin-inline-end: 6px; - white-space: nowrap; - text-overflow: ellipsis; - overflow: hidden; + @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)); + } + + &__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(--g-user-label-border-radius, var(--_--border-radius)); + padding-inline-end: var(--g-user-label-padding, var(--_--padding)); color: var(--g-color-text-secondary); + cursor: pointer; transition-property: color; transition-duration: $transitionDuration; @@ -115,15 +206,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 d65b2591c7..70981a1f09 100644 --- a/src/components/UserLabel/UserLabel.tsx +++ b/src/components/UserLabel/UserLabel.tsx @@ -3,9 +3,11 @@ import * as 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 {BORDER_COLOR, COMPACT_SIZES, DEFAULT_USER_LABEL_SIZE, ICON_SIZES} 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: BORDER_COLOR, ...avatar}; + } + } else if (!avatar && typeof text === 'string') { + avatarProps = {text, borderColor: BORDER_COLOR}; } 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 ? ( ) : null}
diff --git a/src/components/UserLabel/__snapshots__/UserLabel.visual.test.tsx-snapshots/UserLabel-smoke-email-light-chromium-linux.png b/src/components/UserLabel/__snapshots__/UserLabel.visual.test.tsx-snapshots/UserLabel-smoke-email-light-chromium-linux.png deleted file mode 100644 index 1bc2507187..0000000000 Binary files a/src/components/UserLabel/__snapshots__/UserLabel.visual.test.tsx-snapshots/UserLabel-smoke-email-light-chromium-linux.png and /dev/null differ diff --git a/src/components/UserLabel/__snapshots__/UserLabel.visual.test.tsx-snapshots/UserLabel-smoke-person-light-chromium-linux.png b/src/components/UserLabel/__snapshots__/UserLabel.visual.test.tsx-snapshots/UserLabel-smoke-person-light-chromium-linux.png deleted file mode 100644 index a217fbeb1b..0000000000 Binary files a/src/components/UserLabel/__snapshots__/UserLabel.visual.test.tsx-snapshots/UserLabel-smoke-person-light-chromium-linux.png and /dev/null differ diff --git a/src/components/UserLabel/__snapshots__/UserLabel.visual.test.tsx-snapshots/UserLabel-smoke-user-light-chromium-linux.png b/src/components/UserLabel/__snapshots__/UserLabel.visual.test.tsx-snapshots/UserLabel-smoke-user-light-chromium-linux.png deleted file mode 100644 index 10def87e60..0000000000 Binary files a/src/components/UserLabel/__snapshots__/UserLabel.visual.test.tsx-snapshots/UserLabel-smoke-user-light-chromium-linux.png and /dev/null differ diff --git a/src/components/UserLabel/__stories__/UserLabel.stories.tsx b/src/components/UserLabel/__stories__/UserLabel.stories.tsx index 1045b787fa..ddb9851220 100644 --- a/src/components/UserLabel/__stories__/UserLabel.stories.tsx +++ b/src/components/UserLabel/__stories__/UserLabel.stories.tsx @@ -1,11 +1,28 @@ -import {faker} from '@faker-js/faker/locale/en'; +import React from 'react'; + import type {Meta, StoryObj} from '@storybook/react'; +import {Showcase} from '../../../demo/Showcase'; +import {ShowcaseItem} from '../../../demo/ShowcaseItem'; import {UserLabel} from '../UserLabel'; const meta: Meta = { title: 'Components/Data Display/UserLabel', component: UserLabel, + parameters: { + a11y: { + element: '#storybook-root', + config: { + rules: [ + { + id: 'color-contrast', + enabled: false, + selector: '.g-user-label__description', + }, + ], + }, + }, + }, }; export default meta; @@ -13,73 +30,219 @@ export default meta; type Story = StoryObj; const person = 'Charles Darwin'; -const [firstName, lastName] = person.split(' '); -const email = faker.internet.email({firstName, lastName}); -const personImg = faker.image.avatar(); +const email = 'charles.darwin@gmail.com'; +const personImg = + ''; +const avatarAriaProps = { + 'aria-label': `${person}'s avatar`, +}; -export const Default: Story = { - args: { - children: person, - avatar: { - 'aria-label': "Charles Darwin's avatar", - }, +const personProps = { + avatar: avatarAriaProps, + text: person, + description: email, +}; + +const imageProps = { + avatar: { + ...avatarAriaProps, + imgUrl: personImg, + alt: 'Fake person', }, + text: person, + description: email, }; -export const Image: Story = { - args: { - avatar: { - imgUrl: personImg, - 'aria-label': "Charles Darwin's avatar", - alt: 'Fake person', - }, - children: person, +const emailProps = { + type: 'email' as const, + avatar: { + ...avatarAriaProps, + alt: 'Sample envelope icon', }, + text: email, +}; + +const emptyProps = { + type: 'empty' as const, + text: person, + description: email, +}; + +const closeableProps = { + ...personProps, + onCloseClick: () => console.log('closed'), +}; + +export const Default: Story = { + args: personProps, +}; + +export const Image: Story = { + args: imageProps, }; export const Email: Story = { - args: { - type: 'email', - children: email, - avatar: { - 'aria-label': "Charles Darwin's avatar", - alt: 'Sample envelope icon', - }, - }, + args: emailProps, }; export const Empty: Story = { - args: { - type: 'empty', - children: person, - }, + args: emptyProps, }; -export const LongChildren: Story = { +export const LongText: Story = { args: { - children: person.repeat(100), - avatar: { - 'aria-label': "Charles Darwin's avatar", - }, + avatar: avatarAriaProps, + text: person.repeat(100), }, }; export const Clickable: Story = { args: { - children: person, - onClick: (value) => console.log('clicked', value), - avatar: { - 'aria-label': "Charles Darwin's avatar", - }, + avatar: avatarAriaProps, + text: person, + onClick: () => console.log('clicked'), }, }; export const Closable: Story = { - args: { - children: person, - onCloseClick: (value) => console.log('closed', value), - avatar: { - 'aria-label': "Charles Darwin's avatar", - }, + args: closeableProps, +}; + +export const UserLabelShowcase: Story = { + name: 'Showcase', + render: () => { + return ( + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + ); }, }; diff --git a/src/components/UserLabel/__tests__/UserLabel.test.tsx b/src/components/UserLabel/__tests__/UserLabel.test.tsx index acb81955f1..ebf7cdef1e 100644 --- a/src/components/UserLabel/__tests__/UserLabel.test.tsx +++ b/src/components/UserLabel/__tests__/UserLabel.test.tsx @@ -3,6 +3,7 @@ import userEvent from '@testing-library/user-event'; import {queryByAttribute, render, screen} from '../../../../test-utils/utils'; import {getAvatarDisplayText} from '../../Avatar'; import {UserLabel} from '../UserLabel'; +import {DEFAULT_USER_LABEL_SIZE} from '../constants'; import i18n from '../i18n'; const MOCKED_TEXT = 'text'; @@ -14,9 +15,9 @@ describe('UserLabel', () => { 'should return text value as onClick argument', async (text) => { const onClick = jest.fn(); - render({text}); + render(); const user = userEvent.setup(); - const displayText = getAvatarDisplayText(text); + const displayText = getAvatarDisplayText(text, DEFAULT_USER_LABEL_SIZE); const personaNode = screen.getByText(displayText); await user.click(personaNode); expect(onClick).toHaveBeenCalled(); @@ -26,9 +27,7 @@ describe('UserLabel', () => { 'should return text value as onClose argument', async (text) => { const onCloseClick = jest.fn(); - const {container} = render( - {text}, - ); + const {container} = render(); const user = userEvent.setup(); const ariaLabelValue = i18n('label_remove-button'); const closeButtonNode = queryByAttribute('aria-label', container, ariaLabelValue); @@ -42,11 +41,11 @@ describe('UserLabel', () => { }, ); test('should render text as string', () => { - render({MOCKED_TEXT}); + render(); screen.getByText(MOCKED_TEXT); }); test('should render text as react node', () => { - render({MOCKED_TEXT_NODE}); + render(); screen.getByText(MOCKED_TEXT); }); }); diff --git a/src/components/UserLabel/__tests__/UserLabel.visual.test.tsx b/src/components/UserLabel/__tests__/UserLabel.visual.test.tsx index 5e5073cb28..a8fcebd234 100644 --- a/src/components/UserLabel/__tests__/UserLabel.visual.test.tsx +++ b/src/components/UserLabel/__tests__/UserLabel.visual.test.tsx @@ -3,17 +3,54 @@ import {smokeTest, test} from '~playwright/core'; import {createSmokeScenarios} from '../../../stories/tests-factory/create-smoke-scenarios'; import type {UserLabelProps} from '../types'; -import { - childrenCases, - onClickCases, - onCloseClickCases, - sizeCases, - typeCases, - viewCases, -} from './cases'; +import {onClickCases, onCloseClickCases, sizeCases, textCases, typeCases, viewCases} from './cases'; import {TestUserLabel, TestUserLabelWithEmail, TestUserLabelWithPerson} from './helpers'; +import {UserLabelStories} from './stories'; test.describe('UserLabel', {tag: '@UserLabel'}, () => { + test('render story: ', async ({mount, expectScreenshot}) => { + const component = await mount(); + await expectScreenshot({component}); + }); + + test('render story: ', async ({mount, expectScreenshot}) => { + const component = await mount(); + await expectScreenshot({component}); + }); + + test('render story: ', async ({mount, expectScreenshot}) => { + const component = await mount(); + await expectScreenshot({component}); + }); + + test('render story: ', async ({mount, expectScreenshot}) => { + const component = await mount(); + await expectScreenshot({component}); + }); + + test('render story: ', async ({mount, expectScreenshot}) => { + const component = await mount(, {width: 300}); + await expectScreenshot({component}); + }); + + test('render story: ', async ({mount, expectScreenshot}) => { + const component = await mount(); + await expectScreenshot({component, nameSuffix: 'default'}); + + await component.getByText('Charles Darwin').hover(); + await expectScreenshot({component, nameSuffix: 'hovered'}); + }); + + test('render story: ', async ({mount, expectScreenshot}) => { + const component = await mount(); + await expectScreenshot({component}); + }); + + test('render story: ', async ({mount, expectScreenshot}) => { + const component = await mount(); + await expectScreenshot({component}); + }); + smokeTest('user', async ({mount, expectScreenshot}) => { const smokeScenarios = createSmokeScenarios>( {}, @@ -21,7 +58,7 @@ test.describe('UserLabel', {tag: '@UserLabel'}, () => { size: sizeCases, view: viewCases, type: typeCases, - children: childrenCases, + text: textCases, onClick: onClickCases, onCloseClick: onCloseClickCases, }, @@ -51,7 +88,7 @@ test.describe('UserLabel', {tag: '@UserLabel'}, () => { { size: sizeCases, view: viewCases, - children: childrenCases, + text: textCases, onClick: onClickCases, onCloseClick: onCloseClickCases, }, @@ -81,7 +118,7 @@ test.describe('UserLabel', {tag: '@UserLabel'}, () => { { size: sizeCases, view: viewCases, - children: childrenCases, + text: textCases, onClick: onClickCases, onCloseClick: onCloseClickCases, }, diff --git a/src/components/UserLabel/__tests__/cases.tsx b/src/components/UserLabel/__tests__/cases.tsx index 17873dccb1..360554c0e4 100644 --- a/src/components/UserLabel/__tests__/cases.tsx +++ b/src/components/UserLabel/__tests__/cases.tsx @@ -7,7 +7,7 @@ export const viewCases: Cases = ['outlined', 'clear']; export const typeCases: Cases = ['person', 'email', 'empty']; -export const childrenCases: CasesWithName = [ +export const textCases: CasesWithName = [ [ 'long label', 'Charles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles DarwinCharles Darwin', diff --git a/src/components/UserLabel/__tests__/stories.ts b/src/components/UserLabel/__tests__/stories.ts new file mode 100644 index 0000000000..1a9c09fe9b --- /dev/null +++ b/src/components/UserLabel/__tests__/stories.ts @@ -0,0 +1,5 @@ +import {composeStories} from '@storybook/react'; + +import * as Stories from '../__stories__/UserLabel.stories'; + +export const UserLabelStories = composeStories(Stories); diff --git a/src/components/UserLabel/constants.ts b/src/components/UserLabel/constants.ts new file mode 100644 index 0000000000..b210344602 --- /dev/null +++ b/src/components/UserLabel/constants.ts @@ -0,0 +1,17 @@ +import type {UserLabelSize} from './types'; + +export const DEFAULT_USER_LABEL_SIZE: UserLabelSize = 's'; + +export const COMPACT_SIZES: Set = new Set(['m', 's', 'xs', '2xs', '3xs']); + +export const BORDER_COLOR = 'var(--g-color-line-generic-solid)'; + +export const ICON_SIZES: Record = { + '3xs': 12, + '2xs': 12, + xs: 12, + s: 16, + m: 16, + l: 16, + xl: 16, +}; diff --git a/src/components/UserLabel/index.ts b/src/components/UserLabel/index.ts index fcb9ee276f..71f86451ad 100644 --- a/src/components/UserLabel/index.ts +++ b/src/components/UserLabel/index.ts @@ -1,2 +1,3 @@ +export type {UserLabelType, UserLabelView, UserLabelSize, UserLabelProps} from './types'; +export {DEFAULT_USER_LABEL_SIZE} from './constants'; export {UserLabel} from './UserLabel'; -export type {UserLabelType, UserLabelView, UserLabelProps} from './types'; diff --git a/src/components/UserLabel/types.ts b/src/components/UserLabel/types.ts index 6c1a713eda..e059c82af2 100644 --- a/src/components/UserLabel/types.ts +++ b/src/components/UserLabel/types.ts @@ -1,21 +1,23 @@ import type * as React from 'react'; import type {DistributiveOmit} from '../../types/utils'; -import type {AvatarProps, AvatarSize} from '../Avatar'; +import type {AvatarProps} from '../Avatar'; import type {DOMProps, QAProps} from '../types'; export type UserLabelType = 'person' | 'email' | 'empty'; export type UserLabelView = 'outlined' | 'clear'; +export type UserLabelSize = '3xs' | '2xs' | 'xs' | 's' | 'm' | 'l' | 'xl'; export interface UserLabelProps extends DOMProps, QAProps { type?: UserLabelType; + view?: UserLabelView; + size?: UserLabelSize; avatar?: - | DistributiveOmit + | Partial> | string | React.ReactElement; - children: React.ReactNode; - view?: UserLabelView; + text: React.ReactNode; + description?: React.ReactNode; onClick?: React.MouseEventHandler; onCloseClick?: React.MouseEventHandler; - size?: AvatarSize; }