| 
									
										
										
										
											2023-01-03 11:55:46 -08:00
										 |  |  | // Copyright 2021 Signal Messenger, LLC
 | 
					
						
							| 
									
										
										
										
											2021-10-20 16:46:42 -04:00
										 |  |  | // SPDX-License-Identifier: AGPL-3.0-only
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-29 10:42:26 -05:00
										 |  |  | import React, { useState } from 'react'; | 
					
						
							| 
									
										
										
										
											2021-10-20 16:46:42 -04:00
										 |  |  | import { action } from '@storybook/addon-actions'; | 
					
						
							| 
									
										
										
										
											2023-10-11 12:06:43 -07:00
										 |  |  | import type { Meta } from '@storybook/react'; | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { Props } from './MessageBodyReadMore'; | 
					
						
							|  |  |  | import { MessageBodyReadMore } from './MessageBodyReadMore'; | 
					
						
							| 
									
										
										
										
											2021-10-20 16:46:42 -04:00
										 |  |  | import { setupI18n } from '../../util/setupI18n'; | 
					
						
							|  |  |  | import enMessages from '../../../_locales/en/messages.json'; | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  | import type { HydratedBodyRangesType } from '../../types/BodyRange'; | 
					
						
							|  |  |  | import { BodyRange } from '../../types/BodyRange'; | 
					
						
							| 
									
										
										
										
											2023-08-10 18:43:33 +02:00
										 |  |  | import { generateAci } from '../../types/ServiceId'; | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  | import { RenderLocation } from './MessageTextRenderer'; | 
					
						
							| 
									
										
										
										
											2021-10-20 16:46:42 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | const i18n = setupI18n('en', enMessages); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-06 20:48:02 -04:00
										 |  |  | export default { | 
					
						
							|  |  |  |   title: 'Components/Conversation/MessageBodyReadMore', | 
					
						
							| 
									
										
										
										
											2023-10-11 12:06:43 -07:00
										 |  |  | } satisfies Meta<Props>; | 
					
						
							| 
									
										
										
										
											2021-10-20 16:46:42 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | const createProps = (overrideProps: Partial<Props> = {}): Props => ({ | 
					
						
							|  |  |  |   bodyRanges: overrideProps.bodyRanges, | 
					
						
							|  |  |  |   direction: 'incoming', | 
					
						
							| 
									
										
										
										
											2021-11-11 15:45:47 -08:00
										 |  |  |   displayLimit: overrideProps.displayLimit, | 
					
						
							| 
									
										
										
										
											2021-10-20 16:46:42 -04:00
										 |  |  |   i18n, | 
					
						
							| 
									
										
										
										
											2021-11-11 15:45:47 -08:00
										 |  |  |   id: 'some-id', | 
					
						
							| 
									
										
										
										
											2023-05-09 17:40:19 -07:00
										 |  |  |   isSpoilerExpanded: overrideProps.isSpoilerExpanded || {}, | 
					
						
							| 
									
										
										
										
											2021-11-11 15:45:47 -08:00
										 |  |  |   messageExpanded: action('messageExpanded'), | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |   onExpandSpoiler: overrideProps.onExpandSpoiler || action('onExpandSpoiler'), | 
					
						
							|  |  |  |   renderLocation: RenderLocation.Timeline, | 
					
						
							|  |  |  |   text: overrideProps.text || '', | 
					
						
							| 
									
										
										
										
											2021-10-20 16:46:42 -04:00
										 |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-29 10:42:26 -05:00
										 |  |  | function MessageBodyReadMoreTest({ | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |   bodyRanges, | 
					
						
							|  |  |  |   isSpoilerExpanded, | 
					
						
							|  |  |  |   onExpandSpoiler, | 
					
						
							| 
									
										
										
										
											2021-11-29 10:42:26 -05:00
										 |  |  |   text: messageBodyText, | 
					
						
							|  |  |  | }: { | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |   bodyRanges?: HydratedBodyRangesType; | 
					
						
							| 
									
										
										
										
											2023-05-09 17:40:19 -07:00
										 |  |  |   isSpoilerExpanded?: Record<number, boolean>; | 
					
						
							|  |  |  |   onExpandSpoiler?: (data: Record<number, boolean>) => void; | 
					
						
							| 
									
										
										
										
											2021-11-29 10:42:26 -05:00
										 |  |  |   text: string; | 
					
						
							|  |  |  | }): JSX.Element { | 
					
						
							|  |  |  |   const [displayLimit, setDisplayLimit] = useState<number | undefined>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ( | 
					
						
							|  |  |  |     <MessageBodyReadMore | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |       {...createProps({ | 
					
						
							|  |  |  |         bodyRanges, | 
					
						
							|  |  |  |         isSpoilerExpanded, | 
					
						
							|  |  |  |         onExpandSpoiler, | 
					
						
							|  |  |  |         text: messageBodyText, | 
					
						
							|  |  |  |       })} | 
					
						
							| 
									
										
										
										
											2021-11-29 10:42:26 -05:00
										 |  |  |       displayLimit={displayLimit} | 
					
						
							|  |  |  |       messageExpanded={(_, newDisplayLimit) => setDisplayLimit(newDisplayLimit)} | 
					
						
							|  |  |  |     /> | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | export function LongText100More(): JSX.Element { | 
					
						
							|  |  |  |   return ( | 
					
						
							|  |  |  |     <MessageBodyReadMoreTest | 
					
						
							|  |  |  |       text={`${'test '.repeat(160)}${'extra '.repeat(10)}`} | 
					
						
							|  |  |  |     /> | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-20 16:46:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | export function LotsOfCakeWithSomeCherriesOnTop(): JSX.Element { | 
					
						
							|  |  |  |   return ( | 
					
						
							|  |  |  |     <MessageBodyReadMoreTest text={`x${'🍰'.repeat(399)}${'🍒'.repeat(100)}`} /> | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-06-06 20:48:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | export function LeafyNotBuffered(): JSX.Element { | 
					
						
							|  |  |  |   return <MessageBodyReadMoreTest text={`x${'🌿'.repeat(450)}`} />; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-06-06 20:48:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  | export function LongTextWithMention(): JSX.Element { | 
					
						
							|  |  |  |   const bodyRanges = [ | 
					
						
							|  |  |  |     // This is right at boundary for better testing
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       start: 800, | 
					
						
							|  |  |  |       length: 1, | 
					
						
							| 
									
										
										
										
											2023-08-16 22:54:39 +02:00
										 |  |  |       mentionAci: generateAci(), | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |       conversationID: 'x', | 
					
						
							|  |  |  |       replacementText: 'Alice', | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |   ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const text = `${'x '.repeat(400)}\uFFFC woo!${'y '.repeat(100)}`; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return <MessageBodyReadMoreTest bodyRanges={bodyRanges} text={text} />; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function LongTextWithFormatting(): JSX.Element { | 
					
						
							|  |  |  |   const bodyRanges = [ | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       start: 0, | 
					
						
							|  |  |  |       length: 5, | 
					
						
							|  |  |  |       style: BodyRange.Style.ITALIC, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       start: 7, | 
					
						
							|  |  |  |       length: 3, | 
					
						
							|  |  |  |       style: BodyRange.Style.BOLD, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       start: 1019, | 
					
						
							|  |  |  |       length: 4, | 
					
						
							|  |  |  |       style: BodyRange.Style.BOLD, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       start: 1024, | 
					
						
							|  |  |  |       length: 6, | 
					
						
							|  |  |  |       style: BodyRange.Style.ITALIC, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |   ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const text = `ready? set... g${'o'.repeat(1000)}al! bold italic`; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return <MessageBodyReadMoreTest bodyRanges={bodyRanges} text={text} />; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function LongTextMostlySpoiler(): JSX.Element { | 
					
						
							| 
									
										
										
										
											2023-05-09 17:40:19 -07:00
										 |  |  |   const [isSpoilerExpanded, setIsSpoilerExpanded] = React.useState({}); | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |   const bodyRanges = [ | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       start: 7, | 
					
						
							|  |  |  |       length: 1010, | 
					
						
							|  |  |  |       style: BodyRange.Style.SPOILER, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |   ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const text = `ready? set... g${'o'.repeat(1000)}al! bold italic`; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ( | 
					
						
							|  |  |  |     <MessageBodyReadMoreTest | 
					
						
							|  |  |  |       bodyRanges={bodyRanges} | 
					
						
							|  |  |  |       text={text} | 
					
						
							|  |  |  |       isSpoilerExpanded={isSpoilerExpanded} | 
					
						
							| 
									
										
										
										
											2023-05-09 17:40:19 -07:00
										 |  |  |       onExpandSpoiler={data => setIsSpoilerExpanded(data)} | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |     /> | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | export function Links(): JSX.Element { | 
					
						
							|  |  |  |   return ( | 
					
						
							|  |  |  |     <MessageBodyReadMoreTest | 
					
						
							|  |  |  |       text={`${'test '.repeat(176)}https://www.signal.org`} | 
					
						
							|  |  |  |     /> | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-20 16:46:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | export function ExcessiveAmountsOfCake(): JSX.Element { | 
					
						
							|  |  |  |   return <MessageBodyReadMoreTest text={`x${'🍰'.repeat(20000)}`} />; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-20 16:46:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | export function LongText(): JSX.Element { | 
					
						
							|  |  |  |   return ( | 
					
						
							|  |  |  |     <MessageBodyReadMoreTest | 
					
						
							|  |  |  |       text={`
 | 
					
						
							| 
									
										
										
										
											2021-10-20 16:46:42 -04:00
										 |  |  |       SCENE I. Rome. A street. | 
					
						
							|  |  |  |       Enter FLAVIUS, MARULLUS, and certain Commoners | 
					
						
							|  |  |  |       FLAVIUS | 
					
						
							|  |  |  |       Hence! home, you idle creatures get you home: | 
					
						
							|  |  |  |       Is this a holiday? what! know you not, | 
					
						
							|  |  |  |       Being mechanical, you ought not walk | 
					
						
							|  |  |  |       Upon a labouring day without the sign | 
					
						
							|  |  |  |       Of your profession? Speak, what trade art thou? | 
					
						
							|  |  |  |       First Commoner | 
					
						
							|  |  |  |       Why, sir, a carpenter. | 
					
						
							|  |  |  |       MARULLUS | 
					
						
							|  |  |  |       Where is thy leather apron and thy rule? | 
					
						
							|  |  |  |       What dost thou with thy best apparel on? | 
					
						
							|  |  |  |       You, sir, what trade are you? | 
					
						
							|  |  |  |       Second Commoner | 
					
						
							|  |  |  |       Truly, sir, in respect of a fine workman, I am but, | 
					
						
							|  |  |  |       as you would say, a cobbler. | 
					
						
							|  |  |  |       MARULLUS | 
					
						
							|  |  |  |       But what trade art thou? answer me directly. | 
					
						
							|  |  |  |       Second Commoner | 
					
						
							|  |  |  |       A trade, sir, that, I hope, I may use with a safe | 
					
						
							|  |  |  |       conscience; which is, indeed, sir, a mender of bad soles. | 
					
						
							|  |  |  |       MARULLUS | 
					
						
							|  |  |  |       What trade, thou knave? thou naughty knave, what trade? | 
					
						
							|  |  |  |       Second Commoner | 
					
						
							|  |  |  |       Nay, I beseech you, sir, be not out with me: yet, | 
					
						
							|  |  |  |       if you be out, sir, I can mend you. | 
					
						
							|  |  |  |       MARULLUS | 
					
						
							|  |  |  |       What meanest thou by that? mend me, thou saucy fellow! | 
					
						
							|  |  |  |       Second Commoner | 
					
						
							|  |  |  |       Why, sir, cobble you. | 
					
						
							|  |  |  |       FLAVIUS | 
					
						
							|  |  |  |       Thou art a cobbler, art thou? | 
					
						
							|  |  |  |       Second Commoner | 
					
						
							|  |  |  |       Truly, sir, all that I live by is with the awl: I | 
					
						
							|  |  |  |       meddle with no tradesman's matters, nor women's | 
					
						
							|  |  |  |       matters, but with awl. I am, indeed, sir, a surgeon | 
					
						
							|  |  |  |       to old shoes; when they are in great danger, I | 
					
						
							|  |  |  |       recover them. As proper men as ever trod upon | 
					
						
							|  |  |  |       neat's leather have gone upon my handiwork. | 
					
						
							|  |  |  |       FLAVIUS | 
					
						
							|  |  |  |       But wherefore art not in thy shop today? | 
					
						
							|  |  |  |       Why dost thou lead these men about the streets? | 
					
						
							|  |  |  |       Second Commoner | 
					
						
							|  |  |  |       Truly, sir, to wear out their shoes, to get myself | 
					
						
							|  |  |  |       into more work. But, indeed, sir, we make holiday, | 
					
						
							|  |  |  |       to see Caesar and to rejoice in his triumph. | 
					
						
							|  |  |  |       MARULLUS | 
					
						
							|  |  |  |       Wherefore rejoice? What conquest brings he home? | 
					
						
							|  |  |  |       What tributaries follow him to Rome, | 
					
						
							|  |  |  |       To grace in captive bonds his chariot-wheels? | 
					
						
							|  |  |  |       You blocks, you stones, you worse than senseless things! | 
					
						
							|  |  |  |       O you hard hearts, you cruel men of Rome, | 
					
						
							|  |  |  |       Knew you not Pompey? Many a time and oft | 
					
						
							|  |  |  |       Have you climb'd up to walls and battlements, | 
					
						
							|  |  |  |       To towers and windows, yea, to chimney-tops, | 
					
						
							|  |  |  |       Your infants in your arms, and there have sat | 
					
						
							|  |  |  |       The livelong day, with patient expectation, | 
					
						
							|  |  |  |       To see great Pompey pass the streets of Rome: | 
					
						
							|  |  |  |       And when you saw his chariot but appear, | 
					
						
							|  |  |  |       Have you not made an universal shout, | 
					
						
							|  |  |  |       That Tiber trembled underneath her banks, | 
					
						
							|  |  |  |       To hear the replication of your sounds | 
					
						
							|  |  |  |       Made in her concave shores? | 
					
						
							|  |  |  |       And do you now put on your best attire? | 
					
						
							|  |  |  |       And do you now cull out a holiday? | 
					
						
							|  |  |  |       And do you now strew flowers in his way | 
					
						
							|  |  |  |       That comes in triumph over Pompey's blood? Be gone! | 
					
						
							|  |  |  |       Run to your houses, fall upon your knees, | 
					
						
							|  |  |  |       Pray to the gods to intermit the plague | 
					
						
							|  |  |  |       That needs must light on this ingratitude. | 
					
						
							|  |  |  |       FLAVIUS | 
					
						
							|  |  |  |       Go, go, good countrymen, and, for this fault, | 
					
						
							|  |  |  |       Assemble all the poor men of your sort; | 
					
						
							|  |  |  |       Draw them to Tiber banks, and weep your tears | 
					
						
							|  |  |  |       Into the channel, till the lowest stream | 
					
						
							|  |  |  |       Do kiss the most exalted shores of all. | 
					
						
							|  |  |  |       Exeunt all the Commoners | 
					
						
							|  |  |  |       See whether their basest metal be not moved; | 
					
						
							|  |  |  |       They vanish tongue-tied in their guiltiness. | 
					
						
							|  |  |  |       Go you down that way towards the Capitol; | 
					
						
							|  |  |  |       This way will I | 
					
						
							|  |  |  |       disrobe the images, | 
					
						
							|  |  |  |       If you do find them deck'd with ceremonies. | 
					
						
							|  |  |  |       MARULLUS | 
					
						
							|  |  |  |       May we do so? | 
					
						
							|  |  |  |       You know it is the feast of Lupercal. | 
					
						
							|  |  |  |       FLAVIUS | 
					
						
							|  |  |  |       It is no matter; let no images | 
					
						
							|  |  |  |       Be hung with Caesar's trophies. I'll about, | 
					
						
							|  |  |  |       And drive away the vulgar from the streets: | 
					
						
							|  |  |  |       So do you too, where you perceive them thick. | 
					
						
							|  |  |  |       These growing feathers pluck'd from Caesar's wing | 
					
						
							|  |  |  |       Will make him fly an ordinary pitch, | 
					
						
							|  |  |  |       Who else would soar above the view of men | 
					
						
							|  |  |  |       And keep us all in servile fearfulness. | 
					
						
							| 
									
										
										
										
											2021-11-29 10:42:26 -05:00
										 |  |  |       `}
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  |     /> | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | } |