Chore: Hooks app/views/LanguageView (#4537)

* Chore: Hooks app/views/LanguageView

* minor tweak app selector
This commit is contained in:
Reinaldo Neto 2022-09-22 18:40:32 -03:00 committed by GitHub
parent b3a286329c
commit 9d69c10bbd
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 75 additions and 96 deletions

View File

@ -172,7 +172,7 @@ const SettingsStackNavigator = () => {
component={E2EEncryptionSecurityView} component={E2EEncryptionSecurityView}
options={E2EEncryptionSecurityView.navigationOptions} options={E2EEncryptionSecurityView.navigationOptions}
/> />
<SettingsStack.Screen name='LanguageView' component={LanguageView} options={LanguageView.navigationOptions} /> <SettingsStack.Screen name='LanguageView' component={LanguageView} />
<SettingsStack.Screen name='ThemeView' component={ThemeView} /> <SettingsStack.Screen name='ThemeView' component={ThemeView} />
<SettingsStack.Screen name='DefaultBrowserView' component={DefaultBrowserView} /> <SettingsStack.Screen name='DefaultBrowserView' component={DefaultBrowserView} />
<SettingsStack.Screen <SettingsStack.Screen

View File

@ -175,7 +175,7 @@ const ModalStackNavigator = React.memo(({ navigation }: INavigation) => {
component={SettingsView} component={SettingsView}
options={props => SettingsView.navigationOptions!({ ...props, isMasterDetail: true })} options={props => SettingsView.navigationOptions!({ ...props, isMasterDetail: true })}
/> />
<ModalStack.Screen name='LanguageView' component={LanguageView} options={LanguageView.navigationOptions} /> <ModalStack.Screen name='LanguageView' component={LanguageView} />
<ModalStack.Screen name='ThemeView' component={ThemeView} /> <ModalStack.Screen name='ThemeView' component={ThemeView} />
<ModalStack.Screen name='DefaultBrowserView' component={DefaultBrowserView} /> <ModalStack.Screen name='DefaultBrowserView' component={DefaultBrowserView} />
<ModalStack.Screen <ModalStack.Screen

View File

@ -0,0 +1,31 @@
import React from 'react';
import * as List from '../../containers/List';
import { useTheme } from '../../theme';
const LanguageItem = ({
item,
language,
submit
}: {
item: { value: string; label: string };
language: string;
submit: (language: string) => Promise<void>;
}) => {
const { colors } = useTheme();
const { value, label } = item;
const isSelected = language === value;
return (
<List.Item
title={label}
onPress={() => submit(value)}
testID={`language-view-${value}`}
right={() => (isSelected ? <List.Icon name='check' color={colors.tintColor} /> : null)}
translateTitle={false}
/>
);
};
export default LanguageItem;

View File

@ -1,77 +1,53 @@
import React from 'react'; import React, { useLayoutEffect } from 'react';
import { FlatList } from 'react-native'; import { FlatList } from 'react-native';
import RNRestart from 'react-native-restart'; import RNRestart from 'react-native-restart';
import { connect } from 'react-redux'; import { useDispatch } from 'react-redux';
import { useNavigation } from '@react-navigation/native';
import { StackNavigationProp } from '@react-navigation/stack';
import { useAppSelector } from '../../lib/hooks';
import { appStart } from '../../actions/app'; import { appStart } from '../../actions/app';
import { setUser } from '../../actions/login'; import { setUser } from '../../actions/login';
import { themes } from '../../lib/constants';
import * as List from '../../containers/List'; import * as List from '../../containers/List';
import SafeAreaView from '../../containers/SafeAreaView'; import SafeAreaView from '../../containers/SafeAreaView';
import StatusBar from '../../containers/StatusBar'; import StatusBar from '../../containers/StatusBar';
import { IApplicationState, IBaseScreen, IUser, RootEnum } from '../../definitions'; import { RootEnum } from '../../definitions';
import I18n, { isRTL, LANGUAGES } from '../../i18n'; import I18n, { isRTL, LANGUAGES } from '../../i18n';
import database from '../../lib/database'; import database from '../../lib/database';
import { getUserSelector } from '../../selectors/login'; import { getUserSelector } from '../../selectors/login';
import { SettingsStackParamList } from '../../stacks/types'; import { SettingsStackParamList } from '../../stacks/types';
import { withTheme } from '../../theme';
import { showErrorAlert } from '../../lib/methods/helpers/info'; import { showErrorAlert } from '../../lib/methods/helpers/info';
import log, { events, logEvent } from '../../lib/methods/helpers/log'; import log, { events, logEvent } from '../../lib/methods/helpers/log';
import { Services } from '../../lib/services'; import { Services } from '../../lib/services';
import LanguageItem from './LanguageItem';
interface ILanguageViewProps extends IBaseScreen<SettingsStackParamList, 'LanguageView'> { const LanguageView = () => {
user: IUser; const { languageDefault, id } = useAppSelector(state => ({
} languageDefault: getUserSelector(state).language,
id: getUserSelector(state).id
}));
const language = languageDefault || 'en';
interface ILanguageViewState { const dispatch = useDispatch();
language: string; const navigation = useNavigation<StackNavigationProp<SettingsStackParamList, 'LanguageView'>>();
}
class LanguageView extends React.Component<ILanguageViewProps, ILanguageViewState> { useLayoutEffect(() => {
static navigationOptions = () => ({ navigation.setOptions({
title: I18n.t('Change_Language') title: I18n.t('Change_Language')
}); });
}, [navigation]);
constructor(props: ILanguageViewProps) { const submit = async (language: string) => {
super(props); if (languageDefault === language) {
this.state = {
language: props.user ? (props.user.language as string) : 'en'
};
}
shouldComponentUpdate(nextProps: ILanguageViewProps, nextState: ILanguageViewState) {
const { language } = this.state;
const { user, theme } = this.props;
if (nextProps.theme !== theme) {
return true;
}
if (nextState.language !== language) {
return true;
}
if (nextProps.user.language !== user.language) {
return true;
}
return false;
}
formIsChanged = (language: string) => {
const { user } = this.props;
return user.language !== language;
};
submit = async (language: string) => {
if (!this.formIsChanged(language)) {
return; return;
} }
const { dispatch, user } = this.props; const shouldRestart = isRTL(language) || isRTL(languageDefault);
const shouldRestart = isRTL(language) || isRTL(user.language);
dispatch(appStart({ root: RootEnum.ROOT_LOADING, text: I18n.t('Change_language_loading') })); dispatch(appStart({ root: RootEnum.ROOT_LOADING, text: I18n.t('Change_language_loading') }));
// shows loading for at least 300ms // shows loading for at least 300ms
await Promise.all([this.changeLanguage(language), new Promise(resolve => setTimeout(resolve, 300))]); await Promise.all([changeLanguage(language), new Promise(resolve => setTimeout(resolve, 300))]);
if (shouldRestart) { if (shouldRestart) {
await RNRestart.Restart(); await RNRestart.Restart();
@ -80,14 +56,13 @@ class LanguageView extends React.Component<ILanguageViewProps, ILanguageViewStat
} }
}; };
changeLanguage = async (language: string) => { const changeLanguage = async (language: string) => {
logEvent(events.LANG_SET_LANGUAGE); logEvent(events.LANG_SET_LANGUAGE);
const { user, dispatch } = this.props;
const params: { language?: string } = {}; const params: { language?: string } = {};
// language // language
if (user.language !== language) { if (languageDefault !== language) {
params.language = language; params.language = language;
} }
@ -99,7 +74,7 @@ class LanguageView extends React.Component<ILanguageViewProps, ILanguageViewStat
const usersCollection = serversDB.get('users'); const usersCollection = serversDB.get('users');
await serversDB.write(async () => { await serversDB.write(async () => {
try { try {
const userRecord = await usersCollection.find(user.id); const userRecord = await usersCollection.find(id);
await userRecord.update(record => { await userRecord.update(record => {
record.language = params.language; record.language = params.language;
}); });
@ -114,28 +89,6 @@ class LanguageView extends React.Component<ILanguageViewProps, ILanguageViewStat
} }
}; };
renderIcon = () => {
const { theme } = this.props;
return <List.Icon name='check' color={themes[theme].tintColor} />;
};
renderItem = ({ item }: { item: { value: string; label: string } }) => {
const { value, label } = item;
const { language } = this.state;
const isSelected = language === value;
return (
<List.Item
title={label}
onPress={() => this.submit(value)}
testID={`language-view-${value}`}
right={() => (isSelected ? this.renderIcon() : null)}
translateTitle={false}
/>
);
};
render() {
return ( return (
<SafeAreaView testID='language-view'> <SafeAreaView testID='language-view'>
<StatusBar /> <StatusBar />
@ -145,16 +98,11 @@ class LanguageView extends React.Component<ILanguageViewProps, ILanguageViewStat
ListHeaderComponent={List.Separator} ListHeaderComponent={List.Separator}
ListFooterComponent={List.Separator} ListFooterComponent={List.Separator}
contentContainerStyle={List.styles.contentContainerStyleFlatList} contentContainerStyle={List.styles.contentContainerStyleFlatList}
renderItem={this.renderItem} renderItem={({ item }) => <LanguageItem item={item} language={language} submit={submit} />}
ItemSeparatorComponent={List.Separator} ItemSeparatorComponent={List.Separator}
/> />
</SafeAreaView> </SafeAreaView>
); );
} };
}
const mapStateToProps = (state: IApplicationState) => ({ export default LanguageView;
user: getUserSelector(state)
});
export default connect(mapStateToProps)(withTheme(LanguageView));