From a5f18f88be18a3e95dc89361e950ba31747cf26d Mon Sep 17 00:00:00 2001
From: Scott Erickson
Date: Thu, 11 Aug 2016 23:08:09 -0700
Subject: [PATCH 1/7] Add script that migrates users with email-formatted
usernames
---
.../node/fixEmailFormattedUsernames.coffee | 167 ++++++++++++++++++
server/models/User.coffee | 10 +-
server/routes/index.coffee | 2 +-
server/sendwithus.coffee | 3 +
.../fixEmailFormattedUsernames.spec.coffee | 115 ++++++++++++
5 files changed, 293 insertions(+), 4 deletions(-)
create mode 100644 scripts/node/fixEmailFormattedUsernames.coffee
create mode 100644 spec/server/scripts/fixEmailFormattedUsernames.spec.coffee
diff --git a/scripts/node/fixEmailFormattedUsernames.coffee b/scripts/node/fixEmailFormattedUsernames.coffee
new file mode 100644
index 000000000..516daa072
--- /dev/null
+++ b/scripts/node/fixEmailFormattedUsernames.coffee
@@ -0,0 +1,167 @@
+# Usage:
+# > coffee -c scripts/node/fixEmailFormattedUsernames.coffee; node scripts/node/fixEmailFormattedUsernames.js run
+
+require('coffee-script');
+require('coffee-script/register');
+
+_ = require 'lodash'
+sendwithus = require '../../server/sendwithus'
+log = require 'winston'
+str = require 'underscore.string'
+co = require 'co'
+
+filter = /^[A-Z0-9._%+-]+@[A-Z0-9.-]+\.[A-Z]{2,63}$/i
+
+changedUsernameTemplate = _.template("
+
+ Hi, CodeCombat user!
+
+
+
+ Just letting you know we've made a change to your account settings which may change how you log in. Here are your old settings:
+
+
+
+ - Old username: <%= oldUsername %>
+ - Old email: <%= oldEmail %>
+
+
+
+ Your old username was an email address, but to reduce confusion, we now make sure email addresses can't be used as usernames.
+ <%= specialMessage %>
+ Here are your new settings:
+
+
+
+ - New username: <%= newUsername %>
+ - New email: <%= newEmail %>
+
+
+
+ Please visit the site if you would like to update your settings.
+ And let us know if you have any questions!
+
+")
+
+exports.run = ->
+ co ->
+ mongoose = require 'mongoose'
+ User = require '../../server/models/User'
+ users = yield User.find({nameLower: {$regex: filter}}).select({name:1, email:1, anonymous:1, slug:1})
+ console.log 'found', users.length, 'users'
+
+ for user in users
+ oldUsername = user.get('name')
+ oldEmail = user.get('email')
+ newUsername = null
+ newEmail = null
+ specialMessage = ''
+
+ if not oldEmail
+ otherUser = yield User.findByEmail(oldUsername)
+ if otherUser
+ specialMessage = "Since you had no email set, we would have made your old username your new email.
+ But '#{oldUsername}' is already used by another account as an email by another account,
+ so instead we changed your username."
+ newUsername = str.slugify(oldUsername)
+ newEmail = ''
+
+ else
+ specialMessage = "Since you had no email set, we simply made your old username your new email instead."
+ newEmail = oldUsername
+ newUsername = ''
+
+
+ else if oldEmail is oldUsername
+ specialMessage = "Since your email and username are the same, we simply removed your username."
+ newUsername = ''
+ newEmail = oldEmail
+
+
+ else if not filter.test(oldEmail)
+ otherEmailUser = yield User.findByEmail(oldUsername)
+ otherUsernameUser = yield User.findByName(oldEmail)
+ if otherEmailUser
+ specialMessage = "Since your old email looks like a username and your old username looks like an email,
+ we would have swapped them on your account.
+ But '#{oldUsername}' is already used as an email by another account,
+ so instead we changed your username."
+ newUsername = str.slugify(oldUsername)
+ newEmail = oldEmail
+
+ else if otherUsernameUser
+ specialMessage = "Since your old email looks like a username and your old username looks like an email,
+ we would have swapped them on your account.
+ But '#{oldEmail}' is already used as a username by another account,
+ so instead we changed your username."
+ newUsername = str.slugify(oldUsername)
+ newEmail = oldEmail
+ else
+ specialMessage = "Since your old email looks like a username and your old username looks like an email,
+ we swapped them on your account."
+ newUsername = oldEmail
+ newEmail = oldUsername
+
+
+ else if oldUsername and oldEmail
+ # Since oldEmail passed the email filter,
+ specialMessage = "Since your old email is valid, we simply removed your username."
+ newUsername = ''
+ newEmail = oldEmail
+
+
+ else
+ console.log('unhandled user', user.toObject())
+ throw new Error('Unhandled user')
+
+ user.set({name: newUsername, email: newEmail})
+ console.log JSON.stringify({
+ oldUsername, oldEmail, newUsername, newEmail, specialMessage, _id: user.id
+ })
+ yield user.save()
+
+ content = changedUsernameTemplate({
+ oldUsername: oldUsername or '(no username)'
+ oldEmail: oldEmail or '(no email)'
+ newUsername: newUsername or '(no username)'
+ newEmail: newEmail or '(no email)'
+ specialMessage
+ })
+
+ context =
+ template: sendwithus.templates.plain_text_email
+ recipient:
+ address: oldUsername
+ sender:
+ address: 'team@codecombat.com'
+ name: 'CodeCombat Team'
+ template_data:
+ subject: 'Your Username Has Changed'
+ contentHTML: content
+
+ # Also send to the original email if it's valid
+ if filter.test(oldEmail)
+ context.cc = [
+ { address: oldEmail }
+ ]
+
+ yield sendwithus.api.sendAsync(context)
+
+ return 'Done'
+
+if _.last(process.argv) is 'run'
+ database = require '../../server/commons/database'
+ mongoose = require 'mongoose'
+
+ ### SET UP ###
+ do (setupLodash = this) ->
+ GLOBAL._ = require 'lodash'
+ _.str = require 'underscore.string'
+ _.mixin _.str.exports()
+ GLOBAL.tv4 = require('tv4').tv4
+
+ database.connect()
+ co ->
+ yield exports.run()
+ process.exit()
+
diff --git a/server/models/User.coffee b/server/models/User.coffee
index 3d843d00e..b9332d9e6 100644
--- a/server/models/User.coffee
+++ b/server/models/User.coffee
@@ -10,6 +10,7 @@ Classroom = require '../models/Classroom'
languages = require '../routes/languages'
_ = require 'lodash'
errors = require '../commons/errors'
+Promise = require 'bluebird'
config = require '../../server_config'
stripe = require('stripe')(config.stripe.secretKey)
@@ -270,6 +271,8 @@ UserSchema.statics.unconflictName = unconflictName = (name, done) ->
suffix = _.random(0, 9) + ''
unconflictName name + suffix, done
+UserSchema.statics.unconflictNameAsync = Promise.promisify(unconflictName)
+
UserSchema.methods.sendWelcomeEmail = ->
return if not @get('email')
{ welcome_email_student, welcome_email_user } = sendwithus.templates
@@ -361,9 +364,10 @@ UserSchema.pre('save', (next) ->
@set('email', undefined)
@set('emailLower', undefined)
if name = @get('name')
- filter = /^[A-Z0-9._%+-]+@[A-Z0-9.-]+\.[A-Z]{2,63}$/i # https://news.ycombinator.com/item?id=5763990
- if filter.test(name)
- return next(new errors.UnprocessableEntity('Name may not be an email'))
+ if not @allowEmailNames # for testing
+ filter = /^[A-Z0-9._%+-]+@[A-Z0-9.-]+\.[A-Z]{2,63}$/i # https://news.ycombinator.com/item?id=5763990
+ if filter.test(name)
+ return next(new errors.UnprocessableEntity('Name may not be an email'))
@set('nameLower', name.toLowerCase())
else
diff --git a/server/routes/index.coffee b/server/routes/index.coffee
index 9d614b192..56b733aff 100644
--- a/server/routes/index.coffee
+++ b/server/routes/index.coffee
@@ -122,4 +122,4 @@ module.exports.setup = (app) ->
app.put('/db/trial.request/:handle', mw.auth.checkHasPermission(['admin']), mw.trialRequests.put)
app.get('/db/trial.request/-/users', mw.auth.checkHasPermission(['admin']), mw.trialRequests.getUsers)
- app.get('/healthcheck', mw.healthcheck)
+ app.get('/healthcheck', mw.healthcheck)
diff --git a/server/sendwithus.coffee b/server/sendwithus.coffee
index 933f141ed..cba9de8a3 100644
--- a/server/sendwithus.coffee
+++ b/server/sendwithus.coffee
@@ -2,6 +2,7 @@ config = require '../server_config'
sendwithusAPI = require 'sendwithus'
swuAPIKey = config.mail.sendwithusAPIKey
log = require 'winston'
+Promise = require 'bluebird'
module.exports.setupRoutes = (app) ->
return
@@ -14,6 +15,8 @@ module.exports.api =
if swuAPIKey
module.exports.api = new sendwithusAPI swuAPIKey, debug
+
+Promise.promisifyAll(module.exports.api)
module.exports.templates =
parent_subscribe_email: 'tem_2APERafogvwKhmcnouigud'
diff --git a/spec/server/scripts/fixEmailFormattedUsernames.spec.coffee b/spec/server/scripts/fixEmailFormattedUsernames.spec.coffee
new file mode 100644
index 000000000..b432ecdf3
--- /dev/null
+++ b/spec/server/scripts/fixEmailFormattedUsernames.spec.coffee
@@ -0,0 +1,115 @@
+# Don't need to run this regularly, only running script once.
+
+#utils = require '../utils'
+#User = require '../../../server/models/User'
+#request = require '../request'
+#sendwithus = require '../../../server/sendwithus'
+#fixEmailFormattedUsernames = require '../../../scripts/node/fixEmailFormattedUsernames'
+#
+#describe '/scripts/node/fixEmailFormattedUsernames', ->
+#
+# beforeEach utils.wrap (done) ->
+# yield utils.clearModels([User])
+# console.log('spy on send async')
+# spyOn(sendwithus.api, 'sendAsync').and.callThrough()
+# done()
+#
+# afterEach ->
+# expect(sendwithus.api.sendAsync).toHaveBeenCalled()
+#
+# describe "when a user has no email set", ->
+# beforeEach utils.wrap (done) ->
+# @user = new User({name: 'an@email.com', points:100})
+# @user.allowEmailNames = true
+# yield @user.save()
+# done()
+#
+# it 'moves the email-formatted username to be the user\'s email', utils.wrap (done) ->
+# yield fixEmailFormattedUsernames.run()
+# user = yield User.findById(@user.id)
+# expect(user.get('email')).toBe('an@email.com')
+# expect(user.get('name')).toBeUndefined()
+# expect(user.get('points')).toBe(100) # make sure properties aren't removed
+# done()
+#
+#
+# describe "when another user exists with that email", ->
+# beforeEach utils.wrap (done) ->
+# @otherUser = new User({email: 'an@email.com'})
+# yield @otherUser.save()
+# done()
+#
+# it "slugifies the target user's username", utils.wrap (done) ->
+# yield fixEmailFormattedUsernames.run()
+# user = yield User.findById(@user.id)
+# expect(user.get('email')).toBeUndefined()
+# expect(user.get('name')).toBe('anemailcom')
+# done()
+#
+# describe "when a user has the same email and username", ->
+# beforeEach utils.wrap (done) ->
+# @user = new User({name: 'an@email.com', email: 'an@email.com'})
+# @user.allowEmailNames = true
+# yield @user.save()
+# done()
+#
+# it "removes the user's username", utils.wrap (done) ->
+# yield fixEmailFormattedUsernames.run()
+# user = yield User.findById(@user.id)
+# expect(user.get('email')).toBe('an@email.com')
+# expect(user.get('name')).toBeUndefined()
+# done()
+#
+# describe "when the user has an email that isn't formatted like an email", ->
+# beforeEach utils.wrap (done) ->
+# @user = new User({name: 'an@email.com', email: 'a name'})
+# @user.allowEmailNames = true
+# yield @user.save()
+# done()
+#
+# it "swaps the two", utils.wrap (done) ->
+# yield fixEmailFormattedUsernames.run()
+# user = yield User.findById(@user.id)
+# expect(user.get('email')).toBe('an@email.com')
+# expect(user.get('name')).toBe('a name')
+# done()
+#
+# describe "when another user already has the email-formatted name as an email", ->
+# beforeEach utils.wrap (done) ->
+# @otherUser = new User({email: 'an@email.com'})
+# yield @otherUser.save()
+# done()
+#
+# it "slugifies the target user's username", utils.wrap (done) ->
+# yield fixEmailFormattedUsernames.run()
+# user = yield User.findById(@user.id)
+# expect(user.get('email')).toBe('a name')
+# expect(user.get('name')).toBe('anemailcom')
+# done()
+#
+# describe "when another user already has the non-email-formatted email as a username", ->
+# beforeEach utils.wrap (done) ->
+# @otherUser = new User({name: 'a name'})
+# yield @otherUser.save()
+# done()
+#
+# it "slugifies the target user's username", utils.wrap (done) ->
+# yield fixEmailFormattedUsernames.run()
+# user = yield User.findById(@user.id)
+# expect(user.get('email')).toBe('a name')
+# expect(user.get('name')).toBe('anemailcom')
+# done()
+#
+# describe "when the user has a different but well formatted email set", ->
+# beforeEach utils.wrap (done) ->
+# @user = new User({name: 'an@email.com', email: 'another@email.com'})
+# @user.allowEmailNames = true
+# yield @user.save()
+# done()
+#
+# it "removes the target user's username", utils.wrap (done) ->
+# yield fixEmailFormattedUsernames.run()
+# user = yield User.findById(@user.id)
+# expect(user.get('email')).toBe('another@email.com')
+# expect(user.get('name')).toBeUndefined()
+# done()
From 2dadc8458c856db9d264f46d1a2c9b6cb76252f2 Mon Sep 17 00:00:00 2001
From: Scott Erickson
Date: Fri, 12 Aug 2016 14:38:32 -0700
Subject: [PATCH 2/7] Remove hardcoded student ui game dev play instructions
---
app/views/play/level/PlayLevelView.coffee | 20 +-------------------
1 file changed, 1 insertion(+), 19 deletions(-)
diff --git a/app/views/play/level/PlayLevelView.coffee b/app/views/play/level/PlayLevelView.coffee
index 3488d1ec5..e0e982dd9 100644
--- a/app/views/play/level/PlayLevelView.coffee
+++ b/app/views/play/level/PlayLevelView.coffee
@@ -641,25 +641,7 @@ module.exports = class PlayLevelView extends RootView
# Real-time playback
onRealTimePlaybackStarted: (e) ->
@$el.addClass('real-time').focus()
- if @level.isType('game-dev')
- panel = @$('#how-to-play-game-dev-panel')
- panel.removeClass('hide')
- # TODO: Remove this once these levels have studentPlayInstructions set.
- if not @level.get('studentPlayInstructions')
- lines = switch @level.get('slug')
- when 'over-the-garden-wall' then ['Watch to see if the peasants are properly protected.']
- when 'click-gait' then ['Move to each red "X".', 'Click on the screen to move the Knight there.']
- when 'heros-journey' then ['Move to each red "X".', 'Click on the screen to move the Knight there.']
- when 'a-maze-ing' then ['Move to the chest of gems.', 'Click on the screen to move the Duelist there.']
- when 'gemtacular' then ['Move to each of the gems.', 'Click on the screen to move the Captain there.']
- when 'vorpal-mouse' then ['Slay the ogres.', 'Click on the screen to move the Guardian there.', 'Click on the munchkins to attack them!']
- when 'crushing-it' then ['Slay the ogres.', 'Click on the screen to move the Goliath there.', 'Click on the munchkins to attack them!']
- when 'tabula-rasa' then ['Slay any ogres.', 'Collect any coins.', 'Click on the screen to move the Raider there.', 'Click on any munchkins to attack them!']
- else null
- if lines
- html = _.map(lines, (line) -> "#{line}
").join('')
- panel.find('.panel-body').html(html)
-
+ @$('#how-to-play-game-dev-panel').removeClass('hide') if @level.isType('game-dev')
@onWindowResize()
onRealTimePlaybackEnded: (e) ->
From c2ce8048398af0c188afce50700853b008cbe90a Mon Sep 17 00:00:00 2001
From: Phoenix Eliot
Date: Fri, 12 Aug 2016 16:28:29 -0700
Subject: [PATCH 3/7] Fix worker-based syntax annotations
---
app/views/play/level/tome/SpellView.coffee | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/app/views/play/level/tome/SpellView.coffee b/app/views/play/level/tome/SpellView.coffee
index 537158ddf..b4cc1d44b 100644
--- a/app/views/play/level/tome/SpellView.coffee
+++ b/app/views/play/level/tome/SpellView.coffee
@@ -803,7 +803,7 @@ module.exports = class SpellView extends CocoView
# This function itself removes the unwanted annotations on a later tick.
onChangeAnnotation: (event, session) ->
unfilteredAnnotations = session.getAnnotations()
- filteredAnnotations = _.remove unfilteredAnnotations, (annotation) ->
+ filteredAnnotations = _.reject unfilteredAnnotations, (annotation) ->
annotation.text is 'Start tag seen without seeing a doctype first. Expected e.g. .'
if filteredAnnotations.length < unfilteredAnnotations.length
session.setAnnotations(filteredAnnotations)
From 803fc39998cacb206a9f54db5a0eed8b788db2ef Mon Sep 17 00:00:00 2001
From: iammehmetguler
Date: Sat, 13 Aug 2016 13:26:39 +0300
Subject: [PATCH 4/7] Update tr.coffee (#3842)
* Update tr.coffee
calendar
* Update tr.coffee
login, signup
---
app/locale/tr.coffee | 50 ++++++++++++++++++++++----------------------
1 file changed, 25 insertions(+), 25 deletions(-)
diff --git a/app/locale/tr.coffee b/app/locale/tr.coffee
index 70b8f388d..6dbf63e35 100644
--- a/app/locale/tr.coffee
+++ b/app/locale/tr.coffee
@@ -243,7 +243,7 @@ module.exports = nativeDescription: "Türkçe", englishDescription: "Turkish", t
login:
sign_up: "Hesap Oluştur"
-# email_or_username: "Email or username"
+ email_or_username: "E-posta veya kullanıcı adı"
log_in: "Giriş Yap"
logging_in: "Giriş Yapılıyor"
log_out: "Çıkış Yap"
@@ -256,13 +256,13 @@ module.exports = nativeDescription: "Türkçe", englishDescription: "Turkish", t
signup_switch: "Hesap oluşturmak istiyor musun?"
signup:
-# create_student_header: "Create Student Account"
-# create_teacher_header: "Create Teacher Account"
-# create_individual_header: "Create Individual Account"
-# create_header: "Create Account"
+ create_student_header: "Öğrenci Hesabı Oluştur"
+ create_teacher_header: "Öğretmen Hesabı Oluştur"
+ create_individual_header: "Bireysel Hesap Oluştur"
+ create_header: "Hesap Oluştur"
email_announcements: "E-posta duyurularını almak istiyorum" # {change}
creating: "Hesap oluşturuluyor..."
-# create_account: "Create Account"
+ create_account: "Hesap Oluştur"
sign_up: "Kaydol"
log_in: "buradan giriş yapabilirsiniz."
required: "Buraya gidebilmeniz için oturum açmanız gerekli."
@@ -278,46 +278,46 @@ module.exports = nativeDescription: "Türkçe", englishDescription: "Turkish", t
connected_facebook_p: "Kayıt işlemini bitir, artık Facebook hesabınla giriş yapabilirsin."
facebook_exists: "Zaten Facebook ile ilişkilendirilmiş bir hesabın bulunuyor!"
hey_students: "Öğrenciler, öğretmeninizin verdiği sınıf kodunu girin."
-# birthday: "Birthday"
+ birthday: "Doğum günü"
# parent_email_blurb: "We know you can't wait to learn programming — we're excited too! Your parents will receive an email with further instructions on how to create an account for you. Email {{email_link}} if you have any questions."
# classroom_not_found: "No classes exist with this Class Code. Check your spelling or ask your teacher for help."
-# checking: "Checking..."
+ checking: "Kontrol ediliyor..."
# account_exists: "This email is already in use:" # {change}
-# sign_in: "Sign in"
-# email_good: "Email looks good!"
+ sign_in: "Oturum aç"
+ email_good: "E-posta iyi görünüyor!"
# name_taken: "Username already taken! Try {{suggestedName}}?"
-# name_available: "Username available!"
+ name_available: "Kullanıcı adı müsait!"
# name_is_email: "Username may not be an email"
-# choose_type: "Choose your account type:"
+ choose_type: "Hesabınızın türünü seçin:"
# teacher_type_1: "Teach programming using CodeCombat!"
# teacher_type_2: "Set up your class"
# teacher_type_3: "Access Course Guides"
# teacher_type_4: "View student progress"
-# signup_as_teacher: "Sign up as a Teacher"
-# student_type_1: "Learn to program while playing an engaging game!"
-# student_type_2: "Play with your class"
-# student_type_3: "Compete in arenas"
-# student_type_4: "Choose your hero!"
+ signup_as_teacher: "Öğretmen olarak Kaydol"
+ student_type_1: "İlgi çekici bir oyun oynarken program öğren!"
+ student_type_2: "Sınıfınla oyna"
+ student_type_3: "Arenalarda rekabet et"
+ student_type_4: "Kahramanını seç!"
# student_type_5: "Have your Class Code ready!"
-# signup_as_student: "Sign up as a Student"
+ signup_as_student: "Öğrenci olarak Kaydol"
# individuals_or_parents: "Individuals & Parents"
# individual_type: "For players learning to code outside of a class. Parents should sign up for an account here."
-# signup_as_individual: "Sign up as an Individual"
-# enter_class_code: "Enter your Class Code"
-# enter_birthdate: "Enter your birthdate:"
-# ask_teacher_1: "Ask your teacher for your Class Code."
+ signup_as_individual: "Bireysel olarak Kaydol"
+ enter_class_code: "Sınıf Kodunu Gir"
+ enter_birthdate: "Doğum tarihini gir:"
+ ask_teacher_1: "Sınıf kodun için öğretmenine sor."
# ask_teacher_2: "Not part of a class? Create an "
-# ask_teacher_3: "Individual Account"
+ ask_teacher_3: "Bireysel Hesap"
# ask_teacher_4: " instead."
# about_to_join: "You're about to join:"
# enter_parent_email: "Enter your parent’s email address:"
# parent_email_error: "Something went wrong when trying to send the email. Check the email address and try again."
# parent_email_sent: "We’ve sent an email with further instructions on how to create an account. Ask your parent to check their inbox."
-# account_created: "Account Created!"
+ account_created: "Hesap Oluşturuldu!"
# confirm_student_blurb: "Write down your information so that you don't forget it. Your teacher can also help you reset your password at any time."
# confirm_individual_blurb: "Write down your login information in case you need it later. Verify your email so you can recover your account if you ever forget your password - check your inbox!"
# write_this_down: "Write this down:"
-# start_playing: "Start Playing!"
+ start_playing: "Oynamaya Başla!"
# sso_connected: "Successfully connected with:"
recover:
From e9c7edb6be1527398c0ef638d720610968f9f493 Mon Sep 17 00:00:00 2001
From: Scott Erickson
Date: Mon, 15 Aug 2016 11:53:57 -0700
Subject: [PATCH 5/7] Fix populate i18n to also populate thang component
configs
---
app/models/CocoModel.coffee | 8 +++++++-
app/views/editor/level/LevelEditView.coffee | 14 ++++++++++++++
2 files changed, 21 insertions(+), 1 deletion(-)
diff --git a/app/models/CocoModel.coffee b/app/models/CocoModel.coffee
index 90310d5de..8441d2bc0 100644
--- a/app/models/CocoModel.coffee
+++ b/app/models/CocoModel.coffee
@@ -309,6 +309,8 @@ class CocoModel extends Backbone.Model
sum = 0
data ?= $.extend true, {}, @attributes
schema ?= @schema() or {}
+ if schema.oneOf # get populating the Programmable component config to work
+ schema = _.find(schema.oneOf, {type: 'object'})
addedI18N = false
if schema.properties?.i18n and _.isPlainObject(data) and not data.i18n?
data.i18n = {'-':{'-':'-'}} # mongoose doesn't work with empty objects
@@ -318,7 +320,11 @@ class CocoModel extends Backbone.Model
if _.isPlainObject data
for key, value of data
numChanged = 0
- numChanged = @populateI18N(value, childSchema, path+'/'+key) if childSchema = schema.properties?[key]
+ childSchema = schema.properties?[key]
+ if not childSchema and _.isObject(schema.additionalProperties)
+ childSchema = schema.additionalProperties
+ if childSchema
+ numChanged = @populateI18N(value, childSchema, path+'/'+key)
if numChanged and not path # should only do this for the root object
@set key, value
sum += numChanged
diff --git a/app/views/editor/level/LevelEditView.coffee b/app/views/editor/level/LevelEditView.coffee
index 0f9b0d216..230d18a94 100644
--- a/app/views/editor/level/LevelEditView.coffee
+++ b/app/views/editor/level/LevelEditView.coffee
@@ -2,6 +2,7 @@ RootView = require 'views/core/RootView'
template = require 'templates/editor/level/edit'
Level = require 'models/Level'
LevelSystem = require 'models/LevelSystem'
+LevelComponent = require 'models/LevelComponent'
World = require 'lib/world/world'
DocumentFiles = require 'collections/DocumentFiles'
LevelLoader = require 'lib/LevelLoader'
@@ -217,6 +218,19 @@ module.exports = class LevelEditView extends RootView
onPopulateI18N: ->
@level.populateI18N()
+
+ levelComponentMap = _(currentView.supermodel.getModels(LevelComponent))
+ .map((c) -> [c.get('original'), c])
+ .object()
+ .value()
+
+ for thang, thangIndex in @level.get('thangs')
+ for thangComponent, thangComponentIndex in thang.components
+ component = levelComponentMap[thangComponent.original]
+ configSchema = component.get('configSchema')
+ path = "/thangs/#{thangIndex}/components/#{thangComponentIndex}/config"
+ @level.populateI18N(thangComponent.config, configSchema, path)
+
f = -> document.location.reload()
setTimeout(f, 2000)
From b308da0473f6e60291861cdc90b33ee0a01cbc95 Mon Sep 17 00:00:00 2001
From: Rob
Date: Mon, 15 Aug 2016 13:34:06 -0700
Subject: [PATCH 6/7] Add analytics even for LevelLoad errors.
---
app/lib/LevelLoader.coffee | 5 +++++
1 file changed, 5 insertions(+)
diff --git a/app/lib/LevelLoader.coffee b/app/lib/LevelLoader.coffee
index e35af2996..2b3dc9249 100644
--- a/app/lib/LevelLoader.coffee
+++ b/app/lib/LevelLoader.coffee
@@ -79,6 +79,11 @@ module.exports = class LevelLoader extends CocoClass
@listenToOnce @level, 'sync', @onLevelLoaded
reportLoadError: ->
+ window.tracker?.trackEvent 'LevelLoadError',
+ category: 'Error',
+ levelSlug: @work?.level?.slug,
+ unloaded: JSON.stringify(@supermodel.report().map (m) -> _.result(m.model, 'url'))
+
return if me.isAdmin() or /dev=true/.test(window.location?.href ? '') or reportedLoadErrorAlready
reportedLoadErrorAlready = true
context = email: me.get('email')
From e62dbed362b7add7a949f7955da598f7f732f249 Mon Sep 17 00:00:00 2001
From: Matt Lott
Date: Sun, 14 Aug 2016 00:47:59 -0700
Subject: [PATCH 7/7] Shift user code yellow arrow tips inward based on
indentation
Closes #3850
---
app/styles/play/level/tome/spell.sass | 18 ++++++++++++++----
app/views/play/level/tome/SpellView.coffee | 8 ++++++++
2 files changed, 22 insertions(+), 4 deletions(-)
diff --git a/app/styles/play/level/tome/spell.sass b/app/styles/play/level/tome/spell.sass
index cc49b2e07..b3928a4b3 100644
--- a/app/styles/play/level/tome/spell.sass
+++ b/app/styles/play/level/tome/spell.sass
@@ -164,7 +164,6 @@
content: " "
display: inline-block
position: relative
- left: -49px
width: 49px
top: -30px
height: 38px
@@ -172,6 +171,20 @@
background-image: url()
+ .ace_gutter-cell.entry-point:not(.next-entry-point):after
+ opacity: 0.5
+
+ .ace_gutter-cell.entry-point.entry-point-indent-0:after
+ left: -25px
+ .ace_gutter-cell.entry-point.entry-point-indent-4:after
+ left: 5px
+ .ace_gutter-cell.entry-point.entry-point-indent-8:after
+ left: 33px
+ .ace_gutter-cell.entry-point.entry-point-indent-12:after
+ left: 61px
+ .ace_gutter-cell.entry-point.entry-point-indent-16:after
+ left: 89px
+
.ace_gutter-cell.ace_error
background-image: url()
@@ -179,9 +192,6 @@
.ace_gutter-cell.ace_info
background-image: none
- .ace_gutter-cell.entry-point:not(.next-entry-point):after
- opacity: 0.5
-
.ace_marker-layer
.ace_bracket
// Override faint gray
diff --git a/app/views/play/level/tome/SpellView.coffee b/app/views/play/level/tome/SpellView.coffee
index b4cc1d44b..f3a759558 100644
--- a/app/views/play/level/tome/SpellView.coffee
+++ b/app/views/play/level/tome/SpellView.coffee
@@ -1113,6 +1113,7 @@ module.exports = class SpellView extends CocoView
for line, index in lines
session.removeGutterDecoration index, 'entry-point'
session.removeGutterDecoration index, 'next-entry-point'
+ session.removeGutterDecoration index, "entry-point-indent-#{i}" for i in [0, 4, 8, 12, 16]
lineHasComment = @singleLineCommentRegex().test line
lineHasCode = line.trim()[0] and not @singleLineCommentOnlyRegex().test line
@@ -1146,6 +1147,13 @@ module.exports = class SpellView extends CocoView
session.addGutterDecoration index, 'next-entry-point'
seenAnEntryPoint = true
+ # Shift pointer right based on current indentation
+ # TODO: tabs probably need different horizontal offsets than spaces
+ indent = 0
+ indent++ while /\s/.test(line[indent])
+ indent = Math.min(16, Math.floor(indent / 4) * 4)
+ session.addGutterDecoration index, "entry-point-indent-#{indent}"
+
previousLine = line
previousLineHadComment = lineHasComment
previousLineHadCode = lineHasCode