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
+
+ {hasMoreButton ? (
+
+ {renderMore ? (
+ renderMore({count: moreItems})
+ ) : (
+
+ )}
+
+ ) : null}
+ {normalOverflow ? visibleItems.slice(0, max) : visibleItems}
+
+ );
+};
+
+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';