// Copyright 2020 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import * as React from 'react'; import { isBoolean, noop } from 'lodash'; import { action } from '@storybook/addon-actions'; import { boolean, number, select, text } from '@storybook/addon-knobs'; import type { Meta, Story } from '@storybook/react'; import { SignalService } from '../../protobuf'; import { ConversationColors } from '../../types/Colors'; import { EmojiPicker } from '../emoji/EmojiPicker'; import type { AudioAttachmentProps } from './Message'; import type { Props } from './TimelineMessage'; import { TimelineMessage } from './TimelineMessage'; import { GiftBadgeStates, TextDirection } from './Message'; import { AUDIO_MP3, IMAGE_JPEG, IMAGE_PNG, IMAGE_WEBP, VIDEO_MP4, LONG_MESSAGE, stringToMIMEType, IMAGE_GIF, } from '../../types/MIME'; import { ReadStatus } from '../../messages/MessageReadStatus'; import { MessageAudio } from './MessageAudio'; import { computePeaks } from '../VoiceNotesPlaybackContext'; import { setupI18n } from '../../util/setupI18n'; import enMessages from '../../../_locales/en/messages.json'; import { pngUrl } from '../../storybook/Fixtures'; import { getDefaultConversation } from '../../test-both/helpers/getDefaultConversation'; import { WidthBreakpoint } from '../_util'; import { DAY, HOUR, MINUTE, SECOND } from '../../util/durations'; import { ContactFormType } from '../../types/EmbeddedContact'; import { fakeAttachment, fakeThumbnail, } from '../../test-both/helpers/fakeAttachment'; import { getFakeBadge } from '../../test-both/helpers/getFakeBadge'; import { ThemeType } from '../../types/Util'; import { UUID } from '../../types/UUID'; import { BadgeCategory } from '../../badges/BadgeCategory'; import { PaymentEventKind } from '../../types/Payment'; const i18n = setupI18n('en', enMessages); const quoteOptions = { none: undefined, basic: { conversationColor: ConversationColors[2], text: 'The quoted message', isFromMe: false, sentAt: Date.now(), authorId: 'some-id', authorTitle: 'Someone', referencedMessageNotFound: false, isViewOnce: false, isGiftBadge: false, }, }; export default { title: 'Components/Conversation/TimelineMessage', argTypes: { conversationType: { control: 'select', defaultValue: 'direct', options: ['direct', 'group'], }, quote: { control: 'select', defaultValue: undefined, mapping: quoteOptions, options: Object.keys(quoteOptions), }, }, } as Meta; const Template: Story> = args => { return renderBothDirections({ ...createProps(), conversationType: 'direct', quote: undefined, ...args, }); }; const messageIdToAudioUrl = { 'incompetech-com-Agnus-Dei-X': '/fixtures/incompetech-com-Agnus-Dei-X.mp3', }; function getJoyReaction() { return { emoji: '😂', from: getDefaultConversation({ id: '+14155552674', phoneNumber: '+14155552674', name: 'Amelia Briggs', title: 'Amelia', }), timestamp: Date.now() - 10, }; } const renderEmojiPicker: Props['renderEmojiPicker'] = ({ onClose, onPickEmoji, ref, }) => ( ); const renderReactionPicker: Props['renderReactionPicker'] = () =>
; /** * It doesn't handle consecutive playback * since that logic mostly lives in the audioPlayer duck */ function MessageAudioContainer({ played, ...props }: AudioAttachmentProps): JSX.Element { const [isActive, setIsActive] = React.useState(false); const [currentTime, setCurrentTime] = React.useState(0); const [playbackRate, setPlaybackRate] = React.useState(1); const [isPlaying, setIsPlaying] = React.useState(false); const [_played, setPlayed] = React.useState(played); const audioPlayer = React.useMemo(() => { const a = new Audio(); let onLoadedData: () => void = noop; a.addEventListener('timeupdate', () => { setCurrentTime(a.currentTime); }); a.addEventListener('ended', () => { setIsActive(false); }); a.addEventListener('loadeddata', () => onLoadedData()); function play(positionAsRatio?: number) { if (positionAsRatio !== undefined) { a.currentTime = positionAsRatio * a.duration; } void a.play(); } return { loadAndPlay(url: string, positionAsRatio: number) { onLoadedData = () => { play(positionAsRatio); }; a.src = url; }, play, pause() { a.pause(); }, set playbackRate(rate: number) { a.playbackRate = rate; }, set currentTime(value: number) { a.currentTime = value; }, get duration() { return a.duration; }, }; }, []); const handlePlayMessage = (id: string, positionAsRatio: number) => { if (!active) { audioPlayer.loadAndPlay( messageIdToAudioUrl[id as keyof typeof messageIdToAudioUrl], positionAsRatio ); setIsActive(true); setIsPlaying(true); setPlayed(true); } }; const setPlaybackRateAction = (rate: number) => { audioPlayer.playbackRate = rate; setPlaybackRate(rate); }; const setIsPlayingAction = (value: boolean) => { if (value) { audioPlayer.play(); } else { audioPlayer.pause(); } setIsPlaying(value); }; const setPosition = (value: number) => { audioPlayer.currentTime = value * audioPlayer.duration; setCurrentTime(audioPlayer.currentTime); }; const active = isActive ? { playing: isPlaying, playbackRate, currentTime, duration: audioPlayer.duration, } : undefined; return ( ); } const renderAudioAttachment: Props['renderAudioAttachment'] = props => ( ); const createProps = (overrideProps: Partial = {}): Props => ({ attachments: overrideProps.attachments, author: overrideProps.author || getDefaultConversation(), bodyRanges: overrideProps.bodyRanges, canReact: true, canReply: true, canDownload: true, canDeleteForEveryone: overrideProps.canDeleteForEveryone || false, canRetry: overrideProps.canRetry || false, canRetryDeleteForEveryone: overrideProps.canRetryDeleteForEveryone || false, checkForAccount: action('checkForAccount'), clearTargetedMessage: action('clearSelectedMessage'), containerElementRef: React.createRef(), containerWidthBreakpoint: WidthBreakpoint.Wide, conversationColor: overrideProps.conversationColor || select('conversationColor', ConversationColors, ConversationColors[0]), conversationTitle: overrideProps.conversationTitle || text('conversationTitle', 'Conversation Title'), conversationId: text('conversationId', overrideProps.conversationId || ''), conversationType: overrideProps.conversationType || 'direct', contact: overrideProps.contact, deletedForEveryone: overrideProps.deletedForEveryone, deleteMessages: action('deleteMessages'), deleteMessageForEveryone: action('deleteMessageForEveryone'), // disableMenu: overrideProps.disableMenu, disableScroll: overrideProps.disableScroll, direction: overrideProps.direction || 'incoming', showLightboxForViewOnceMedia: action('showLightboxForViewOnceMedia'), doubleCheckMissingQuoteReference: action('doubleCheckMissingQuoteReference'), expirationLength: number('expirationLength', overrideProps.expirationLength || 0) || undefined, expirationTimestamp: number('expirationTimestamp', overrideProps.expirationTimestamp || 0) || undefined, getPreferredBadge: overrideProps.getPreferredBadge || (() => undefined), giftBadge: overrideProps.giftBadge, i18n, platform: 'darwin', id: text('id', overrideProps.id || 'random-message-id'), // renderingContext: 'storybook', interactionMode: overrideProps.interactionMode || 'keyboard', isSticker: isBoolean(overrideProps.isSticker) ? overrideProps.isSticker : false, isBlocked: isBoolean(overrideProps.isBlocked) ? overrideProps.isBlocked : false, isMessageRequestAccepted: isBoolean(overrideProps.isMessageRequestAccepted) ? overrideProps.isMessageRequestAccepted : true, isSelected: isBoolean(overrideProps.isSelected) ? overrideProps.isSelected : false, isSelectMode: isBoolean(overrideProps.isSelectMode) ? overrideProps.isSelectMode : false, isTapToView: overrideProps.isTapToView, isTapToViewError: overrideProps.isTapToViewError, isTapToViewExpired: overrideProps.isTapToViewExpired, kickOffAttachmentDownload: action('kickOffAttachmentDownload'), markAttachmentAsCorrupted: action('markAttachmentAsCorrupted'), messageExpanded: action('messageExpanded'), showConversation: action('showConversation'), openGiftBadge: action('openGiftBadge'), previews: overrideProps.previews || [], quote: overrideProps.quote || undefined, reactions: overrideProps.reactions, reactToMessage: action('reactToMessage'), readStatus: overrideProps.readStatus === undefined ? ReadStatus.Read : overrideProps.readStatus, renderEmojiPicker, renderReactionPicker, renderAudioAttachment, saveAttachment: action('saveAttachment'), setQuoteByMessageId: action('setQuoteByMessageId'), retryMessageSend: action('retryMessageSend'), retryDeleteForEveryone: action('retryDeleteForEveryone'), scrollToQuotedMessage: action('scrollToQuotedMessage'), targetMessage: action('targetMessage'), toggleSelectMessage: overrideProps.toggleSelectMessage == null ? action('toggleSelectMessage') : overrideProps.toggleSelectMessage, shouldCollapseAbove: isBoolean(overrideProps.shouldCollapseAbove) ? overrideProps.shouldCollapseAbove : false, shouldCollapseBelow: isBoolean(overrideProps.shouldCollapseBelow) ? overrideProps.shouldCollapseBelow : false, shouldHideMetadata: isBoolean(overrideProps.shouldHideMetadata) ? overrideProps.shouldHideMetadata : false, pushPanelForConversation: action('pushPanelForConversation'), showContactModal: action('showContactModal'), showExpiredIncomingTapToViewToast: action( 'showExpiredIncomingTapToViewToast' ), showExpiredOutgoingTapToViewToast: action( 'showExpiredOutgoingTapToViewToast' ), toggleForwardMessagesModal: action('toggleForwardMessagesModal'), showLightbox: action('showLightbox'), startConversation: action('startConversation'), status: overrideProps.status || 'sent', text: overrideProps.text || text('text', ''), textDirection: overrideProps.textDirection || TextDirection.Default, textAttachment: overrideProps.textAttachment || { contentType: LONG_MESSAGE, size: 123, pending: boolean('textPending', false), }, theme: ThemeType.light, timestamp: number('timestamp', overrideProps.timestamp || Date.now()), viewStory: action('viewStory'), }); const renderMany = (propsArray: ReadonlyArray) => ( <> {propsArray.map((message, index) => ( ))} ); const renderThree = (props: Props) => renderMany([props, props, props]); const renderBothDirections = (props: Props) => ( <> {renderThree(props)} {renderThree({ ...props, author: { ...props.author, id: getDefaultConversation().id }, direction: 'outgoing', })} ); export const PlainMessage = Template.bind({}); PlainMessage.args = { text: 'Hello there from a pal! I am sending a long message so that it will wrap a bit, since I like that look.', }; export const PlainRtlMessage = Template.bind({}); PlainRtlMessage.args = { text: 'الأسانسير، علشان القطط ماتاكلش منها. وننساها، ونعود الى أوراقنا موصدين الباب بإحكام. نتنحنح، ونقول: البتاع. كلمة تدلّ على لا شيء، وعلى كلّ شيء. وهي مركز أبحاث شعبية كثيرة، تتعجّب من غرابتها والقومية المصرية الخاصة التي تعكسها، الى جانب الشيء الكثير من العفوية وحلاوة الروح. نعم، نحن قرأنا وسمعنا وعرفنا كل هذا. لكنه محلّ اهتمامنا اليوم لأسباب غير تلك الأسباب. كذلك، فإننا لعاقدون عزمنا على أن نتجاوز قضية الفصحى والعامية، وثنائية النخبة والرعاع، التي كثيراً ما ينحو نحوها الحديث عن الكلمة المذكورة. وفوق هذا كله، لسنا بصدد تفسير معاني "البتاع" كما تأتي في قصيدة الحاج أحمد فؤاد نجم، ولا التحذلق والتفذلك في الألغاز والأسرار المكنونة. هذا البتاع - أم هذه البت', textDirection: TextDirection.RightToLeft, }; PlainRtlMessage.story = { name: 'Plain RTL Message', }; export function EmojiMessages(): JSX.Element { return ( <>









); } export const Delivered = Template.bind({}); Delivered.args = { status: 'delivered', text: 'Hello there from a pal! I am sending a long message so that it will wrap a bit, since I like that look.', }; export const Read = Template.bind({}); Read.args = { status: 'read', text: 'Hello there from a pal! I am sending a long message so that it will wrap a bit, since I like that look.', }; export const Sending = Template.bind({}); Sending.args = { status: 'sending', text: 'Hello there from a pal! I am sending a long message so that it will wrap a bit, since I like that look.', }; export const Expiring = Template.bind({}); Expiring.args = { expirationLength: 30 * 1000, expirationTimestamp: Date.now() + 30 * 1000, text: 'Hello there from a pal! I am sending a long message so that it will wrap a bit, since I like that look.', }; export const WillExpireButStillSending = Template.bind({}); WillExpireButStillSending.args = { status: 'sending', expirationLength: 30 * 1000, text: 'We always show the timer if a message has an expiration length, even if unread or still sending.', }; WillExpireButStillSending.story = { name: 'Will expire but still sending', }; export const Pending = Template.bind({}); Pending.args = { text: 'Hello there from a pal! I am sending a long message so that it will wrap a bit, since I like that look.', textAttachment: { contentType: LONG_MESSAGE, size: 123, pending: true, }, }; export const LongBodyCanBeDownloaded = Template.bind({}); LongBodyCanBeDownloaded.args = { text: 'Hello there from a pal! I am sending a long message so that it will wrap a bit, since I like that look.', textAttachment: { contentType: LONG_MESSAGE, size: 123, pending: false, error: true, digest: 'abc', key: 'def', }, }; LongBodyCanBeDownloaded.story = { name: 'Long body can be downloaded', }; export const Recent = Template.bind({}); Recent.args = { text: 'Hello there from a pal!', timestamp: Date.now() - 30 * 60 * 1000, }; export const Older = Template.bind({}); Older.args = { text: 'Hello there from a pal!', timestamp: Date.now() - 180 * 24 * 60 * 60 * 1000, }; export const ReactionsWiderMessage = Template.bind({}); ReactionsWiderMessage.args = { text: 'Hello there from a pal!', timestamp: Date.now() - 180 * 24 * 60 * 60 * 1000, reactions: [ { emoji: '👍', from: getDefaultConversation({ isMe: true, id: '+14155552672', phoneNumber: '+14155552672', name: 'Me', title: 'Me', }), timestamp: Date.now() - 10, }, { emoji: '👍', from: getDefaultConversation({ id: '+14155552672', phoneNumber: '+14155552672', name: 'Amelia Briggs', title: 'Amelia', }), timestamp: Date.now() - 10, }, { emoji: '👍', from: getDefaultConversation({ id: '+14155552673', phoneNumber: '+14155552673', name: 'Amelia Briggs', title: 'Amelia', }), timestamp: Date.now() - 10, }, { emoji: '😂', from: getDefaultConversation({ id: '+14155552674', phoneNumber: '+14155552674', name: 'Amelia Briggs', title: 'Amelia', }), timestamp: Date.now() - 10, }, { emoji: '😡', from: getDefaultConversation({ id: '+14155552677', phoneNumber: '+14155552677', name: 'Amelia Briggs', title: 'Amelia', }), timestamp: Date.now() - 10, }, { emoji: '👎', from: getDefaultConversation({ id: '+14155552678', phoneNumber: '+14155552678', name: 'Amelia Briggs', title: 'Amelia', }), timestamp: Date.now() - 10, }, { emoji: '❤️', from: getDefaultConversation({ id: '+14155552679', phoneNumber: '+14155552679', name: 'Amelia Briggs', title: 'Amelia', }), timestamp: Date.now() - 10, }, ], }; ReactionsWiderMessage.story = { name: 'Reactions (wider message)', }; const joyReactions = Array.from({ length: 52 }, () => getJoyReaction()); export const ReactionsShortMessage = Template.bind({}); ReactionsShortMessage.args = { text: 'h', timestamp: Date.now(), reactions: [ ...joyReactions, { emoji: '👍', from: getDefaultConversation({ isMe: true, id: '+14155552672', phoneNumber: '+14155552672', name: 'Me', title: 'Me', }), timestamp: Date.now(), }, { emoji: '👍', from: getDefaultConversation({ id: '+14155552672', phoneNumber: '+14155552672', name: 'Amelia Briggs', title: 'Amelia', }), timestamp: Date.now(), }, { emoji: '👍', from: getDefaultConversation({ id: '+14155552673', phoneNumber: '+14155552673', name: 'Amelia Briggs', title: 'Amelia', }), timestamp: Date.now(), }, { emoji: '😡', from: getDefaultConversation({ id: '+14155552677', phoneNumber: '+14155552677', name: 'Amelia Briggs', title: 'Amelia', }), timestamp: Date.now(), }, { emoji: '👎', from: getDefaultConversation({ id: '+14155552678', phoneNumber: '+14155552678', name: 'Amelia Briggs', title: 'Amelia', }), timestamp: Date.now(), }, { emoji: '❤️', from: getDefaultConversation({ id: '+14155552679', phoneNumber: '+14155552679', name: 'Amelia Briggs', title: 'Amelia', }), timestamp: Date.now(), }, ], }; ReactionsShortMessage.story = { name: 'Reactions (short message)', }; export const AvatarInGroup = Template.bind({}); AvatarInGroup.args = { author: getDefaultConversation({ avatarPath: pngUrl }), conversationType: 'group', status: 'sent', text: 'Hello it is me, the saxophone.', }; AvatarInGroup.story = { name: 'Avatar in Group', }; export const BadgeInGroup = Template.bind({}); BadgeInGroup.args = { conversationType: 'group', getPreferredBadge: () => getFakeBadge(), status: 'sent', text: 'Hello it is me, the saxophone.', }; BadgeInGroup.story = { name: 'Badge in Group', }; export const Sticker = Template.bind({}); Sticker.args = { attachments: [ fakeAttachment({ url: '/fixtures/512x515-thumbs-up-lincoln.webp', fileName: '512x515-thumbs-up-lincoln.webp', contentType: IMAGE_WEBP, width: 128, height: 128, }), ], isSticker: true, status: 'sent', }; export const Quote = Template.bind({}); Quote.args = { quote: { text: 'hello my good friend', conversationColor: 'ultramarine', conversationTitle: 'Convo', isFromMe: false, sentAt: 0, authorId: '', authorTitle: 'Author', referencedMessageNotFound: false, isViewOnce: false, isGiftBadge: false, }, author: { id: '', isMe: false, title: 'Quoter Dude', sharedGroupNames: [], acceptedMessageRequest: true, badges: [], }, conversationType: 'group', }; export function Deleted(): JSX.Element { const propsSent = createProps({ conversationType: 'direct', deletedForEveryone: true, status: 'sent', }); const propsSending = createProps({ conversationType: 'direct', deletedForEveryone: true, status: 'sending', }); return ( <> {renderBothDirections(propsSent)} {renderBothDirections(propsSending)} ); } export const DeletedWithExpireTimer = Template.bind({}); DeletedWithExpireTimer.args = { timestamp: Date.now() - 60 * 1000, conversationType: 'group', deletedForEveryone: true, expirationLength: 5 * 60 * 1000, expirationTimestamp: Date.now() + 3 * 60 * 1000, status: 'sent', }; DeletedWithExpireTimer.story = { name: 'Deleted with expireTimer', }; export function DeletedWithError(): JSX.Element { const propsPartialError = createProps({ timestamp: Date.now() - 60 * 1000, // canDeleteForEveryone: true, conversationType: 'group', deletedForEveryone: true, status: 'partial-sent', direction: 'outgoing', }); const propsError = createProps({ timestamp: Date.now() - 60 * 1000, // canDeleteForEveryone: true, conversationType: 'group', deletedForEveryone: true, status: 'error', direction: 'outgoing', }); return ( <> {renderThree(propsPartialError)} {renderThree(propsError)} ); } DeletedWithError.story = { name: 'Deleted with error', }; export const CanDeleteForEveryone = Template.bind({}); CanDeleteForEveryone.args = { status: 'read', text: 'I hope you get this.', // canDeleteForEveryone: true, direction: 'outgoing', }; CanDeleteForEveryone.story = { name: 'Can delete for everyone', }; export const Error = Template.bind({}); Error.args = { status: 'error', // canRetry: true, text: 'I hope you get this.', }; export const Paused = Template.bind({}); Paused.args = { status: 'paused', text: 'I am up to a challenge', }; export const PartialSend = Template.bind({}); PartialSend.args = { status: 'partial-sent', text: 'I hope you get this.', }; export const LinkPreviewInGroup = Template.bind({}); LinkPreviewInGroup.args = { previews: [ { domain: 'signal.org', image: fakeAttachment({ contentType: IMAGE_PNG, fileName: 'the-sax.png', height: 240, url: pngUrl, width: 320, }), isStickerPack: false, title: 'Signal', description: 'Say "hello" to a different messaging experience. An unexpected focus on privacy, combined with all of the features you expect.', url: 'https://www.signal.org', date: new Date(2020, 2, 10).valueOf(), }, ], status: 'sent', text: 'Be sure to look at https://www.signal.org', conversationType: 'group', }; LinkPreviewInGroup.story = { name: 'Link Preview in Group', }; export const LinkPreviewWithQuote = Template.bind({}); LinkPreviewWithQuote.args = { quote: { conversationColor: ConversationColors[2], conversationTitle: getDefaultConversation().title, text: 'The quoted message', isFromMe: false, sentAt: Date.now(), authorId: 'some-id', authorTitle: 'Someone', referencedMessageNotFound: false, isViewOnce: false, isGiftBadge: false, }, previews: [ { domain: 'signal.org', image: fakeAttachment({ contentType: IMAGE_PNG, fileName: 'the-sax.png', height: 240, url: pngUrl, width: 320, }), isStickerPack: false, title: 'Signal', description: 'Say "hello" to a different messaging experience. An unexpected focus on privacy, combined with all of the features you expect.', url: 'https://www.signal.org', date: new Date(2020, 2, 10).valueOf(), }, ], status: 'sent', text: 'Be sure to look at https://www.signal.org', conversationType: 'group', }; LinkPreviewWithQuote.story = { name: 'Link Preview with Quote', }; export const LinkPreviewWithSmallImage = Template.bind({}); LinkPreviewWithSmallImage.args = { previews: [ { domain: 'signal.org', image: fakeAttachment({ contentType: IMAGE_PNG, fileName: 'the-sax.png', height: 50, url: pngUrl, width: 50, }), isStickerPack: false, title: 'Signal', description: 'Say "hello" to a different messaging experience. An unexpected focus on privacy, combined with all of the features you expect.', url: 'https://www.signal.org', date: new Date(2020, 2, 10).valueOf(), }, ], status: 'sent', text: 'Be sure to look at https://www.signal.org', }; LinkPreviewWithSmallImage.story = { name: 'Link Preview with Small Image', }; export const LinkPreviewWithoutImage = Template.bind({}); LinkPreviewWithoutImage.args = { previews: [ { domain: 'signal.org', isStickerPack: false, title: 'Signal', description: 'Say "hello" to a different messaging experience. An unexpected focus on privacy, combined with all of the features you expect.', url: 'https://www.signal.org', date: new Date(2020, 2, 10).valueOf(), }, ], status: 'sent', text: 'Be sure to look at https://www.signal.org', }; LinkPreviewWithoutImage.story = { name: 'Link Preview without Image', }; export const LinkPreviewWithNoDescription = Template.bind({}); LinkPreviewWithNoDescription.args = { previews: [ { domain: 'signal.org', isStickerPack: false, title: 'Signal', url: 'https://www.signal.org', date: Date.now(), }, ], status: 'sent', text: 'Be sure to look at https://www.signal.org', }; LinkPreviewWithNoDescription.story = { name: 'Link Preview with no description', }; export const LinkPreviewWithLongDescription = Template.bind({}); LinkPreviewWithLongDescription.args = { previews: [ { domain: 'signal.org', isStickerPack: false, title: 'Signal', description: Array(10) .fill( 'Say "hello" to a different messaging experience. An unexpected focus on privacy, combined with all of the features you expect.' ) .join(' '), url: 'https://www.signal.org', date: Date.now(), }, ], status: 'sent', text: 'Be sure to look at https://www.signal.org', }; LinkPreviewWithLongDescription.story = { name: 'Link Preview with long description', }; export const LinkPreviewWithSmallImageLongDescription = Template.bind({}); LinkPreviewWithSmallImageLongDescription.args = { previews: [ { domain: 'signal.org', image: fakeAttachment({ contentType: IMAGE_PNG, fileName: 'the-sax.png', height: 50, url: pngUrl, width: 50, }), isStickerPack: false, title: 'Signal', description: Array(10) .fill( 'Say "hello" to a different messaging experience. An unexpected focus on privacy, combined with all of the features you expect.' ) .join(' '), url: 'https://www.signal.org', date: Date.now(), }, ], status: 'sent', text: 'Be sure to look at https://www.signal.org', }; LinkPreviewWithSmallImageLongDescription.story = { name: 'Link Preview with small image, long description', }; export const LinkPreviewWithNoDate = Template.bind({}); LinkPreviewWithNoDate.args = { previews: [ { domain: 'signal.org', image: fakeAttachment({ contentType: IMAGE_PNG, fileName: 'the-sax.png', height: 240, url: pngUrl, width: 320, }), isStickerPack: false, title: 'Signal', description: 'Say "hello" to a different messaging experience. An unexpected focus on privacy, combined with all of the features you expect.', url: 'https://www.signal.org', }, ], status: 'sent', text: 'Be sure to look at https://www.signal.org', }; LinkPreviewWithNoDate.story = { name: 'Link Preview with no date', }; export const LinkPreviewWithTooNewADate = Template.bind({}); LinkPreviewWithTooNewADate.args = { previews: [ { domain: 'signal.org', image: fakeAttachment({ contentType: IMAGE_PNG, fileName: 'the-sax.png', height: 240, url: pngUrl, width: 320, }), isStickerPack: false, title: 'Signal', description: 'Say "hello" to a different messaging experience. An unexpected focus on privacy, combined with all of the features you expect.', url: 'https://www.signal.org', date: Date.now() + 3000000000, }, ], status: 'sent', text: 'Be sure to look at https://www.signal.org', }; LinkPreviewWithTooNewADate.story = { name: 'Link Preview with too new a date', }; export function Image(): JSX.Element { const darkImageProps = createProps({ attachments: [ fakeAttachment({ url: '/fixtures/tina-rolf-269345-unsplash.jpg', fileName: 'tina-rolf-269345-unsplash.jpg', contentType: IMAGE_JPEG, width: 128, height: 128, }), ], status: 'sent', }); const lightImageProps = createProps({ attachments: [ fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), ], status: 'sent', }); return ( <> {renderBothDirections(darkImageProps)} {renderBothDirections(lightImageProps)} ); } export const MultipleImages2 = Template.bind({}); MultipleImages2.args = { attachments: [ fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), ], status: 'sent', }; export const MultipleImages3 = Template.bind({}); MultipleImages3.args = { attachments: [ fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), ], status: 'sent', }; export const MultipleImages4 = Template.bind({}); MultipleImages4.args = { attachments: [ fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), ], status: 'sent', }; export const MultipleImages5 = Template.bind({}); MultipleImages5.args = { attachments: [ fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), fakeAttachment({ url: pngUrl, fileName: 'the-sax.png', contentType: IMAGE_PNG, height: 240, width: 320, }), ], status: 'sent', }; export const ImageWithCaption = Template.bind({}); ImageWithCaption.args = { attachments: [ fakeAttachment({ url: '/fixtures/tina-rolf-269345-unsplash.jpg', fileName: 'tina-rolf-269345-unsplash.jpg', contentType: IMAGE_JPEG, width: 128, height: 128, }), ], status: 'sent', text: 'This is my home.', }; ImageWithCaption.story = { name: 'Image with Caption', }; export const Gif = Template.bind({}); Gif.args = { attachments: [ fakeAttachment({ contentType: VIDEO_MP4, flags: SignalService.AttachmentPointer.Flags.GIF, fileName: 'cat-gif.mp4', url: '/fixtures/cat-gif.mp4', width: 400, height: 332, }), ], status: 'sent', }; Gif.story = { name: 'GIF', }; export const GifInAGroup = Template.bind({}); GifInAGroup.args = { attachments: [ fakeAttachment({ contentType: VIDEO_MP4, flags: SignalService.AttachmentPointer.Flags.GIF, fileName: 'cat-gif.mp4', url: '/fixtures/cat-gif.mp4', width: 400, height: 332, }), ], conversationType: 'group', status: 'sent', }; GifInAGroup.story = { name: 'GIF in a group', }; export const NotDownloadedGif = Template.bind({}); NotDownloadedGif.args = { attachments: [ fakeAttachment({ contentType: VIDEO_MP4, flags: SignalService.AttachmentPointer.Flags.GIF, fileName: 'cat-gif.mp4', fileSize: '188.61 KB', blurHash: 'LDA,FDBnm+I=p{tkIUI;~UkpELV]', width: 400, height: 332, }), ], status: 'sent', }; NotDownloadedGif.story = { name: 'Not Downloaded GIF', }; export const PendingGif = Template.bind({}); PendingGif.args = { attachments: [ fakeAttachment({ pending: true, contentType: VIDEO_MP4, flags: SignalService.AttachmentPointer.Flags.GIF, fileName: 'cat-gif.mp4', fileSize: '188.61 KB', blurHash: 'LDA,FDBnm+I=p{tkIUI;~UkpELV]', width: 400, height: 332, }), ], status: 'sent', }; PendingGif.story = { name: 'Pending GIF', }; export const _Audio = (): JSX.Element => { function Wrapper() { const [isPlayed, setIsPlayed] = React.useState(false); const messageProps = createProps({ id: 'incompetech-com-Agnus-Dei-X', attachments: [ fakeAttachment({ contentType: AUDIO_MP3, fileName: 'incompetech-com-Agnus-Dei-X.mp3', url: messageIdToAudioUrl['incompetech-com-Agnus-Dei-X'], path: 'somepath', }), ], ...(isPlayed ? { status: 'viewed', readStatus: ReadStatus.Viewed, } : { status: 'read', readStatus: ReadStatus.Read, }), }); return ( <> {renderBothDirections(messageProps)} ); } return ; }; export const LongAudio = Template.bind({}); LongAudio.args = { attachments: [ fakeAttachment({ contentType: AUDIO_MP3, fileName: 'long-audio.mp3', url: '/fixtures/long-audio.mp3', path: 'somepath', }), ], status: 'sent', }; export const AudioWithCaption = Template.bind({}); AudioWithCaption.args = { attachments: [ fakeAttachment({ contentType: AUDIO_MP3, fileName: 'incompetech-com-Agnus-Dei-X.mp3', url: '/fixtures/incompetech-com-Agnus-Dei-X.mp3', path: 'somepath', }), ], status: 'sent', text: 'This is what I sound like.', }; AudioWithCaption.story = { name: 'Audio with Caption', }; export const AudioWithNotDownloadedAttachment = Template.bind({}); AudioWithNotDownloadedAttachment.args = { attachments: [ fakeAttachment({ contentType: AUDIO_MP3, fileName: 'incompetech-com-Agnus-Dei-X.mp3', }), ], status: 'sent', }; AudioWithNotDownloadedAttachment.story = { name: 'Audio with Not Downloaded Attachment', }; export const AudioWithPendingAttachment = Template.bind({}); AudioWithPendingAttachment.args = { attachments: [ fakeAttachment({ contentType: AUDIO_MP3, fileName: 'incompetech-com-Agnus-Dei-X.mp3', pending: true, }), ], status: 'sent', }; AudioWithPendingAttachment.story = { name: 'Audio with Pending Attachment', }; export const OtherFileType = Template.bind({}); OtherFileType.args = { attachments: [ fakeAttachment({ contentType: stringToMIMEType('text/plain'), fileName: 'my-resume.txt', url: 'my-resume.txt', fileSize: '10MB', }), ], status: 'sent', }; export const OtherFileTypeWithCaption = Template.bind({}); OtherFileTypeWithCaption.args = { attachments: [ fakeAttachment({ contentType: stringToMIMEType('text/plain'), fileName: 'my-resume.txt', url: 'my-resume.txt', fileSize: '10MB', }), ], status: 'sent', text: 'This is what I have done.', }; OtherFileTypeWithCaption.story = { name: 'Other File Type with Caption', }; export const OtherFileTypeWithLongFilename = Template.bind({}); OtherFileTypeWithLongFilename.args = { attachments: [ fakeAttachment({ contentType: stringToMIMEType('text/plain'), fileName: 'INSERT-APP-NAME_INSERT-APP-APPLE-ID_AppStore_AppsGamesWatch.psd.zip', url: 'a2/a2334324darewer4234', fileSize: '10MB', }), ], status: 'sent', text: 'This is what I have done.', }; OtherFileTypeWithLongFilename.story = { name: 'Other File Type with Long Filename', }; export const TapToViewImage = Template.bind({}); TapToViewImage.args = { attachments: [ fakeAttachment({ url: '/fixtures/tina-rolf-269345-unsplash.jpg', fileName: 'tina-rolf-269345-unsplash.jpg', contentType: IMAGE_JPEG, width: 128, height: 128, }), ], isTapToView: true, status: 'sent', }; TapToViewImage.story = { name: 'TapToView Image', }; export const TapToViewVideo = Template.bind({}); TapToViewVideo.args = { attachments: [ fakeAttachment({ contentType: VIDEO_MP4, fileName: 'pixabay-Soap-Bubble-7141.mp4', height: 128, url: '/fixtures/pixabay-Soap-Bubble-7141.mp4', width: 128, }), ], isTapToView: true, status: 'sent', }; TapToViewVideo.story = { name: 'TapToView Video', }; export const TapToViewGif = Template.bind({}); TapToViewGif.args = { attachments: [ fakeAttachment({ contentType: VIDEO_MP4, flags: SignalService.AttachmentPointer.Flags.GIF, fileName: 'cat-gif.mp4', url: '/fixtures/cat-gif.mp4', width: 400, height: 332, }), ], isTapToView: true, status: 'sent', }; TapToViewGif.story = { name: 'TapToView GIF', }; export const TapToViewExpired = Template.bind({}); TapToViewExpired.args = { attachments: [ fakeAttachment({ url: '/fixtures/tina-rolf-269345-unsplash.jpg', fileName: 'tina-rolf-269345-unsplash.jpg', contentType: IMAGE_JPEG, width: 128, height: 128, }), ], isTapToView: true, isTapToViewExpired: true, status: 'sent', }; TapToViewExpired.story = { name: 'TapToView Expired', }; export const TapToViewError = Template.bind({}); TapToViewError.args = { attachments: [ fakeAttachment({ url: '/fixtures/tina-rolf-269345-unsplash.jpg', fileName: 'tina-rolf-269345-unsplash.jpg', contentType: IMAGE_JPEG, width: 128, height: 128, }), ], isTapToView: true, isTapToViewError: true, status: 'sent', }; TapToViewError.story = { name: 'TapToView Error', }; export const DangerousFileType = Template.bind({}); DangerousFileType.args = { attachments: [ fakeAttachment({ contentType: stringToMIMEType( 'application/vnd.microsoft.portable-executable' ), fileName: 'terrible.exe', url: 'terrible.exe', }), ], status: 'sent', }; export function Colors(): JSX.Element { return ( <> {ConversationColors.map(color => (
{renderBothDirections( createProps({ conversationColor: color, text: `Here is a preview of the chat color: ${color}. The color is visible to only you.`, }) )}
))} ); } export const Mentions = Template.bind({}); Mentions.args = { bodyRanges: [ { start: 0, length: 1, mentionUuid: 'zap', replacementText: 'Zapp Brannigan', conversationID: 'x', }, ], text: '\uFFFC This Is It. The Moment We Should Have Trained For.', }; Mentions.story = { name: '@Mentions', }; export function AllTheContextMenus(): JSX.Element { const props = createProps({ attachments: [ fakeAttachment({ url: '/fixtures/tina-rolf-269345-unsplash.jpg', fileName: 'tina-rolf-269345-unsplash.jpg', contentType: IMAGE_JPEG, width: 128, height: 128, }), ], status: 'partial-sent', canDeleteForEveryone: true, canRetry: true, canRetryDeleteForEveryone: true, }); return ; } AllTheContextMenus.story = { name: 'All the context menus', }; export const NotApprovedWithLinkPreview = Template.bind({}); NotApprovedWithLinkPreview.args = { previews: [ { domain: 'signal.org', image: fakeAttachment({ contentType: IMAGE_PNG, fileName: 'the-sax.png', height: 240, url: pngUrl, width: 320, }), isStickerPack: false, title: 'Signal', description: 'Say "hello" to a different messaging experience. An unexpected focus on privacy, combined with all of the features you expect.', url: 'https://www.signal.org', date: new Date(2020, 2, 10).valueOf(), }, ], status: 'sent', text: 'Be sure to look at https://www.signal.org', isMessageRequestAccepted: false, }; NotApprovedWithLinkPreview.story = { name: 'Not approved, with link preview', }; export function CustomColor(): JSX.Element { return ( <> {renderThree({ ...createProps({ text: 'Solid.' }), direction: 'outgoing', customColor: { start: { hue: 82, saturation: 35 }, }, })}
{renderThree({ ...createProps({ text: 'Gradient.' }), direction: 'outgoing', customColor: { deg: 192, start: { hue: 304, saturation: 85 }, end: { hue: 231, saturation: 76 }, }, })} ); } export const CollapsingTextOnlyDMs = (): JSX.Element => { const them = getDefaultConversation(); const me = getDefaultConversation({ isMe: true }); return renderMany([ createProps({ author: them, text: 'One', timestamp: Date.now() - 5 * MINUTE, }), createProps({ author: them, text: 'Two', timestamp: Date.now() - 4 * MINUTE, }), createProps({ author: them, text: 'Three', timestamp: Date.now() - 3 * MINUTE, }), createProps({ author: me, direction: 'outgoing', text: 'Four', timestamp: Date.now() - 2 * MINUTE, }), createProps({ text: 'Five', author: me, timestamp: Date.now() - MINUTE, direction: 'outgoing', }), createProps({ author: me, direction: 'outgoing', text: 'Six', }), ]); }; CollapsingTextOnlyDMs.story = { name: 'Collapsing text-only DMs', }; export const CollapsingTextOnlyGroupMessages = (): JSX.Element => { const author = getDefaultConversation(); return renderMany([ createProps({ author, conversationType: 'group', text: 'One', timestamp: Date.now() - 2 * MINUTE, }), createProps({ author, conversationType: 'group', text: 'Two', timestamp: Date.now() - MINUTE, }), createProps({ author, conversationType: 'group', text: 'Three', }), ]); }; CollapsingTextOnlyGroupMessages.story = { name: 'Collapsing text-only group messages', }; export const StoryReply = (): JSX.Element => { const conversation = getDefaultConversation(); return renderThree({ ...createProps({ direction: 'outgoing', text: 'Wow!' }), storyReplyContext: { authorTitle: conversation.firstName || conversation.title, conversationColor: ConversationColors[0], isFromMe: false, rawAttachment: fakeAttachment({ url: '/fixtures/snow.jpg', thumbnail: fakeThumbnail('/fixtures/snow.jpg'), }), text: 'Photo', }, }); }; StoryReply.story = { name: 'Story reply', }; export const StoryReplyYours = (): JSX.Element => { const conversation = getDefaultConversation(); return renderThree({ ...createProps({ direction: 'incoming', text: 'Wow!' }), storyReplyContext: { authorTitle: conversation.firstName || conversation.title, conversationColor: ConversationColors[0], isFromMe: true, rawAttachment: fakeAttachment({ url: '/fixtures/snow.jpg', thumbnail: fakeThumbnail('/fixtures/snow.jpg'), }), text: 'Photo', }, }); }; StoryReplyYours.story = { name: 'Story reply (yours)', }; export const StoryReplyEmoji = (): JSX.Element => { const conversation = getDefaultConversation(); return renderThree({ ...createProps({ direction: 'outgoing', text: 'Wow!' }), storyReplyContext: { authorTitle: conversation.firstName || conversation.title, conversationColor: ConversationColors[0], emoji: '💄', isFromMe: false, rawAttachment: fakeAttachment({ url: '/fixtures/snow.jpg', thumbnail: fakeThumbnail('/fixtures/snow.jpg'), }), text: 'Photo', }, }); }; StoryReplyEmoji.story = { name: 'Story reply (emoji)', }; const fullContact = { avatar: { avatar: fakeAttachment({ path: '/fixtures/giphy-GVNvOUpeYmI7e.gif', contentType: IMAGE_GIF, }), isProfile: true, }, email: [ { value: 'jerjor@fakemail.com', type: ContactFormType.HOME, }, ], name: { givenName: 'Jerry', familyName: 'Jordan', prefix: 'Dr.', suffix: 'Jr.', middleName: 'James', displayName: 'Jerry Jordan', }, number: [ { value: '555-444-2323', type: ContactFormType.HOME, }, ], }; export const EmbeddedContactFullContact = Template.bind({}); EmbeddedContactFullContact.args = { contact: fullContact, }; EmbeddedContactFullContact.story = { name: 'EmbeddedContact: Full Contact', }; export const EmbeddedContactWithSendMessage = Template.bind({}); EmbeddedContactWithSendMessage.args = { contact: { ...fullContact, firstNumber: fullContact.number[0].value, uuid: UUID.generate().toString(), }, direction: 'incoming', }; EmbeddedContactWithSendMessage.story = { name: 'EmbeddedContact: with Send Message', }; export const EmbeddedContactOnlyEmail = Template.bind({}); EmbeddedContactOnlyEmail.args = { contact: { email: fullContact.email, }, }; EmbeddedContactOnlyEmail.story = { name: 'EmbeddedContact: Only Email', }; export const EmbeddedContactGivenName = Template.bind({}); EmbeddedContactGivenName.args = { contact: { name: { givenName: 'Jerry', }, }, }; EmbeddedContactGivenName.story = { name: 'EmbeddedContact: Given Name', }; export const EmbeddedContactOrganization = Template.bind({}); EmbeddedContactOrganization.args = { contact: { organization: 'Company 5', }, }; EmbeddedContactOrganization.story = { name: 'EmbeddedContact: Organization', }; export const EmbeddedContactGivenFamilyName = Template.bind({}); EmbeddedContactGivenFamilyName.args = { contact: { name: { givenName: 'Jerry', familyName: 'FamilyName', }, }, }; EmbeddedContactGivenFamilyName.story = { name: 'EmbeddedContact: Given + Family Name', }; export const EmbeddedContactFamilyName = Template.bind({}); EmbeddedContactFamilyName.args = { contact: { name: { familyName: 'FamilyName', }, }, }; EmbeddedContactFamilyName.story = { name: 'EmbeddedContact: Family Name', }; export const EmbeddedContactLoadingAvatar = Template.bind({}); EmbeddedContactLoadingAvatar.args = { contact: { name: { displayName: 'Jerry Jordan', }, avatar: { avatar: fakeAttachment({ pending: true, contentType: IMAGE_GIF, }), isProfile: true, }, }, }; EmbeddedContactLoadingAvatar.story = { name: 'EmbeddedContact: Loading Avatar', }; export const GiftBadgeUnopened = Template.bind({}); GiftBadgeUnopened.args = { giftBadge: { id: 'GIFT', expiration: Date.now() + DAY * 30, level: 3, state: GiftBadgeStates.Unopened, }, }; GiftBadgeUnopened.story = { name: 'Gift Badge: Unopened', }; const getPreferredBadge = () => ({ category: BadgeCategory.Donor, descriptionTemplate: 'This is a description of the badge', id: 'GIFT', images: [ { transparent: { localPath: '/fixtures/orange-heart.svg', url: 'http://someplace', }, }, ], name: 'heart', }); export const GiftBadgeRedeemed30Days = Template.bind({}); GiftBadgeRedeemed30Days.args = { getPreferredBadge, giftBadge: { expiration: Date.now() + DAY * 30 + SECOND, id: 'GIFT', level: 3, state: GiftBadgeStates.Redeemed, }, }; GiftBadgeRedeemed30Days.story = { name: 'Gift Badge: Redeemed (30 days)', }; export const GiftBadgeRedeemed24Hours = Template.bind({}); GiftBadgeRedeemed24Hours.args = { getPreferredBadge, giftBadge: { expiration: Date.now() + DAY + SECOND, id: 'GIFT', level: 3, state: GiftBadgeStates.Redeemed, }, }; GiftBadgeRedeemed24Hours.story = { name: 'Gift Badge: Redeemed (24 hours)', }; export const GiftBadgeOpened60Minutes = Template.bind({}); GiftBadgeOpened60Minutes.args = { getPreferredBadge, giftBadge: { expiration: Date.now() + HOUR + SECOND, id: 'GIFT', level: 3, state: GiftBadgeStates.Opened, }, }; GiftBadgeOpened60Minutes.story = { name: 'Gift Badge: Opened (60 minutes)', }; export const GiftBadgeRedeemed1Minute = Template.bind({}); GiftBadgeRedeemed1Minute.args = { getPreferredBadge, giftBadge: { expiration: Date.now() + MINUTE + SECOND, id: 'GIFT', level: 3, state: GiftBadgeStates.Redeemed, }, }; GiftBadgeRedeemed1Minute.story = { name: 'Gift Badge: Redeemed (1 minute)', }; export const GiftBadgeOpenedExpired = Template.bind({}); GiftBadgeOpenedExpired.args = { getPreferredBadge, giftBadge: { expiration: Date.now(), id: 'GIFT', level: 3, state: GiftBadgeStates.Opened, }, }; GiftBadgeOpenedExpired.story = { name: 'Gift Badge: Opened (expired)', }; export const GiftBadgeMissingBadge = Template.bind({}); GiftBadgeMissingBadge.args = { getPreferredBadge: () => undefined, giftBadge: { expiration: Date.now() + MINUTE + SECOND, id: 'MISSING', level: 3, state: GiftBadgeStates.Redeemed, }, }; GiftBadgeMissingBadge.story = { name: 'Gift Badge: Missing Badge', }; export const PaymentNotification = Template.bind({}); PaymentNotification.args = { canReply: false, canReact: false, payment: { kind: PaymentEventKind.Notification, note: 'Hello there', }, }; function MultiSelectMessage() { const [selected, setSelected] = React.useState(false); return ( ); } export function MultiSelect(): JSX.Element { return ( <> ); } MultiSelect.args = { name: 'Multi Select', };