1
0
mirror of https://github.com/laurent22/joplin.git synced 2024-12-24 10:27:10 +02:00
joplin/packages/app-desktop/gui/SyncWizard/Dialog.tsx

362 lines
10 KiB
TypeScript

import * as React from 'react';
import { useState, useRef, useCallback } from 'react';
import { _ } from '@joplin/lib/locale';
import DialogButtonRow from '../DialogButtonRow';
import Dialog from '../Dialog';
import styled from 'styled-components';
import DialogTitle from '../DialogTitle';
import SyncTargetRegistry, { SyncTargetInfo } from '@joplin/lib/SyncTargetRegistry';
import useElementSize from '@joplin/lib/hooks/useElementSize';
import Button, { ButtonLevel } from '../Button/Button';
import bridge from '../../services/bridge';
import StyledInput from '../style/StyledInput';
import Setting from '@joplin/lib/models/Setting';
import SyncTargetJoplinCloud from '@joplin/lib/SyncTargetJoplinCloud';
import StyledLink from '../style/StyledLink';
interface Props {
themeId: number;
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
dispatch: Function;
}
const StyledRoot = styled.div`
min-width: 500px;
max-width: 1200px;
`;
const SyncTargetDescription = styled.div<{ height: number }>`
${props => props.height ? `height: ${props.height}px` : ''};
margin-bottom: 1.3em;
line-height: ${props => props.theme.lineHeight};
font-size: 16px;
`;
const CreateAccountLink = styled(StyledLink)`
font-size: 16px;
`;
const ContentRoot = styled.div`
background-color: ${props => props.theme.backgroundColor3};
padding: 1em;
padding-right: 0;
`;
const SelfHostingMessage = styled.div`
color: ${props => props.theme.color};
padding-right: 1em;
font-style: italic;
margin-top: 1em;
opacity: 0.6;
`;
const SyncTargetBoxes = styled.div`
display: flex;
flex-direction: row;
justify-content: center;
`;
const SyncTargetTitle = styled.p`
display: flex;
flex-direction: row;
font-weight: bold;
font-size: 1.7em;
align-items: center;
white-space: nowrap;
`;
const SyncTargetLogo = styled.img`
height: 1.3em;
margin-right: 0.4em;
`;
const SyncTargetBox = styled.div<{ faded: boolean }>`
display: flex;
flex: 1;
flex-direction: column;
font-family: ${props => props.theme.fontFamily};
color: ${props => props.theme.color};
background-color: ${props => props.theme.backgroundColor};
border: 1px solid ${props => props.theme.dividerColor};
border-radius: 8px;
padding: 2em 2.2em 2em 2.2em;
margin-right: 1em;
max-width: 400px;
opacity: ${props => props.faded ? 0.5 : 1};
`;
const FeatureList = styled.div`
margin-bottom: 1em;
`;
const FeatureIcon = styled.i`
display: inline-flex;
width: 16px;
justify-content: center;
color: ${props => props.theme.color4};
position: absolute;
`;
const FeatureLine = styled.div<{ enabled: boolean }>`
margin-bottom: .5em;
opacity: ${props => props.enabled ? 1 : 0.5};
position: relative;
font-size: 16px;
`;
const FeatureLabel = styled.div`
margin-left: 24px;
line-height: ${props => props.theme.lineHeight};
`;
const SelectButton = styled(Button)`
padding: 10px 10px;
height: auto;
min-height: auto;
max-height: fit-content;
font-size: 1em;
`;
const JoplinCloudLoginForm = styled.div`
display: flex;
flex-direction: column;
`;
const FormLabel = styled.label`
font-weight: bold;
margin: 1em 0 0.6em 0;
`;
const SlowSyncWarning = styled.div`
margin-top: 1em;
opacity: 0.8;
font-family: ${props => props.theme.fontFamily};
color: ${props => props.theme.color};
font-size: 14px;
`;
const syncTargetNames: string[] = [
'joplinCloud',
'dropbox',
'onedrive',
'nextcloud',
'webdav',
'amazon_s3',
'joplinServer',
];
const logosImageNames: Record<string, string> = {
'dropbox': 'SyncTarget_Dropbox.svg',
'joplinCloud': 'SyncTarget_JoplinCloud.svg',
'onedrive': 'SyncTarget_OneDrive.svg',
};
export default function(props: Props) {
const [showJoplinCloudForm, setShowJoplinCloudForm] = useState(false);
const joplinCloudDescriptionRef = useRef(null);
const [joplinCloudEmail, setJoplinCloudEmail] = useState('');
const [joplinCloudPassword, setJoplinCloudPassword] = useState('');
const [joplinCloudLoginInProgress, setJoplinCloudLoginInProgress] = useState(false);
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
function closeDialog(dispatch: Function) {
dispatch({
type: 'DIALOG_CLOSE',
name: 'syncWizard',
});
}
const onButtonRowClick = useCallback(() => {
closeDialog(props.dispatch);
}, [props.dispatch]);
const { height: descriptionHeight } = useElementSize(joplinCloudDescriptionRef);
function renderFeature(enabled: boolean, label: string) {
const className = enabled ? 'fas fa-check' : 'fas fa-times';
return (
<FeatureLine enabled={enabled} key={label}><FeatureIcon className={className}></FeatureIcon> <FeatureLabel>{label}</FeatureLabel></FeatureLine>
);
}
function renderFeatures(name: string) {
return (
<FeatureList>
{[
renderFeature(true, _('Sync your notes')),
renderFeature(name === 'joplinCloud', _('Publish notes to the internet')),
renderFeature(name === 'joplinCloud', _('Collaborate on notebooks with others')),
]}
</FeatureList>
);
}
const onJoplinCloudEmailChange = useCallback((event: any) => {
setJoplinCloudEmail(event.target.value);
}, []);
const onJoplinCloudPasswordChange = useCallback((event: any) => {
setJoplinCloudPassword(event.target.value);
}, []);
const onJoplinCloudLoginClick = useCallback(async () => {
setJoplinCloudLoginInProgress(true);
let result = null;
try {
result = await SyncTargetJoplinCloud.checkConfig({
password: () => joplinCloudPassword,
path: () => Setting.value('sync.10.path'),
userContentPath: () => Setting.value('sync.10.userContentPath'),
username: () => joplinCloudEmail,
});
} finally {
setJoplinCloudLoginInProgress(false);
}
if (result.ok) {
Setting.setValue('sync.target', 10);
Setting.setValue('sync.10.username', joplinCloudEmail);
Setting.setValue('sync.10.password', joplinCloudPassword);
await Setting.saveAll();
alert(_('Thank you! Your Joplin Cloud account is now setup and ready to use.'));
closeDialog(props.dispatch);
props.dispatch({
type: 'NAV_GO',
routeName: 'Main',
});
} else {
alert(_('There was an error setting up your Joplin Cloud account. Please verify your email and password and try again. Error was:\n\n%s', result.errorMessage));
}
}, [joplinCloudEmail, joplinCloudPassword, props.dispatch]);
const onJoplinCloudCreateAccountClick = useCallback(() => {
void bridge().openExternal('https://joplinapp.org/plans/');
}, []);
function renderJoplinCloudLoginForm() {
return (
<JoplinCloudLoginForm>
<div style={{ fontSize: '16px' }}>{_('Login below.')} <CreateAccountLink href="#" onClick={onJoplinCloudCreateAccountClick}>{_('Or create an account.')}</CreateAccountLink></div>
<FormLabel>{_('Email')}</FormLabel>
<StyledInput type="email" onChange={onJoplinCloudEmailChange}/>
<FormLabel>{_('Password')}</FormLabel>
<StyledInput type="password" onChange={onJoplinCloudPasswordChange}/>
<SelectButton mt="1.3em" disabled={joplinCloudLoginInProgress} level={ButtonLevel.Primary} title={_('Login')} onClick={onJoplinCloudLoginClick}/>
</JoplinCloudLoginForm>
);
}
const onSelectButtonClick = useCallback(async (name: string) => {
if (name === 'joplinCloud') {
setShowJoplinCloudForm(true);
} else {
Setting.setValue('sync.target', name === 'dropbox' ? 7 : 3);
await Setting.saveAll();
closeDialog(props.dispatch);
props.dispatch({
type: 'NAV_GO',
routeName: name === 'dropbox' ? 'DropboxLogin' : 'OneDriveLogin',
});
}
}, [props.dispatch]);
function renderSelectArea(info: SyncTargetInfo) {
if (info.name === 'joplinCloud' && showJoplinCloudForm) {
return renderJoplinCloudLoginForm();
} else {
return (
<SelectButton
level={ButtonLevel.Primary}
title={_('Select')}
onClick={() => onSelectButtonClick(info.name)}
disabled={joplinCloudLoginInProgress}
/>
);
}
}
function renderSyncTarget(info: SyncTargetInfo) {
const key = `syncTarget_${info.name}`;
const height = info.name !== 'joplinCloud' ? descriptionHeight : null;
const logoImageName = logosImageNames[info.name];
const logoImageSrc = logoImageName ? `${bridge().buildDir()}/images/${logoImageName}` : '';
const logo = logoImageSrc ? <SyncTargetLogo src={logoImageSrc}/> : null;
const descriptionComp = <SyncTargetDescription height={height} ref={info.name === 'joplinCloud' ? joplinCloudDescriptionRef : null}>{info.description}</SyncTargetDescription>;
const featuresComp = showJoplinCloudForm && info.name === 'joplinCloud' ? null : renderFeatures(info.name);
const renderSlowSyncWarning = () => {
if (info.name === 'joplinCloud') return null;
return <SlowSyncWarning>{`⚠️ ${_('%s is not optimised for synchronising many small files so your initial synchronisation will be slow.', info.label)}`}</SlowSyncWarning>;
};
return (
<SyncTargetBox id={key} key={key} faded={showJoplinCloudForm && info.name !== 'joplinCloud'}>
<SyncTargetTitle>{logo}{info.label}</SyncTargetTitle>
{descriptionComp}
{featuresComp}
{renderSelectArea(info)}
{renderSlowSyncWarning()}
</SyncTargetBox>
);
}
const onSelfHostingClick = useCallback(() => {
closeDialog(props.dispatch);
props.dispatch({
type: 'NAV_GO',
routeName: 'Config',
props: {
defaultSection: 'sync',
},
});
}, [props.dispatch]);
function renderContent() {
const boxes: any[] = [];
for (const name of syncTargetNames) {
const info = SyncTargetRegistry.infoByName(name);
if (info.supportsSelfHosted) continue;
boxes.push(renderSyncTarget(info));
}
const selfHostingMessage = showJoplinCloudForm ? null : <SelfHostingMessage>Self-hosting? Joplin also supports various self-hosting options such as Nextcloud, WebDAV, AWS S3 and Joplin Server. <a href="#" onClick={onSelfHostingClick}>Click here to select one</a>.</SelfHostingMessage>;
return (
<ContentRoot>
<SyncTargetBoxes>
{boxes}
</SyncTargetBoxes>
{selfHostingMessage}
</ContentRoot>
);
}
function renderDialogWrapper() {
return (
<StyledRoot>
<DialogTitle title={_('Joplin can synchronise your notes using various providers. Select one from the list below.')} justifyContent="center"/>
{renderContent()}
<DialogButtonRow
themeId={props.themeId}
onClick={onButtonRowClick}
okButtonShow={false}
cancelButtonLabel={_('Close')}
/>
</StyledRoot>
);
}
return (
<Dialog renderContent={renderDialogWrapper}/>
);
}