Skip to content

Commit

Permalink
Merge branch 'develop' into fix.pdf-view
Browse files Browse the repository at this point in the history
  • Loading branch information
diegolmello authored Jun 7, 2024
2 parents 07f0805 + 2e47590 commit 72c684b
Show file tree
Hide file tree
Showing 26 changed files with 401 additions and 116 deletions.
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import React, { useState } from 'react';
import React, { useState, useRef } from 'react';
import { StyleSheet, Text, View } from 'react-native';

import { CustomIcon, TIconsName } from '../../CustomIcon';
Expand Down Expand Up @@ -42,22 +42,28 @@ const FooterButtons = ({
confirmTitle = '',
disabled = false,
cancelBackgroundColor = '',
confirmBackgroundColor = ''
confirmBackgroundColor = '',
testID = ''
}): React.ReactElement => {
const { colors } = useTheme();
return (
<View style={styles.footerButtonsContainer}>
<Button
style={[styles.buttonSeparator, { flex: 1, backgroundColor: cancelBackgroundColor || colors.buttonBackgroundSecondaryDefault }]}
style={[
styles.buttonSeparator,
{ flex: 1, backgroundColor: cancelBackgroundColor || colors.buttonBackgroundSecondaryDefault }
]}
color={colors.backdropColor}
title={cancelTitle}
onPress={cancelAction}
testID={`${testID}-cancel`}
/>
<Button
style={{ flex: 1, backgroundColor: confirmBackgroundColor || colors.buttonBackgroundDangerDefault }}
title={confirmTitle}
onPress={confirmAction}
disabled={disabled}
testID={`${testID}-confirm`}
/>
</View>
);
Expand All @@ -76,62 +82,104 @@ const ActionSheetContentWithInputAndSubmit = ({
iconColor,
customText,
confirmBackgroundColor,
showInput = true
showInput = true,
inputs = [],
isDisabled
}: {
onSubmit: (inputValue: string) => void;
onSubmit: (inputValue: string | string[]) => void;
onCancel?: () => void;
title: string;
description: string;
description?: string;
testID: string;
secureTextEntry?: boolean;
placeholder: string;
placeholder?: string;
confirmTitle?: string;
iconName?: TIconsName;
iconColor?: string;
customText?: React.ReactElement;
confirmBackgroundColor?: string;
showInput?: boolean;
inputs?: { placeholder: string; secureTextEntry?: boolean; key: string }[];
isDisabled?: (inputValues: string[]) => boolean;
}): React.ReactElement => {
const { colors } = useTheme();
const [inputValue, setInputValue] = useState('');
const [inputValues, setInputValues] = useState(inputs.map(() => ''));
const inputRefs = useRef(inputs.map(() => React.createRef()));

const handleInputChange = (value: string, index: number) => {
const newInputValues = [...inputValues];
newInputValues[index] = value;
setInputValues(newInputValues);
};

const { hideActionSheet } = useActionSheet();

const renderInputs = () => {
if (inputs.length > 0) {
return inputs.map((inputConfig, index) => (
<FormTextInput
key={inputConfig.key}
value={inputValues[index]}
placeholder={inputConfig.placeholder}
onChangeText={value => handleInputChange(value, index)}
onSubmitEditing={() => {
if (index < inputs.length - 1) {
(inputRefs.current[index + 1] as any).current.focus();
} else {
setTimeout(() => {
hideActionSheet();
}, 100);
if (inputValues.every(value => value)) onSubmit(inputValues);
}
}}
inputRef={inputRefs.current[index] as any}
testID={`${testID}-input-${inputConfig.key}`}
secureTextEntry={inputConfig.secureTextEntry}
bottomSheet={isIOS}
/>
));
}

return (
<FormTextInput
value={inputValues[0]}
placeholder={placeholder}
onChangeText={value => handleInputChange(value, 0)}
onSubmitEditing={() => {
setTimeout(() => {
hideActionSheet();
}, 100);
if (inputValues[0]) onSubmit(inputValues[0]);
}}
testID={`${testID}-input`}
secureTextEntry={secureTextEntry}
bottomSheet={isIOS}
/>
);
};

const defaultDisabled = showInput && inputValues.some(value => !value);
const disabled = isDisabled ? isDisabled(inputValues) : defaultDisabled;

return (
<View style={sharedStyles.containerScrollView} testID='action-sheet-content-with-input-and-submit'>
<>
<View style={styles.titleContainer}>
{iconName ? <CustomIcon name={iconName} size={32} color={iconColor || colors.buttonBackgroundDangerDefault} /> : null}
<Text style={[styles.titleContainerText, { color: colors.fontDefault, paddingLeft: iconName ? 16 : 0 }]}>
{title}
</Text>
<Text style={[styles.titleContainerText, { color: colors.fontDefault, paddingLeft: iconName ? 16 : 0 }]}>{title}</Text>
</View>
<Text style={[styles.subtitleText, { color: colors.fontTitlesLabels }]}>{description}</Text>
{description ? <Text style={[styles.subtitleText, { color: colors.fontTitlesLabels }]}>{description}</Text> : null}
{customText}
</>
{showInput ? (
<FormTextInput
value={inputValue}
placeholder={placeholder}
onChangeText={value => setInputValue(value)}
onSubmitEditing={() => {
// fix android animation
setTimeout(() => {
hideActionSheet();
}, 100);
if (inputValue) onSubmit(inputValue);
}}
testID={testID}
secureTextEntry={secureTextEntry}
bottomSheet={isIOS}
/>
) : null}
{showInput ? renderInputs() : null}
<FooterButtons
confirmBackgroundColor={confirmBackgroundColor || colors.fontHint}
cancelAction={onCancel || hideActionSheet}
confirmAction={() => onSubmit(inputValue)}
confirmAction={() => onSubmit(inputs.length > 0 ? inputValues : inputValues[0])}
cancelTitle={i18n.t('Cancel')}
confirmTitle={confirmTitle || i18n.t('Save')}
disabled={!showInput ? false : !inputValue}
disabled={disabled}
testID={testID}
/>
</View>
);
Expand Down
107 changes: 107 additions & 0 deletions app/containers/ChangePasswordRequired.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,107 @@
import React, { useState } from 'react';
import { StyleSheet, Text, View } from 'react-native';
import { useDispatch } from 'react-redux';

import { logout, setUser } from '../actions/login';
import I18n from '../i18n';
import { useSetting } from '../lib/hooks/useSetting';
import { showErrorAlert } from '../lib/methods/helpers';
import { Services } from '../lib/services';
import { useTheme } from '../theme';
import sharedStyles from '../views/Styles';
import { useActionSheet } from './ActionSheet';
import ActionSheetContentWithInputAndSubmit from './ActionSheet/ActionSheetContentWithInputAndSubmit';
import Button from './Button';
import { CustomIcon } from './CustomIcon';

export const ChangePasswordRequired = () => {
const [loading, setLoading] = useState(false);
const { colors } = useTheme();
const dispatch = useDispatch();
const { showActionSheet, hideActionSheet } = useActionSheet();

const requiresPasswordConfirmation = useSetting('Accounts_RequirePasswordConfirmation');
const passwordPlaceholder = useSetting('Accounts_PasswordPlaceholder') as string;
const passwordConfirmationPlaceholder = useSetting('Accounts_ConfirmPasswordPlaceholder') as string;

const changePassword = async (password: string) => {
setLoading(true);
try {
await Services.setUserPassword(password);
dispatch(setUser({ requirePasswordChange: false }));
hideActionSheet();
} catch (error: any) {
showErrorAlert(error?.reason || error?.message, I18n.t('Oops'));
}
setLoading(false);
};

const showActionSheetPassword = () => {
const inputs = [{ placeholder: passwordPlaceholder || I18n.t('Password'), secureTextEntry: true, key: 'password' }];
if (requiresPasswordConfirmation) {
inputs.push({
placeholder: passwordConfirmationPlaceholder || I18n.t('Confirm_your_password'),
secureTextEntry: true,
key: 'confirm-password'
});
}
showActionSheet({
children: (
<ActionSheetContentWithInputAndSubmit
title={I18n.t('Please_Enter_your_new_password')}
testID='change-password-required-sheet'
inputs={inputs}
onSubmit={input => changePassword(input[0])}
isDisabled={input => (loading || input[0] === '' || requiresPasswordConfirmation ? input[0] !== input[1] : false)}
/>
)
});
};

return (
<View style={[styles.container, { paddingTop: 120, backgroundColor: colors.surfaceLight }]}>
<View style={styles.iconContainer}>
<CustomIcon name='info' size={36} color={colors.statusFontWarning} />
</View>
<Text style={[styles.title, { color: colors.fontTitlesLabels }]}>{I18n.t('You_need_to_change_your_password')}</Text>
<Text style={[styles.description, { color: colors.fontDefault }]}>{I18n.t('To_continue_using_RocketChat')}</Text>
<Button
testID='change-password-required-button'
title={I18n.t('Change_password')}
type='primary'
onPress={showActionSheetPassword}
/>
<Button
testID='change-password-required-logout'
title={I18n.t('Logout')}
type='secondary'
backgroundColor={colors.surfaceTint}
onPress={() => dispatch(logout())}
/>
</View>
);
};

const styles = StyleSheet.create({
container: {
flex: 1,
padding: 16,
backgroundColor: '#fff'
},
iconContainer: {
alignItems: 'center',
padding: 24
},
title: {
fontSize: 20,
lineHeight: 30,
marginBottom: 24,
...sharedStyles.textBold
},
description: {
fontSize: 16,
lineHeight: 24,
marginBottom: 24,
...sharedStyles.textRegular
}
});
45 changes: 22 additions & 23 deletions app/containers/MessageComposer/hooks/useChooseMedia.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { Alert } from 'react-native';
import DocumentPicker from 'react-native-document-picker';
import * as DocumentPicker from 'expo-document-picker';

import { IMAGE_PICKER_CONFIG, LIBRARY_PICKER_CONFIG, VIDEO_PICKER_CONFIG } from '../constants';
import { forceJpgExtension } from '../helpers';
Expand Down Expand Up @@ -89,29 +89,28 @@ export const useChooseMedia = ({

const chooseFile = async () => {
try {
const res = await DocumentPicker.pickSingle({
type: [DocumentPicker.types.allFiles]
});
const file = {
filename: res.name,
size: res.size,
mime: res.type,
path: res.uri
} as any;
const canUploadResult = canUploadFile({
file,
allowList,
maxFileSize,
permissionToUploadFile: permissionToUpload
});
if (canUploadResult.success) {
return openShareView([file]);
}
handleError(canUploadResult.error);
} catch (e: any) {
if (!DocumentPicker.isCancel(e)) {
log(e);
const res = await DocumentPicker.getDocumentAsync({ copyToCacheDirectory: false });
if (!res.canceled) {
const [asset] = res.assets;
const file = {
filename: asset.name,
size: asset.size,
mime: asset.mimeType,
path: asset.uri
} as any;
const canUploadResult = canUploadFile({
file,
allowList,
maxFileSize,
permissionToUploadFile: permissionToUpload
});
if (canUploadResult.success) {
return openShareView([file]);
}
handleError(canUploadResult.error);
}
} catch (e) {
log(e);
}
};

Expand Down
1 change: 1 addition & 0 deletions app/definitions/ILoggedUser.ts
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ export interface ILoggedUser {
alsoSendThreadToChannel: 'default' | 'always' | 'never';
bio?: string;
nickname?: string;
requirePasswordChange?: boolean;
}

export interface ILoggedUserResultFromServer
Expand Down
5 changes: 5 additions & 0 deletions app/i18n/locales/en.json
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,7 @@
"Certificate_password": "Certificate password",
"Change_Language": "Change language",
"Change_language_loading": "Changing language.",
"Change_password": "Change password",
"changed_room_announcement": "changed room announcement to: {{announcement}}",
"changed_room_description": "changed room description to: {{description}}",
"changing_avatar": "changing avatar",
Expand Down Expand Up @@ -127,6 +128,7 @@
"Condensed": "Condensed",
"conference_call": "Conference call",
"Confirm": "Confirm",
"Confirm_your_password": "Confirm your password",
"Confirmation": "Confirmation",
"Connect": "Connect",
"Connecting": "Connecting...",
Expand Down Expand Up @@ -523,6 +525,7 @@
"Pinned_a_message": "Pinned a message:",
"Place_chat_on_hold": "Place chat on hold",
"Please_add_a_comment": "Please add a comment",
"Please_Enter_your_new_password": "Please enter your new password",
"Please_enter_your_password": "Please enter your password",
"Please_wait": "Please wait.",
"Preferences": "Preferences",
Expand Down Expand Up @@ -724,6 +727,7 @@
"Threads_displaying_following": "Displaying following",
"Threads_displaying_unread": "Displaying unread",
"Timezone": "Timezone",
"To_continue_using_RocketChat": "To continue using the mobile app, you need to change your password.",
"Token_expired": "Your session has expired. Please log in again.",
"Topic": "Topic",
"topic": "topic",
Expand Down Expand Up @@ -822,6 +826,7 @@
"You_colon": "You: ",
"You_dont_have_permission_to_perform_this_action": "You don’t have permission to perform this action. Check with a workspace administrator.",
"You_need_to_access_at_least_one_RocketChat_server_to_share_something": "You need to access at least one Rocket.Chat workspace to share something.",
"You_need_to_change_your_password": "You need to change your password",
"You_need_to_verifiy_your_email_address_to_get_notications": "You need to verify your email address to get notifications",
"you_were_mentioned": "you were mentioned",
"You_were_removed_from_channel": "You were removed from {{channel}}",
Expand Down
Loading

0 comments on commit 72c684b

Please sign in to comment.