// Copyright 2020-2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { Meta, Story } from '@storybook/react'; import * as React from 'react'; import { action } from '@storybook/addon-actions'; import { expect } from '@storybook/jest'; import { isBoolean } from 'lodash'; import { within, userEvent } from '@storybook/testing-library'; import type { AvatarColorType } from '../types/Colors'; import type { Props } from './Avatar'; import enMessages from '../../_locales/en/messages.json'; import { Avatar, AvatarBlur, AvatarSize } from './Avatar'; import { AvatarColors } from '../types/Colors'; import { HasStories } from '../types/Stories'; import { ThemeType } from '../types/Util'; import { getFakeBadge } from '../test-both/helpers/getFakeBadge'; import { setupI18n } from '../util/setupI18n'; const i18n = setupI18n('en', enMessages); const colorMap: Record = AvatarColors.reduce( (m, color) => ({ ...m, [color]: color, }), {} ); const conversationTypeMap: Record = { direct: 'direct', group: 'group', }; export default { title: 'Components/Avatar', component: Avatar, argTypes: { badge: { control: false, }, blur: { control: { type: 'radio' }, defaultValue: AvatarBlur.NoBlur, options: { NoBlur: AvatarBlur.NoBlur, BlurPicture: AvatarBlur.BlurPicture, BlurPictureWithClickToView: AvatarBlur.BlurPictureWithClickToView, }, }, color: { defaultValue: AvatarColors[0], options: colorMap, }, conversationType: { control: { type: 'radio' }, options: conversationTypeMap, }, onClick: { action: true }, size: { control: false, }, storyRing: { control: { type: 'radio' }, options: [undefined, ...Object.values(HasStories)], }, theme: { control: { type: 'radio' }, defaultValue: ThemeType.light, options: ThemeType, }, }, } as Meta; const createProps = (overrideProps: Partial = {}): Props => ({ acceptedMessageRequest: isBoolean(overrideProps.acceptedMessageRequest) ? overrideProps.acceptedMessageRequest : true, avatarPath: overrideProps.avatarPath || '', badge: overrideProps.badge, blur: overrideProps.blur || AvatarBlur.NoBlur, color: overrideProps.color || AvatarColors[0], conversationType: overrideProps.conversationType || 'direct', i18n, isMe: false, loading: Boolean(overrideProps.loading), name: overrideProps.name || '', noteToSelf: Boolean(overrideProps.noteToSelf), onClick: action('onClick'), onClickBadge: action('onClickBadge'), phoneNumber: overrideProps.phoneNumber || '', searchResult: Boolean(overrideProps.searchResult), sharedGroupNames: [], size: 80, title: overrideProps.title || '', theme: overrideProps.theme || ThemeType.light, }); const sizes = Object.values(AvatarSize).filter( x => typeof x === 'number' ) as Array; const Template: Story = args => ( <> {sizes.map(size => ( ))} ); const TemplateSingle: Story = args => ( ); export const Default = Template.bind({}); Default.args = createProps({ avatarPath: '/fixtures/giphy-GVNvOUpeYmI7e.gif', }); Default.play = async ({ args, canvasElement }) => { const canvas = within(canvasElement); const [avatar] = canvas.getAllByRole('button'); await userEvent.click(avatar); await expect(args.onClick).toHaveBeenCalled(); }; Default.story = { name: 'Avatar', }; export const WithBadge = Template.bind({}); WithBadge.args = createProps({ avatarPath: '/fixtures/kitten-3-64-64.jpg', badge: getFakeBadge(), }); WithBadge.story = { name: 'With badge', }; export const WideImage = Template.bind({}); WideImage.args = createProps({ avatarPath: '/fixtures/wide.jpg', }); WideImage.story = { name: 'Wide image', }; export const OneWordName = Template.bind({}); OneWordName.args = createProps({ title: 'John', }); OneWordName.story = { name: 'One-word Name', }; export const TwoWordName = Template.bind({}); TwoWordName.args = createProps({ title: 'John Smith', }); TwoWordName.story = { name: 'Two-word Name', }; export const WideInitials = Template.bind({}); WideInitials.args = createProps({ title: 'Walter White', }); WideInitials.story = { name: 'Wide initials', }; export const ThreeWordName = Template.bind({}); ThreeWordName.args = createProps({ title: 'Walter H. White', }); ThreeWordName.story = { name: 'Three-word name', }; export const NoteToSelf = Template.bind({}); NoteToSelf.args = createProps({ noteToSelf: true, }); NoteToSelf.story = { name: 'Note to Self', }; export const ContactIcon = Template.bind({}); ContactIcon.args = createProps(); export const GroupIcon = Template.bind({}); GroupIcon.args = createProps({ conversationType: 'group', }); export const SearchIcon = Template.bind({}); SearchIcon.args = createProps({ searchResult: true, }); export const Colors = (): JSX.Element => { const props = createProps(); return ( <> {AvatarColors.map(color => ( ))} ); }; export const BrokenColor = Template.bind({}); BrokenColor.args = createProps({ color: 'nope' as AvatarColorType, }); export const BrokenAvatar = Template.bind({}); BrokenAvatar.args = createProps({ avatarPath: 'badimage.png', }); export const BrokenAvatarForGroup = Template.bind({}); BrokenAvatarForGroup.args = createProps({ avatarPath: 'badimage.png', conversationType: 'group', }); BrokenAvatarForGroup.story = { name: 'Broken Avatar for Group', }; export const Loading = Template.bind({}); Loading.args = createProps({ loading: true, }); export const BlurredBasedOnProps = TemplateSingle.bind({}); BlurredBasedOnProps.args = createProps({ acceptedMessageRequest: false, avatarPath: '/fixtures/kitten-3-64-64.jpg', }); BlurredBasedOnProps.story = { name: 'Blurred based on props', }; export const ForceBlurred = TemplateSingle.bind({}); ForceBlurred.args = createProps({ avatarPath: '/fixtures/kitten-3-64-64.jpg', blur: AvatarBlur.BlurPicture, }); ForceBlurred.story = { name: 'Force-blurred', }; export const BlurredWithClickToView = TemplateSingle.bind({}); BlurredWithClickToView.args = createProps({ avatarPath: '/fixtures/kitten-3-64-64.jpg', blur: AvatarBlur.BlurPictureWithClickToView, }); BlurredWithClickToView.story = { name: 'Blurred with "click to view"', }; export const StoryUnread = TemplateSingle.bind({}); StoryUnread.args = createProps({ avatarPath: '/fixtures/kitten-3-64-64.jpg', storyRing: HasStories.Unread, }); StoryUnread.story = { name: 'Story: unread', }; export const StoryRead = TemplateSingle.bind({}); StoryRead.args = createProps({ avatarPath: '/fixtures/kitten-3-64-64.jpg', storyRing: HasStories.Read, }); StoryRead.story = { name: 'Story: read', };