diff --git a/CODEOWNERS b/CODEOWNERS index 3a004158c2..fa0631e2e0 100644 --- a/CODEOWNERS +++ b/CODEOWNERS @@ -3,6 +3,7 @@ /src/components/Alert @IsaevAlexandr /src/components/ArrowToggle @Marginy605 /src/components/Avatar @DakEnviy +/src/components/AvatarStack @ogonkov #/src/components/Breadcrumbs /src/components/Button @amje /src/components/Card @Lunory diff --git a/package.json b/package.json index 491931f709..096cfcc0a7 100644 --- a/package.json +++ b/package.json @@ -108,7 +108,7 @@ "lint": "run-p lint:*", "typecheck": "tsc --noEmit", "prepublishOnly": "npm run build && npm pkg delete engines", - "playwright:install": "playwright install --with-deps", + "playwright:install": "playwright install chromium webkit --with-deps", "playwright": "playwright test --config=playwright/playwright.config.ts", "playwright:update": "npm run playwright -- -u", "playwright:docker": "./scripts/playwright-docker.sh 'npm run playwright'", diff --git a/src/components/AvatarStack/AvatarStack.scss b/src/components/AvatarStack/AvatarStack.scss new file mode 100644 index 0000000000..b0663c6a2e --- /dev/null +++ b/src/components/AvatarStack/AvatarStack.scss @@ -0,0 +1,61 @@ +@use '../../../styles/mixins'; +@use '../Avatar/variables' as avatar-variables; +@use '../variables'; + +$block: '.#{variables.$ns}avatar-stack'; + +#{$block} { + --_--more-button-size: #{avatar-variables.$default-size}; + --_--more-button-border-width: 1px; + + display: inline-flex; + justify-content: flex-end; + flex-direction: row-reverse; + + margin: 0; + padding: 0; + + &_overlap-size_s { + --_--overlap: var(--g-spacing-1); + } + + &_overlap-size_m { + --_--overlap: var(--g-spacing-2); + } + + &_overlap-size_l { + --_--overlap: var(--g-spacing-3); + } + + &__item { + display: flex; + z-index: 0; + border-radius: 100%; + + &:not(:first-child) { + margin-inline-end: calc(-1 * var(--_--overlap)); + } + } + + &__more-button { + @include mixins.button-reset; + + border-radius: 100%; + + width: var(--_--more-button-size); + height: var(--_--more-button-size); + + &:focus-visible { + outline: var(--g-color-line-focus) solid 2px; + outline-offset: 0; + } + + &_size { + @each $size-name, $size-value in avatar-variables.$sizes { + &_#{$size-name} { + --_--more-button-size: #{$size-value}; + } + } + } + } +} diff --git a/src/components/AvatarStack/AvatarStack.tsx b/src/components/AvatarStack/AvatarStack.tsx new file mode 100644 index 0000000000..3e03fb63c5 --- /dev/null +++ b/src/components/AvatarStack/AvatarStack.tsx @@ -0,0 +1,69 @@ +import React from 'react'; + +import {Avatar} from '../Avatar'; +import {block} from '../utils/cn'; + +import {AvatarStackItem} from './AvatarStackItem'; +import {AvatarStackMoreButton} from './AvatarStackMoreButton'; +import i18n from './i18n'; +import type {AvatarStackProps} from './types'; + +import './AvatarStack.scss'; + +const b = block('avatar-stack'); + +const AvatarStackComponent = ({ + max = 3, + overlapSize = 's', + size, + children, + className, + renderMore, +}: AvatarStackProps) => { + const visibleItems: React.ReactElement[] = []; + let moreItems = 0; + + React.Children.forEach(children, (child) => { + if (!React.isValidElement(child)) { + return; + } + + const item = {child}; + + if (visibleItems.length <= max) { + visibleItems.unshift(item); + } else { + moreItems += 1; + } + }); + + const hasMoreButton = moreItems > 0; + /** Avatars + more button, or just avatars, when avatars count is equal to `max` or less */ + const normalOverflow = moreItems >= 1; + + return ( + // Safari remove role=list with some styles, applied to li items, so we need + // to restore role manually + // eslint-disable-next-line jsx-a11y/no-redundant-roles + + ); +}; + +AvatarStackComponent.displayName = 'AvatarStack'; + +export const AvatarStack = Object.assign(AvatarStackComponent, {MoreButton: AvatarStackMoreButton}); diff --git a/src/components/AvatarStack/AvatarStackItem.tsx b/src/components/AvatarStack/AvatarStackItem.tsx new file mode 100644 index 0000000000..b21aaa415f --- /dev/null +++ b/src/components/AvatarStack/AvatarStackItem.tsx @@ -0,0 +1,13 @@ +import React from 'react'; + +import {block} from '../utils/cn'; + +const b = block('avatar-stack'); + +type Props = React.PropsWithChildren<{}>; + +export const AvatarStackItem = ({children}: Props) => { + return
  • {children}
  • ; +}; + +AvatarStackItem.displayName = 'AvatarStack.Item'; diff --git a/src/components/AvatarStack/AvatarStackMoreButton.tsx b/src/components/AvatarStack/AvatarStackMoreButton.tsx new file mode 100644 index 0000000000..c49d410a98 --- /dev/null +++ b/src/components/AvatarStack/AvatarStackMoreButton.tsx @@ -0,0 +1,39 @@ +import React from 'react'; + +import type {AvatarSize} from '../Avatar'; +import {Avatar, DEFAULT_AVATAR_SIZE} from '../Avatar'; +import {block} from '../utils/cn'; + +import i18n from './i18n'; + +const b = block('avatar-stack'); + +export type AvatarStackMoreButtonProps = Pick< + React.HTMLProps, + 'className' | 'onClick' | 'aria-label' +> & { + size?: AvatarSize; + count: number; +}; + +export const AvatarStackMoreButton = React.forwardRef< + HTMLButtonElement, + AvatarStackMoreButtonProps +>(({className, size = DEFAULT_AVATAR_SIZE, onClick, count, 'aria-label': ariaLabel}, ref) => { + return ( + + ); +}); + +AvatarStackMoreButton.displayName = 'AvatarStack.MoreButton'; diff --git a/src/components/AvatarStack/README.md b/src/components/AvatarStack/README.md new file mode 100644 index 0000000000..46198c1e3a --- /dev/null +++ b/src/components/AvatarStack/README.md @@ -0,0 +1,52 @@ + + +# AvatarStack + + + +```ts +import {AvatarStack} from '@gravity-ui/uikit'; +``` + +Stack of images with overlap over next image and optional control. This is usually users avatars. + +## Usage + +Component is not limit you to what components to render, basic usage is: + +```tsx + + + + + +``` + +## Properties + +| Name | Description | Type | Default | +| :---------- | :----------------------------------------------------------------------------------------------------------------------------------------------------------------- | :------------------------------------------------: | :-----: | +| max | How much avatars should be visible before more button. If avatars count is only 1 short from `max`, than more button would be replaced with avatar. | `number` | 3 | +| overlapSize | How much each item should overlap next one. `s` recommended for `Avatar`'s of sizes `xs`-`m`, `m` recomended for `l` size avatars and `l` overlap for `xl` avatars | `s`, `m`, `l` | `s` | +| size | Size for control displaying extra avatars. Value same to `Avatar` size. | `AvatarSize` | | +| className | Class name of root DOM node | `string` | | +| children | List of avatars, probably with some extra wrappers | `Object[]` | | +| renderMore | Custom render for control displaying extra avatars | `function(options: {count: number}): ReactElement` | | + +### AvatarStack.MoreButton + +Component for overriding more button + +```tsx + ( + + + + )} +> + + + + +``` diff --git a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-dark-chromium-linux.png b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-dark-chromium-linux.png new file mode 100644 index 0000000000..c9d1bcd627 Binary files /dev/null and b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-dark-chromium-linux.png differ diff --git a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-dark-webkit-linux.png b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-dark-webkit-linux.png new file mode 100644 index 0000000000..cafbd09942 Binary files /dev/null and b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-dark-webkit-linux.png differ diff --git a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-light-chromium-linux.png b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-light-chromium-linux.png new file mode 100644 index 0000000000..7c9ffe1618 Binary files /dev/null and b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-light-chromium-linux.png differ diff --git a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-light-webkit-linux.png b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-light-webkit-linux.png new file mode 100644 index 0000000000..697111e72f Binary files /dev/null and b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-light-webkit-linux.png differ diff --git a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButtonOmit-dark-chromium-linux.png b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButtonOmit-dark-chromium-linux.png new file mode 100644 index 0000000000..9584de0536 Binary files /dev/null and b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButtonOmit-dark-chromium-linux.png differ diff --git a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButtonOmit-dark-webkit-linux.png b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButtonOmit-dark-webkit-linux.png new file mode 100644 index 0000000000..d40c16dffc Binary files /dev/null and b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButtonOmit-dark-webkit-linux.png differ diff --git a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButtonOmit-light-chromium-linux.png b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButtonOmit-light-chromium-linux.png new file mode 100644 index 0000000000..6d1abcf1b5 Binary files /dev/null and b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButtonOmit-light-chromium-linux.png differ diff --git a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButtonOmit-light-webkit-linux.png b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButtonOmit-light-webkit-linux.png new file mode 100644 index 0000000000..3ab541002c Binary files /dev/null and b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButtonOmit-light-webkit-linux.png differ diff --git a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-SingleItem-dark-chromium-linux.png b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-SingleItem-dark-chromium-linux.png new file mode 100644 index 0000000000..4346f27ec6 Binary files /dev/null and b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-SingleItem-dark-chromium-linux.png differ diff --git a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-SingleItem-dark-webkit-linux.png b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-SingleItem-dark-webkit-linux.png new file mode 100644 index 0000000000..cef93a0a01 Binary files /dev/null and b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-SingleItem-dark-webkit-linux.png differ diff --git a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-SingleItem-light-chromium-linux.png b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-SingleItem-light-chromium-linux.png new file mode 100644 index 0000000000..5f0b07b72a Binary files /dev/null and b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-SingleItem-light-chromium-linux.png differ diff --git a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-SingleItem-light-webkit-linux.png b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-SingleItem-light-webkit-linux.png new file mode 100644 index 0000000000..2238b45c9d Binary files /dev/null and b/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-SingleItem-light-webkit-linux.png differ diff --git a/src/components/AvatarStack/__stories__/AvatarStack.stories.tsx b/src/components/AvatarStack/__stories__/AvatarStack.stories.tsx new file mode 100644 index 0000000000..30c1d63c70 --- /dev/null +++ b/src/components/AvatarStack/__stories__/AvatarStack.stories.tsx @@ -0,0 +1,139 @@ +import React from 'react'; + +import {faker} from '@faker-js/faker/locale/en'; +import type {Meta, StoryObj} from '@storybook/react'; + +import type {AvatarSize} from '../../Avatar'; +import {AVATAR_SIZES, Avatar, DEFAULT_AVATAR_SIZE} from '../../Avatar'; +import {Tooltip} from '../../Tooltip'; +import {AvatarStack} from '../AvatarStack'; + +type ComponentType = React.ComponentProps & { + avatarSize: AvatarSize; + randomAvatar: boolean; +}; + +const imgUrl = + ''; + +function getChildren({ + count = faker.number.int({min: 1, max: 30}), + avatarSize = DEFAULT_AVATAR_SIZE, + randomAvatar, +}: Partial<{count: number; avatarSize: AvatarSize; randomAvatar: boolean}>) { + return faker.helpers.uniqueArray( + () => ( + + ), + count, + ); +} + +const meta: Meta = { + title: 'Components/Data Display/AvatarStack', + component: AvatarStack, + args: { + overlapSize: 's', + avatarSize: DEFAULT_AVATAR_SIZE, + randomAvatar: true, + }, + argTypes: { + avatarSize: { + control: 'select', + options: Object.keys(AVATAR_SIZES), + name: 'Size of avatar', + description: 'Not part of component API', + }, + randomAvatar: { + control: 'boolean', + name: 'Use random avatars', + description: 'Not part of component API', + }, + }, + parameters: { + a11y: { + element: '#storybook-root', + config: { + rules: [ + { + id: 'color-contrast', + enabled: false, + selector: '.g-avatar__text', + }, + ], + }, + }, + }, +}; + +export default meta; + +type Story = StoryObj; + +function getTemplate({count}: Partial<{count: number}> = {}) { + // eslint-disable-next-line react/display-name + return (args: ComponentType) => { + const {avatarSize, randomAvatar, size, ...props} = args; + return ( + + {getChildren({count, avatarSize, randomAvatar})} + + ); + }; +} + +export const Default: Story = { + render: getTemplate(), +}; + +export const SingleItem: Story = { + render: getTemplate({count: 1}), +}; + +export const MoreButton: Story = { + render: getTemplate({count: 6}), +}; + +export const MoreButtonOmit: Story = { + render: getTemplate({count: 4}), + parameters: { + docs: { + description: { + component: 'In case when only one avatar is hidden, we omit rendering more button', + }, + }, + }, + args: { + max: 3, + }, +}; + +export const CustomMoreButton: Story = { + render(args) { + const {avatarSize, randomAvatar, size, ...props} = args; + + return ( + ( + + + + )} + > + {getChildren({count: 26, avatarSize, randomAvatar})} + + ); + }, +}; diff --git a/src/components/AvatarStack/__tests__/AvatarStack.visual.test.tsx b/src/components/AvatarStack/__tests__/AvatarStack.visual.test.tsx new file mode 100644 index 0000000000..fb3cded5ab --- /dev/null +++ b/src/components/AvatarStack/__tests__/AvatarStack.visual.test.tsx @@ -0,0 +1,25 @@ +import React from 'react'; + +import {test} from '~playwright/core'; + +import {AvatarStackStories} from './stories'; + +test.describe('AvatarStack', () => { + test('render story ', async ({mount, expectScreenshot}) => { + await mount(); + + await expectScreenshot(); + }); + + test('render story ', async ({mount, expectScreenshot}) => { + await mount(); + + await expectScreenshot(); + }); + + test('render story ', async ({mount, expectScreenshot}) => { + await mount(); + + await expectScreenshot(); + }); +}); diff --git a/src/components/AvatarStack/__tests__/stories.ts b/src/components/AvatarStack/__tests__/stories.ts new file mode 100644 index 0000000000..9662f19af8 --- /dev/null +++ b/src/components/AvatarStack/__tests__/stories.ts @@ -0,0 +1,5 @@ +import {composeStories} from '@storybook/react'; + +import * as CSFStories from '../__stories__/AvatarStack.stories'; + +export const AvatarStackStories = composeStories(CSFStories); diff --git a/src/components/AvatarStack/i18n/en.json b/src/components/AvatarStack/i18n/en.json new file mode 100644 index 0000000000..7a925041fb --- /dev/null +++ b/src/components/AvatarStack/i18n/en.json @@ -0,0 +1,3 @@ +{ + "more": ["and {{count}} more", "and {{count}} more", "and {{count}} more"] +} diff --git a/src/components/AvatarStack/i18n/index.ts b/src/components/AvatarStack/i18n/index.ts new file mode 100644 index 0000000000..600a77e666 --- /dev/null +++ b/src/components/AvatarStack/i18n/index.ts @@ -0,0 +1,8 @@ +import {addComponentKeysets} from '../../utils/addComponentKeysets'; + +import en from './en.json'; +import ru from './ru.json'; + +const COMPONENT = 'AvatarStack'; + +export default addComponentKeysets({en, ru}, COMPONENT); diff --git a/src/components/AvatarStack/i18n/ru.json b/src/components/AvatarStack/i18n/ru.json new file mode 100644 index 0000000000..8d7e17ac08 --- /dev/null +++ b/src/components/AvatarStack/i18n/ru.json @@ -0,0 +1,3 @@ +{ + "more": ["И eщё {{count}}", "И eщё {{count}}", "И eщё {{count}}"] +} diff --git a/src/components/AvatarStack/index.ts b/src/components/AvatarStack/index.ts new file mode 100644 index 0000000000..370ed35cb0 --- /dev/null +++ b/src/components/AvatarStack/index.ts @@ -0,0 +1,2 @@ +export {AvatarStack} from './AvatarStack'; +export type {AvatarStackProps, AvatarStackOverlapSize} from './types'; diff --git a/src/components/AvatarStack/types.ts b/src/components/AvatarStack/types.ts new file mode 100644 index 0000000000..c6e1af6bd9 --- /dev/null +++ b/src/components/AvatarStack/types.ts @@ -0,0 +1,41 @@ +import type React from 'react'; + +import type {AvatarSize} from '../Avatar'; + +export type AvatarStackOverlapSize = 's' | 'm' | 'l'; + +export interface AvatarStackProps { + /** Amount of avatars to be shown before more button. Default 3. */ + max?: number; + /** + * How much each avatar should overlap next one + * | Avatar sizes | Recommended overlap | + * | :----------: | :-----------------: | + * | `xs`-`m` | `s` | + * | `l` | `m` | + * | `xl` | `l` | + */ + overlapSize?: AvatarStackOverlapSize; + /** + * Size for control displaying count of extra avatars + */ + size?: AvatarSize; + className?: string; + /** + * Children would be wrapped for "stacking" + * @example + * + * + * + * + */ + children?: React.ReactNode; + /** + * Custom render for control displaying extra data + * @example + * }> + * + * + */ + renderMore?: (options: {count: number}) => React.ReactElement; +} diff --git a/src/components/index.ts b/src/components/index.ts index 1f00c0fa38..9257d9dca5 100644 --- a/src/components/index.ts +++ b/src/components/index.ts @@ -19,6 +19,7 @@ export * from './Divider'; export * from './DropdownMenu'; export * from './Hotkey'; export * from './Icon'; +export * from './AvatarStack'; export * from './Label'; export * from './Link'; export * from './List';