// Copyright 2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import React from 'react'; import { times, omit } from 'lodash'; import { storiesOf } from '@storybook/react'; import { action } from '@storybook/addon-actions'; import { boolean, date, select, text } from '@storybook/addon-knobs'; import { ConversationList, PropsType, RowType, Row } from './ConversationList'; import { MessageSearchResult } from './conversationList/MessageSearchResult'; import { PropsData as ConversationListItemPropsType, MessageStatuses, } from './conversationList/ConversationListItem'; import { ContactCheckboxDisabledReason } from './conversationList/ContactCheckbox'; import { getDefaultConversation } from '../test-both/helpers/getDefaultConversation'; import { setupI18n } from '../util/setupI18n'; import enMessages from '../../_locales/en/messages.json'; const i18n = setupI18n('en', enMessages); const story = storiesOf('Components/ConversationList', module); const defaultConversations: Array = [ getDefaultConversation({ id: 'fred-convo', title: 'Fred Willard', }), getDefaultConversation({ id: 'marc-convo', isSelected: true, unreadCount: 12, title: 'Marc Barraca', }), getDefaultConversation({ id: 'long-name-convo', title: 'Pablo Diego José Francisco de Paula Juan Nepomuceno María de los Remedios Cipriano de la Santísima Trinidad Ruiz y Picasso', }), getDefaultConversation(), ]; const createProps = (rows: ReadonlyArray): PropsType => ({ dimensions: { width: 300, height: 350, }, rowCount: rows.length, getRow: (index: number) => rows[index], shouldRecomputeRowHeights: false, i18n, onSelectConversation: action('onSelectConversation'), onClickArchiveButton: action('onClickArchiveButton'), onClickContactCheckbox: action('onClickContactCheckbox'), renderMessageSearchResult: (id: string) => ( ), showChooseGroupMembers: action('showChooseGroupMembers'), startNewConversationFromPhoneNumber: action( 'startNewConversationFromPhoneNumber' ), }); story.add('Archive button', () => ( )); story.add('Contact: note to self', () => ( )); story.add('Contact: direct', () => ( )); story.add('Contact: direct with short about', () => ( )); story.add('Contact: direct with long about', () => ( )); story.add('Contact: group', () => ( )); story.add('Contact checkboxes', () => ( )); story.add('Contact checkboxes: disabled', () => ( )); { const createConversation = ( overrideProps: Partial = {} ): ConversationListItemPropsType => ({ ...overrideProps, acceptedMessageRequest: boolean( 'acceptedMessageRequest', overrideProps.acceptedMessageRequest !== undefined ? overrideProps.acceptedMessageRequest : true ), isMe: boolean('isMe', overrideProps.isMe || false), avatarPath: text('avatarPath', overrideProps.avatarPath || ''), id: overrideProps.id || '', isSelected: boolean('isSelected', overrideProps.isSelected || false), title: text('title', overrideProps.title || 'Some Person'), name: overrideProps.name || 'Some Person', type: overrideProps.type || 'direct', markedUnread: boolean('markedUnread', overrideProps.markedUnread || false), lastMessage: overrideProps.lastMessage || { text: text('lastMessage.text', 'Hi there!'), status: select( 'status', MessageStatuses.reduce((m, s) => ({ ...m, [s]: s }), {}), 'read' ), deletedForEveryone: false, }, lastUpdated: date( 'lastUpdated', new Date(overrideProps.lastUpdated || Date.now() - 5 * 60 * 1000) ), sharedGroupNames: [], }); const renderConversation = ( overrideProps: Partial = {} ) => ( ); story.add('Conversation: name', () => renderConversation()); story.add('Conversation: name and avatar', () => renderConversation({ avatarPath: '/fixtures/kitten-1-64-64.jpg', }) ); story.add('Conversation: with yourself', () => renderConversation({ lastMessage: { text: 'Just a second', status: 'read', deletedForEveryone: false, }, name: 'Myself', title: 'Myself', isMe: true, }) ); story.add('Conversations: Message Statuses', () => ( ({ type: RowType.Conversation, conversation: createConversation({ lastMessage: { text: status, status, deletedForEveryone: false }, }), })) )} /> )); story.add('Conversation: Typing Status', () => renderConversation({ typingContact: { ...getDefaultConversation(), name: 'Someone Here', }, }) ); story.add('Conversation: With draft', () => renderConversation({ shouldShowDraft: true, draftPreview: "I'm in the middle of typing this...", }) ); story.add('Conversation: Deleted for everyone', () => renderConversation({ lastMessage: { deletedForEveryone: true }, }) ); story.add('Conversation: Message Request', () => renderConversation({ acceptedMessageRequest: false, lastMessage: { text: 'A Message', status: 'delivered', deletedForEveryone: false, }, }) ); story.add('Conversations: unread count', () => ( ({ type: RowType.Conversation, conversation: createConversation({ lastMessage: { text: 'Hey there!', status: 'delivered', deletedForEveryone: false, }, unreadCount, }), })) )} /> )); story.add('Conversation: marked unread', () => renderConversation({ markedUnread: true }) ); story.add('Conversation: Selected', () => renderConversation({ lastMessage: { text: 'Hey there!', status: 'read', deletedForEveryone: false, }, isSelected: true, }) ); story.add('Conversation: Emoji in Message', () => renderConversation({ lastMessage: { text: '🔥', status: 'read', deletedForEveryone: false, }, }) ); story.add('Conversation: Link in Message', () => renderConversation({ lastMessage: { text: 'Download at http://signal.org', status: 'read', deletedForEveryone: false, }, }) ); story.add('Conversation: long name', () => { const name = 'Long contact name. Esquire. The third. And stuff. And more! And more!'; return renderConversation({ name, title: name, }); }); story.add('Conversation: Long Message', () => { const messages = [ "Long line. This is a really really really long line. Really really long. Because that's just how it is", `Many lines. This is a many-line message. Line 2 is really exciting but it shouldn't be seen. Line three is even better. Line 4, well.`, ]; return ( ({ type: RowType.Conversation, conversation: createConversation({ lastMessage: { text: messageText, status: 'read', deletedForEveryone: false, }, }), })) )} /> ); }); story.add('Conversations: Various Times', () => { const pairs: Array<[number, string]> = [ [Date.now() - 5 * 60 * 60 * 1000, 'Five hours ago'], [Date.now() - 24 * 60 * 60 * 1000, 'One day ago'], [Date.now() - 7 * 24 * 60 * 60 * 1000, 'One week ago'], [Date.now() - 365 * 24 * 60 * 60 * 1000, 'One year ago'], ]; return ( ({ type: RowType.Conversation, conversation: createConversation({ lastUpdated, lastMessage: { text: messageText, status: 'read', deletedForEveryone: false, }, }), })) )} /> ); }); story.add('Conversation: Missing Date', () => { const row = { type: RowType.Conversation as const, conversation: omit(createConversation(), 'lastUpdated'), }; return ; }); story.add('Conversation: Missing Message', () => { const row = { type: RowType.Conversation as const, conversation: omit(createConversation(), 'lastMessage'), }; return ; }); story.add('Conversation: Missing Text', () => renderConversation({ lastMessage: { text: '', status: 'sent', deletedForEveryone: false, }, }) ); story.add('Conversation: Muted Conversation', () => renderConversation({ muteExpiresAt: Date.now() + 1000 * 60 * 60, }) ); story.add('Conversation: At Mention', () => renderConversation({ title: 'The Rebellion', type: 'group', lastMessage: { text: '@Leia Organa I know', status: 'read', deletedForEveryone: false, }, }) ); } story.add('Headers', () => ( )); story.add('Start new conversation', () => ( )); story.add('Search results loading skeleton', () => ( ({ type: RowType.SearchResultsLoadingFakeRow as const, })), ])} /> )); story.add('Kitchen sink', () => ( ));