forked from Ivasoft/mattermost-mobile
224 lines
6.8 KiB
TypeScript
224 lines
6.8 KiB
TypeScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import React, {useCallback, useEffect, useRef, useState} from 'react';
|
|
import {useIntl} from 'react-intl';
|
|
import {
|
|
ActivityIndicator,
|
|
Image,
|
|
Keyboard,
|
|
KeyboardAvoidingView,
|
|
Platform,
|
|
TextInput,
|
|
TouchableWithoutFeedback,
|
|
View,
|
|
} from 'react-native';
|
|
import Button from 'react-native-button';
|
|
import {SafeAreaView} from 'react-native-safe-area-context';
|
|
|
|
import ErrorText from '@components/error_text';
|
|
import FormattedText from '@components/formatted_text';
|
|
import {login} from '@actions/remote/user';
|
|
import {Config} from '@typings/database/models/servers/config';
|
|
import {License} from '@typings/database/models/servers/license';
|
|
import {t} from '@utils/i18n';
|
|
import {preventDoubleTap} from '@utils/tap';
|
|
import {changeOpacity, makeStyleSheetFromTheme} from '@utils/theme';
|
|
|
|
type MFAProps = {
|
|
config: Partial<Config>;
|
|
goToChannel: () => void;
|
|
license: Partial<License>;
|
|
loginId: string;
|
|
password: string;
|
|
serverUrl: string;
|
|
theme: Theme;
|
|
}
|
|
|
|
const MFA = ({config, goToChannel, license, loginId, password, serverUrl, theme}: MFAProps) => {
|
|
const [token, setToken] = useState<string>('');
|
|
const [error, setError] = useState<string>('');
|
|
const [isLoading, setIsLoading] = useState<boolean>(false);
|
|
const {formatMessage} = useIntl();
|
|
const textInputRef = useRef<TextInput>(null);
|
|
|
|
const styles = getStyleSheet(theme);
|
|
|
|
const onBlur = useCallback(() => {
|
|
textInputRef?.current?.blur();
|
|
}, []);
|
|
|
|
useEffect(() => {
|
|
if (Platform.OS === 'android') {
|
|
Keyboard.addListener('keyboardDidHide', onBlur);
|
|
}
|
|
return () => {
|
|
if (Platform.OS === 'android') {
|
|
Keyboard.removeListener('keyboardDidHide', onBlur);
|
|
}
|
|
};
|
|
}, []);
|
|
|
|
const handleInput = (userToken: string) => {
|
|
setToken(userToken);
|
|
setError('');
|
|
};
|
|
|
|
const submit = preventDoubleTap(async () => {
|
|
Keyboard.dismiss();
|
|
if (!token) {
|
|
setError(
|
|
formatMessage({
|
|
id: t('login_mfa.tokenReq'),
|
|
defaultMessage: 'Please enter an MFA token',
|
|
}),
|
|
);
|
|
return;
|
|
}
|
|
setIsLoading(true);
|
|
const result = await login(serverUrl, {loginId, password, mfaToken: token, config, license});
|
|
setIsLoading(false);
|
|
if (result?.error) {
|
|
setError(result?.error);
|
|
return;
|
|
}
|
|
goToChannel();
|
|
});
|
|
|
|
const getProceedView = () => {
|
|
if (isLoading) {
|
|
return (
|
|
<ActivityIndicator
|
|
animating={true}
|
|
size='small'
|
|
/>
|
|
);
|
|
}
|
|
return (
|
|
<Button
|
|
testID={'login_mfa.submit'}
|
|
onPress={submit}
|
|
containerStyle={styles.signupButton}
|
|
>
|
|
<FormattedText
|
|
style={styles.signupButtonText}
|
|
id='mobile.components.select_server_view.proceed'
|
|
defaultMessage='Proceed'
|
|
/>
|
|
</Button>
|
|
);
|
|
};
|
|
|
|
return (
|
|
<SafeAreaView
|
|
testID='mfa.screen'
|
|
style={styles.flex}
|
|
>
|
|
<KeyboardAvoidingView
|
|
behavior='padding'
|
|
style={styles.flex}
|
|
keyboardVerticalOffset={5}
|
|
enabled={Platform.OS === 'ios'}
|
|
>
|
|
<TouchableWithoutFeedback onPress={onBlur}>
|
|
<View style={[styles.container, styles.signupContainer]}>
|
|
<Image
|
|
source={require('@assets/images/logo.png')}
|
|
style={styles.containerImage}
|
|
/>
|
|
<View>
|
|
<FormattedText
|
|
style={[styles.header, styles.label]}
|
|
id='login_mfa.enterToken'
|
|
defaultMessage="To complete the sign in process, please enter a token from your smartphone's authenticator"
|
|
/>
|
|
</View>
|
|
<ErrorText
|
|
error={error}
|
|
testID='mfa.error.text'
|
|
theme={theme}
|
|
/>
|
|
<TextInput
|
|
testID={'login_mfa.input'}
|
|
ref={textInputRef}
|
|
value={token}
|
|
onChangeText={handleInput}
|
|
onSubmitEditing={submit}
|
|
style={styles.inputBox}
|
|
autoCapitalize='none'
|
|
autoCorrect={false}
|
|
keyboardType='numeric'
|
|
placeholder={formatMessage({id: 'login_mfa.token', defaultMessage: 'MFA Token'})}
|
|
placeholderTextColor={changeOpacity(theme.centerChannelColor, 0.5)}
|
|
returnKeyType='go'
|
|
underlineColorAndroid='transparent'
|
|
disableFullscreenUI={true}
|
|
/>
|
|
{getProceedView()}
|
|
</View>
|
|
</TouchableWithoutFeedback>
|
|
</KeyboardAvoidingView>
|
|
</SafeAreaView>
|
|
);
|
|
};
|
|
|
|
const getStyleSheet = makeStyleSheetFromTheme((theme: Theme) => ({
|
|
flex: {
|
|
flex: 1,
|
|
},
|
|
signupButton: {
|
|
borderRadius: 3,
|
|
borderColor: theme.buttonBg,
|
|
borderWidth: 1,
|
|
alignItems: 'center',
|
|
alignSelf: 'stretch',
|
|
marginTop: 10,
|
|
padding: 15,
|
|
},
|
|
inputBox: {
|
|
fontSize: 16,
|
|
height: 45,
|
|
borderColor: 'gainsboro',
|
|
borderWidth: 1,
|
|
marginTop: 5,
|
|
marginBottom: 5,
|
|
paddingLeft: 10,
|
|
alignSelf: 'stretch',
|
|
borderRadius: 3,
|
|
color: theme.centerChannelColor,
|
|
},
|
|
header: {
|
|
textAlign: 'center',
|
|
marginTop: 15,
|
|
marginBottom: 15,
|
|
fontSize: 32,
|
|
fontWeight: '600',
|
|
},
|
|
label: {
|
|
color: changeOpacity(theme.centerChannelColor, 0.6),
|
|
fontSize: 20,
|
|
fontWeight: '400',
|
|
},
|
|
container: {
|
|
flex: 1,
|
|
flexDirection: 'column',
|
|
justifyContent: 'center',
|
|
alignItems: 'center',
|
|
},
|
|
signupContainer: {
|
|
paddingRight: 15,
|
|
paddingLeft: 15,
|
|
},
|
|
signupButtonText: {
|
|
color: theme.buttonBg,
|
|
textAlign: 'center',
|
|
fontSize: 17,
|
|
},
|
|
containerImage: {
|
|
height: 72,
|
|
resizeMode: 'contain',
|
|
},
|
|
}));
|
|
|
|
export default MFA;
|