forked from Ivasoft/mattermost-mobile
256 lines
8.7 KiB
TypeScript
256 lines
8.7 KiB
TypeScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import React, {useEffect, useMemo, useRef} from 'react';
|
|
import {Platform, useWindowDimensions, View} from 'react-native';
|
|
import {KeyboardAwareScrollView} from 'react-native-keyboard-aware-scroll-view';
|
|
import {Navigation} from 'react-native-navigation';
|
|
import Animated, {useAnimatedStyle, useSharedValue, withTiming} from 'react-native-reanimated';
|
|
import {SafeAreaView} from 'react-native-safe-area-context';
|
|
|
|
import FormattedText from '@components/formatted_text';
|
|
import {Screens} from '@constants';
|
|
import useAndroidHardwareBackHandler from '@hooks/android_back_handler';
|
|
import {useIsTablet} from '@hooks/device';
|
|
import useNavButtonPressed from '@hooks/navigation_button_pressed';
|
|
import NetworkManager from '@managers/network_manager';
|
|
import Background from '@screens/background';
|
|
import {dismissModal, goToScreen, loginAnimationOptions, popTopScreen} from '@screens/navigation';
|
|
import {preventDoubleTap} from '@utils/tap';
|
|
import {changeOpacity, makeStyleSheetFromTheme} from '@utils/theme';
|
|
import {typography} from '@utils/typography';
|
|
|
|
import Form from './form';
|
|
import LoginOptionsSeparator from './login_options_separator';
|
|
import SsoOptions from './sso_options';
|
|
|
|
import type {LaunchProps} from '@typings/launch';
|
|
import type {AvailableScreens} from '@typings/screens/navigation';
|
|
|
|
export interface LoginOptionsProps extends LaunchProps {
|
|
closeButtonId?: string;
|
|
componentId: AvailableScreens;
|
|
config: ClientConfig;
|
|
hasLoginForm: boolean;
|
|
license: ClientLicense;
|
|
serverDisplayName: string;
|
|
serverUrl: string;
|
|
ssoOptions: Record<string, boolean>;
|
|
theme: Theme;
|
|
}
|
|
|
|
const getStyles = makeStyleSheetFromTheme((theme: Theme) => ({
|
|
centered: {
|
|
width: '100%',
|
|
maxWidth: 600,
|
|
},
|
|
container: {
|
|
flex: 1,
|
|
...Platform.select({
|
|
android: {
|
|
marginTop: 56,
|
|
},
|
|
}),
|
|
},
|
|
flex: {
|
|
flex: 1,
|
|
},
|
|
header: {
|
|
color: theme.mentionColor,
|
|
marginBottom: 12,
|
|
...typography('Heading', 1000, 'SemiBold'),
|
|
},
|
|
innerContainer: {
|
|
alignItems: 'center',
|
|
justifyContent: 'center',
|
|
paddingHorizontal: 24,
|
|
},
|
|
subheader: {
|
|
color: changeOpacity(theme.centerChannelColor, 0.6),
|
|
marginBottom: 12,
|
|
...typography('Body', 200, 'Regular'),
|
|
},
|
|
}));
|
|
|
|
const AnimatedSafeArea = Animated.createAnimatedComponent(SafeAreaView);
|
|
|
|
const LoginOptions = ({
|
|
closeButtonId, componentId, config, extra,
|
|
hasLoginForm, launchType, launchError, license,
|
|
serverDisplayName, serverUrl, ssoOptions, theme,
|
|
}: LoginOptionsProps) => {
|
|
const styles = getStyles(theme);
|
|
const keyboardAwareRef = useRef<KeyboardAwareScrollView>(null);
|
|
const dimensions = useWindowDimensions();
|
|
const isTablet = useIsTablet();
|
|
const translateX = useSharedValue(dimensions.width);
|
|
const numberSSOs = useMemo(() => {
|
|
return Object.values(ssoOptions).filter((v) => v).length;
|
|
}, [ssoOptions]);
|
|
const description = useMemo(() => {
|
|
if (hasLoginForm) {
|
|
return (
|
|
<FormattedText
|
|
style={styles.subheader}
|
|
id='mobile.login_options.enter_credentials'
|
|
testID='login_options.description.enter_credentials'
|
|
defaultMessage='Enter your login details below.'
|
|
/>
|
|
);
|
|
} else if (numberSSOs) {
|
|
return (
|
|
<FormattedText
|
|
style={styles.subheader}
|
|
id='mobile.login_options.select_option'
|
|
testID='login_options.description.select_option'
|
|
defaultMessage='Select a login option below.'
|
|
/>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<FormattedText
|
|
style={styles.subheader}
|
|
id='mobile.login_options.none'
|
|
testID='login_options.description.none'
|
|
defaultMessage="You can't log in to your account yet. At least one login option must be configured. Contact your System Admin for assistance."
|
|
/>
|
|
);
|
|
}, [hasLoginForm, numberSSOs, theme]);
|
|
|
|
const goToSso = preventDoubleTap((ssoType: string) => {
|
|
goToScreen(Screens.SSO, '', {config, extra, launchError, launchType, license, theme, ssoType, serverDisplayName, serverUrl}, loginAnimationOptions());
|
|
});
|
|
|
|
const optionsSeparator = hasLoginForm && Boolean(numberSSOs) && (
|
|
<LoginOptionsSeparator
|
|
theme={theme}
|
|
/>
|
|
);
|
|
|
|
const transform = useAnimatedStyle(() => {
|
|
const duration = Platform.OS === 'android' ? 250 : 350;
|
|
return {
|
|
transform: [{translateX: withTiming(translateX.value, {duration})}],
|
|
};
|
|
}, []);
|
|
|
|
const dismiss = () => {
|
|
dismissModal({componentId});
|
|
};
|
|
|
|
const pop = () => {
|
|
popTopScreen(componentId);
|
|
};
|
|
|
|
useEffect(() => {
|
|
const navigationEvents = Navigation.events().registerNavigationButtonPressedListener(({buttonId}) => {
|
|
if (closeButtonId && buttonId === closeButtonId) {
|
|
NetworkManager.invalidateClient(serverUrl);
|
|
dismissModal({componentId});
|
|
}
|
|
});
|
|
|
|
return () => navigationEvents.remove();
|
|
}, []);
|
|
|
|
useEffect(() => {
|
|
translateX.value = 0;
|
|
}, []);
|
|
|
|
useEffect(() => {
|
|
const listener = {
|
|
componentDidAppear: () => {
|
|
translateX.value = 0;
|
|
},
|
|
componentDidDisappear: () => {
|
|
translateX.value = -dimensions.width;
|
|
},
|
|
};
|
|
const unsubscribe = Navigation.events().registerComponentListener(listener, Screens.LOGIN);
|
|
|
|
return () => unsubscribe.remove();
|
|
}, [dimensions]);
|
|
|
|
useNavButtonPressed(closeButtonId || '', componentId, dismiss, []);
|
|
useAndroidHardwareBackHandler(componentId, pop);
|
|
|
|
let additionalContainerStyle;
|
|
if (numberSSOs < 3 || !hasLoginForm || (isTablet && dimensions.height > dimensions.width)) {
|
|
additionalContainerStyle = styles.flex;
|
|
}
|
|
|
|
let title;
|
|
if (hasLoginForm || numberSSOs > 0) {
|
|
title = (
|
|
<FormattedText
|
|
defaultMessage='Log In to Your Account'
|
|
id={'mobile.login_options.heading'}
|
|
testID={'login_options.title.login_to_account'}
|
|
style={styles.header}
|
|
/>
|
|
);
|
|
} else {
|
|
title = (
|
|
<FormattedText
|
|
defaultMessage="Can't Log In"
|
|
id={'mobile.login_options.cant_heading'}
|
|
testID={'login_options.title.cant_login'}
|
|
style={styles.header}
|
|
/>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<View
|
|
style={styles.flex}
|
|
testID='login.screen'
|
|
>
|
|
<Background theme={theme}/>
|
|
<AnimatedSafeArea style={[styles.container, transform]}>
|
|
<KeyboardAwareScrollView
|
|
bounces={true}
|
|
contentContainerStyle={[styles.innerContainer, additionalContainerStyle]}
|
|
enableAutomaticScroll={true}
|
|
enableOnAndroid={false}
|
|
enableResetScrollToCoords={true}
|
|
extraScrollHeight={0}
|
|
keyboardDismissMode='interactive'
|
|
keyboardShouldPersistTaps='handled'
|
|
ref={keyboardAwareRef}
|
|
scrollToOverflowEnabled={true}
|
|
style={styles.flex}
|
|
>
|
|
<View style={styles.centered}>
|
|
{title}
|
|
{description}
|
|
{hasLoginForm &&
|
|
<Form
|
|
config={config}
|
|
extra={extra}
|
|
license={license}
|
|
launchError={launchError}
|
|
launchType={launchType}
|
|
theme={theme}
|
|
serverDisplayName={serverDisplayName}
|
|
serverUrl={serverUrl}
|
|
/>
|
|
}
|
|
{optionsSeparator}
|
|
{numberSSOs > 0 &&
|
|
<SsoOptions
|
|
goToSso={goToSso}
|
|
ssoOnly={!hasLoginForm}
|
|
ssoOptions={ssoOptions}
|
|
theme={theme}
|
|
/>
|
|
}
|
|
</View>
|
|
</KeyboardAwareScrollView>
|
|
</AnimatedSafeArea>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
export default LoginOptions;
|