import { Reducer } from 'react'; import Setting from '../models/Setting'; import { ApplicationPlatform, ApplicationType } from '../types'; import shim from '../shim'; import { _ } from '../locale'; type ActionType = 'LINK_USED' | 'COMPLETED' | 'ERROR'; type Action = { type: ActionType; payload?: any; }; type DefaultState = { className: 'text' | 'bold'; message: ()=> string; next: ActionType; active: ActionType | 'INITIAL'; errorMessage?: string; }; export const defaultState: DefaultState = { className: 'text', message: ()=> _('Waiting for authorisation...'), next: 'LINK_USED', active: 'INITIAL', }; export const reducer: Reducer = (state: DefaultState, action: Action) => { switch (action.type) { case 'LINK_USED': { return { className: 'text', message: () => _('If you have already authorised, please wait for the application to sync to Joplin Cloud.'), next: 'COMPLETED', active: 'LINK_USED', }; } case 'COMPLETED': { return { className: 'bold', message: () => _('You are logged in into Joplin Cloud, you can leave this screen now.'), active: 'COMPLETED', next: 'COMPLETED', }; } case 'ERROR': { return { className: 'text', message: () => _('You were unable to connect to Joplin Cloud. Please check your credentials and try again. Error:'), active: 'ERROR', next: 'COMPLETED', errorMessage: action.payload, }; } default: { return state; } } }; export const getApplicationInformation = async () => { const platformName = await shim.platformName(); switch (platformName) { case 'ios': return { type: ApplicationType.Mobile, platform: ApplicationPlatform.Ios }; case 'android': return { type: ApplicationType.Mobile, platform: ApplicationPlatform.Android }; case 'darwin': return { type: ApplicationType.Desktop, platform: ApplicationPlatform.MacOs }; case 'win32': return { type: ApplicationType.Desktop, platform: ApplicationPlatform.Windows }; case 'linux': return { type: ApplicationType.Desktop, platform: ApplicationPlatform.Linux }; default: return { type: ApplicationType.Unknown, platform: ApplicationPlatform.Unknown }; } }; export const generateApplicationConfirmUrl = async (confirmUrl: string) => { const applicationInfo = await getApplicationInformation(); const searchParams = new URLSearchParams(); searchParams.append('platform', applicationInfo.platform.toString()); searchParams.append('type', applicationInfo.type.toString()); searchParams.append('version', shim.appVersion()); return `${confirmUrl}?${searchParams.toString()}`; }; // We have isWaitingResponse inside the function to avoid any state from lingering // after an error occurs. E.g.: if the function would throw an error while isWaitingResponse // was set to true the next time we call the function the value would still be true. // The closure function prevents that. export const checkIfLoginWasSuccessful = async (applicationsUrl: string) => { let isWaitingResponse = false; const performLoginRequest = async () => { if (isWaitingResponse) return undefined; isWaitingResponse = true; const response = await fetch(applicationsUrl); const jsonBody = await response.json(); if (!response.ok || jsonBody.status !== 'finished') { isWaitingResponse = false; return undefined; } Setting.setValue('sync.10.username', jsonBody.id); Setting.setValue('sync.10.password', jsonBody.password); return { success: true }; }; return performLoginRequest(); };