diff --git a/CODEOWNERS b/CODEOWNERS index d76f2ca9b0..c60304b61d 100644 --- a/CODEOWNERS +++ b/CODEOWNERS @@ -2,6 +2,7 @@ /src/components/ActionTooltip @amje /src/components/Alert @IsaevAlexandr /src/components/ArrowToggle @Marginy605 +/src/components/Avatar @DakEnviy @DaffPunks #/src/components/Breadcrumbs /src/components/Button @amje /src/components/Card @Lunory @@ -30,8 +31,8 @@ /src/components/Radio @zamkovskaya /src/components/RadioButton @zamkovskaya /src/components/RadioGroup @zamkovskaya -/src/components/User @DaffPunks -/src/components/UserAvatar @DaffPunks +/src/components/User @DakEnviy @DaffPunks +/src/components/UserWrapper @DakEnviy @DaffPunks /src/components/Select @korvin89 /src/components/Sheet @mournfulCoroner /src/components/Skeleton @SeqviriouM diff --git a/src/components/Avatar/Avatar.scss b/src/components/Avatar/Avatar.scss new file mode 100644 index 0000000000..8e175533a4 --- /dev/null +++ b/src/components/Avatar/Avatar.scss @@ -0,0 +1,157 @@ +@use '../variables'; + +$block: '.#{variables.$ns-new}avatar'; + +#{$block} { + overflow: hidden; + display: inline-flex; + justify-content: center; + align-items: center; + border-radius: 50%; + background-color: var(--g-color-base-background); + + &__image { + display: block; + width: 100%; + height: 100%; + object-fit: cover; + } + + &__icon > svg { + display: block; + } + + &__text { + font-weight: 500; + } + + &_withBorder, + &_view_outlined { + position: relative; + + &::before, + &::after { + content: ''; + z-index: 1; + position: absolute; + inset: 0; + border-radius: 50%; + } + + &::before { + border: 3px solid var(--g-color-base-background); + } + + &::after { + border: 2px solid currentColor; + } + } + + &_size { + &_3xs { + width: 17px; + height: 17px; + } + + &_2xs { + width: 20px; + height: 20px; + } + + &_xs { + width: 24px; + height: 24px; + } + + &_s { + width: 28px; + height: 28px; + } + + &_m { + width: 32px; + height: 32px; + } + + &_l { + width: 42px; + height: 42px; + } + + &_xl { + width: 50px; + height: 50px; + } + + &_3xs, + &_2xs, + &_xs, + &_s { + #{$block}__text { + font-size: var(--g-text-caption-1-font-size); + line-height: var(--g-text-caption-1-line-height); + } + } + + &_m, + &_l { + #{$block}__text { + font-size: var(--g-text-body-1-font-size); + line-height: var(--g-text-body-1-line-height); + } + } + + &_xl { + #{$block}__text { + font-size: var(--g-text-body-2-font-size); + line-height: var(--g-text-body-2-line-height); + } + } + } + + &_theme { + &_normal { + &#{$block}_view_filled { + background-color: var(--g-color-base-misc-light); + color: unset; + + #{$block}__icon, + #{$block}__text { + color: var(--g-color-text-misc); + } + } + + &#{$block}_view_outlined { + background-color: var(--g-color-base-background); + color: var(--g-color-text-misc); + + #{$block}__icon, + #{$block}__text { + color: var(--g-color-text-misc); + } + } + } + + &_brand { + &#{$block}_view_filled { + background-color: var(--g-color-base-brand); + color: unset; + + #{$block}__icon, + #{$block}__text { + color: var(--g-color-text-brand-contrast); + } + } + + &#{$block}_view_outlined { + background-color: var(--g-color-base-background); + color: var(--g-color-text-brand); + + #{$block}__icon, + #{$block}__text { + color: var(--g-color-text-brand); + } + } + } + } +} diff --git a/src/components/Avatar/Avatar.tsx b/src/components/Avatar/Avatar.tsx new file mode 100644 index 0000000000..3099d40e2e --- /dev/null +++ b/src/components/Avatar/Avatar.tsx @@ -0,0 +1,87 @@ +import React from 'react'; + +import {blockNew} from '../utils/cn'; + +import {AvatarIcon} from './AvatarIcon'; +import {AvatarImage} from './AvatarImage'; +import {AvatarText} from './AvatarText'; +import type {AvatarProps} from './types/main'; + +import './Avatar.scss'; + +const b = blockNew('avatar'); + +export const Avatar = React.forwardRef((props, ref) => { + const { + size = 'm', + theme = 'normal', + view = 'filled', + backgroundColor, + borderColor, + title, + ariaAttributes, + className, + qa, + } = props; + + const style = React.useMemo( + () => ({backgroundColor, color: borderColor}), + [backgroundColor, borderColor], + ); + + const renderContent = () => { + if ('imgUrl' in props && props.imgUrl) { + return ( + + ); + } + + if ('icon' in props && props.icon) { + return ( + + ); + } + + if ('text' in props && props.text) { + return ( + + ); + } + + return null; + }; + + return ( +
+ {renderContent()} +
+ ); +}); + +Avatar.displayName = 'Avatar'; diff --git a/src/components/Avatar/AvatarIcon/AvatarIcon.tsx b/src/components/Avatar/AvatarIcon/AvatarIcon.tsx new file mode 100644 index 0000000000..33e4d12c03 --- /dev/null +++ b/src/components/Avatar/AvatarIcon/AvatarIcon.tsx @@ -0,0 +1,26 @@ +import React from 'react'; + +import {Icon} from '../../Icon'; +import type {AvatarSize} from '../types/common'; + +import type {AvatarIconProps} from './types'; + +const avatarSizeToIconSize: Record = { + '3xs': 11, + '2xs': 14, + xs: 14, + s: 16, + m: 16, + l: 20, + xl: 24, +}; + +export const AvatarIcon = ({icon, color, size, className}: AvatarIconProps) => { + const style = React.useMemo(() => ({color}), [color]); + + return ( +
+ +
+ ); +}; diff --git a/src/components/Avatar/AvatarIcon/index.ts b/src/components/Avatar/AvatarIcon/index.ts new file mode 100644 index 0000000000..9cb755cdfc --- /dev/null +++ b/src/components/Avatar/AvatarIcon/index.ts @@ -0,0 +1,2 @@ +export type {AvatarIconProps} from './types'; +export {AvatarIcon} from './AvatarIcon'; diff --git a/src/components/Avatar/AvatarIcon/types.ts b/src/components/Avatar/AvatarIcon/types.ts new file mode 100644 index 0000000000..fa8213b1ec --- /dev/null +++ b/src/components/Avatar/AvatarIcon/types.ts @@ -0,0 +1,7 @@ +import type {IconData} from '../../Icon'; +import type {AvatarCommonProps} from '../types/common'; + +export interface AvatarIconProps extends AvatarCommonProps { + icon: IconData; + color?: string; +} diff --git a/src/components/Avatar/AvatarImage/AvatarImage.tsx b/src/components/Avatar/AvatarImage/AvatarImage.tsx new file mode 100644 index 0000000000..303686b7cd --- /dev/null +++ b/src/components/Avatar/AvatarImage/AvatarImage.tsx @@ -0,0 +1,41 @@ +import React from 'react'; + +import {AVATAR_SIZES} from '../constants'; + +import type {AvatarImageProps} from './types'; + +export const AvatarImage = ({ + imgUrl, + fallbackImgUrl, + sizes, + srcSet, + alt, + loading, + size, + className, +}: AvatarImageProps) => { + const [isErrored, setIsErrored] = React.useState(false); + + const handleError = React.useCallback(() => { + setIsErrored(true); + }, []); + + // Reset error if `imgUrl` was changed to check it again + React.useEffect(() => { + setIsErrored(false); + }, [imgUrl]); + + return ( + {alt} + ); +}; diff --git a/src/components/Avatar/AvatarImage/index.ts b/src/components/Avatar/AvatarImage/index.ts new file mode 100644 index 0000000000..210d9ea26b --- /dev/null +++ b/src/components/Avatar/AvatarImage/index.ts @@ -0,0 +1,2 @@ +export type {AvatarImageProps} from './types'; +export {AvatarImage} from './AvatarImage'; diff --git a/src/components/Avatar/AvatarImage/types.ts b/src/components/Avatar/AvatarImage/types.ts new file mode 100644 index 0000000000..f0bc72d0cb --- /dev/null +++ b/src/components/Avatar/AvatarImage/types.ts @@ -0,0 +1,10 @@ +import type {AvatarCommonProps} from '../types/common'; + +export interface AvatarImageProps extends AvatarCommonProps { + imgUrl: string; + fallbackImgUrl?: string; + sizes?: string; + srcSet?: string; + alt?: string; + loading?: 'eager' | 'lazy'; +} diff --git a/src/components/Avatar/AvatarText/AvatarText.tsx b/src/components/Avatar/AvatarText/AvatarText.tsx new file mode 100644 index 0000000000..5e789fad16 --- /dev/null +++ b/src/components/Avatar/AvatarText/AvatarText.tsx @@ -0,0 +1,15 @@ +import React from 'react'; + +import type {AvatarTextProps} from './types'; +import {getAvatarDisplayText} from './utils'; + +export const AvatarText = ({text, color, size, className}: AvatarTextProps) => { + const style = React.useMemo(() => ({color}), [color]); + const displayText = React.useMemo(() => getAvatarDisplayText(text, size), [size, text]); + + return ( +
+ {displayText} +
+ ); +}; diff --git a/src/components/Avatar/AvatarText/index.ts b/src/components/Avatar/AvatarText/index.ts new file mode 100644 index 0000000000..e1daf58bce --- /dev/null +++ b/src/components/Avatar/AvatarText/index.ts @@ -0,0 +1,3 @@ +export type {AvatarTextProps} from './types'; +export {getAvatarDisplayText} from './utils'; +export {AvatarText} from './AvatarText'; diff --git a/src/components/Avatar/AvatarText/types.ts b/src/components/Avatar/AvatarText/types.ts new file mode 100644 index 0000000000..8b8cf06208 --- /dev/null +++ b/src/components/Avatar/AvatarText/types.ts @@ -0,0 +1,6 @@ +import type {AvatarCommonProps} from '../types/common'; + +export interface AvatarTextProps extends AvatarCommonProps { + text: string; + color?: string; +} diff --git a/src/components/Avatar/AvatarText/utils.ts b/src/components/Avatar/AvatarText/utils.ts new file mode 100644 index 0000000000..f5d27653e6 --- /dev/null +++ b/src/components/Avatar/AvatarText/utils.ts @@ -0,0 +1,13 @@ +import type {AvatarSize} from '../types/common'; + +export const getAvatarDisplayText = (text: string, size: AvatarSize) => { + if (size === '3xs') { + return text[0] || ''; + } + + const words = text.split(' '); + const result = + words.length > 1 ? [words[0][0], words[1][0]].filter(Boolean).join('') : text.slice(0, 2); + + return result.toUpperCase(); +}; diff --git a/src/components/Avatar/README.md b/src/components/Avatar/README.md new file mode 100644 index 0000000000..440b5699ce --- /dev/null +++ b/src/components/Avatar/README.md @@ -0,0 +1,211 @@ + + +# Avatar + + + +```tsx +import {Avatar} from '@gravity-ui/uikit'; +``` + +The component intended to render avatars. It has three basic types of avatars: image, icon and text (initials). All of these types have special props to configure behaviour and appearance. + +## Types + +### Image + +This component can be used to render avatars using images. Provide the image via `imgUrl` property. + + + +Also, you can provide `srcSet` property to load images of different sizes. + + + +Avatar component has `fallbackImgUrl` property which allows you to provide the image that is shown when an image loading error occurs via the link `imgUrl` (CSP error or no original image). + + + +### Icon + +This component can be used to render avatars using icons. Provide the icon via `icon` property like in `Icon` component. + + + +### Text + +This component can be used to render avatars using text. Provide the text via `text` property. The text renders like initials (2 first letters of words) or just 2 first letters of a single word. If the size is `3xs`, only the first letter will appear. + + + +## Appearance + +### Theme and view + +The Avatar component has predefined themes (`normal`, `brand`) and views (`filled`, `outlined`) + +Default theme: `normal` +Default view: `filled` + + + +### Custom colors + +Also, you can provide custom colors via props `backgroundColor`, `borderColor` and `color` (works only for icon and text avatars). These colors have a higher priority than the colors from the theme. + + + +### Size + +To control the size of the `Avatar` use the `size` property. The default size is `m`. Possible values: `3xs`, `2xs`, `xs`, `s`, `m`, `l`, `xl`. + + + +## Properties + +### Common + +| Name | Description | Type | Default | +| :-------------- | :-------------------------------------- | :---------------------------------------------: | :------: | +| size | Avatar size | `'3xs'` `'2xs'` `'xs'` `'s'` `'m'` `'l'` `'xl'` | `m` | +| theme | Avatar theme | `'normal'` `'brand'` | `normal` | +| view | Avatar view | `'filled'` `'outlined'` | `filled` | +| backgroundColor | Custom background color | `string` | | +| borderColor | Custom border color | `string` | | +| title | HTML `title` attributes | `string` | | +| ariaAttributes | HTML `aria-*` attributes | `React.AriaAttributes` | | +| className | Custom CSS class for root element | `string` | | +| qa | HTML `data-qa` attribute, used in tests | `string` | | + +### Image-specific + +| Name | Description | Type | Default | +| :------------- | :-------------------------------------- | :----------------: | :---------: | +| imgUrl | HTML img `src` attribute | `string` | | +| fallbackImgUrl | Fallback image, shown if error happened | `string` | | +| sizes | HTML img `sizes` attribute | `string` | | +| srcSet | HTML img `srcSet` attribute | `string` | | +| alt | HTML img `alt` attribute | `string` | props.title | +| loading | HTML img `loading` attribute | `'eager'` `'lazy'` | | + +### Icon-specific + +| Name | Description | Type | Default | +| :---- | :----------------- | :--------: | :-----: | +| icon | Source of SVG icon | `IconData` | | +| color | Custom icon color | `string` | | + +### Text-specific + +| Name | Description | Type | Default | +| :---- | :---------------- | :------: | :-----: | +| text | Avatar text | `string` | | +| color | Custom text color | `string` | | diff --git a/src/components/Avatar/__stories__/Avatar.stories.tsx b/src/components/Avatar/__stories__/Avatar.stories.tsx new file mode 100644 index 0000000000..0c2a3dd5bc --- /dev/null +++ b/src/components/Avatar/__stories__/Avatar.stories.tsx @@ -0,0 +1,285 @@ +import React from 'react'; + +import {faker} from '@faker-js/faker/locale/en'; +import {FaceRobot} from '@gravity-ui/icons'; +import {useArgs} from '@storybook/client-api'; +import type {Meta, StoryFn, StoryObj} from '@storybook/react'; + +import {Showcase} from '../../../demo/Showcase'; +import {ShowcaseItem} from '../../../demo/ShowcaseItem'; +import {Avatar} from '../Avatar'; + +import {getAvatarSrcSet} from './utils/getAvatarSrcSet'; + +const meta: Meta = { + title: 'Components/Data Display/Avatar', + component: Avatar, +}; + +export default meta; + +type Story = StoryObj; +type StoryFunc = StoryFn; + +const imgUrl = + 'https://avatars.mds.yandex.net/get-yapic/69015/enc-137b8b64288fa6fc5ec58c6b83aea00e7723c8fa5638c078312a1134d8ee32ac/islands-middle'; + +const randomAvatars = faker.helpers + .uniqueArray(() => faker.number.int({min: 20, max: 500}), 30) + .reduce( + (sizes, num) => ({ + ...sizes, + [num]: faker.image.urlLoremFlickr({ + category: 'cats', + width: num, + height: Math.round((num / 640) * 480), + }), + }), + {}, + ); + +const imageProps = { + imgUrl, +}; + +const iconProps = { + backgroundColor: 'var(--g-color-base-brand)', + icon: FaceRobot, + color: 'var(--g-color-text-brand-contrast)', +}; + +const textProps = { + backgroundColor: 'var(--g-color-base-generic-medium)', + text: 'Charles Darwin', + color: 'var(--g-color-text-primary)', +}; + +const BORDER_COLOR = 'var(--g-color-line-misc)'; + +export const Image: Story = { + args: { + imgUrl, + }, +}; + +export const ImageSrcSet: StoryFunc = (args) => { + const [, setArgs] = useArgs(); + + React.useEffect(() => { + if (args.size) { + setArgs({srcSet: getAvatarSrcSet(args.size, randomAvatars)}); + } + }, [args.size, setArgs]); + + return ; +}; + +ImageSrcSet.args = { + imgUrl: faker.image.urlLoremFlickr({category: 'cats'}), + size: 'xl', +}; + +export const ImageFallback: Story = { + args: { + imgUrl: imgUrl + '1', + fallbackImgUrl: imgUrl, + }, +}; + +export const Icon: Story = { + args: { + theme: 'brand', + icon: FaceRobot, + }, +}; + +export const Text: Story = { + args: { + theme: 'brand', + text: 'UI', + }, +}; + +export const TextInitials: Story = { + args: { + theme: 'brand', + text: 'Charles Darwin', + }, +}; + +export const WithBorder: Story = { + args: { + imgUrl, + borderColor: 'var(--g-color-line-misc)', + }, +}; + +export const AvatarShowcase: Story = { + name: 'Showcase', + render: () => { + return ( + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + ); + }, +}; diff --git a/src/components/UserAvatar/__stories__/getAvatarSrcSet.ts b/src/components/Avatar/__stories__/utils/getAvatarSrcSet.ts similarity index 71% rename from src/components/UserAvatar/__stories__/getAvatarSrcSet.ts rename to src/components/Avatar/__stories__/utils/getAvatarSrcSet.ts index 83549be241..78766fa85a 100644 --- a/src/components/UserAvatar/__stories__/getAvatarSrcSet.ts +++ b/src/components/Avatar/__stories__/utils/getAvatarSrcSet.ts @@ -1,19 +1,19 @@ -import {SIZES} from '../constants'; -import type {UserAvatarSize} from '../types'; +import {AVATAR_SIZES} from '../../constants'; +import type {AvatarSize} from '../../types/common'; import {getClosestNumber} from './getClosestNumber'; import {getSrcSet} from './getSrcSet'; import type {SrcSetType} from './types'; -export function getAvatarSrcSet( - size: UserAvatarSize, +export const getAvatarSrcSet = ( + size: AvatarSize, sizes: Record, {multipliers = [1, 2, 3, 4]} = {}, -) { +) => { const availableSizes = Object.keys(sizes) - .map((size) => Number(size)) + .map((item) => Number(item)) .sort((a, b) => a - b); - const baseSize = SIZES[size]; + const baseSize = AVATAR_SIZES[size]; const srcSet = multipliers.map((multiplier) => { const targetSize = multiplier * baseSize; const appropriateSize = getClosestNumber(targetSize, availableSizes); @@ -22,4 +22,4 @@ export function getAvatarSrcSet( }); return getSrcSet(srcSet as SrcSetType); -} +}; diff --git a/src/components/UserAvatar/__stories__/getClosestNumber.ts b/src/components/Avatar/__stories__/utils/getClosestNumber.ts similarity index 86% rename from src/components/UserAvatar/__stories__/getClosestNumber.ts rename to src/components/Avatar/__stories__/utils/getClosestNumber.ts index 2736a67b57..35a2226db9 100644 --- a/src/components/UserAvatar/__stories__/getClosestNumber.ts +++ b/src/components/Avatar/__stories__/utils/getClosestNumber.ts @@ -1,4 +1,4 @@ -export function getClosestNumber(targetNumber: number, availableNumbers: number[]) { +export const getClosestNumber = (targetNumber: number, availableNumbers: number[]) => { const stack = [...availableNumbers]; let previousNumber: number | undefined; @@ -23,4 +23,4 @@ export function getClosestNumber(targetNumber: number, availableNumbers: number[ } return previousNumber as number; -} +}; diff --git a/src/components/UserAvatar/__stories__/getSrcSet.ts b/src/components/Avatar/__stories__/utils/getSrcSet.ts similarity index 86% rename from src/components/UserAvatar/__stories__/getSrcSet.ts rename to src/components/Avatar/__stories__/utils/getSrcSet.ts index 268302c885..621a3c2c91 100644 --- a/src/components/UserAvatar/__stories__/getSrcSet.ts +++ b/src/components/Avatar/__stories__/utils/getSrcSet.ts @@ -1,6 +1,6 @@ import type {SrcSetType} from './types'; -export function getSrcSet(srcSet: SrcSetType) { +export const getSrcSet = (srcSet: SrcSetType) => { let srcSetString = ''; for (const item of srcSet) { @@ -16,4 +16,4 @@ export function getSrcSet(srcSet: SrcSetType) { } return srcSetString; -} +}; diff --git a/src/components/UserAvatar/__stories__/types.ts b/src/components/Avatar/__stories__/utils/types.ts similarity index 100% rename from src/components/UserAvatar/__stories__/types.ts rename to src/components/Avatar/__stories__/utils/types.ts diff --git a/src/components/Avatar/constants.ts b/src/components/Avatar/constants.ts new file mode 100644 index 0000000000..03cbd6b9a2 --- /dev/null +++ b/src/components/Avatar/constants.ts @@ -0,0 +1,11 @@ +import type {AvatarSize} from './types/common'; + +export const AVATAR_SIZES: Record = { + '3xs': 17, + '2xs': 20, + xs: 24, + s: 28, + m: 32, + l: 42, + xl: 50, +}; diff --git a/src/components/Avatar/index.ts b/src/components/Avatar/index.ts new file mode 100644 index 0000000000..6e01fd1865 --- /dev/null +++ b/src/components/Avatar/index.ts @@ -0,0 +1,5 @@ +export type {AvatarSize} from './types/common'; +export type {AvatarProps, AvatarTheme, AvatarView} from './types/main'; +export {AVATAR_SIZES} from './constants'; +export {Avatar} from './Avatar'; +export {getAvatarDisplayText} from './AvatarText'; diff --git a/src/components/Avatar/types/common.ts b/src/components/Avatar/types/common.ts new file mode 100644 index 0000000000..32aadd5f8b --- /dev/null +++ b/src/components/Avatar/types/common.ts @@ -0,0 +1,6 @@ +export type AvatarSize = '3xs' | '2xs' | 'xs' | 's' | 'm' | 'l' | 'xl'; + +export interface AvatarCommonProps { + size: AvatarSize; + className?: string; +} diff --git a/src/components/Avatar/types/main.ts b/src/components/Avatar/types/main.ts new file mode 100644 index 0000000000..1c11dac1a6 --- /dev/null +++ b/src/components/Avatar/types/main.ts @@ -0,0 +1,26 @@ +import type React from 'react'; + +import type {DistributiveOmit} from '../../../types/utils'; +import type {QAProps} from '../../types'; +import type {AvatarIconProps} from '../AvatarIcon'; +import type {AvatarImageProps} from '../AvatarImage'; +import type {AvatarTextProps} from '../AvatarText'; + +import type {AvatarCommonProps, AvatarSize} from './common'; + +export type AvatarTheme = 'normal' | 'brand'; +export type AvatarView = 'filled' | 'outlined'; + +interface AvatarBaseProps extends QAProps { + size?: AvatarSize; + theme?: AvatarTheme; + view?: AvatarView; + backgroundColor?: string; + borderColor?: string; + title?: string; + ariaAttributes?: React.AriaAttributes; + className?: string; +} + +export type AvatarProps = AvatarBaseProps & + DistributiveOmit; diff --git a/src/components/Icon/README.md b/src/components/Icon/README.md index 4fd8547b1e..0e636703aa 100644 --- a/src/components/Icon/README.md +++ b/src/components/Icon/README.md @@ -55,7 +55,7 @@ import CheckIcon from './check.svg'; | Name | Description | Type | Default | | :-------- | :-------------------------------------- | :---------------: | :--------------: | -| data | Source of SVG icon | `any` | | +| data | Source of SVG icon | `IconData` | | | width | `width` SVG attribute | `number` `string` | | | height | `height` SVG attribute | `number` `string` | | | size | Both `width` and `height` SVG attribute | `number` `string` | | diff --git a/src/components/Link/__stories__/LinkShowcase.tsx b/src/components/Link/__stories__/LinkShowcase.tsx index be2ff97cf6..3178d45272 100644 --- a/src/components/Link/__stories__/LinkShowcase.tsx +++ b/src/components/Link/__stories__/LinkShowcase.tsx @@ -1,7 +1,7 @@ import React from 'react'; import {Showcase} from '../../../demo/Showcase'; -import {ShowcaseGrid} from '../../../demo/ShowcaseGrid/ShowcaseGrid'; +import {ShowcaseGrid} from '../../../demo/ShowcaseGrid'; import {Link} from '../Link'; export function LinkShowcase() { diff --git a/src/components/Persona/Persona.tsx b/src/components/Persona/Persona.tsx index fb8c5b597d..f4abf64183 100644 --- a/src/components/Persona/Persona.tsx +++ b/src/components/Persona/Persona.tsx @@ -2,12 +2,12 @@ import React from 'react'; import {Envelope} from '@gravity-ui/icons'; -import {Icon} from '../Icon'; +import {Avatar} from '../Avatar'; import {PersonaWrap} from '../PersonaWrap'; import i18n from './i18n'; import type {PersonaProps} from './types'; -import {extractTextValue, extractTextView, getTwoLetters} from './utils'; +import {extractTextValue, extractTextView} from './utils'; export function Persona({ size = 's', @@ -31,10 +31,14 @@ export function Persona({ switch (type) { case 'person': - avatar = image ? {''} : {getTwoLetters(textValue)}; + avatar = image ? ( + + ) : ( + + ); break; case 'email': - avatar = ; + avatar = ; break; case 'empty': avatar = null; diff --git a/src/components/Persona/__tests__/Persona.test.tsx b/src/components/Persona/__tests__/Persona.test.tsx index 5f81b10a96..153e952875 100644 --- a/src/components/Persona/__tests__/Persona.test.tsx +++ b/src/components/Persona/__tests__/Persona.test.tsx @@ -3,10 +3,11 @@ import React from 'react'; import {queryByAttribute, render, screen} from '@testing-library/react'; import userEvent from '@testing-library/user-event'; +import {getAvatarDisplayText} from '../../Avatar'; import {Persona} from '../Persona'; import i18n from '../i18n'; import type {PersonaProps} from '../types'; -import {extractTextValue, getTwoLetters} from '../utils'; +import {extractTextValue} from '../utils'; const MOCKED_TEXT = 'text'; const MOCKED_TEXT_NODE_CONTENT_VALUE = 'Some view'; @@ -24,8 +25,8 @@ describe('Persona', () => { render(); const user = userEvent.setup(); const textValue = extractTextValue(text); - const twoLetters = getTwoLetters(textValue); - const personaNode = screen.getByText(twoLetters); + const displayText = getAvatarDisplayText(textValue, 's'); + const personaNode = screen.getByText(displayText); await user.click(personaNode); expect(onClick).toBeCalledWith(textValue); }, diff --git a/src/components/Persona/utils.ts b/src/components/Persona/utils.ts index 68c17dcc44..ef07849bff 100644 --- a/src/components/Persona/utils.ts +++ b/src/components/Persona/utils.ts @@ -1,5 +1,3 @@ -import get from 'lodash/get'; - import type {PersonaText} from './types'; export const extractTextValue = (text: PersonaText = '') => { @@ -17,8 +15,3 @@ export const extractTextView = (text: PersonaText = '') => { return text; }; - -export function getTwoLetters(text: string) { - const words = text.split(' '); - return [get(words, '[0][0]'), get(words, '[1][0]')].filter(Boolean).join(''); -} diff --git a/src/components/PersonaWrap/PersonaWrap.scss b/src/components/PersonaWrap/PersonaWrap.scss index 1ccbafe1d3..442a0b3284 100644 --- a/src/components/PersonaWrap/PersonaWrap.scss +++ b/src/components/PersonaWrap/PersonaWrap.scss @@ -2,17 +2,17 @@ @use '../variables'; $block: '.#{variables.$ns}persona'; +$blockAvatar: '.#{variables.$ns-new}avatar'; #{$block} { $avatarSize: 28px; $transitionDuration: 0.1s; $transitionTimingFunction: ease-in-out; - height: $avatarSize; - display: inline-flex; - align-items: center; position: relative; z-index: 0; + display: inline-flex; + height: $avatarSize; border-radius: 20px; transition-property: background-color; @@ -59,11 +59,10 @@ $block: '.#{variables.$ns}persona'; &__main { @include mixins.button-reset(); - height: $avatarSize; display: inline-flex; align-items: center; - padding-right: 6px; border-radius: inherit; + padding-right: 6px; #{$block}_closeable & { padding-right: 0; @@ -74,6 +73,7 @@ $block: '.#{variables.$ns}persona'; &:focus { outline: 2px solid var(--g-color-line-focus); } + &:focus:not(:focus-visible) { outline: 0; } @@ -81,19 +81,6 @@ $block: '.#{variables.$ns}persona'; } &__avatar { - width: $avatarSize; - height: $avatarSize; - display: flex; - align-items: center; - border-radius: 50%; - overflow: hidden; - justify-content: center; - background-color: var(--g-color-base-generic-accent); - - transition-property: background-color; - transition-duration: $transitionDuration; - transition-timing-function: $transitionTimingFunction; - #{$block}_size_n & { margin-right: 12px; } @@ -102,21 +89,21 @@ $block: '.#{variables.$ns}persona'; margin-right: 6px; } - #{$block}_clickable:hover & { - background-color: var(--g-color-base-generic); - } + #{$blockAvatar} { + background-color: var(--g-color-base-generic-accent); - span { - @include mixins.text-accent; - text-transform: uppercase; - font-size: 11px; - line-height: 11px; - } + transition-property: background-color; + transition-duration: $transitionDuration; + transition-timing-function: $transitionTimingFunction; - img { - width: $avatarSize; - height: $avatarSize; - object-fit: cover; + & #{$blockAvatar}__icon, + & #{$blockAvatar}__text { + color: var(--g-color-text-primary); + } + + #{$block}_clickable:hover & { + background-color: var(--g-color-base-generic); + } } } @@ -135,15 +122,14 @@ $block: '.#{variables.$ns}persona'; &__close { @include mixins.button-reset(); - height: $avatarSize; - width: 16px; + box-sizing: initial; display: inline-flex; - align-items: center; justify-content: center; + align-items: center; + width: 16px; cursor: pointer; padding-right: 6px; color: var(--g-color-text-secondary); - box-sizing: initial; transition-property: color; transition-duration: $transitionDuration; @@ -160,6 +146,7 @@ $block: '.#{variables.$ns}persona'; #{$block}__close:focus & { outline: 2px solid var(--g-color-line-focus); } + #{$block}__close:focus:not(:focus-visible) & { outline: 0; } diff --git a/src/components/User/README.md b/src/components/User/README.md index eb23d1f231..b3ea19b59a 100644 --- a/src/components/User/README.md +++ b/src/components/User/README.md @@ -1,14 +1,89 @@ -## User + -Display user avatar and his brief info. +# User -### PropTypes + -| Name | Type | Required | Default | Description | -| :---------- | :--------------- | :------- | :------ | :-------------------------------------------------------------------------------------------------------------- | -| imgUrl | `string` | | | Url of user avatar | -| className | `string` | | | Root element class name | -| name | `string` | | | User name (first line of info) | -| description | `string` | | | User additional data (second line of info) | -| size | `UserAvatarSize` | | 'm' | Component size. Supported values is: `xs`, `s`, `m`, `l`, `xl`. With a smallest size user info is not rendered. | -| qa | `string` | | | HTML `data-qa` attribute, used in tests | +```tsx +import {User} from '@gravity-ui/uikit'; +``` + +General component for displaying a user avatar with a info block. It uses [Avatar](https://gravity-ui.com/components/uikit/avatar) component to render the avatar and based on [UserWrapper](https://gravity-ui.com/components/uikit/user-wrapper). + +## Name and description + +`User` component has properties `name` and `description` to display a info block. + + + +## Size + +To control the size of the `User` use the `size` property. The default size is `m`. Possible values: `3xs`, `2xs`, `xs`, `s`, `m`, `l`, `xl`. + +This propeperty passes to the internal `Avatar` component too. + + + +## Properties + +| Name | Description | Type | Default | +| :------------- | :-------------------------------------- | :----------------------------------------------------------------------: | :-----: | +| avatar | User avatar | [AvatarProps](https://gravity-ui.com/components/uikit/avatar#properties) | | +| name | User name | `React.ReactNode` | | +| description | User description | `React.ReactNode` | | +| size | User block size | `'3xs'` `'2xs'` `'xs'` `'s'` `'m'` `'l'` `'xl'` | `m` | +| ariaAttributes | HTML `aria-*` attributes | `React.AriaAttributes` | | +| className | Custom CSS class for root element | `string` | | +| qa | HTML `data-qa` attribute, used in tests | `string` | | diff --git a/src/components/User/User.tsx b/src/components/User/User.tsx index c1be92ce4e..f94fb58809 100644 --- a/src/components/User/User.tsx +++ b/src/components/User/User.tsx @@ -1,36 +1,25 @@ import React from 'react'; -import {UserAvatar} from '../UserAvatar'; -import type {UserAvatarSize} from '../UserAvatar'; -import type {QAProps} from '../types'; -import {block} from '../utils/cn'; +import {Avatar} from '../Avatar'; +import {UserWrapper} from '../UserWrapper'; -import './User.scss'; +import type {UserProps} from './types'; -const b = block('user'); +export const User = React.forwardRef( + ({avatar, name, description, size, ariaAttributes, className, qa}, ref) => { + return ( + : null} + name={name} + description={description} + size={size} + className={className} + ariaAttributes={ariaAttributes} + qa={qa} + ref={ref} + /> + ); + }, +); -export interface UserProps extends QAProps { - name?: string; - description?: string; - imgUrl?: string; - size?: UserAvatarSize; - className?: string; -} - -export function User({name, description, imgUrl, size = 'm', className, qa}: UserProps) { - const compact = size === 'xs'; - - return ( -
- {imgUrl && } - {(name || description) && ( -
- {name && {name}} - {!compact && description && ( - {description} - )} -
- )} -
- ); -} +User.displayName = 'User'; diff --git a/src/components/User/__stories__/User.stories.tsx b/src/components/User/__stories__/User.stories.tsx index 1efe19ba94..3c3bec6252 100644 --- a/src/components/User/__stories__/User.stories.tsx +++ b/src/components/User/__stories__/User.stories.tsx @@ -1,18 +1,22 @@ -import React from 'react'; - -import type {Meta, StoryFn} from '@storybook/react'; +import type {Meta, StoryObj} from '@storybook/react'; import {User} from '../User'; -import type {UserProps} from '../User'; -export default { +const meta: Meta = { title: 'Components/Data Display/User', component: User, -} as Meta; +}; + +export default meta; + +type Story = StoryObj; -export const Default: StoryFn = (args) => ; -Default.args = { - name: 'Isaac', - description: 'user@gravity-ui.com', - imgUrl: '', +export const Default: Story = { + args: { + avatar: { + imgUrl: '', + }, + name: 'Isaac', + description: 'user@gravity-ui.com', + }, }; diff --git a/src/components/User/index.ts b/src/components/User/index.ts index f6b9f36c6e..50afdeac0b 100644 --- a/src/components/User/index.ts +++ b/src/components/User/index.ts @@ -1 +1,2 @@ -export * from './User'; +export type {UserProps} from './types'; +export {User} from './User'; diff --git a/src/components/User/types.ts b/src/components/User/types.ts new file mode 100644 index 0000000000..72ecbba110 --- /dev/null +++ b/src/components/User/types.ts @@ -0,0 +1,15 @@ +import type React from 'react'; + +import type {DistributiveOmit} from '../../types/utils'; +import type {AvatarProps} from '../Avatar'; +import type {UserWrapperSize} from '../UserWrapper'; +import type {QAProps} from '../types'; + +export interface UserProps extends QAProps { + avatar?: DistributiveOmit; + name?: React.ReactNode; + description?: React.ReactNode; + size?: UserWrapperSize; + ariaAttributes?: React.AriaAttributes; + className?: string; +} diff --git a/src/components/UserAvatar/README.md b/src/components/UserAvatar/README.md deleted file mode 100644 index 8fac13d836..0000000000 --- a/src/components/UserAvatar/README.md +++ /dev/null @@ -1,17 +0,0 @@ -## UserAvatar - -Component for displaying user avatar. - -### PropTypes - -| Name | Type | Required | Default | Description | -| :------------- | :--------------- | :------- | :------ | :------------------------------------------------------------------------------------------------------ | -| imgUrl | `string` | | | Link to image | -| fallbackImgUrl | `string` | | | Link to fallback image | -| size | `UserAvatarSize` | | 'm' | Component size. Possible values: `xs`, `s`, `m`, `l`, `xl` | -| srcSet | `string` | | | `srcSet` attribute of the image | -| sizes | `string` | | | `sizes` attribute of the image | -| title | `string` | | | Tooltip text on hover | -| className | `string` | | | Class name | -| loading | `eager \| lazy` | | | The loading attribute specifies whether a browser should load an image immediately or to defer loading. | -| qa | `string` | | | HTML `data-qa` attribute, used in tests | diff --git a/src/components/UserAvatar/UserAvatar.scss b/src/components/UserAvatar/UserAvatar.scss deleted file mode 100644 index 483bd10804..0000000000 --- a/src/components/UserAvatar/UserAvatar.scss +++ /dev/null @@ -1,47 +0,0 @@ -@use '../variables'; - -$block: '.#{variables.$ns}user-avatar'; - -#{$block} { - display: inline-block; - overflow: hidden; - - box-sizing: border-box; - - border-radius: 50%; - background-color: var(--g-color-base-background); - background-repeat: no-repeat; - background-position: center; - background-size: cover; - - &_size { - &_xs { - width: 24px; - height: 24px; - } - - &_s { - width: 28px; - height: 28px; - } - - &_m { - width: 32px; - height: 32px; - } - - &_l { - width: 42px; - height: 42px; - } - - &_xl { - width: 50px; - height: 50px; - } - } - - &__figure { - object-fit: cover; - } -} diff --git a/src/components/UserAvatar/UserAvatar.tsx b/src/components/UserAvatar/UserAvatar.tsx deleted file mode 100644 index a73f83325a..0000000000 --- a/src/components/UserAvatar/UserAvatar.tsx +++ /dev/null @@ -1,68 +0,0 @@ -import React from 'react'; - -import type {QAProps} from '../types'; -import {block} from '../utils/cn'; - -import {SIZES} from './constants'; -import type {UserAvatarSize} from './types'; - -import './UserAvatar.scss'; - -export interface UserAvatarProps extends QAProps { - imgUrl?: string; - fallbackImgUrl?: string; - size?: UserAvatarSize; - srcSet?: string; - sizes?: string; - title?: string; - className?: string; - loading?: 'eager' | 'lazy'; - /** @deprecated Use appropriate component, like `