From 8dcc87c034669553bf52166aa3e9f417f6a8a287 Mon Sep 17 00:00:00 2001 From: Hellen Date: Fri, 2 Aug 2024 14:24:11 +0300 Subject: [PATCH 1/3] fix(DefinitionList): should be semantic correct (#211) --- .../DefinitionList/DefinitionList.scss | 24 ++- .../DefinitionList/DefinitionList.tsx | 165 +++++++++++------- .../DefinitionList/components/GroupLabel.tsx | 2 +- src/components/DefinitionList/types.ts | 11 +- src/components/DefinitionList/utils.ts | 64 +++---- 5 files changed, 149 insertions(+), 117 deletions(-) diff --git a/src/components/DefinitionList/DefinitionList.scss b/src/components/DefinitionList/DefinitionList.scss index 15d4a14c..eef1bd25 100644 --- a/src/components/DefinitionList/DefinitionList.scss +++ b/src/components/DefinitionList/DefinitionList.scss @@ -4,9 +4,11 @@ $block: '.#{variables.$ns}definition-list'; #{$block} { - margin: 0; + &__list { + margin: 0; + } - &__title { + &__group-title { margin-block-end: var(--g-spacing-3); &:not(:first-of-type) { @@ -14,10 +16,6 @@ $block: '.#{variables.$ns}definition-list'; } } - #{$block}__item:is(#{$block}__item_grouped) + #{$block}__item:not(#{$block}__item_grouped) { - margin-block-start: var(--g-spacing-5); - } - &__item { display: flex; align-items: baseline; @@ -34,8 +32,16 @@ $block: '.#{variables.$ns}definition-list'; } } + &_margin { + &:not(:first-of-type) { + margin-block-start: var(--g-spacing-5); + } + } + &__term-container { - flex: 0 0 300px; + flex: 0 0 auto; + width: 300px; + max-width: 300px; display: flex; align-items: baseline; @@ -133,10 +139,10 @@ $block: '.#{variables.$ns}definition-list'; #{$block}__item + #{$block}__item { margin-block-start: var(--g-spacing-3); } - #{$block}__title:not(:first-of-type) { + #{$block}__group-title:not(:first-of-type) { margin-block-start: var(--g-spacing-8); } - #{$block}__item:is(#{$block}__item_grouped) + #{$block}__item:not(#{$block}__item_grouped) { + #{$block}_margin:not(:first-of-type) { margin-block-start: var(--g-spacing-8); } } diff --git a/src/components/DefinitionList/DefinitionList.tsx b/src/components/DefinitionList/DefinitionList.tsx index f04dfe86..a42a1eb3 100644 --- a/src/components/DefinitionList/DefinitionList.tsx +++ b/src/components/DefinitionList/DefinitionList.tsx @@ -3,20 +3,16 @@ import React from 'react'; import {Definition} from './components/Definition'; import {GroupLabel} from './components/GroupLabel'; import {Term} from './components/Term'; -import {DefinitionListProps} from './types'; import { - b, - getFlattenItems, - getKeyStyles, - getTitle, - getValueStyles, - isGroup, - isUnbreakableOver, -} from './utils'; + DefinitionListGranularProps, + DefinitionListGroupedProps, + DefinitionListProps, +} from './types'; +import {b, getAllItemsAsGroups, getTitle, isUnbreakableOver, onlySingleItems} from './utils'; import './DefinitionList.scss'; -export function DefinitionList({ +function DefinitionListGranular({ items, responsive, direction = 'horizontal', @@ -26,68 +22,115 @@ export function DefinitionList({ itemClassName, copyPosition = 'outside', qa, -}: DefinitionListProps) { - const keyStyle = getKeyStyles({nameMaxWidth, direction}); +}: DefinitionListGranularProps) { + const keyStyle = nameMaxWidth ? {maxWidth: nameMaxWidth, width: nameMaxWidth} : {}; - const valueStyle = getValueStyles({contentMaxWidth, direction}); + const valueStyle = + typeof contentMaxWidth === 'number' + ? {width: contentMaxWidth, maxWidth: contentMaxWidth} + : {}; const normalizedItems = React.useMemo(() => { - return getFlattenItems(items).map((value, index) => ({...value, key: index})); + return items.map((value, index) => ({...value, key: index})); }, [items]); return ( -
+
+
+ {normalizedItems.map((item) => { + const { + name, + key, + content, + contentTitle, + nameTitle, + copyText, + note, + multilineName, + } = item; + + return ( +
+
+ +
+
+ +
+
+ ); + })} +
+
+ ); +} + +function DefinitionListGrouped({ + items, + className, + itemClassName, + ...rest +}: DefinitionListGroupedProps) { + const normalizedItems = React.useMemo(() => { + return items.map((value, index) => ({...value, key: index})); + }, [items]); + + return ( +
{normalizedItems.map((item) => { - if (isGroup(item)) { - const {key, label} = item; - return ; - } - const { - name, - key, - content, - contentTitle, - nameTitle, - copyText, - note, - multilineName, - isGrouped, - } = item; + const {key, label} = item; return ( -
-
- -
-
- + {label && } + {item.items && ( + -
-
+ )} + ); })} -
+ ); } + +export function DefinitionList({items, ...rest}: DefinitionListProps) { + if (onlySingleItems(items)) { + return ; + } + + const preparedItems = getAllItemsAsGroups(items); + + return ; +} diff --git a/src/components/DefinitionList/components/GroupLabel.tsx b/src/components/DefinitionList/components/GroupLabel.tsx index fd59f54f..c02e76c9 100644 --- a/src/components/DefinitionList/components/GroupLabel.tsx +++ b/src/components/DefinitionList/components/GroupLabel.tsx @@ -10,7 +10,7 @@ interface GroupLabelProps { export function GroupLabel({label}: GroupLabelProps) { return ( -
+
{label} diff --git a/src/components/DefinitionList/types.ts b/src/components/DefinitionList/types.ts index cd55a30c..3132f3c0 100644 --- a/src/components/DefinitionList/types.ts +++ b/src/components/DefinitionList/types.ts @@ -20,10 +20,6 @@ export interface DefinitionListSingleItem { multilineName?: boolean; } -export interface DefinitionListItemGrouped extends DefinitionListSingleItem { - isGrouped?: boolean; -} - export type DefinitionListItem = DefinitionListSingleItem | DefinitionListGroup; export type DefinitionListDirection = 'vertical' | 'horizontal'; @@ -38,3 +34,10 @@ export interface DefinitionListProps extends QAProps { className?: string; itemClassName?: string; } + +export interface DefinitionListGranularProps extends Omit { + items: DefinitionListSingleItem[]; +} +export interface DefinitionListGroupedProps extends Omit { + items: DefinitionListGroup[]; +} diff --git a/src/components/DefinitionList/utils.ts b/src/components/DefinitionList/utils.ts index 62eb2f60..7a12d6ff 100644 --- a/src/components/DefinitionList/utils.ts +++ b/src/components/DefinitionList/utils.ts @@ -2,13 +2,7 @@ import React from 'react'; import {block} from '../utils/cn'; -import type { - DefinitionListGroup, - DefinitionListItem, - DefinitionListItemGrouped, - DefinitionListProps, - DefinitionListSingleItem, -} from './types'; +import type {DefinitionListGroup, DefinitionListItem, DefinitionListSingleItem} from './types'; export const b = block('definition-list'); @@ -23,19 +17,31 @@ export function isUnbreakableOver(limit: number) { export const isGroup = (item: DefinitionListItem): item is DefinitionListGroup => 'label' in item && !('name' in item); -export function getFlattenItems( +export const onlySingleItems = (items: DefinitionListItem[]): items is DefinitionListSingleItem[] => + !items.some((el) => isGroup(el)); + +export function getAllItemsAsGroups( items: (DefinitionListSingleItem | DefinitionListGroup)[], -): (DefinitionListItemGrouped | DefinitionListGroup)[] { - return items.reduce<(DefinitionListSingleItem | DefinitionListGroup)[]>((acc, item) => { +): DefinitionListGroup[] { + const result: DefinitionListGroup[] = []; + let temporaryList: DefinitionListSingleItem[] = []; + for (const item of items) { if (isGroup(item)) { - acc.push({label: item.label}); - const items = [...(item.items ?? [])].map((el) => ({...el, isGrouped: true})); - acc.push(...items); + if (temporaryList.length) { + result.push({items: temporaryList, label: null}); + temporaryList = []; + } + + result.push(item); } else { - acc.push(item); + temporaryList.push(item); } - return acc; - }, []); + } + if (temporaryList.length) { + result.push({items: temporaryList, label: null}); + temporaryList = []; + } + return result; } export function getTitle(title?: string, content?: React.ReactNode) { @@ -49,29 +55,3 @@ export function getTitle(title?: string, content?: React.ReactNode) { return undefined; } - -export function getKeyStyles({ - nameMaxWidth, - direction, -}: Pick) { - if (!nameMaxWidth) { - return {}; - } - if (direction === 'vertical') { - return {maxWidth: nameMaxWidth}; - } - return {flexBasis: nameMaxWidth}; -} - -export function getValueStyles({ - contentMaxWidth, - direction, -}: Pick) { - if (!(typeof contentMaxWidth === 'number')) { - return {}; - } - if (direction === 'vertical') { - return {maxWidth: contentMaxWidth}; - } - return {flexBasis: contentMaxWidth, maxWidth: contentMaxWidth}; -} From be23d4e5949ac918ac945bbf829044520c7d0106 Mon Sep 17 00:00:00 2001 From: Vlad Furman Date: Tue, 20 Aug 2024 16:55:40 +0300 Subject: [PATCH 2/3] feat(Reactions): added new Reactions component (#197) --- CODEOWNERS | 2 + src/components/Reactions/README.md | 108 +++++++++++ src/components/Reactions/Reaction.tsx | 94 ++++++++++ src/components/Reactions/Reactions.scss | 48 +++++ src/components/Reactions/Reactions.tsx | 174 ++++++++++++++++++ .../__stories__/Reactions.stories.tsx | 66 +++++++ .../Reactions/__tests__/Reactions.test.tsx | 74 ++++++++ .../Reactions/__tests__/mock/mockData.ts | 57 ++++++ .../Reactions/__tests__/mock/mockHooks.tsx | 108 +++++++++++ src/components/Reactions/context.ts | 24 +++ src/components/Reactions/hooks.ts | 111 +++++++++++ src/components/Reactions/i18n/en.json | 3 + src/components/Reactions/i18n/index.ts | 8 + src/components/Reactions/i18n/ru.json | 3 + src/components/Reactions/index.ts | 1 + 15 files changed, 881 insertions(+) create mode 100644 src/components/Reactions/README.md create mode 100644 src/components/Reactions/Reaction.tsx create mode 100644 src/components/Reactions/Reactions.scss create mode 100644 src/components/Reactions/Reactions.tsx create mode 100644 src/components/Reactions/__stories__/Reactions.stories.tsx create mode 100644 src/components/Reactions/__tests__/Reactions.test.tsx create mode 100644 src/components/Reactions/__tests__/mock/mockData.ts create mode 100644 src/components/Reactions/__tests__/mock/mockHooks.tsx create mode 100644 src/components/Reactions/context.ts create mode 100644 src/components/Reactions/hooks.ts create mode 100644 src/components/Reactions/i18n/en.json create mode 100644 src/components/Reactions/i18n/index.ts create mode 100644 src/components/Reactions/i18n/ru.json create mode 100644 src/components/Reactions/index.ts diff --git a/CODEOWNERS b/CODEOWNERS index b0894972..f95a046c 100644 --- a/CODEOWNERS +++ b/CODEOWNERS @@ -4,9 +4,11 @@ /src/components/FilePreview @KirillDyachkovskiy /src/components/FormRow @ogonkov /src/components/HelpPopover @Raubzeug +/src/components/Notifications @Ruminat /src/components/OnboardingMenu @nikita-jpg /src/components/PlaceholderContainer @Marginy605 /src/components/PromoSheet @Avol-V +/src/components/Reactions @Ruminat /src/components/SharePopover @niktverd /src/components/StoreBadge @NikitaCG /src/components/Stories @darkgenius diff --git a/src/components/Reactions/README.md b/src/components/Reactions/README.md new file mode 100644 index 00000000..b3971ec9 --- /dev/null +++ b/src/components/Reactions/README.md @@ -0,0 +1,108 @@ +## Reactions + +Component for user reactions (e.g. ๐Ÿ‘, ๐Ÿ˜Š, ๐Ÿ˜Ž etc) as new GitHub comments for example. + +### Usage example + +```typescript +import React from 'react'; + +import {PaletteOption} from '@gravity-ui/uikit'; +import {ReactionState, Reactions} from '@gravity-ui/components'; + +const user = { + spongeBob: {name: 'Sponge Bob'}, + patrick: {name: 'Patrick'}, +}; + +const currentUser = user.spongeBob; + +const option = { + 'thumbs-up': {content: '๐Ÿ‘', value: 'thumbs-up'}, + cool: {content: '๐Ÿ˜Ž', value: 'cool'}, +} satisfies Record; + +const options = Object.values(option); + +const YourComponent = () => { + // You can set up a mapping: reaction.value -> users reacted + const [usersReacted, setUsersReacted] = React.useState({ + [option.cool.value]: [user.spongeBob], + }); + + // And then convert that mapping into an array of ReactionState + const reactions = React.useMemo( + () => + Object.entries(usersReacted).map( + ([value, users]): ReactionState => ({ + value, + counter: users.length, + selected: users.some(({name}) => name === currentUser.name), + }), + ), + [usersReacted], + ); + + // You can then handle clicking on a reaction with changing the inital mapping, + // and the array of ReactionState will change accordingly + const onToggle = React.useCallback( + (value: string) => { + if (!usersReacted[value]) { + // If the reaction is not present yet + setUsersReacted((current) => ({...current, [value]: [currentUser]})); + } else if (!usersReacted[value].some(({name}) => name === currentUser.name)) { + // If the reaction is present, but current user hasn't selected it yet + setUsersReacted((current) => ({ + ...current, + [value]: [...usersReacted[value], currentUser], + })); + } else if (usersReacted[value].length > 1) { + // If the user used that reaction, and he's not the only one who used it + setUsersReacted((current) => ({ + ...current, + [value]: usersReacted[value].filter(({name}) => name !== currentUser.name), + })); + } else { + // If the user used that reaction, and he's the only one who used it + setUsersReacted((current) => { + const newValue = {...current}; + delete newValue[value]; + return newValue; + }); + } + }, + [usersReacted], + ); + + return ( + + ); +}; +``` + +For more code examples go to [Reactions.stories.tsx](https://github.com/gravity-ui/components/blob/main/src/components/Reactions/__stories__/Reactions.stories.tsx). + +### Props + +**ReactionsProps** (main component props โ€” Reactions' list): + +| Property | Type | Required | Default | Description | +| :--------------- | :------------------------------------------ | :------: | :------ | :--------------------------------------------------------------------------------------------- | +| `className` | `string` | | | HTML `class` attribute | +| `onToggle` | `(value: string) => void` | | | Fires when a user clicks on a Reaction (in a Palette or in the Reactions' list) | +| `paletteProps` | `ReactionsPaletteProps` | `true` | | Notifications' palette props โ€” it's a `Palette` component with available reactions to the user | +| `qa` | `string` | | | `qa` attribute for testing | +| `reactions` | `PaletteOption[]` | `true` | | List of all available reactions | +| `reactionsState` | `ReactionState[]` | `true` | | List of reactions that were used | +| `readOnly` | `boolean` | | `false` | readOnly state (usage example: only signed in users can react) | +| `renderTooltip` | `(state: ReactionState) => React.ReactNode` | | | Reaction's tooltip with the list of reacted users for example | +| `size` | `ButtonSize` | | `m` | Buttons's size | +| `style` | `React.CSSProperties` | | | HTML `style` attribute | + +**ReactionState** (single reaction props): + +| Property | Type | Required | Default | Description | +| :--------- | :---------------- | :------: | :------ | :-------------------------------- | +| `counter` | `React.ReactNode` | | | How many users used this reaction | +| `selected` | `boolean` | | | Is reaction selected by the user | +| `value` | `string` | | | Reaction's unique value (ID) | diff --git a/src/components/Reactions/Reaction.tsx b/src/components/Reactions/Reaction.tsx new file mode 100644 index 00000000..2ac0092c --- /dev/null +++ b/src/components/Reactions/Reaction.tsx @@ -0,0 +1,94 @@ +import React from 'react'; + +import {Button, ButtonSize, PaletteOption, PopoverProps, Popup} from '@gravity-ui/uikit'; + +import {block} from '../utils/cn'; + +import {useReactionsContext} from './context'; +import {useReactionsPopup} from './hooks'; + +export type ReactionProps = Pick; + +export interface ReactionState { + /** + * Reaction's unique value (ID). + */ + value: string; + /** + * Should be true when the user used this reaction. + */ + selected?: boolean; + /** + * Display a number after the icon. + * Represents the number of users who used this reaction. + */ + counter?: React.ReactNode; +} + +interface ReactionInnerProps extends Pick { + reaction: ReactionState; + size: ButtonSize; + tooltip?: React.ReactNode; + onClick?: (value: string) => void; +} + +const popupDefaultPlacement: PopoverProps['placement'] = [ + 'bottom-start', + 'bottom', + 'bottom-end', + 'top-start', + 'top', + 'top-end', +]; + +const b = block('reactions'); + +export function Reaction(props: ReactionInnerProps) { + const {value, selected, counter} = props.reaction; + const {size, content, tooltip, onClick} = props; + + const onClickCallback = React.useCallback(() => onClick?.(value), [onClick, value]); + + const buttonRef = React.useRef(null); + const {onMouseEnter, onMouseLeave} = useReactionsPopup(props.reaction, buttonRef); + const {openedTooltip: currentHoveredReaction} = useReactionsContext(); + + const button = ( + + ); + + return tooltip ? ( +
+ {button} + + {currentHoveredReaction && currentHoveredReaction.reaction.value === value ? ( + + {tooltip} + + ) : null} +
+ ) : ( + button + ); +} diff --git a/src/components/Reactions/Reactions.scss b/src/components/Reactions/Reactions.scss new file mode 100644 index 00000000..5a0c3e62 --- /dev/null +++ b/src/components/Reactions/Reactions.scss @@ -0,0 +1,48 @@ +@use '../variables'; + +$block: '.#{variables.$ns}reactions'; + +#{$block} { + &__popup { + padding: 8px; + } + + &__add-reaction-popover { + max-width: unset; + } + + &__reaction-button-content_size_xs { + font-size: 12px; + } + &__reaction-button-content_size_xs#{&}__reaction-button-content_text { + font-size: var(--g-text-caption-1-font-size); + } + + &__reaction-button-content_size_s { + font-size: 16px; + } + &__reaction-button-content_size_s#{&}__reaction-button-content_text { + font-size: var(--g-text-caption-2-font-size); + } + + &__reaction-button-content_size_m { + font-size: 16px; + } + &__reaction-button-content_size_m#{&}__reaction-button-content_text { + font-size: var(--g-text-body-1-font-size); + } + + &__reaction-button-content_size_l { + font-size: 16px; + } + &__reaction-button-content_size_l#{&}__reaction-button-content_text { + font-size: var(--g-text-subheader-1-font-size); + } + + &__reaction-button-content_size_xl { + font-size: 20px; + } + &__reaction-button-content_size_xl#{&}__reaction-button-content_text { + font-size: var(--g-text-subheader-2-font-size); + } +} diff --git a/src/components/Reactions/Reactions.tsx b/src/components/Reactions/Reactions.tsx new file mode 100644 index 00000000..08e783e9 --- /dev/null +++ b/src/components/Reactions/Reactions.tsx @@ -0,0 +1,174 @@ +import React from 'react'; + +import {FaceSmile} from '@gravity-ui/icons'; +import { + Button, + DOMProps, + Flex, + Icon, + Palette, + PaletteOption, + PaletteProps, + Popover, + QAProps, +} from '@gravity-ui/uikit'; +import xor from 'lodash/xor'; + +import {block} from '../utils/cn'; + +import {Reaction, ReactionProps, ReactionState} from './Reaction'; +import {ReactionsContextProvider, ReactionsContextTooltipProps} from './context'; +import {i18n} from './i18n'; + +import './Reactions.scss'; + +const b = block('reactions'); + +export type ReactionsPaletteProps = Pick< + PaletteProps, + 'columns' | 'rowClassName' | 'optionClassName' +>; + +export interface ReactionsProps extends Pick, QAProps, DOMProps { + /** + * All available reactions. + */ + reactions: ReactionProps[]; + /** + * Users' reactions. + */ + reactionsState: ReactionState[]; + /** + * Reactions' palette props. + */ + paletteProps?: ReactionsPaletteProps; + /** + * Reactions' readonly state (when a user is unable to react for some reason). + */ + readOnly?: boolean; + /** + * If present, when a user hovers over the reaction, a popover appears with renderTooltip(state) content. + * Can be used to display users who used this reaction. + */ + renderTooltip?: (state: ReactionState) => React.ReactNode; + /** + * Callback for clicking on a reaction in the Palette or directly in the reactions' list. + */ + onToggle?: (value: string) => void; +} + +const buttonSizeToIconSize = { + xs: '12px', + s: '16px', + m: '16px', + l: '16px', + xl: '20px', +}; + +export function Reactions({ + reactions, + reactionsState, + className, + style, + size = 'm', + paletteProps, + readOnly, + qa, + renderTooltip, + onToggle, +}: ReactionsProps) { + const [currentHoveredReaction, setCurrentHoveredReaction] = React.useState< + ReactionsContextTooltipProps | undefined + >(undefined); + + const paletteOptionsMap = React.useMemo( + () => + reactions.reduce>((acc, current) => { + // eslint-disable-next-line no-param-reassign + acc[current.value] = current; + return acc; + }, {}), + [reactions], + ); + + const paletteValue = React.useMemo( + () => + reactionsState + .filter((reaction) => reaction.selected) + .map((reaction) => reaction.value), + [reactionsState], + ); + + const onUpdatePalette = React.useCallback( + (updated: string[]) => { + const diffValues = xor(paletteValue, updated); + for (const diffValue of diffValues) { + onToggle?.(diffValue); + } + }, + [onToggle, paletteValue], + ); + + const paletteContent = React.useMemo( + () => ( + + ), + [paletteProps, reactions, paletteValue, size, onUpdatePalette], + ); + + return ( + + + {/* Reactions' list */} + {reactionsState.map((reaction) => { + const content = paletteOptionsMap[reaction.value]?.content ?? '?'; + + return ( + + ); + })} + + {/* Add reaction button */} + {readOnly ? null : ( + + + + )} + + + ); +} diff --git a/src/components/Reactions/__stories__/Reactions.stories.tsx b/src/components/Reactions/__stories__/Reactions.stories.tsx new file mode 100644 index 00000000..435b4b51 --- /dev/null +++ b/src/components/Reactions/__stories__/Reactions.stories.tsx @@ -0,0 +1,66 @@ +import React from 'react'; + +import {Flex, Text} from '@gravity-ui/uikit'; +import {Meta, StoryFn} from '@storybook/react'; + +import {Reactions} from '../Reactions'; +import {useMockReactions} from '../__tests__/mock/mockHooks'; + +export default { + title: 'Components/Reactions', + component: Reactions, +} as Meta; + +export const Default: StoryFn = () => { + return ; +}; + +export const Readonly: StoryFn = () => { + const {reactions, reactionsState, renderTooltip, onToggle} = useMockReactions(); + + return ( + ( + + You must be singed in to react + {renderTooltip(state)} + + ) + : undefined + } + reactionsState={reactionsState} + onToggle={onToggle} + readOnly={true} + /> + ); +}; + +export const Size: StoryFn = () => { + return ( + + + Size XS + + + + Size S + + + + Size M + + + + Size L + + + + Size XL + + + + ); +}; diff --git a/src/components/Reactions/__tests__/Reactions.test.tsx b/src/components/Reactions/__tests__/Reactions.test.tsx new file mode 100644 index 00000000..45da5db6 --- /dev/null +++ b/src/components/Reactions/__tests__/Reactions.test.tsx @@ -0,0 +1,74 @@ +import React from 'react'; + +import {ButtonSize} from '@gravity-ui/uikit'; +import userEvent from '@testing-library/user-event'; + +import {render, screen, within} from '../../../../test-utils/utils'; + +import {reactionsPalletteMockOption as option} from './mock/mockData'; +import {TestReactions} from './mock/mockHooks'; + +const qaId = 'reactions-component'; + +describe('Reactions', () => { + test('render Reactions', async () => { + render(); + + const reactions = screen.getByTestId(qaId); + expect(reactions).toBeVisible(); + }); + + test.each(new Array('xs', 's', 'm', 'l', 'xl'))( + 'render with given "%s" size', + (size) => { + render(); + + const $component = screen.getByTestId(qaId); + const $reactions = within($component).getAllByRole('button'); + + $reactions.forEach(($reaction: HTMLElement) => { + expect($reaction).toHaveClass(`g-button_size_${size}`); + }); + }, + ); + + test('show given reaction', () => { + render(); + + const text = screen.getByText(option.cool.content as string); + + expect(text).toBeVisible(); + }); + + test('add className and style', () => { + const className = 'my-class'; + const style = {color: 'red'}; + + render(); + + const $component = screen.getByTestId(qaId); + + expect($component).toHaveClass(className); + expect($component).toHaveStyle(style); + }); + + test('can (un)select an option', async () => { + render(); + + const $component = screen.getByTestId(qaId); + const $reactions = within($component).getAllByRole('button'); + + const $firstReaction = await screen.findByText(option.cool.content as string); + const $secondReaction = await screen.findByText(option.laughing.content as string); + + expect($reactions[0].getAttribute('aria-pressed')).toBe('false'); + expect($reactions[1].getAttribute('aria-pressed')).toBe('true'); + + const user = userEvent.setup(); + await user.click($firstReaction); + await user.click($secondReaction); + + expect($reactions[0].getAttribute('aria-pressed')).toBe('true'); + expect($reactions[1].getAttribute('aria-pressed')).toBe('false'); + }); +}); diff --git a/src/components/Reactions/__tests__/mock/mockData.ts b/src/components/Reactions/__tests__/mock/mockData.ts new file mode 100644 index 00000000..1dcceb8e --- /dev/null +++ b/src/components/Reactions/__tests__/mock/mockData.ts @@ -0,0 +1,57 @@ +import {PaletteOption} from '@gravity-ui/uikit'; + +const reactionsAvatar = { + spongeBob: + '', + patrick: + '', + squidward: + '', +}; + +export type ReactionsMockUser = {name: string; avatar: string}; + +export const reactionsMockUser = { + spongeBob: {name: 'Sponge Bob', avatar: reactionsAvatar.spongeBob}, + patrick: {name: 'Patrick', avatar: reactionsAvatar.patrick}, + squidward: {name: 'Squidward', avatar: reactionsAvatar.squidward}, +} satisfies Record; + +const baseMockOption = { + 'smiling-face': {content: '๐Ÿ˜Š'}, + heart: {content: 'โค๏ธ'}, + 'thumbs-up': {content: '๐Ÿ‘'}, + laughing: {content: '๐Ÿ˜‚'}, + 'hearts-eyes': {content: '๐Ÿ˜'}, + cool: {content: '๐Ÿ˜Ž'}, + tongue: {content: '๐Ÿ˜›'}, + angry: {content: '๐Ÿ˜ก'}, + sad: {content: '๐Ÿ˜ข'}, + surprised: {content: '๐Ÿ˜ฏ'}, + 'face-screaming': {content: '๐Ÿ˜ฑ'}, + 'smiling-face-with-open-hands': { + content: '๐Ÿค—', + }, + nauseated: {content: '๐Ÿคข'}, + 'lying-face': {content: '๐Ÿคฅ'}, + 'star-struck': {content: '๐Ÿคฉ'}, + 'face-with-hand-over-mouth': { + content: '๐Ÿคญ', + }, + vomiting: {content: '๐Ÿคฎ'}, + partying: {content: '๐Ÿฅณ'}, + woozy: {content: '๐Ÿฅด'}, + 'cold-face': {content: '๐Ÿฅถ'}, +}; + +export const reactionsPalletteMockOption = baseMockOption as Record< + keyof typeof baseMockOption, + PaletteOption +>; + +for (const value of Object.keys(reactionsPalletteMockOption)) { + reactionsPalletteMockOption[value as keyof typeof baseMockOption].value = value; + reactionsPalletteMockOption[value as keyof typeof baseMockOption].title = value; +} + +export const reactionsPalletteMockOptions = Object.values(reactionsPalletteMockOption); diff --git a/src/components/Reactions/__tests__/mock/mockHooks.tsx b/src/components/Reactions/__tests__/mock/mockHooks.tsx new file mode 100644 index 00000000..92546ab8 --- /dev/null +++ b/src/components/Reactions/__tests__/mock/mockHooks.tsx @@ -0,0 +1,108 @@ +import React from 'react'; + +import {Flex, Text, User} from '@gravity-ui/uikit'; + +import {ReactionState} from '../../Reaction'; +import {Reactions, ReactionsProps} from '../../Reactions'; + +import { + ReactionsMockUser, + reactionsPalletteMockOption as option, + reactionsPalletteMockOptions as options, + reactionsMockUser as user, +} from './mockData'; + +const currentUser = user.spongeBob; + +const renderUserReacted = ({avatar, name}: ReactionsMockUser) => { + return ( + + {name} (you) + + ) : ( + name + ) + } + key={name} + size={'xs'} + /> + ); +}; + +const renderUsersReacted = (users: ReactionsMockUser[]) => { + return ( + + {users.map(renderUserReacted)} + + ); +}; + +const getTooltip = (users: ReactionsMockUser[]) => renderUsersReacted(users); + +export function useMockReactions(): ReactionsProps { + const [usersReacted, setUsersReacted] = React.useState({ + [option.cool.value]: [user.patrick], + [option.laughing.value]: [user.patrick, user.spongeBob], + [option['thumbs-up'].value]: [user.patrick, user.spongeBob, user.squidward], + [option['hearts-eyes'].value]: [user.spongeBob], + [option['cold-face'].value]: [user.squidward], + [option.sad.value]: [user.squidward], + }); + + const reactionsState = React.useMemo( + () => + Object.entries(usersReacted).map( + ([value, users]): ReactionState => ({ + value, + counter: users.length, + selected: users.some(({name}) => name === currentUser.name), + }), + ), + [usersReacted], + ); + + const onToggle = React.useCallback( + (value: string) => { + if (!usersReacted[value]) { + setUsersReacted((current) => ({...current, [value]: [currentUser]})); + } else if (!usersReacted[value].some(({name}) => name === currentUser.name)) { + setUsersReacted((current) => ({ + ...current, + [value]: [...usersReacted[value], currentUser], + })); + } else if (usersReacted[value].length > 1) { + setUsersReacted((current) => ({ + ...current, + [value]: usersReacted[value].filter(({name}) => name !== currentUser.name), + })); + } else { + setUsersReacted((current) => { + const newValue = {...current}; + delete newValue[value]; + return newValue; + }); + } + }, + [usersReacted], + ); + + const renderTooltip = React.useCallback( + (reaciton: ReactionState) => getTooltip(usersReacted[reaciton.value]), + [usersReacted], + ); + + return { + reactions: options, + reactionsState, + renderTooltip, + onToggle, + }; +} + +export function TestReactions(props: Partial) { + return ; +} diff --git a/src/components/Reactions/context.ts b/src/components/Reactions/context.ts new file mode 100644 index 00000000..833ed652 --- /dev/null +++ b/src/components/Reactions/context.ts @@ -0,0 +1,24 @@ +import React from 'react'; + +import type {ReactionState} from './Reaction'; + +export interface ReactionsContextTooltipProps { + reaction: ReactionState; + ref: React.RefObject; + open: boolean; +} + +export interface ReactionsContext { + openedTooltip?: ReactionsContextTooltipProps; + setOpenedTooltip: (props: ReactionsContextTooltipProps | undefined) => void; +} + +const context = React.createContext({ + setOpenedTooltip: () => undefined, +}); + +export const ReactionsContextProvider = context.Provider; + +export function useReactionsContext(): ReactionsContext { + return React.useContext(context); +} diff --git a/src/components/Reactions/hooks.ts b/src/components/Reactions/hooks.ts new file mode 100644 index 00000000..72a2a839 --- /dev/null +++ b/src/components/Reactions/hooks.ts @@ -0,0 +1,111 @@ +import React from 'react'; + +import type {ReactionState} from './Reaction'; +import {useReactionsContext} from './context'; + +const DELAY = { + focusTimeout: 600, + openTimeout: 200, + closeTimeout: 200, +} as const; + +export function useReactionsPopup( + reaction: ReactionState, + ref: React.RefObject, +) { + const {value} = reaction; + + const {openedTooltip: currentHoveredReaction, setOpenedTooltip: setCurrentHoveredReaction} = + useReactionsContext(); + + const {delayedCall: setDelayedOpen, clearTimeoutRef: clearOpenTimeout} = useTimeoutRef(); + const {delayedCall: setDelayedClose, clearTimeoutRef: clearCloseTimeout} = useTimeoutRef(); + + const open = React.useCallback(() => { + setCurrentHoveredReaction({reaction, open: true, ref}); + }, [reaction, ref, setCurrentHoveredReaction]); + + const close = React.useCallback(() => { + clearOpenTimeout(); + + if (currentHoveredReaction?.reaction.value === value && currentHoveredReaction.open) { + setCurrentHoveredReaction({...currentHoveredReaction, open: false}); + } + }, [clearOpenTimeout, currentHoveredReaction, setCurrentHoveredReaction, value]); + + const focus = React.useCallback(() => { + clearCloseTimeout(); + + // If already hovered over current reaction + if (currentHoveredReaction && currentHoveredReaction.reaction.value === reaction.value) { + // But if it's not opened yet + if (!currentHoveredReaction.open) { + setDelayedOpen(open, DELAY.openTimeout); + } + } else { + setCurrentHoveredReaction({reaction, open: false, ref}); + + setDelayedOpen(open, DELAY.openTimeout); + } + }, [ + clearCloseTimeout, + currentHoveredReaction, + open, + reaction, + ref, + setCurrentHoveredReaction, + setDelayedOpen, + ]); + + const delayedOpenPopup = React.useCallback(() => { + clearCloseTimeout(); + setDelayedOpen(focus, DELAY.focusTimeout); + }, [clearCloseTimeout, focus, setDelayedOpen]); + + const delayedClosePopup = React.useCallback(() => { + clearOpenTimeout(); + + setDelayedClose(close, DELAY.closeTimeout); + }, [clearOpenTimeout, close, setDelayedClose]); + + const onMouseEnter: React.MouseEventHandler< + HTMLDivElement | HTMLButtonElement | HTMLAnchorElement + > = delayedOpenPopup; + + const onMouseLeave: React.MouseEventHandler< + HTMLDivElement | HTMLButtonElement | HTMLAnchorElement + > = delayedClosePopup; + + React.useEffect(() => { + // When the tab gets focus we need to hide the popup, + // because the user might have changed the cursor position. + window.addEventListener('focus', close); + + return () => { + window.removeEventListener('focus', close); + }; + }, [close]); + + return {onMouseEnter, onMouseLeave}; +} + +function useTimeoutRef() { + const timeoutRef = React.useRef | null>(null); + + const clearTimeoutRef = React.useCallback(() => { + if (timeoutRef.current !== null) { + clearTimeout(timeoutRef.current); + timeoutRef.current = null; + } + }, []); + + const delayedCall = React.useCallback( + (handler: () => void, delay: number) => { + clearTimeoutRef(); + timeoutRef.current = setTimeout(handler, delay); + }, + [clearTimeoutRef], + ); + + return {delayedCall, clearTimeoutRef}; +} diff --git a/src/components/Reactions/i18n/en.json b/src/components/Reactions/i18n/en.json new file mode 100644 index 00000000..6f79bdaf --- /dev/null +++ b/src/components/Reactions/i18n/en.json @@ -0,0 +1,3 @@ +{ + "add-reaction": "Add reaction" +} diff --git a/src/components/Reactions/i18n/index.ts b/src/components/Reactions/i18n/index.ts new file mode 100644 index 00000000..b113f612 --- /dev/null +++ b/src/components/Reactions/i18n/index.ts @@ -0,0 +1,8 @@ +import {addComponentKeysets} from '@gravity-ui/uikit/i18n'; + +import {NAMESPACE} from '../../utils/cn'; + +import en from './en.json'; +import ru from './ru.json'; + +export const i18n = addComponentKeysets({en, ru}, `${NAMESPACE}reactions`); diff --git a/src/components/Reactions/i18n/ru.json b/src/components/Reactions/i18n/ru.json new file mode 100644 index 00000000..dd01278d --- /dev/null +++ b/src/components/Reactions/i18n/ru.json @@ -0,0 +1,3 @@ +{ + "add-reaction": "ะ”ะพะฑะฐะฒะธั‚ัŒ ั€ะตะฐะบั†ะธัŽ" +} diff --git a/src/components/Reactions/index.ts b/src/components/Reactions/index.ts new file mode 100644 index 00000000..1ef7c4f1 --- /dev/null +++ b/src/components/Reactions/index.ts @@ -0,0 +1 @@ +export * from './Reactions'; From 70bf9f073c5592eb38e903ccdce6b5a49e8d50fe Mon Sep 17 00:00:00 2001 From: Gravity UI Bot <111915794+gravity-ui-bot@users.noreply.github.com> Date: Thu, 22 Aug 2024 17:48:15 +0300 Subject: [PATCH 3/3] chore(main): release 3.8.0 (#209) --- CHANGELOG.md | 16 ++++++++++++++++ package-lock.json | 4 ++-- package.json | 2 +- 3 files changed, 19 insertions(+), 3 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index b435d608..dc1eb7d7 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,21 @@ # Changelog +## [3.8.0](https://github.com/gravity-ui/components/compare/v3.7.0...v3.8.0) (2024-08-20) + + +### Features + +* **DefinitionList:** add mobile view ([#205](https://github.com/gravity-ui/components/issues/205)) ([b94ee82](https://github.com/gravity-ui/components/commit/b94ee824f69f7ed075167ff6d16263a30ca62d71)) +* **Reactions:** added new Reactions component ([#197](https://github.com/gravity-ui/components/issues/197)) ([be23d4e](https://github.com/gravity-ui/components/commit/be23d4e5949ac918ac945bbf829044520c7d0106)) + + +### Bug Fixes + +* add aria-label for DefinitionList help popover ([#208](https://github.com/gravity-ui/components/issues/208)) ([ad93fec](https://github.com/gravity-ui/components/commit/ad93fec84c88702c52aad69e9b6542690c66e4d4)) +* **DefinitionList:** should be semantic correct ([#211](https://github.com/gravity-ui/components/issues/211)) ([8dcc87c](https://github.com/gravity-ui/components/commit/8dcc87c034669553bf52166aa3e9f417f6a8a287)) +* **Notification:** correct `onClick` type for action ([#212](https://github.com/gravity-ui/components/issues/212)) ([b87bad1](https://github.com/gravity-ui/components/commit/b87bad101aeaf29cf1b97138e34d1a2f283b3688)) +* **PasswordInput:** fix incorrect props usage ([#210](https://github.com/gravity-ui/components/issues/210)) ([c740204](https://github.com/gravity-ui/components/commit/c7402042ea8aac95823f2a7f852429d8fd8ab957)) + ## [3.7.0](https://github.com/gravity-ui/components/compare/v3.6.2...v3.7.0) (2024-06-14) diff --git a/package-lock.json b/package-lock.json index c3742f0a..141ffe1d 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "@gravity-ui/components", - "version": "3.7.0", + "version": "3.8.0", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "@gravity-ui/components", - "version": "3.7.0", + "version": "3.8.0", "license": "MIT", "dependencies": { "@bem-react/classname": "^1.6.0", diff --git a/package.json b/package.json index 526f03c4..a6e532e1 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "@gravity-ui/components", - "version": "3.7.0", + "version": "3.8.0", "description": "", "license": "MIT", "main": "./build/cjs/index.js",