mirror of
https://github.com/Foundry376/Mailspring.git
synced 2025-01-08 01:04:39 +08:00
461 lines
16 KiB
CoffeeScript
461 lines
16 KiB
CoffeeScript
React = require 'react'
|
||
ReactDOM = require 'react-dom'
|
||
_ = require 'underscore'
|
||
{ipcRenderer, dialog, remote} = require 'electron'
|
||
{RetinaImg} = require 'nylas-component-kit'
|
||
{RegExpUtils, EdgehillAPI, NylasAPI, APIError, Actions, AccountStore} = require 'nylas-exports'
|
||
|
||
OnboardingActions = require './onboarding-actions'
|
||
NylasApiEnvironmentStore = require './nylas-api-environment-store'
|
||
Providers = require './account-types'
|
||
|
||
class AccountSettingsPage extends React.Component
|
||
@displayName: "AccountSettingsPage"
|
||
|
||
constructor: (@props) ->
|
||
@state =
|
||
provider: @props.pageData.provider
|
||
settings: {}
|
||
fields: {}
|
||
pageNumber: 0
|
||
errorFieldNames: []
|
||
errorMessage: null
|
||
show_advanced: false
|
||
|
||
@props.pageData.provider.settings.forEach (field) =>
|
||
if field.default?
|
||
@state.settings[field.name] = field.default
|
||
|
||
# Special case for gmail. Rather than showing a form, we poll in the
|
||
# background for completion of the gmail auth on the server.
|
||
if @state.provider.name is 'gmail'
|
||
pollAttemptId = 0
|
||
done = false
|
||
# polling with capped exponential backoff
|
||
delay = 1000
|
||
tries = 0
|
||
poll = (id,initial_delay) =>
|
||
_retry = =>
|
||
tries++
|
||
@_pollForGmailAccount((account_data) =>
|
||
if account_data?
|
||
done = true
|
||
{data} = account_data
|
||
account = JSON.parse(data)
|
||
@_onAccountReceived(account)
|
||
else if tries < 20 and id is pollAttemptId
|
||
setTimeout(_retry, delay)
|
||
delay *= 1.2 # exponential backoff
|
||
)
|
||
setTimeout(_retry,initial_delay)
|
||
|
||
ipcRenderer.on('browser-window-focus', ->
|
||
if not done # hack to deactivate this listener when done
|
||
pollAttemptId++
|
||
poll(pollAttemptId,0)
|
||
)
|
||
poll(pollAttemptId,5000)
|
||
|
||
render: ->
|
||
<div className="page account-setup">
|
||
<div className="logo-container">
|
||
<RetinaImg
|
||
name={@state.provider.header_icon}
|
||
mode={RetinaImg.Mode.ContentPreserve}
|
||
className="logo"/>
|
||
</div>
|
||
|
||
{@_renderTitle()}
|
||
|
||
<div className="back" onClick={@_fireMoveToPrevPage}>
|
||
<RetinaImg
|
||
name="onboarding-back.png"
|
||
mode={RetinaImg.Mode.ContentPreserve}/>
|
||
</div>
|
||
|
||
{@_renderErrorMessage()}
|
||
|
||
<form className="settings">
|
||
{@_renderFields()}
|
||
{@_renderSettings()}
|
||
{@_renderButton()}
|
||
</form>
|
||
</div>
|
||
|
||
componentDidMount: =>
|
||
@_applyFocus()
|
||
|
||
componentDidUpdate: =>
|
||
@_applyFocus()
|
||
|
||
_applyFocus: =>
|
||
firstInput = ReactDOM.findDOMNode(@).querySelector('input')
|
||
anyInputFocused = document.activeElement and document.activeElement.nodeName is 'INPUT'
|
||
|
||
if firstInput and not anyInputFocused
|
||
firstInput.focus()
|
||
|
||
_onSettingsChanged: (event) =>
|
||
# NOTE: This code is largely duplicated in _onValueChanged. TODO Fix!
|
||
{field, format} = event.target.dataset
|
||
intFormatter = (a) ->
|
||
i = parseInt(a)
|
||
if isNaN(i) then "" else i
|
||
formatter = if format is 'integer' then intFormatter else (a) -> a
|
||
settings = @state.settings
|
||
if event.target.type is 'checkbox'
|
||
settings[field] = event.target.checked
|
||
else
|
||
settings[field] = formatter(event.target.value)
|
||
|
||
settingField = _.findWhere(@state.provider.settings, {name: field})
|
||
|
||
# If the field defines an isValid method, try to validate
|
||
# the input.
|
||
if settingField
|
||
valueIsValid = not settingField.isValid? or settingField.isValid(event.target.value)
|
||
valueIsPresent = event.target.value and event.target.value.length > 0
|
||
valueIsRequired = settingField.required is true
|
||
|
||
if (not valueIsPresent and valueIsRequired) or (valueIsPresent and not valueIsValid)
|
||
errorFields = _.uniq(@state.errorFieldNames.concat([field]))
|
||
else
|
||
errorFields = _.uniq(_.without(@state.errorFieldNames, field))
|
||
@setState({errorFieldNames: errorFields})
|
||
|
||
@setState({settings})
|
||
|
||
_noFormErrors: =>
|
||
allFields = @state.provider.fields.concat(@state.provider.settings || [])
|
||
fieldsOnThisPage = allFields.filter(@_fieldOnCurrentPage)
|
||
fieldNames = _.pluck(fieldsOnThisPage, 'name')
|
||
return _.intersection(fieldNames, @state.errorFieldNames).length == 0
|
||
|
||
_fieldRequired: (f) =>
|
||
return f?.required == true
|
||
|
||
_allRequiredFieldsFilled: =>
|
||
allFields = @state.provider.fields.concat(@state.provider.settings || [])
|
||
requiredFields = allFields.filter(@_fieldOnCurrentPage).filter(@_fieldRequired)
|
||
fields = _.extend({}, @state.fields, @state.settings)
|
||
|
||
for field in requiredFields
|
||
fieldName = field['name']
|
||
if not (fieldName of fields) or fields[fieldName] == ''
|
||
return false
|
||
|
||
return true
|
||
|
||
_onValueChanged: (event) =>
|
||
# NOTE: This code is largely duplicated in _onSettingsChanged. TODO Fix!
|
||
field = event.target.dataset.field
|
||
fields = @state.fields
|
||
fields[field] = event.target.value
|
||
|
||
providerField = _.find(@state.provider.fields, ((e) -> return e['name'] == field))
|
||
|
||
# If the field defines an isValid method, try to validate
|
||
# the input.
|
||
if providerField
|
||
valueIsValid = not providerField.isValid? or providerField.isValid(event.target.value)
|
||
valueIsPresent = event.target.value and event.target.value.length > 0
|
||
valueIsRequired = providerField.required is true
|
||
|
||
if (not valueIsPresent and valueIsRequired) or (valueIsPresent and not valueIsValid)
|
||
errorFields = _.uniq(@state.errorFieldNames.concat([field]))
|
||
else
|
||
errorFields = _.uniq(_.without(@state.errorFieldNames, field))
|
||
@setState({errorFieldNames: errorFields})
|
||
|
||
if providerField.type == "email" and event.target.value
|
||
if event.target.value.endsWith('@gmail.com')
|
||
# set a state that contains a "this is a gmail account" message
|
||
errorFields = _.uniq(@state.errorFieldNames.concat([field]))
|
||
@setState
|
||
errorMessage: "This looks like a Gmail account. You should go back and sign in to Gmail instead."
|
||
errorFieldNames: errorFields
|
||
@_resize()
|
||
else
|
||
@setState({errorMessage: null})
|
||
@_resize()
|
||
|
||
@setState({fields})
|
||
|
||
_onFieldKeyPress: (event) =>
|
||
if event.key in ['Enter', 'Return']
|
||
pages = @state.provider.pages || []
|
||
if pages.length > @state.pageNumber + 1
|
||
@_onNextButton()
|
||
else
|
||
@_onSubmit()
|
||
|
||
_renderTitle: =>
|
||
if @state.provider.name is 'gmail'
|
||
<h2>
|
||
Sign in to Google in<br/>your browser.
|
||
</h2>
|
||
else if @state.provider.pages?.length > 0
|
||
<h2>
|
||
{@state.provider.pages[@state.pageNumber]}
|
||
</h2>
|
||
else
|
||
<h2>
|
||
Sign in to {@state.provider.displayName}
|
||
</h2>
|
||
|
||
_renderErrorMessage: =>
|
||
return unless @state.errorMessage
|
||
|
||
text = @state.errorMessage
|
||
result = RegExpUtils.urlRegex(matchEntireString: false).exec(text)
|
||
|
||
if result
|
||
link = result[0]
|
||
beforeText = text.substr(0, result.index)
|
||
afterText = text.substr(result.index + link.length)
|
||
return (
|
||
<div className="errormsg">
|
||
{beforeText}<a href={link}>{link}</a>{afterText}
|
||
</div>
|
||
)
|
||
else
|
||
return (
|
||
<div className="errormsg">
|
||
{text}
|
||
</div>
|
||
)
|
||
|
||
_fieldOnCurrentPage: (field) =>
|
||
!@state.provider.pages || field.page is @state.pageNumber
|
||
|
||
_renderFields: =>
|
||
@state.provider.fields?.filter(@_fieldOnCurrentPage)
|
||
.map (field, idx) =>
|
||
errclass = if field.name in @state.errorFieldNames then "error " else ""
|
||
<label className={(field.className || "")} key={field.name}>
|
||
{field.label}
|
||
<input type={field.type}
|
||
tabIndex={idx + 1}
|
||
value={@state.fields[field.name]}
|
||
onChange={@_onValueChanged}
|
||
onKeyPress={@_onFieldKeyPress}
|
||
data-field={field.name}
|
||
data-format={field.format ? ""}
|
||
disabled={@state.tryingToAuthenticate}
|
||
className={errclass}
|
||
placeholder={field.placeholder} />
|
||
</label>
|
||
|
||
_renderSettings: =>
|
||
@state.provider.settings?.filter(@_fieldOnCurrentPage)
|
||
.map (field, idx) =>
|
||
if field.type is 'checkbox'
|
||
<label className={"checkbox #{field.className ? ""}"} key={field.name}>
|
||
<input type={field.type}
|
||
tabIndex={idx + 5}
|
||
checked={@state.settings[field.name]}
|
||
onChange={@_onSettingsChanged}
|
||
onKeyPress={@_onFieldKeyPress}
|
||
data-field={field.name}
|
||
disabled={@state.tryingToAuthenticate}
|
||
data-format={field.format ? ""}
|
||
className={field.className ? ""} />
|
||
{field.label}
|
||
</label>
|
||
else
|
||
errclass = if field.name in @state.errorFieldNames then "error " else ""
|
||
<label className={field.className ? ""}
|
||
style={if field.advanced and not @state.show_advanced then {display:'none'} else {}}
|
||
key={field.name}>
|
||
{field.label}
|
||
<input type={field.type}
|
||
tabIndex={idx + 5}
|
||
value={@state.settings[field.name]}
|
||
onChange={@_onSettingsChanged}
|
||
onKeyPress={@_onFieldKeyPress}
|
||
data-field={field.name}
|
||
data-format={field.format ? ""}
|
||
disabled={@state.tryingToAuthenticate}
|
||
className={errclass+(field.className ? "")}
|
||
placeholder={field.placeholder} />
|
||
</label>
|
||
|
||
_renderButton: =>
|
||
pages = @state.provider.pages || []
|
||
|
||
if pages.length > @state.pageNumber + 1
|
||
# We're not on the last page.
|
||
if @_noFormErrors() and @_allRequiredFieldsFilled()
|
||
<button className="btn btn-large btn-gradient" onClick={@_onNextButton}>Continue</button>
|
||
else
|
||
# Disable the "Continue" button if the fields haven't been filled correctly.
|
||
<button className="btn btn-large btn-gradient btn-disabled">Continue</button>
|
||
else if @state.provider.name isnt 'gmail'
|
||
if @state.tryingToAuthenticate
|
||
<button className="btn btn-large btn-disabled btn-add-account-spinning">
|
||
<RetinaImg name="sending-spinner.gif" width={15} height={15} mode={RetinaImg.Mode.ContentPreserve} /> Adding account…
|
||
</button>
|
||
else
|
||
if @_noFormErrors() and @_allRequiredFieldsFilled()
|
||
<button className="btn btn-large btn-gradient btn-add-account" onClick={@_onSubmit}>Add account</button>
|
||
else
|
||
# Disable the "Add Account" button if the fields haven't been filled correctly.
|
||
<button className="btn btn-large btn-gradient btn-add-account btn-disabled">Add account</button>
|
||
|
||
_onNextButton: (event) =>
|
||
return unless @_noFormErrors() and @_allRequiredFieldsFilled()
|
||
@setState(pageNumber: @state.pageNumber + 1)
|
||
@_resize()
|
||
|
||
_onSubmit: (event) =>
|
||
return unless @_noFormErrors() and @_allRequiredFieldsFilled()
|
||
return if @state.tryingToAuthenticate
|
||
|
||
data = settings: {}
|
||
for own k,v of @state.fields when v isnt ''
|
||
data[k] = v
|
||
for own k,v of @state.settings when v isnt ''
|
||
data.settings[k] = v
|
||
data.provider = @state.provider.name
|
||
|
||
# if there's an account with this email, get the ID for it to notify the backend of re-auth
|
||
account = AccountStore.accountForEmail(data.email)
|
||
reauthParam = if account then "&reauth=#{account.id}" else ""
|
||
|
||
# handle special case for exchange/outlook/hotmail username field
|
||
if data.provider in ['exchange','outlook','hotmail'] and not data.settings.username?.trim().length
|
||
data.settings.username = data.email
|
||
|
||
@setState(tryingToAuthenticate: true)
|
||
|
||
# Send the form data directly to Nylas to get code
|
||
# If this succeeds, send the received code to N1 server to register the account
|
||
# Otherwise process the error message from the server and highlight UI as needed
|
||
NylasAPI.makeRequest
|
||
path: "/auth?client_id=#{NylasAPI.AppID}&n1_id=#{NylasEnv.config.get('updateIdentity')}#{reauthParam}"
|
||
method: 'POST'
|
||
body: data
|
||
returnsModel: false
|
||
timeout: 60000
|
||
auth:
|
||
user: ''
|
||
pass: ''
|
||
sendImmediately: true
|
||
.then (json) =>
|
||
invite_code = NylasEnv.config.get('invitationCode')
|
||
|
||
json.invite_code = invite_code
|
||
json.email = data.email
|
||
|
||
EdgehillAPI.request
|
||
path: "/connect/nylas"
|
||
method: "POST"
|
||
timeout: 60000
|
||
body: json
|
||
success: @_onAccountReceived
|
||
error: @_onNetworkError
|
||
.catch(@_onNetworkError)
|
||
|
||
_onAccountReceived: (json) =>
|
||
Actions.recordUserEvent('Auth Successful', {
|
||
provider: @state.provider.name
|
||
})
|
||
|
||
try
|
||
OnboardingActions.accountJSONReceived(json)
|
||
catch e
|
||
NylasEnv.reportError(e)
|
||
@setState
|
||
tryingToAuthenticate: false
|
||
errorMessage: "Sorry, something went wrong on the Nylas server. Please try again. If you're still having issues, contact us at support@nylas.com."
|
||
@_resize()
|
||
|
||
_onNetworkError: (err) =>
|
||
errorMessage = err.message
|
||
|
||
Actions.recordUserEvent('Auth Failed', {
|
||
errorMessage: errorMessage
|
||
provider: @state.provider.name
|
||
})
|
||
|
||
if errorMessage is "Invite code required"
|
||
choice = dialog.showMessageBox(remote.getCurrentWindow(), {
|
||
type: 'info',
|
||
buttons: ['Okay'],
|
||
title: 'Confirm',
|
||
message: 'Due to a large number of sign-ups this week, you’ll need an invitation code to add another account! Visit http://invite.nylas.com/ to grab one, or hold tight!'
|
||
})
|
||
OnboardingActions.moveToPage("token-auth")
|
||
|
||
if errorMessage is "Invalid invite code"
|
||
OnboardingActions.moveToPage("token-auth")
|
||
|
||
pageNumber = @state.pageNumber
|
||
errorFieldNames = err.body?.missing_fields || err.body?.missing_settings
|
||
|
||
if err.errorTitle is "setting_update_error"
|
||
@setState
|
||
tryingToAuthenticate: false
|
||
errorMessage: 'The IMAP/SMTP servers for this account do not match our records. Please verify that any server names you entered are correct. If your IMAP/SMTP server has changed, first remove this account from N1, then try logging in again.'
|
||
@_resize()
|
||
OnboardingActions.moveToPage("account-settings")
|
||
return
|
||
|
||
if errorFieldNames
|
||
{pageNumber, errorMessage} = @_stateForMissingFieldNames(errorFieldNames)
|
||
if err.statusCode is -123 # timeout
|
||
errorMessage = "Request timed out. Please try again."
|
||
|
||
@setState
|
||
pageNumber: pageNumber
|
||
errorMessage: errorMessage
|
||
errorFieldNames: errorFieldNames || []
|
||
tryingToAuthenticate: false
|
||
@_resize()
|
||
|
||
_stateForMissingFieldNames: (fieldNames) ->
|
||
fieldLabels = []
|
||
fields = [].concat(@state.provider.settings, @state.provider.fields)
|
||
pageNumbers = [@state.pageNumber]
|
||
|
||
for fieldName in fieldNames
|
||
for s in fields when s.name is fieldName
|
||
fieldLabels.push(s.label.toLowerCase())
|
||
if s.page isnt undefined
|
||
pageNumbers.push(s.page)
|
||
|
||
pageNumber = Math.min.apply(null, pageNumbers)
|
||
errorMessage = @_messageForFieldLabels(fieldLabels)
|
||
|
||
{pageNumber, errorMessage}
|
||
|
||
_messageForFieldLabels: (labels) ->
|
||
if labels.length > 2
|
||
return "Please fix the highlighted fields."
|
||
else if labels.length is 2
|
||
return "Please provide your #{labels[0]} and #{labels[1]}."
|
||
else
|
||
return "Please provide your #{labels[0]}."
|
||
|
||
_pollForGmailAccount: (callback) =>
|
||
EdgehillAPI.request
|
||
path: "/oauth/google/token?key="+@state.provider.clientKey
|
||
method: "GET"
|
||
success: (json) =>
|
||
callback(json)
|
||
error: (err) =>
|
||
callback()
|
||
|
||
_resize: =>
|
||
setTimeout( =>
|
||
@props.onResize?()
|
||
, 10)
|
||
|
||
_fireMoveToPrevPage: =>
|
||
if @state.pageNumber > 0
|
||
@setState(pageNumber: @state.pageNumber - 1)
|
||
@_resize()
|
||
else
|
||
OnboardingActions.moveToPreviousPage()
|
||
|
||
module.exports = AccountSettingsPage
|