From 0d7af4088a434b1e852aca101094153de62a3745 Mon Sep 17 00:00:00 2001 From: Diego Mello Date: Mon, 9 Oct 2023 13:18:02 -0300 Subject: [PATCH 01/12] fix(Android): Remove RefreshControl from messages list (#5251) --- app/views/RoomView/List/components/List.tsx | 2 +- .../RoomView/List/components/NavBottomFAB.tsx | 12 +---- .../List/components/RefreshControl.tsx | 38 -------------- app/views/RoomView/List/components/index.ts | 1 - app/views/RoomView/List/index.tsx | 49 ++++++++----------- app/views/RoomView/index.tsx | 2 +- 6 files changed, 24 insertions(+), 80 deletions(-) delete mode 100644 app/views/RoomView/List/components/RefreshControl.tsx diff --git a/app/views/RoomView/List/components/List.tsx b/app/views/RoomView/List/components/List.tsx index 22c07570f..1976075c5 100644 --- a/app/views/RoomView/List/components/List.tsx +++ b/app/views/RoomView/List/components/List.tsx @@ -43,7 +43,7 @@ export const List = ({ listRef, jumpToBottom, isThread, ...props }: IListProps) keyExtractor={item => item.id} contentContainerStyle={styles.contentContainer} style={styles.list} - inverted={isIOS} + inverted removeClippedSubviews={isIOS} initialNumToRender={7} onEndReachedThreshold={0.5} diff --git a/app/views/RoomView/List/components/NavBottomFAB.tsx b/app/views/RoomView/List/components/NavBottomFAB.tsx index a0d9a65c6..9ecb6681e 100644 --- a/app/views/RoomView/List/components/NavBottomFAB.tsx +++ b/app/views/RoomView/List/components/NavBottomFAB.tsx @@ -1,5 +1,5 @@ import React from 'react'; -import { StyleSheet, View, Platform } from 'react-native'; +import { StyleSheet, View } from 'react-native'; import { CustomIcon } from '../../../../containers/CustomIcon'; import { useTheme } from '../../../../theme'; @@ -43,15 +43,7 @@ const NavBottomFAB = ({ style={[ styles.container, { - ...Platform.select({ - ios: { - bottom: 100 + (isThread ? 40 : 0) - }, - android: { - top: 15, - scaleY: -1 - } - }) + bottom: 100 + (isThread ? 40 : 0) } ]} testID='nav-jump-to-bottom' diff --git a/app/views/RoomView/List/components/RefreshControl.tsx b/app/views/RoomView/List/components/RefreshControl.tsx deleted file mode 100644 index ef72298ab..000000000 --- a/app/views/RoomView/List/components/RefreshControl.tsx +++ /dev/null @@ -1,38 +0,0 @@ -import React from 'react'; -import { RefreshControl as RNRefreshControl, RefreshControlProps, StyleSheet } from 'react-native'; - -import { useTheme } from '../../../../theme'; -import { isAndroid } from '../../../../lib/methods/helpers'; - -const style = StyleSheet.create({ - container: { - flex: 1 - }, - inverted: { - scaleY: -1 - } -}); - -interface IRefreshControl extends RefreshControlProps { - children: React.ReactElement; -} - -export const RefreshControl = ({ children, onRefresh, refreshing }: IRefreshControl): React.ReactElement => { - const { colors } = useTheme(); - if (isAndroid) { - return ( - - {children} - - ); - } - - const refreshControl = ; - - return React.cloneElement(children, { refreshControl }); -}; diff --git a/app/views/RoomView/List/components/index.ts b/app/views/RoomView/List/components/index.ts index 7f17ff1d9..c45143de1 100644 --- a/app/views/RoomView/List/components/index.ts +++ b/app/views/RoomView/List/components/index.ts @@ -1,4 +1,3 @@ export * from './NavBottomFAB'; -export * from './RefreshControl'; export * from './EmptyRoom'; export * from './List'; diff --git a/app/views/RoomView/List/index.tsx b/app/views/RoomView/List/index.tsx index 6846f208f..c38080468 100644 --- a/app/views/RoomView/List/index.tsx +++ b/app/views/RoomView/List/index.tsx @@ -1,21 +1,12 @@ import React, { forwardRef, useImperativeHandle } from 'react'; -import { View, Platform, StyleSheet } from 'react-native'; +import { RefreshControl } from 'react-native'; import ActivityIndicator from '../../../containers/ActivityIndicator'; import { useMessages, useRefresh, useScroll } from './hooks'; -import { useDebounce } from '../../../lib/methods/helpers'; -import { RefreshControl, EmptyRoom, List } from './components'; +import { isIOS, useDebounce } from '../../../lib/methods/helpers'; +import { EmptyRoom, List } from './components'; import { IListContainerProps, IListContainerRef, IListProps } from './definitions'; - -const styles = StyleSheet.create({ - inverted: { - ...Platform.select({ - android: { - scaleY: -1 - } - }) - } -}); +import { useTheme } from '../../../theme'; const ListContainer = forwardRef( ({ rid, tmid, renderRow, showMessageInMainThread, serverVersion, hideSystemMessages, listRef, loading }, ref) => { @@ -26,6 +17,7 @@ const ListContainer = forwardRef( serverVersion, hideSystemMessages }); + const { colors } = useTheme(); const [refreshing, refresh] = useRefresh({ rid, tmid, messagesLength: messages.length }); const { jumpToBottom, @@ -52,26 +44,25 @@ const ListContainer = forwardRef( return null; }; - const renderItem: IListProps['renderItem'] = ({ item, index }) => ( - {renderRow(item, messages[index + 1], highlightedMessageId)} - ); + const renderItem: IListProps['renderItem'] = ({ item, index }) => renderRow(item, messages[index + 1], highlightedMessageId); return ( <> - - - + : undefined + } + /> ); } diff --git a/app/views/RoomView/index.tsx b/app/views/RoomView/index.tsx index 62338423f..4aea96c2c 100644 --- a/app/views/RoomView/index.tsx +++ b/app/views/RoomView/index.tsx @@ -1361,8 +1361,8 @@ class RoomView extends React.Component { if (showUnreadSeparator || dateSeparator) { return ( <> - {content} + ); } From 62559c852b45e3f30962c592c326b473823c03ca Mon Sep 17 00:00:00 2001 From: Gleidson Daniel Silva Date: Mon, 9 Oct 2023 17:24:07 -0300 Subject: [PATCH 02/12] regression: disable mediaPlaybackRequiresUserAction on Android (#5263) --- app/views/JitsiMeetView/index.tsx | 1 + 1 file changed, 1 insertion(+) diff --git a/app/views/JitsiMeetView/index.tsx b/app/views/JitsiMeetView/index.tsx index 9f3b26c10..00c7997f1 100644 --- a/app/views/JitsiMeetView/index.tsx +++ b/app/views/JitsiMeetView/index.tsx @@ -86,6 +86,7 @@ const JitsiMeetView = (): React.ReactElement => { domStorageEnabled allowsInlineMediaPlayback mediaCapturePermissionGrantType={'grant'} + mediaPlaybackRequiresUserAction={isIOS} /> ); From 6a02fde4539892042e409bdf6e7c6fc38a257f3c Mon Sep 17 00:00:00 2001 From: Diego Mello Date: Tue, 10 Oct 2023 13:43:05 -0300 Subject: [PATCH 03/12] chore: Fix CI yarn cache not working (#5265) --- yarn.lock | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/yarn.lock b/yarn.lock index e12bd923b..0e3aaeaa5 100644 --- a/yarn.lock +++ b/yarn.lock @@ -17704,7 +17704,7 @@ react-native-easy-grid@^0.2.2: dependencies: lodash "^4.17.15" -react-native-easy-toast@^1.2.0: +react-native-easy-toast@1.2.0: version "1.2.0" resolved "https://registry.yarnpkg.com/react-native-easy-toast/-/react-native-easy-toast-1.2.0.tgz#0f70bcb99e3306cda4800c244bfb4a67d42276ed" integrity sha512-UtpxnRn1ME+035Uey4VR+9K0P4aVoTcWNOx5QkioWBe3LBKMPb/kZjrQ1LtvWzOyeGP4TeTUTtMX3IOPWv5MtA== From 74cd85a069769a5d7a55fcd44a0c8eb467167750 Mon Sep 17 00:00:00 2001 From: Reinaldo Neto <47038980+reinaldonetof@users.noreply.github.com> Date: Wed, 11 Oct 2023 11:23:11 -0300 Subject: [PATCH 04/12] feat: image base64 (#5264) * improvement: handle base64 * update tests * minor tweak * remove imageprocess and remove the save to gallery from image base64 * Update app/containers/message/Image.tsx * Update app/views/AttachmentView.tsx * Update app/views/AttachmentView.tsx * Update app/views/AttachmentView.tsx * Update app/containers/message/Image.tsx * chore: change file name and fix implementation * chore: remove useless const * chore: update regex * chore: update name * Update app/lib/methods/isImageBase64.ts * Update app/lib/methods/isImageBase64.ts * early return --------- Co-authored-by: Gleidson Daniel Silva Co-authored-by: Diego Mello --- .../__snapshots__/Message.stories.storyshot | 2 +- app/containers/message/Image.tsx | 8 +++++++- app/containers/message/Message.stories.tsx | 12 ++++++++++++ app/lib/methods/handleMediaDownload.ts | 7 +++---- app/lib/methods/helpers/formatAttachmentUrl.ts | 6 +++++- app/lib/methods/index.ts | 1 + app/lib/methods/isImageBase64.test.ts | 16 ++++++++++++++++ app/lib/methods/isImageBase64.ts | 8 ++++++++ app/views/AttachmentView.tsx | 4 +++- 9 files changed, 56 insertions(+), 8 deletions(-) create mode 100644 app/lib/methods/isImageBase64.test.ts create mode 100644 app/lib/methods/isImageBase64.ts diff --git a/__tests__/containers/message/__snapshots__/Message.stories.storyshot b/__tests__/containers/message/__snapshots__/Message.stories.storyshot index 76b5b01b0..ea77bfa7d 100644 --- a/__tests__/containers/message/__snapshots__/Message.stories.storyshot +++ b/__tests__/containers/message/__snapshots__/Message.stories.storyshot @@ -78,7 +78,7 @@ exports[`Storyshots Message With Audio 1`] = `"{\\"type\\":\\"RCTScrollView\\",\ exports[`Storyshots Message With File 1`] = `"{\\"type\\":\\"RCTScrollView\\",\\"props\\":{\\"style\\":{\\"backgroundColor\\":\\"#ffffff\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"paddingVertical\\":4,\\"width\\":\\"100%\\",\\"paddingHorizontal\\":14,\\"flexDirection\\":\\"column\\"},null]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4},{\\"marginTop\\":4}],\\"testID\\":\\"avatar\\"},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4}]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/avatar/diego.mello?format=png&size=72\\",\\"headers\\":{\\"User-Agent\\":\\"RC Mobile; ios unknown; vunknown (unknown)\\"},\\"priority\\":\\"high\\"},\\"resizeMode\\":\\"cover\\"},\\"children\\":null}]}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"flex\\":1,\\"marginLeft\\":46},{\\"marginLeft\\":10}]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flex\\":1,\\"flexDirection\\":\\"row\\",\\"justifyContent\\":\\"space-between\\",\\"alignItems\\":\\"center\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flexShrink\\":1,\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"center\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"flexShrink\\":1,\\"fontSize\\":16,\\"lineHeight\\":22,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"600\\"},{\\"color\\":\\"#0d0e12\\"}],\\"numberOfLines\\":1},\\"children\\":[\\"diego.mello\\"]},{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":13,\\"marginLeft\\":8,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#9ca2a8\\"}]},\\"children\\":[\\"10:00 AM\\"]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":null}]},{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":null},{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"testID\\":\\"reply-undefined-File.pdf\\",\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flex\\":1,\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"center\\",\\"marginVertical\\":4,\\"alignSelf\\":\\"flex-start\\",\\"borderLeftWidth\\":2,\\"marginBottom\\":4,\\"borderColor\\":\\"#e1e5e8\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flex\\":1,\\"borderRadius\\":4,\\"flexDirection\\":\\"column\\",\\"paddingVertical\\":4,\\"paddingLeft\\":8}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flex\\":1,\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"center\\",\\"marginBottom\\":8}},\\"children\\":null},{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#2f343d\\"},{\\"color\\":\\"#6C727A\\",\\"fontSize\\":14}]},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"accessibilityLabel\\":\\"File.pdf\\",\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},[{},{\\"marginTop\\":0,\\"marginBottom\\":0,\\"flexWrap\\":\\"wrap\\",\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"flex-start\\",\\"justifyContent\\":\\"flex-start\\"}]]},\\"children\\":[\\"File.pdf\\"]}]}]}]},{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#2f343d\\"}]},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"accessibilityLabel\\":\\"This is a description \\",\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},[{},{\\"marginTop\\":0,\\"marginBottom\\":0,\\"flexWrap\\":\\"wrap\\",\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"flex-start\\",\\"justifyContent\\":\\"flex-start\\"}]]},\\"children\\":[\\"This is a description \\"]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},[{\\"width\\":20,\\"height\\":20}]]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/emoji-custom/nyan_rocket.png\\",\\"priority\\":\\"high\\"},\\"resizeMode\\":\\"contain\\"},\\"children\\":null}]}]}]}]}]}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"paddingVertical\\":4,\\"width\\":\\"100%\\",\\"paddingHorizontal\\":14,\\"flexDirection\\":\\"column\\"},null]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"flex\\":1,\\"marginLeft\\":46},false]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":null},{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"testID\\":\\"reply-undefined-File.pdf\\",\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flex\\":1,\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"center\\",\\"marginVertical\\":4,\\"alignSelf\\":\\"flex-start\\",\\"borderLeftWidth\\":2,\\"marginBottom\\":4,\\"borderColor\\":\\"#e1e5e8\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flex\\":1,\\"borderRadius\\":4,\\"flexDirection\\":\\"column\\",\\"paddingVertical\\":4,\\"paddingLeft\\":8}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flex\\":1,\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"center\\",\\"marginBottom\\":8}},\\"children\\":null},{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#2f343d\\"},{\\"color\\":\\"#6C727A\\",\\"fontSize\\":14}]},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"accessibilityLabel\\":\\"File.pdf\\",\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},[{},{\\"marginTop\\":0,\\"marginBottom\\":0,\\"flexWrap\\":\\"wrap\\",\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"flex-start\\",\\"justifyContent\\":\\"flex-start\\"}]]},\\"children\\":[\\"File.pdf\\"]}]}]}]},{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#2f343d\\"}]},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"accessibilityLabel\\":\\"This is a description \\",\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},[{},{\\"marginTop\\":0,\\"marginBottom\\":0,\\"flexWrap\\":\\"wrap\\",\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"flex-start\\",\\"justifyContent\\":\\"flex-start\\"}]]},\\"children\\":[\\"This is a description \\"]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},[{\\"width\\":20,\\"height\\":20}]]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/emoji-custom/nyan_rocket.png\\",\\"priority\\":\\"high\\"},\\"resizeMode\\":\\"contain\\"},\\"children\\":null}]}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":null}]}]}]}]}]}]}"`; -exports[`Storyshots Message With Image 1`] = `"{\\"type\\":\\"RCTScrollView\\",\\"props\\":{\\"style\\":{\\"backgroundColor\\":\\"#ffffff\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"paddingVertical\\":4,\\"width\\":\\"100%\\",\\"paddingHorizontal\\":14,\\"flexDirection\\":\\"column\\"},null]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4},{\\"marginTop\\":4}],\\"testID\\":\\"avatar\\"},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4}]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/avatar/diego.mello?format=png&size=72\\",\\"headers\\":{\\"User-Agent\\":\\"RC Mobile; ios unknown; vunknown (unknown)\\"},\\"priority\\":\\"high\\"},\\"resizeMode\\":\\"cover\\"},\\"children\\":null}]}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"flex\\":1,\\"marginLeft\\":46},{\\"marginLeft\\":10}]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flex\\":1,\\"flexDirection\\":\\"row\\",\\"justifyContent\\":\\"space-between\\",\\"alignItems\\":\\"center\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flexShrink\\":1,\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"center\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"flexShrink\\":1,\\"fontSize\\":16,\\"lineHeight\\":22,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"600\\"},{\\"color\\":\\"#0d0e12\\"}],\\"numberOfLines\\":1},\\"children\\":[\\"diego.mello\\"]},{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":13,\\"marginLeft\\":8,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#9ca2a8\\"}]},\\"children\\":[\\"10:00 AM\\"]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":null}]},{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":null},{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#2f343d\\"},null]},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"accessibilityLabel\\":\\"This is a description\\",\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},[{},{\\"marginTop\\":0,\\"marginBottom\\":0,\\"flexWrap\\":\\"wrap\\",\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"flex-start\\",\\"justifyContent\\":\\"flex-start\\"}]]},\\"children\\":[\\"This is a description\\"]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flexDirection\\":\\"column\\",\\"borderRadius\\":4,\\"overflow\\":\\"visible\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},[{\\"width\\":\\"100%\\",\\"minHeight\\":200,\\"borderRadius\\":4,\\"borderWidth\\":1,\\"overflow\\":\\"hidden\\"},{\\"borderColor\\":\\"#e1e5e8\\"}]]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/dummypath?rc_token=abc&rc_uid=y8bd77ptZswPj3EW8\\"},\\"resizeMode\\":\\"cover\\"},\\"children\\":null}]},{\\"type\\":\\"BlurView\\",\\"props\\":{\\"style\\":[{\\"backgroundColor\\":\\"transparent\\"},[[{\\"width\\":\\"100%\\",\\"minHeight\\":200,\\"borderRadius\\":4,\\"borderWidth\\":1,\\"overflow\\":\\"hidden\\"},{\\"height\\":\\"100%\\"}],{\\"position\\":\\"absolute\\",\\"borderWidth\\":0,\\"top\\":0,\\"left\\":0,\\"bottom\\":0,\\"right\\":0}]],\\"blurType\\":\\"dark\\",\\"blurAmount\\":2},\\"children\\":null},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[[{\\"width\\":\\"100%\\",\\"minHeight\\":200,\\"borderRadius\\":4,\\"borderWidth\\":1,\\"overflow\\":\\"hidden\\"},{\\"height\\":\\"100%\\"}],{\\"position\\":\\"absolute\\",\\"justifyContent\\":\\"center\\",\\"alignItems\\":\\"center\\"}]},\\"children\\":[{\\"type\\":\\"ActivityIndicator\\",\\"props\\":{\\"style\\":[{\\"padding\\":16,\\"flex\\":1},null],\\"color\\":\\"#9ca2a8\\",\\"size\\":54},\\"children\\":null}]}]}]}]}]}]}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"paddingVertical\\":4,\\"width\\":\\"100%\\",\\"paddingHorizontal\\":14,\\"flexDirection\\":\\"column\\"},null]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4},{\\"marginTop\\":4}],\\"testID\\":\\"avatar\\"},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4}]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/avatar/diego.mello?format=png&size=72\\",\\"headers\\":{\\"User-Agent\\":\\"RC Mobile; ios unknown; vunknown (unknown)\\"},\\"priority\\":\\"high\\"},\\"resizeMode\\":\\"cover\\"},\\"children\\":null}]}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"flex\\":1,\\"marginLeft\\":46},{\\"marginLeft\\":10}]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flex\\":1,\\"flexDirection\\":\\"row\\",\\"justifyContent\\":\\"space-between\\",\\"alignItems\\":\\"center\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flexShrink\\":1,\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"center\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"flexShrink\\":1,\\"fontSize\\":16,\\"lineHeight\\":22,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"600\\"},{\\"color\\":\\"#0d0e12\\"}],\\"numberOfLines\\":1},\\"children\\":[\\"diego.mello\\"]},{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":13,\\"marginLeft\\":8,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#9ca2a8\\"}]},\\"children\\":[\\"10:00 AM\\"]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":null}]},{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":null},{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#2f343d\\"},null]},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"accessibilityLabel\\":\\"This is a description \\",\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},[{},{\\"marginTop\\":0,\\"marginBottom\\":0,\\"flexWrap\\":\\"wrap\\",\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"flex-start\\",\\"justifyContent\\":\\"flex-start\\"}]]},\\"children\\":[\\"This is a description \\"]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},[{\\"width\\":20,\\"height\\":20}]]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/emoji-custom/nyan_rocket.png\\",\\"priority\\":\\"high\\"},\\"resizeMode\\":\\"contain\\"},\\"children\\":null}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flexDirection\\":\\"column\\",\\"borderRadius\\":4,\\"overflow\\":\\"visible\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},[{\\"width\\":\\"100%\\",\\"minHeight\\":200,\\"borderRadius\\":4,\\"borderWidth\\":1,\\"overflow\\":\\"hidden\\"},{\\"borderColor\\":\\"#e1e5e8\\"}]]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/dummypath?rc_token=abc&rc_uid=y8bd77ptZswPj3EW8\\"},\\"resizeMode\\":\\"cover\\"},\\"children\\":null}]},{\\"type\\":\\"BlurView\\",\\"props\\":{\\"style\\":[{\\"backgroundColor\\":\\"transparent\\"},[[{\\"width\\":\\"100%\\",\\"minHeight\\":200,\\"borderRadius\\":4,\\"borderWidth\\":1,\\"overflow\\":\\"hidden\\"},{\\"height\\":\\"100%\\"}],{\\"position\\":\\"absolute\\",\\"borderWidth\\":0,\\"top\\":0,\\"left\\":0,\\"bottom\\":0,\\"right\\":0}]],\\"blurType\\":\\"dark\\",\\"blurAmount\\":2},\\"children\\":null},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[[{\\"width\\":\\"100%\\",\\"minHeight\\":200,\\"borderRadius\\":4,\\"borderWidth\\":1,\\"overflow\\":\\"hidden\\"},{\\"height\\":\\"100%\\"}],{\\"position\\":\\"absolute\\",\\"justifyContent\\":\\"center\\",\\"alignItems\\":\\"center\\"}]},\\"children\\":[{\\"type\\":\\"ActivityIndicator\\",\\"props\\":{\\"style\\":[{\\"padding\\":16,\\"flex\\":1},null],\\"color\\":\\"#9ca2a8\\",\\"size\\":54},\\"children\\":null}]}]}]}]}]}]}]}]}]}]}"`; +exports[`Storyshots Message With Image 1`] = `"{\\"type\\":\\"RCTScrollView\\",\\"props\\":{\\"style\\":{\\"backgroundColor\\":\\"#ffffff\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"paddingVertical\\":4,\\"width\\":\\"100%\\",\\"paddingHorizontal\\":14,\\"flexDirection\\":\\"column\\"},null]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4},{\\"marginTop\\":4}],\\"testID\\":\\"avatar\\"},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4}]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/avatar/diego.mello?format=png&size=72\\",\\"headers\\":{\\"User-Agent\\":\\"RC Mobile; ios unknown; vunknown (unknown)\\"},\\"priority\\":\\"high\\"},\\"resizeMode\\":\\"cover\\"},\\"children\\":null}]}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"flex\\":1,\\"marginLeft\\":46},{\\"marginLeft\\":10}]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flex\\":1,\\"flexDirection\\":\\"row\\",\\"justifyContent\\":\\"space-between\\",\\"alignItems\\":\\"center\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flexShrink\\":1,\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"center\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"flexShrink\\":1,\\"fontSize\\":16,\\"lineHeight\\":22,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"600\\"},{\\"color\\":\\"#0d0e12\\"}],\\"numberOfLines\\":1},\\"children\\":[\\"diego.mello\\"]},{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":13,\\"marginLeft\\":8,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#9ca2a8\\"}]},\\"children\\":[\\"10:00 AM\\"]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":null}]},{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":null},{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#2f343d\\"},null]},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"accessibilityLabel\\":\\"This is a description\\",\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},[{},{\\"marginTop\\":0,\\"marginBottom\\":0,\\"flexWrap\\":\\"wrap\\",\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"flex-start\\",\\"justifyContent\\":\\"flex-start\\"}]]},\\"children\\":[\\"This is a description\\"]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flexDirection\\":\\"column\\",\\"borderRadius\\":4,\\"overflow\\":\\"visible\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},[{\\"width\\":\\"100%\\",\\"minHeight\\":200,\\"borderRadius\\":4,\\"borderWidth\\":1,\\"overflow\\":\\"hidden\\"},{\\"borderColor\\":\\"#e1e5e8\\"}]]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/dummypath?rc_token=abc&rc_uid=y8bd77ptZswPj3EW8\\"},\\"resizeMode\\":\\"cover\\"},\\"children\\":null}]},{\\"type\\":\\"BlurView\\",\\"props\\":{\\"style\\":[{\\"backgroundColor\\":\\"transparent\\"},[[{\\"width\\":\\"100%\\",\\"minHeight\\":200,\\"borderRadius\\":4,\\"borderWidth\\":1,\\"overflow\\":\\"hidden\\"},{\\"height\\":\\"100%\\"}],{\\"position\\":\\"absolute\\",\\"borderWidth\\":0,\\"top\\":0,\\"left\\":0,\\"bottom\\":0,\\"right\\":0}]],\\"blurType\\":\\"dark\\",\\"blurAmount\\":2},\\"children\\":null},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[[{\\"width\\":\\"100%\\",\\"minHeight\\":200,\\"borderRadius\\":4,\\"borderWidth\\":1,\\"overflow\\":\\"hidden\\"},{\\"height\\":\\"100%\\"}],{\\"position\\":\\"absolute\\",\\"justifyContent\\":\\"center\\",\\"alignItems\\":\\"center\\"}]},\\"children\\":[{\\"type\\":\\"ActivityIndicator\\",\\"props\\":{\\"style\\":[{\\"padding\\":16,\\"flex\\":1},null],\\"color\\":\\"#9ca2a8\\",\\"size\\":54},\\"children\\":null}]}]}]}]}]}]}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"paddingVertical\\":4,\\"width\\":\\"100%\\",\\"paddingHorizontal\\":14,\\"flexDirection\\":\\"column\\"},null]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4},{\\"marginTop\\":4}],\\"testID\\":\\"avatar\\"},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4}]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/avatar/diego.mello?format=png&size=72\\",\\"headers\\":{\\"User-Agent\\":\\"RC Mobile; ios unknown; vunknown (unknown)\\"},\\"priority\\":\\"high\\"},\\"resizeMode\\":\\"cover\\"},\\"children\\":null}]}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"flex\\":1,\\"marginLeft\\":46},{\\"marginLeft\\":10}]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flex\\":1,\\"flexDirection\\":\\"row\\",\\"justifyContent\\":\\"space-between\\",\\"alignItems\\":\\"center\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flexShrink\\":1,\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"center\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"flexShrink\\":1,\\"fontSize\\":16,\\"lineHeight\\":22,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"600\\"},{\\"color\\":\\"#0d0e12\\"}],\\"numberOfLines\\":1},\\"children\\":[\\"diego.mello\\"]},{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":13,\\"marginLeft\\":8,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#9ca2a8\\"}]},\\"children\\":[\\"10:00 AM\\"]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":null}]},{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":null},{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#2f343d\\"},null]},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"accessibilityLabel\\":\\"This is a description \\",\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},[{},{\\"marginTop\\":0,\\"marginBottom\\":0,\\"flexWrap\\":\\"wrap\\",\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"flex-start\\",\\"justifyContent\\":\\"flex-start\\"}]]},\\"children\\":[\\"This is a description \\"]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},[{\\"width\\":20,\\"height\\":20}]]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/emoji-custom/nyan_rocket.png\\",\\"priority\\":\\"high\\"},\\"resizeMode\\":\\"contain\\"},\\"children\\":null}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flexDirection\\":\\"column\\",\\"borderRadius\\":4,\\"overflow\\":\\"visible\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},[{\\"width\\":\\"100%\\",\\"minHeight\\":200,\\"borderRadius\\":4,\\"borderWidth\\":1,\\"overflow\\":\\"hidden\\"},{\\"borderColor\\":\\"#e1e5e8\\"}]]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/dummypath?rc_token=abc&rc_uid=y8bd77ptZswPj3EW8\\"},\\"resizeMode\\":\\"cover\\"},\\"children\\":null}]},{\\"type\\":\\"BlurView\\",\\"props\\":{\\"style\\":[{\\"backgroundColor\\":\\"transparent\\"},[[{\\"width\\":\\"100%\\",\\"minHeight\\":200,\\"borderRadius\\":4,\\"borderWidth\\":1,\\"overflow\\":\\"hidden\\"},{\\"height\\":\\"100%\\"}],{\\"position\\":\\"absolute\\",\\"borderWidth\\":0,\\"top\\":0,\\"left\\":0,\\"bottom\\":0,\\"right\\":0}]],\\"blurType\\":\\"dark\\",\\"blurAmount\\":2},\\"children\\":null},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[[{\\"width\\":\\"100%\\",\\"minHeight\\":200,\\"borderRadius\\":4,\\"borderWidth\\":1,\\"overflow\\":\\"hidden\\"},{\\"height\\":\\"100%\\"}],{\\"position\\":\\"absolute\\",\\"justifyContent\\":\\"center\\",\\"alignItems\\":\\"center\\"}]},\\"children\\":[{\\"type\\":\\"ActivityIndicator\\",\\"props\\":{\\"style\\":[{\\"padding\\":16,\\"flex\\":1},null],\\"color\\":\\"#9ca2a8\\",\\"size\\":54},\\"children\\":null}]}]}]}]}]}]}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"paddingVertical\\":4,\\"width\\":\\"100%\\",\\"paddingHorizontal\\":14,\\"flexDirection\\":\\"column\\"},null]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4},{\\"marginTop\\":4}],\\"testID\\":\\"avatar\\"},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4}]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/avatar/diego.mello?format=png&size=72\\",\\"headers\\":{\\"User-Agent\\":\\"RC Mobile; ios unknown; vunknown (unknown)\\"},\\"priority\\":\\"high\\"},\\"resizeMode\\":\\"cover\\"},\\"children\\":null}]}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"flex\\":1,\\"marginLeft\\":46},{\\"marginLeft\\":10}]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flex\\":1,\\"flexDirection\\":\\"row\\",\\"justifyContent\\":\\"space-between\\",\\"alignItems\\":\\"center\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flexShrink\\":1,\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"center\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"flexShrink\\":1,\\"fontSize\\":16,\\"lineHeight\\":22,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"600\\"},{\\"color\\":\\"#0d0e12\\"}],\\"numberOfLines\\":1},\\"children\\":[\\"diego.mello\\"]},{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":13,\\"marginLeft\\":8,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#9ca2a8\\"}]},\\"children\\":[\\"10:00 AM\\"]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":null}]},{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":null},{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#2f343d\\"},null]},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"accessibilityLabel\\":\\"This is a description for Base64 Attachment \\",\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},[{},{\\"marginTop\\":0,\\"marginBottom\\":0,\\"flexWrap\\":\\"wrap\\",\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"flex-start\\",\\"justifyContent\\":\\"flex-start\\"}]]},\\"children\\":[\\"This is a description for Base64 Attachment \\"]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},[{\\"width\\":20,\\"height\\":20}]]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/emoji-custom/nyan_rocket.png\\",\\"priority\\":\\"high\\"},\\"resizeMode\\":\\"contain\\"},\\"children\\":null}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flexDirection\\":\\"column\\",\\"borderRadius\\":4,\\"overflow\\":\\"visible\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},[{\\"width\\":\\"100%\\",\\"minHeight\\":200,\\"borderRadius\\":4,\\"borderWidth\\":1,\\"overflow\\":\\"hidden\\"},{\\"borderColor\\":\\"#e1e5e8\\"}]]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"\\"},\\"resizeMode\\":\\"cover\\"},\\"children\\":null}]}]}]}]}]}]}]}]}]}]}"`; exports[`Storyshots Message With Video 1`] = `"{\\"type\\":\\"RCTScrollView\\",\\"props\\":{\\"style\\":{\\"backgroundColor\\":\\"#ffffff\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"paddingVertical\\":4,\\"width\\":\\"100%\\",\\"paddingHorizontal\\":14,\\"flexDirection\\":\\"column\\"},null]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4},{\\"marginTop\\":4}],\\"testID\\":\\"avatar\\"},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4}]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/avatar/diego.mello?format=png&size=72\\",\\"headers\\":{\\"User-Agent\\":\\"RC Mobile; ios unknown; vunknown (unknown)\\"},\\"priority\\":\\"high\\"},\\"resizeMode\\":\\"cover\\"},\\"children\\":null}]}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"flex\\":1,\\"marginLeft\\":46},{\\"marginLeft\\":10}]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flex\\":1,\\"flexDirection\\":\\"row\\",\\"justifyContent\\":\\"space-between\\",\\"alignItems\\":\\"center\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flexShrink\\":1,\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"center\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"flexShrink\\":1,\\"fontSize\\":16,\\"lineHeight\\":22,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"600\\"},{\\"color\\":\\"#0d0e12\\"}],\\"numberOfLines\\":1},\\"children\\":[\\"diego.mello\\"]},{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":13,\\"marginLeft\\":8,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#9ca2a8\\"}]},\\"children\\":[\\"10:00 AM\\"]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":null}]},{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":null},{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#2f343d\\"},null]},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"accessibilityLabel\\":\\"This is a description \\",\\"style\\":[{\\"fontSize\\":16,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},[{},{\\"marginTop\\":0,\\"marginBottom\\":0,\\"flexWrap\\":\\"wrap\\",\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"flex-start\\",\\"justifyContent\\":\\"flex-start\\"}]]},\\"children\\":[\\"This is a description \\"]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},[{\\"width\\":20,\\"height\\":20}]]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/emoji-custom/nyan_rocket.png\\",\\"priority\\":\\"high\\"},\\"resizeMode\\":\\"contain\\"},\\"children\\":null}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flex\\":1,\\"borderRadius\\":4,\\"height\\":150,\\"marginBottom\\":6,\\"alignItems\\":\\"center\\",\\"justifyContent\\":\\"center\\",\\"overflow\\":\\"visible\\",\\"backgroundColor\\":\\"#1f2329\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"BlurView\\",\\"props\\":{\\"style\\":[{\\"backgroundColor\\":\\"transparent\\"},[{\\"flex\\":1,\\"borderRadius\\":4,\\"height\\":150,\\"marginBottom\\":6,\\"alignItems\\":\\"center\\",\\"justifyContent\\":\\"center\\"},{\\"position\\":\\"absolute\\",\\"borderWidth\\":0,\\"top\\":0,\\"left\\":0,\\"bottom\\":0,\\"right\\":0}]],\\"blurType\\":\\"dark\\",\\"blurAmount\\":2},\\"children\\":null},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"flex\\":1,\\"borderRadius\\":4,\\"height\\":150,\\"marginBottom\\":6,\\"alignItems\\":\\"center\\",\\"justifyContent\\":\\"center\\"},{\\"position\\":\\"absolute\\",\\"justifyContent\\":\\"center\\",\\"alignItems\\":\\"center\\"}]},\\"children\\":[{\\"type\\":\\"ActivityIndicator\\",\\"props\\":{\\"style\\":[{\\"padding\\":16,\\"flex\\":1},null],\\"color\\":\\"#9ca2a8\\",\\"size\\":54},\\"children\\":null}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"top\\":8,\\"right\\":8}},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\",\\"fontSize\\":12},{\\"color\\":\\"#9ca2a8\\"}]},\\"children\\":[\\"Cancel\\"]}]}]}]}]}]}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"paddingVertical\\":4,\\"width\\":\\"100%\\",\\"paddingHorizontal\\":14,\\"flexDirection\\":\\"column\\"},null]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4},{\\"marginTop\\":4}],\\"testID\\":\\"avatar\\"},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"overflow\\":\\"hidden\\"},{\\"width\\":36,\\"height\\":36,\\"borderRadius\\":4}]},\\"children\\":[{\\"type\\":\\"FastImageView\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"left\\":0,\\"right\\":0,\\"top\\":0,\\"bottom\\":0},\\"source\\":{\\"uri\\":\\"https://open.rocket.chat/avatar/diego.mello?format=png&size=72\\",\\"headers\\":{\\"User-Agent\\":\\"RC Mobile; ios unknown; vunknown (unknown)\\"},\\"priority\\":\\"high\\"},\\"resizeMode\\":\\"cover\\"},\\"children\\":null}]}]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"flex\\":1,\\"marginLeft\\":46},{\\"marginLeft\\":10}]},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flex\\":1,\\"flexDirection\\":\\"row\\",\\"justifyContent\\":\\"space-between\\",\\"alignItems\\":\\"center\\"}},\\"children\\":[{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"accessibilityState\\":{\\"disabled\\":false},\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flexShrink\\":1,\\"flexDirection\\":\\"row\\",\\"alignItems\\":\\"center\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"flexShrink\\":1,\\"fontSize\\":16,\\"lineHeight\\":22,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"600\\"},{\\"color\\":\\"#0d0e12\\"}],\\"numberOfLines\\":1},\\"children\\":[\\"diego.mello\\"]},{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"fontSize\\":13,\\"marginLeft\\":8,\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\"},{\\"color\\":\\"#9ca2a8\\"}]},\\"children\\":[\\"10:00 AM\\"]}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"flexDirection\\":\\"row\\"}},\\"children\\":null}]},{\\"type\\":\\"View\\",\\"props\\":{},\\"children\\":null},{\\"type\\":\\"View\\",\\"props\\":{\\"accessible\\":true,\\"focusable\\":true,\\"collapsable\\":false,\\"style\\":{\\"flex\\":1,\\"borderRadius\\":4,\\"height\\":150,\\"marginBottom\\":6,\\"alignItems\\":\\"center\\",\\"justifyContent\\":\\"center\\",\\"overflow\\":\\"visible\\",\\"backgroundColor\\":\\"#1f2329\\",\\"opacity\\":1}},\\"children\\":[{\\"type\\":\\"BlurView\\",\\"props\\":{\\"style\\":[{\\"backgroundColor\\":\\"transparent\\"},[{\\"flex\\":1,\\"borderRadius\\":4,\\"height\\":150,\\"marginBottom\\":6,\\"alignItems\\":\\"center\\",\\"justifyContent\\":\\"center\\"},{\\"position\\":\\"absolute\\",\\"borderWidth\\":0,\\"top\\":0,\\"left\\":0,\\"bottom\\":0,\\"right\\":0}]],\\"blurType\\":\\"dark\\",\\"blurAmount\\":2},\\"children\\":null},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":[{\\"flex\\":1,\\"borderRadius\\":4,\\"height\\":150,\\"marginBottom\\":6,\\"alignItems\\":\\"center\\",\\"justifyContent\\":\\"center\\"},{\\"position\\":\\"absolute\\",\\"justifyContent\\":\\"center\\",\\"alignItems\\":\\"center\\"}]},\\"children\\":[{\\"type\\":\\"ActivityIndicator\\",\\"props\\":{\\"style\\":[{\\"padding\\":16,\\"flex\\":1},null],\\"color\\":\\"#9ca2a8\\",\\"size\\":54},\\"children\\":null}]},{\\"type\\":\\"View\\",\\"props\\":{\\"style\\":{\\"position\\":\\"absolute\\",\\"top\\":8,\\"right\\":8}},\\"children\\":[{\\"type\\":\\"Text\\",\\"props\\":{\\"style\\":[{\\"textAlign\\":\\"left\\",\\"backgroundColor\\":\\"transparent\\",\\"fontFamily\\":\\"Inter\\",\\"fontWeight\\":\\"400\\",\\"fontSize\\":12},{\\"color\\":\\"#9ca2a8\\"}]},\\"children\\":[\\"Cancel\\"]}]}]}]}]}]}]}]}]}]}"`; diff --git a/app/containers/message/Image.tsx b/app/containers/message/Image.tsx index 61bfe1f3b..43fd1428c 100644 --- a/app/containers/message/Image.tsx +++ b/app/containers/message/Image.tsx @@ -13,6 +13,7 @@ import BlurComponent from './Components/BlurComponent'; import MessageContext from './Context'; import Touchable from './Touchable'; import styles from './styles'; +import { isImageBase64 } from '../../lib/methods'; interface IMessageButton { children: React.ReactElement; @@ -110,7 +111,12 @@ const ImageContainer = ({ await handleAutoDownload(); } }; - handleCache(); + if (isImageBase64(imgUrlToCache)) { + setLoading(false); + setCached(true); + } else { + handleCache(); + } }, []); if (!img) { diff --git a/app/containers/message/Message.stories.tsx b/app/containers/message/Message.stories.tsx index 33994b5e3..d59daa02b 100644 --- a/app/containers/message/Message.stories.tsx +++ b/app/containers/message/Message.stories.tsx @@ -356,6 +356,9 @@ export const DateAndUnreadSeparators = () => ( ); +const base64 = + ''; + export const WithImage = () => ( <> ( } ]} /> + ); diff --git a/app/lib/methods/handleMediaDownload.ts b/app/lib/methods/handleMediaDownload.ts index 9fb3c5c9b..21beb2792 100644 --- a/app/lib/methods/handleMediaDownload.ts +++ b/app/lib/methods/handleMediaDownload.ts @@ -199,8 +199,7 @@ export function downloadMediaFile({ try { const path = getFilePath({ type, mimeType, urlToCache: downloadUrl }); if (!path) { - reject(); - return; + return reject(); } downloadKey = mediaDownloadKey(downloadUrl); downloadQueue[downloadKey] = FileSystem.createDownloadResumable(downloadUrl, path); @@ -208,9 +207,9 @@ export function downloadMediaFile({ if (result?.uri) { return resolve(result.uri); } - reject(); + return reject(); } catch { - reject(); + return reject(); } finally { delete downloadQueue[downloadKey]; } diff --git a/app/lib/methods/helpers/formatAttachmentUrl.ts b/app/lib/methods/helpers/formatAttachmentUrl.ts index de8c204f6..55055f4dc 100644 --- a/app/lib/methods/helpers/formatAttachmentUrl.ts +++ b/app/lib/methods/helpers/formatAttachmentUrl.ts @@ -1,6 +1,7 @@ import { URL } from 'react-native-url-polyfill'; import { LOCAL_DOCUMENT_DIRECTORY } from '../handleMediaDownload'; +import { isImageBase64 } from '../isImageBase64'; function setParamInUrl({ url, token, userId }: { url: string; token: string; userId: string }) { const urlObj = new URL(url); @@ -10,7 +11,10 @@ function setParamInUrl({ url, token, userId }: { url: string; token: string; use } export const formatAttachmentUrl = (attachmentUrl: string | undefined, userId: string, token: string, server: string): string => { - if (LOCAL_DOCUMENT_DIRECTORY && attachmentUrl?.startsWith(LOCAL_DOCUMENT_DIRECTORY)) { + if ( + (attachmentUrl && isImageBase64(attachmentUrl)) || + (LOCAL_DOCUMENT_DIRECTORY && attachmentUrl?.startsWith(LOCAL_DOCUMENT_DIRECTORY)) + ) { return attachmentUrl; } if (attachmentUrl && attachmentUrl.startsWith('http')) { diff --git a/app/lib/methods/index.ts b/app/lib/methods/index.ts index a1ac3ae1a..d86b7ea14 100644 --- a/app/lib/methods/index.ts +++ b/app/lib/methods/index.ts @@ -38,3 +38,4 @@ export * from './parseSettings'; export * from './subscribeRooms'; export * from './serializeAsciiUrl'; export * from './isRoomFederated'; +export * from './isImageBase64'; diff --git a/app/lib/methods/isImageBase64.test.ts b/app/lib/methods/isImageBase64.test.ts new file mode 100644 index 000000000..69f34b51d --- /dev/null +++ b/app/lib/methods/isImageBase64.test.ts @@ -0,0 +1,16 @@ +import { isImageBase64 } from './isImageBase64'; + +// We aren't testing the content, only the header +const base64 = + ''; + +describe('Test the isImageBase64', () => { + it.each([ + ['test', false], + ['/file-upload/oTQmb2zRCsYF4pdHv/help-image-url.png', false], + [base64, true] + ])('return properly the boolean', (data, res) => { + const result = isImageBase64(data); + expect(result).toBe(res); + }); +}); diff --git a/app/lib/methods/isImageBase64.ts b/app/lib/methods/isImageBase64.ts new file mode 100644 index 000000000..3ec0e5b3a --- /dev/null +++ b/app/lib/methods/isImageBase64.ts @@ -0,0 +1,8 @@ +const imageBase64RegExp = new RegExp(/^data:image\/([a-zA-Z]*);base64,([^\"]*)$/); + +export function isImageBase64(data?: string): boolean { + if (!data) { + return false; + } + return imageBase64RegExp.test(data); +} diff --git a/app/views/AttachmentView.tsx b/app/views/AttachmentView.tsx index d77bfbd8e..89fd71f15 100644 --- a/app/views/AttachmentView.tsx +++ b/app/views/AttachmentView.tsx @@ -8,6 +8,7 @@ import { useSafeAreaInsets } from 'react-native-safe-area-context'; import { shallowEqual } from 'react-redux'; import RNFetchBlob from 'rn-fetch-blob'; +import { isImageBase64 } from '../lib/methods'; import RCActivityIndicator from '../containers/ActivityIndicator'; import * as HeaderButton from '../containers/HeaderButton'; import { ImageViewer } from '../containers/ImageViewer'; @@ -109,6 +110,7 @@ const AttachmentView = (): React.ReactElement => { const setHeader = () => { let { title } = attachment; + try { if (title) { title = decodeURI(title); @@ -128,7 +130,7 @@ const AttachmentView = (): React.ReactElement => { ), headerRight: () => - Allow_Save_Media_to_Gallery ? ( + Allow_Save_Media_to_Gallery && !isImageBase64(attachment.image_url) ? ( ) : null, headerBackground: () => ( From 231057af10c9243960e7f1e40a3d3916f143e969 Mon Sep 17 00:00:00 2001 From: Gleidson Daniel Silva Date: Fri, 13 Oct 2023 16:34:59 -0300 Subject: [PATCH 05/12] regression(Android): Poor performance in messages list (#5267) * fix: remove RefreshControl from RoomView * fix: add container to verify android or ios --- app/views/RoomView/List/components/List.tsx | 2 +- .../RoomView/List/components/NavBottomFAB.tsx | 12 ++++- app/views/RoomView/List/hooks/index.ts | 1 - app/views/RoomView/List/hooks/useRefresh.ts | 27 ---------- app/views/RoomView/List/index.tsx | 53 +++++++++++-------- app/views/RoomView/index.tsx | 2 +- 6 files changed, 44 insertions(+), 53 deletions(-) delete mode 100644 app/views/RoomView/List/hooks/useRefresh.ts diff --git a/app/views/RoomView/List/components/List.tsx b/app/views/RoomView/List/components/List.tsx index 1976075c5..22c07570f 100644 --- a/app/views/RoomView/List/components/List.tsx +++ b/app/views/RoomView/List/components/List.tsx @@ -43,7 +43,7 @@ export const List = ({ listRef, jumpToBottom, isThread, ...props }: IListProps) keyExtractor={item => item.id} contentContainerStyle={styles.contentContainer} style={styles.list} - inverted + inverted={isIOS} removeClippedSubviews={isIOS} initialNumToRender={7} onEndReachedThreshold={0.5} diff --git a/app/views/RoomView/List/components/NavBottomFAB.tsx b/app/views/RoomView/List/components/NavBottomFAB.tsx index 9ecb6681e..a0d9a65c6 100644 --- a/app/views/RoomView/List/components/NavBottomFAB.tsx +++ b/app/views/RoomView/List/components/NavBottomFAB.tsx @@ -1,5 +1,5 @@ import React from 'react'; -import { StyleSheet, View } from 'react-native'; +import { StyleSheet, View, Platform } from 'react-native'; import { CustomIcon } from '../../../../containers/CustomIcon'; import { useTheme } from '../../../../theme'; @@ -43,7 +43,15 @@ const NavBottomFAB = ({ style={[ styles.container, { - bottom: 100 + (isThread ? 40 : 0) + ...Platform.select({ + ios: { + bottom: 100 + (isThread ? 40 : 0) + }, + android: { + top: 15, + scaleY: -1 + } + }) } ]} testID='nav-jump-to-bottom' diff --git a/app/views/RoomView/List/hooks/index.ts b/app/views/RoomView/List/hooks/index.ts index 12e970eca..0e9e991f8 100644 --- a/app/views/RoomView/List/hooks/index.ts +++ b/app/views/RoomView/List/hooks/index.ts @@ -1,3 +1,2 @@ export * from './useMessages'; -export * from './useRefresh'; export * from './useScroll'; diff --git a/app/views/RoomView/List/hooks/useRefresh.ts b/app/views/RoomView/List/hooks/useRefresh.ts deleted file mode 100644 index 79b744d13..000000000 --- a/app/views/RoomView/List/hooks/useRefresh.ts +++ /dev/null @@ -1,27 +0,0 @@ -import moment from 'moment'; -import { useState } from 'react'; - -import log from '../../../../lib/methods/helpers/log'; -import { loadMissedMessages, loadThreadMessages } from '../../../../lib/methods'; - -export const useRefresh = ({ rid, tmid, messagesLength }: { rid: string; tmid?: string; messagesLength: number }) => { - const [refreshing, setRefreshing] = useState(false); - - const refresh = async () => { - if (messagesLength) { - setRefreshing(true); - try { - if (tmid) { - await loadThreadMessages({ tmid, rid }); - } else { - await loadMissedMessages({ rid, lastOpen: moment().subtract(7, 'days').toDate() }); - } - } catch (e) { - log(e); - } - setRefreshing(false); - } - }; - - return [refreshing, refresh] as const; -}; diff --git a/app/views/RoomView/List/index.tsx b/app/views/RoomView/List/index.tsx index c38080468..a4c666be5 100644 --- a/app/views/RoomView/List/index.tsx +++ b/app/views/RoomView/List/index.tsx @@ -1,12 +1,24 @@ import React, { forwardRef, useImperativeHandle } from 'react'; -import { RefreshControl } from 'react-native'; +import { Platform, StyleSheet, View } from 'react-native'; import ActivityIndicator from '../../../containers/ActivityIndicator'; -import { useMessages, useRefresh, useScroll } from './hooks'; -import { isIOS, useDebounce } from '../../../lib/methods/helpers'; +import { isAndroid, useDebounce } from '../../../lib/methods/helpers'; import { EmptyRoom, List } from './components'; import { IListContainerProps, IListContainerRef, IListProps } from './definitions'; -import { useTheme } from '../../../theme'; +import { useMessages, useScroll } from './hooks'; + +const styles = StyleSheet.create({ + inverted: { + ...Platform.select({ + android: { + scaleY: -1 + } + }) + } +}); + +const Container = ({ children }: { children: React.ReactElement }) => + isAndroid ? {children} : <>{children}; const ListContainer = forwardRef( ({ rid, tmid, renderRow, showMessageInMainThread, serverVersion, hideSystemMessages, listRef, loading }, ref) => { @@ -17,8 +29,6 @@ const ListContainer = forwardRef( serverVersion, hideSystemMessages }); - const { colors } = useTheme(); - const [refreshing, refresh] = useRefresh({ rid, tmid, messagesLength: messages.length }); const { jumpToBottom, jumpToMessage, @@ -44,25 +54,26 @@ const ListContainer = forwardRef( return null; }; - const renderItem: IListProps['renderItem'] = ({ item, index }) => renderRow(item, messages[index + 1], highlightedMessageId); + const renderItem: IListProps['renderItem'] = ({ item, index }) => ( + {renderRow(item, messages[index + 1], highlightedMessageId)} + ); return ( <> - : undefined - } - /> + + + ); } diff --git a/app/views/RoomView/index.tsx b/app/views/RoomView/index.tsx index 4aea96c2c..62338423f 100644 --- a/app/views/RoomView/index.tsx +++ b/app/views/RoomView/index.tsx @@ -1361,8 +1361,8 @@ class RoomView extends React.Component { if (showUnreadSeparator || dateSeparator) { return ( <> - {content} + {content} ); } From 199331f8f6bec7079781eaa3b0732aad1f1b25d3 Mon Sep 17 00:00:00 2001 From: Reinaldo Neto <47038980+reinaldonetof@users.noreply.github.com> Date: Mon, 16 Oct 2023 17:55:38 -0300 Subject: [PATCH 06/12] fix: check the permissions properly to create a discussion and reply in dm (#5217) * fix: check the permissions properly to create a discussion * add the permission start-discussion-other-user and fix the permission at messagebox --------- Co-authored-by: Gleidson Daniel Silva --- app/containers/MessageActions/index.tsx | 31 ++++++++++++------ app/containers/MessageBox/index.tsx | 43 ++++++++++++++++++------- app/lib/methods/getPermissions.ts | 1 + 3 files changed, 53 insertions(+), 22 deletions(-) diff --git a/app/containers/MessageActions/index.tsx b/app/containers/MessageActions/index.tsx index 9ee4be937..2f9864659 100644 --- a/app/containers/MessageActions/index.tsx +++ b/app/containers/MessageActions/index.tsx @@ -44,6 +44,7 @@ export interface IMessageActionsProps { deleteOwnMessagePermission?: string[]; pinMessagePermission?: string[]; createDirectMessagePermission?: string[]; + createDiscussionOtherUserPermission?: string[]; } export interface IMessageActions { @@ -76,6 +77,7 @@ const MessageActions = React.memo( deleteOwnMessagePermission, pinMessagePermission, createDirectMessagePermission, + createDiscussionOtherUserPermission, serverVersion }, ref @@ -85,7 +87,9 @@ const MessageActions = React.memo( hasDeletePermission: false, hasForceDeletePermission: false, hasPinPermission: false, - hasDeleteOwnPermission: false + hasDeleteOwnPermission: false, + hasCreateDirectMessagePermission: false, + hasCreateDiscussionOtherUserPermission: false }; const { showActionSheet, hideActionSheet } = useActionSheet(); @@ -96,7 +100,9 @@ const MessageActions = React.memo( deleteMessagePermission, forceDeleteMessagePermission, pinMessagePermission, - deleteOwnMessagePermission + deleteOwnMessagePermission, + createDirectMessagePermission, + createDiscussionOtherUserPermission ]; const result = await hasPermission(permission, room.rid); permissions = { @@ -104,7 +110,9 @@ const MessageActions = React.memo( hasDeletePermission: result[1], hasForceDeletePermission: result[2], hasPinPermission: result[3], - hasDeleteOwnPermission: result[4] + hasDeleteOwnPermission: result[4], + hasCreateDirectMessagePermission: result[5], + hasCreateDiscussionOtherUserPermission: result[6] }; } catch { // Do nothing @@ -385,7 +393,7 @@ const MessageActions = React.memo( } // Reply in DM - if (room.t !== 'd' && room.t !== 'l' && createDirectMessagePermission && !videoConfBlock) { + if (room.t !== 'd' && room.t !== 'l' && permissions.hasCreateDirectMessagePermission && !videoConfBlock) { options.push({ title: I18n.t('Reply_in_direct_message'), icon: 'arrow-back', @@ -394,11 +402,13 @@ const MessageActions = React.memo( } // Create Discussion - options.push({ - title: I18n.t('Start_a_Discussion'), - icon: 'discussions', - onPress: () => handleCreateDiscussion(message) - }); + if (permissions.hasCreateDiscussionOtherUserPermission) { + options.push({ + title: I18n.t('Start_a_Discussion'), + icon: 'discussions', + onPress: () => handleCreateDiscussion(message) + }); + } if (compareServerVersion(serverVersion, 'greaterThanOrEqualTo', '6.2.0') && !videoConfBlock) { options.push({ @@ -541,7 +551,8 @@ const mapStateToProps = (state: IApplicationState) => ({ deleteOwnMessagePermission: state.permissions['delete-own-message'], forceDeleteMessagePermission: state.permissions['force-delete-message'], pinMessagePermission: state.permissions['pin-message'], - createDirectMessagePermission: state.permissions['create-d'] + createDirectMessagePermission: state.permissions['create-d'], + createDiscussionOtherUserPermission: state.permissions['start-discussion-other-user'] }); export default connect(mapStateToProps, null, null, { forwardRef: true })(MessageActions); diff --git a/app/containers/MessageBox/index.tsx b/app/containers/MessageBox/index.tsx index cbf4dfd94..695d8572d 100644 --- a/app/containers/MessageBox/index.tsx +++ b/app/containers/MessageBox/index.tsx @@ -112,6 +112,7 @@ export interface IMessageBoxProps extends IBaseScreen void | null; serverVersion: string; } @@ -130,6 +131,7 @@ interface IMessageBoxState { tshow: boolean; mentionLoading: boolean; permissionToUpload: boolean; + hasCreateDiscussionPermission: boolean; showEmojiSearchbar: boolean; } @@ -184,7 +186,8 @@ class MessageBox extends Component { tshow: this.sendThreadToChannel, mentionLoading: false, permissionToUpload: true, - showEmojiSearchbar: false + showEmojiSearchbar: false, + hasCreateDiscussionPermission: false }; this.text = ''; this.selection = { start: 0, end: 0 }; @@ -325,6 +328,7 @@ class MessageBox extends Component { mentionLoading, trackingType, permissionToUpload, + hasCreateDiscussionPermission, showEmojiSearchbar } = this.state; @@ -337,6 +341,7 @@ class MessageBox extends Component { theme, usedCannedResponse, uploadFilePermission, + createDiscussionPermission, goToCannedResponses } = this.props; if (nextProps.theme !== theme) { @@ -378,6 +383,9 @@ class MessageBox extends Component { if (nextState.permissionToUpload !== permissionToUpload) { return true; } + if (nextState.hasCreateDiscussionPermission !== hasCreateDiscussionPermission) { + return true; + } if (!dequal(nextState.mentions, mentions)) { return true; } @@ -390,6 +398,9 @@ class MessageBox extends Component { if (!dequal(nextProps.uploadFilePermission, uploadFilePermission)) { return true; } + if (!dequal(nextProps.createDiscussionPermission, createDiscussionPermission)) { + return true; + } if (nextProps.usedCannedResponse !== usedCannedResponse) { return true; } @@ -400,13 +411,18 @@ class MessageBox extends Component { } componentDidUpdate(prevProps: IMessageBoxProps) { - const { uploadFilePermission, goToCannedResponses, replyWithMention, threadsEnabled } = this.props; + const { uploadFilePermission, goToCannedResponses, replyWithMention, threadsEnabled, createDiscussionPermission } = + this.props; if (prevProps.replyWithMention !== replyWithMention) { if (threadsEnabled && replyWithMention) { this.setState({ tshow: this.sendThreadToChannel }); } } - if (!dequal(prevProps.uploadFilePermission, uploadFilePermission) || prevProps.goToCannedResponses !== goToCannedResponses) { + if ( + !dequal(prevProps.uploadFilePermission, uploadFilePermission) || + !dequal(prevProps.createDiscussionPermission, createDiscussionPermission) || + prevProps.goToCannedResponses !== goToCannedResponses + ) { this.setOptions(); } } @@ -441,7 +457,7 @@ class MessageBox extends Component { } setOptions = async () => { - const { uploadFilePermission, rid } = this.props; + const { uploadFilePermission, rid, createDiscussionPermission } = this.props; // Servers older than 4.2 if (!uploadFilePermission) { @@ -449,8 +465,8 @@ class MessageBox extends Component { return; } - const permissionToUpload = await hasPermission([uploadFilePermission], rid); - this.setState({ permissionToUpload: permissionToUpload[0] }); + const permissions = await hasPermission([uploadFilePermission, createDiscussionPermission], rid); + this.setState({ permissionToUpload: permissions[0], hasCreateDiscussionPermission: permissions[1] }); }; onChangeText: any = (text: string): void => { @@ -877,7 +893,7 @@ class MessageBox extends Component { showMessageBoxActions = () => { logEvent(events.ROOM_SHOW_BOX_ACTIONS); - const { permissionToUpload } = this.state; + const { permissionToUpload, hasCreateDiscussionPermission } = this.state; const { showActionSheet, goToCannedResponses } = this.props; const options: TActionSheetOptionsItem[] = []; @@ -913,11 +929,13 @@ class MessageBox extends Component { ); } - options.push({ - title: I18n.t('Create_Discussion'), - icon: 'discussions', - onPress: this.createDiscussion - }); + if (hasCreateDiscussionPermission) { + options.push({ + title: I18n.t('Create_Discussion'), + icon: 'discussions', + onPress: this.createDiscussion + }); + } this.closeEmojiAndAction(showActionSheet, { options }); }; @@ -1316,6 +1334,7 @@ const mapStateToProps = (state: IApplicationState) => ({ FileUpload_MaxFileSize: state.settings.FileUpload_MaxFileSize, Message_AudioRecorderEnabled: state.settings.Message_AudioRecorderEnabled, uploadFilePermission: state.permissions['mobile-upload-file'], + createDiscussionPermission: state.permissions['start-discussion'], serverVersion: state.server.version }); diff --git a/app/lib/methods/getPermissions.ts b/app/lib/methods/getPermissions.ts index aab7a432c..cd741ee62 100644 --- a/app/lib/methods/getPermissions.ts +++ b/app/lib/methods/getPermissions.ts @@ -23,6 +23,7 @@ export const SUPPORTED_PERMISSIONS = [ 'create-p', 'create-d', 'start-discussion', + 'start-discussion-other-user', 'create-team', 'delete-c', 'delete-message', From 3cf08c29e180b64742f2dac81f498449e091de40 Mon Sep 17 00:00:00 2001 From: Reinaldo Neto <47038980+reinaldonetof@users.noreply.github.com> Date: Tue, 17 Oct 2023 16:28:02 -0300 Subject: [PATCH 07/12] feat: add the support for CDN_PREFIX (#5266) * add the support for cdn_prefix * add cdn-prefix to video and audio too * refactor where call the store to cdn * fix e2e test * add connect redux --- app/containers/Avatar/Avatar.tsx | 6 ++++-- app/containers/Avatar/AvatarContainer.tsx | 6 ++++-- app/containers/Avatar/interfaces.ts | 1 + app/containers/message/Audio.tsx | 15 ++++++++++----- app/lib/constants/defaultSettings.ts | 3 +++ app/lib/methods/helpers/formatAttachmentUrl.ts | 5 +++++ app/lib/methods/helpers/getAvatarUrl.ts | 7 ++++++- e2e/tests/room/02-room.spec.ts | 2 ++ e2e/tests/room/05-threads.spec.ts | 3 +++ e2e/tests/room/11-autoTranslate.spec.ts | 4 +++- 10 files changed, 41 insertions(+), 11 deletions(-) diff --git a/app/containers/Avatar/Avatar.tsx b/app/containers/Avatar/Avatar.tsx index 60bbfe14f..7d10b2936 100644 --- a/app/containers/Avatar/Avatar.tsx +++ b/app/containers/Avatar/Avatar.tsx @@ -30,7 +30,8 @@ const Avatar = React.memo( borderRadius = 4, type = SubscriptionType.DIRECT, avatarExternalProviderUrl, - roomAvatarExternalProviderUrl + roomAvatarExternalProviderUrl, + cdnPrefix }: IAvatar) => { if ((!text && !avatar && !emoji && !rid) || !server) { return null; @@ -61,7 +62,8 @@ const Avatar = React.memo( rid, blockUnauthenticatedAccess, avatarExternalProviderUrl, - roomAvatarExternalProviderUrl + roomAvatarExternalProviderUrl, + cdnPrefix }); } diff --git a/app/containers/Avatar/AvatarContainer.tsx b/app/containers/Avatar/AvatarContainer.tsx index acfb0978a..e8e6a0754 100644 --- a/app/containers/Avatar/AvatarContainer.tsx +++ b/app/containers/Avatar/AvatarContainer.tsx @@ -32,9 +32,10 @@ const AvatarContainer = ({ shallowEqual ); - const { avatarExternalProviderUrl, roomAvatarExternalProviderUrl } = useSelector((state: IApplicationState) => ({ + const { avatarExternalProviderUrl, roomAvatarExternalProviderUrl, cdnPrefix } = useSelector((state: IApplicationState) => ({ avatarExternalProviderUrl: state.settings.Accounts_AvatarExternalProviderUrl as string, - roomAvatarExternalProviderUrl: state.settings.Accounts_RoomAvatarExternalProviderUrl as string + roomAvatarExternalProviderUrl: state.settings.Accounts_RoomAvatarExternalProviderUrl as string, + cdnPrefix: state.settings.CDN_PREFIX as string })); const blockUnauthenticatedAccess = useSelector( (state: IApplicationState) => @@ -67,6 +68,7 @@ const AvatarContainer = ({ roomAvatarExternalProviderUrl={roomAvatarExternalProviderUrl} avatarETag={avatarETag} serverVersion={serverVersion} + cdnPrefix={cdnPrefix} /> ); }; diff --git a/app/containers/Avatar/interfaces.ts b/app/containers/Avatar/interfaces.ts index 6ae020202..625ae4b87 100644 --- a/app/containers/Avatar/interfaces.ts +++ b/app/containers/Avatar/interfaces.ts @@ -24,4 +24,5 @@ export interface IAvatar { serverVersion?: string | null; avatarExternalProviderUrl?: string; roomAvatarExternalProviderUrl?: string; + cdnPrefix?: string; } diff --git a/app/containers/message/Audio.tsx b/app/containers/message/Audio.tsx index 315a26442..7f6afdf54 100644 --- a/app/containers/message/Audio.tsx +++ b/app/containers/message/Audio.tsx @@ -6,6 +6,7 @@ import moment from 'moment'; import { dequal } from 'dequal'; import { activateKeepAwake, deactivateKeepAwake } from 'expo-keep-awake'; import { Sound } from 'expo-av/build/Audio/Sound'; +import { connect } from 'react-redux'; import Touchable from './Touchable'; import Markdown from '../markdown'; @@ -17,7 +18,7 @@ import MessageContext from './Context'; import ActivityIndicator from '../ActivityIndicator'; import { withDimensions } from '../../dimensions'; import { TGetCustomEmoji } from '../../definitions/IEmoji'; -import { IAttachment, IUserMessage } from '../../definitions'; +import { IApplicationState, IAttachment, IUserMessage } from '../../definitions'; import { TSupportedThemes, useTheme } from '../../theme'; import { downloadMediaFile, getMediaCache } from '../../lib/methods/handleMediaDownload'; import EventEmitter from '../../lib/methods/helpers/events'; @@ -41,6 +42,7 @@ interface IMessageAudioProps { scale?: number; author?: IUserMessage; msg?: string; + cdnPrefix?: string; } interface IMessageAudioState { @@ -208,13 +210,12 @@ class MessageAudio extends React.Component { - const { file } = this.props; + const { file, cdnPrefix } = this.props; // @ts-ignore can't use declare to type this const { baseUrl } = this.context; - let url = file.audio_url; if (url && !url.startsWith('http')) { - url = `${baseUrl}${file.audio_url}`; + url = `${cdnPrefix || baseUrl}${file.audio_url}`; } return url; }; @@ -393,4 +394,8 @@ class MessageAudio extends React.Component ({ + cdnPrefix: state.settings.CDN_PREFIX as string +}); + +export default connect(mapStateToProps)(withDimensions(MessageAudio)); diff --git a/app/lib/constants/defaultSettings.ts b/app/lib/constants/defaultSettings.ts index 164832190..4551f6319 100644 --- a/app/lib/constants/defaultSettings.ts +++ b/app/lib/constants/defaultSettings.ts @@ -246,5 +246,8 @@ export const defaultSettings = { Omnichannel_call_provider: { type: 'valueAsBoolean' }, + CDN_PREFIX: { + type: 'valueAsString' + }, ...deprecatedSettings } as const; diff --git a/app/lib/methods/helpers/formatAttachmentUrl.ts b/app/lib/methods/helpers/formatAttachmentUrl.ts index 55055f4dc..143188594 100644 --- a/app/lib/methods/helpers/formatAttachmentUrl.ts +++ b/app/lib/methods/helpers/formatAttachmentUrl.ts @@ -2,6 +2,7 @@ import { URL } from 'react-native-url-polyfill'; import { LOCAL_DOCUMENT_DIRECTORY } from '../handleMediaDownload'; import { isImageBase64 } from '../isImageBase64'; +import { store } from '../../store/auxStore'; function setParamInUrl({ url, token, userId }: { url: string; token: string; userId: string }) { const urlObj = new URL(url); @@ -23,5 +24,9 @@ export const formatAttachmentUrl = (attachmentUrl: string | undefined, userId: s } return setParamInUrl({ url: attachmentUrl, token, userId }); } + const cdnPrefix = store?.getState().settings.CDN_PREFIX as string; + if (cdnPrefix) { + server = cdnPrefix.trim().replace(/\/+$/, ''); + } return setParamInUrl({ url: `${server}${attachmentUrl}`, token, userId }); }; diff --git a/app/lib/methods/helpers/getAvatarUrl.ts b/app/lib/methods/helpers/getAvatarUrl.ts index 476b54896..1702369ef 100644 --- a/app/lib/methods/helpers/getAvatarUrl.ts +++ b/app/lib/methods/helpers/getAvatarUrl.ts @@ -22,7 +22,8 @@ export const getAvatarURL = ({ blockUnauthenticatedAccess, serverVersion, avatarExternalProviderUrl, - roomAvatarExternalProviderUrl + roomAvatarExternalProviderUrl, + cdnPrefix }: IAvatar): string => { let room; if (type === SubscriptionType.DIRECT) { @@ -48,6 +49,10 @@ export const getAvatarURL = ({ query += `&etag=${avatarETag}`; } + if (cdnPrefix) { + server = cdnPrefix.trim().replace(/\/+$/, ''); + } + if (avatar) { if (avatar.startsWith('http')) { return avatar; diff --git a/e2e/tests/room/02-room.spec.ts b/e2e/tests/room/02-room.spec.ts index 8a863f9de..a7f091106 100644 --- a/e2e/tests/room/02-room.spec.ts +++ b/e2e/tests/room/02-room.spec.ts @@ -478,6 +478,8 @@ describe('Room screen', () => { .toExist() .withTimeout(2000); await expect(element(by.id('action-sheet-handle'))).toBeVisible(); + // Fix android flaky test. Close the action sheet, then re-open again + await element(by.id('action-sheet-handle')).swipe('up', 'fast', 0.5); await element(by.id('action-sheet')).swipe('up', 'fast', 0.5); await sleep(300); // wait for animation await waitFor(element(by[textMatcher]('Delete'))) diff --git a/e2e/tests/room/05-threads.spec.ts b/e2e/tests/room/05-threads.spec.ts index 9f46eceab..5ef800674 100644 --- a/e2e/tests/room/05-threads.spec.ts +++ b/e2e/tests/room/05-threads.spec.ts @@ -237,6 +237,9 @@ describe('Threads', () => { .withTimeout(5000); await element(by.id(`message-thread-button-${thread}`)).tap(); await tryTapping(element(by[textMatcher]('replied')).atIndex(0), 2000, true); + // Fix android flaky test. Close the action sheet, then re-open again + await element(by.id('action-sheet-handle')).swipe('up', 'fast', 0.5); + await sleep(1000); // wait for animation await element(by.id('action-sheet')).swipe('up', 'fast', 0.5); await sleep(300); // wait for animation await element(by[textMatcher]('Delete')).atIndex(0).tap(); diff --git a/e2e/tests/room/11-autoTranslate.spec.ts b/e2e/tests/room/11-autoTranslate.spec.ts index 44fa2d81d..814ad5f37 100644 --- a/e2e/tests/room/11-autoTranslate.spec.ts +++ b/e2e/tests/room/11-autoTranslate.spec.ts @@ -137,9 +137,10 @@ describe('Auto Translate', () => { // verify default language is checked await waitFor(element(by.id(`auto-translate-view-${languages.default}`))) - .toBeVisible() + .toExist() .whileElement(by.id('auto-translate-view')) .scroll(750, 'down'); + await element(by.id('auto-translate-view')).swipe('up', 'slow', 0.5); await waitForVisible(`auto-translate-view-${languages.default}-check`); // enable translated language @@ -219,6 +220,7 @@ describe('Auto Translate', () => { }); it(`should don't see action to View original when disable auto translate`, async () => { + await searchMessage(oldMessage[languages.default] as string, textMatcher); // will scroll the messages list to the last one await waitForVisibleTextMatcher(oldMessage[languages.default] as string, textMatcher); await tryTapping(element(by[textMatcher](oldMessage[languages.default] as string)).atIndex(0), 2000, true); From c0660db06d2d3ac8ac45bcae0ab32cbf96cd2670 Mon Sep 17 00:00:00 2001 From: Reinaldo Neto <47038980+reinaldonetof@users.noreply.github.com> Date: Wed, 18 Oct 2023 14:13:42 -0300 Subject: [PATCH 08/12] chore: migrate AuthenticationWebView to hooks (#5054) * chore: migrate AuthenticationWebView to hooks * minor tweak * remove navigation, tweak at useRoute --------- Co-authored-by: GleidsonDaniel --- app/lib/methods/helpers/debounce.ts | 6 +- app/stacks/OutsideStack.tsx | 6 +- app/views/AuthenticationWebView.tsx | 181 ++++++++++------------------ 3 files changed, 71 insertions(+), 122 deletions(-) diff --git a/app/lib/methods/helpers/debounce.ts b/app/lib/methods/helpers/debounce.ts index 5061e322b..4a58b6d76 100644 --- a/app/lib/methods/helpers/debounce.ts +++ b/app/lib/methods/helpers/debounce.ts @@ -1,4 +1,4 @@ -import { useDebouncedCallback } from 'use-debounce'; +import { useDebouncedCallback, Options } from 'use-debounce'; export function debounce(func: Function, wait?: number, immediate?: boolean) { let timeout: ReturnType | null; @@ -24,6 +24,6 @@ export function debounce(func: Function, wait?: number, immediate?: boolean) { return _debounce; } -export function useDebounce(func: (...args: any) => any, wait?: number): (...args: any[]) => void { - return useDebouncedCallback(func, wait || 1000); +export function useDebounce(func: (...args: any) => any, wait?: number, options?: Options): (...args: any[]) => void { + return useDebouncedCallback(func, wait || 1000, options); } diff --git a/app/stacks/OutsideStack.tsx b/app/stacks/OutsideStack.tsx index ec19c2319..40a3c9cb1 100644 --- a/app/stacks/OutsideStack.tsx +++ b/app/stacks/OutsideStack.tsx @@ -50,11 +50,7 @@ const OutsideStackModal = () => { screenOptions={{ ...defaultHeader, ...themedHeader(theme), ...ModalAnimation, presentation: 'transparentModal' }} > - + ); }; diff --git a/app/views/AuthenticationWebView.tsx b/app/views/AuthenticationWebView.tsx index 894c524b9..4874f13a1 100644 --- a/app/views/AuthenticationWebView.tsx +++ b/app/views/AuthenticationWebView.tsx @@ -1,20 +1,20 @@ -import React from 'react'; -import { WebView, WebViewNavigation } from 'react-native-webview'; -import { connect } from 'react-redux'; -import parse from 'url-parse'; -import { StackNavigationProp } from '@react-navigation/stack'; -import { WebViewMessage } from 'react-native-webview/lib/WebViewTypes'; import { RouteProp } from '@react-navigation/core'; +import { useNavigation, useRoute } from '@react-navigation/native'; +import { StackNavigationProp } from '@react-navigation/stack'; +import React, { useLayoutEffect, useState } from 'react'; +import { WebView, WebViewNavigation } from 'react-native-webview'; +import { WebViewMessage } from 'react-native-webview/lib/WebViewTypes'; +import parse from 'url-parse'; -import { OutsideModalParamList } from '../stacks/types'; -import StatusBar from '../containers/StatusBar'; import ActivityIndicator from '../containers/ActivityIndicator'; -import { TSupportedThemes, withTheme } from '../theme'; -import { userAgent } from '../lib/constants'; -import { debounce } from '../lib/methods/helpers'; import * as HeaderButton from '../containers/HeaderButton'; +import StatusBar from '../containers/StatusBar'; +import { ICredentials } from '../definitions'; +import { userAgent } from '../lib/constants'; +import { useAppSelector } from '../lib/hooks'; +import { useDebounce } from '../lib/methods/helpers'; import { Services } from '../lib/services'; -import { IApplicationState, ICredentials } from '../definitions'; +import { OutsideModalParamList } from '../stacks/types'; // iframe uses a postMessage to send the token to the client // We'll handle this sending the token to the hash of the window.location @@ -40,95 +40,56 @@ window.addEventListener('popstate', function() { }); `; -interface INavigationOption { - navigation: StackNavigationProp; - route: RouteProp; -} +const AuthenticationWebView = () => { + const [logging, setLogging] = useState(false); + const [loading, setLoading] = useState(false); -interface IAuthenticationWebView extends INavigationOption { - server: string; - Accounts_Iframe_api_url: string; - Accounts_Iframe_api_method: string; - theme: TSupportedThemes; -} + const navigation = useNavigation>(); + const { + params: { authType, url, ssoToken } + } = useRoute>(); -interface IState { - logging: boolean; - loading: boolean; -} + const { Accounts_Iframe_api_method, Accounts_Iframe_api_url, server } = useAppSelector(state => ({ + server: state.server.server, + Accounts_Iframe_api_url: state.settings.Accounts_Iframe_api_url as string, + Accounts_Iframe_api_method: state.settings.Accounts_Iframe_api_method as string + })); -class AuthenticationWebView extends React.PureComponent { - private oauthRedirectRegex: RegExp; - private iframeRedirectRegex: RegExp; + const oauthRedirectRegex = new RegExp(`(?=.*(${server}))(?=.*(credentialToken))(?=.*(credentialSecret))`, 'g'); + const iframeRedirectRegex = new RegExp(`(?=.*(${server}))(?=.*(event|loginToken|token))`, 'g'); - static navigationOptions = ({ route, navigation }: INavigationOption) => { - const { authType } = route.params; - return { - headerLeft: () => , - title: ['saml', 'cas', 'iframe'].includes(authType) ? 'SSO' : 'OAuth' - }; - }; + // Force 3s delay so the server has time to evaluate the token + const debouncedLogin = useDebounce((params: ICredentials) => login(params), 3000); - constructor(props: IAuthenticationWebView) { - super(props); - this.state = { - logging: false, - loading: false - }; - this.oauthRedirectRegex = new RegExp(`(?=.*(${props.server}))(?=.*(credentialToken))(?=.*(credentialSecret))`, 'g'); - this.iframeRedirectRegex = new RegExp(`(?=.*(${props.server}))(?=.*(event|loginToken|token))`, 'g'); - } - - componentWillUnmount() { - if (this.debouncedLogin && this.debouncedLogin.stop) { - this.debouncedLogin.stop(); - } - } - - dismiss = () => { - const { navigation } = this.props; - navigation.pop(); - }; - - login = (params: ICredentials) => { - const { logging } = this.state; + const login = (params: ICredentials) => { if (logging) { return; } - - this.setState({ logging: true }); - + setLogging(true); try { Services.loginOAuthOrSso(params); } catch (e) { console.warn(e); } - this.setState({ logging: false }); - this.dismiss(); + setLogging(false); + navigation.pop(); }; - // Force 3s delay so the server has time to evaluate the token - debouncedLogin = debounce((params: ICredentials) => this.login(params), 3000); - - tryLogin = debounce( + const tryLogin = useDebounce( async () => { - const { Accounts_Iframe_api_url, Accounts_Iframe_api_method } = this.props; const data = await fetch(Accounts_Iframe_api_url, { method: Accounts_Iframe_api_method }).then(response => response.json()); const resume = data?.login || data?.loginToken; if (resume) { - this.login({ resume }); + login({ resume }); } }, 3000, - true + { leading: true } ); - onNavigationStateChange = (webViewState: WebViewNavigation | WebViewMessage) => { + const onNavigationStateChange = (webViewState: WebViewNavigation | WebViewMessage) => { const url = decodeURIComponent(webViewState.url); - const { route } = this.props; - const { authType } = route.params; if (authType === 'saml' || authType === 'cas') { - const { ssoToken } = route.params; const parsedUrl = parse(url, true); // ticket -> cas / validate & saml_idp_credentialToken -> saml if (parsedUrl.pathname?.includes('validate') || parsedUrl.query?.ticket || parsedUrl.query?.saml_idp_credentialToken) { @@ -140,28 +101,28 @@ class AuthenticationWebView extends React.PureComponent - - this.onNavigationStateChange(nativeEvent)} - onNavigationStateChange={this.onNavigationStateChange} - injectedJavaScript={isIframe ? injectedJavaScript : undefined} - onLoadStart={() => { - this.setState({ loading: true }); - }} - onLoadEnd={() => { - this.setState({ loading: false }); - }} - /> - {loading ? : null} - - ); - } -} + useLayoutEffect(() => { + navigation.setOptions({ + headerLeft: () => , + title: ['saml', 'cas', 'iframe'].includes(authType) ? 'SSO' : 'OAuth' + }); + }, [authType, navigation]); -const mapStateToProps = (state: IApplicationState) => ({ - server: state.server.server, - Accounts_Iframe_api_url: state.settings.Accounts_Iframe_api_url as string, - Accounts_Iframe_api_method: state.settings.Accounts_Iframe_api_method as string -}); + return ( + <> + + onNavigationStateChange(nativeEvent)} + onNavigationStateChange={onNavigationStateChange} + injectedJavaScript={isIframe ? injectedJavaScript : undefined} + onLoadStart={() => setLoading(true)} + onLoadEnd={() => setLoading(false)} + /> + {loading ? : null} + + ); +}; -export default connect(mapStateToProps)(withTheme(AuthenticationWebView)); +export default AuthenticationWebView; From 3f2e5ced197e9a56f70e106e72671d54213e4c16 Mon Sep 17 00:00:00 2001 From: Reinaldo Neto <47038980+reinaldonetof@users.noreply.github.com> Date: Wed, 18 Oct 2023 14:23:12 -0300 Subject: [PATCH 09/12] chore: migrate LoginView to hooks (#5092) * chore: migrate LoginView to hooks * minor tweak * minor tweak * fix ref --------- Co-authored-by: GleidsonDaniel --- .../TextInput/ControlledFormTextInput.tsx | 6 +- app/views/LoginView.tsx | 264 ------------------ app/views/LoginView/UserForm.tsx | 171 ++++++++++++ app/views/LoginView/index.tsx | 40 +++ app/views/LoginView/styles.ts | 36 +++ 5 files changed, 251 insertions(+), 266 deletions(-) delete mode 100644 app/views/LoginView.tsx create mode 100644 app/views/LoginView/UserForm.tsx create mode 100644 app/views/LoginView/index.tsx create mode 100644 app/views/LoginView/styles.ts diff --git a/app/containers/TextInput/ControlledFormTextInput.tsx b/app/containers/TextInput/ControlledFormTextInput.tsx index 5b69b0e20..d2aa1d446 100644 --- a/app/containers/TextInput/ControlledFormTextInput.tsx +++ b/app/containers/TextInput/ControlledFormTextInput.tsx @@ -3,7 +3,7 @@ import { Control, Controller } from 'react-hook-form'; import { FormTextInput, IRCTextInputProps } from './FormTextInput'; -interface IControlledFormTextInputProps extends IRCTextInputProps { +interface IControlledFormTextInputProps extends Omit { control: Control; name: string; } @@ -12,6 +12,8 @@ export const ControlledFormTextInput = ({ control, name, ...props }: IControlled } + render={({ field: { onChange, value, ref } }) => ( + + )} /> ); diff --git a/app/views/LoginView.tsx b/app/views/LoginView.tsx deleted file mode 100644 index b43830f73..000000000 --- a/app/views/LoginView.tsx +++ /dev/null @@ -1,264 +0,0 @@ -import { dequal } from 'dequal'; -import React from 'react'; -import { Alert, Keyboard, StyleSheet, Text, View, TextInput as RNTextInput } from 'react-native'; -import { connect } from 'react-redux'; - -import { loginRequest } from '../actions/login'; -import { themes } from '../lib/constants'; -import Button from '../containers/Button'; -import FormContainer, { FormContainerInner } from '../containers/FormContainer'; -import * as HeaderButton from '../containers/HeaderButton'; -import LoginServices from '../containers/LoginServices'; -import { FormTextInput } from '../containers/TextInput'; -import { IApplicationState, IBaseScreen } from '../definitions'; -import I18n from '../i18n'; -import { OutsideParamList } from '../stacks/types'; -import { withTheme } from '../theme'; -import sharedStyles from './Styles'; -import UGCRules from '../containers/UserGeneratedContentRules'; - -const styles = StyleSheet.create({ - registerDisabled: { - ...sharedStyles.textRegular, - ...sharedStyles.textAlignCenter, - fontSize: 16 - }, - title: { - ...sharedStyles.textBold, - fontSize: 22 - }, - inputContainer: { - marginVertical: 16 - }, - bottomContainer: { - flexDirection: 'column', - alignItems: 'center' - }, - bottomContainerText: { - ...sharedStyles.textRegular, - fontSize: 13 - }, - bottomContainerTextBold: { - ...sharedStyles.textSemibold, - fontSize: 13 - }, - loginButton: { - marginTop: 16 - }, - ugcContainer: { - marginTop: 32 - } -}); - -interface ILoginViewProps extends IBaseScreen { - Site_Name: string; - Accounts_RegistrationForm: string; - Accounts_RegistrationForm_LinkReplacementText: string; - Accounts_EmailOrUsernamePlaceholder: string; - Accounts_PasswordPlaceholder: string; - Accounts_PasswordReset: boolean; - Accounts_ShowFormLogin: boolean; - isFetching: boolean; - error: { - error: string; - }; - failure: boolean; - loginRequest: Function; - inviteLinkToken: string; -} - -interface ILoginViewState { - user: string; - password: string; -} - -class LoginView extends React.Component { - private passwordInput: RNTextInput | null | undefined; - - static navigationOptions = ({ route, navigation }: ILoginViewProps) => ({ - title: route?.params?.title ?? 'Rocket.Chat', - headerRight: () => - }); - - constructor(props: ILoginViewProps) { - super(props); - this.state = { - user: props.route.params?.username ?? '', - password: '' - }; - } - - UNSAFE_componentWillReceiveProps(nextProps: ILoginViewProps) { - const { error } = this.props; - if (nextProps.failure && !dequal(error, nextProps.error)) { - if (nextProps.error?.error === 'error-invalid-email') { - this.resendEmailConfirmation(); - } else { - Alert.alert(I18n.t('Oops'), I18n.t('Login_error')); - } - } - } - - get showRegistrationButton() { - const { Accounts_RegistrationForm, inviteLinkToken } = this.props; - return Accounts_RegistrationForm === 'Public' || (Accounts_RegistrationForm === 'Secret URL' && inviteLinkToken?.length); - } - - login = () => { - const { navigation, Site_Name } = this.props; - navigation.navigate('LoginView', { title: Site_Name }); - }; - - register = () => { - const { navigation, Site_Name } = this.props; - navigation.navigate('RegisterView', { title: Site_Name }); - }; - - forgotPassword = () => { - const { navigation, Site_Name } = this.props; - navigation.navigate('ForgotPasswordView', { title: Site_Name }); - }; - - resendEmailConfirmation = () => { - const { user } = this.state; - const { navigation } = this.props; - navigation.navigate('SendEmailConfirmationView', { user }); - }; - - valid = () => { - const { user, password } = this.state; - return user.trim() && password.trim(); - }; - - submit = () => { - if (!this.valid()) { - return; - } - - const { user, password } = this.state; - const { dispatch } = this.props; - Keyboard.dismiss(); - dispatch(loginRequest({ user, password })); - }; - - renderUserForm = () => { - const { user } = this.state; - const { - Accounts_EmailOrUsernamePlaceholder, - Accounts_PasswordPlaceholder, - Accounts_PasswordReset, - Accounts_RegistrationForm_LinkReplacementText, - isFetching, - theme, - Accounts_ShowFormLogin - } = this.props; - - if (!Accounts_ShowFormLogin) { - return null; - } - - return ( - <> - {I18n.t('Login')} - this.setState({ user: value })} - onSubmitEditing={() => { - this.passwordInput?.focus(); - }} - testID='login-view-email' - textContentType='username' - autoComplete='username' - value={user} - /> - { - this.passwordInput = e; - }} - placeholder={Accounts_PasswordPlaceholder || I18n.t('Password')} - returnKeyType='send' - secureTextEntry - onSubmitEditing={this.submit} - onChangeText={(value: string) => this.setState({ password: value })} - testID='login-view-password' - textContentType='password' - autoComplete='password' - /> -