forked from Ivasoft/mattermost-mobile
* Show login screen when selecting a previous server * Update app/screens/home/channel_list/servers/servers_list/server_item/server_item.tsx Co-authored-by: Jason Frerich <jason.frerich@mattermost.com> Co-authored-by: Jason Frerich <jason.frerich@mattermost.com>
362 lines
12 KiB
TypeScript
362 lines
12 KiB
TypeScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import {useManagedConfig} from '@mattermost/react-native-emm';
|
|
import React, {useCallback, useEffect, useRef, useState} from 'react';
|
|
import {useIntl} from 'react-intl';
|
|
import {Alert, 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 {doPing} from '@actions/remote/general';
|
|
import {fetchConfigAndLicense} from '@actions/remote/systems';
|
|
import LocalConfig from '@assets/config.json';
|
|
import ClientError from '@client/rest/error';
|
|
import AppVersion from '@components/app_version';
|
|
import {Screens} from '@constants';
|
|
import DatabaseManager from '@database/manager';
|
|
import {t} from '@i18n';
|
|
import NetworkManager from '@managers/network_manager';
|
|
import {queryServerByDisplayName, queryServerByIdentifier} from '@queries/app/servers';
|
|
import Background from '@screens/background';
|
|
import {dismissModal, goToScreen, loginAnimationOptions} from '@screens/navigation';
|
|
import {DeepLinkWithData, LaunchProps, LaunchType} from '@typings/launch';
|
|
import {getErrorMessage} from '@utils/client_error';
|
|
import {loginOptions} from '@utils/server';
|
|
import {changeOpacity, makeStyleSheetFromTheme} from '@utils/theme';
|
|
import {getServerUrlAfterRedirect, isValidUrl, sanitizeUrl} from '@utils/url';
|
|
|
|
import ServerForm from './form';
|
|
import ServerHeader from './header';
|
|
|
|
interface ServerProps extends LaunchProps {
|
|
closeButtonId?: string;
|
|
componentId: string;
|
|
theme: Theme;
|
|
}
|
|
|
|
let cancelPing: undefined | (() => void);
|
|
|
|
const defaultServerUrlMessage = {
|
|
id: t('mobile.server_url.empty'),
|
|
defaultMessage: 'Please enter a valid server URL',
|
|
};
|
|
|
|
const AnimatedSafeArea = Animated.createAnimatedComponent(SafeAreaView);
|
|
|
|
const Server = ({
|
|
closeButtonId,
|
|
componentId,
|
|
displayName: defaultDisplayName,
|
|
extra,
|
|
launchType,
|
|
launchError,
|
|
serverUrl: defaultServerUrl,
|
|
theme,
|
|
}: ServerProps) => {
|
|
const intl = useIntl();
|
|
const managedConfig = useManagedConfig<ManagedConfig>();
|
|
const dimensions = useWindowDimensions();
|
|
const translateX = useSharedValue(0);
|
|
const keyboardAwareRef = useRef<KeyboardAwareScrollView>();
|
|
const [connecting, setConnecting] = useState(false);
|
|
const [displayName, setDisplayName] = useState<string>('');
|
|
const [buttonDisabled, setButtonDisabled] = useState(true);
|
|
const [url, setUrl] = useState<string>('');
|
|
const [displayNameError, setDisplayNameError] = useState<string | undefined>();
|
|
const [urlError, setUrlError] = useState<string | undefined>();
|
|
const styles = getStyleSheet(theme);
|
|
const {formatMessage} = intl;
|
|
|
|
useEffect(() => {
|
|
let serverName: string | undefined = defaultDisplayName || managedConfig?.serverName || LocalConfig.DefaultServerName;
|
|
let serverUrl: string | undefined = defaultServerUrl || managedConfig?.serverUrl || LocalConfig.DefaultServerUrl;
|
|
let autoconnect = managedConfig?.allowOtherServers === 'false' || LocalConfig.AutoSelectServerUrl;
|
|
|
|
if (launchType === LaunchType.DeepLink) {
|
|
const deepLinkServerUrl = (extra as DeepLinkWithData).data?.serverUrl;
|
|
if (managedConfig) {
|
|
autoconnect = (managedConfig.allowOtherServers === 'false' && managedConfig.serverUrl === deepLinkServerUrl);
|
|
if (managedConfig.serverUrl !== deepLinkServerUrl || launchError) {
|
|
Alert.alert('', intl.formatMessage({
|
|
id: 'mobile.server_url.deeplink.emm.denied',
|
|
defaultMessage: 'This app is controlled by an EMM and the DeepLink server url does not match the EMM allowed server',
|
|
}));
|
|
}
|
|
} else {
|
|
autoconnect = true;
|
|
serverUrl = deepLinkServerUrl;
|
|
}
|
|
} else if (launchType === LaunchType.AddServer) {
|
|
serverName = defaultDisplayName;
|
|
serverUrl = defaultServerUrl;
|
|
}
|
|
|
|
if (serverUrl) {
|
|
// If a server Url is set by the managed or local configuration, use it.
|
|
setUrl(serverUrl);
|
|
}
|
|
|
|
if (serverName) {
|
|
setDisplayName(serverName);
|
|
}
|
|
|
|
if (serverUrl && serverName && autoconnect) {
|
|
// If no other servers are allowed or the local config for AutoSelectServerUrl is set, attempt to connect
|
|
handleConnect(managedConfig?.serverUrl || LocalConfig.DefaultServerUrl);
|
|
}
|
|
}, []);
|
|
|
|
useEffect(() => {
|
|
if (url && displayName) {
|
|
setButtonDisabled(false);
|
|
} else {
|
|
setButtonDisabled(true);
|
|
}
|
|
}, [url, displayName]);
|
|
|
|
useEffect(() => {
|
|
const listener = {
|
|
componentDidAppear: () => {
|
|
translateX.value = 0;
|
|
if (url) {
|
|
NetworkManager.invalidateClient(url);
|
|
}
|
|
},
|
|
componentDidDisappear: () => {
|
|
translateX.value = -dimensions.width;
|
|
},
|
|
};
|
|
const unsubscribe = Navigation.events().registerComponentListener(listener, componentId);
|
|
|
|
return () => unsubscribe.remove();
|
|
}, [componentId, url, dimensions]);
|
|
|
|
useEffect(() => {
|
|
const navigationEvents = Navigation.events().registerNavigationButtonPressedListener(({buttonId}) => {
|
|
if (closeButtonId && buttonId === closeButtonId) {
|
|
NetworkManager.invalidateClient(url);
|
|
dismissModal({componentId});
|
|
}
|
|
});
|
|
|
|
return () => navigationEvents.remove();
|
|
}, []);
|
|
|
|
const displayLogin = (serverUrl: string, config: ClientConfig, license: ClientLicense) => {
|
|
const {enabledSSOs, hasLoginForm, numberSSOs, ssoOptions} = loginOptions(config, license);
|
|
const passProps = {
|
|
config,
|
|
extra,
|
|
hasLoginForm,
|
|
launchError,
|
|
launchType,
|
|
license,
|
|
serverDisplayName: displayName,
|
|
serverUrl,
|
|
ssoOptions,
|
|
theme,
|
|
};
|
|
|
|
const redirectSSO = !hasLoginForm && numberSSOs === 1;
|
|
const screen = redirectSSO ? Screens.SSO : Screens.LOGIN;
|
|
if (redirectSSO) {
|
|
// @ts-expect-error ssoType not in definition
|
|
passProps.ssoType = enabledSSOs[0];
|
|
}
|
|
|
|
goToScreen(screen, '', passProps, loginAnimationOptions());
|
|
setConnecting(false);
|
|
setButtonDisabled(false);
|
|
setUrl(serverUrl);
|
|
};
|
|
|
|
const handleConnect = async (manualUrl?: string) => {
|
|
if (buttonDisabled && !manualUrl) {
|
|
return;
|
|
}
|
|
|
|
if (connecting && cancelPing) {
|
|
cancelPing();
|
|
return;
|
|
}
|
|
|
|
const serverUrl = typeof manualUrl === 'string' ? manualUrl : url;
|
|
if (!serverUrl || serverUrl.trim() === '') {
|
|
setUrlError(formatMessage(defaultServerUrlMessage));
|
|
return;
|
|
}
|
|
|
|
if (!isServerUrlValid(serverUrl)) {
|
|
return;
|
|
}
|
|
|
|
if (displayNameError) {
|
|
setDisplayNameError(undefined);
|
|
}
|
|
|
|
if (urlError) {
|
|
setUrlError(undefined);
|
|
}
|
|
|
|
const server = await queryServerByDisplayName(DatabaseManager.appDatabase!.database, displayName);
|
|
if (server && server.lastActiveAt > 0) {
|
|
setButtonDisabled(true);
|
|
setDisplayNameError(formatMessage({
|
|
id: 'mobile.server_name.exists',
|
|
defaultMessage: 'You are using this name for another server.',
|
|
}));
|
|
setConnecting(false);
|
|
return;
|
|
}
|
|
|
|
pingServer(serverUrl);
|
|
};
|
|
|
|
const handleDisplayNameTextChanged = useCallback((text: string) => {
|
|
setDisplayName(text);
|
|
setDisplayNameError(undefined);
|
|
}, []);
|
|
|
|
const handleUrlTextChanged = useCallback((text: string) => {
|
|
setUrlError(undefined);
|
|
setUrl(text);
|
|
}, []);
|
|
|
|
const isServerUrlValid = (serverUrl?: string) => {
|
|
const testUrl = sanitizeUrl(serverUrl ?? url);
|
|
if (!isValidUrl(testUrl)) {
|
|
setUrlError(intl.formatMessage({
|
|
id: 'mobile.server_url.invalid_format',
|
|
defaultMessage: 'URL must start with http:// or https://',
|
|
}));
|
|
return false;
|
|
}
|
|
return true;
|
|
};
|
|
|
|
const pingServer = async (pingUrl: string, retryWithHttp = true) => {
|
|
let canceled = false;
|
|
setConnecting(true);
|
|
cancelPing = () => {
|
|
canceled = true;
|
|
setConnecting(false);
|
|
cancelPing = undefined;
|
|
};
|
|
|
|
const serverUrl = await getServerUrlAfterRedirect(pingUrl, !retryWithHttp);
|
|
const result = await doPing(serverUrl);
|
|
|
|
if (canceled) {
|
|
return;
|
|
}
|
|
|
|
if (result.error) {
|
|
if (retryWithHttp) {
|
|
const nurl = serverUrl.replace('https:', 'http:');
|
|
pingServer(nurl, false);
|
|
} else {
|
|
setUrlError(getErrorMessage(result.error as ClientError, intl));
|
|
setButtonDisabled(true);
|
|
setConnecting(false);
|
|
}
|
|
return;
|
|
}
|
|
|
|
const data = await fetchConfigAndLicense(serverUrl, true);
|
|
if (data.error) {
|
|
setButtonDisabled(true);
|
|
setUrlError(getErrorMessage(data.error as ClientError, intl));
|
|
setConnecting(false);
|
|
return;
|
|
}
|
|
|
|
const server = await queryServerByIdentifier(DatabaseManager.appDatabase!.database, data.config!.DiagnosticId);
|
|
setConnecting(false);
|
|
|
|
if (server && server.lastActiveAt > 0) {
|
|
setButtonDisabled(true);
|
|
setUrlError(formatMessage({
|
|
id: 'mobile.server_identifier.exists',
|
|
defaultMessage: 'You are already connected to this server.',
|
|
}));
|
|
return;
|
|
}
|
|
|
|
displayLogin(serverUrl, data.config!, data.license!);
|
|
};
|
|
|
|
const transform = useAnimatedStyle(() => {
|
|
const duration = Platform.OS === 'android' ? 250 : 350;
|
|
return {
|
|
transform: [{translateX: withTiming(translateX.value, {duration})}],
|
|
};
|
|
}, []);
|
|
|
|
return (
|
|
<View
|
|
style={styles.flex}
|
|
testID='server.screen'
|
|
>
|
|
<Background theme={theme}/>
|
|
<AnimatedSafeArea
|
|
key={'server_content'}
|
|
style={[styles.flex, transform]}
|
|
>
|
|
<KeyboardAwareScrollView
|
|
bounces={false}
|
|
contentContainerStyle={styles.scrollContainer}
|
|
enableAutomaticScroll={Platform.OS === 'android'}
|
|
enableOnAndroid={false}
|
|
enableResetScrollToCoords={true}
|
|
extraScrollHeight={20}
|
|
keyboardDismissMode='on-drag'
|
|
keyboardShouldPersistTaps='handled'
|
|
|
|
// @ts-expect-error legacy ref
|
|
ref={keyboardAwareRef}
|
|
scrollToOverflowEnabled={true}
|
|
style={styles.flex}
|
|
>
|
|
<ServerHeader
|
|
additionalServer={launchType === LaunchType.AddServer}
|
|
theme={theme}
|
|
/>
|
|
<ServerForm
|
|
buttonDisabled={buttonDisabled}
|
|
connecting={connecting}
|
|
displayName={displayName}
|
|
displayNameError={displayNameError}
|
|
handleConnect={handleConnect}
|
|
handleDisplayNameTextChanged={handleDisplayNameTextChanged}
|
|
handleUrlTextChanged={handleUrlTextChanged}
|
|
keyboardAwareRef={keyboardAwareRef}
|
|
theme={theme}
|
|
url={url}
|
|
urlError={urlError}
|
|
/>
|
|
</KeyboardAwareScrollView>
|
|
<AppVersion textStyle={styles.appInfo}/>
|
|
</AnimatedSafeArea>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const getStyleSheet = makeStyleSheetFromTheme((theme: Theme) => ({
|
|
appInfo: {
|
|
color: changeOpacity(theme.centerChannelColor, 0.56),
|
|
},
|
|
flex: {
|
|
flex: 1,
|
|
},
|
|
scrollContainer: {
|
|
alignItems: 'center',
|
|
height: '100%',
|
|
justifyContent: 'center',
|
|
},
|
|
}));
|
|
|
|
export default Server;
|