diff --git a/src/components/AvatarStack/AvatarStack.tsx b/src/components/AvatarStack/AvatarStack.tsx index 0f2340f02c..f7af7de699 100644 --- a/src/components/AvatarStack/AvatarStack.tsx +++ b/src/components/AvatarStack/AvatarStack.tsx @@ -5,6 +5,7 @@ import {block} from '../utils/cn'; import {AvatarStackItem} from './AvatarStackItem'; import {AvatarStackMore} from './AvatarStackMore'; import {AvatarStackMoreButton} from './AvatarStackMoreButton'; +import {AVATAR_STACK_DEFAULT_MAX} from './constants'; import type {AvatarStackProps} from './types'; import './AvatarStack.scss'; @@ -12,7 +13,7 @@ import './AvatarStack.scss'; const b = block('avatar-stack'); const AvatarStackComponent = ({ - max = 3, + max = AVATAR_STACK_DEFAULT_MAX, total, overlapSize = 's', size, @@ -21,8 +22,17 @@ const AvatarStackComponent = ({ renderMore, }: AvatarStackProps) => { const visibleItems: React.ReactElement[] = []; - let moreItems = 0; - const normalizedMax = max < 1 ? 1 : max; + + /** All avatars amount */ + const normalizedTotal = total ? Math.max(total, max) : React.Children.count(children); + + /** Amount avatars to be visible (doesn't include badge with remaining avatars) */ + let normalizedMax = max < 1 ? 1 : max; + // Skip rendering badge with +1, just show avatar instead + normalizedMax = normalizedTotal - normalizedMax > 1 ? normalizedMax : normalizedTotal; + + /** Remaining avatars */ + const moreItems = normalizedTotal - normalizedMax; React.Children.forEach(children, (child) => { if (!React.isValidElement(child)) { @@ -31,18 +41,12 @@ const AvatarStackComponent = ({ const item = {child}; - if (visibleItems.length <= normalizedMax) { + if (visibleItems.length < normalizedMax) { visibleItems.unshift(item); - } else { - moreItems += 1; } }); - moreItems = Math.max(moreItems, total ? total - normalizedMax : 0); - 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 @@ -58,7 +62,7 @@ const AvatarStackComponent = ({ )} ) : null} - {normalOverflow ? visibleItems.slice(0, normalizedMax) : visibleItems} + {visibleItems} ); }; 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 index 40e2daf352..6c0545af9b 100644 Binary files a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-dark-chromium-linux.png 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 index 7d018e56d3..2ab381bc7e 100644 Binary files a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-dark-webkit-linux.png 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 index 89bc9d30fc..b7f562dd03 100644 Binary files a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-light-chromium-linux.png 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 index 2666a959ed..1226f21dc4 100644 Binary files a/src/components/AvatarStack/__snapshots__/AvatarStack.visual.test.tsx-snapshots/AvatarStack-render-story-MoreButton-light-webkit-linux.png 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/__stories__/AvatarStack.stories.tsx b/src/components/AvatarStack/__stories__/AvatarStack.stories.tsx index 1963ab45e0..d968c95a26 100644 --- a/src/components/AvatarStack/__stories__/AvatarStack.stories.tsx +++ b/src/components/AvatarStack/__stories__/AvatarStack.stories.tsx @@ -7,6 +7,7 @@ import type {AvatarSize} from '../../Avatar'; import {AVATAR_SIZES, Avatar, DEFAULT_AVATAR_SIZE} from '../../Avatar'; import {Tooltip} from '../../Tooltip'; import {AvatarStack} from '../AvatarStack'; +import {AVATAR_STACK_DEFAULT_MAX} from '../constants'; type ComponentType = React.ComponentProps & { avatarSize: AvatarSize; @@ -17,10 +18,14 @@ const imgUrl = ''; function getChildren({ - count = faker.number.int({min: 1, max: 30}), + count, avatarSize = DEFAULT_AVATAR_SIZE, randomAvatar, -}: Partial<{count: number; avatarSize: AvatarSize; randomAvatar: boolean}>) { +}: { + count: number; + avatarSize: AvatarSize; + randomAvatar: boolean; +}) { return faker.helpers.uniqueArray( () => ( = { title: 'Components/Data Display/AvatarStack', component: AvatarStack, args: { + max: AVATAR_STACK_DEFAULT_MAX, overlapSize: 's', avatarSize: DEFAULT_AVATAR_SIZE, randomAvatar: true, @@ -76,7 +82,7 @@ export default meta; type Story = StoryObj; -function getTemplate({count}: Partial<{count: number}> = {}) { +function getTemplate({count = faker.number.int({min: 1, max: 30})}: Partial<{count: number}> = {}) { // eslint-disable-next-line react/display-name return (args: ComponentType) => { const {avatarSize, randomAvatar, size, ...props} = args; diff --git a/src/components/AvatarStack/constants.ts b/src/components/AvatarStack/constants.ts new file mode 100644 index 0000000000..5a96b4c2df --- /dev/null +++ b/src/components/AvatarStack/constants.ts @@ -0,0 +1 @@ +export const AVATAR_STACK_DEFAULT_MAX = 3;