2023-01-03 19:55:46 +00:00
|
|
|
// Copyright 2020 Signal Messenger, LLC
|
2020-10-30 20:34:04 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2023-09-27 21:23:52 +00:00
|
|
|
import React, { useEffect, useState } from 'react';
|
2023-09-18 21:17:26 +00:00
|
|
|
import { times } from 'lodash';
|
|
|
|
import { action } from '@storybook/addon-actions';
|
2023-10-11 19:06:43 +00:00
|
|
|
import type { Meta } from '@storybook/react';
|
2023-09-27 21:23:52 +00:00
|
|
|
import { getDefaultConversation } from '../../test-both/helpers/getDefaultConversation';
|
2021-09-18 00:30:08 +00:00
|
|
|
import { setupI18n } from '../../util/setupI18n';
|
2020-08-26 18:40:23 +00:00
|
|
|
import enMessages from '../../../_locales/en/messages.json';
|
2023-09-27 21:23:52 +00:00
|
|
|
import type { TypingBubblePropsType } from './TypingBubble';
|
2021-10-26 19:15:33 +00:00
|
|
|
import { TypingBubble } from './TypingBubble';
|
2021-05-28 16:15:17 +00:00
|
|
|
import { AvatarColors } from '../../types/Colors';
|
2021-11-15 20:01:58 +00:00
|
|
|
import { getFakeBadge } from '../../test-both/helpers/getFakeBadge';
|
|
|
|
import { ThemeType } from '../../types/Util';
|
2020-08-26 18:40:23 +00:00
|
|
|
|
|
|
|
const i18n = setupI18n('en', enMessages);
|
|
|
|
|
2022-06-07 00:48:02 +00:00
|
|
|
export default {
|
|
|
|
title: 'Components/Conversation/TypingBubble',
|
2023-10-11 19:06:43 +00:00
|
|
|
argTypes: {},
|
|
|
|
args: {},
|
|
|
|
} satisfies Meta<TypingBubblePropsType>;
|
2020-08-26 18:40:23 +00:00
|
|
|
|
2023-09-27 21:23:52 +00:00
|
|
|
const CONTACTS = times(10, index => {
|
2023-09-18 21:17:26 +00:00
|
|
|
const letter = (index + 10).toString(36).toUpperCase();
|
2023-09-27 21:23:52 +00:00
|
|
|
return getDefaultConversation({
|
2023-09-18 21:17:26 +00:00
|
|
|
id: `contact-${index}`,
|
|
|
|
acceptedMessageRequest: false,
|
2024-07-11 19:44:09 +00:00
|
|
|
avatarUrl: '',
|
2023-09-27 21:23:52 +00:00
|
|
|
badges: [],
|
2023-09-18 21:17:26 +00:00
|
|
|
color: AvatarColors[index],
|
|
|
|
name: `${letter} ${letter}`,
|
|
|
|
phoneNumber: '(202) 555-0001',
|
|
|
|
profileName: `${letter} ${letter}`,
|
|
|
|
isMe: false,
|
|
|
|
sharedGroupNames: [],
|
|
|
|
title: `${letter} ${letter}`,
|
2023-09-27 21:23:52 +00:00
|
|
|
});
|
2023-09-18 21:17:26 +00:00
|
|
|
});
|
2023-09-27 21:23:52 +00:00
|
|
|
const CONTACT_IDS = CONTACTS.map(contact => contact.id);
|
|
|
|
const CONTACTS_BY_ID = new Map(CONTACTS.map(contact => [contact.id, contact]));
|
|
|
|
const getConversation = (id: string) =>
|
|
|
|
CONTACTS_BY_ID.get(id) || getDefaultConversation();
|
|
|
|
|
|
|
|
const CONTACTS_WITH_BADGES = CONTACTS.map(contact => {
|
|
|
|
return { ...contact, badges: [getFakeBadge()] };
|
|
|
|
});
|
|
|
|
const CONTACTS_WITH_BADGES_BY_ID = new Map(
|
|
|
|
CONTACTS_WITH_BADGES.map(contact => [contact.id, contact])
|
|
|
|
);
|
|
|
|
const getConversationWithBadges = (id: string) =>
|
|
|
|
CONTACTS_WITH_BADGES_BY_ID.get(id) || getDefaultConversation();
|
|
|
|
|
|
|
|
const getTypingContactIdTimestamps = (count: number) =>
|
2023-10-11 19:06:43 +00:00
|
|
|
Object.fromEntries(CONTACT_IDS.slice(0, count).map(id => [id, Date.now()]));
|
2023-09-18 21:17:26 +00:00
|
|
|
|
|
|
|
const createProps = (
|
|
|
|
overrideProps: Partial<TypingBubblePropsType> = {}
|
2023-09-27 21:23:52 +00:00
|
|
|
): TypingBubblePropsType => {
|
|
|
|
return {
|
|
|
|
typingContactIdTimestamps:
|
|
|
|
overrideProps.typingContactIdTimestamps ??
|
|
|
|
getTypingContactIdTimestamps(1),
|
|
|
|
lastItemAuthorId: '123',
|
|
|
|
lastItemTimestamp: undefined,
|
|
|
|
i18n,
|
|
|
|
conversationId: '123',
|
2023-10-11 19:06:43 +00:00
|
|
|
conversationType: overrideProps.conversationType ?? 'direct',
|
2023-09-27 21:23:52 +00:00
|
|
|
getConversation: overrideProps.getConversation || getConversation,
|
|
|
|
getPreferredBadge: badges =>
|
|
|
|
badges.length > 0 ? getFakeBadge() : undefined,
|
|
|
|
showContactModal: action('showContactModal'),
|
|
|
|
theme: ThemeType.light,
|
|
|
|
};
|
|
|
|
};
|
2020-08-26 18:40:23 +00:00
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
export function Direct(): JSX.Element {
|
2020-08-26 18:40:23 +00:00
|
|
|
const props = createProps();
|
|
|
|
|
|
|
|
return <TypingBubble {...props} />;
|
2022-11-18 00:45:19 +00:00
|
|
|
}
|
2020-08-26 18:40:23 +00:00
|
|
|
|
2023-09-27 21:23:52 +00:00
|
|
|
export function DirectStoppedTyping(): JSX.Element {
|
|
|
|
const props = createProps();
|
|
|
|
const [afterTimeoutProps, setAfterTimeoutProps] = useState({});
|
|
|
|
useEffect(() => {
|
|
|
|
setTimeout(
|
|
|
|
() =>
|
|
|
|
setAfterTimeoutProps({
|
|
|
|
typingContactIdTimestamps: {},
|
|
|
|
}),
|
|
|
|
500
|
|
|
|
);
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
return <TypingBubble {...props} {...afterTimeoutProps} />;
|
|
|
|
}
|
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
export function Group(): JSX.Element {
|
2020-08-26 18:40:23 +00:00
|
|
|
const props = createProps({ conversationType: 'group' });
|
|
|
|
return <TypingBubble {...props} />;
|
2022-11-18 00:45:19 +00:00
|
|
|
}
|
2021-11-15 20:01:58 +00:00
|
|
|
|
2023-10-06 21:30:36 +00:00
|
|
|
export function GroupStartsTyping(): JSX.Element {
|
|
|
|
const props = createProps({
|
|
|
|
conversationType: 'group',
|
|
|
|
typingContactIdTimestamps: {},
|
|
|
|
});
|
|
|
|
const [afterTimeoutProps, setAfterTimeoutProps] = useState({});
|
|
|
|
useEffect(() => {
|
|
|
|
setTimeout(
|
|
|
|
() =>
|
|
|
|
setAfterTimeoutProps({
|
|
|
|
typingContactIdTimestamps: getTypingContactIdTimestamps(1),
|
|
|
|
}),
|
|
|
|
500
|
|
|
|
);
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
return <TypingBubble {...props} {...afterTimeoutProps} />;
|
|
|
|
}
|
|
|
|
|
2023-09-27 21:23:52 +00:00
|
|
|
export function GroupStoppedTyping(): JSX.Element {
|
2023-09-18 21:17:26 +00:00
|
|
|
const props = createProps({
|
|
|
|
conversationType: 'group',
|
2023-09-27 21:23:52 +00:00
|
|
|
typingContactIdTimestamps: getTypingContactIdTimestamps(1),
|
2023-09-18 21:17:26 +00:00
|
|
|
});
|
2023-09-27 21:23:52 +00:00
|
|
|
const [afterTimeoutProps, setAfterTimeoutProps] = useState({});
|
|
|
|
useEffect(() => {
|
|
|
|
setTimeout(
|
|
|
|
() => setAfterTimeoutProps({ typingContactIdTimestamps: {} }),
|
|
|
|
500
|
|
|
|
);
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
return <TypingBubble {...props} {...afterTimeoutProps} />;
|
2023-09-18 21:17:26 +00:00
|
|
|
}
|
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
export function GroupWithBadge(): JSX.Element {
|
2021-11-15 20:01:58 +00:00
|
|
|
const props = createProps({
|
|
|
|
conversationType: 'group',
|
2023-09-27 21:23:52 +00:00
|
|
|
typingContactIdTimestamps: getTypingContactIdTimestamps(1),
|
|
|
|
getConversation: getConversationWithBadges,
|
2021-11-15 20:01:58 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
return <TypingBubble {...props} />;
|
2022-11-18 00:45:19 +00:00
|
|
|
}
|
2022-06-07 00:48:02 +00:00
|
|
|
|
2023-09-27 21:23:52 +00:00
|
|
|
export function GroupMultiTyping1To2(): JSX.Element {
|
|
|
|
const props = createProps({
|
|
|
|
conversationType: 'group',
|
|
|
|
typingContactIdTimestamps: getTypingContactIdTimestamps(1),
|
|
|
|
});
|
|
|
|
const [afterTimeoutProps, setAfterTimeoutProps] = useState({});
|
|
|
|
useEffect(() => {
|
|
|
|
setTimeout(
|
|
|
|
() =>
|
|
|
|
setAfterTimeoutProps({
|
|
|
|
typingContactIdTimestamps: getTypingContactIdTimestamps(2),
|
|
|
|
}),
|
|
|
|
500
|
|
|
|
);
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
return <TypingBubble {...props} {...afterTimeoutProps} />;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function GroupMultiTyping2Then1PersonStops(): JSX.Element {
|
2023-09-18 21:17:26 +00:00
|
|
|
const props = createProps({
|
|
|
|
conversationType: 'group',
|
2023-09-27 21:23:52 +00:00
|
|
|
typingContactIdTimestamps: getTypingContactIdTimestamps(2),
|
2023-09-18 21:17:26 +00:00
|
|
|
});
|
2023-09-27 21:23:52 +00:00
|
|
|
const [afterTimeoutProps, setAfterTimeoutProps] = useState({});
|
|
|
|
useEffect(() => {
|
|
|
|
setTimeout(
|
|
|
|
() =>
|
|
|
|
setAfterTimeoutProps({
|
|
|
|
typingContactIdTimestamps: getTypingContactIdTimestamps(1),
|
|
|
|
}),
|
|
|
|
500
|
|
|
|
);
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
return <TypingBubble {...props} {...afterTimeoutProps} />;
|
2023-09-18 21:17:26 +00:00
|
|
|
}
|
|
|
|
|
2023-09-27 21:23:52 +00:00
|
|
|
export function GroupMultiTyping3To4(): JSX.Element {
|
2023-09-18 21:17:26 +00:00
|
|
|
const props = createProps({
|
|
|
|
conversationType: 'group',
|
2023-09-27 21:23:52 +00:00
|
|
|
typingContactIdTimestamps: getTypingContactIdTimestamps(3),
|
2023-09-18 21:17:26 +00:00
|
|
|
});
|
2023-09-27 21:23:52 +00:00
|
|
|
const [afterTimeoutProps, setAfterTimeoutProps] = useState({});
|
|
|
|
useEffect(() => {
|
|
|
|
setTimeout(
|
|
|
|
() =>
|
|
|
|
setAfterTimeoutProps({
|
|
|
|
typingContactIdTimestamps: getTypingContactIdTimestamps(4),
|
|
|
|
}),
|
|
|
|
500
|
|
|
|
);
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
return <TypingBubble {...props} {...afterTimeoutProps} />;
|
2023-09-18 21:17:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function GroupMultiTyping10(): JSX.Element {
|
|
|
|
const props = createProps({
|
|
|
|
conversationType: 'group',
|
2023-09-27 21:23:52 +00:00
|
|
|
typingContactIdTimestamps: getTypingContactIdTimestamps(10),
|
2023-09-18 21:17:26 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
return <TypingBubble {...props} />;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function GroupMultiTypingWithBadges(): JSX.Element {
|
|
|
|
const props = createProps({
|
|
|
|
conversationType: 'group',
|
2023-09-27 21:23:52 +00:00
|
|
|
typingContactIdTimestamps: getTypingContactIdTimestamps(3),
|
|
|
|
getConversation: getConversationWithBadges,
|
2023-09-18 21:17:26 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
return <TypingBubble {...props} />;
|
|
|
|
}
|