import { AccountStore, Account, Actions, IdentityStore } from 'mailspring-exports'; import { ipcRenderer } from 'electron'; import MailspringStore from 'mailspring-store'; import OnboardingActions from './onboarding-actions'; class OnboardingStore extends MailspringStore { constructor() { super(); this.listenTo(OnboardingActions.moveToPreviousPage, this._onMoveToPreviousPage); this.listenTo(OnboardingActions.moveToPage, this._onMoveToPage); this.listenTo(OnboardingActions.setAccount, this._onSetAccount); this.listenTo(OnboardingActions.chooseAccountProvider, this._onChooseAccountProvider); this.listenTo(OnboardingActions.finishAndAddAccount, this._onFinishAndAddAccount); this.listenTo(OnboardingActions.identityJSONReceived, this._onIdentityJSONReceived); ipcRenderer.on('set-account-provider', (e, provider) => { if (provider) { this._onChooseAccountProvider(provider); } else { this._pageStack = ['account-choose']; this.trigger(); } }); const { existingAccount, addingAccount, accountProvider } = AppEnv.getWindowProps(); const hasAccounts = AccountStore.accounts().length > 0; const identity = IdentityStore.identity(); this._account = new Account({ name: identity ? `${identity.firstName || ''} ${identity.lastName || ''}` : '', emailAddress: identity ? identity.emailAddress : '', settings: {}, }); if (existingAccount) { // Used when re-adding an account after re-connecting this._pageStack = ['account-choose']; this._account.name = existingAccount.name; this._account.emailAddress = existingAccount.emailAddress; this._onChooseAccountProvider(existingAccount.provider); } else if (addingAccount) { // Adding a new, unknown account this._pageStack = ['account-choose']; if (accountProvider) { this._onChooseAccountProvider(accountProvider); } } else if (identity) { // Should only happen if config was edited to remove all accounts, // but don't want to re-login to Nylas account. Very useful when // switching environments. this._pageStack = ['account-choose']; } else if (hasAccounts) { // Should only happen when the user has "signed out" of their Nylas ID, // but already has accounts synced. Or is upgrading from a very old build. // We used to show "Welcome Back", but now just jump to sign in. this._pageStack = ['authenticate']; } else { // Standard new user onboarding flow. this._pageStack = ['welcome']; } } _onOnboardingComplete = () => { // When account JSON is received, we want to notify external services // that it succeeded. Unfortunately in this case we're likely to // close the window before those requests can be made. We add a short // delay here to ensure that any pending requests have a chance to // clear before the window closes. setTimeout(() => { ipcRenderer.send('account-setup-successful'); }, 100); }; _onChooseAccountProvider = provider => { let nextPage = 'account-settings'; if (provider === 'gmail') { nextPage = 'account-settings-gmail'; } else if (provider === 'exchange') { nextPage = 'account-settings-exchange'; } Actions.recordUserEvent('Selected Account Provider', { provider, }); // Don't carry over any type-specific account information this._onSetAccount( new Account({ emailAddress: this._account.emailAddress, name: this._account.name, settings: {}, provider, }) ); this._onMoveToPage(nextPage); }; _onSetAccount = acct => { if (!(acct instanceof Account)) { throw new Error('OnboardingActions.setAccount expects an Account instance.'); } this._account = acct; this.trigger(); }; _onMoveToPreviousPage = () => { this._pageStack.pop(); this.trigger(); }; _onMoveToPage = page => { this._pageStack.push(page); this.trigger(); }; _onIdentityJSONReceived = async json => { const isFirstAccount = AccountStore.accounts().length === 0; await IdentityStore.saveIdentity(json); setTimeout(() => { if (isFirstAccount) { const next = this._account.clone(); next.name = `${json.firstName || ''} ${json.lastName || ''}`; next.emailAddress = json.emailAddress; this._onSetAccount(next); OnboardingActions.moveToPage('account-choose'); } else { this._onOnboardingComplete(); } }, 1000); }; _onFinishAndAddAccount = async account => { const isFirstAccount = AccountStore.accounts().length === 0; try { AccountStore.addAccount(account); } catch (e) { AppEnv.reportError(e); AppEnv.showErrorDialog({ title: 'Unable to Add Account', message: 'Sorry, something went wrong when this account was added to Mailspring. ' + `If you do not see the account, try linking it again. ${e.toString()}`, }); } AppEnv.displayWindow(); Actions.recordUserEvent('Email Account Auth Succeeded', { provider: account.provider, }); if (isFirstAccount) { this._onMoveToPage('initial-preferences'); Actions.recordUserEvent('First Account Linked', { provider: account.provider, }); } else { // let them see the "success" screen for a moment // before the window is closed. setTimeout(() => { this._onOnboardingComplete(); }, 2000); } }; page() { return this._pageStack[this._pageStack.length - 1]; } pageDepth() { return this._pageStack.length; } account() { return this._account; } } export default new OnboardingStore();