mirror of
https://github.com/codeninjasllc/codecombat.git
synced 2024-11-29 18:45:48 -05:00
e748417007
anonymous users are now warned if their new name is already chosen User Settings is now without auto-save Upon name conflict, a user will be suggested a new name which is then submitted if the user chooses to save after all. Refactored conflicted name checking so it can be used in more places Signup form now has an optional name field Covered extra case where the debounced check happened too late. Support for submitting on enter. Worked in scott's comments and got tests working again
230 lines
7.7 KiB
CoffeeScript
230 lines
7.7 KiB
CoffeeScript
View = require 'views/kinds/RootView'
|
|
template = require 'templates/account/settings'
|
|
{me} = require 'lib/auth'
|
|
forms = require 'lib/forms'
|
|
User = require 'models/User'
|
|
AuthModalView = require 'views/modal/auth_modal'
|
|
|
|
WizardSettingsView = require './wizard_settings_view'
|
|
JobProfileView = require './job_profile_view'
|
|
|
|
module.exports = class SettingsView extends View
|
|
id: 'account-settings-view'
|
|
template: template
|
|
changedFields: [] # DOM input fields
|
|
|
|
events:
|
|
'click #save-button': 'save'
|
|
'change #settings-panes input:checkbox': (e) -> @trigger 'checkboxToggled', e
|
|
'keyup #settings-panes input:text, #settings-panes input:password': (e) -> @trigger 'inputChanged', e
|
|
'keyup #name': 'onNameChange'
|
|
'click #toggle-all-button': 'toggleEmailSubscriptions'
|
|
'keypress #settings-panes': 'onKeyPress'
|
|
|
|
constructor: (options) ->
|
|
@save = _.debounce(@save, 200)
|
|
@onNameChange = _.debounce @checkNameExists, 500
|
|
super options
|
|
return unless me
|
|
|
|
@listenTo(me, 'invalid', (errors) -> forms.applyErrorsToForm(@$el, me.validationError))
|
|
@on 'checkboxToggled', @onToggle
|
|
@on 'checkboxToggled', @onInputChanged
|
|
@on 'inputChanged', @onInputChanged
|
|
@on 'enterPressed', @onEnter
|
|
|
|
onInputChanged: (e) ->
|
|
return @enableSaveButton() unless e?.currentTarget
|
|
that = e.currentTarget
|
|
$that = $(that)
|
|
savedValue = $that.data 'saved-value'
|
|
currentValue = $that.val()
|
|
if savedValue isnt currentValue
|
|
@changedFields.push that unless that in @changedFields
|
|
@enableSaveButton()
|
|
else
|
|
_.pull @changedFields, that
|
|
@disableSaveButton() if _.isEmpty @changedFields
|
|
|
|
onToggle: (e) ->
|
|
$that = $(e.currentTarget)
|
|
$that.val $that[0].checked
|
|
|
|
onEnter: ->
|
|
@save()
|
|
|
|
onKeyPress: (e) ->
|
|
@trigger 'enterPressed', e if e.which is 13
|
|
|
|
enableSaveButton: ->
|
|
$('#save-button', @$el).removeClass 'disabled'
|
|
$('#save-button', @$el).removeClass 'btn-danger'
|
|
$('#save-button', @$el).removeAttr 'disabled'
|
|
$('#save-button', @$el).text 'Save'
|
|
|
|
disableSaveButton: ->
|
|
$('#save-button', @$el).addClass 'disabled'
|
|
$('#save-button', @$el).removeClass 'btn-danger'
|
|
$('#save-button', @$el).attr 'disabled', "true"
|
|
$('#save-button', @$el).text 'No Changes'
|
|
|
|
checkNameExists: =>
|
|
name = $('#name', @$el).val()
|
|
return if name is me.get 'name'
|
|
User.getUnconflictedName name, (newName) =>
|
|
forms.clearFormAlerts(@$el)
|
|
if name is newName
|
|
@suggestedName = undefined
|
|
else
|
|
@suggestedName = newName
|
|
forms.setErrorToProperty @$el, 'name', "That name is taken! How about #{newName}?", true
|
|
|
|
afterRender: ->
|
|
super()
|
|
$('#settings-tabs a', @$el).click((e) =>
|
|
e.preventDefault()
|
|
$(e.target).tab('show')
|
|
|
|
# make sure errors show up in the general pane, but keep the password pane clean
|
|
$('#password-pane input').val('')
|
|
@save() unless $(e.target).attr('href') is '#password-pane'
|
|
forms.clearFormAlerts($('#password-pane', @$el))
|
|
)
|
|
|
|
@chooseTab(location.hash.replace('#', ''))
|
|
|
|
wizardSettingsView = new WizardSettingsView()
|
|
@listenTo wizardSettingsView, 'change', @save
|
|
@insertSubView wizardSettingsView
|
|
|
|
@jobProfileView = new JobProfileView()
|
|
@listenTo @jobProfileView, 'change', @save
|
|
@insertSubView @jobProfileView
|
|
_.defer => @buildPictureTreema() # Not sure why, but the Treemas don't fully build without this if you reload the page.
|
|
|
|
afterInsert: ->
|
|
super()
|
|
if me.get('anonymous')
|
|
@openModalView new AuthModalView()
|
|
@updateSavedValues()
|
|
|
|
chooseTab: (category) ->
|
|
id = "##{category}-pane"
|
|
pane = $(id, @$el)
|
|
return @chooseTab('general') unless pane.length or category is 'general'
|
|
loc = "a[href=#{id}]"
|
|
$(loc, @$el).tab('show')
|
|
$('.tab-pane').removeClass('active')
|
|
pane.addClass('active')
|
|
@currentTab = category
|
|
|
|
getRenderData: ->
|
|
c = super()
|
|
return c unless me
|
|
c.subs = {}
|
|
c.subs[sub] = 1 for sub in c.me.getEnabledEmails()
|
|
c.showsJobProfileTab = me.isAdmin() or me.get('jobProfile') or location.hash.search('job-profile-') isnt -1
|
|
c
|
|
|
|
getSubscriptions: ->
|
|
inputs = ($(i) for i in $('#email-pane input[type="checkbox"].changed', @$el))
|
|
emailNames = (i.attr('name').replace('email_', '') for i in inputs)
|
|
enableds = (i.prop('checked') for i in inputs)
|
|
_.zipObject emailNames, enableds
|
|
|
|
toggleEmailSubscriptions: =>
|
|
subs = @getSubscriptions()
|
|
$('#email-pane input[type="checkbox"]', @$el).prop('checked', not _.any(_.values(subs))).addClass('changed')
|
|
@save()
|
|
|
|
buildPictureTreema: ->
|
|
data = photoURL: me.get('photoURL')
|
|
data.photoURL = null if data.photoURL?.search('gravatar') isnt -1 # Old style
|
|
schema = $.extend true, {}, me.schema()
|
|
schema.properties = _.pick me.schema().properties, 'photoURL'
|
|
schema.required = ['photoURL']
|
|
treemaOptions =
|
|
filePath: "db/user/#{me.id}"
|
|
schema: schema
|
|
data: data
|
|
callbacks: {change: @onPictureChanged}
|
|
|
|
@pictureTreema = @$el.find('#picture-treema').treema treemaOptions
|
|
@pictureTreema?.build()
|
|
@pictureTreema?.open()
|
|
@$el.find('.gravatar-fallback').toggle not me.get 'photoURL'
|
|
|
|
onPictureChanged: (e) =>
|
|
@trigger 'inputChanged', e
|
|
@$el.find('.gravatar-fallback').toggle not me.get 'photoURL'
|
|
|
|
save: (e) ->
|
|
$(e.target).addClass('changed') if e
|
|
forms.clearFormAlerts(@$el)
|
|
@grabData()
|
|
res = me.validate()
|
|
if res?
|
|
console.error 'Couldn\'t save because of validation errors:', res
|
|
forms.applyErrorsToForm(@$el, res)
|
|
return
|
|
|
|
return unless me.hasLocalChanges()
|
|
|
|
res = me.patch()
|
|
return unless res
|
|
save = $('#save-button', @$el).text($.i18n.t('common.saving', defaultValue: 'Saving...'))
|
|
.removeClass('btn-danger').addClass('btn-success').show()
|
|
|
|
res.error ->
|
|
errors = JSON.parse(res.responseText)
|
|
forms.applyErrorsToForm(@$el, errors)
|
|
save.text($.i18n.t('account_settings.error_saving', defaultValue: 'Error Saving')).removeClass('btn-success').addClass('btn-danger', 500)
|
|
res.success (model, response, options) =>
|
|
@changedFields = []
|
|
@updateSavedValues()
|
|
save.text($.i18n.t('account_settings.saved', defaultValue: 'Changes Saved')).removeClass('btn-success', 500)
|
|
|
|
grabData: ->
|
|
@grabPasswordData()
|
|
@grabOtherData()
|
|
|
|
grabPasswordData: ->
|
|
password1 = $('#password', @$el).val()
|
|
password2 = $('#password2', @$el).val()
|
|
bothThere = Boolean(password1) and Boolean(password2)
|
|
if bothThere and password1 isnt password2
|
|
message = $.i18n.t('account_settings.password_mismatch', defaultValue: 'Password does not match.')
|
|
err = [message: message, property: 'password2', formatted: true]
|
|
forms.applyErrorsToForm(@$el, err)
|
|
return
|
|
if bothThere
|
|
me.set('password', password1)
|
|
|
|
grabOtherData: ->
|
|
$('#name', @$el).val @suggestedName if @suggestedName
|
|
me.set 'name', $('#name', @$el).val()
|
|
me.set 'email', $('#email', @$el).val()
|
|
for emailName, enabled of @getSubscriptions()
|
|
me.setEmailSubscription emailName, enabled
|
|
me.set 'photoURL', @pictureTreema.get('/photoURL')
|
|
|
|
adminCheckbox = @$el.find('#admin')
|
|
if adminCheckbox.length
|
|
permissions = []
|
|
permissions.push 'admin' if adminCheckbox.prop('checked')
|
|
me.set('permissions', permissions)
|
|
|
|
jobProfile = me.get('jobProfile') ? {}
|
|
updated = false
|
|
for key, val of @jobProfileView.getData()
|
|
updated = updated or jobProfile[key] isnt val
|
|
jobProfile[key] = val
|
|
if updated
|
|
jobProfile.updated = (new Date()).toISOString()
|
|
me.set 'jobProfile', jobProfile
|
|
|
|
updateSavedValues: ->
|
|
$('#settings-panes input:text').each ->
|
|
$(@).data 'saved-value', $(@).val()
|
|
$('#settings-panes input:checkbox').each ->
|
|
$(@).data 'saved-value', JSON.stringify $(@)[0].checked
|