From 5b5af5a55ac8486707a8d640e5d78637d92e63d5 Mon Sep 17 00:00:00 2001
From: Finn <finnfantparadis@gmail.com>
Date: Fri, 28 Feb 2014 12:06:26 +0100
Subject: [PATCH 001/178] Update names.coffee

Added the name "Cid" under "Soldier M".
---
 app/lib/world/names.coffee | 1 +
 1 file changed, 1 insertion(+)

diff --git a/app/lib/world/names.coffee b/app/lib/world/names.coffee
index d47f92425..7ba332f60 100644
--- a/app/lib/world/names.coffee
+++ b/app/lib/world/names.coffee
@@ -44,6 +44,7 @@ module.exports.thangNames = thangNames =
     "Huburt"
     "Sterling"
     "Alistair"
+    "Cid"
     "Remy"
     "Stormy"
     "Halle"

From 9c107f24f868b5b2ce210b60a54f91a203ebf426 Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Sat, 1 Mar 2014 03:31:36 +0400
Subject: [PATCH 002/178] Finishing ru.coffee

---
 app/locale/ru.coffee | 54 ++++++++++++++++++++++----------------------
 1 file changed, 27 insertions(+), 27 deletions(-)

diff --git a/app/locale/ru.coffee b/app/locale/ru.coffee
index b1102bc8e..985282a6c 100644
--- a/app/locale/ru.coffee
+++ b/app/locale/ru.coffee
@@ -201,15 +201,15 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     hud_continue: "Продолжить (нажмите Shift+Пробел)"
     spell_saved: "Заклинание сохранено"
 
-#  admin:
-#    av_title: "Admin Views"
-#    av_entities_sub_title: "Entities"
-#    av_entities_users_url: "Users"
-#    av_entities_active_instances_url: "Active Instances"
-#    av_other_sub_title: "Other"
-#    av_other_debug_base_url: "Base (for debugging base.jade)"
-#    u_title: "User List"
-#    lg_title: "Latest Games"
+  admin:
+    av_title: "Админ панель"
+    av_entities_sub_title: "Сущности"
+    av_entities_users_url: "Пользователи"
+    av_entities_active_instances_url: "Активные экземпляры"
+    av_other_sub_title: "Другое"
+    av_other_debug_base_url: "База (для отладки base.jade)"
+    u_title: "Список пользователей"
+    lg_title: "Последние игры"
 
   editor:
     main_title: "Редакторы CodeCombat"
@@ -224,7 +224,7 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     contact_us: "свяжитесь с нами!"
     hipchat_prefix: "Также вы можете найти нас в нашей"
     hipchat_url: "комнате HipChat."
-#    level_some_options: "Some Options?"
+    level_some_options: "Ещё опции"
     level_tab_thangs: "Объекты"
     level_tab_scripts: "Скрипты"
     level_tab_settings: "Настройки"
@@ -254,7 +254,7 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
   general:
     and: "и"
     name: "Имя"
-#    body: "Body"
+    body: "Содержание"
     version: "Версия"
     commit_msg: "Сопроводительное сообщение"
     version_history_for: "История версий для: "
@@ -418,22 +418,22 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     diplomat_join_suf_github: ", отредактируйте его онлайн и отправьте запрос на включение изменений. Кроме того, установите флажок ниже, чтобы быть в курсе новых разработок интернационализации!"
     more_about_diplomat: "Узнать больше о том, как стать Дипломатом"
     diplomat_subscribe_desc: "Получать email-ы о i18n разработках и уровнях для перевода."
-#    ambassador_summary: "Мы пытаемся создать сообщество, и каждое сообщество нуждается в службе поддержки, когда есть проблемы. У нас есть чаты, электронная почта и социальные сети, чтобы наши пользователи могли познакомиться с игрой. Если вы хотите помочь людям втянуться, получать удовольствие и учиться программированию, этот класс для вас."  # Not done yet
-#    ambassador_introduction: "This is a community we're building, and you are the connections. We've got Olark chats, emails, and social networks with lots of people to talk with and help get acquainted with the game and learn from. If you want to help people get involved and have fun, and get a good feel of the pulse of CodeCombat and where we're going, then this class might be for you."
-#    ambassador_attribute_1: "Communication skills. Be able to identify the problems players are having and help them solve them. Also, keep the rest of us informed about what players are saying, what they like and don't like and want more of!"
-#    ambassador_join_desc: "tell us a little about yourself, what you've done and what you'd be interested in doing. We'll go from there!"
-#    ambassador_join_note_strong: "Note"
-#    ambassador_join_note_desc: "One of our top priorities is to build multiplayer where players having difficulty solving levels can summon higher level wizards to help them. This will be a great way for ambassadors to do their thing. We'll keep you posted!"
-#    more_about_ambassador: "Узнать больше о том, как стать Ambassador"  # Not done yet
-#    ambassador_subscribe_desc: "Get emails on support updates and multiplayer developments."
-#    counselor_summary: "None of the above roles fit what you are interested in? Do not worry, we are on the lookout for anybody who wants a hand in the development of CodeCombat! If you are interested in teaching, game development, open source management, or anything else that you think will be relevant to us, then this class is for you."
-#    counselor_introduction_1: "Do you have life experience? A different perspective on things that can help us decide how to shape CodeCombat? Of all these roles, this will probably take the least time, but individually you may make the most difference. We're on the lookout for wisened sages, particularly in areas like: teaching, game development, open source project management, technical recruiting, entrepreneurship, or design."
-#    counselor_introduction_2: "Or really anything that is relevant to the development of CodeCombat. If you have knowledge and want to share it to help grow this project, then this class might be for you."
-#    counselor_attribute_1: "Experience, in any of the areas above or something you think might be helpful."
-#    counselor_attribute_2: "A little bit of free time!"
-#    counselor_join_desc: "tell us a little about yourself, what you've done and what you'd be interested in doing. We'll put you in our contact list and be in touch when we could use advice (not too often)."
-#    more_about_counselor: "Узнать больше о том, как стать Counselor"  # Not done yet
-#    changes_auto_save: "Changes are saved automatically when you toggle checkboxes."
+    ambassador_summary: "Мы пытаемся создать сообщество, и каждое сообщество нуждается в службе поддержки, когда есть проблемы. У нас есть чаты, электронная почта и социальные сети, чтобы наши пользователи могли познакомиться с игрой. Если вы хотите помочь людям втянуться, получать удовольствие и учиться программированию, этот класс для вас."
+    ambassador_introduction: "Это сообщество, которое мы создаём, и вы соединяете. У нас есть Olark чаты, электронная почта и социальные сети с уймой людей, с которыми нужно поговорить, помочь в ознакомлении с игрой и обучении из неё. Если вы хотите помочь людям втянуться, получать удовольствие, наслаждаться и и куда мы идём, этот класс для вас."
+    ambassador_attribute_1: "Навыки общения. Уметь определять проблемы игроков и помогать решить их. Кроме того, держите всех нас в курсе о том, что игроки говорят, что им нравится, не нравится и чего хотят больше!"
+    ambassador_join_desc: "расскажите нам немного о себе, чем вы занимались и чем хотели бы заниматься. Отсюда и начнём!"
+    ambassador_join_note_strong: "Примечание"
+    ambassador_join_note_desc: "Одним из наших главных приоритетов является создание мультиплеера, где игроки столкнутся с труднорешаемыми уровнями и могут призвать более высокоуровневых волшебников для помощи. Это будет отличным способом для послов делать свое дело. Мы будем держать вас в курсе!"
+    more_about_ambassador: "Узнать больше о том, как стать Послом"
+    ambassador_subscribe_desc: "Получать email-ы о разработке мультиплеера и обновлениях в системе поддержки."
+    counselor_summary: "Ни одна из вышеупомянутых ролей не соответствует тому, в чём вы заинтересованы? Не волнуйтесь, мы в поисках тех, кто хочет приложить руку к разработке CodeCombat! Если вы заинтересованы в обучении, разработке игр, управлением проектами с открытым исходным кодом, или в чём-нибудь ещё, что, как вы думаете, будет актуально для нас, то этот класс для вас."
+    counselor_introduction_1: "У вас есть жизненный опыт? Другая точка зрения на вещи, которые могут помочь нам решить, как формировать CodeCombat? Из всех этих ролей, эта, возможно, займёт меньше всего времени, но по отдельности, вы можете сделать наибольшие изменения. Мы в поисках морщинистых мудрецов, особенно в таких областях, как: обучение, разработка игр, управление проектами с открытым исходным кодом, технической рекрутинг, предпринимательство или дизайн."
+    counselor_introduction_2: "Или действительно всё, что имеет отношение к развитию CodeCombat. Если у вас есть знания и вы хотите поделиться ими, чтобы помочь вырастить этот проект, то этот класс для вас."
+    counselor_attribute_1: "Опыт, в любой из областей выше, или в том, что, как вы думаете, может быть полезным."
+    counselor_attribute_2: "Немного свободного времени!"
+    counselor_join_desc: "расскажите нам немного о себе, чем вы занимались и чем хотели бы заниматься. Мы поместим вас в наш список контактов и выйдем на связь, когда нам понадобится совет(не слишком часто)."
+    more_about_counselor: "Узнать больше о том, как стать Советником"
+    changes_auto_save: "Изменения сохраняются автоматически при переключении флажков."
     diligent_scribes: "Наши старательные Писари:"
     powerful_archmages: "Наши могущественные Архимаги:"
     creative_artisans: "Наши творческие Ремесленники:"

From 789b49e15f5349cba6f8060290feee0e1e698f36 Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Sat, 1 Mar 2014 03:37:11 +0400
Subject: [PATCH 003/178] Update diplomat.jade

---
 app/templates/contribute/diplomat.jade | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/templates/contribute/diplomat.jade b/app/templates/contribute/diplomat.jade
index 1c7863680..6b02823a6 100644
--- a/app/templates/contribute/diplomat.jade
+++ b/app/templates/contribute/diplomat.jade
@@ -86,7 +86,7 @@ block content
           li Slovak - Anon
           li Persian - Reza Habibi (Rehb)
           li Czech - vanous
-          li Russian - fess89, ser-storchak
+          li Russian - fess89, ser-storchak, Mr.A
           li Ukrainian - fess89
           li Italian - flauta
           li Norwegian - bardeh

From d68ff5a8997f82b433c030c65088479ad1fad633 Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Sat, 1 Mar 2014 04:54:54 +0400
Subject: [PATCH 004/178] Fix diplomat.jade building

---
 app/templates/contribute/diplomat.jade | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/templates/contribute/diplomat.jade b/app/templates/contribute/diplomat.jade
index 6b02823a6..06743761c 100644
--- a/app/templates/contribute/diplomat.jade
+++ b/app/templates/contribute/diplomat.jade
@@ -86,7 +86,7 @@ block content
           li Slovak - Anon
           li Persian - Reza Habibi (Rehb)
           li Czech - vanous
-          li Russian - fess89, ser-storchak, Mr.A
+          li Russian - fess89, ser-storchak, Mr A
           li Ukrainian - fess89
           li Italian - flauta
           li Norwegian - bardeh

From 8cc8103288d9a280e1bac2f9f440cd8e548d3fab Mon Sep 17 00:00:00 2001
From: Muhammed Thanish <mnmtanish@gmail.com>
Date: Sat, 1 Mar 2014 19:58:15 +0530
Subject: [PATCH 005/178] Add keyboard shortcuts to move the wizard

---
 app/lib/LevelBus.coffee                   | 10 ++++++++++
 app/lib/surface/Camera.coffee             | 14 +++++++++-----
 app/views/play/level/playback_view.coffee | 16 ++++++++++++++++
 3 files changed, 35 insertions(+), 5 deletions(-)

diff --git a/app/lib/LevelBus.coffee b/app/lib/LevelBus.coffee
index 18985716b..707b0cef8 100644
--- a/app/lib/LevelBus.coffee
+++ b/app/lib/LevelBus.coffee
@@ -22,6 +22,7 @@ module.exports = class LevelBus extends Bus
     'level-show-victory': 'onVictory'
     'tome:spell-changed': 'onSpellChanged'
     'tome:spell-created': 'onSpellCreated'
+    'self-wizard:move': 'moveWizard'
 
   constructor: ->
     super(arguments...)
@@ -240,3 +241,12 @@ module.exports = class LevelBus extends Bus
   destroy: ->
     @session.off 'change:multiplayer', @onMultiplayerChanged, @
     super()
+
+  moveWizard : (x, y) =>
+    wizardSprite = @getSelfWizard()
+    position = wizardSprite.getCurrentPosition()
+    position.x += x
+    position.y += y
+    wizardSprite.setTarget(position,1000)
+    wizardSprite.updatePosition()
+    Backbone.Mediator.publish 'camera-zoom-to', position
diff --git a/app/lib/surface/Camera.coffee b/app/lib/surface/Camera.coffee
index 6c5ec946f..70ee37884 100644
--- a/app/lib/surface/Camera.coffee
+++ b/app/lib/surface/Camera.coffee
@@ -42,6 +42,7 @@ module.exports = class Camera extends CocoClass
     'level:restarted': 'onLevelRestarted'
     'sprite:mouse-down': 'onMouseDown'
     'sprite:dragged': 'onMouseDragged'
+    'camera-zoom-to': 'onZoomTo'
 
   # TODO: Fix tests to not use mainLayer
   constructor: (@canvasWidth, @canvasHeight, angle=Math.asin(0.75), hFOV=d2r(30)) ->
@@ -169,7 +170,7 @@ module.exports = class Camera extends CocoClass
   onMouseDown: (e) ->
     return if @dragDisabled
     @lastPos = {x: e.originalEvent.rawX, y: e.originalEvent.rawY}
-    
+
   onMouseDragged: (e) ->
     return if @dragDisabled
     target = @boundTarget(@target, @zoom)
@@ -180,7 +181,7 @@ module.exports = class Camera extends CocoClass
     @zoomTo newPos, @zoom, 0
     @lastPos = {x: e.originalEvent.rawX, y: e.originalEvent.rawY}
     Backbone.Mediator.publish 'camera:dragged'
-    
+
   onLevelRestarted: ->
     @setBounds(@firstBounds, false)
 
@@ -220,7 +221,7 @@ module.exports = class Camera extends CocoClass
     newTarget ?= {x:0, y:0}
     newTarget = (@newTarget or @target) if @locked
     newZoom = Math.min((Math.max @minZoom, newZoom), MAX_ZOOM)
-    
+
     thangType = @target?.sprite?.thangType
     if thangType
       @offset = _.clone(thangType.get('positions')?.torso or {x: 0, y:0})
@@ -229,7 +230,7 @@ module.exports = class Camera extends CocoClass
       @offset.y *= scale
     else
       @offset = {x: 0, y:0}
-      
+
     return if @zoom is newZoom and newTarget is newTarget.x and newTarget.y is newTarget.y
 
     @finishTween(true)
@@ -247,7 +248,7 @@ module.exports = class Camera extends CocoClass
       @target = newTarget
       @zoom = newZoom
       @updateZoom true
-      
+
   focusedOnSprite: ->
     return @target?.name
 
@@ -308,3 +309,6 @@ module.exports = class Camera extends CocoClass
     createjs.Tween.removeTweens @
     @finishTween = null
     super()
+
+  onZoomTo: (pos) ->
+    @zoomTo(@worldToSurface(pos), @zoom)
diff --git a/app/views/play/level/playback_view.coffee b/app/views/play/level/playback_view.coffee
index f9c13a9e3..404201e9e 100644
--- a/app/views/play/level/playback_view.coffee
+++ b/app/views/play/level/playback_view.coffee
@@ -36,6 +36,10 @@ module.exports = class PlaybackView extends View
     '⌘+p, p, ctrl+p': 'onTogglePlay'
     '⌘+[, ctrl+[': 'onScrubBack'
     '⌘+], ctrl+]': 'onScrubForward'
+    'w, up': 'onMoveUpKey'
+    's, down': 'onMoveDownKey'
+    'a, left': 'onMoveLeftKey'
+    'd, right': 'onMoveRightKey'
 
   constructor: ->
     super(arguments...)
@@ -215,3 +219,15 @@ module.exports = class PlaybackView extends View
     $(window).off('resize', @onWindowResize)
     @onWindowResize = null
     super()
+
+  onMoveUpKey: ->
+    Backbone.Mediator.publish 'self-wizard:move', 0, 10
+
+  onMoveDownKey: ->
+    Backbone.Mediator.publish 'self-wizard:move', 0, -10
+
+  onMoveLeftKey: ->
+    Backbone.Mediator.publish 'self-wizard:move', -10, 0
+
+  onMoveRightKey: ->
+    Backbone.Mediator.publish 'self-wizard:move', 10, 0

From 7ea92603229c362999805e9ff9b9f30681ae5b73 Mon Sep 17 00:00:00 2001
From: Muhammed Thanish <mnmtanish@gmail.com>
Date: Sun, 2 Mar 2014 01:45:49 +0530
Subject: [PATCH 006/178] Use WizardSprite

---
 app/lib/LevelBus.coffee                   | 10 ----------
 app/lib/surface/Camera.coffee             |  4 ----
 app/lib/surface/WizardSprite.coffee       | 18 ++++++++++++++----
 app/views/play/level/playback_view.coffee |  8 ++++----
 4 files changed, 18 insertions(+), 22 deletions(-)

diff --git a/app/lib/LevelBus.coffee b/app/lib/LevelBus.coffee
index 707b0cef8..18985716b 100644
--- a/app/lib/LevelBus.coffee
+++ b/app/lib/LevelBus.coffee
@@ -22,7 +22,6 @@ module.exports = class LevelBus extends Bus
     'level-show-victory': 'onVictory'
     'tome:spell-changed': 'onSpellChanged'
     'tome:spell-created': 'onSpellCreated'
-    'self-wizard:move': 'moveWizard'
 
   constructor: ->
     super(arguments...)
@@ -241,12 +240,3 @@ module.exports = class LevelBus extends Bus
   destroy: ->
     @session.off 'change:multiplayer', @onMultiplayerChanged, @
     super()
-
-  moveWizard : (x, y) =>
-    wizardSprite = @getSelfWizard()
-    position = wizardSprite.getCurrentPosition()
-    position.x += x
-    position.y += y
-    wizardSprite.setTarget(position,1000)
-    wizardSprite.updatePosition()
-    Backbone.Mediator.publish 'camera-zoom-to', position
diff --git a/app/lib/surface/Camera.coffee b/app/lib/surface/Camera.coffee
index 70ee37884..9e4701868 100644
--- a/app/lib/surface/Camera.coffee
+++ b/app/lib/surface/Camera.coffee
@@ -42,7 +42,6 @@ module.exports = class Camera extends CocoClass
     'level:restarted': 'onLevelRestarted'
     'sprite:mouse-down': 'onMouseDown'
     'sprite:dragged': 'onMouseDragged'
-    'camera-zoom-to': 'onZoomTo'
 
   # TODO: Fix tests to not use mainLayer
   constructor: (@canvasWidth, @canvasHeight, angle=Math.asin(0.75), hFOV=d2r(30)) ->
@@ -309,6 +308,3 @@ module.exports = class Camera extends CocoClass
     createjs.Tween.removeTweens @
     @finishTween = null
     super()
-
-  onZoomTo: (pos) ->
-    @zoomTo(@worldToSurface(pos), @zoom)
diff --git a/app/lib/surface/WizardSprite.coffee b/app/lib/surface/WizardSprite.coffee
index 665a93bf4..7b57b7b87 100644
--- a/app/lib/surface/WizardSprite.coffee
+++ b/app/lib/surface/WizardSprite.coffee
@@ -21,6 +21,7 @@ module.exports = class WizardSprite extends IndieSprite
     'surface:sprite-selected': 'onSpriteSelected'
     'echo-self-wizard-sprite': 'onEchoSelfWizardSprite'
     'echo-all-wizard-sprites': 'onEchoAllWizardSprites'
+    'self-wizard:move': 'moveWizard'
 
   constructor: (thangType, options) ->
     if options?.isSelf
@@ -102,7 +103,7 @@ module.exports = class WizardSprite extends IndieSprite
   defaultPos: -> x: 35, y: 24, z: @thang.depth / 2 + @thang.bobHeight
   move: (pos, duration) -> @setTarget(pos, duration)
 
-  setTarget: (newTarget, duration) ->
+  setTarget: (newTarget, duration, isLinear=false) ->
     # ignore targets you're already heading for
     targetPos = @getPosFromTarget(newTarget)
     return if @targetPos and @targetPos.x is targetPos.x and @targetPos.y is targetPos.y
@@ -115,7 +116,7 @@ module.exports = class WizardSprite extends IndieSprite
     @shoveOtherWizards(true) if @targetSprite
     @targetSprite = if isSprite then newTarget else null
     @targetPos = targetPos
-    @beginMoveTween(duration)
+    @beginMoveTween(duration, isLinear)
     @shoveOtherWizards()
     Backbone.Mediator.publish('self-wizard:target-changed', {sender:@}) if @isSelf
 
@@ -127,7 +128,7 @@ module.exports = class WizardSprite extends IndieSprite
     return target if target.x?
     return target.thang.pos
 
-  beginMoveTween: (duration=1000) ->
+  beginMoveTween: (duration=1000, isLinear=false) ->
     # clear the old tween
     createjs.Tween.removeTweens(@)
 
@@ -140,8 +141,11 @@ module.exports = class WizardSprite extends IndieSprite
       @updatePosition()
       @endMoveTween()
       return
+    if isLinear
+      ease = createjs.Ease.linear
+    else
+      ease = createjs.Ease.getPowInOut(3.0)
 
-    ease = createjs.Ease.getPowInOut(3.0)
     createjs.Tween
       .get(@)
       .to({tweenPercentage:0.0}, duration, ease)
@@ -225,3 +229,9 @@ module.exports = class WizardSprite extends IndieSprite
 
   updateMarks: ->
     super() if @displayObject.visible  # not if we hid the wiz
+
+  moveWizard : (x, y) =>
+    console.log x, y
+    position = {x: @targetPos.x+x, y: @targetPos.y+y}
+    @setTarget(position, 500, true)
+    @updatePosition()
diff --git a/app/views/play/level/playback_view.coffee b/app/views/play/level/playback_view.coffee
index 404201e9e..4a3b8cadb 100644
--- a/app/views/play/level/playback_view.coffee
+++ b/app/views/play/level/playback_view.coffee
@@ -221,13 +221,13 @@ module.exports = class PlaybackView extends View
     super()
 
   onMoveUpKey: ->
-    Backbone.Mediator.publish 'self-wizard:move', 0, 10
+    Backbone.Mediator.publish 'self-wizard:move', 0, 1
 
   onMoveDownKey: ->
-    Backbone.Mediator.publish 'self-wizard:move', 0, -10
+    Backbone.Mediator.publish 'self-wizard:move', 0, -1
 
   onMoveLeftKey: ->
-    Backbone.Mediator.publish 'self-wizard:move', -10, 0
+    Backbone.Mediator.publish 'self-wizard:move', -1, 0
 
   onMoveRightKey: ->
-    Backbone.Mediator.publish 'self-wizard:move', 10, 0
+    Backbone.Mediator.publish 'self-wizard:move', 1, 0

From a5f633961ff293cba0fc6679c482e9744e7ed650 Mon Sep 17 00:00:00 2001
From: Aditya Raisinghani <aditya.ajeet@gmail.com>
Date: Sun, 2 Mar 2014 01:48:21 +0530
Subject: [PATCH 007/178] Added log setting for production.

---
 server_setup.coffee | 10 ++++++++--
 1 file changed, 8 insertions(+), 2 deletions(-)

diff --git a/server_setup.coffee b/server_setup.coffee
index 9f94da179..3332e0428 100644
--- a/server_setup.coffee
+++ b/server_setup.coffee
@@ -8,7 +8,6 @@ database = require './server/commons/database'
 baseRoute = require './server/routes/base'
 user = require './server/users/user_handler'
 logging = require './server/commons/logging'
-
 config = require './server_config'
 
 ###Middleware setup functions implementation###
@@ -20,9 +19,16 @@ setupRequestTimeoutMiddleware = (app) ->
       self.emit('pass',message)
     next()
 
+productionLogging = (tokens, req, res)->
+  status = res.statusCode
+  color = 31
+  if(status != 200 && status != 304)
+    return '\x1b[90m' + req.method+ ' ' + req.originalUrl + ' '+ '\x1b[' + color + 'm' + res.statusCode+ ' \x1b[90m'+ (new Date - req._startTime)+ 'ms' + '\x1b[0m';
+
 setupExpressMiddleware = (app) ->
   setupRequestTimeoutMiddleware app
-  app.use(express.logger('dev'))
+  express.logger.format('prod', productionLogging)
+  app.use(express.logger('prod'))
   app.use(express.static(path.join(__dirname, 'public')))
   app.use(useragent.express())
 

From a092557973e8edca74f5bd3d58160ba971893d47 Mon Sep 17 00:00:00 2001
From: Muhammed Thanish <mnmtanish@gmail.com>
Date: Sun, 2 Mar 2014 02:13:37 +0530
Subject: [PATCH 008/178] Add camera zoomTo(pos)

---
 app/lib/surface/Camera.coffee       | 4 ++++
 app/lib/surface/WizardSprite.coffee | 2 +-
 2 files changed, 5 insertions(+), 1 deletion(-)

diff --git a/app/lib/surface/Camera.coffee b/app/lib/surface/Camera.coffee
index 9e4701868..76323e5e8 100644
--- a/app/lib/surface/Camera.coffee
+++ b/app/lib/surface/Camera.coffee
@@ -42,6 +42,7 @@ module.exports = class Camera extends CocoClass
     'level:restarted': 'onLevelRestarted'
     'sprite:mouse-down': 'onMouseDown'
     'sprite:dragged': 'onMouseDragged'
+    'camera-zoom-to': 'onZoomTo'
 
   # TODO: Fix tests to not use mainLayer
   constructor: (@canvasWidth, @canvasHeight, angle=Math.asin(0.75), hFOV=d2r(30)) ->
@@ -308,3 +309,6 @@ module.exports = class Camera extends CocoClass
     createjs.Tween.removeTweens @
     @finishTween = null
     super()
+
+  onZoomTo: (pos, time) ->
+    @zoomTo(@worldToSurface(pos), @zoom, time)
diff --git a/app/lib/surface/WizardSprite.coffee b/app/lib/surface/WizardSprite.coffee
index 7b57b7b87..1ec5ad812 100644
--- a/app/lib/surface/WizardSprite.coffee
+++ b/app/lib/surface/WizardSprite.coffee
@@ -231,7 +231,7 @@ module.exports = class WizardSprite extends IndieSprite
     super() if @displayObject.visible  # not if we hid the wiz
 
   moveWizard : (x, y) =>
-    console.log x, y
     position = {x: @targetPos.x+x, y: @targetPos.y+y}
     @setTarget(position, 500, true)
     @updatePosition()
+    Backbone.Mediator.publish 'camera-zoom-to', position, 500
\ No newline at end of file

From d7e031fd00ea3e8425a67d8243f07adc3eac8c44 Mon Sep 17 00:00:00 2001
From: Muhammed Thanish <mnmtanish@gmail.com>
Date: Sun, 2 Mar 2014 03:27:06 +0530
Subject: [PATCH 009/178] Use key.isPressed() to get movement direction

---
 app/lib/surface/WizardSprite.coffee       |  5 ++--
 app/views/play/level/playback_view.coffee | 29 +++++++++++------------
 2 files changed, 17 insertions(+), 17 deletions(-)

diff --git a/app/lib/surface/WizardSprite.coffee b/app/lib/surface/WizardSprite.coffee
index 1ec5ad812..236d2d77d 100644
--- a/app/lib/surface/WizardSprite.coffee
+++ b/app/lib/surface/WizardSprite.coffee
@@ -231,7 +231,8 @@ module.exports = class WizardSprite extends IndieSprite
     super() if @displayObject.visible  # not if we hid the wiz
 
   moveWizard : (x, y) =>
+    interval = 250
     position = {x: @targetPos.x+x, y: @targetPos.y+y}
-    @setTarget(position, 500, true)
+    @setTarget(position, interval, true)
     @updatePosition()
-    Backbone.Mediator.publish 'camera-zoom-to', position, 500
\ No newline at end of file
+    Backbone.Mediator.publish 'camera-zoom-to', position, interval
\ No newline at end of file
diff --git a/app/views/play/level/playback_view.coffee b/app/views/play/level/playback_view.coffee
index 4a3b8cadb..c9d11c9f0 100644
--- a/app/views/play/level/playback_view.coffee
+++ b/app/views/play/level/playback_view.coffee
@@ -36,10 +36,10 @@ module.exports = class PlaybackView extends View
     '⌘+p, p, ctrl+p': 'onTogglePlay'
     '⌘+[, ctrl+[': 'onScrubBack'
     '⌘+], ctrl+]': 'onScrubForward'
-    'w, up': 'onMoveUpKey'
-    's, down': 'onMoveDownKey'
-    'a, left': 'onMoveLeftKey'
-    'd, right': 'onMoveRightKey'
+    'up': 'onMoveKey'
+    'down': 'onMoveKey'
+    'left': 'onMoveKey'
+    'right': 'onMoveKey'
 
   constructor: ->
     super(arguments...)
@@ -220,14 +220,13 @@ module.exports = class PlaybackView extends View
     @onWindowResize = null
     super()
 
-  onMoveUpKey: ->
-    Backbone.Mediator.publish 'self-wizard:move', 0, 1
-
-  onMoveDownKey: ->
-    Backbone.Mediator.publish 'self-wizard:move', 0, -1
-
-  onMoveLeftKey: ->
-    Backbone.Mediator.publish 'self-wizard:move', -1, 0
-
-  onMoveRightKey: ->
-    Backbone.Mediator.publish 'self-wizard:move', 1, 0
+  onMoveKey: (e) ->
+    e?.preventDefault()
+    x = 0
+    y = 0
+    y = 1 if key.isPressed('up')
+    y = -1 if key.isPressed('down')
+    x = 1 if key.isPressed('right')
+    x = -1 if key.isPressed('left')
+    console.log 'onMoveKey', x, y
+    Backbone.Mediator.publish 'self-wizard:move', x, y

From 2db9fa708b1db809307fbc64c4020883844b89e1 Mon Sep 17 00:00:00 2001
From: YF <yfdyh000@gmail.com>
Date: Sun, 2 Mar 2014 07:27:15 +0800
Subject: [PATCH 010/178] Update zh-HANS.coffee

---
 app/locale/zh-HANS.coffee | 256 +++++++++++++++++++-------------------
 1 file changed, 128 insertions(+), 128 deletions(-)

diff --git a/app/locale/zh-HANS.coffee b/app/locale/zh-HANS.coffee
index 2be4960c8..13c3d3161 100644
--- a/app/locale/zh-HANS.coffee
+++ b/app/locale/zh-HANS.coffee
@@ -3,7 +3,7 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
     loading: "读取中……"
     saving: "保存中……"
     sending: "发送中……"
-    cancel: "退出"
+    cancel: "取消"
     save: "保存"
     delay_1_sec: "1 秒"
     delay_3_sec: "3 秒"
@@ -14,31 +14,31 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
 
   modal:
     close: "关闭"
-    okay: "好"
+    okay: "好的"
 
   not_found:
     page_not_found: "找不到网页"
 
   nav:
     play: "玩"
-    editor: "编辑"
+    editor: "编辑器"
     blog: "博客"
     forum: "论坛"
-    admin: "超级管理员"
+    admin: "管理"
     home: "首页"
     contribute: "贡献"
     legal: "法律"
     about: "关于"
-    contact: "联系我们"
+    contact: "联系"
     twitter_follow: "关注"
     employers: "雇佣我们"
 
   versions:
     save_version_title: "保存新版本"
-#    new_major_version: "New Major Version"
-#    cla_prefix: "To save changes, first you must agree to our"
-#    cla_url: "CLA"
-#    cla_suffix: "."
+    new_major_version: "新的重要版本"
+    cla_prefix: "要想保存更改,您必须先同意我们的"
+    cla_url: "贡献者许可协议"
+    cla_suffix: "。"
     cla_agree: "我同意"
 
   login:
@@ -63,40 +63,40 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
 
   home:
     slogan: "通过游戏学习 Javascript"
-    no_ie: "抱歉!Internet Explorer 9等更旧的预览器打不开此网站。"
-    no_mobile: "CodeCombat 不是针对手机设备设计的,所以可能无法达到做好的体验!"
+    no_ie: "抱歉!Internet Explorer 9 等旧式预览器无法使用本网站。"
+    no_mobile: "CodeCombat 不是针对手机设备设计的,所以可能无法达到最好的体验!"
     play: "开始游戏"
 
   play:
     choose_your_level: "选取难度"
-    adventurer_prefix: "你可以选择以下任意关卡,或者讨论以上的关卡。"
+    adventurer_prefix: "你可以选择以下任意关卡,或者讨论以上的关卡。到"
     adventurer_forum: "冒险者论坛"
-    adventurer_suffix: "."
+    adventurer_suffix: "。"
     campaign_beginner: "新手作战"
-    campaign_beginner_description: "……在这里可以学到编程技巧。"
+    campaign_beginner_description: "……在这里你可以学习到编程技巧。"
     campaign_dev: "随机困难关卡"
     campaign_dev_description: "……在这里你可以学到做一些复杂功能的接口。"
     campaign_multiplayer: "多人竞技场"
-    campaign_multiplayer_description: "……在这里你可以和其他玩家们进行代码肉搏战。"
-    campaign_player_created: "已创建的玩家"
+    campaign_multiplayer_description: "……在这里你可以与其他玩家进行代码肉搏战。"
+    campaign_player_created: "创建玩家"
     campaign_player_created_description: "……在这里你可以与你的小伙伴的创造力战斗 <a href=\"/contribute#artisan\">技术指导</a>."
-    level_difficulty: "难度"
+    level_difficulty: "难度:"
 
   contact:
     contact_us: "联系我们"
-    welcome: "很高兴收到你的信!用这个表格给我们发邮件。 "
-    contribute_prefix: "如果你想贡献代码,请看我们的 "
-    contribute_page: "贡献代码页面"
+    welcome: "我们很乐意收到你的信!用这个表单给我们发邮件。 "
+    contribute_prefix: "如果你想贡献什么,请看我们的 "
+    contribute_page: "贡献页面"
     contribute_suffix: "!"
-    forum_prefix: "对任何公共部分,放手去干吧 "
+    forum_prefix: "对于任何公开部分,请尝试用"
     forum_page: "我们的论坛"
-    forum_suffix: "代替 "
+    forum_suffix: "代替"
     send: "意见反馈"
 
   diplomat_suggestion:
     title: "帮我们翻译 CodeCombat"
     sub_heading: "我们需要您的语言技能"
-    pitch_body: "我们开发了 CodeCombat 的英文版,但是现在我们的玩家遍布全球。很多人想玩中文(简体)版的,却不会说英语,所以如果你中英文都会,请考虑一下参加我们的翻译工作,帮忙把 CodeCombat 网站还有所有的关卡翻译成中文(简体)。"
+    pitch_body: "我们开发了 CodeCombat 的英文版,但是现在我们的玩家遍布全球。很多人英语不熟练,但很想玩简体中文版的游戏,所以如果你中英文都很熟练,请考虑参加我们的翻译工作,帮忙把 CodeCombat 网站还有所有的关卡翻译成简体中文。"
     missing_translations: "未翻译的文本将显示为英文。"
     learn_more: "了解更多有关成为翻译人员的说明"
     subscribe_as_diplomat: "提交翻译人员申请"
@@ -130,28 +130,28 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
     new_password_verify: "核实"
     email_subscriptions: "邮箱验证"
     email_announcements: "通知"
-    email_notifications_description: "定期接受来自你的账户的通知。"
+    email_notifications_description: "接收来自你的账户的定期通知。"
     email_announcements_description: "接收关于 CodeCombat 最近的新闻和发展的邮件。"
     contributor_emails: "贡献者通知"
-    contribute_prefix: "我们在寻找志同道合的人!请到 "
+    contribute_prefix: "我们在寻找志同道合的人!请到"
     contribute_page: "贡献页面"
-    contribute_suffix: " 查看更多信息。"
+    contribute_suffix: "查看更多信息。"
     email_toggle: "切换所有"
     error_saving: "保存时出错"
-    saved: "保存修改"
+    saved: "更改已保存"
     password_mismatch: "密码不匹配。"
 
   account_profile:
     edit_settings: "编辑设置"
-    profile_for_prefix: "关于TA的基本资料:"
+    profile_for_prefix: "关于他的基本资料:"
     profile_for_suffix: ""
     profile: "基本资料"
-    user_not_found: "没有找到用户。检查URL?"
-    gravatar_not_found_mine: "我们找不到TA的基本资料:"
-    gravatar_not_found_email_suffix: "."
+    user_not_found: "没有找到用户。网址有错?"
+    gravatar_not_found_mine: "我们找不到他的基本资料:"
+    gravatar_not_found_email_suffix: "。"
     gravatar_signup_prefix: "去注册 "
-    gravatar_signup_suffix: " 去设置!"
-    gravatar_not_found_other: "哎呀,没有与这个人的邮箱相关的资料。"
+    gravatar_signup_suffix: " 来设置!"
+    gravatar_not_found_other: "哎呀,没有与这个邮箱相关的资料。"
     gravatar_contact: "联系"
     gravatar_websites: "网站"
     gravatar_accounts: "显示为"
@@ -198,33 +198,33 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
     tome_select_spell: "选择一个法术"
     tome_select_a_thang: "选择人物来 "
     tome_available_spells: "可用的法术"
-    hud_continue: "继续(按 shift-空格)"
-    spell_saved: "输入已保存"
+    hud_continue: "继续(按 Shift-空格)"
+    spell_saved: "咒语已保存"
 
-#  admin:
-#    av_title: "Admin Views"
-#    av_entities_sub_title: "Entities"
-#    av_entities_users_url: "Users"
-#    av_entities_active_instances_url: "Active Instances"
-#    av_other_sub_title: "Other"
-#    av_other_debug_base_url: "Base (for debugging base.jade)"
-#    u_title: "User List"
-#    lg_title: "Latest Games"
+  admin:
+    av_title: "管理员视图"
+    av_entities_sub_title: "实体"
+    av_entities_users_url: "用户"
+    av_entities_active_instances_url: "活动实例"
+    av_other_sub_title: "其他"
+    av_other_debug_base_url: "Base(用于调试 base.jade)"
+    u_title: "用户列表"
+    lg_title: "最新的游戏"
 
   editor:
     main_title: "CodeCombat 编辑器"
-    main_description: "建立自己的关卡, 战役, 单位 和教育内容。我们提供所有你需要的工具!"
+    main_description: "建立你自己的关卡、 战役、单元和教育内容。我们会提供所有你需要的工具!"
     article_title: "提示编辑器"
-    article_description: "编写提示,让玩家可以使用编程概念,来通过各种的关卡和战役。"
+    article_description: "编写提示,让玩家可以使用编程概念来通过各种关卡和战役。"
     thang_title: "物体编辑器"
-    thang_description: "创建单位,并定义单位的逻辑、图形和音频。目前只支持导入Flash导出的矢量图形。"
+    thang_description: "创建单元,并定义单元的逻辑、图形和音频。目前只支持导入 Flash 导出的矢量图形。"
     level_title: "关卡编辑器"
     level_description: "所有用来创造所有难度的关卡的工具,包括脚本、上传音频和构建自定义逻辑。"
-    security_notice: "编辑器的许多主要功能并不是目前默认启动的。当我们改善完这些系统的安全性的时候,他们将会变成常用的。如果你想要马上使用这些功能,"
+    security_notice: "编辑器的许多主要功能并不是目前默认启动的。当我们改善完这些系统的安全性的时候,它们就会成为通常可用的。如果你想要马上使用这些功能,"
     contact_us: "联系我们!"
     hipchat_prefix: "你也可以在这里找到我们"
-    hipchat_url: "HipChat room."
-#    level_some_options: "Some Options?"
+    hipchat_url: "HipChat 房间。"
+    level_some_options: "有些选项?"
     level_tab_thangs: "物体"
     level_tab_scripts: "脚本"
     level_tab_settings: "设定"
@@ -239,7 +239,7 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
     level_systems_tab_title: "目前所有系统"
     level_systems_btn_new: "创建新的系统"
     level_systems_btn_add: "增加系统"
-    level_components_title: "返回到所以物体主页"
+    level_components_title: "返回到所有物体主页"
     level_components_type: "类型"
     level_component_edit_title: "编辑组件"
     level_system_edit_title: "编辑系统"
@@ -249,12 +249,12 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
 
   article:
     edit_btn_preview: "预览"
-    edit_article_title: "编辑文章"
+    edit_article_title: "编辑提示"
 
   general:
     and: "和"
     name: "姓名"
-#    body: "Body"
+    body: "正文"
     version: "版本"
     commit_msg: "提交信息"
     version_history_for: "版本历史: "
@@ -265,11 +265,11 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
     message: "消息"
 
   about:
-    who_is_codecombat: "什么是CodeCombat?"
-    why_codecombat: "为什么选择CodeCombat?"
-    who_description_prefix: "一起在2013开始编写CodeCombat。在2008年时,我们也创造了"
-    who_description_suffix: "并且发展它成为第一的学习如何写中文字和日文字的Web和IOS应用"
-    who_description_ending: "现在是时候教人们如何写代码。"
+    who_is_codecombat: "什么是 CodeCombat?"
+    why_codecombat: "为什么选择 CodeCombat?"
+    who_description_prefix: "在2013年开始一起编写 CodeCombat。在2008年时,我们还创造"
+    who_description_suffix: "并且发展出了首选的学习如何写中文和日文的Web和IOS应用"
+    who_description_ending: "现在是时候教人们如何写代码了。"
 #    why_paragraph_1: "When making Skritter, George didn't know how to program and was constantly frustrated by his inability to implement his ideas. Afterwards, he tried learning, but the lessons were too slow. His housemate, wanting to reskill and stop teaching, tried Codecademy, but \"got bored.\" Each week another friend started Codecademy, then dropped off. We realized it was the same problem we'd solved with Skritter: people learning a skill via slow, intensive lessons when what they need is fast, extensive practice. We know how to fix that."
 #    why_paragraph_2: "Need to learn to code? You don't need lessons. You need to write a lot of code and have a great time doing it."
 #    why_paragraph_3_prefix: "That's what programming is about. It's gotta be fun. Not fun like"
@@ -286,61 +286,61 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
 #    jeremy_description: "Customer support mage, usability tester, and community organizer; you've probably already spoken with Jeremy."
 #    michael_description: "Programmer, sys-admin, and undergrad technical wunderkind, Michael is the person keeping our servers online."
 
-#  legal:
-#    page_title: "Legal"
-#    opensource_intro: "CodeCombat is free to play and completely open source."
-#    opensource_description_prefix: "Check out "
-#    github_url: "our GitHub"
-#    opensource_description_center: "and help out if you like! CodeCombat is built on dozens of open source projects, and we love them. See "
-#    archmage_wiki_url: "our Archmage wiki"
-#    opensource_description_suffix: "for a list of the software that makes this game possible."
-#    practices_title: "Respectful Best Practices"
-#    practices_description: "These are our promises to you, the player, in slightly less legalese."
-#    privacy_title: "Privacy"
-#    privacy_description: "We will not sell any of your personal information. We intend to make money through recruitment eventually, but rest assured we will not distribute your personal information to interested companies without your explicit consent."
-#    security_title: "Security"
-#    security_description: "We strive to keep your personal information safe. As an open source project, our site is freely open to anyone to review and improve our security systems."
-#    email_title: "Email"
-#    email_description_prefix: "We will not inundate you with spam. Through"
-#    email_settings_url: "your email settings"
-#    email_description_suffix: "or through links in the emails we send, you can change your preferences and easily unsubscribe at any time."
-#    cost_title: "Cost"
-#    cost_description: "Currently, CodeCombat is 100% free! One of our main goals is to keep it that way, so that as many people can play as possible, regardless of place in life. If the sky darkens, we might have to charge subscriptions or for some content, but we'd rather not. With any luck, we'll be able to sustain the company with:"
-#    recruitment_title: "Recruitment"
-#    recruitment_description_prefix: "Here on CodeCombat, you're going to become a powerful wizard–not just in the game, but also in real life."
-#    url_hire_programmers: "No one can hire programmers fast enough"
-#    recruitment_description_suffix: "so once you've sharpened your skills and if you agree, we will demo your best coding accomplishments to the thousands of employers who are drooling for the chance to hire you. They pay us a little, they pay you"
-#    recruitment_description_italic: "a lot"
-#    recruitment_description_ending: "the site remains free and everybody's happy. That's the plan."
-#    copyrights_title: "Copyrights and Licenses"
-#    contributor_title: "Contributor License Agreement"
-#    contributor_description_prefix: "All contributions, both on the site and on our GitHub repository, are subject to our"
-#    cla_url: "CLA"
-#    contributor_description_suffix: "to which you should agree before contributing."
-#    code_title: "Code - MIT"
-#    code_description_prefix: "All code owned by CodeCombat or hosted on codecombat.com, both in the GitHub repository or in the codecombat.com database, is licensed under the"
-#    mit_license_url: "MIT license"
+  legal:
+    page_title: "法律"
+    opensource_intro: "CodeCombat 是一个自由发挥,完全开源的项目。"
+    opensource_description_prefix: "查看"
+    github_url: "我们的 GitHub"
+    opensource_description_center: "并做你想做的修改吧!CodeCombat 是构筑在几十个开源项目的基础之上,我们爱它们。见"
+    archmage_wiki_url: "我们的 Archmage wiki"
+    opensource_description_suffix: "了解让这个游戏成为可能的名单。"
+    practices_title: "尊重最佳实践"
+    practices_description: "这是我们对您的承诺,即玩家,尽管这在法律用语中略显不足。"
+    privacy_title: "隐私"
+    privacy_description: "我们不会出售您的任何个人信息。我们计划最终通过招聘来盈利,但请您放心,未经您的明确同意,我们不会将您的个人信息出售有兴趣的公司。"
+    security_title: "安全"
+    security_description: "我们竭力保证您的个人信息安全性。作为一个开源项目,任何人都可以检讨并改善我们自由开放的网站的安全性。"
+    email_title: "电子邮件"
+    email_description_prefix: "我们不会发给您垃圾邮件。通过"
+    email_settings_url: "您的电子邮件设置"
+    email_description_suffix: "或者我们发送的邮件中的链接,您可以随时更改您的偏好设置或者随时取消订阅。"
+    cost_title: "花费"
+    cost_description: "目前来说,CodeCombat 是完全免费的!我们的主要目标之一也是保持目前这种方式,让尽可能多的人玩得更好,不论是否是生活中。如果天空变暗,我们可能会对某些内容采取订阅收费,但我们宁愿不那么做。运气好的话,我们可以维持公司,通过:"
+    recruitment_title: "招募"
+    recruitment_description_prefix: "在 CodeCombat 这里,你将得以成为一名法力强大的“巫师”,不只是在游戏中,更在生活中。"
+    url_hire_programmers: "没有人能以足够快速度招聘程序员,"
+    recruitment_description_suffix: "所以一旦你的技能成熟并且得到你的同意,我们将战士你的最佳编码成就给上万名雇主,希望他们垂涎欲滴。而他们支付给我们一点点报酬,并且付给你工资,"
+    recruitment_description_italic: "“一大笔”"
+    recruitment_description_ending: "。而这网站也就能保持免费,皆大欢喜。计划就是这样。"
+    copyrights_title: "版权与许可"
+    contributor_title: "贡献者许可协议"
+    contributor_description_prefix: "在本网站或者我们的 GitHub 版本库的所有贡献都依照我们的"
+    cla_url: "贡献者许可协议(CLA)"
+    contributor_description_suffix: ",而这在您贡献之前就应该已经同意。"
+    code_title: "代码 - MIT"
+    code_description_prefix: "所有由 CodeCombat 拥有或者托管在 codecombat.com 的代码,在 GitHub 版本库或者 codecombat.com 数据库,以上许可协议都依照"
+    mit_license_url: "MIT 许可证"
 #    code_description_suffix: "This includes all code in Systems and Components that are made available by CodeCombat for the purpose of creating levels."
 #    art_title: "Art/Music - Creative Commons "
 #    art_description_prefix: "All common content is available under the"
 #    cc_license_url: "Creative Commons Attribution 4.0 International License"
 #    art_description_suffix: "Common content is anything made generally available by CodeCombat for the purpose of creating Levels. This includes:"
-#    art_music: "Music"
-#    art_sound: "Sound"
-#    art_artwork: "Artwork"
-#    art_sprites: "Sprites"
+    art_music: "音乐"
+    art_sound: "声效"
+    art_artwork: "艺术品"
+    art_sprites: "小妖精"
 #    art_other: "Any and all other non-code creative works that are made available when creating Levels."
 #    art_access: "Currently there is no universal, easy system for fetching these assets. In general, fetch them from the URLs as used by the site, contact us for assistance, or help us in extending the site to make these assets more easily accessible."
 #    art_paragraph_1: "For attribution, please name and link to codecombat.com near where the source is used or where appropriate for the medium. For example:"
 #    use_list_1: "If used in a movie or another game, include codecombat.com in the credits."
 #    use_list_2: "If used on a website, include a link near the usage, for example underneath an image, or in a general attributions page where you might also mention other Creative Commons works and open source software being used on the site. Something that's already clearly referencing CodeCombat, such as a blog post mentioning CodeCombat, does not need some separate attribution."
 #    art_paragraph_2: "If the content being used is created not by CodeCombat but instead by a user of codecombat.com, attribute them instead, and follow attribution directions provided in that resource's description if there are any."
-#    rights_title: "Rights Reserved"
-#    rights_desc: "All rights are reserved for Levels themselves. This includes"
-#    rights_scripts: "Scripts"
-#    rights_unit: "Unit configuration"
-#    rights_description: "Description"
-#    rights_writings: "Writings"
+    rights_title: "版权所有"
+    rights_desc: "所有关卡由他们自己版权所有。这包括"
+    rights_scripts: "脚本"
+    rights_unit: "单元配置"
+    rights_description: "描述"
+    rights_writings: "作品"
 #    rights_media: "Media (sounds, music) and any other creative content made specifically for that Level and not made generally available when creating Levels."
 #    rights_clarification: "To clarify, anything that is made available in the Level Editor for the purpose of making levels is under CC, whereas the content created with the Level Editor or uploaded in the course of creation of Levels is not."
 #    nutshell_title: "In a Nutshell"
@@ -408,16 +408,16 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
 #    scribe_join_description: "tell us a little about yourself, your experience with programming and what sort of things you'd like to write about. We'll go from there!"
 #    more_about_scribe: "Learn More About Becoming a Scribe"
 #    scribe_subscribe_desc: "Get emails about article writing announcements."
-    diplomat_summary: "在其他国家不讲英语,很多人对于CodeCombat有很大的兴趣。我们正在寻找译者,愿意花时间翻译网站的语料库的词语,这样CodeCombat就能尽快访问到世界各地。如果你想帮助CodeCombat国际化,那么这个类就是给你的。"
+    diplomat_summary: "在其他国家不讲英语,很多人对于CodeCombat有很大的兴趣。我们正在寻找愿意花时间翻译网站语料库的词语的译者,这样 CodeCombat 就能尽快地遍及世界各地。如果你想帮助 CodeCombat 的国际化,那么这个类就是给你的。"
     diplomat_introduction_pref: "如果有一件事情是从 "
     diplomat_launch_url: "launch in October"
     diplomat_introduction_suf: "学来的,Combat有相当大的兴趣在其他国家发展。我们正在构建一个译者兵团把单词一个一个的翻译,让CodeCombat尽可能地访问到世界各地。如果你喜欢偷偷地瞄一眼即将到来的内容,并让你的国民尽快学习到CodeCombat,那么这个类可能适合你。"
     diplomat_attribute_1: "会流利的英语和能翻译的语言。当传递复杂思想,难得的是能很好的同时掌握这两个。"
     diplomat_join_pref_github: "找到你自己的语言文件 "
     diplomat_github_url: "在GitHub网站"
-    diplomat_join_suf_github: "在线编辑它,然后提交一个合并请求。同时,检查下面这个盒子来关注最新的国际化开发!"
-    more_about_diplomat: "学习更多关于“成为一名外交官(翻译者)”"
-    diplomat_subscribe_desc: "接受邮件,关于国际化开发和翻译情况"
+    diplomat_join_suf_github: "在线编辑它,然后提交一个合并请求。同时,选中下面这个复选框来关注最新的国际化开发!"
+    more_about_diplomat: "了解更多“如何成为一名外交官(翻译者)”"
+    diplomat_subscribe_desc: "接受有关国际化开发和翻译情况的邮件"
 #    ambassador_summary: "We are trying to build a community, and every community needs a support team when there are troubles. We have got chats, emails, and social networks so that our users can get acquainted with the game. If you want to help people get involved, have fun, and learn some programming, then this class is for you."
 #    ambassador_introduction: "This is a community we're building, and you are the connections. We've got Olark chats, emails, and social networks with lots of people to talk with and help get acquainted with the game and learn from. If you want to help people get involved and have fun, and get a good feel of the pulse of CodeCombat and where we're going, then this class might be for you."
 #    ambassador_attribute_1: "Communication skills. Be able to identify the problems players are having and help them solve them. Also, keep the rest of us informed about what players are saying, what they like and don't like and want more of!"
@@ -433,26 +433,26 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
 #    counselor_attribute_2: "A little bit of free time!"
 #    counselor_join_desc: "tell us a little about yourself, what you've done and what you'd be interested in doing. We'll put you in our contact list and be in touch when we could use advice (not too often)."
 #    more_about_counselor: "Learn More About Becoming a Counselor"
-#    changes_auto_save: "Changes are saved automatically when you toggle checkboxes."
-#    diligent_scribes: "Our Diligent Scribes:"
-#    powerful_archmages: "Our Powerful Archmages:"
-#    creative_artisans: "Our Creative Artisans:"
-#    brave_adventurers: "Our Brave Adventurers:"
-#    translating_diplomats: "Our Translating Diplomats:"
-#    helpful_ambassadors: "Our Helpful Ambassadors:"
+    changes_auto_save: "在您切换复选框时,更改将自动保存。"
+    diligent_scribes: "我们勤奋的文书:"
+    powerful_archmages: "我们强力的大法师:"
+    creative_artisans: "我们极具创意的工匠:"
+    brave_adventurers: "我们勇敢的冒险家:"
+    translating_diplomats: "我们遍及世界的外交官:"
+    helpful_ambassadors: "我们亲切的使节:"
 
-#  classes:
-#    archmage_title: "Archmage"
-#    archmage_title_description: "(Coder)"
-#    artisan_title: "Artisan"
-#    artisan_title_description: "(Level Builder)"
-#    adventurer_title: "Adventurer"
-#    adventurer_title_description: "(Level Playtester)"
-#    scribe_title: "Scribe"
-#    scribe_title_description: "(Article Editor)"
-#    diplomat_title: "Diplomat"
-#    diplomat_title_description: "(Translator)"
-#    ambassador_title: "Ambassador"
-#    ambassador_title_description: "(Support)"
-#    counselor_title: "Counselor"
-#    counselor_title_description: "(Expert/Teacher)"
+  classes:
+    archmage_title: "大法师"
+    archmage_title_description: "(代码编写人员)"
+    artisan_title: "工匠"
+    artisan_title_description: "(关卡建立人员)"
+    adventurer_title: "冒险家"
+    adventurer_title_description: "(关卡测试人员)"
+    scribe_title: "文书"
+    scribe_title_description: "(提示编辑人员)"
+    diplomat_title: "外交官"
+    diplomat_title_description: "(翻译人员)"
+    ambassador_title: "使节"
+    ambassador_title_description: "(用户支持人员)"
+    counselor_title: "顾问"
+    counselor_title_description: "(专家/导师)"

From e69e23115fe5c611ffa7fd3bf06dc1d1e3f874b3 Mon Sep 17 00:00:00 2001
From: Michael Schmatz <schmatz@umich.edu>
Date: Sat, 1 Mar 2014 16:52:17 -0800
Subject: [PATCH 011/178] Improvements to game scheduling, removed cap

---
 server/queues/scoring.coffee | 79 ++++++++++++++++++++----------------
 1 file changed, 45 insertions(+), 34 deletions(-)

diff --git a/server/queues/scoring.coffee b/server/queues/scoring.coffee
index 08f0794d0..692a08b75 100644
--- a/server/queues/scoring.coffee
+++ b/server/queues/scoring.coffee
@@ -130,7 +130,7 @@ module.exports.processTaskResult = (req, res) ->
                 opponentID = _.pull(_.keys(newScoresObject), originalSessionID)
                 sessionNewScore = newScoresObject[originalSessionID].totalScore
                 opponentNewScore = newScoresObject[opponentID].totalScore
-                findNearestBetterSessionID sessionNewScore, opponentNewScore, opponentID ,opposingTeam, (err, opponentSessionID) ->
+                findNearestBetterSessionID originalSessionID, sessionNewScore, opponentNewScore, opponentID ,opposingTeam, (err, opponentSessionID) ->
                   if err? then return errors.serverError res, "There was an error finding the nearest sessionID!"
                   unless opponentSessionID then return sendResponseObject req, res, {"message":"There were no more games to rank(game is at top!"}
                     
@@ -162,9 +162,6 @@ determineIfSessionShouldContinueAndUpdateLog = (sessionID, sessionRank, cb) ->
     if totalNumberOfGamesPlayed < 5
       console.log "Number of games played is less than 5, continuing..."
       cb null, true
-    else if totalNumberOfGamesPlayed > 15
-      console.log "Too many games played, ending..."
-      cb null, false
     else
       ratio = (updatedSession.numberOfLosses - 5) / (totalNumberOfGamesPlayed)
       if ratio > 0.66
@@ -175,39 +172,53 @@ determineIfSessionShouldContinueAndUpdateLog = (sessionID, sessionRank, cb) ->
         cb null, true
       
     
-findNearestBetterSessionID = (sessionTotalScore, opponentSessionTotalScore, opponentSessionID, opposingTeam, cb) ->
-  queryParameters =
-    totalScore: 
-      $gt:opponentSessionTotalScore + 0.5
-    _id: 
-      $ne: opponentSessionID
-    "level.original": "52d97ecd32362bc86e004e87"
-    "level.majorVersion": 0
-    submitted: true
-    submittedCode:
-      $exists: true
-    team: opposingTeam
+findNearestBetterSessionID = (sessionID, sessionTotalScore, opponentSessionTotalScore, opponentSessionID, opposingTeam, cb) ->
+  retrieveAllOpponentSessionIDs sessionID, (err, opponentSessionIDs) ->
+    if err? then return cb err, null
     
-  limitNumber = 1
-
-  sortParameters =
-    totalScore: 1
-    
-  selectString = '_id totalScore'
-    
-  query = LevelSession.findOne(queryParameters)
-    .sort(sortParameters)
-    .limit(limitNumber)
-    .select(selectString)
-    .lean()
+    queryParameters =
+      totalScore: 
+        $gt:opponentSessionTotalScore
+      _id: 
+        $nin: opponentSessionIDs
+      "level.original": "52d97ecd32362bc86e004e87"
+      "level.majorVersion": 0
+      submitted: true
+      submittedCode:
+        $exists: true
+      team: opposingTeam
+      
+    limitNumber = 1
   
-  console.log "Finding session with score near #{opponentSessionTotalScore}"
-  query.exec (err, session) ->
-    if err? then return cb err, session
-    unless session then return cb err, null
-    console.log "Found session with score #{session.totalScore}"
-    cb err, session._id
+    sortParameters =
+      totalScore: 1
+      
+    selectString = '_id totalScore'
+      
+    query = LevelSession.findOne(queryParameters)
+      .sort(sortParameters)
+      .limit(limitNumber)
+      .select(selectString)
+      .lean()
+    
+    console.log "Finding session with score near #{opponentSessionTotalScore}"
+    query.exec (err, session) ->
+      if err? then return cb err, session
+      unless session then return cb err, null
+      console.log "Found session with score #{session.totalScore}"
+      cb err, session._id
 
+    
+retrieveAllOpponentSessionIDs = (sessionID, cb) ->
+  query = LevelSession.findOne({"_id":sessionID})
+    .select('matches.opponents.sessionID')
+    .lean()
+  query.exec (err, session) ->
+    if err? then return cb err, null
+    opponentSessionIDs = (match.opponents[0].sessionID for match in session.matches)
+    cb err, opponentSessionIDs
+    
+      
 calculateOpposingTeam = (sessionTeam) ->
   teams = ['ogres','humans']
   opposingTeams = _.pull teams, sessionTeam

From 488be26fd3de738dcdd8bb613acb0d7e3a1fc98e Mon Sep 17 00:00:00 2001
From: Michael Schmatz <schmatz@umich.edu>
Date: Sat, 1 Mar 2014 16:56:04 -0800
Subject: [PATCH 012/178] Tweaked ratio calculation

---
 server/queues/scoring.coffee | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/server/queues/scoring.coffee b/server/queues/scoring.coffee
index 692a08b75..0079d34a8 100644
--- a/server/queues/scoring.coffee
+++ b/server/queues/scoring.coffee
@@ -163,7 +163,7 @@ determineIfSessionShouldContinueAndUpdateLog = (sessionID, sessionRank, cb) ->
       console.log "Number of games played is less than 5, continuing..."
       cb null, true
     else
-      ratio = (updatedSession.numberOfLosses - 5) / (totalNumberOfGamesPlayed)
+      ratio = (updatedSession.numberOfLosses) / (totalNumberOfGamesPlayed)
       if ratio > 0.66
         cb null, false
         console.log "Ratio(#{ratio}) is bad, ending simulation"

From 6c48660922ecd8c9e3f852e047d05473f4db0935 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sat, 1 Mar 2014 16:57:29 -0800
Subject: [PATCH 013/178] No save button when there's nothing to save. Messing
 with AudioPlayer playSound API a bit.

---
 app/lib/AudioPlayer.coffee              | 5 +++--
 app/lib/surface/CocoSprite.coffee       | 2 +-
 app/templates/editor/level/save.jade    | 4 ++++
 app/views/editor/level/save_view.coffee | 1 +
 4 files changed, 9 insertions(+), 3 deletions(-)

diff --git a/app/lib/AudioPlayer.coffee b/app/lib/AudioPlayer.coffee
index 122509dcc..b5eb4f70d 100644
--- a/app/lib/AudioPlayer.coffee
+++ b/app/lib/AudioPlayer.coffee
@@ -78,8 +78,9 @@ class AudioPlayer extends CocoClass
       @preloadInterfaceSounds [name] unless filename of cache
       @soundsToPlayWhenLoaded[name] = volume
 
-  playSound: (name, volume=1) ->
-    createjs.Sound.play name, {volume: (me.get('volume') ? 1) * volume}
+  playSound: (name, volume=1, delay=0) ->
+    instance = createjs.Sound.play name, {volume: (me.get('volume') ? 1) * volume, delay: delay}
+    instance
 
 #  # TODO: load Interface sounds somehow, somewhere, somewhen
 
diff --git a/app/lib/surface/CocoSprite.coffee b/app/lib/surface/CocoSprite.coffee
index 29ee32371..008519599 100644
--- a/app/lib/surface/CocoSprite.coffee
+++ b/app/lib/surface/CocoSprite.coffee
@@ -475,6 +475,6 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
     return null unless sound
     delay = if withDelay and sound.delay then 1000 * sound.delay / createjs.Ticker.getFPS() else 0
     name = AudioPlayer.nameForSoundReference sound
-    instance = createjs.Sound.play name, "none", delay, 0, 0, volume
+    instance = AudioPlayer.playSound name, volume, delay
 #    console.log @thang?.id, "played sound", name, "with delay", delay, "volume", volume, "and got sound instance", instance
     instance
diff --git a/app/templates/editor/level/save.jade b/app/templates/editor/level/save.jade
index 8ada52b23..00e9e43f2 100644
--- a/app/templates/editor/level/save.jade
+++ b/app/templates/editor/level/save.jade
@@ -52,3 +52,7 @@ block modal-body-content
           label.control-label(for=id + "-version-is-major") Major Changes?
           input(id=id + "-version-is-major", name="version-is-major", type="checkbox")
           span.help-block (Could this update break anything depending on this System?)
+
+if noSaveButton
+  block modal-footer-content
+    button.btn(data-dismiss="modal", data-i18n="common.cancel") Cancel
diff --git a/app/views/editor/level/save_view.coffee b/app/views/editor/level/save_view.coffee
index a1f0ee01b..4edf03ca0 100644
--- a/app/views/editor/level/save_view.coffee
+++ b/app/views/editor/level/save_view.coffee
@@ -23,6 +23,7 @@ module.exports = class LevelSaveView extends SaveVersionModal
     context.levelNeedsSave = @level.hasLocalChanges()
     context.modifiedComponents = _.filter @supermodel.getModels(LevelComponent), @shouldSaveEntity
     context.modifiedSystems = _.filter @supermodel.getModels(LevelSystem), @shouldSaveEntity
+    context.noSaveButton = context.levelNeedsSave or context.modifiedComponents.length or context.modifiedSystems.length
     context
 
   shouldSaveEntity: (m) ->

From da2ebf857a6d2d86468c911ee6705dc0404a08e3 Mon Sep 17 00:00:00 2001
From: Muhammed Thanish <mnmtanish@gmail.com>
Date: Sun, 2 Mar 2014 14:02:30 +0530
Subject: [PATCH 014/178] Increase delay

---
 app/lib/surface/WizardSprite.coffee       |  2 +-
 app/views/play/level/playback_view.coffee | 16 ++++++++--------
 2 files changed, 9 insertions(+), 9 deletions(-)

diff --git a/app/lib/surface/WizardSprite.coffee b/app/lib/surface/WizardSprite.coffee
index 236d2d77d..b98ae44b5 100644
--- a/app/lib/surface/WizardSprite.coffee
+++ b/app/lib/surface/WizardSprite.coffee
@@ -231,7 +231,7 @@ module.exports = class WizardSprite extends IndieSprite
     super() if @displayObject.visible  # not if we hid the wiz
 
   moveWizard : (x, y) =>
-    interval = 250
+    interval = 500
     position = {x: @targetPos.x+x, y: @targetPos.y+y}
     @setTarget(position, interval, true)
     @updatePosition()
diff --git a/app/views/play/level/playback_view.coffee b/app/views/play/level/playback_view.coffee
index c9d11c9f0..20ec9295c 100644
--- a/app/views/play/level/playback_view.coffee
+++ b/app/views/play/level/playback_view.coffee
@@ -222,11 +222,11 @@ module.exports = class PlaybackView extends View
 
   onMoveKey: (e) ->
     e?.preventDefault()
-    x = 0
-    y = 0
-    y = 1 if key.isPressed('up')
-    y = -1 if key.isPressed('down')
-    x = 1 if key.isPressed('right')
-    x = -1 if key.isPressed('left')
-    console.log 'onMoveKey', x, y
-    Backbone.Mediator.publish 'self-wizard:move', x, y
+    yMovement = 0
+    xMovement = 0
+    yMovement += 2 if key.isPressed('up')
+    yMovement -= 2 if key.isPressed('down')
+    xMovement += 2 if key.isPressed('right')
+    xMovement -= 2 if key.isPressed('left')
+    console.log 'onMoveKey', xMovement, yMovement
+    Backbone.Mediator.publish 'self-wizard:move', xMovement, yMovement

From fb57e634a5469a88c60530c23eaa6cb8cc8c2d64 Mon Sep 17 00:00:00 2001
From: Ruben Vereecken <rubenvereecken@gmail.com>
Date: Sun, 2 Mar 2014 18:35:37 +0100
Subject: [PATCH 015/178] Patched up some grammar mistakes (Dutch-Nederlands).

---
 app/locale/nl.coffee | 6 +++---
 1 file changed, 3 insertions(+), 3 deletions(-)

diff --git a/app/locale/nl.coffee b/app/locale/nl.coffee
index dd5904abe..c644b5d99 100644
--- a/app/locale/nl.coffee
+++ b/app/locale/nl.coffee
@@ -213,13 +213,13 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
 
   editor:
     main_title: "CodeCombat Editors"
-    main_description: "Maak je eigen levels, campagnes, eenheden en leermateriaal. Wij bieden alle programma's die u nodig heeft!"
+    main_description: "Maak je eigen levels, campagnes, eenheden en leermateriaal. Wij bieden alle programma's aan die u nodig heeft!"
     article_title: "Artikel Editor"
     article_description: "Schrijf artikels dat spelers een overzicht geven over programmeer concepten die kunnen gebruikt worden over een variëteit van levels en campagnes."
     thang_title: "Thang Editor"
-    thang_description: "Maak eenheden, beschrijf hun default logica, graphics en audio. Momenteel is enkel het importeren van vector graphics geëxporteerd in Flash ondersteunt."
+    thang_description: "Maak eenheden, beschrijf hun default logica, graphics en audio. Momenteel is enkel het importeren van vector graphics geëxporteerd in Flash ondersteund."
     level_title: "Level Editor"
-    level_description: "Bevat programmeurs om te programmeren, audio te uploaden, en om aangepaste logica om alle soorten levels te maken. Het is alles wat wijzelf ook gebruiken!"
+    level_description: "Bevat het gereedschap om te programmeren, audio te uploaden en aangepaste logica om alle soorten levels te maken. Het is alles wat wijzelf ook gebruiken!"
     security_notice: "Veel belangrijke elementen in deze editors zijn momenteel niet actief. Met dat wij de veiligheid van deze systemen verbeteren, zullen ook deze elementen beschikbaar worden. Indien u deze elementen al eerder wil gebruiken, "
     contact_us: "contacteer ons!"
     hipchat_prefix: "Je kan ons ook vinden in ons"

From 60f79e261d86f61ea3d6a94c64af7a85112b736e Mon Sep 17 00:00:00 2001
From: Ruben Vereecken <rubenvereecken@gmail.com>
Date: Sun, 2 Mar 2014 19:42:56 +0100
Subject: [PATCH 016/178] Patched up tonnes of minor language mistakes
 (Dutch-Nederlands)

---
 app/locale/nl.coffee | 22 +++++++++++-----------
 1 file changed, 11 insertions(+), 11 deletions(-)

diff --git a/app/locale/nl.coffee b/app/locale/nl.coffee
index c644b5d99..0942c88a1 100644
--- a/app/locale/nl.coffee
+++ b/app/locale/nl.coffee
@@ -219,7 +219,7 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     thang_title: "Thang Editor"
     thang_description: "Maak eenheden, beschrijf hun default logica, graphics en audio. Momenteel is enkel het importeren van vector graphics geëxporteerd in Flash ondersteund."
     level_title: "Level Editor"
-    level_description: "Bevat het gereedschap om te programmeren, audio te uploaden en aangepaste logica om alle soorten levels te maken. Het is alles wat wijzelf ook gebruiken!"
+    level_description: "Bevat het programma om te programmeren, audio te uploaden en aangepaste logica om alle soorten levels te maken. Het is alles wat wijzelf ook gebruiken!"
     security_notice: "Veel belangrijke elementen in deze editors zijn momenteel niet actief. Met dat wij de veiligheid van deze systemen verbeteren, zullen ook deze elementen beschikbaar worden. Indien u deze elementen al eerder wil gebruiken, "
     contact_us: "contacteer ons!"
     hipchat_prefix: "Je kan ons ook vinden in ons"
@@ -267,16 +267,16 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
   about:
     who_is_codecombat: "Wie is CodeCombat?"
     why_codecombat: "Waarom CodeCombat?"
-    who_description_prefix: "hebben samen CodeCombat opgericht in 2013. We creerden ook "
+    who_description_prefix: "hebben samen CodeCombat opgericht in 2013. We creëerden ook "
     who_description_suffix: "en in 2008, groeide het uit tot de #1 web en iOS applicatie om Chinese en Japanse karakters te leren schrijven."
     who_description_ending: "Nu is het tijd om mensen te leren programmeren."
-    why_paragraph_1: "Tijdens het maken van Skritter wist George niet hoe hij moest programmeren. Hij constant gefrustreerd doordat hij zijn ideeën niet kon verwezelijken. Nadien probeerde hij te studeren, maar te lessen gingen te traag. Ook zijn huisgenoot wou opnieuw studeren en stopte met lesgeven. Hij probeerde Codecademy maar was al snel \"verveeld\". Iedere week starte een andere vriend met Codecademy, met telkens als resultaat dat hij/zij vrij snel met de lessen stopte. We realiseerde ons dat het hetzelfde probleem was zoals we al eerder hadden opgelost met Skritter: mensen leren iets via langzame en intensieve lessen, terwijl ze het eigenlijk zo snel mogelijk nodig hebben via uitgebreide oefeningen. Wij weten hoe dat op te lossen."
+    why_paragraph_1: "Tijdens het maken van Skritter wist George niet hoe hij moest programmeren en was hij constant gefrustreerd doordat hij zijn ideeën niet kon verwezelijken. Nadien probeerde hij te studeren maar de lessen gingen te traag. Ook zijn huisgenoot wou opnieuw studeren en stopte met lesgeven. Hij probeerde Codecademy maar was al snel \"verveeld\". Iedere week startte een andere vriend met Codecademy, met telkens als resultaat dat hij/zij vrij snel met de lessen stopte. We realiseerden ons dat het hetzelfde probleem was zoals we al eerder hadden opgelost met Skritter: mensen leren iets via langzame en intensieve lessen, terwijl ze het eigenlijk zo snel mogelijk nodig hebben via uitgebreide oefeningen. Wij weten hoe dat op te lossen."
     why_paragraph_2: "Wil je leren programmeren? Je hebt geen lessen nodig. Je moet vooral veel code schrijven en je amuseren terwijl je dit doet."
     why_paragraph_3_prefix: "Dat is waar programmeren om draait. Het moet tof zijn. Niet tof zoals"
     why_paragraph_3_italic: "joepie een medaille"
     why_paragraph_3_center: "maar tof zoals"
     why_paragraph_3_italic_caps: "NEE MAMA IK MOET DIT LEVEL AF MAKEN!"
-    why_paragraph_3_suffix: "Dat is waarom CodeCombat een multiplayergame is, en niet zomaar lessen gegoten in spelformaat. We zullen niet stoppen totdat jij niet meer kan stoppen--maar deze keer, is dat een goed."
+    why_paragraph_3_suffix: "Dat is waarom CodeCombat een multiplayergame is, en niet zomaar lessen gegoten in spelformaat. We zullen niet stoppen totdat jij niet meer kan stoppen--maar deze keer, is dat iets goeds."
     why_paragraph_4: "Als je verslaafd gaat zijn aan een spel, dan is het beter om hieraan verslaafd te raken en een tovenaar van het technisch tijdperk te worden."
     why_ending: "En hallo, het is gratis."
     why_ending_url: "Start nu met toveren!"
@@ -297,15 +297,15 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     practices_title: "Goede Respectvolle gewoonten"
     practices_description: "Dit zijn onze beloften aan u, de speler, en iets minder juridische jargon."
     privacy_title: "Privacy"
-    privacy_description: "We zullen nooit jouw persoonlijke informatie verkopen. We willen geld verdienen dankzij aanwerving in verloop van tijd, maar je mag op je twee oren slapen dat wij nooit jouw persoonlijke informatie zullen verspreiden aan geïnteresseerde bedrijven zonder dat jij daar explicit met akkoord gaat."
+    privacy_description: "We zullen nooit jouw persoonlijke informatie verkopen. We willen geld verdienen dankzij aanwerving in verloop van tijd, maar je mag op je twee oren slapen dat wij nooit jouw persoonlijke informatie zullen verspreiden aan geïnteresseerde bedrijven zonder dat jij daar expliciet mee akkoord gaat."
     security_title: "Beveiliging"
     security_description: "We streven ernaar om jouw persoonlijke informatie veilig te bewaren. Onze website is open en beschikbaar voor iedereen, opdat ons beveiliging systeem kan worden nagekeken en geoptimaliseerd door iedereen die dat wil. Dit alles is mogelijk doordat we volledig open source en transparant zijn."
     email_title: "E-mail"
     email_description_prefix: "We zullen je niet overspoelen met spam. Door"
     email_settings_url: "jouw e-mail instellingen"
-    email_description_suffix: "of via urls in de emails die wij verzanden, kan je jouw instellingen wijzigen en te alle tijden uitschrijven."
+    email_description_suffix: "of via urls in de emails die wij verzenden, kan je jouw instellingen wijzigen en ten allen tijden uitschrijven."
     cost_title: "Kosten"
-    cost_description: "Momenteel, CodeCombat is 100% gratis! Één van onze doestellingen is om dit zo te houden, opdat zoveel mogelijk mensen kunnen spelen, onafhankelijk van waar je leeft of wie je bent. Als het financieel moeilijker wordt, kan het mogelijk zijn dat we gaan beginnen met abonnementen of een prijs zetten op bepaalde zaken, maar we streven ernaar om dit te verkomen. Met een beetje geluk zullen we dit voor altijd kunnen garanderen met:"
+    cost_description: "Momenteel is CodeCombat 100% gratis! Één van onze doestellingen is om dit zo te houden, opdat zoveel mogelijk mensen kunnen spelen, onafhankelijk van waar je leeft of wie je bent. Als het financieel moeilijker wordt, kan het mogelijk zijn dat we gaan beginnen met abonnementen of een prijs zetten op bepaalde zaken, maar we streven ernaar om dit te voorkomen. Met een beetje geluk zullen we dit voor altijd kunnen garanderen met:"
     recruitment_title: "Aanwervingen"
     recruitment_description_prefix: "Hier bij CodeCombat, ga je ontplooien tot een krachtige tovenoor-niet enkel virtueel, maar ook in het echt."
     url_hire_programmers: "Niemand kan snel genoeg programmeurs aanwerven"
@@ -320,20 +320,20 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     code_title: "Code - MIT"
     code_description_prefix: "Alle code in het bezit van CodeCombat of aanwezig op codecombat.com, zowel in de GitHub respository of in de codecombat.com database, is erkend onder de"
     mit_license_url: "MIT licentie"
-    code_description_suffix: "Dit geld zowel voor code in Systemen en Componenten dat publiekelijk is gemaakt met als doelstellingen het maken van levels."
+    code_description_suffix: "Dit geldt ook voor code in Systemen en Componenten dat publiekelijk is gemaakt met als doelstellingen het maken van levels."
     art_title: "Art/Music - Creative Commons "
     art_description_prefix: "Alle gemeenschappelijke inhoud valt onder de"
     cc_license_url: "Creative Commons Attribution 4.0 Internationale Licentie"
-    art_description_suffix: "Gemeenschappelijke inhoud is alles dat algemeen verkrijgen is bij CodeCombat voor het doel levels te maken. Dit omvat:"
+    art_description_suffix: "Gemeenschappelijke inhoud is alles dat algemeen verkrijgbaar is bij CodeCombat voor het doel levels te maken. Dit omvat:"
     art_music: "Muziek"
     art_sound: "Geluid"
     art_artwork: "Artwork"
     art_sprites: "Sprites"
     art_other: "Eender wat en al het creatief werk dat niet als code aanzien wordt en verkrijgbaar is bij het aanmaken van levels."
-    art_access: "Momenteel is er geen universeel en gebruiksvriendelijk voor het ophalen van deze assets. In het algemeen, worden deze opgehaald via de links zoals gebruikt door de website. Contacteer ons voor assitentie, of help ons met de website uit te breiden en de assets bereikbaarder maken."
+    art_access: "Momenteel is er geen universeel en gebruiksvriendelijk systeem voor het ophalen van deze assets. In het algemeen, worden deze opgehaald via de links zoals gebruikt door de website. Contacteer ons voor assitentie, of help ons met de website uit te breiden en de assets bereikbaarder te maken."
     art_paragraph_1: "Voor toekenning, gelieve de naam en link naar codecombat.com te plaatsen waar dit passend is voor de vorm waarin het voorkomt. Bijvoorbeeld:"
     use_list_1: "Wanneer gebruikt in een film of een ander spel, voeg codecombat.com toe in de credits."
-    use_list_2: "Wanneer toegepast op een website, inclusief een link naar het gebruik, bijvorbeeld onderaan een afbeelding. Of in een algemene webpagina waar je eventueel ook andere Create Commons werken en open source software vernoemd die je gebruikt op de website. Iets dat alreeds duidelijk is gespecificeerd met CodeCombat, zoals een blog artikel, dat CodeCombat vernoemt, heeft geen aparte vermelding nodig."
+    use_list_2: "Wanneer toegepast op een website, inclusief een link naar het gebruik, bijvoorbeeld onderaan een afbeelding. Of in een algemene webpagina waar je eventueel ook andere Create Commons werken en open source software vernoemd die je gebruikt op de website. Iets dat alreeds duidelijk is gespecificeerd met CodeCombat, zoals een blog artikel, dat CodeCombat vernoemt, heeft geen aparte vermelding nodig."
     art_paragraph_2: "Wanneer de gebruikte inhoud is gemaakt door een gebruiker van codecombat.com, vernoem hem/haar in plaats van ons. en volg verspreidingsaanwijzingen van die brons als die er zijn."
     rights_title: "Rechten Voorbehouden"
     rights_desc: "Alle rechten zijn voorbehouden voor de Levels. Dit omvat:"

From e4bf752e2030bd25eba272331c9bc854ecbf1484 Mon Sep 17 00:00:00 2001
From: Ruben Vereecken <rubenvereecken@gmail.com>
Date: Sun, 2 Mar 2014 20:30:18 +0100
Subject: [PATCH 017/178] Changed Dutch (NL) ambassador_summary to match the
 English version

---
 app/locale/nl.coffee | 26 +++++++++++++-------------
 1 file changed, 13 insertions(+), 13 deletions(-)

diff --git a/app/locale/nl.coffee b/app/locale/nl.coffee
index 0942c88a1..57a366206 100644
--- a/app/locale/nl.coffee
+++ b/app/locale/nl.coffee
@@ -334,7 +334,7 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     art_paragraph_1: "Voor toekenning, gelieve de naam en link naar codecombat.com te plaatsen waar dit passend is voor de vorm waarin het voorkomt. Bijvoorbeeld:"
     use_list_1: "Wanneer gebruikt in een film of een ander spel, voeg codecombat.com toe in de credits."
     use_list_2: "Wanneer toegepast op een website, inclusief een link naar het gebruik, bijvoorbeeld onderaan een afbeelding. Of in een algemene webpagina waar je eventueel ook andere Create Commons werken en open source software vernoemd die je gebruikt op de website. Iets dat alreeds duidelijk is gespecificeerd met CodeCombat, zoals een blog artikel, dat CodeCombat vernoemt, heeft geen aparte vermelding nodig."
-    art_paragraph_2: "Wanneer de gebruikte inhoud is gemaakt door een gebruiker van codecombat.com, vernoem hem/haar in plaats van ons. en volg verspreidingsaanwijzingen van die brons als die er zijn."
+    art_paragraph_2: "Wanneer de gebruikte inhoud is gemaakt door een gebruiker van codecombat.com, vernoem hem/haar in plaats van ons en volg verspreidingsaanwijzingen van die brons als die er zijn."
     rights_title: "Rechten Voorbehouden"
     rights_desc: "Alle rechten zijn voorbehouden voor de Levels. Dit omvat:"
     rights_scripts: "Scripts"
@@ -342,15 +342,15 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     rights_description: "Beschrijvingen"
     rights_writings: "Literaire werken"
     rights_media: "Media (geluid, muziek) en eender welke creatieve inhoud, specifiek gemaakt voor dat level en niet verkrijgbaar bij het maken van levels."
-    rights_clarification: "Om het duidelijk te maken, iets dat beschikbaar is in de Level editor voor het maken van level, valt onder de CC licentie. Terwijl de inhoud gemaakt met de Level Editor of geupload in de loop van de creatie van de levels, hier not onder vallen."
+    rights_clarification: "Om het duidelijk te maken, iets dat beschikbaar is in de Level editor voor het maken van levels, valt onder de CC licentie. Terwijl de inhoud gemaakt met de Level Editor of geüpload in de loop van de creatie van de levels, hier niet onder vallen."
     nutshell_title: "In een notendop"
-    nutshell_description: "Alle midellen die wij aanbieden in de Level Editor zijn gratis te gebruiken om levels aan te maken. Maar wij behouden ons het recht om levels die gemaakt zijn op codecombat.com te beperken, en hier in de toekomst geld voor te vragen, moest dat ooit gebeuren."
+    nutshell_description: "Alle middelen die wij aanbieden in de Level Editor zijn gratis te gebruiken om levels aan te maken. Wij behouden ons echter het recht voor om levels die gemaakt zijn op codecombat.com te beperken, en hier in de toekomst geld voor te vragen, moest dat ooit gebeuren."
     canonical: "De Engelse versie van dit document is de definitieve en kanonieke versie. Bij verschillen tussen vertalingen heeft de Engelse versie voorrang."
 
   contribute:
     page_title: "Bijdragen"
     character_classes_title: "Karakter Klassen"
-    introduction_desc_intro: "We hebben hoge verwachten over CodeCombat."
+    introduction_desc_intro: "We hebben hoge verwachtingen over CodeCombat."
     introduction_desc_pref: "We willen zijn waar programmeurs van alle niveaus komen om te leren en samen te spelen, anderen introduceren aan de wondere wereld van code, en de beste delen van de gemeenschap te reflecteren. We kunnen en willen dit niet alleen doen; wat projecten zoals GitHub, Stack Overflow en Linux groots en succesvol maken, zijn de mensen die deze software gebruiken en verbeteren. Daartoe, "
     introduction_desc_github_url: "CodeCombat is volledig open source"
     introduction_desc_suf: ", en we mikken ernaar om zoveel mogelijk manieren mogelijk maken voor u om deel te nemen en dit project van zowel jou als ons te maken."
@@ -360,7 +360,7 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     alert_account_message_pref: "Om je te abonneren voor de klasse e-mails, moet je eerst "
     alert_account_message_suf: "."
     alert_account_message_create_url: "een account aanmaken"
-    archmage_summary: "Geïnteresserd in werken aan game graphics, user interface design, database- en serverorganisatie, multiplayer networking, physics, geluid of game engine  prestaties? Wil jij helpen een game te bouwen wat anderen leert waar jij goed in bent? We moeten nog veel doen en als jij een ervaren programmeur bent en wil ontwikkelen voor CodeCombat, dan is dit de klasse voor jou. We zouden graag je hulp hebben bij het maken van de beste programmeergame ooit."
+    archmage_summary: "Geïnteresserd in werken aan game graphics, user interface design, database- en serverorganisatie, multiplayer networking, physics, geluid of game engine prestaties? Wil jij helpen een game te bouwen wat anderen leert waar jij goed in bent? We moeten nog veel doen en als jij een ervaren programmeur bent en wil ontwikkelen voor CodeCombat, dan is dit de klasse voor jou. We zouden graag je hulp hebben bij het maken van de beste programmeergame ooit."
     archmage_introduction: "Een van de beste aspecten aan het maken van spelletjes is dat zij zoveel verschillende zaken omvatten. Visualisaties, geluid, real-time netwerken, sociale netwerken, en natuurlijk veel van de voorkomende aspecten van programmeren, van low-level database beheer en server administratie tot gebruiksvriendelijke interfaces maken. Er is veel te doen, en als jij een ervaren programmeur bent met de motivatie om je handen veel te maken met CodeCombat, dan ben je de tovenaar die wij zoeken! We zouden graag jouw help hebben met het bouwen aan het allerbeste programmeerspel ooit."
     class_attributes: "Klasse kenmerken"
     archmage_attribute_1_pref: "Ervaring met "
@@ -386,7 +386,7 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     artisan_join_step1: "Lees de documentatie."
     artisan_join_step2: "Maak een nieuw level en bestudeer reeds bestaande levels."
     artisan_join_step3: "Praat met ons in ons publieke (Engelstalige) HipChat kanaal voor hulp. (optioneel)"
-    artisan_join_step4: "Maak een bericht over jou level op ons forum voor feedback."
+    artisan_join_step4: "Maak een bericht over jouw level op ons forum voor feedback."
     more_about_artisan: "Leer meer over hoe je een Creatieve Ambachtsman kan worden."
     artisan_subscribe_desc: "Ontvang e-mails met nieuws over de Level Editor."
     adventurer_sumamry: "Laten we duidelijk zijn over je rol: jij bent de tank. Jij krijgt de zware klappen te verduren. We hebben mensen nodig om spiksplinternieuwe levels te proberen en te kijken hoe deze beter kunnen. De pijn zal groot zijn, het maken van een goede game is een lang proces en niemand doet het de eerste keer goed. Als jij dit kan verduren en een hoge constitution score hebt, dan is dit de klasse voor jou."
@@ -403,7 +403,7 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     scribe_introduction_pref: "CodeCombat is meer dan slechts een aantal levels, het zal ook een bron van kennis kennis zijn en een wiki met programmeerconcepten waar levels op in kunnen gaan. Op die manier zal elk Ambachtslied niet in detail hoeven uit te leggen wat een vergelijkingsoperator is, maar een link kunnen geven naar een artikel wat deze informatie bevat voor de speler. Net zoiets als het  "    
     scribe_introduction_url_mozilla: "Mozilla Developer Network"
     scribe_introduction_suf: " heeft gebouwd. Als jij het leuk vindt om programmeerconcepten uit te leggen in Markdown-vorm, dan is deze klasse wellicht iets voor jou."
-    scribe_attribute_1: "Taal-skills zijn praktisch alles wat je nodig hebt. Niet alleen grammatica of spelling, maar ook moeilijke ideas overbrengen aan anderen."    
+    scribe_attribute_1: "Taal-skills zijn praktisch alles wat je nodig hebt. Niet alleen grammatica of spelling, maar ook moeilijke ideeën overbrengen aan anderen."
     contact_us_url: "Contacteer ons"
     scribe_join_description: "vertel ons wat over jezelf, je ervaring met programmeren en over wat voor soort dingen je graag zou schrijven. Verder zien we wel!"    
     more_about_scribe: "Leer meer over het worden van een ijverige Klerk."    
@@ -412,22 +412,22 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     diplomat_summary: "Er is grote interesse in CodeCombat in landen waar geen Engels wordt gesproken! We zijn op zoek naar vertalers wie tijd willen spenderen aan het vertalen van de site's corpus aan woorden zodat CodeCombat zo snel mogelijk toegankelijk wordt voor heel de wereld. Als jij wilt helpen met CodeCombat internationaal maken, dan is dit de klasse voor jou."
     diplomat_introduction_pref: "Dus, als er iets is wat we geleerd hebben van de "    
     diplomat_launch_url: "release in oktober"
-    diplomat_introduction_suf: "dan is het wel dat er een significante interesse is in CodeCombat in andere landen, vooral Brazilïe! We zijn een corps aan vertalers aan het creëren dat ijverig de ene set woorden in een andere omzet om CodeCombat zo toegankelijk te maken als mogelijk in hel de wereld. Als jij het leuk vindt glimpsen op te vangen van aankomende content en deze levels zo snel mogelijk naar je landgenoten te krijgen, dan is dit de klasse voor jou."
-    diplomat_attribute_1: "Vloeiend Engels en de taal waar naar je wilt vertalen kunnen spreken. Wanneer je moeilijke ideas wilt overbrengen, is het belangrijk beide goed te kunnen!"
+    diplomat_introduction_suf: "dan is het wel dat er een significante interesse is in CodeCombat in andere landen, vooral Brazilië! We zijn een corps aan vertalers aan het creëren dat ijverig de ene set woorden in een andere omzet om CodeCombat zo toegankelijk te maken als mogelijk in heel de wereld. Als jij het leuk vindt glimpsen op te vangen van aankomende content en deze levels zo snel mogelijk naar je landgenoten te krijgen, dan is dit de klasse voor jou."
+    diplomat_attribute_1: "Vloeiend Engels en de taal waar naar je wilt vertalen kunnen spreken. Wanneer je moeilijke ideeën wilt overbrengen, is het belangrijk beide goed te kunnen!"
     diplomat_join_pref_github: "Vind jouw taal haar locale bestand "
     diplomat_github_url: "op GitHub"
-    diplomat_join_suf_github: ", edit het online, en submit een pull request. Daarnaast kun je hieronder aanvinken alsj e up-to-date wilt worden gehouden met nieuwe internationalisatie-ontwikkelingen."
+    diplomat_join_suf_github: ", edit het online, en submit een pull request. Daarnaast kun je hieronder aanvinken als je up-to-date wilt worden gehouden met nieuwe internationalisatie-ontwikkelingen."
     more_about_diplomat: "Leer meer over het worden van een geweldige Diplomaat"
     diplomat_subscribe_desc: "Ontvang e-mails over i18n ontwikkelingen en levels om te vertalen."
-    ambassador_summary: "We proberen een gemeenschap te bouwen en elke gemeenschap heeft een supportteam nodig wanneer er problemen zijn. We hebben chats, e-mails en sociale netwerken met een hoop mensen om mee te praten en wie je kunt helpen bekend te worden met het spel en er van te leren. Als jij mensen wilt helpen betrokken te raken en plezier te hebben, én een goed gevoel van de levenslijn van CodeCombat te krijgen en waar we naar toe gaan, dan is dit wellicht de klasse voor jou."    
+    ambassador_summary: "We proberen een gemeenschap te bouwen en elke gemeenschap heeft een supportteam nodig wanneer er problemen zijn. We hebben chats, e-mails en sociale netwerken zodat onze gebruikers het spel kunnen leren kennen. Als jij mensen wilt helpen betrokken te raken, plezier te hebben en wat te leren programmeren, dan is dit wellicht de klasse voor jou."
     ambassador_attribute_1: "Communicatieskills. Problemen die spelers hebben kunnen identificeren en ze helpen deze op te lossen. Verder zul je ook de rest van ons geïnformeerd houden over wat de spelers zeggen, wat ze leuk vinden, wat ze minder vinden en waar er meer van moet zijn!"
     ambassador_join_desc: "vertel ons wat over jezelf, wat je hebt gedaan en wat je graag zou doen. We zien verder wel!"    
     ambassador_join_note_strong: "Opmerking"
     ambassador_join_note_desc: "Een van onze topprioriteiten is om een multiplayer te bouwen waar spelers die moeite hebben een level op te lossen een wizard met een hoger level kunnen oproepen om te helpen. Dit zal een goede manier zijn voor ambassadeurs om hun ding te doen. We houden je op de hoogte!"    
     more_about_ambassador: "Leer meer over het worden van een behulpzame Ambassadeur"
     ambassador_subscribe_desc: "Ontvang e-mails met updates over ondersteuning en multiplayer-ontwikkelingen."    
-    counselor_summary: "Geen van de rollen hierboven in jouw interessegebied? Maak je geen zorgen, we zijn op zoek naar iedereen die wil helpen met het ontwikkelen van CodeCombat! Als je geïnteresseerd bent in lesgeven, gameontwikkeling, open source management of iets anders waarvan je denk dat het relevant voor ons is, dan is dit de klasse voor jou."
-    counselor_introduction_1: "Heb jij levenservaring? Een afwijkend perspectief op zaken die ons kunnen helpen CodeCombat te vormen? Van alle rollen neemt deze wellicht de minste tijd in, maar individiueel maak je misschien het grootste verschil. We zijn op zoek naar wijze tovenaars, vooral in het gebied van lesgeven, gameontwikkeling, open source projectmanagement, technische recrutering, ondernemerschap of design."    
+    counselor_summary: "Geen van de rollen hierboven in jouw interessegebied? Maak je geen zorgen, we zijn op zoek naar iedereen die wil helpen met het ontwikkelen van CodeCombat! Als je geïnteresseerd bent in lesgeven, gameontwikkeling, open source management of iets anders waarvan je denkt dat het relevant voor ons is, dan is dit de klasse voor jou."
+    counselor_introduction_1: "Heb jij levenservaring? Een afwijkend perspectief op zaken die ons kunnen helpen CodeCombat te vormen? Van alle rollen neemt deze wellicht de minste tijd in, maar individueel maak je misschien het grootste verschil. We zijn op zoek naar wijze tovenaars, vooral in het gebied van lesgeven, gameontwikkeling, open source projectmanagement, technische recrutering, ondernemerschap of design."
     counselor_introduction_2: "Of eigenlijk alles wat relevant is voor de ontwikkeling van CodeCombat. Als jij kennis hebt en deze wilt dezen om dit project te laten groeien, dan is dit misschien de klasse voor jou."    
     counselor_attribute_1: "Ervaring, in enig van de bovenstaande gebieden of iets anders waarvan je denkt dat het behulpzaam zal zijn."
     counselor_attribute_2: "Een beetje vrije tijd!"

From d0f416f66880a6db26aced8d02892a21fd361353 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Sun, 2 Mar 2014 12:43:21 -0800
Subject: [PATCH 018/178] Rearranging the ladder view.

---
 app/styles/play/ladder.sass                   |  10 +-
 app/templates/play/ladder.jade                |  78 ++++----
 app/templates/play/ladder/ladder_tab.jade     |  23 +++
 app/templates/play/ladder/my_matches_tab.jade |  45 +++++
 app/views/play/ladder/ladder_tab.coffee       |  97 ++++++++++
 app/views/play/ladder/my_matches_tab.coffee   | 101 ++++++++++
 app/views/play/ladder/utils.coffee            |  23 +++
 app/views/play/ladder_view.coffee             | 183 ++++--------------
 8 files changed, 372 insertions(+), 188 deletions(-)
 create mode 100644 app/templates/play/ladder/ladder_tab.jade
 create mode 100644 app/templates/play/ladder/my_matches_tab.jade
 create mode 100644 app/views/play/ladder/ladder_tab.coffee
 create mode 100644 app/views/play/ladder/my_matches_tab.coffee
 create mode 100644 app/views/play/ladder/utils.coffee

diff --git a/app/styles/play/ladder.sass b/app/styles/play/ladder.sass
index 027e755db..3ca1754e3 100644
--- a/app/styles/play/ladder.sass
+++ b/app/styles/play/ladder.sass
@@ -1,4 +1,10 @@
 #ladder-view
+  h1
+    text-align: center
+    
+  .tab-pane
+    margin-top: 10px
+  
   .score-cell
     width: 50px
     
@@ -10,10 +16,6 @@
     width: 45%
     margin: 0 2.5%
 
-  #simulation-status-text
-    display: inline
-    margin-left: 10px
-
   .name-col-cell
     max-width: 300px
     text-overflow: ellipsis
diff --git a/app/templates/play/ladder.jade b/app/templates/play/ladder.jade
index 1eb85a6f0..9e03f5f3c 100644
--- a/app/templates/play/ladder.jade
+++ b/app/templates/play/ladder.jade
@@ -2,17 +2,11 @@ extends /templates/base
 block content
 
   div#level-column
-    h3= level.get('name')
-    div#level-description
-      !{description}
-  
-    if !me.get('anonymous')
-      //a(href="http://www.youtube.com/watch?v=IFvfZiJGDsw&list=HL1392928835&feature=mh_lolz").intro-button.btn.btn-primary.btn-lg Watch the Video
-      
-      a(href="/play/level/brawlwood-tutorial").intro-button.btn.btn-primary.btn-lg Play the Tutorial
+    h1= level.get('name')
+    
+    //if !me.get('anonymous')
+    //  a(href="/play/level/brawlwood-tutorial").intro-button.btn.btn-primary.btn-lg Play the Tutorial
 
-  hr
-  
   if me.get('anonymous')
     div#must-log-in
       p
@@ -23,41 +17,37 @@ block content
 
   else
     div#columns.row
+      div.column.col-md-2
       for team in teams
-        div.column.col-md-6
+        div.column.col-md-4
           a(href="/play/ladder/#{levelID}/team/#{team.id}", style="background-color: #{team.primaryColor}").play-button.btn.btn-danger.btn-block.btn-lg
             span Play As 
             span= team.name
-  
-          table.table.table-bordered.table-condensed.table-hover
-            //(style="background-color: #{team.bgColor}")
-            tr
-              th(colspan=3, style="color: #{team.primaryColor}")
-                span= team.name
-                span  Leaderboard
-            tr
-              th Score
-              th.name-col-cell Name
-              th
-  
-            for session in team.leaderboard.topPlayers.models
-              - var myRow = session.get('creator') == me.id
-              tr(class=myRow ? "success" : "")
-                td.score-cell= session.get('totalScore').toFixed(2)
-                td.name-col-cell= session.get('creatorName') || "Anonymous"
-                td
-                  if(!myRow)
-                    a(href="/play/level/#{level.get('slug') || level.id}/?team=#{team.otherTeam}&opponent=#{session.id}") Battle as #{team.otherTeam}!
-                  else
-                    a(href="/play/ladder/#{levelID}/team/#{team.id}") View your #{team.id} matches.
-  
-    unless me.attributes.anonymous
-      hr
-      button.btn.btn-warning.btn-lg.highlight#simulate-button(style="margin-bottom:10px;") Simulate Games!
-      p(id="simulation-status-text", style="display:inline; margin-left:10px;")
-        if simulationStatus
-          | #{simulationStatus}
-        else
-          | By simulating games you can get your game ranked faster!
-    if me.isAdmin()
-      button.btn.btn-danger.btn-lg.highlight#simulate-all-button(style="margin-bottom:10px; float: right;") RESET AND SIMULATE GAMES
\ No newline at end of file
+      div.column.col-md-2
+    
+    hr
+      
+    ul.nav.nav-pills
+      li.active
+        a(href="#ladder", data-toggle="tab") Ladder
+      li
+        a(href="#my-matches", data-toggle="tab") My Matches
+      li
+        a(href="#simulate", data-toggle="tab") Simulate
+
+    div.tab-content
+      .tab-pane.active.well#ladder
+        #ladder-tab-view
+      .tab-pane.well#my-matches
+        | My Matches Pane
+      .tab-pane.well#simulate
+        p(id="simulation-status-text")
+          if simulationStatus
+            | #{simulationStatus}
+          else
+            | By simulating games you can get your game ranked faster!
+        p
+          button.btn.btn-warning.btn-lg.highlight#simulate-button() Simulate Games!
+        if me.isAdmin()
+          p
+            button.btn.btn-danger.btn-lg.highlight#simulate-all-button() RESET AND SIMULATE GAMES
\ No newline at end of file
diff --git a/app/templates/play/ladder/ladder_tab.jade b/app/templates/play/ladder/ladder_tab.jade
new file mode 100644
index 000000000..f0c7a1b1f
--- /dev/null
+++ b/app/templates/play/ladder/ladder_tab.jade
@@ -0,0 +1,23 @@
+div#columns.row
+  for team in teams
+    div.column.col-md-6
+      table.table.table-bordered.table-condensed.table-hover
+        tr
+          th(colspan=3, style="color: #{team.primaryColor}")
+            span= team.name
+            span  Leaderboard
+        tr
+          th Score
+          th.name-col-cell Name
+          th
+
+        for session in team.leaderboard.topPlayers.models
+          - var myRow = session.get('creator') == me.id
+          tr(class=myRow ? "success" : "")
+            td.score-cell= session.get('totalScore').toFixed(2)
+            td.name-col-cell= session.get('creatorName') || "Anonymous"
+            td
+              if(!myRow)
+                a(href="/play/level/#{level.get('slug') || level.id}/?team=#{team.otherTeam}&opponent=#{session.id}") Battle as #{team.otherTeam}!
+              else
+                a(href="/play/ladder/#{levelID}/team/#{team.id}") View your #{team.id} matches.
diff --git a/app/templates/play/ladder/my_matches_tab.jade b/app/templates/play/ladder/my_matches_tab.jade
new file mode 100644
index 000000000..d07432a83
--- /dev/null
+++ b/app/templates/play/ladder/my_matches_tab.jade
@@ -0,0 +1,45 @@
+//if matches.length
+//  p#your-score
+//    span Your Current Score:
+//    span  
+//    strong= score
+
+h3.pull-left Ranked Games
+div#columns.row
+  for team in teams
+    div#matches-column.col-md-6
+      button.btn.btn-warning.pull-right.rank-button(data-session-id=team.session.id)
+        span.unavailable.hidden No New Code to Rank
+        span.rank.hidden Rank My Game!
+        span.ranking.hidden Submitting...
+        span.ranked.hidden Submitted for Ranking
+        span.failed.hidden Failed to Rank
+      
+      hr.clearfix(style="clear: both")
+
+      if matches.length
+        table.table.table-bordered.table-condensed
+          tr
+            th Result
+            th Opponent
+            th When
+          for match in matches
+            tr
+              td.state-cell
+                if match.state === 'win'
+                  span.win Win
+                if match.state === 'loss'
+                  span.loss Loss
+                if match.state === 'tie'
+                  span.tie Tie
+              td.name-cell= match.opponentName || "Anonymous"
+              td.time-cell= match.when
+              td.battle-cell
+                - var text = match.state === 'win' ? 'Watch your victory' : 'Defeat the ' + otherTeamID
+                a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{match.sessionID}")= text
+      
+      else
+        div.alert.alert-warning
+          | No ranked matches played yet!
+          | Play some competitors on the right and then come back to get your game ranked.
+      
diff --git a/app/views/play/ladder/ladder_tab.coffee b/app/views/play/ladder/ladder_tab.coffee
new file mode 100644
index 000000000..f7d22f98f
--- /dev/null
+++ b/app/views/play/ladder/ladder_tab.coffee
@@ -0,0 +1,97 @@
+CocoView = require 'views/kinds/CocoView'
+Level = require 'models/Level'
+LevelSession = require 'models/LevelSession'
+CocoCollection = require 'models/CocoCollection'
+LeaderboardCollection  = require 'collections/LeaderboardCollection'
+{teamDataFromLevel} = require './utils'
+
+HIGHEST_SCORE = 1000000
+
+class LevelSessionsCollection extends CocoCollection
+  url: ''
+  model: LevelSession
+
+  constructor: (levelID) ->
+    super()
+    @url = "/db/level/#{levelID}/all_sessions"
+
+module.exports = class LadderView extends CocoView
+  id: 'ladder-tab-view'
+  template: require 'templates/play/ladder/ladder_tab'
+  startsLoading: true
+
+  constructor: (options, @level, @sessions) ->
+    super(options)
+    @teams = teamDataFromLevel @level
+    @leaderboards = {}
+    for team in @teams
+#      teamSession = _.find @sessions.models, (session) -> session.get('team') is team.id
+      teamSession = null
+      console.log "Team session: #{JSON.stringify teamSession}"
+      @leaderboards[team.id] = new LeaderboardData(@level, team.id, teamSession)
+      @leaderboards[team.id].once 'sync', @onLeaderboardLoaded, @
+
+  onChallengersLoaded: -> @renderMaybe()
+  onLeaderboardLoaded: -> @renderMaybe()
+
+  renderMaybe: ->
+    leaderboardModels = _.values(@leaderboards)
+    return unless _.every leaderboardModels, (loader) -> loader.loaded
+    @startsLoading = false
+    @render()
+
+  getRenderData: ->
+    ctx = super()
+    ctx.level = @level
+    ctx.link = "/play/level/#{@level.get('name')}"
+    ctx.teams = @teams
+    team.leaderboard = @leaderboards[team.id] for team in @teams
+    ctx.levelID = @levelID
+    ctx
+    
+class LeaderboardData
+  constructor: (@level, @team, @session) ->
+    _.extend @, Backbone.Events
+    @topPlayers = new LeaderboardCollection(@level, {order:-1, scoreOffset: HIGHEST_SCORE, team: @team, limit: if @session then 10 else 20})
+    @topPlayers.fetch()
+    @topPlayers.comparator = (model) ->
+      return -model.get('totalScore')
+    @topPlayers.sort()
+
+    @topPlayers.once 'sync', @leaderboardPartLoaded, @
+
+#    if @session
+#      score = @session.get('totalScore') or 25
+#      @playersAbove = new LeaderboardCollection(@level, {order:1, scoreOffset: score, limit: 4, team: @team})
+#      @playersAbove.fetch()
+#      @playersAbove.once 'sync', @leaderboardPartLoaded, @
+#      @playersBelow = new LeaderboardCollection(@level, {order:-1, scoreOffset: score, limit: 4, team: @team})
+#      @playersBelow.fetch()
+#      @playersBelow.once 'sync', @leaderboardPartLoaded, @
+
+  leaderboardPartLoaded: ->
+    if @session
+      if @topPlayers.loaded # and @playersAbove.loaded and @playersBelow.loaded
+        @loaded = true
+        @fetchNames()
+    else
+      @loaded = true
+      @fetchNames()
+
+  fetchNames: ->
+    sessionCollections = [@topPlayers, @playersAbove, @playersBelow]
+    sessionCollections = (s for s in sessionCollections when s)
+    ids = []
+    for collection in sessionCollections
+      ids.push model.get('creator') for model in collection.models
+
+    success = (nameMap) =>
+      for collection in sessionCollections
+        session.set('creatorName', nameMap[session.get('creator')]) for session in collection.models
+      @trigger 'sync'
+
+    $.ajax('/db/user/-/names', {
+      data: {ids: ids}
+      type: 'POST'
+      success: success
+    })
diff --git a/app/views/play/ladder/my_matches_tab.coffee b/app/views/play/ladder/my_matches_tab.coffee
new file mode 100644
index 000000000..3a4859c83
--- /dev/null
+++ b/app/views/play/ladder/my_matches_tab.coffee
@@ -0,0 +1,101 @@
+CocoView = require 'views/kinds/CocoView'
+Level = require 'models/Level'
+LevelSession = require 'models/LevelSession'
+LeaderboardCollection  = require 'collections/LeaderboardCollection'
+{teamDataFromLevel} = require './utils'
+
+module.exports = class LadderTeamView extends CocoView
+  id: 'ladder-team-view'
+  template: require 'templates/play/ladder/team'
+  startsLoading: true
+
+  events:
+    'click #rank-button': 'rankSession'
+
+  constructor: (options, @level, @sessions) ->
+    super(options)
+    @teams = teamDataFromLevel @level
+    @loadNames()
+
+  loadNames: ->
+    ids = []
+    for session in @sessions.models
+      ids.push match.opponents[0].userID for match in session.get('matches') or []
+
+    success = (@nameMap) =>
+      for match in @session.get('matches') or []
+        opponent = match.opponents[0]
+        opponent.userName = @nameMap[opponent.userID]
+      @finishRendering()
+
+    $.ajax('/db/user/-/names', {
+      data: {ids: ids}
+      type: 'POST'
+      success: success
+    })
+
+  finishRendering: ->
+    @startsLoading = false
+    @render()
+
+  getRenderData: ->
+    ctx = super()
+    ctx.level = @level
+    ctx.levelID = @level.get('slug') or @level.id
+    ctx.teams = @teams
+
+    convertMatch = (match) =>
+      opponent = match.opponents[0]
+      state = 'win'
+      state = 'loss' if match.metrics.rank > opponent.metrics.rank
+      state = 'tie' if match.metrics.rank is opponent.metrics.rank
+      {
+        state: state
+        opponentName: @nameMap[opponent.userID]
+        opponentID: opponent.userID
+        when: moment(match.date).fromNow()
+        sessionID: opponent.sessionID
+      }
+
+    for team in @teams
+      team.session = (s for s in @sessions.models when s.get('team') is team.id)[0]
+      team.readyToRank = @readyToRank(team.session)
+      team.matches = (convertMatch(match) for match in team.session.get('matches') or [])
+      team.matches.reverse()
+      team.score = (team.session.get('totalScore') or 10).toFixed(2)
+      
+    ctx
+
+  afterRender: ->
+    super()
+    @$el.find('.rank-button').each (i, el) =>
+      sessionID = button.data('session-id')
+      session = _.find @sessions.models, { id: sessionID }
+      @setRankingButtonText $(el), if @readyToRank(session) then 'rank' else 'unavailable'
+
+  readyToRank: (session) ->
+    c1 = session.get('code')
+    c2 = session.get('submittedCode')
+    c1 and not _.isEqual(c1, c2)
+
+  rankSession: (e) ->
+    button = $(e.target).closest('.rank-button')
+    sessionID = button.data('session-id')
+    session = _.find @sessions.models, { id: sessionID }
+    return unless @readyToRank(session)
+
+    @setRankingButtonText(button, 'ranking')
+    success = => @setRankingButtonText(button, 'ranked')
+    failure = => @setRankingButtonText(button, 'failed')
+
+    $.ajax '/queue/scoring', {
+      type: 'POST'
+      data: { session: sessionID }
+      success: success
+      failure: failure
+    }
+
+  setRankingButtonText: (rankButton, spanClass) ->
+    rankButton.find('span').addClass('hidden')
+    rankButton.find(".#{spanClass}").removeClass('hidden')
+    rankButton.toggleClass 'disabled', spanClass isnt 'rank'
diff --git a/app/views/play/ladder/utils.coffee b/app/views/play/ladder/utils.coffee
new file mode 100644
index 000000000..17667cbd2
--- /dev/null
+++ b/app/views/play/ladder/utils.coffee
@@ -0,0 +1,23 @@
+{hslToHex} = require 'lib/utils'
+
+module.exports.teamDataFromLevel = (level) ->
+  alliedSystem = _.find level.get('systems'), (value) -> value.config?.teams?
+  teamNames = (teamName for teamName, teamConfig of alliedSystem.config.teams when teamConfig.playable)
+  teamConfigs = alliedSystem.config.teams
+
+  teams = []
+  for team in teamNames or []
+    otherTeam = if team is 'ogres' then 'humans' else 'ogres'
+    color = teamConfigs[team].color
+    bgColor = hslToHex([color.hue, color.saturation, color.lightness + (1 - color.lightness) * 0.5])
+    primaryColor = hslToHex([color.hue, 0.5, 0.5])
+    teams.push({
+      id: team
+      name: _.string.titleize(team)
+      otherTeam: otherTeam
+      bgColor: bgColor
+      primaryColor: primaryColor
+    })
+
+  console.log 'created teams', teams
+  teams
\ No newline at end of file
diff --git a/app/views/play/ladder_view.coffee b/app/views/play/ladder_view.coffee
index 0ff4e6717..c331816b7 100644
--- a/app/views/play/ladder_view.coffee
+++ b/app/views/play/ladder_view.coffee
@@ -4,7 +4,8 @@ Simulator = require 'lib/simulator/Simulator'
 LevelSession = require 'models/LevelSession'
 CocoCollection = require 'models/CocoCollection'
 LeaderboardCollection  = require 'collections/LeaderboardCollection'
-{hslToHex} = require 'lib/utils'
+{teamDataFromLevel} = require './ladder/utils'
+LadderTabView = require './ladder/ladder_tab'
 
 HIGHEST_SCORE = 1000000
 
@@ -25,8 +26,48 @@ module.exports = class LadderView extends RootView
     'click #simulate-button': 'onSimulateButtonClick'
     'click #simulate-all-button': 'onSimulateAllButtonClick'
 
+  constructor: (options, @levelID) ->
+    super(options)
+    @level = new Level(_id:@levelID)
+    @level.fetch()
+    @level.once 'sync', @onLevelLoaded, @
+    #    @sessions = new LevelSessionsCollection(levelID)
+    #    @sessions.fetch({})
+    #    @sessions.once 'sync', @onMySessionsLoaded, @
+    @simulator = new Simulator()
+    @simulator.on 'statusUpdate', @updateSimulationStatus, @
+    @teams = []
+
+  onLevelLoaded: -> @renderMaybe()
+  onMySessionsLoaded: -> @renderMaybe()
+
+  renderMaybe: ->
+    return unless @level.loaded # and @sessions.loaded
+    @teams = teamDataFromLevel @level
+    console.log 'made teams', @teams
+    @startsLoading = false
+    @render()
+
+  getRenderData: ->
+    ctx = super()
+    ctx.level = @level
+    ctx.link = "/play/level/#{@level.get('name')}"
+    ctx.simulationStatus = @simulationStatus
+    ctx.teams = @teams
+    console.log 'ctx teams', ctx.teams
+    ctx.levelID = @levelID
+    ctx
+
+  afterRender: ->
+    super()
+    return if @startsLoading
+    @ladderTab = new LadderTabView({}, @level, @sessions)
+    @insertSubView(@ladderTab)
+
+  # Simulations
+
   onSimulateAllButtonClick: (e) ->
-    submitIDs = _.pluck @leaderboards[@teams[0]].topPlayers.models, "id"
+    submitIDs = _.pluck @leaderboards[@teams[0].id].topPlayers.models, "id"
     for ID in submitIDs
       $.ajax
         url: '/queue/scoring'
@@ -57,141 +98,3 @@ module.exports = class LadderView extends RootView
     catch e
       console.log "There was a problem with the named simulation status: #{e}"
     $("#simulation-status-text").text @simulationStatus
-
-
-  constructor: (options, @levelID) ->
-    super(options)
-    @level = new Level(_id:@levelID)
-    @level.fetch()
-    @level.once 'sync', @onLevelLoaded, @
-    @simulator = new Simulator()
-    @simulator.on 'statusUpdate', @updateSimulationStatus, @
-
-#    @sessions = new LevelSessionsCollection(levelID)
-#    @sessions.fetch({})
-#    @sessions.once 'sync', @onMySessionsLoaded, @
-
-  onLevelLoaded: -> @startLoadingPhaseTwoMaybe()
-  onMySessionsLoaded: ->
-    @startLoadingPhaseTwoMaybe()
-
-  startLoadingPhaseTwoMaybe: ->
-    return unless @level.loaded # and @sessions.loaded
-    @loadPhaseTwo()
-
-  loadPhaseTwo: ->
-    alliedSystem = _.find @level.get('systems'), (value) -> value.config?.teams?
-    teams = []
-    for teamName, teamConfig of alliedSystem.config.teams
-      continue unless teamConfig.playable
-      teams.push teamName
-    @teams = teams
-    @teamConfigs = alliedSystem.config.teams
-
-    @leaderboards = {}
-    @challengers = {}
-    for team in teams
-#      teamSession = _.find @sessions.models, (session) -> session.get('team') is team
-      teamSession = null
-      console.log "Team session: #{JSON.stringify teamSession}"
-      @leaderboards[team] = new LeaderboardData(@level, team, teamSession)
-      @leaderboards[team].once 'sync', @onLeaderboardLoaded, @
-
-  onChallengersLoaded: -> @renderMaybe()
-  onLeaderboardLoaded: -> @renderMaybe()
-
-  renderMaybe: ->
-    loaders = _.values(@leaderboards) # .concat(_.values(@challengers))
-    return unless _.every loaders, (loader) -> loader.loaded
-    @startsLoading = false
-    @render()
-
-  getRenderData: ->
-    ctx = super()
-    ctx.level = @level
-    description = @level.get('description')
-    ctx.description = if description then marked(description) else ''
-    ctx.link = "/play/level/#{@level.get('name')}"
-    ctx.simulationStatus = @simulationStatus
-    ctx.teams = []
-    ctx.levelID = @levelID
-    for team in @teams or []
-      otherTeam = if team is 'ogres' then 'humans' else 'ogres'
-      color = @teamConfigs[team].color
-      bgColor = hslToHex([color.hue, color.saturation, color.lightness + (1 - color.lightness) * 0.5])
-      primaryColor = hslToHex([color.hue, 0.5, 0.5])
-      ctx.teams.push({
-        id: team
-        name: _.string.titleize(team)
-        leaderboard: @leaderboards[team]
-        otherTeam: otherTeam
-        bgColor: bgColor
-        primaryColor: primaryColor
-      })
-    ctx
-
-class LeaderboardData
-  constructor: (@level, @team, @session) ->
-    _.extend @, Backbone.Events
-    @topPlayers = new LeaderboardCollection(@level, {order:-1, scoreOffset: HIGHEST_SCORE, team: @team, limit: if @session then 10 else 20})
-    @topPlayers.fetch()
-    @topPlayers.comparator = (model) ->
-      return -model.get('totalScore')
-    @topPlayers.sort()
-
-    @topPlayers.once 'sync', @leaderboardPartLoaded, @
-
-#    if @session
-#      score = @session.get('totalScore') or 25
-#      @playersAbove = new LeaderboardCollection(@level, {order:1, scoreOffset: score, limit: 4, team: @team})
-#      @playersAbove.fetch()
-#      @playersAbove.once 'sync', @leaderboardPartLoaded, @
-#      @playersBelow = new LeaderboardCollection(@level, {order:-1, scoreOffset: score, limit: 4, team: @team})
-#      @playersBelow.fetch()
-#      @playersBelow.once 'sync', @leaderboardPartLoaded, @
-
-  leaderboardPartLoaded: ->
-    if @session
-      if @topPlayers.loaded # and @playersAbove.loaded and @playersBelow.loaded
-        @loaded = true
-        @fetchNames()
-    else
-      @loaded = true
-      @fetchNames()
-
-  fetchNames: ->
-    sessionCollections = [@topPlayers, @playersAbove, @playersBelow]
-    sessionCollections = (s for s in sessionCollections when s)
-    ids = []
-    for collection in sessionCollections
-      ids.push model.get('creator') for model in collection.models
-
-    success = (nameMap) =>
-      for collection in sessionCollections
-        session.set('creatorName', nameMap[session.get('creator')]) for session in collection.models
-      @trigger 'sync'
-
-    $.ajax('/db/user/-/names', {
-      data: {ids: ids}
-      type: 'POST'
-      success: success
-    })
-
-class ChallengersData
-  constructor: (@level, @team, @session) ->
-    _.extend @, Backbone.Events
-    score = @session?.get('totalScore') or 25
-    @easyPlayer = new LeaderboardCollection(@level, {order:1, scoreOffset: score - 5, limit: 1, team: @team})
-    @easyPlayer.fetch()
-    @easyPlayer.once 'sync', @challengerLoaded, @
-    @mediumPlayer = new LeaderboardCollection(@level, {order:1, scoreOffset: score, limit: 1, team: @team})
-    @mediumPlayer.fetch()
-    @mediumPlayer.once 'sync', @challengerLoaded, @
-    @hardPlayer = new LeaderboardCollection(@level, {order:-1, scoreOffset: score + 5, limit: 1, team: @team})
-    @hardPlayer.fetch()
-    @hardPlayer.once 'sync', @challengerLoaded, @
-
-  challengerLoaded: ->
-    if @easyPlayer.loaded and @mediumPlayer.loaded and @hardPlayer.loaded
-      @loaded = true
-      @trigger 'sync'

From 956e2b3c40f94c577f6ca7d7892a3d729dab3097 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Sun, 2 Mar 2014 13:24:41 -0800
Subject: [PATCH 019/178] Set up the my matches tab in the new ladder view.

---
 app/templates/play/ladder.jade                |  2 +-
 app/templates/play/ladder/my_matches_tab.jade | 35 +++++++++++--------
 app/views/play/ladder/ladder_tab.coffee       |  2 +-
 app/views/play/ladder/my_matches_tab.coffee   | 19 +++++-----
 app/views/play/ladder/utils.coffee            |  1 -
 app/views/play/ladder_view.coffee             | 17 +++++----
 server/levels/level_handler.coffee            | 19 +++-------
 7 files changed, 46 insertions(+), 49 deletions(-)

diff --git a/app/templates/play/ladder.jade b/app/templates/play/ladder.jade
index 9e03f5f3c..4d056ce97 100644
--- a/app/templates/play/ladder.jade
+++ b/app/templates/play/ladder.jade
@@ -39,7 +39,7 @@ block content
       .tab-pane.active.well#ladder
         #ladder-tab-view
       .tab-pane.well#my-matches
-        | My Matches Pane
+        #my-matches-tab-view
       .tab-pane.well#simulate
         p(id="simulation-status-text")
           if simulationStatus
diff --git a/app/templates/play/ladder/my_matches_tab.jade b/app/templates/play/ladder/my_matches_tab.jade
index d07432a83..b9af08530 100644
--- a/app/templates/play/ladder/my_matches_tab.jade
+++ b/app/templates/play/ladder/my_matches_tab.jade
@@ -4,26 +4,33 @@
 //    span  
 //    strong= score
 
-h3.pull-left Ranked Games
 div#columns.row
   for team in teams
-    div#matches-column.col-md-6
-      button.btn.btn-warning.pull-right.rank-button(data-session-id=team.session.id)
-        span.unavailable.hidden No New Code to Rank
-        span.rank.hidden Rank My Game!
-        span.ranking.hidden Submitting...
-        span.ranked.hidden Submitted for Ranking
-        span.failed.hidden Failed to Rank
-      
-      hr.clearfix(style="clear: both")
-
-      if matches.length
+    div.matches-column.col-md-6
+      if team.matches.length
         table.table.table-bordered.table-condensed
+
+          tr
+            th(colspan=4, style="color: #{team.primaryColor}")
+              span Your
+              span  
+              span= team.name
+              span  
+              span Matches
+
+              button.btn.btn-sm.btn-warning.pull-right.rank-button(data-session-id=team.session.id)
+                span.unavailable.hidden No New Code to Rank
+                span.rank.hidden Rank My Game!
+                span.ranking.hidden Submitting...
+                span.ranked.hidden Submitted for Ranking
+                span.failed.hidden Failed to Rank
+
           tr
             th Result
             th Opponent
             th When
-          for match in matches
+            th
+          for match in team.matches
             tr
               td.state-cell
                 if match.state === 'win'
@@ -35,7 +42,7 @@ div#columns.row
               td.name-cell= match.opponentName || "Anonymous"
               td.time-cell= match.when
               td.battle-cell
-                - var text = match.state === 'win' ? 'Watch your victory' : 'Defeat the ' + otherTeamID
+                - var text = match.state === 'win' ? 'Watch your victory' : 'Defeat the ' + team.otherTeam
                 a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{match.sessionID}")= text
       
       else
diff --git a/app/views/play/ladder/ladder_tab.coffee b/app/views/play/ladder/ladder_tab.coffee
index f7d22f98f..96e268e3c 100644
--- a/app/views/play/ladder/ladder_tab.coffee
+++ b/app/views/play/ladder/ladder_tab.coffee
@@ -27,7 +27,7 @@ module.exports = class LadderView extends CocoView
     for team in @teams
 #      teamSession = _.find @sessions.models, (session) -> session.get('team') is team.id
       teamSession = null
-      console.log "Team session: #{JSON.stringify teamSession}"
+#      console.log "Team session: #{JSON.stringify teamSession}"
       @leaderboards[team.id] = new LeaderboardData(@level, team.id, teamSession)
       @leaderboards[team.id].once 'sync', @onLeaderboardLoaded, @
 
diff --git a/app/views/play/ladder/my_matches_tab.coffee b/app/views/play/ladder/my_matches_tab.coffee
index 3a4859c83..110a289db 100644
--- a/app/views/play/ladder/my_matches_tab.coffee
+++ b/app/views/play/ladder/my_matches_tab.coffee
@@ -4,17 +4,18 @@ LevelSession = require 'models/LevelSession'
 LeaderboardCollection  = require 'collections/LeaderboardCollection'
 {teamDataFromLevel} = require './utils'
 
-module.exports = class LadderTeamView extends CocoView
-  id: 'ladder-team-view'
-  template: require 'templates/play/ladder/team'
+module.exports = class MyMatchesTabView extends CocoView
+  id: 'my-matches-tab-view'
+  template: require 'templates/play/ladder/my_matches_tab'
   startsLoading: true
 
   events:
-    'click #rank-button': 'rankSession'
+    'click .rank-button': 'rankSession'
 
   constructor: (options, @level, @sessions) ->
     super(options)
     @teams = teamDataFromLevel @level
+    @nameMap = {}
     @loadNames()
 
   loadNames: ->
@@ -23,9 +24,10 @@ module.exports = class LadderTeamView extends CocoView
       ids.push match.opponents[0].userID for match in session.get('matches') or []
 
     success = (@nameMap) =>
-      for match in @session.get('matches') or []
-        opponent = match.opponents[0]
-        opponent.userName = @nameMap[opponent.userID]
+      for session in @sessions.models
+        for match in session.get('matches') or []
+          opponent = match.opponents[0]
+          opponent.userName = @nameMap[opponent.userID]
       @finishRendering()
 
     $.ajax('/db/user/-/names', {
@@ -69,9 +71,10 @@ module.exports = class LadderTeamView extends CocoView
   afterRender: ->
     super()
     @$el.find('.rank-button').each (i, el) =>
+      button = $(el)
       sessionID = button.data('session-id')
       session = _.find @sessions.models, { id: sessionID }
-      @setRankingButtonText $(el), if @readyToRank(session) then 'rank' else 'unavailable'
+      @setRankingButtonText button, if @readyToRank(session) then 'rank' else 'unavailable'
 
   readyToRank: (session) ->
     c1 = session.get('code')
diff --git a/app/views/play/ladder/utils.coffee b/app/views/play/ladder/utils.coffee
index 17667cbd2..10f088093 100644
--- a/app/views/play/ladder/utils.coffee
+++ b/app/views/play/ladder/utils.coffee
@@ -19,5 +19,4 @@ module.exports.teamDataFromLevel = (level) ->
       primaryColor: primaryColor
     })
 
-  console.log 'created teams', teams
   teams
\ No newline at end of file
diff --git a/app/views/play/ladder_view.coffee b/app/views/play/ladder_view.coffee
index c331816b7..25d15f484 100644
--- a/app/views/play/ladder_view.coffee
+++ b/app/views/play/ladder_view.coffee
@@ -6,6 +6,7 @@ CocoCollection = require 'models/CocoCollection'
 LeaderboardCollection  = require 'collections/LeaderboardCollection'
 {teamDataFromLevel} = require './ladder/utils'
 LadderTabView = require './ladder/ladder_tab'
+MyMatchesTabView = require './ladder/my_matches_tab'
 
 HIGHEST_SCORE = 1000000
 
@@ -15,7 +16,7 @@ class LevelSessionsCollection extends CocoCollection
 
   constructor: (levelID) ->
     super()
-    @url = "/db/level/#{levelID}/all_sessions"
+    @url = "/db/level/#{levelID}/my_sessions"
 
 module.exports = class LadderView extends RootView
   id: 'ladder-view'
@@ -31,9 +32,9 @@ module.exports = class LadderView extends RootView
     @level = new Level(_id:@levelID)
     @level.fetch()
     @level.once 'sync', @onLevelLoaded, @
-    #    @sessions = new LevelSessionsCollection(levelID)
-    #    @sessions.fetch({})
-    #    @sessions.once 'sync', @onMySessionsLoaded, @
+    @sessions = new LevelSessionsCollection(levelID)
+    @sessions.fetch({})
+    @sessions.once 'sync', @onMySessionsLoaded, @
     @simulator = new Simulator()
     @simulator.on 'statusUpdate', @updateSimulationStatus, @
     @teams = []
@@ -42,9 +43,8 @@ module.exports = class LadderView extends RootView
   onMySessionsLoaded: -> @renderMaybe()
 
   renderMaybe: ->
-    return unless @level.loaded # and @sessions.loaded
+    return unless @level.loaded and @sessions.loaded
     @teams = teamDataFromLevel @level
-    console.log 'made teams', @teams
     @startsLoading = false
     @render()
 
@@ -54,15 +54,14 @@ module.exports = class LadderView extends RootView
     ctx.link = "/play/level/#{@level.get('name')}"
     ctx.simulationStatus = @simulationStatus
     ctx.teams = @teams
-    console.log 'ctx teams', ctx.teams
     ctx.levelID = @levelID
     ctx
 
   afterRender: ->
     super()
     return if @startsLoading
-    @ladderTab = new LadderTabView({}, @level, @sessions)
-    @insertSubView(@ladderTab)
+    @insertSubView(@ladderTab = new LadderTabView({}, @level, @sessions))
+    @insertSubView(@myMatchesTab = new MyMatchesTabView({}, @level, @sessions))
 
   # Simulations
 
diff --git a/server/levels/level_handler.coffee b/server/levels/level_handler.coffee
index b26e729bd..0cf0a2066 100644
--- a/server/levels/level_handler.coffee
+++ b/server/levels/level_handler.coffee
@@ -27,7 +27,7 @@ LevelHandler = class LevelHandler extends Handler
   getByRelationship: (req, res, args...) ->
     return @getSession(req, res, args[0]) if args[1] is 'session'
     return @getLeaderboard(req, res, args[0]) if args[1] is 'leaderboard'
-    return @getAllSessions(req, res, args[0]) if args[1] is 'all_sessions'
+    return @getMySessions(req, res, args[0]) if args[1] is 'my_sessions'
     return @getFeedback(req, res, args[0]) if args[1] is 'feedback'
     return @sendNotFoundError(res)
 
@@ -86,26 +86,15 @@ LevelHandler = class LevelHandler extends Handler
       # associated with the handler, because the handler might return a different type
       # of model, like in this case. Refactor to move that logic to the model instead.
 
-  getAllSessions: (req, res, id) ->
+  getMySessions: (req, res, id) ->
     @fetchLevelByIDAndHandleErrors id, req, res, (err, level) =>
       sessionQuery =
         level:
           original: level.original.toString()
           majorVersion: level.version.major
-        submitted: true
-
-      propertiesToReturn = [
-        '_id'
-        'totalScore'
-        'submitted'
-        'team'
-        'creatorName'
-      ]
-
-      query = Session
-        .find(sessionQuery)
-        .select(propertiesToReturn.join ' ')
+        creator: req.user._id+''
 
+      query = Session.find(sessionQuery)
       query.exec (err, results) =>
         if err then @sendDatabaseError(res, err) else @sendSuccess res, results
 

From bea002a8d49856e9f6a2e41e5db2a5aef1bf212f Mon Sep 17 00:00:00 2001
From: Ruben Vereecken <ruben.vereecken@student.uantwerpen.be>
Date: Sun, 2 Mar 2014 22:59:15 +0100
Subject: [PATCH 020/178] Fixed some wrong translations

---
 app/locale/nl.coffee | 8 ++++----
 1 file changed, 4 insertions(+), 4 deletions(-)

diff --git a/app/locale/nl.coffee b/app/locale/nl.coffee
index 57a366206..9182c54db 100644
--- a/app/locale/nl.coffee
+++ b/app/locale/nl.coffee
@@ -8,7 +8,7 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     delay_1_sec: "1 seconde"
     delay_3_sec: "3 secondes"
     delay_5_sec: "5 secondes"
-    manual: "Handboek"
+    manual: "Handmatig"
     fork: "Fork"
     play: "Spelen"
 
@@ -215,11 +215,11 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     main_title: "CodeCombat Editors"
     main_description: "Maak je eigen levels, campagnes, eenheden en leermateriaal. Wij bieden alle programma's aan die u nodig heeft!"
     article_title: "Artikel Editor"
-    article_description: "Schrijf artikels dat spelers een overzicht geven over programmeer concepten die kunnen gebruikt worden over een variëteit van levels en campagnes."
+    article_description: "Schrijf artikels die spelers een overzicht geven over programmeer concepten die kunnen gebruikt worden over een variëteit van levels en campagnes."
     thang_title: "Thang Editor"
-    thang_description: "Maak eenheden, beschrijf hun default logica, graphics en audio. Momenteel is enkel het importeren van vector graphics geëxporteerd in Flash ondersteund."
+    thang_description: "Maak eenheden, beschrijf hun standaard logica, graphics en audio. Momenteel is enkel het importeren van vector graphics geëxporteerd in Flash ondersteund."
     level_title: "Level Editor"
-    level_description: "Bevat het programma om te programmeren, audio te uploaden en aangepaste logica om alle soorten levels te maken. Het is alles wat wijzelf ook gebruiken!"
+    level_description: "Bevat het programma om te programmeren, audio te uploaden en aangepaste logica te creëren om alle soorten levels te maken. Het is alles wat wijzelf ook gebruiken!"
     security_notice: "Veel belangrijke elementen in deze editors zijn momenteel niet actief. Met dat wij de veiligheid van deze systemen verbeteren, zullen ook deze elementen beschikbaar worden. Indien u deze elementen al eerder wil gebruiken, "
     contact_us: "contacteer ons!"
     hipchat_prefix: "Je kan ons ook vinden in ons"

From c3c699df490ecd8cf2d78115c7d64eb28d4e9c16 Mon Sep 17 00:00:00 2001
From: Ruben Vereecken <ruben.vereecken@student.uantwerpen.be>
Date: Sun, 2 Mar 2014 23:28:16 +0100
Subject: [PATCH 021/178] Fixed wrong i18n name 'account_settings.autosave'.
 Added missing i18n name 'account_settings.email_notifications'

---
 app/locale/en.coffee                | 1 +
 app/locale/nl.coffee                | 3 ++-
 app/templates/account/settings.jade | 2 +-
 3 files changed, 4 insertions(+), 2 deletions(-)

diff --git a/app/locale/en.coffee b/app/locale/en.coffee
index 63b228460..d72b872cd 100644
--- a/app/locale/en.coffee
+++ b/app/locale/en.coffee
@@ -130,6 +130,7 @@ module.exports = nativeDescription: "English", englishDescription: "English", tr
     new_password_verify: "Verify"
     email_subscriptions: "Email Subscriptions"
     email_announcements: "Announcements"
+    email_notifications: "Notifications"
     email_notifications_description: "Get periodic notifications for your account."
     email_announcements_description: "Get emails on the latest news and developments at CodeCombat."
     contributor_emails: "Contributor Class Emails"
diff --git a/app/locale/nl.coffee b/app/locale/nl.coffee
index 9182c54db..8e762eaeb 100644
--- a/app/locale/nl.coffee
+++ b/app/locale/nl.coffee
@@ -116,7 +116,7 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
   account_settings:
     title: "Account Instellingen"
     not_logged_in: "Log in of maak een account om je instellingen aan te passen."
-    autosave: "Aanpassingen Worden Automatisch Opgeslagen"
+    autosave: "Aanpassingen Automatisch Opgeslagen"
     me_tab: "Ik"
     picture_tab: "Afbeelding"
     wizard_tab: "Tovenaar"
@@ -130,6 +130,7 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     new_password_verify: "Verifieer"
     email_subscriptions: "E-mail Abonnementen"
     email_announcements: "Aankondigingen"
+    email_notifications: "Notificaties"
     email_notifications_description: "Krijg periodieke meldingen voor jouw account."
     email_announcements_description: "Verkrijg emails over het laatste nieuws en de ontwikkelingen bij CodeCombat."
     contributor_emails: "Medewerker Klasse emails"
diff --git a/app/templates/account/settings.jade b/app/templates/account/settings.jade
index bae178865..d64936645 100644
--- a/app/templates/account/settings.jade
+++ b/app/templates/account/settings.jade
@@ -8,7 +8,7 @@ block content
     p(data-i18n="account_settings.not_logged_in") Log in or create an account to change your settings.
 
   else
-    button.btn#save-button.disabled.secret(data-i18n="account_settings.saveBackups") Changes Save Automatically
+    button.btn#save-button.disabled.secret(data-i18n="account_settings.autosave") Changes Save Automatically
   
     ul.nav.nav-pills#settings-tabs
       li

From c2d247e686d0709bca5b280f638d511cb9a2f627 Mon Sep 17 00:00:00 2001
From: Ruben Vereecken <ruben.vereecken@student.uantwerpen.be>
Date: Mon, 3 Mar 2014 00:31:49 +0100
Subject: [PATCH 022/178] Editor tables now load i18n correctly after search.

---
 app/views/kinds/SearchView.coffee | 1 +
 1 file changed, 1 insertion(+)

diff --git a/app/views/kinds/SearchView.coffee b/app/views/kinds/SearchView.coffee
index e89d080e4..18648e386 100644
--- a/app/views/kinds/SearchView.coffee
+++ b/app/views/kinds/SearchView.coffee
@@ -77,6 +77,7 @@ module.exports = class ThangTypeHomeView extends View
     documents = @collection.models
     table = $(@tableTemplate(documents:documents))
     @$el.find('table').replaceWith(table)
+    @$el.find('table').i18n()
 
   removeOldSearch: ->
     return unless @collection?

From a8ef87c9ea0255e3c28e6c79f716f70d40778fc8 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sun, 2 Mar 2014 16:06:22 -0800
Subject: [PATCH 023/178] Fixed a few bugs; re-enabled protectAPI for
 Brawlwood.

---
 app/lib/God.coffee                         |  6 +++---
 app/templates/editor/level/save.jade       |  4 ----
 app/templates/modal/save_version.jade      | 16 +++++++++-------
 app/views/editor/article/edit.coffee       |  2 +-
 app/views/editor/level/save_view.coffee    |  2 +-
 app/views/play/level/tome/tome_view.coffee |  2 +-
 6 files changed, 15 insertions(+), 17 deletions(-)

diff --git a/app/lib/God.coffee b/app/lib/God.coffee
index 3f80500bf..ff92f7e31 100644
--- a/app/lib/God.coffee
+++ b/app/lib/God.coffee
@@ -194,8 +194,8 @@ class Angel
     @ids[@lastID]
 
   # https://github.com/codecombat/codecombat/issues/81 -- TODO: we need to wait for worker initialization first
-  infiniteLoopIntervalDuration: 5000  # check this often (must be more than the others added)
-  infiniteLoopTimeoutDuration: 1500  # wait this long when we check
+  infiniteLoopIntervalDuration: 7500  # check this often (must be more than the others added)
+  infiniteLoopTimeoutDuration: 2500  # wait this long when we check
   abortTimeoutDuration: 500  # give in-process or dying workers this long to give up
   constructor: (@god) ->
     @id = Angel.nextID()
@@ -227,7 +227,7 @@ class Angel
       _.delay ->
         worker.terminate()
         worker.removeEventListener 'message', onWorkerMessage
-      , 1000
+      , 2000
       @worker = null
     @
 
diff --git a/app/templates/editor/level/save.jade b/app/templates/editor/level/save.jade
index 00e9e43f2..8ada52b23 100644
--- a/app/templates/editor/level/save.jade
+++ b/app/templates/editor/level/save.jade
@@ -52,7 +52,3 @@ block modal-body-content
           label.control-label(for=id + "-version-is-major") Major Changes?
           input(id=id + "-version-is-major", name="version-is-major", type="checkbox")
           span.help-block (Could this update break anything depending on this System?)
-
-if noSaveButton
-  block modal-footer-content
-    button.btn(data-dismiss="modal", data-i18n="common.cancel") Cancel
diff --git a/app/templates/modal/save_version.jade b/app/templates/modal/save_version.jade
index 3c6a35f48..d1f8fc219 100644
--- a/app/templates/modal/save_version.jade
+++ b/app/templates/modal/save_version.jade
@@ -17,12 +17,14 @@ block modal-body-wait-content
   h3(data-i18n="common.saving") Saving...
 
 block modal-footer-content
-  #accept-cla-wrapper.alert.alert-info
-    span(data-i18n="versions.cla_prefix") To save changes, first you must agree to our
-    |  
-    strong#cla-link(data-i18n="versions.cla_url") CLA
-    span(data-i18n="versions.cla_suffix") .
-    button.btn#agreement-button(data-i18n="versions.cla_agree") I AGREE
+  if !noSaveButton
+    #accept-cla-wrapper.alert.alert-info
+      span(data-i18n="versions.cla_prefix") To save changes, first you must agree to our
+      |  
+      strong#cla-link(data-i18n="versions.cla_url") CLA
+      span(data-i18n="versions.cla_suffix") .
+      button.btn#agreement-button(data-i18n="versions.cla_agree") I AGREE
     
   button.btn(data-dismiss="modal", data-i18n="common.cancel") Cancel
-  button.btn.btn-primary#save-version-button(data-i18n="common.save") Save
+  if !noSaveButton
+    button.btn.btn-primary#save-version-button(data-i18n="common.save") Save
diff --git a/app/views/editor/article/edit.coffee b/app/views/editor/article/edit.coffee
index 05df5291d..7a2894a09 100644
--- a/app/views/editor/article/edit.coffee
+++ b/app/views/editor/article/edit.coffee
@@ -59,7 +59,7 @@ module.exports = class ArticleEditView extends View
     context
 
   openPreview: =>
-    @preview = window.open('http://localhost:3000/editor/article/x/preview', 'preview', 'height=800,width=600')
+    @preview = window.open('/editor/article/x/preview', 'preview', 'height=800,width=600')
     @preview.focus() if window.focus
     @preview.onload = => @pushChangesToPreview()
     return false
diff --git a/app/views/editor/level/save_view.coffee b/app/views/editor/level/save_view.coffee
index 4edf03ca0..e3e5ad25c 100644
--- a/app/views/editor/level/save_view.coffee
+++ b/app/views/editor/level/save_view.coffee
@@ -23,7 +23,7 @@ module.exports = class LevelSaveView extends SaveVersionModal
     context.levelNeedsSave = @level.hasLocalChanges()
     context.modifiedComponents = _.filter @supermodel.getModels(LevelComponent), @shouldSaveEntity
     context.modifiedSystems = _.filter @supermodel.getModels(LevelSystem), @shouldSaveEntity
-    context.noSaveButton = context.levelNeedsSave or context.modifiedComponents.length or context.modifiedSystems.length
+    context.noSaveButton = not (context.levelNeedsSave or context.modifiedComponents.length or context.modifiedSystems.length)
     context
 
   shouldSaveEntity: (m) ->
diff --git a/app/views/play/level/tome/tome_view.coffee b/app/views/play/level/tome/tome_view.coffee
index 11fad104b..124e54e0f 100644
--- a/app/views/play/level/tome/tome_view.coffee
+++ b/app/views/play/level/tome/tome_view.coffee
@@ -128,7 +128,7 @@ module.exports = class TomeView extends View
         spellKey = pathComponents.join '/'
         @thangSpells[thang.id].push spellKey
         unless method.cloneOf
-          skipProtectAPI = true  #@getQueryVariable("skip_protect_api") is "true"
+          skipProtectAPI = @getQueryVariable("skip_protect_api") is "true" or @options.levelID isnt 'brawlwood'
           skipFlow = @getQueryVariable("skip_flow") is "true" or @options.levelID is 'brawlwood'
           spell = @spells[spellKey] = new Spell programmableMethod: method, spellKey: spellKey, pathComponents: pathPrefixComponents.concat(pathComponents), session: @options.session, supermodel: @supermodel, skipFlow: skipFlow, skipProtectAPI: skipProtectAPI, worker: @worker
     for thangID, spellKeys of @thangSpells

From 82b250b9a3e0431918b86661744735c675c5a1cc Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sun, 2 Mar 2014 17:51:57 -0800
Subject: [PATCH 024/178] protectAPI off again for Brawlwood, too slow for now.

---
 app/views/play/level/tome/tome_view.coffee |  1 +
 server/queues/scoring.coffee               | 74 +++++++++++-----------
 2 files changed, 38 insertions(+), 37 deletions(-)

diff --git a/app/views/play/level/tome/tome_view.coffee b/app/views/play/level/tome/tome_view.coffee
index 124e54e0f..f7be355f6 100644
--- a/app/views/play/level/tome/tome_view.coffee
+++ b/app/views/play/level/tome/tome_view.coffee
@@ -129,6 +129,7 @@ module.exports = class TomeView extends View
         @thangSpells[thang.id].push spellKey
         unless method.cloneOf
           skipProtectAPI = @getQueryVariable("skip_protect_api") is "true" or @options.levelID isnt 'brawlwood'
+          skipProtectAPI = true  # gah, it's so slow :( and somehow still affects simulation
           skipFlow = @getQueryVariable("skip_flow") is "true" or @options.levelID is 'brawlwood'
           spell = @spells[spellKey] = new Spell programmableMethod: method, spellKey: spellKey, pathComponents: pathPrefixComponents.concat(pathComponents), session: @options.session, supermodel: @supermodel, skipFlow: skipFlow, skipProtectAPI: skipProtectAPI, worker: @worker
     for thangID, spellKeys of @thangSpells
diff --git a/server/queues/scoring.coffee b/server/queues/scoring.coffee
index 0079d34a8..a1119378b 100644
--- a/server/queues/scoring.coffee
+++ b/server/queues/scoring.coffee
@@ -23,13 +23,13 @@ connectToScoringQueue = ->
       if error? then throw new Error  "There was an error registering the scoring queue: #{error}"
       scoringTaskQueue = data
       log.info "Connected to scoring task queue!"
-      
+
 module.exports.addPairwiseTaskToQueueFromRequest = (req, res) ->
   taskPair = req.body.sessions
   addPairwiseTaskToQueue req.body.sessions (err, success) ->
     if err? then return errors.serverError res, "There was an error adding pairwise tasks: #{err}"
     sendResponseObject req, res, {"message":"All task pairs were succesfully sent to the queue"}
-    
+
 
 addPairwiseTaskToQueue = (taskPair, cb) ->
   LevelSession.findOne(_id:taskPair[0]).lean().exec (err, firstSession) =>
@@ -42,9 +42,9 @@ addPairwiseTaskToQueue = (taskPair, cb) ->
         if e then return cb e, false
 
       sendEachTaskPairToTheQueue taskPairs, (taskPairError) ->
-        if taskPairError? then return cb taskPairError,false 
+        if taskPairError? then return cb taskPairError,false
         cb null, true
-        
+
 
 module.exports.createNewTask = (req, res) ->
   requestSessionID = req.body.session
@@ -105,6 +105,7 @@ module.exports.processTaskResult = (req, res) ->
     return handleTimedOutTask req, res, clientResponseObject if hasTaskTimedOut taskLogJSON.sentDate
 
     scoringTaskQueue.deleteMessage clientResponseObject.receiptHandle, (err) ->
+      console.log "Deleted message."
       if err? then return errors.badInput res, "The queue message is already back in the queue, rejecting results."
 
       logTaskComputation clientResponseObject, taskLog, (logErr) ->
@@ -117,14 +118,14 @@ module.exports.processTaskResult = (req, res) ->
 
           addMatchToSessions clientResponseObject, newScoresObject, (err, data) ->
             if err? then return errors.serverError res, "There was an error updating the sessions with the match! #{JSON.stringify err}"
-              
+
             originalSessionID = clientResponseObject.originalSessionID
             originalSessionTeam = clientResponseObject.originalSessionTeam
             originalSessionRank = parseInt clientResponseObject.originalSessionRank
-            
+
             determineIfSessionShouldContinueAndUpdateLog originalSessionID, originalSessionRank, (err, sessionShouldContinue) ->
               if err? then return errors.serverError res, "There was an error determining if the session should continue, #{err}"
-                
+
               if sessionShouldContinue
                 opposingTeam = calculateOpposingTeam(originalSessionTeam)
                 opponentID = _.pull(_.keys(newScoresObject), originalSessionID)
@@ -133,7 +134,7 @@ module.exports.processTaskResult = (req, res) ->
                 findNearestBetterSessionID originalSessionID, sessionNewScore, opponentNewScore, opponentID ,opposingTeam, (err, opponentSessionID) ->
                   if err? then return errors.serverError res, "There was an error finding the nearest sessionID!"
                   unless opponentSessionID then return sendResponseObject req, res, {"message":"There were no more games to rank(game is at top!"}
-                    
+
                   addPairwiseTaskToQueue [originalSessionID, opponentSessionID], (err, success) ->
                     if err? then return errors.serverError res, "There was an error sending the pairwise tasks to the queue!"
                     sendResponseObject req, res, {"message":"The scores were updated successfully and more games were sent to the queue!"}
@@ -141,19 +142,19 @@ module.exports.processTaskResult = (req, res) ->
                 console.log "Player lost, achieved rank #{originalSessionRank}"
                 sendResponseObject req, res, {"message":"The scores were updated successfully, person lost so no more games are being inserted!"}
 
-              
+
 determineIfSessionShouldContinueAndUpdateLog = (sessionID, sessionRank, cb) ->
-  queryParameters = 
+  queryParameters =
     _id: sessionID
-  
-  updateParameters = 
+
+  updateParameters =
     "$inc": {}
-  
-  if sessionRank is 0 
+
+  if sessionRank is 0
     updateParameters["$inc"] = {numberOfWinsAndTies: 1}
   else
     updateParameters["$inc"] = {numberOfLosses: 1}
- 
+
   LevelSession.findOneAndUpdate queryParameters, updateParameters,{select: 'numberOfWinsAndTies numberOfLosses'}, (err, updatedSession) ->
     if err? then return cb err, updatedSession
     updatedSession = updatedSession.toObject()
@@ -170,16 +171,16 @@ determineIfSessionShouldContinueAndUpdateLog = (sessionID, sessionRank, cb) ->
       else
         console.log "Ratio(#{ratio}) is good, so continuing simulations"
         cb null, true
-      
-    
+
+
 findNearestBetterSessionID = (sessionID, sessionTotalScore, opponentSessionTotalScore, opponentSessionID, opposingTeam, cb) ->
   retrieveAllOpponentSessionIDs sessionID, (err, opponentSessionIDs) ->
     if err? then return cb err, null
-    
+
     queryParameters =
-      totalScore: 
+      totalScore:
         $gt:opponentSessionTotalScore
-      _id: 
+      _id:
         $nin: opponentSessionIDs
       "level.original": "52d97ecd32362bc86e004e87"
       "level.majorVersion": 0
@@ -187,20 +188,20 @@ findNearestBetterSessionID = (sessionID, sessionTotalScore, opponentSessionTotal
       submittedCode:
         $exists: true
       team: opposingTeam
-      
+
     limitNumber = 1
-  
+
     sortParameters =
       totalScore: 1
-      
+
     selectString = '_id totalScore'
-      
+
     query = LevelSession.findOne(queryParameters)
       .sort(sortParameters)
       .limit(limitNumber)
       .select(selectString)
       .lean()
-    
+
     console.log "Finding session with score near #{opponentSessionTotalScore}"
     query.exec (err, session) ->
       if err? then return cb err, session
@@ -208,7 +209,7 @@ findNearestBetterSessionID = (sessionID, sessionTotalScore, opponentSessionTotal
       console.log "Found session with score #{session.totalScore}"
       cb err, session._id
 
-    
+
 retrieveAllOpponentSessionIDs = (sessionID, cb) ->
   query = LevelSession.findOne({"_id":sessionID})
     .select('matches.opponents.sessionID')
@@ -217,14 +218,14 @@ retrieveAllOpponentSessionIDs = (sessionID, cb) ->
     if err? then return cb err, null
     opponentSessionIDs = (match.opponents[0].sessionID for match in session.matches)
     cb err, opponentSessionIDs
-    
-      
+
+
 calculateOpposingTeam = (sessionTeam) ->
   teams = ['ogres','humans']
   opposingTeams = _.pull teams, sessionTeam
   return opposingTeams[0]
-  
-  
+
+
 validatePermissions = (req, sessionID, callback) ->
   if isUserAnonymous req then return callback null, false
   if isUserAdmin req then return callback null, true
@@ -298,17 +299,17 @@ fetchInitialSessionsToRankAgainst = (opposingTeam, callback) ->
     submittedCode:
       $exists: true
     team: opposingTeam
-  
-  sortParameters = 
+
+  sortParameters =
     totalScore: 1
-  
+
   limitNumber = 1
-  
+
   query = LevelSession.find(findParameters)
     .sort(sortParameters)
     .limit(limitNumber)
-  
-  
+
+
   query.exec callback
 
 generateTaskPairs = (submittedSessions, sessionToScore) ->
@@ -444,4 +445,3 @@ retrieveOldSessionData = (sessionID, callback) ->
       "totalScore":session.totalScore ? (25 - 1.8*(25/3))
       "id": sessionID
     callback err, oldScoreObject
-

From 4b6ebd6532a53c143a157d3d8e13c1587d35333b Mon Sep 17 00:00:00 2001
From: George Saines <gsaines@gmail.com>
Date: Sun, 2 Mar 2014 18:21:42 -0800
Subject: [PATCH 025/178] styled the victory modal a bit

---
 app/styles/play/level/modal/victory.sass    | 27 +++++++++++++++++----
 app/templates/play/level/modal/victory.jade | 16 ++++++------
 2 files changed, 31 insertions(+), 12 deletions(-)

diff --git a/app/styles/play/level/modal/victory.sass b/app/styles/play/level/modal/victory.sass
index d29b67bcd..5e2b9e727 100644
--- a/app/styles/play/level/modal/victory.sass
+++ b/app/styles/play/level/modal/victory.sass
@@ -2,12 +2,18 @@
   p.sign-up-poke
     text-align: left
     margin-bottom: 10px
+
   .sign-up-button
+    margin-left: 20px
+
+  .next-level-button
+    margin-bottom: 10px
+    margin-right: 30px
     float: right
-    margin-left: 10px
     
   .rating
-    float: left
+    float: center
+    margin-right: 22%
     span
       margin-right: 5px
     i
@@ -30,7 +36,18 @@
     clear: both
 
   .modal-header
-    text-align: center
+    text-align: left
 
-    .victory-banner
-      width: 200px
+  .victory-banner
+    width: 450px
+    position: absolute
+    left: -150px
+
+  .modal-dialog
+    margin-left: 30%
+    padding-left: 300px
+    width: 700px
+
+  .modal-footer
+    margin: 0px
+    padding: 0px
diff --git a/app/templates/play/level/modal/victory.jade b/app/templates/play/level/modal/victory.jade
index 3509e4fe9..7a0637f8a 100644
--- a/app/templates/play/level/modal/victory.jade
+++ b/app/templates/play/level/modal/victory.jade
@@ -1,5 +1,9 @@
 // TODO: refactor to be like other modals
+
 .modal-dialog
+
+  img.victory-banner(src="/images/level/victory.png", alt="")
+
   .modal-header
     button(type='button', data-dismiss="modal", aria-hidden="true").close &times;
     h3
@@ -7,15 +11,17 @@
       span= levelName
       span(data-i18n="play_level.victory_title_suffix") Complete
 
-    img.victory-banner(src="/images/level/victory.png", alt="")
-  
   .modal-body!= body
   
   .modal-footer
+    if hasNextLevel
+      button.btn.btn-primary.next-level-button(data-dismiss="modal", data-i18n="play_level.victory_play_next_level") Play Next Level
+    else
+      a.btn.btn-primary(href="/", data-dismiss="modal", data-i18n="play_level.victory_go_home") Go Home
     if me.get('anonymous')
       p.sign-up-poke
         button.btn.btn-success.sign-up-button.btn-large(data-toggle="coco-modal", data-target="modal/signup", data-i18n="play_level.victory_sign_up") Sign Up to Save Progress
-        span(data-i18n="play_level.victory_sign_up_poke") Want to save your code? Create a free account!
+          span(data-i18n="play_level.victory_sign_up_poke") Want to save your code? Create a free account!
       p.clearfix
     else
       div.rating.secret
@@ -25,10 +31,6 @@
         i.icon-star-empty
         i.icon-star-empty
         i.icon-star-empty
-    if hasNextLevel
-      button.btn.btn-primary.next-level-button(data-dismiss="modal", data-i18n="play_level.victory_play_next_level") Play Next Level
-    else
-      a.btn.btn-primary(href="/", data-dismiss="modal", data-i18n="play_level.victory_go_home") Go Home
     if !me.get('anonymous')
       div.review.secret
         span(data-i18n="play_level.victory_review") Tell us more!

From 81466675ec4c87e8c2eb4c74c4d8e144ddfdce1c Mon Sep 17 00:00:00 2001
From: Michael Schmatz <schmatz@umich.edu>
Date: Sun, 2 Mar 2014 18:55:07 -0800
Subject: [PATCH 026/178] Scheduling fixes

---
 server/queues/scoring.coffee | 77 ++++++++++++++++++++----------------
 1 file changed, 43 insertions(+), 34 deletions(-)

diff --git a/server/queues/scoring.coffee b/server/queues/scoring.coffee
index a1119378b..eb0815e38 100644
--- a/server/queues/scoring.coffee
+++ b/server/queues/scoring.coffee
@@ -107,40 +107,49 @@ module.exports.processTaskResult = (req, res) ->
     scoringTaskQueue.deleteMessage clientResponseObject.receiptHandle, (err) ->
       console.log "Deleted message."
       if err? then return errors.badInput res, "The queue message is already back in the queue, rejecting results."
-
-      logTaskComputation clientResponseObject, taskLog, (logErr) ->
-        if logErr? then return errors.serverError res, "There as a problem logging the task computation: #{logErr}"
-
-        updateSessions clientResponseObject, (updateError, newScoreArray) ->
-          if updateError? then return errors.serverError res, "There was an error updating the scores.#{updateError}"
-
-          newScoresObject = _.indexBy newScoreArray, 'id'
-
-          addMatchToSessions clientResponseObject, newScoresObject, (err, data) ->
-            if err? then return errors.serverError res, "There was an error updating the sessions with the match! #{JSON.stringify err}"
-
-            originalSessionID = clientResponseObject.originalSessionID
-            originalSessionTeam = clientResponseObject.originalSessionTeam
-            originalSessionRank = parseInt clientResponseObject.originalSessionRank
-
-            determineIfSessionShouldContinueAndUpdateLog originalSessionID, originalSessionRank, (err, sessionShouldContinue) ->
-              if err? then return errors.serverError res, "There was an error determining if the session should continue, #{err}"
-
-              if sessionShouldContinue
-                opposingTeam = calculateOpposingTeam(originalSessionTeam)
-                opponentID = _.pull(_.keys(newScoresObject), originalSessionID)
-                sessionNewScore = newScoresObject[originalSessionID].totalScore
-                opponentNewScore = newScoresObject[opponentID].totalScore
-                findNearestBetterSessionID originalSessionID, sessionNewScore, opponentNewScore, opponentID ,opposingTeam, (err, opponentSessionID) ->
-                  if err? then return errors.serverError res, "There was an error finding the nearest sessionID!"
-                  unless opponentSessionID then return sendResponseObject req, res, {"message":"There were no more games to rank(game is at top!"}
-
-                  addPairwiseTaskToQueue [originalSessionID, opponentSessionID], (err, success) ->
-                    if err? then return errors.serverError res, "There was an error sending the pairwise tasks to the queue!"
-                    sendResponseObject req, res, {"message":"The scores were updated successfully and more games were sent to the queue!"}
-              else
-                console.log "Player lost, achieved rank #{originalSessionRank}"
-                sendResponseObject req, res, {"message":"The scores were updated successfully, person lost so no more games are being inserted!"}
+        
+      LevelSession.findOne(_id: clientResponseObject.originalSessionID).lean().exec (err, levelSession) ->
+        if err? then return errors.serverError res, "There was a problem finding the level session:#{err}"
+          
+        console.log "Queue message created at: #{taskLogJSON.createdAt}, level session submitted at #{levelSession.submitDate}"
+        
+        if taskLogJSON.createdAt <= levelSession.submitDate
+          console.log "Task has been resubmitted!"
+          return sendResponseObject req, res, {"message":"The game has been resubmitted. Removing from queue..."}
+  
+        logTaskComputation clientResponseObject, taskLog, (logErr) ->
+          if logErr? then return errors.serverError res, "There as a problem logging the task computation: #{logErr}"
+  
+          updateSessions clientResponseObject, (updateError, newScoreArray) ->
+            if updateError? then return errors.serverError res, "There was an error updating the scores.#{updateError}"
+  
+            newScoresObject = _.indexBy newScoreArray, 'id'
+  
+            addMatchToSessions clientResponseObject, newScoresObject, (err, data) ->
+              if err? then return errors.serverError res, "There was an error updating the sessions with the match! #{JSON.stringify err}"
+  
+              originalSessionID = clientResponseObject.originalSessionID
+              originalSessionTeam = clientResponseObject.originalSessionTeam
+              originalSessionRank = parseInt clientResponseObject.originalSessionRank
+  
+              determineIfSessionShouldContinueAndUpdateLog originalSessionID, originalSessionRank, (err, sessionShouldContinue) ->
+                if err? then return errors.serverError res, "There was an error determining if the session should continue, #{err}"
+  
+                if sessionShouldContinue
+                  opposingTeam = calculateOpposingTeam(originalSessionTeam)
+                  opponentID = _.pull(_.keys(newScoresObject), originalSessionID)
+                  sessionNewScore = newScoresObject[originalSessionID].totalScore
+                  opponentNewScore = newScoresObject[opponentID].totalScore
+                  findNearestBetterSessionID originalSessionID, sessionNewScore, opponentNewScore, opponentID ,opposingTeam, (err, opponentSessionID) ->
+                    if err? then return errors.serverError res, "There was an error finding the nearest sessionID!"
+                    unless opponentSessionID then return sendResponseObject req, res, {"message":"There were no more games to rank(game is at top!"}
+  
+                    addPairwiseTaskToQueue [originalSessionID, opponentSessionID], (err, success) ->
+                      if err? then return errors.serverError res, "There was an error sending the pairwise tasks to the queue!"
+                      sendResponseObject req, res, {"message":"The scores were updated successfully and more games were sent to the queue!"}
+                else
+                  console.log "Player lost, achieved rank #{originalSessionRank}"
+                  sendResponseObject req, res, {"message":"The scores were updated successfully, person lost so no more games are being inserted!"}
 
 
 determineIfSessionShouldContinueAndUpdateLog = (sessionID, sessionRank, cb) ->

From 239ca83a84718f93a6ed7d28d1a3341211e10803 Mon Sep 17 00:00:00 2001
From: Michael Schmatz <schmatz@umich.edu>
Date: Sun, 2 Mar 2014 19:11:42 -0800
Subject: [PATCH 027/178] Perhaps buggy fix to scheduling

---
 server/queues/scoring.coffee | 9 +++++----
 1 file changed, 5 insertions(+), 4 deletions(-)

diff --git a/server/queues/scoring.coffee b/server/queues/scoring.coffee
index eb0815e38..5aa252679 100644
--- a/server/queues/scoring.coffee
+++ b/server/queues/scoring.coffee
@@ -84,7 +84,7 @@ module.exports.dispatchTaskToConsumer = (req, res) ->
       message.changeMessageVisibilityTimeout scoringTaskTimeoutInSeconds, (err) ->
         if err? then return errors.serverError res, "There was an error changing the message visibility timeout."
         console.log "Changed visibility timeout"
-        constructTaskLogObject getUserIDFromRequest(req),message.getReceiptHandle(), (taskLogError, taskLogObject) ->
+        constructTaskLogObject getUserIDFromRequest(req),messageBody.registrationTime, message.getReceiptHandle(), (taskLogError, taskLogObject) ->
           if taskLogError? then return errors.serverError res, "There was an error creating the task log object."
 
           taskObject.taskID = taskLogObject._id
@@ -113,7 +113,7 @@ module.exports.processTaskResult = (req, res) ->
           
         console.log "Queue message created at: #{taskLogJSON.createdAt}, level session submitted at #{levelSession.submitDate}"
         
-        if taskLogJSON.createdAt <= levelSession.submitDate
+        if taskLogJSON.registrationTime <= levelSession.submitDate
           console.log "Task has been resubmitted!"
           return sendResponseObject req, res, {"message":"The game has been resubmitted. Removing from queue..."}
   
@@ -333,7 +333,7 @@ generateTaskPairs = (submittedSessions, sessionToScore) ->
   return taskPairs
 
 sendTaskPairToQueue = (taskPair, callback) ->
-  scoringTaskQueue.sendMessage {sessions: taskPair}, 0, (err,data) -> callback? err,data
+  scoringTaskQueue.sendMessage {sessions: taskPair, registrationTime:new Date()}, 0, (err,data) -> callback? err,data
 
 getUserIDFromRequest = (req) -> if req.user? then return req.user._id else return null
 
@@ -380,9 +380,10 @@ getSessionInformation = (sessionIDString, callback) ->
     callback err, sessionInformation
 
 
-constructTaskLogObject = (calculatorUserID, messageIdentifierString, callback) ->
+constructTaskLogObject = (calculatorUserID, registrationTime, messageIdentifierString, callback) ->
   taskLogObject = new TaskLog
     "createdAt": new Date()
+    "registrationTime": registrationTime
     "calculator":calculatorUserID
     "sentDate": Date.now()
     "messageIdentifierString":messageIdentifierString

From b8b22bb739c2e36a78c2073292bb5c5633e18581 Mon Sep 17 00:00:00 2001
From: Ruben Vereecken <ruben.vereecken@student.uantwerpen.be>
Date: Mon, 3 Mar 2014 12:45:04 +0100
Subject: [PATCH 028/178] Localized 'Create New ..' and 'Search .. Here'

---
 app/locale/en.coffee              | 6 ++++++
 app/locale/nl.coffee              | 6 ++++++
 app/templates/kinds/search.jade   | 5 +++--
 app/views/kinds/SearchView.coffee | 8 +++++++-
 4 files changed, 22 insertions(+), 3 deletions(-)

diff --git a/app/locale/en.coffee b/app/locale/en.coffee
index d72b872cd..2afc376cd 100644
--- a/app/locale/en.coffee
+++ b/app/locale/en.coffee
@@ -247,6 +247,12 @@ module.exports = nativeDescription: "English", englishDescription: "English", tr
     create_system_title: "Create New System"
     new_component_title: "Create New Component"
     new_component_field_system: "System"
+    new_article_title: "Create a New Article"
+    new_thang_title: "Create a New Thang Type"
+    new_level_title: "Create a New Level"
+    article_search_title: "Search Articles Here"
+    thang_search_title: "Search Thang Types Here"
+    level_search_title: "Search Levels Here"
 
   article:
     edit_btn_preview: "Preview"
diff --git a/app/locale/nl.coffee b/app/locale/nl.coffee
index 8e762eaeb..929741d23 100644
--- a/app/locale/nl.coffee
+++ b/app/locale/nl.coffee
@@ -247,6 +247,12 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     create_system_title: "Maak een nieuw Systeem aan"
     new_component_title: "Maak een nieuw Component aan"
     new_component_field_system: "Systeem"
+    new_article_title: "Maak een Nieuw Artikel"
+    new_thang_title: "Maak een Nieuw Thang Type"
+    new_level_title: "Maak een Nieuw Level"
+    article_search_title: "Zoek Artikels Hier"
+    thang_search_title: "Zoek Thang Types Hier"
+    level_search_title: "Zoek Levels Hier"
 
   article:
     edit_btn_preview: "Voorbeeld"
diff --git a/app/templates/kinds/search.jade b/app/templates/kinds/search.jade
index 24547e166..eda4c0f2b 100644
--- a/app/templates/kinds/search.jade
+++ b/app/templates/kinds/search.jade
@@ -10,8 +10,9 @@ block content
   if me.get('anonymous')
     a.btn.btn-primary.open-modal-button(data-toggle="coco-modal", data-target="modal/signup", role="button") Sign Up to Create a New #{modelLabel}
   else
-    a.btn.btn-primary.open-modal-button(href='#new-model-modal', role="button", data-toggle="modal") Create a New #{modelLabel}
-  input#search(placeholder="Search #{modelLabel}s Here")
+    a.btn.btn-primary.open-modal-button(href='#new-model-modal', role="button", data-toggle="modal" data-i18n="#{currentNew}") Create a New Something
+
+  input#search(data-i18n="[placeholder]#{currentSearch}")
   hr
   div.results
     table
diff --git a/app/views/kinds/SearchView.coffee b/app/views/kinds/SearchView.coffee
index 18648e386..1198f74a1 100644
--- a/app/views/kinds/SearchView.coffee
+++ b/app/views/kinds/SearchView.coffee
@@ -28,14 +28,20 @@ module.exports = class ThangTypeHomeView extends View
 
   getRenderData: ->
     context = super()
-    context.modelLabel = @modelLabel
     switch @modelLabel
       when 'Level'
         context.currentEditor = 'editor.level_title'
+        context.currentNew = 'editor.new_level_title'
+        context.currentSearch = 'editor.level_search_title'
       when 'Thang Type'
         context.currentEditor = 'editor.thang_title'
+        context.currentNew = 'editor.new_thang_title'
+        context.currentSearch = 'editor.thang_search_title'
       when 'Article'
         context.currentEditor = 'editor.article_title'
+        context.currentNew = 'editor.new_article_title'
+        context.currentSearch = 'editor.article_search_title'
+    @$el.i18n()
     context
 
   constructor: (options) ->

From 86fd09815930e992303fef4d7711bb7648b9959c Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 3 Mar 2014 07:28:40 -0800
Subject: [PATCH 029/178] Localized WizardSprite movement hey handling to
 WizardSprite file.

---
 app/lib/surface/WizardSprite.coffee       | 25 +++++++++++++++++++----
 app/views/play/level/playback_view.coffee | 15 --------------
 2 files changed, 21 insertions(+), 19 deletions(-)

diff --git a/app/lib/surface/WizardSprite.coffee b/app/lib/surface/WizardSprite.coffee
index b98ae44b5..58eec1dc7 100644
--- a/app/lib/surface/WizardSprite.coffee
+++ b/app/lib/surface/WizardSprite.coffee
@@ -21,7 +21,12 @@ module.exports = class WizardSprite extends IndieSprite
     'surface:sprite-selected': 'onSpriteSelected'
     'echo-self-wizard-sprite': 'onEchoSelfWizardSprite'
     'echo-all-wizard-sprites': 'onEchoAllWizardSprites'
-    'self-wizard:move': 'moveWizard'
+
+  shortcuts:
+    'up': 'onMoveKey'
+    'down': 'onMoveKey'
+    'left': 'onMoveKey'
+    'right': 'onMoveKey'
 
   constructor: (thangType, options) ->
     if options?.isSelf
@@ -230,9 +235,21 @@ module.exports = class WizardSprite extends IndieSprite
   updateMarks: ->
     super() if @displayObject.visible  # not if we hid the wiz
 
-  moveWizard : (x, y) =>
+
+  onMoveKey: (e) ->
+    return unless @isSelf
+    e?.preventDefault()
+    yMovement = 0
+    xMovement = 0
+    yMovement += 2 if key.isPressed('up')
+    yMovement -= 2 if key.isPressed('down')
+    xMovement += 2 if key.isPressed('right')
+    xMovement -= 2 if key.isPressed('left')
+    @moveWizard xMovement, yMovement
+
+  moveWizard: (x, y) ->
     interval = 500
-    position = {x: @targetPos.x+x, y: @targetPos.y+y}
+    position = {x: @targetPos.x + x, y: @targetPos.y + y}
     @setTarget(position, interval, true)
     @updatePosition()
-    Backbone.Mediator.publish 'camera-zoom-to', position, interval
\ No newline at end of file
+    Backbone.Mediator.publish 'camera-zoom-to', position, interval
diff --git a/app/views/play/level/playback_view.coffee b/app/views/play/level/playback_view.coffee
index 20ec9295c..f9c13a9e3 100644
--- a/app/views/play/level/playback_view.coffee
+++ b/app/views/play/level/playback_view.coffee
@@ -36,10 +36,6 @@ module.exports = class PlaybackView extends View
     '⌘+p, p, ctrl+p': 'onTogglePlay'
     '⌘+[, ctrl+[': 'onScrubBack'
     '⌘+], ctrl+]': 'onScrubForward'
-    'up': 'onMoveKey'
-    'down': 'onMoveKey'
-    'left': 'onMoveKey'
-    'right': 'onMoveKey'
 
   constructor: ->
     super(arguments...)
@@ -219,14 +215,3 @@ module.exports = class PlaybackView extends View
     $(window).off('resize', @onWindowResize)
     @onWindowResize = null
     super()
-
-  onMoveKey: (e) ->
-    e?.preventDefault()
-    yMovement = 0
-    xMovement = 0
-    yMovement += 2 if key.isPressed('up')
-    yMovement -= 2 if key.isPressed('down')
-    xMovement += 2 if key.isPressed('right')
-    xMovement -= 2 if key.isPressed('left')
-    console.log 'onMoveKey', xMovement, yMovement
-    Backbone.Mediator.publish 'self-wizard:move', xMovement, yMovement

From 357ef45cb09d31178b49ddd47d7d74b96b6c117a Mon Sep 17 00:00:00 2001
From: Karthig <karthig@karthig-HP-Mini-5101.(none)>
Date: Sun, 2 Mar 2014 22:33:02 -0500
Subject: [PATCH 030/178] - insert random error image into 404 page

---
 app/templates/not_found.jade | 5 +++++
 1 file changed, 5 insertions(+)

diff --git a/app/templates/not_found.jade b/app/templates/not_found.jade
index ab9d5809e..fd838b2c2 100644
--- a/app/templates/not_found.jade
+++ b/app/templates/not_found.jade
@@ -3,3 +3,8 @@ extends /templates/base
 block content
 
   h1.text-center(data-i18n="not_found.page_not_found") Page Not Found
+
+  num = Math.floor(Math.random() * 3) + 1
+
+  img(src="/images/pages/not_found/404_#{num}.png")
+

From a5adf5a1bf35fa6d917694b3da163e1e10b1fd82 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 3 Mar 2014 08:10:36 -0800
Subject: [PATCH 031/178] Tweaked status colors for production logging;
 applying it only when in production.

---
 server_setup.coffee | 22 ++++++++++++++--------
 1 file changed, 14 insertions(+), 8 deletions(-)

diff --git a/server_setup.coffee b/server_setup.coffee
index 3332e0428..9d00188a7 100644
--- a/server_setup.coffee
+++ b/server_setup.coffee
@@ -21,14 +21,23 @@ setupRequestTimeoutMiddleware = (app) ->
 
 productionLogging = (tokens, req, res)->
   status = res.statusCode
-  color = 31
-  if(status != 200 && status != 304)
-    return '\x1b[90m' + req.method+ ' ' + req.originalUrl + ' '+ '\x1b[' + color + 'm' + res.statusCode+ ' \x1b[90m'+ (new Date - req._startTime)+ 'ms' + '\x1b[0m';
+  color = 32
+  if status >= 500 then color = 31
+  else if status >= 400 then color = 33
+  else if status >= 300 then color = 36
+  elapsed = (new Date()) - req._startTime
+  elapsedColor = if elapsed < 500 then 90 else 31
+  if (status isnt 200 and status isnt 304) or elapsed > 500
+    return "\x1b[90m#{req.method} #{req.originalUrl} \x1b[#{color}m#{res.statusCode} \x1b[#{elapsedColor}m#{elapsed}ms\x1b[0m"
+  null
 
 setupExpressMiddleware = (app) ->
   setupRequestTimeoutMiddleware app
-  express.logger.format('prod', productionLogging)
-  app.use(express.logger('prod'))
+  if config.isProduction
+    express.logger.format('prod', productionLogging)
+    app.use(express.logger('prod'))
+  else
+    app.use(express.logger('dev'))
   app.use(express.static(path.join(__dirname, 'public')))
   app.use(useragent.express())
 
@@ -105,6 +114,3 @@ exports.setExpressConfigurationOptions = (app) ->
   app.set('views', __dirname + '/app/views')
   app.set('view engine', 'jade')
   app.set('view options', { layout: false })
-
-
-

From 967d6cd8890c9baf6ca3eaac51a2406aff8ae2e5 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 3 Mar 2014 08:13:53 -0800
Subject: [PATCH 032/178] Let's try this without the request timeout
 middleware, to see if we still need that.

---
 server_setup.coffee | 17 +++++++++--------
 1 file changed, 9 insertions(+), 8 deletions(-)

diff --git a/server_setup.coffee b/server_setup.coffee
index 9d00188a7..ad23a6bf1 100644
--- a/server_setup.coffee
+++ b/server_setup.coffee
@@ -11,13 +11,14 @@ logging = require './server/commons/logging'
 config = require './server_config'
 
 ###Middleware setup functions implementation###
-setupRequestTimeoutMiddleware = (app) ->
-  app.use (req, res, next) ->
-    req.setTimeout 15000, ->
-      console.log 'timed out!'
-      req.abort()
-      self.emit('pass',message)
-    next()
+# 2014-03-03: Try not using this and see if it's still a problem
+#setupRequestTimeoutMiddleware = (app) ->
+#  app.use (req, res, next) ->
+#    req.setTimeout 15000, ->
+#      console.log 'timed out!'
+#      req.abort()
+#      self.emit('pass',message)
+#    next()
 
 productionLogging = (tokens, req, res)->
   status = res.statusCode
@@ -32,7 +33,7 @@ productionLogging = (tokens, req, res)->
   null
 
 setupExpressMiddleware = (app) ->
-  setupRequestTimeoutMiddleware app
+  #setupRequestTimeoutMiddleware app
   if config.isProduction
     express.logger.format('prod', productionLogging)
     app.use(express.logger('prod'))

From 9a40eb623c6537296679aaf561650c6ba43cbaed Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 3 Mar 2014 08:21:15 -0800
Subject: [PATCH 033/178] Added translation by @rdxiang of server_config.js to
 server_config.coffee

---
 server_config.coffee | 49 +++++++++++++++++++++++++++++++++++++++++
 server_config.js     | 52 --------------------------------------------
 2 files changed, 49 insertions(+), 52 deletions(-)
 create mode 100644 server_config.coffee
 delete mode 100644 server_config.js

diff --git a/server_config.coffee b/server_config.coffee
new file mode 100644
index 000000000..b13bead19
--- /dev/null
+++ b/server_config.coffee
@@ -0,0 +1,49 @@
+config = {}
+
+config.unittest = process.argv.indexOf("--unittest") > -1
+
+config.port = process.env.COCO_PORT or process.env.COCO_NODE_PORT or 3000
+config.ssl_port = process.env.COCO_SSL_PORT or process.env.COCO_SSL_NODE_PORT or 3443
+
+config.mongo =
+  port: process.env.COCO_MONGO_PORT or 27017
+  host: process.env.COCO_MONGO_HOST or "localhost"
+  db: process.env.COCO_MONGO_DATABASE_NAME or "coco"
+  mongoose_replica_string: process.env.COCO_MONGO_MONGOOSE_REPLICA_STRING or ""
+
+if config.unittest
+  config.port += 1
+  config.ssl_port += 1
+  config.mongo.host = "localhost"
+else
+  config.mongo.username = process.env.COCO_MONGO_USERNAME or ""
+  config.mongo.password = process.env.COCO_MONGO_PASSWORD or ""
+
+config.mail =
+  service: process.env.COCO_MAIL_SERVICE_NAME or "Zoho"
+  username: process.env.COCO_MAIL_SERVICE_USERNAME or ""
+  password: process.env.COCO_MAIL_SERVICE_PASSWORD or ""
+  mailchimpAPIKey: process.env.COCO_MAILCHIMP_API_KEY or ""
+  mailchimpWebhook: process.env.COCO_MAILCHIMP_WEBHOOK or "/mail/webhook"
+  sendwithusAPIKey: process.env.COCO_SENDWITHUS_API_KEY or ""
+
+config.queue =
+  accessKeyId: process.env.COCO_AWS_ACCESS_KEY_ID or ""
+  secretAccessKey: process.env.COCO_AWS_SECRET_ACCESS_KEY or ""
+  region: "us-east-1"
+  simulationQueueName: "simulationQueue"
+
+config.mongoQueue =
+  queueDatabaseName: "coco_queue"
+
+config.salt = process.env.COCO_SALT or "pepper"
+config.cookie_secret = process.env.COCO_COOKIE_SECRET or "chips ahoy"
+
+config.isProduction = config.mongo.host isnt "localhost"
+
+if not config.unittest and  not config.isProduction
+  # change artificially slow down non-static requests for testing
+  config.slow_down = false
+
+
+module.exports = config
diff --git a/server_config.js b/server_config.js
deleted file mode 100644
index 880fcfd78..000000000
--- a/server_config.js
+++ /dev/null
@@ -1,52 +0,0 @@
-var config = {};
-
-config.unittest = process.argv.indexOf('--unittest') > -1;
-
-config.port = process.env.COCO_PORT || process.env.COCO_NODE_PORT || 3000;
-config.ssl_port = 
-  process.env.COCO_SSL_PORT || process.env.COCO_SSL_NODE_PORT || 3443;
-
-config.mongo = {};
-config.mongo.port = process.env.COCO_MONGO_PORT || 27017;
-config.mongo.host = process.env.COCO_MONGO_HOST || 'localhost';
-config.mongo.db = process.env.COCO_MONGO_DATABASE_NAME || 'coco';
-config.mongo.mongoose_replica_string = process.env.COCO_MONGO_MONGOOSE_REPLICA_STRING || '';
-
-if(config.unittest) {
-  config.port += 1;
-  config.ssl_port += 1;
-  config.mongo.host = 'localhost';
-}
-
-else {
-  config.mongo.username = process.env.COCO_MONGO_USERNAME || '';
-  config.mongo.password = process.env.COCO_MONGO_PASSWORD || '';
-}
-
-config.mail = {};
-config.mail.service = process.env.COCO_MAIL_SERVICE_NAME || "Zoho";
-config.mail.username = process.env.COCO_MAIL_SERVICE_USERNAME || "";
-config.mail.password = process.env.COCO_MAIL_SERVICE_PASSWORD || "";
-config.mail.mailchimpAPIKey = process.env.COCO_MAILCHIMP_API_KEY || '';
-config.mail.mailchimpWebhook = process.env.COCO_MAILCHIMP_WEBHOOK || '/mail/webhook';
-config.mail.sendwithusAPIKey = process.env.COCO_SENDWITHUS_API_KEY || '';
-
-config.queue = {};
-config.queue.accessKeyId = process.env.COCO_AWS_ACCESS_KEY_ID || '';
-config.queue.secretAccessKey = process.env.COCO_AWS_SECRET_ACCESS_KEY || '';
-config.queue.region = 'us-east-1';
-config.queue.simulationQueueName = "simulationQueue";
-config.mongoQueue = {};
-config.mongoQueue.queueDatabaseName = "coco_queue";
-
-config.salt = process.env.COCO_SALT || 'pepper';
-config.cookie_secret = process.env.COCO_COOKIE_SECRET || 'chips ahoy';
-
-config.isProduction = config.mongo.host != 'localhost';
-
-if(!config.unittest && !config.isProduction) {
-  // change artificially slow down non-static requests for testing
-  config.slow_down = false; 
-}
-
-module.exports = config;

From b04d63d5bf41345d2ddc8926fc83f33663c7f1d4 Mon Sep 17 00:00:00 2001
From: Karthig <karthig@karthig-HP-Mini-5101.(none)>
Date: Mon, 3 Mar 2014 11:42:07 -0500
Subject: [PATCH 034/178] - create sass file to center 404 image on page

---
 app/styles/not_found.sass    | 9 +++++++++
 app/templates/not_found.jade | 2 +-
 2 files changed, 10 insertions(+), 1 deletion(-)
 create mode 100644 app/styles/not_found.sass

diff --git a/app/styles/not_found.sass b/app/styles/not_found.sass
new file mode 100644
index 000000000..677900b6d
--- /dev/null
+++ b/app/styles/not_found.sass
@@ -0,0 +1,9 @@
+@import "bootstrap/mixins"
+@import "bootstrap/variables"
+
+#not-found-view
+
+  .not-found-image
+    display: block
+    margin-left: auto
+    margin-right: auto
diff --git a/app/templates/not_found.jade b/app/templates/not_found.jade
index fd838b2c2..bc2d32d13 100644
--- a/app/templates/not_found.jade
+++ b/app/templates/not_found.jade
@@ -6,5 +6,5 @@ block content
 
   num = Math.floor(Math.random() * 3) + 1
 
-  img(src="/images/pages/not_found/404_#{num}.png")
+  img(src="/images/pages/not_found/404_#{num}.png" class="not-found-image")
 

From 8399667b49bdd0241c1b653cbfcd0cd57d404b7b Mon Sep 17 00:00:00 2001
From: Michael Schmatz <schmatz@umich.edu>
Date: Mon, 3 Mar 2014 08:47:09 -0800
Subject: [PATCH 035/178] Fixed submit date checks

---
 server/queues/scoring.coffee | 14 ++++++--------
 1 file changed, 6 insertions(+), 8 deletions(-)

diff --git a/server/queues/scoring.coffee b/server/queues/scoring.coffee
index 5aa252679..c527f921f 100644
--- a/server/queues/scoring.coffee
+++ b/server/queues/scoring.coffee
@@ -84,7 +84,7 @@ module.exports.dispatchTaskToConsumer = (req, res) ->
       message.changeMessageVisibilityTimeout scoringTaskTimeoutInSeconds, (err) ->
         if err? then return errors.serverError res, "There was an error changing the message visibility timeout."
         console.log "Changed visibility timeout"
-        constructTaskLogObject getUserIDFromRequest(req),messageBody.registrationTime, message.getReceiptHandle(), (taskLogError, taskLogObject) ->
+        constructTaskLogObject getUserIDFromRequest(req), message.getReceiptHandle(), (taskLogError, taskLogObject) ->
           if taskLogError? then return errors.serverError res, "There was an error creating the task log object."
 
           taskObject.taskID = taskLogObject._id
@@ -111,10 +111,9 @@ module.exports.processTaskResult = (req, res) ->
       LevelSession.findOne(_id: clientResponseObject.originalSessionID).lean().exec (err, levelSession) ->
         if err? then return errors.serverError res, "There was a problem finding the level session:#{err}"
           
-        console.log "Queue message created at: #{taskLogJSON.createdAt}, level session submitted at #{levelSession.submitDate}"
-        
-        if taskLogJSON.registrationTime <= levelSession.submitDate
-          console.log "Task has been resubmitted!"
+        supposedSubmissionDate = new Date(clientResponseObject.sessions[0].submitDate)
+          
+        if Number(supposedSubmissionDate) isnt Number(levelSession.submitDate)
           return sendResponseObject req, res, {"message":"The game has been resubmitted. Removing from queue..."}
   
         logTaskComputation clientResponseObject, taskLog, (logErr) ->
@@ -333,7 +332,7 @@ generateTaskPairs = (submittedSessions, sessionToScore) ->
   return taskPairs
 
 sendTaskPairToQueue = (taskPair, callback) ->
-  scoringTaskQueue.sendMessage {sessions: taskPair, registrationTime:new Date()}, 0, (err,data) -> callback? err,data
+  scoringTaskQueue.sendMessage {sessions: taskPair}, 0, (err,data) -> callback? err,data
 
 getUserIDFromRequest = (req) -> if req.user? then return req.user._id else return null
 
@@ -380,10 +379,9 @@ getSessionInformation = (sessionIDString, callback) ->
     callback err, sessionInformation
 
 
-constructTaskLogObject = (calculatorUserID, registrationTime, messageIdentifierString, callback) ->
+constructTaskLogObject = (calculatorUserID, messageIdentifierString, callback) ->
   taskLogObject = new TaskLog
     "createdAt": new Date()
-    "registrationTime": registrationTime
     "calculator":calculatorUserID
     "sentDate": Date.now()
     "messageIdentifierString":messageIdentifierString

From 22e80f6d24737b6ee19674deb7c43e6bf037a2f7 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Mon, 3 Mar 2014 09:03:44 -0800
Subject: [PATCH 036/178] Set up a play modal, starting to refactor wizard
 thang type loading to be shared across the site.

---
 app/models/ThangType.coffee               |   7 ++
 app/styles/play/ladder/play_modal.sass    | 108 +++++++++++++++++
 app/templates/play/ladder.jade            |   2 +-
 app/templates/play/ladder/play_modal.jade |  64 ++++++++++
 app/views/home_view.coffee                |  10 +-
 app/views/play/ladder/play_modal.coffee   | 140 ++++++++++++++++++++++
 app/views/play/ladder_view.coffee         |  11 +-
 7 files changed, 333 insertions(+), 9 deletions(-)
 create mode 100644 app/styles/play/ladder/play_modal.sass
 create mode 100644 app/templates/play/ladder/play_modal.jade
 create mode 100644 app/views/play/ladder/play_modal.coffee

diff --git a/app/models/ThangType.coffee b/app/models/ThangType.coffee
index 6aaff80a5..c162f9287 100644
--- a/app/models/ThangType.coffee
+++ b/app/models/ThangType.coffee
@@ -210,3 +210,10 @@ module.exports = class ThangType extends CocoModel
 
   onFileUploaded: =>
     console.log 'Image uploaded'
+
+wizOriginal = "52a00d55cf1818f2be00000b"
+url = "/db/thang_type/#{wizOriginal}/version"
+wizardType = new module.exports()
+wizardType.url = -> url
+wizardType.fetch()
+module.exports.wizardType = wizardType
\ No newline at end of file
diff --git a/app/styles/play/ladder/play_modal.sass b/app/styles/play/ladder/play_modal.sass
new file mode 100644
index 000000000..8dc69a5ae
--- /dev/null
+++ b/app/styles/play/ladder/play_modal.sass
@@ -0,0 +1,108 @@
+#ladder-play-modal
+  .tutorial-suggestion
+    text-align: center
+    font-size: 18px
+  
+  .play-option
+    margin-bottom: 15px
+    width: 100%
+    height: 100px
+    overflow: hidden
+    background: white
+    border: 1px solid #333
+    position: relative
+
+    -webkit-transition: opacity 0.3s ease-in-out
+    -moz-transition: opacity 0.3s ease-in-out
+    -ms-transition: opacity 0.3s ease-in-out
+    -o-transition: opacity 0.3s ease-in-out
+    transition: opacity 0.3s ease-in-out
+    
+    opacity: 0.4
+    
+    border-radius: 5px
+    
+  .play-option:hover
+    opacity: 1
+    
+  .my-icon
+    position: relative
+    left: 0
+    top: -10px
+    
+  .opponent-icon
+    position: relative
+    float: right
+    right: 0
+    top: -10px
+    -moz-transform: scaleX(-1)
+    -o-transform: scaleX(-1)
+    -webkit-transform: scaleX(-1)
+    transform: scaleX(-1)
+    filter: FlipH
+    -ms-filter: "FlipH"
+    
+  .name-label
+    border-bottom: 20px solid lightslategray
+    height: 0
+    width: 40%
+    position: absolute
+    bottom: 0
+    color: black
+    font-weight: bold
+    text-align: center
+    
+    span
+      position: relative
+      top: 1px
+      
+  .my-name
+    border-right: 15px solid transparent
+    left: 0
+    span
+      left: 3px
+    
+  .opponent-name
+    border-left: 15px solid transparent
+    right: 0
+    //text-align: right
+    span
+      right: 3px
+  
+  .difficulty
+    border-top: 25px solid darkgray
+    border-left: 20px solid transparent
+    border-right: 20px solid transparent
+    height: 0
+    width: 30%
+    position: absolute
+    left: 35%
+    top: 0
+    color: black
+    text-align: center
+    font-size: 18px
+    font-weight: bold
+
+    span
+      position: relative
+      top: -25px
+      
+  .easy-option .difficulty
+    border-top: 25px solid limegreen
+      
+  .medium-option .difficulty
+    border-top: 25px solid darkorange
+    
+  .hard-option .difficulty
+    border-top: 25px solid black
+    color: white
+
+  .vs
+    position: absolute
+    left: 40%
+    right: 40%
+    text-align: center
+    top: 35px
+    font-size: 40px
+    font-weight: bolder
+    color: black
\ No newline at end of file
diff --git a/app/templates/play/ladder.jade b/app/templates/play/ladder.jade
index 4d056ce97..18802f55d 100644
--- a/app/templates/play/ladder.jade
+++ b/app/templates/play/ladder.jade
@@ -20,7 +20,7 @@ block content
       div.column.col-md-2
       for team in teams
         div.column.col-md-4
-          a(href="/play/ladder/#{levelID}/team/#{team.id}", style="background-color: #{team.primaryColor}").play-button.btn.btn-danger.btn-block.btn-lg
+          a(style="background-color: #{team.primaryColor}", data-team=team.id).play-button.btn.btn-danger.btn-block.btn-lg
             span Play As 
             span= team.name
       div.column.col-md-2
diff --git a/app/templates/play/ladder/play_modal.jade b/app/templates/play/ladder/play_modal.jade
new file mode 100644
index 000000000..6eae0c87c
--- /dev/null
+++ b/app/templates/play/ladder/play_modal.jade
@@ -0,0 +1,64 @@
+extends /templates/modal/modal_base
+
+block modal-header-content
+  h3 Choose an Opponent
+
+block modal-body-content
+  
+  p.tutorial-suggestion
+    span Not sure what's going on?
+    |  
+    a(href="/play/level/brawlwood-tutorial") Play the tutorial first.
+
+  a(href="/play/level/#{levelID}?team=#{teamID}")
+    div.play-option
+      img(src=portraitSRC).my-icon
+      img(src=portraitSRC).opponent-icon
+      div.my-name.name-label
+        span= myName
+      div.opponent-name.name-label
+        span Simple AI
+      div.difficulty
+        span Warmup
+      div.vs VS
+  
+  if challengers.easy
+    a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.easy.sessionID}")
+      div.play-option.easy-option
+        img(src=portraitSRC).my-icon
+        img(src=portraitSRC).opponent-icon
+        div.my-name.name-label
+          span= myName
+        div.opponent-name.name-label
+          span= challengers.easy.opponentName
+        div.difficulty
+          span Easy
+        div.vs VS
+
+  if challengers.medium
+    a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.medium.sessionID}")
+      div.play-option.medium-option
+        img(src=portraitSRC).my-icon
+        img(src=portraitSRC).opponent-icon
+        div.my-name.name-label
+          span= myName
+        div.opponent-name.name-label
+          span= challengers.medium.opponentName
+        div.difficulty
+          span Medium
+        div.vs VS
+
+  if challengers.hard
+    a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.hard.sessionID}")
+      div.play-option.hard-option
+        img(src=portraitSRC).my-icon
+        img(src=portraitSRC).opponent-icon
+        div.my-name.name-label
+          span= myName
+        div.opponent-name.name-label
+          span= challengers.hard.opponentName
+        div.difficulty
+          span Hard
+        div.vs VS
+
+block modal-footer
\ No newline at end of file
diff --git a/app/views/home_view.coffee b/app/views/home_view.coffee
index 7289b4ca9..a16c1a9ba 100644
--- a/app/views/home_view.coffee
+++ b/app/views/home_view.coffee
@@ -28,12 +28,8 @@ module.exports = class HomeView extends View
     @$el.find('.modal').on 'shown.bs.modal', ->
       $('input:visible:first', @).focus()
 
-    wizOriginal = "52a00d55cf1818f2be00000b"
-    url = "/db/thang_type/#{wizOriginal}/version"
-    @wizardType = new ThangType()
-    @wizardType.url = -> url
-    @wizardType.fetch()
-    @wizardType.once 'sync', @initCanvas
+    @wizardType = ThangType.wizardType
+    if @wizardType.loaded then @initCanvas else @wizardType.once 'sync', @initCanvas, @
 
     # Try to find latest level and set "Play" link to go to that level
     if localStorage?
@@ -48,7 +44,7 @@ module.exports = class HomeView extends View
     else
       console.log("TODO: Insert here code to get latest level played from the database. If this can't be found, we just let the user play the first level.")
 
-  initCanvas: =>
+  initCanvas: ->
     @stage = new createjs.Stage($('#beginner-campaign canvas', @$el)[0])
     @createWizard()
 
diff --git a/app/views/play/ladder/play_modal.coffee b/app/views/play/ladder/play_modal.coffee
new file mode 100644
index 000000000..560397d37
--- /dev/null
+++ b/app/views/play/ladder/play_modal.coffee
@@ -0,0 +1,140 @@
+View = require 'views/kinds/ModalView'
+template = require 'templates/play/ladder/play_modal'
+ThangType = require 'models/ThangType'
+{me} = require 'lib/auth'
+
+module.exports = class LadderPlayModal extends View
+  id: "ladder-play-modal"
+  template: template
+  closeButton: true
+  startsLoading = true
+
+  constructor: (options, @level, @session, @team) ->
+    super(options)
+    @nameMap = {}
+    @otherTeam = if team is 'ogres' then 'humans' else 'ogres'
+    @startLoadingChallengersMaybe()
+    @wizardType = ThangType.wizardType
+    
+  # PART 1: Load challengers from the db unless some are in the matches
+
+  startLoadingChallengersMaybe: ->
+    matches = @session.get('matches')
+    if matches?.length then @loadNames() else @loadChallengers()
+
+  loadChallengers: ->
+    @challengers = new ChallengersData(@level, @team, @otherTeam, @session)
+    @challengers.on 'sync', @loadNames, @
+
+  # PART 2: Loading the names of the other users
+
+  loadNames: ->
+    @challengers = @getChallengers()
+    ids = (challenger.opponentID for challenger in _.values @challengers)
+
+    success = (@nameMap) =>
+      for challenger in _.values(@challengers)
+        challenger.opponentName = @nameMap[challenger.opponentID] or 'Anoner'
+      @checkWizardLoaded()
+
+    $.ajax('/db/user/-/names', {
+      data: {ids: ids}
+      type: 'POST'
+      success: success
+    })
+    
+  # PART 3: Make sure wizard is loaded
+  
+  checkWizardLoaded: ->
+    if @wizardType.loaded then @finishRendering() else @wizardType.once 'sync', @finishRendering, @
+    
+  # PART 4: Render
+
+  finishRendering: ->
+    @startsLoading = false
+    @render()
+
+  getRenderData: ->
+    ctx = super()
+    ctx.level = @level
+    ctx.levelID = @level.get('slug') or @level.id
+    ctx.teamName = _.string.titleize @team
+    ctx.teamID = @team
+    ctx.otherTeamID = @otherTeam
+    ctx.challengers = if not @startsLoading then @challengers else {}
+    ctx.portraitSRC = @wizardType.getPortraitSource()
+    ctx.myName = me.get('name') || 'Newcomer'
+    ctx
+    
+  # Choosing challengers
+
+  getChallengers: ->
+    # make an object of challengers to everything needed to link to them
+    challengers = {}
+    if @challengers
+      easyInfo = @challengeInfoFromSession(@challengers.easyPlayer.models[0])
+      mediumInfo = @challengeInfoFromSession(@challengers.mediumPlayer.models[0])
+      hardInfo = @challengeInfoFromSession(@challengers.hardPlayer.models[0])
+    else
+      matches = @session.get('matches')
+      won = (m for m in matches when m.metrics.rank < m.opponents[0].metrics.rank)
+      lost = (m for m in matches when m.metrics.rank > m.opponents[0].metrics.rank)
+      tied = (m for m in matches when m.metrics.rank is m.opponents[0].metrics.rank)
+      easyInfo = @challengeInfoFromMatches(won)
+      mediumInfo = @challengeInfoFromMatches(tied)
+      hardInfo = @challengeInfoFromMatches(lost)
+    @addChallenger easyInfo, challengers, 'easy'
+    @addChallenger mediumInfo, challengers, 'medium'
+    @addChallenger hardInfo, challengers, 'hard'
+    challengers
+
+  addChallenger: (info, challengers, title) ->
+    # check for duplicates first
+    return unless info
+    for key, value of challengers
+      return if value.sessionID is info.sessionID
+    challengers[title] = info
+
+  challengeInfoFromSession: (session) ->
+    # given a model from the db, return info needed for a link to the match
+    return unless session
+    return {
+      sessionID: session.id
+      opponentID: session.get('creator')
+    }
+
+  challengeInfoFromMatches: (matches) ->
+    return unless matches?.length
+    match = _.sample matches
+    opponent = match.opponents[0]
+    return {
+      sessionID: opponent.sessionID
+      opponentID: opponent.userID
+    }
+
+
+class ChallengersData
+  constructor: (@level, @team, @otherTeam, @session) ->
+    _.extend @, Backbone.Events
+    score = @session?.get('totalScore') or 25
+    @easyPlayer = new LeaderboardCollection(@level, {order:1, scoreOffset: score - 5, limit: 1, team: @otherTeam})
+    @easyPlayer.fetch()
+    @easyPlayer.once 'sync', @challengerLoaded, @
+    @mediumPlayer = new LeaderboardCollection(@level, {order:1, scoreOffset: score, limit: 1, team: @otherTeam})
+    @mediumPlayer.fetch()
+    @mediumPlayer.once 'sync', @challengerLoaded, @
+    @hardPlayer = new LeaderboardCollection(@level, {order:-1, scoreOffset: score + 5, limit: 1, team: @otherTeam})
+    @hardPlayer.fetch()
+    @hardPlayer.once 'sync', @challengerLoaded, @
+
+  challengerLoaded: ->
+    if @allLoaded()
+      @loaded = true
+      @trigger 'sync'
+
+  playerIDs: ->
+    collections = [@easyPlayer, @mediumPlayer, @hardPlayer]
+    (c.models[0].get('creator') for c in collections when c?.models[0])
+
+  allLoaded: ->
+    _.all [@easyPlayer.loaded, @mediumPlayer.loaded, @hardPlayer.loaded]
diff --git a/app/views/play/ladder_view.coffee b/app/views/play/ladder_view.coffee
index 25d15f484..b1d74b182 100644
--- a/app/views/play/ladder_view.coffee
+++ b/app/views/play/ladder_view.coffee
@@ -3,10 +3,11 @@ Level = require 'models/Level'
 Simulator = require 'lib/simulator/Simulator'
 LevelSession = require 'models/LevelSession'
 CocoCollection = require 'models/CocoCollection'
-LeaderboardCollection  = require 'collections/LeaderboardCollection'
 {teamDataFromLevel} = require './ladder/utils'
+
 LadderTabView = require './ladder/ladder_tab'
 MyMatchesTabView = require './ladder/my_matches_tab'
+LadderPlayModal = require './ladder/play_modal'
 
 HIGHEST_SCORE = 1000000
 
@@ -26,6 +27,7 @@ module.exports = class LadderView extends RootView
   events:
     'click #simulate-button': 'onSimulateButtonClick'
     'click #simulate-all-button': 'onSimulateAllButtonClick'
+    'click .play-button': 'onClickPlayButton'
 
   constructor: (options, @levelID) ->
     super(options)
@@ -97,3 +99,10 @@ module.exports = class LadderView extends RootView
     catch e
       console.log "There was a problem with the named simulation status: #{e}"
     $("#simulation-status-text").text @simulationStatus
+
+  onClickPlayButton: (e) ->
+    button = $(e.target).closest('.play-button')
+    teamID = button.data('team')
+    session = (s for s in @sessions.models when s.get('team') is teamID)[0]
+    modal = new LadderPlayModal({}, @level, session, teamID)
+    @openModalView modal
\ No newline at end of file

From d3b6836a06386c882890d4b266ea1734f45ec61b Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Mon, 3 Mar 2014 09:20:14 -0800
Subject: [PATCH 037/178] Fixed some bugs with the new ladder view when you
 haven't played before.

---
 app/templates/play/ladder/my_matches_tab.jade | 65 ++++++++++---------
 app/views/play/ladder/my_matches_tab.coffee   |  8 +--
 app/views/play/ladder/play_modal.coffee       |  5 +-
 3 files changed, 41 insertions(+), 37 deletions(-)

diff --git a/app/templates/play/ladder/my_matches_tab.jade b/app/templates/play/ladder/my_matches_tab.jade
index b9af08530..310beca34 100644
--- a/app/templates/play/ladder/my_matches_tab.jade
+++ b/app/templates/play/ladder/my_matches_tab.jade
@@ -7,17 +7,17 @@
 div#columns.row
   for team in teams
     div.matches-column.col-md-6
-      if team.matches.length
-        table.table.table-bordered.table-condensed
+      table.table.table-bordered.table-condensed
 
-          tr
-            th(colspan=4, style="color: #{team.primaryColor}")
-              span Your
-              span  
-              span= team.name
-              span  
-              span Matches
+        tr
+          th(colspan=4, style="color: #{team.primaryColor}")
+            span Your
+            span  
+            span= team.name
+            span  
+            span Matches
 
+            if team.session
               button.btn.btn-sm.btn-warning.pull-right.rank-button(data-session-id=team.session.id)
                 span.unavailable.hidden No New Code to Rank
                 span.rank.hidden Rank My Game!
@@ -25,28 +25,29 @@ div#columns.row
                 span.ranked.hidden Submitted for Ranking
                 span.failed.hidden Failed to Rank
 
+        tr
+          th Result
+          th Opponent
+          th When
+          th
+        for match in team.matches
           tr
-            th Result
-            th Opponent
-            th When
-            th
-          for match in team.matches
-            tr
-              td.state-cell
-                if match.state === 'win'
-                  span.win Win
-                if match.state === 'loss'
-                  span.loss Loss
-                if match.state === 'tie'
-                  span.tie Tie
-              td.name-cell= match.opponentName || "Anonymous"
-              td.time-cell= match.when
-              td.battle-cell
-                - var text = match.state === 'win' ? 'Watch your victory' : 'Defeat the ' + team.otherTeam
-                a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{match.sessionID}")= text
-      
-      else
-        div.alert.alert-warning
-          | No ranked matches played yet!
-          | Play some competitors on the right and then come back to get your game ranked.
+            td.state-cell
+              if match.state === 'win'
+                span.win Win
+              if match.state === 'loss'
+                span.loss Loss
+              if match.state === 'tie'
+                span.tie Tie
+            td.name-cell= match.opponentName || "Anonymous"
+            td.time-cell= match.when
+            td.battle-cell
+              - var text = match.state === 'win' ? 'Watch your victory' : 'Defeat the ' + team.otherTeam
+              a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{match.sessionID}")= text
+
+        if !team.matches.length
+          tr
+            td(colspan=4).alert.alert-warning
+              | No ranked matches for this team!
+              | Play against some competitors and then come back here to get your game ranked.
       
diff --git a/app/views/play/ladder/my_matches_tab.coffee b/app/views/play/ladder/my_matches_tab.coffee
index 110a289db..f4994d519 100644
--- a/app/views/play/ladder/my_matches_tab.coffee
+++ b/app/views/play/ladder/my_matches_tab.coffee
@@ -62,9 +62,9 @@ module.exports = class MyMatchesTabView extends CocoView
     for team in @teams
       team.session = (s for s in @sessions.models when s.get('team') is team.id)[0]
       team.readyToRank = @readyToRank(team.session)
-      team.matches = (convertMatch(match) for match in team.session.get('matches') or [])
+      team.matches = (convertMatch(match) for match in team.session?.get('matches') or [])
       team.matches.reverse()
-      team.score = (team.session.get('totalScore') or 10).toFixed(2)
+      team.score = (team.session?.get('totalScore') or 10).toFixed(2)
       
     ctx
 
@@ -77,8 +77,8 @@ module.exports = class MyMatchesTabView extends CocoView
       @setRankingButtonText button, if @readyToRank(session) then 'rank' else 'unavailable'
 
   readyToRank: (session) ->
-    c1 = session.get('code')
-    c2 = session.get('submittedCode')
+    c1 = session?.get('code')
+    c2 = session?.get('submittedCode')
     c1 and not _.isEqual(c1, c2)
 
   rankSession: (e) ->
diff --git a/app/views/play/ladder/play_modal.coffee b/app/views/play/ladder/play_modal.coffee
index 560397d37..1a31618ad 100644
--- a/app/views/play/ladder/play_modal.coffee
+++ b/app/views/play/ladder/play_modal.coffee
@@ -2,6 +2,7 @@ View = require 'views/kinds/ModalView'
 template = require 'templates/play/ladder/play_modal'
 ThangType = require 'models/ThangType'
 {me} = require 'lib/auth'
+LeaderboardCollection = require 'collections/LeaderboardCollection'
 
 module.exports = class LadderPlayModal extends View
   id: "ladder-play-modal"
@@ -19,7 +20,7 @@ module.exports = class LadderPlayModal extends View
   # PART 1: Load challengers from the db unless some are in the matches
 
   startLoadingChallengersMaybe: ->
-    matches = @session.get('matches')
+    matches = @session?.get('matches')
     if matches?.length then @loadNames() else @loadChallengers()
 
   loadChallengers: ->
@@ -126,8 +127,10 @@ class ChallengersData
     @hardPlayer = new LeaderboardCollection(@level, {order:-1, scoreOffset: score + 5, limit: 1, team: @otherTeam})
     @hardPlayer.fetch()
     @hardPlayer.once 'sync', @challengerLoaded, @
+    console.log 'fetching challengers yes'
 
   challengerLoaded: ->
+    console.log 'challenger loaded'
     if @allLoaded()
       @loaded = true
       @trigger 'sync'

From feaeb1dc980d6b8a5732e49b22d86b87e4d94c3d Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Mon, 3 Mar 2014 10:21:51 -0800
Subject: [PATCH 038/178] Made the wizard icons colored based on user settings.

---
 app/models/ThangType.coffee               |  5 +++-
 app/templates/play/ladder/play_modal.jade | 16 +++++------
 app/views/play/ladder/play_modal.coffee   | 34 ++++++++++++++---------
 server/users/user_handler.coffee          | 12 ++++++--
 4 files changed, 43 insertions(+), 24 deletions(-)

diff --git a/app/models/ThangType.coffee b/app/models/ThangType.coffee
index c162f9287..00065b3ed 100644
--- a/app/models/ThangType.coffee
+++ b/app/models/ThangType.coffee
@@ -172,7 +172,10 @@ module.exports = class ThangType extends CocoModel
     key = spriteOptionsOrKey
     key = if _.isString(key) then key else @spriteSheetKey(@fillOptions(key))
     spriteSheet = @spriteSheets[key]
-    spriteSheet ?= @buildSpriteSheet({portraitOnly:true})
+    if not spriteSheet
+      options = if _.isPlainObject spriteOptionsOrKey then spriteOptionsOrKey else {}
+      options.portraitOnly = true
+      spriteSheet = @buildSpriteSheet(options)
     return unless spriteSheet
     canvas = $("<canvas width='#{size}' height='#{size}'></canvas>")
     stage = new createjs.Stage(canvas[0])
diff --git a/app/templates/play/ladder/play_modal.jade b/app/templates/play/ladder/play_modal.jade
index 6eae0c87c..d7f3b2c34 100644
--- a/app/templates/play/ladder/play_modal.jade
+++ b/app/templates/play/ladder/play_modal.jade
@@ -12,8 +12,8 @@ block modal-body-content
 
   a(href="/play/level/#{levelID}?team=#{teamID}")
     div.play-option
-      img(src=portraitSRC).my-icon
-      img(src=portraitSRC).opponent-icon
+      img(src=myPortrait).my-icon
+      img(src=genericPortrait).opponent-icon
       div.my-name.name-label
         span= myName
       div.opponent-name.name-label
@@ -25,8 +25,8 @@ block modal-body-content
   if challengers.easy
     a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.easy.sessionID}")
       div.play-option.easy-option
-        img(src=portraitSRC).my-icon
-        img(src=portraitSRC).opponent-icon
+        img(src=myPortrait).my-icon
+        img(src=challengers.easy.opponentImageSource||genericPortrait).opponent-icon
         div.my-name.name-label
           span= myName
         div.opponent-name.name-label
@@ -38,8 +38,8 @@ block modal-body-content
   if challengers.medium
     a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.medium.sessionID}")
       div.play-option.medium-option
-        img(src=portraitSRC).my-icon
-        img(src=portraitSRC).opponent-icon
+        img(src=myPortrait).my-icon
+        img(src=challengers.medium.opponentImageSource||genericPortrait).opponent-icon
         div.my-name.name-label
           span= myName
         div.opponent-name.name-label
@@ -51,8 +51,8 @@ block modal-body-content
   if challengers.hard
     a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.hard.sessionID}")
       div.play-option.hard-option
-        img(src=portraitSRC).my-icon
-        img(src=portraitSRC).opponent-icon
+        img(src=myPortrait).my-icon
+        img(src=challengers.hard.opponentImageSource||genericPortrait).opponent-icon
         div.my-name.name-label
           span= myName
         div.opponent-name.name-label
diff --git a/app/views/play/ladder/play_modal.coffee b/app/views/play/ladder/play_modal.coffee
index 1a31618ad..5ef5fc74c 100644
--- a/app/views/play/ladder/play_modal.coffee
+++ b/app/views/play/ladder/play_modal.coffee
@@ -24,8 +24,8 @@ module.exports = class LadderPlayModal extends View
     if matches?.length then @loadNames() else @loadChallengers()
 
   loadChallengers: ->
-    @challengers = new ChallengersData(@level, @team, @otherTeam, @session)
-    @challengers.on 'sync', @loadNames, @
+    @challengersCollection = new ChallengersData(@level, @team, @otherTeam, @session)
+    @challengersCollection.on 'sync', @loadNames, @
 
   # PART 2: Loading the names of the other users
 
@@ -35,11 +35,12 @@ module.exports = class LadderPlayModal extends View
 
     success = (@nameMap) =>
       for challenger in _.values(@challengers)
-        challenger.opponentName = @nameMap[challenger.opponentID] or 'Anoner'
+        challenger.opponentName = @nameMap[challenger.opponentID]?.name or 'Anoner'
+        challenger.opponentWizard = @nameMap[challenger.opponentID]?.wizard or {}
       @checkWizardLoaded()
 
     $.ajax('/db/user/-/names', {
-      data: {ids: ids}
+      data: {ids: ids, wizard: true}
       type: 'POST'
       success: success
     })
@@ -62,9 +63,18 @@ module.exports = class LadderPlayModal extends View
     ctx.teamName = _.string.titleize @team
     ctx.teamID = @team
     ctx.otherTeamID = @otherTeam
-    ctx.challengers = if not @startsLoading then @challengers else {}
-    ctx.portraitSRC = @wizardType.getPortraitSource()
+
+    ctx.challengers = @challengers or {}
+    for challenger in _.values ctx.challengers
+      continue unless challenger
+      if (not challenger.opponentImageSource) and challenger.opponentWizard?.colorConfig
+        challenger.opponentImageSource = @wizardType.getPortraitSource(
+          {colorConfig: challenger.opponentWizard.colorConfig})
+      
+    ctx.genericPortrait = @wizardType.getPortraitSource()
     ctx.myName = me.get('name') || 'Newcomer'
+    myColorConfig = me.get('wizard')?.colorConfig
+    ctx.myPortrait = if myColorConfig then @wizardType.getPortraitSource({colorConfig: myColorConfig}) else ctx.genericPortrait
     ctx
     
   # Choosing challengers
@@ -72,10 +82,10 @@ module.exports = class LadderPlayModal extends View
   getChallengers: ->
     # make an object of challengers to everything needed to link to them
     challengers = {}
-    if @challengers
-      easyInfo = @challengeInfoFromSession(@challengers.easyPlayer.models[0])
-      mediumInfo = @challengeInfoFromSession(@challengers.mediumPlayer.models[0])
-      hardInfo = @challengeInfoFromSession(@challengers.hardPlayer.models[0])
+    if @challengersCollection
+      easyInfo = @challengeInfoFromSession(@challengersCollection.easyPlayer.models[0])
+      mediumInfo = @challengeInfoFromSession(@challengersCollection.mediumPlayer.models[0])
+      hardInfo = @challengeInfoFromSession(@challengersCollection.hardPlayer.models[0])
     else
       matches = @session.get('matches')
       won = (m for m in matches when m.metrics.rank < m.opponents[0].metrics.rank)
@@ -127,10 +137,8 @@ class ChallengersData
     @hardPlayer = new LeaderboardCollection(@level, {order:-1, scoreOffset: score + 5, limit: 1, team: @otherTeam})
     @hardPlayer.fetch()
     @hardPlayer.once 'sync', @challengerLoaded, @
-    console.log 'fetching challengers yes'
-
+    
   challengerLoaded: ->
-    console.log 'challenger loaded'
     if @allLoaded()
       @loaded = true
       @trigger 'sync'
diff --git a/server/users/user_handler.coffee b/server/users/user_handler.coffee
index 465aae27f..87c414a55 100644
--- a/server/users/user_handler.coffee
+++ b/server/users/user_handler.coffee
@@ -114,11 +114,19 @@ UserHandler = class UserHandler extends Handler
     ids = ids.split(',') if _.isString ids
     ids = _.uniq ids
     
+    # TODO: Extend and repurpose this handler to return other public info about a user more flexibly,
+    #   say by a query parameter that lists public properties to return.
+    returnWizard = req.query.wizard or req.body.wizard
+    query = if returnWizard then {name:1, wizard:1} else {name:1}
+    
     makeFunc = (id) ->
       (callback) ->
-        User.findById(id, {name:1}).exec (err, document) ->
+        User.findById(id, query).exec (err, document) ->
           return done(err) if err
-          callback(null, document?.get('name') or '')
+          if document and returnWizard
+            callback(null, {name:document.get('name'), wizard:document.get('wizard') or {}})
+          else
+            callback(null, document?.get('name') or '')
           
     funcs = {}
     for id in ids

From 0d054bec8e8d53cd89e2e03df8b14fa91e3c17ff Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Mon, 3 Mar 2014 10:42:11 -0800
Subject: [PATCH 039/178] Refactored the universal wizard loading to happen
 silently when you import ThangType.

---
 app/models/ThangType.coffee             | 14 ++++++++------
 app/views/home_view.coffee              |  4 ++++
 app/views/play/ladder/play_modal.coffee | 16 +++++++++-------
 3 files changed, 21 insertions(+), 13 deletions(-)

diff --git a/app/models/ThangType.coffee b/app/models/ThangType.coffee
index 00065b3ed..62f9eaa4e 100644
--- a/app/models/ThangType.coffee
+++ b/app/models/ThangType.coffee
@@ -214,9 +214,11 @@ module.exports = class ThangType extends CocoModel
   onFileUploaded: =>
     console.log 'Image uploaded'
 
-wizOriginal = "52a00d55cf1818f2be00000b"
-url = "/db/thang_type/#{wizOriginal}/version"
-wizardType = new module.exports()
-wizardType.url = -> url
-wizardType.fetch()
-module.exports.wizardType = wizardType
\ No newline at end of file
+  @loadUniversalWizard: ->
+    return @wizardType if @wizardType
+    wizOriginal = "52a00d55cf1818f2be00000b"
+    url = "/db/thang_type/#{wizOriginal}/version"
+    @wizardType = new module.exports()
+    @wizardType.url = -> url
+    @wizardType.fetch()
+    @wizardType
\ No newline at end of file
diff --git a/app/views/home_view.coffee b/app/views/home_view.coffee
index a16c1a9ba..3d9f30db2 100644
--- a/app/views/home_view.coffee
+++ b/app/views/home_view.coffee
@@ -11,6 +11,10 @@ module.exports = class HomeView extends View
   events:
     'mouseover #beginner-campaign': 'onMouseOverButton'
     'mouseout #beginner-campaign': 'onMouseOutButton'
+    
+  constructor: ->
+    super(arguments...)
+    ThangType.loadUniversalWizard()
 
   getRenderData: ->
     c = super()
diff --git a/app/views/play/ladder/play_modal.coffee b/app/views/play/ladder/play_modal.coffee
index 5ef5fc74c..30c3a116d 100644
--- a/app/views/play/ladder/play_modal.coffee
+++ b/app/views/play/ladder/play_modal.coffee
@@ -8,14 +8,14 @@ module.exports = class LadderPlayModal extends View
   id: "ladder-play-modal"
   template: template
   closeButton: true
-  startsLoading = true
+  startsLoading: true
 
   constructor: (options, @level, @session, @team) ->
     super(options)
     @nameMap = {}
     @otherTeam = if team is 'ogres' then 'humans' else 'ogres'
     @startLoadingChallengersMaybe()
-    @wizardType = ThangType.wizardType
+    @wizardType = ThangType.loadUniversalWizard()
     
   # PART 1: Load challengers from the db unless some are in the matches
 
@@ -66,15 +66,17 @@ module.exports = class LadderPlayModal extends View
 
     ctx.challengers = @challengers or {}
     for challenger in _.values ctx.challengers
-      continue unless challenger
+      continue unless challenger and @wizardType.loaded
       if (not challenger.opponentImageSource) and challenger.opponentWizard?.colorConfig
         challenger.opponentImageSource = @wizardType.getPortraitSource(
           {colorConfig: challenger.opponentWizard.colorConfig})
-      
-    ctx.genericPortrait = @wizardType.getPortraitSource()
+
+    if @wizardType.loaded
+      ctx.genericPortrait = @wizardType.getPortraitSource()
+      myColorConfig = me.get('wizard')?.colorConfig
+      ctx.myPortrait = if myColorConfig then @wizardType.getPortraitSource({colorConfig: myColorConfig}) else ctx.genericPortrait
+
     ctx.myName = me.get('name') || 'Newcomer'
-    myColorConfig = me.get('wizard')?.colorConfig
-    ctx.myPortrait = if myColorConfig then @wizardType.getPortraitSource({colorConfig: myColorConfig}) else ctx.genericPortrait
     ctx
     
   # Choosing challengers

From c5521a13114431b70a85ba7c7707f7c9e798ae7f Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Mon, 3 Mar 2014 11:04:03 -0800
Subject: [PATCH 040/178] Level now uses the universal wizard thang type.

---
 app/models/Level.coffee | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/models/Level.coffee b/app/models/Level.coffee
index cb82d325a..fca1f8249 100644
--- a/app/models/Level.coffee
+++ b/app/models/Level.coffee
@@ -117,7 +117,6 @@ module.exports = class Level extends CocoModel
     else if path is '/'
       # We also we need to make sure we grab the Wizard ThangType and the Marks. Hackitrooooid!
       for [type, original] in [
-        ["Wizard", "52a00d55cf1818f2be00000b"]
         ["Highlight", "529f8fdbdacd325127000003"]
         ["Selection", "52aa5f7520fccb0000000002"]
         ["Target", "52b32ad97385ec3d03000001"]
@@ -126,4 +125,5 @@ module.exports = class Level extends CocoModel
         link = "/db/thang_type/#{original}/version"
         model = CocoModel.getOrMakeModelFromLink link, shouldLoadProjection
         models.push model if model
+      models.push ThangType.loadUniversalWizard()
     models

From ff3f3e5a07edd23845962592e12edfb54446f5e2 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Mon, 3 Mar 2014 11:11:27 -0800
Subject: [PATCH 041/178] Removed a link to the old team view.

---
 app/templates/play/ladder/ladder_tab.jade | 6 ++----
 1 file changed, 2 insertions(+), 4 deletions(-)

diff --git a/app/templates/play/ladder/ladder_tab.jade b/app/templates/play/ladder/ladder_tab.jade
index f0c7a1b1f..457dd4914 100644
--- a/app/templates/play/ladder/ladder_tab.jade
+++ b/app/templates/play/ladder/ladder_tab.jade
@@ -17,7 +17,5 @@ div#columns.row
             td.score-cell= session.get('totalScore').toFixed(2)
             td.name-col-cell= session.get('creatorName') || "Anonymous"
             td
-              if(!myRow)
-                a(href="/play/level/#{level.get('slug') || level.id}/?team=#{team.otherTeam}&opponent=#{session.id}") Battle as #{team.otherTeam}!
-              else
-                a(href="/play/ladder/#{levelID}/team/#{team.id}") View your #{team.id} matches.
+              a(href="/play/level/#{level.get('slug') || level.id}/?team=#{team.otherTeam}&opponent=#{session.id}")
+                span Battle as #{team.otherTeam}!

From 65d39e658d8b57409c7b243a3f645fad57b9ba56 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Mon, 3 Mar 2014 11:15:46 -0800
Subject: [PATCH 042/178] Removed the old team view cruft.

---
 app/lib/Router.coffee                  |   2 +-
 app/styles/play/ladder.sass            |   4 -
 app/styles/play/ladder/team.sass       |  12 --
 app/templates/play/ladder.jade         |   3 -
 app/templates/play/ladder/team.jade    | 112 --------------
 app/views/play/ladder/team_view.coffee | 195 -------------------------
 6 files changed, 1 insertion(+), 327 deletions(-)
 delete mode 100644 app/styles/play/ladder/team.sass
 delete mode 100644 app/templates/play/ladder/team.jade
 delete mode 100644 app/views/play/ladder/team_view.coffee

diff --git a/app/lib/Router.coffee b/app/lib/Router.coffee
index 43d53c479..435fdb7d4 100644
--- a/app/lib/Router.coffee
+++ b/app/lib/Router.coffee
@@ -19,7 +19,7 @@ module.exports = class CocoRouter extends Backbone.Router
     'editor/:model(/:slug_or_id)(/:subview)': 'editorModelView'
     
     # Experimenting with direct links
-    'play/ladder/:levelID/team/:team': go('play/ladder/team_view')
+#    'play/ladder/:levelID/team/:team': go('play/ladder/team_view')
 
     # db and file urls call the server directly
     'db/*path': 'routeToServer'
diff --git a/app/styles/play/ladder.sass b/app/styles/play/ladder.sass
index 3ca1754e3..6904c5739 100644
--- a/app/styles/play/ladder.sass
+++ b/app/styles/play/ladder.sass
@@ -12,10 +12,6 @@
     margin-bottom: 10px
     background-image: none
 
-  .intro-button
-    width: 45%
-    margin: 0 2.5%
-
   .name-col-cell
     max-width: 300px
     text-overflow: ellipsis
diff --git a/app/styles/play/ladder/team.sass b/app/styles/play/ladder/team.sass
deleted file mode 100644
index ca4cc9676..000000000
--- a/app/styles/play/ladder/team.sass
+++ /dev/null
@@ -1,12 +0,0 @@
-#ladder-team-view
-  #rank-button
-    margin-top: 15px
-  
-  #competitors-column .well
-    font-size: 18px
-    padding: 7px
-
-  #your-score
-    margin-top: 20px
-    text-align: center
-    font-size: 20px
\ No newline at end of file
diff --git a/app/templates/play/ladder.jade b/app/templates/play/ladder.jade
index 18802f55d..41360c85b 100644
--- a/app/templates/play/ladder.jade
+++ b/app/templates/play/ladder.jade
@@ -3,9 +3,6 @@ block content
 
   div#level-column
     h1= level.get('name')
-    
-    //if !me.get('anonymous')
-    //  a(href="/play/level/brawlwood-tutorial").intro-button.btn.btn-primary.btn-lg Play the Tutorial
 
   if me.get('anonymous')
     div#must-log-in
diff --git a/app/templates/play/ladder/team.jade b/app/templates/play/ladder/team.jade
deleted file mode 100644
index 22bfd5404..000000000
--- a/app/templates/play/ladder/team.jade
+++ /dev/null
@@ -1,112 +0,0 @@
-extends /templates/base
-block content
-
-  ol.breadcrumb
-    li
-      a(href="/") Home
-    li
-      a(href="/play/ladder/#{levelID}")= level.get('name')
-    li.active= teamName
-  
-  p
-    | In this level, you play against everyone who has ever written strategies for the opposing forces.
-    | Choose from the suggested players on the right, playing as many and as long as you like,
-    | and when you are ready to test your grand strategy against the whole ladder, return and click the rank button.
-    
-  p
-    | After your first submission, your code will also continuously run against other players as they rank themselves.
-
-  if matches.length
-    p#your-score
-      span Your Current Score:
-      span  
-      strong= score
-
-
-  div#columns.row
-    div#matches-column.col-md-6
-      h3.pull-left Ranked Games
-      button.btn.btn-warning.pull-right#rank-button
-        span.unavailable.hidden No New Code to Rank
-        span.rank.hidden Rank My Game!
-        span.ranking.hidden Submitting...
-        span.ranked.hidden Submitted for Ranking
-        span.failed.hidden Failed to Rank
-      
-      hr.clearfix(style="clear: both")
-
-      if matches.length
-        table.table.table-bordered.table-condensed
-          tr
-            th Result
-            th Opponent
-            th When
-          for match in matches
-            tr
-              td.state-cell
-                if match.state === 'win'
-                  span.win Win
-                if match.state === 'loss'
-                  span.loss Loss
-                if match.state === 'tie'
-                  span.tie Tie
-              td.name-cell= match.opponentName || "Anonymous"
-              td.time-cell= match.when
-              td.battle-cell
-                - var text = match.state === 'win' ? 'Watch your victory' : 'Defeat the ' + otherTeamID
-                a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{match.sessionID}")= text
-      
-      else
-        div.alert.alert-warning
-          | No ranked matches played yet!
-          | Play some competitors on the right and then come back to get your game ranked.
-      
-      // finish this once matches are available
-        
-    div#competitors-column.col-md-6
-      h3 Your Competitors
-
-      .well.text-muted
-        div.row
-          div.col-md-2
-            span.warmup Warmup
-            span : 
-          div.col-md-10
-            a(href="/play/level/#{levelID}?team=#{teamID}")
-              span.warmup Play #{teamID} vs Default #{otherTeamID}
-
-      if challengers.easy
-        .well
-          div.row.text-info.bg-info
-            div.col-md-2
-              span.easy Easy
-              span : 
-            div.col-md-10
-              a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.easy.sessionID}")
-                span Play #{teamID} vs 
-                strong= challengers.easy.opponentName
-                span  #{otherTeamID}
-            
-      if challengers.medium
-        .well
-          div.row.text-warning.bg-warning
-            div.col-md-2
-              span.medium Medium
-              span : 
-            div.col-md-10
-              a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.medium.sessionID}")
-                span Play #{teamID} vs 
-                strong= challengers.medium.opponentName
-                span  #{otherTeamID}
-
-      if challengers.hard
-        .well
-          div.row.text-danger.bg-danger
-            div.col-md-2
-              span.hard Hard
-              span : 
-            div.col-md-10
-              a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.hard.sessionID}")
-                span Play #{teamID} vs 
-                strong= challengers.hard.opponentName
-                span  #{otherTeamID}
diff --git a/app/views/play/ladder/team_view.coffee b/app/views/play/ladder/team_view.coffee
deleted file mode 100644
index 32adcbc24..000000000
--- a/app/views/play/ladder/team_view.coffee
+++ /dev/null
@@ -1,195 +0,0 @@
-RootView = require 'views/kinds/RootView'
-Level = require 'models/Level'
-LevelSession = require 'models/LevelSession'
-LeaderboardCollection  = require 'collections/LeaderboardCollection'
-
-module.exports = class LadderTeamView extends RootView
-  id: 'ladder-team-view'
-  template: require 'templates/play/ladder/team'
-  startsLoading: true
-
-  events:
-    'click #rank-button': 'rankSession'
-
-  # PART 1: Loading Level/Session
-
-  constructor: (options, @levelID, @team) ->
-    super(options)
-    @otherTeam = if team is 'ogres' then 'humans' else 'ogres'
-    @level = new Level(_id:@levelID)
-    @level.fetch()
-    @level.once 'sync', @onLevelLoaded, @
-
-    url = "/db/level/#{@levelID}/session?team=#{@team}"
-    @session = new LevelSession()
-    @session.url = -> url
-    @session.fetch()
-    @session.once 'sync', @onSessionLoaded, @
-
-  onLevelLoaded: -> @startLoadingChallengersMaybe()
-  onSessionLoaded: -> @startLoadingChallengersMaybe()
-
-  # PART 2: Loading some challengers if we don't have any matches yet
-
-  startLoadingChallengersMaybe: ->
-    return unless @level.loaded and @session.loaded
-    matches = @session.get('matches')
-    if matches?.length then @loadNames() else @loadChallengers()
-
-  loadChallengers: ->
-    @challengers = new ChallengersData(@level, @team, @otherTeam, @session)
-    @challengers.on 'sync', @loadNames, @
-
-  # PART 3: Loading the names of the other users
-
-  loadNames: ->
-    ids = []
-    ids.push match.opponents[0].userID for match in @session.get('matches') or []
-    ids = ids.concat(@challengers.playerIDs()) if @challengers
-
-    success = (@nameMap) =>
-      for match in @session.get('matches') or []
-        opponent = match.opponents[0]
-        opponent.userName = @nameMap[opponent.userID]
-      @finishRendering()
-
-    $.ajax('/db/user/-/names', {
-      data: {ids: ids}
-      type: 'POST'
-      success: success
-    })
-
-  # PART 4: Rendering
-
-  finishRendering: ->
-    @startsLoading = false
-    @render()
-
-  getRenderData: ->
-    ctx = super()
-    ctx.level = @level
-    ctx.levelID = @levelID
-    ctx.teamName = _.string.titleize @team
-    ctx.teamID = @team
-    ctx.otherTeamID = @otherTeam
-    ctx.challengers = if not @startsLoading then @getChallengers() else {}
-    ctx.readyToRank = @readyToRank()
-
-    convertMatch = (match) =>
-      opponent = match.opponents[0]
-      state = 'win'
-      state = 'loss' if match.metrics.rank > opponent.metrics.rank
-      state = 'tie' if match.metrics.rank is opponent.metrics.rank
-      {
-        state: state
-        opponentName: @nameMap[opponent.userID]
-        opponentID: opponent.userID
-        when: moment(match.date).fromNow()
-        sessionID: opponent.sessionID
-      }
-
-    ctx.matches = (convertMatch(match) for match in @session.get('matches') or [])
-    ctx.matches.reverse()
-    ctx.score = (@session.get('totalScore') or 10).toFixed(2)
-    ctx
-
-  afterRender: ->
-    super()
-    @setRankingButtonText(if @readyToRank() then 'rank' else 'unavailable')
-
-  readyToRank: ->
-    c1 = @session.get('code')
-    c2 = @session.get('submittedCode')
-    c1 and not _.isEqual(c1, c2)
-
-  getChallengers: ->
-    # make an object of challengers to everything needed to link to them
-    challengers = {}
-    if @challengers
-      easyInfo = @challengeInfoFromSession(@challengers.easyPlayer.models[0])
-      mediumInfo = @challengeInfoFromSession(@challengers.mediumPlayer.models[0])
-      hardInfo = @challengeInfoFromSession(@challengers.hardPlayer.models[0])
-    else
-      matches = @session.get('matches')
-      won = (m for m in matches when m.metrics.rank < m.opponents[0].metrics.rank)
-      lost = (m for m in matches when m.metrics.rank > m.opponents[0].metrics.rank)
-      tied = (m for m in matches when m.metrics.rank is m.opponents[0].metrics.rank)
-      easyInfo = @challengeInfoFromMatches(won)
-      mediumInfo = @challengeInfoFromMatches(tied)
-      hardInfo = @challengeInfoFromMatches(lost)
-    @addChallenger easyInfo, challengers, 'easy'
-    @addChallenger mediumInfo, challengers, 'medium'
-    @addChallenger hardInfo, challengers, 'hard'
-    challengers
-
-  addChallenger: (info, challengers, title) ->
-    # check for duplicates first
-    return unless info
-    for key, value of challengers
-      return if value.sessionID is info.sessionID
-    challengers[title] = info
-
-  challengeInfoFromSession: (session) ->
-    # given a model from the db, return info needed for a link to the match
-    return unless session
-    return {
-      sessionID: session.id
-      opponentName: @nameMap[session.get('creator')] or 'Anoner'
-      opponentID: session.get('creator')
-    }
-
-  challengeInfoFromMatches: (matches) ->
-    return unless matches?.length
-    match = _.sample matches
-    opponent = match.opponents[0]
-    return {
-      sessionID: opponent.sessionID
-      opponentName: opponent.userName or 'Anoner'
-      opponentID: opponent.userID
-    }
-
-  rankSession: ->
-    return unless @readyToRank()
-    @setRankingButtonText('ranking')
-
-    success = => @setRankingButtonText('ranked')
-    failure = => @setRankingButtonText('failed')
-
-    $.ajax '/queue/scoring', {
-      type: 'POST'
-      data: { session: @session.id }
-      success: success
-      failure: failure
-    }
-
-  setRankingButtonText: (spanClass) ->
-    rankButton = $('#rank-button')
-    rankButton.find('span').addClass('hidden')
-    rankButton.find(".#{spanClass}").removeClass('hidden')
-    rankButton.toggleClass 'disabled', spanClass isnt 'rank'
-
-class ChallengersData
-  constructor: (@level, @team, @otherTeam, @session) ->
-    _.extend @, Backbone.Events
-    score = @session?.get('totalScore') or 25
-    @easyPlayer = new LeaderboardCollection(@level, {order:1, scoreOffset: score - 5, limit: 1, team: @otherTeam})
-    @easyPlayer.fetch()
-    @easyPlayer.once 'sync', @challengerLoaded, @
-    @mediumPlayer = new LeaderboardCollection(@level, {order:1, scoreOffset: score, limit: 1, team: @otherTeam})
-    @mediumPlayer.fetch()
-    @mediumPlayer.once 'sync', @challengerLoaded, @
-    @hardPlayer = new LeaderboardCollection(@level, {order:-1, scoreOffset: score + 5, limit: 1, team: @otherTeam})
-    @hardPlayer.fetch()
-    @hardPlayer.once 'sync', @challengerLoaded, @
-
-  challengerLoaded: ->
-    if @allLoaded()
-      @loaded = true
-      @trigger 'sync'
-
-  playerIDs: ->
-    collections = [@easyPlayer, @mediumPlayer, @hardPlayer]
-    (c.models[0].get('creator') for c in collections when c?.models[0])
-
-  allLoaded: ->
-    _.all [@easyPlayer.loaded, @mediumPlayer.loaded, @hardPlayer.loaded]

From 942b69ea9f1f12464972ca1881f6199e8710b212 Mon Sep 17 00:00:00 2001
From: Ruben Vereecken <ruben.vereecken@student.uantwerpen.be>
Date: Mon, 3 Mar 2014 20:41:35 +0100
Subject: [PATCH 043/178] Created hasWriteAccess on Cocomodel, used it for
 level access

---
 app/models/CocoModel.coffee          | 21 +++++++++++++++++++++
 app/templates/editor/level/edit.jade |  7 +++----
 app/templates/kinds/search.jade      |  1 -
 app/views/editor/level/edit.coffee   |  1 +
 4 files changed, 25 insertions(+), 5 deletions(-)

diff --git a/app/models/CocoModel.coffee b/app/models/CocoModel.coffee
index f4551b904..74dc336bf 100644
--- a/app/models/CocoModel.coffee
+++ b/app/models/CocoModel.coffee
@@ -184,4 +184,25 @@ class CocoModel extends Backbone.Model
   @isObjectID: (s) ->
     s.length is 24 and s.match(/[a-z0-9]/gi)?.length is 24
 
+  hasReadAccess: (actor) ->
+    # actor is a User object
+
+    if @get('permissions')?
+      for permission in @get('permissions')
+        if permission.target is 'public' or actor.get('_id') is permission.target
+          return true if permission.access in ['owner', 'read']
+
+    return false
+
+    hasWriteAccess: (actor) ->
+      # actor is a User object
+
+    if @get('permissions')?
+      for permission in @get('permissions')
+        if permission.target is 'public' or actor.get('_id') is permission.target
+          return true if permission.access in ['owner', 'write']
+
+    return false
+
+
 module.exports = CocoModel
diff --git a/app/templates/editor/level/edit.jade b/app/templates/editor/level/edit.jade
index cdc97cd2a..e63cdd2b9 100644
--- a/app/templates/editor/level/edit.jade
+++ b/app/templates/editor/level/edit.jade
@@ -29,10 +29,9 @@ block outer_content
             
             
           ul.nav.navbar-nav.navbar-right
-            li(data-toggle="coco-modal", data-target="modal/revert", data-i18n="revert.revert").btn.btn-primary.navbar-btn#revert-button Revert
-
-            li(data-i18n="common.save").btn.btn-primary.navbar-btn#commit-level-start-button Save
-            li(data-i18n="common.fork").btn.btn-primary.navbar-btn#fork-level-start-button Fork
+            li(data-toggle="coco-modal", data-target="modal/revert", data-i18n="revert.revert", disabled=authorized === true ? undefined : "true").btn.btn-primary.navbar-btn#revert-button Revert
+            li(data-i18n="common.save", disabled=authorized === true ? undefined : "true").btn.btn-primary.navbar-btn#commit-level-start-button Save
+            li(data-i18n="common.fork", disabled=authorized === true ? undefined : "true").btn.btn-primary.navbar-btn#fork-level-start-button Fork
             li(title="⌃↩ or ⌘↩: Play preview of current level", data-i18n="common.play")#play-button.btn.btn-inverse.banner.navbar-btn Play!
 
             li.divider
diff --git a/app/templates/kinds/search.jade b/app/templates/kinds/search.jade
index eda4c0f2b..296581a47 100644
--- a/app/templates/kinds/search.jade
+++ b/app/templates/kinds/search.jade
@@ -11,7 +11,6 @@ block content
     a.btn.btn-primary.open-modal-button(data-toggle="coco-modal", data-target="modal/signup", role="button") Sign Up to Create a New #{modelLabel}
   else
     a.btn.btn-primary.open-modal-button(href='#new-model-modal', role="button", data-toggle="modal" data-i18n="#{currentNew}") Create a New Something
-
   input#search(data-i18n="[placeholder]#{currentSearch}")
   hr
   div.results
diff --git a/app/views/editor/level/edit.coffee b/app/views/editor/level/edit.coffee
index bbafd529d..c79439800 100644
--- a/app/views/editor/level/edit.coffee
+++ b/app/views/editor/level/edit.coffee
@@ -63,6 +63,7 @@ module.exports = class EditorLevelView extends View
   getRenderData: (context={}) ->
     context = super(context)
     context.level = @level
+    context.authorized = me.isAdmin() or @level.hasWriteAccess(me)
     context
 
   afterRender: ->

From cc9cf64539592332f1793b2e447bec64fc820fb2 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 3 Mar 2014 11:55:26 -0800
Subject: [PATCH 044/178] Corrected link to view your own matches in
 leaderboard view.

---
 app/templates/play/ladder/ladder_tab.jade | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/templates/play/ladder/ladder_tab.jade b/app/templates/play/ladder/ladder_tab.jade
index f0c7a1b1f..8f79bd624 100644
--- a/app/templates/play/ladder/ladder_tab.jade
+++ b/app/templates/play/ladder/ladder_tab.jade
@@ -20,4 +20,4 @@ div#columns.row
               if(!myRow)
                 a(href="/play/level/#{level.get('slug') || level.id}/?team=#{team.otherTeam}&opponent=#{session.id}") Battle as #{team.otherTeam}!
               else
-                a(href="/play/ladder/#{levelID}/team/#{team.id}") View your #{team.id} matches.
+                a(href="/play/ladder/#{level.get('slug') || level.id}/team/#{team.id}") View your #{team.id} matches.

From fb75ad36e7740133dcd1fcc577bcf7f97758fe66 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Mon, 3 Mar 2014 12:04:31 -0800
Subject: [PATCH 045/178] Added team icons to the ladder play view.

---
 .../pages/play/ladder/humans_ladder_easy.png  | Bin 0 -> 22579 bytes
 .../pages/play/ladder/humans_ladder_hard.png  | Bin 0 -> 24237 bytes
 .../play/ladder/humans_ladder_medium.png      | Bin 0 -> 18770 bytes
 .../play/ladder/humans_ladder_tutorial.png    | Bin 0 -> 22473 bytes
 .../pages/play/ladder/ogres_ladder_easy.png   | Bin 0 -> 23086 bytes
 .../pages/play/ladder/ogres_ladder_hard.png   | Bin 0 -> 25821 bytes
 .../pages/play/ladder/ogres_ladder_medium.png | Bin 0 -> 23199 bytes
 .../play/ladder/ogres_ladder_tutorial.png     | Bin 0 -> 26542 bytes
 app/styles/play/ladder/play_modal.sass        |  33 ++++++++++++++++++
 app/templates/play/ladder/play_modal.jade     |  24 ++++++++-----
 app/views/play/ladder/play_modal.coffee       |   9 +++++
 11 files changed, 58 insertions(+), 8 deletions(-)
 create mode 100644 app/assets/images/pages/play/ladder/humans_ladder_easy.png
 create mode 100644 app/assets/images/pages/play/ladder/humans_ladder_hard.png
 create mode 100644 app/assets/images/pages/play/ladder/humans_ladder_medium.png
 create mode 100644 app/assets/images/pages/play/ladder/humans_ladder_tutorial.png
 create mode 100644 app/assets/images/pages/play/ladder/ogres_ladder_easy.png
 create mode 100644 app/assets/images/pages/play/ladder/ogres_ladder_hard.png
 create mode 100644 app/assets/images/pages/play/ladder/ogres_ladder_medium.png
 create mode 100644 app/assets/images/pages/play/ladder/ogres_ladder_tutorial.png

diff --git a/app/assets/images/pages/play/ladder/humans_ladder_easy.png b/app/assets/images/pages/play/ladder/humans_ladder_easy.png
new file mode 100644
index 0000000000000000000000000000000000000000..ea34dcc5b75a4a7be749482e9a4b5173f8036175
GIT binary patch
literal 22579
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

literal 0
HcmV?d00001

diff --git a/app/assets/images/pages/play/ladder/humans_ladder_hard.png b/app/assets/images/pages/play/ladder/humans_ladder_hard.png
new file mode 100644
index 0000000000000000000000000000000000000000..8cd03225d4ac184ef24c71f9bebe9e72e7a4c193
GIT binary patch
literal 24237
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
KMNUMnLSTXojnLNs

literal 0
HcmV?d00001

diff --git a/app/assets/images/pages/play/ladder/humans_ladder_medium.png b/app/assets/images/pages/play/ladder/humans_ladder_medium.png
new file mode 100644
index 0000000000000000000000000000000000000000..f4b5fdf946770cba7874cef46e7925bcd61796ff
GIT binary patch
literal 18770
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
A$p8QV

literal 0
HcmV?d00001

diff --git a/app/assets/images/pages/play/ladder/humans_ladder_tutorial.png b/app/assets/images/pages/play/ladder/humans_ladder_tutorial.png
new file mode 100644
index 0000000000000000000000000000000000000000..8e34fc9244e077c17f06102068a0a2b8254abfc9
GIT binary patch
literal 22473
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

literal 0
HcmV?d00001

diff --git a/app/assets/images/pages/play/ladder/ogres_ladder_easy.png b/app/assets/images/pages/play/ladder/ogres_ladder_easy.png
new file mode 100644
index 0000000000000000000000000000000000000000..d5e4695ff0715bd69e0db789e8b693742c214eed
GIT binary patch
literal 23086
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
LNkvXXu0mjfL5d8f

literal 0
HcmV?d00001

diff --git a/app/assets/images/pages/play/ladder/ogres_ladder_hard.png b/app/assets/images/pages/play/ladder/ogres_ladder_hard.png
new file mode 100644
index 0000000000000000000000000000000000000000..36204caf5d95697e81c8a0e1a5e17af8f41d061f
GIT binary patch
literal 25821
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

literal 0
HcmV?d00001

diff --git a/app/assets/images/pages/play/ladder/ogres_ladder_medium.png b/app/assets/images/pages/play/ladder/ogres_ladder_medium.png
new file mode 100644
index 0000000000000000000000000000000000000000..5e327d74b5ba960224a571b95ea626a5ea4ba0a6
GIT binary patch
literal 23199
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
CbO8VW

literal 0
HcmV?d00001

diff --git a/app/assets/images/pages/play/ladder/ogres_ladder_tutorial.png b/app/assets/images/pages/play/ladder/ogres_ladder_tutorial.png
new file mode 100644
index 0000000000000000000000000000000000000000..16e9527285183e766fd6af67707b93a6398cb80a
GIT binary patch
literal 26542
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%0DeL%**fUcS+&aW~=h%5-VKjBBCq|`jVr1gVPkr
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
Z{6Di-IP(%LT)+ST002ovPDHLkV1hVB$;1Ev

literal 0
HcmV?d00001

diff --git a/app/styles/play/ladder/play_modal.sass b/app/styles/play/ladder/play_modal.sass
index 8dc69a5ae..7d87d475b 100644
--- a/app/styles/play/ladder/play_modal.sass
+++ b/app/styles/play/ladder/play_modal.sass
@@ -21,14 +21,45 @@
     opacity: 0.4
     
     border-radius: 5px
+    .only-one
+      -webkit-transition: opacity 0.3s ease-in-out
+      -moz-transition: opacity 0.3s ease-in-out
+      -ms-transition: opacity 0.3s ease-in-out
+      -o-transition: opacity 0.3s ease-in-out
+      transition: opacity 0.3s ease-in-out
+      opacity: 0
     
   .play-option:hover
     opacity: 1
+    .only-one
+      opacity: 1
     
   .my-icon
     position: relative
     left: 0
     top: -10px
+    z-index: 1
+    
+  .my-team-icon
+    height: 60px
+    position: relative
+    top: -10px
+    left: 10px
+    z-index: 0
+    
+  .opponent-team-icon
+    height: 60px
+    position: relative
+    top: 10px
+    right: 10px
+    z-index: 0
+    float: right
+    -moz-transform: scaleX(-1)
+    -o-transform: scaleX(-1)
+    -webkit-transform: scaleX(-1)
+    transform: scaleX(-1)
+    filter: FlipH
+    -ms-filter: "FlipH"
     
   .opponent-icon
     position: relative
@@ -41,6 +72,7 @@
     transform: scaleX(-1)
     filter: FlipH
     -ms-filter: "FlipH"
+    z-index: 1
     
   .name-label
     border-bottom: 20px solid lightslategray
@@ -51,6 +83,7 @@
     color: black
     font-weight: bold
     text-align: center
+    z-index: 2
     
     span
       position: relative
diff --git a/app/templates/play/ladder/play_modal.jade b/app/templates/play/ladder/play_modal.jade
index d7f3b2c34..6c1cd22d6 100644
--- a/app/templates/play/ladder/play_modal.jade
+++ b/app/templates/play/ladder/play_modal.jade
@@ -12,9 +12,11 @@ block modal-body-content
 
   a(href="/play/level/#{levelID}?team=#{teamID}")
     div.play-option
-      img(src=myPortrait).my-icon
+      img(src=myPortrait).my-icon.only-one
+      img(src="/images/pages/play/ladder/"+teamID+"_ladder_tutorial.png", style="border: 1px solid #{teamColor}; background: #{teamBackgroundColor}").my-team-icon.img-circle.only-one
       img(src=genericPortrait).opponent-icon
-      div.my-name.name-label
+      img(src="/images/pages/play/ladder/"+otherTeamID+"_ladder_tutorial.png", style="border: 1px solid #{opponentTeamColor}; background: #{opponentTeamBackgroundColor}").opponent-team-icon.img-circle
+      div.my-name.name-label.only-one
         span= myName
       div.opponent-name.name-label
         span Simple AI
@@ -25,9 +27,11 @@ block modal-body-content
   if challengers.easy
     a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.easy.sessionID}")
       div.play-option.easy-option
-        img(src=myPortrait).my-icon
+        img(src=myPortrait).my-icon.only-one
+        img(src="/images/pages/play/ladder/"+teamID+"_ladder_easy.png", style="border: 1px solid #{teamColor}; background: #{teamBackgroundColor}").my-team-icon.img-circle.only-one
         img(src=challengers.easy.opponentImageSource||genericPortrait).opponent-icon
-        div.my-name.name-label
+        img(src="/images/pages/play/ladder/"+otherTeamID+"_ladder_easy.png", style="border: 1px solid #{opponentTeamColor}; background: #{opponentTeamBackgroundColor}").opponent-team-icon.img-circle
+        div.my-name.name-label.only-one
           span= myName
         div.opponent-name.name-label
           span= challengers.easy.opponentName
@@ -38,9 +42,11 @@ block modal-body-content
   if challengers.medium
     a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.medium.sessionID}")
       div.play-option.medium-option
-        img(src=myPortrait).my-icon
+        img(src=myPortrait).my-icon.only-one
+        img(src="/images/pages/play/ladder/"+teamID+"_ladder_medium.png", style="border: 1px solid #{teamColor}; background: #{teamBackgroundColor}").my-team-icon.img-circle.only-one
         img(src=challengers.medium.opponentImageSource||genericPortrait).opponent-icon
-        div.my-name.name-label
+        img(src="/images/pages/play/ladder/"+otherTeamID+"_ladder_medium.png", style="border: 1px solid #{opponentTeamColor}; background: #{opponentTeamBackgroundColor}").opponent-team-icon.img-circle
+        div.my-name.name-label.only-one
           span= myName
         div.opponent-name.name-label
           span= challengers.medium.opponentName
@@ -51,9 +57,11 @@ block modal-body-content
   if challengers.hard
     a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.hard.sessionID}")
       div.play-option.hard-option
-        img(src=myPortrait).my-icon
+        img(src=myPortrait).my-icon.only-one
+        img(src="/images/pages/play/ladder/"+teamID+"_ladder_hard.png", style="border: 1px solid #{teamColor}; background: #{teamBackgroundColor}").my-team-icon.img-circle.only-one
         img(src=challengers.hard.opponentImageSource||genericPortrait).opponent-icon
-        div.my-name.name-label
+        img(src="/images/pages/play/ladder/"+otherTeamID+"_ladder_hard.png", style="border: 1px solid #{opponentTeamColor}; background: #{opponentTeamBackgroundColor}").opponent-team-icon.img-circle
+        div.my-name.name-label.only-one
           span= myName
         div.opponent-name.name-label
           span= challengers.hard.opponentName
diff --git a/app/views/play/ladder/play_modal.coffee b/app/views/play/ladder/play_modal.coffee
index 30c3a116d..28f689b14 100644
--- a/app/views/play/ladder/play_modal.coffee
+++ b/app/views/play/ladder/play_modal.coffee
@@ -3,6 +3,7 @@ template = require 'templates/play/ladder/play_modal'
 ThangType = require 'models/ThangType'
 {me} = require 'lib/auth'
 LeaderboardCollection = require 'collections/LeaderboardCollection'
+{teamDataFromLevel} = require './utils'
 
 module.exports = class LadderPlayModal extends View
   id: "ladder-play-modal"
@@ -63,6 +64,14 @@ module.exports = class LadderPlayModal extends View
     ctx.teamName = _.string.titleize @team
     ctx.teamID = @team
     ctx.otherTeamID = @otherTeam
+    
+    teamsList = teamDataFromLevel @level
+    teams = {}
+    teams[team.id] = team for team in teamsList
+    ctx.teamColor = teams[@team].primaryColor
+    ctx.teamBackgroundColor = teams[@team].bgColor
+    ctx.opponentTeamColor = teams[@otherTeam].primaryColor
+    ctx.opponentTeamBackgroundColor = teams[@otherTeam].bgColor
 
     ctx.challengers = @challengers or {}
     for challenger in _.values ctx.challengers

From 1efb96f24ac06b9dfea414cb38ded8735daa2f37 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Mon, 3 Mar 2014 12:08:11 -0800
Subject: [PATCH 046/178] Merge branch 'master' of
 https://github.com/codecombat/codecombat

Conflicts:
	app/templates/play/ladder/ladder_tab.jade
---
 app/templates/play/ladder/ladder_tab.jade | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/templates/play/ladder/ladder_tab.jade b/app/templates/play/ladder/ladder_tab.jade
index 457dd4914..eed7e9d2c 100644
--- a/app/templates/play/ladder/ladder_tab.jade
+++ b/app/templates/play/ladder/ladder_tab.jade
@@ -18,4 +18,4 @@ div#columns.row
             td.name-col-cell= session.get('creatorName') || "Anonymous"
             td
               a(href="/play/level/#{level.get('slug') || level.id}/?team=#{team.otherTeam}&opponent=#{session.id}")
-                span Battle as #{team.otherTeam}!
+                span Battle as #{team.otherTeam}!
\ No newline at end of file

From a5864c4f0c7bd5e6047e838e537ec707ef23a277 Mon Sep 17 00:00:00 2001
From: Michael Schmatz <schmatz@umich.edu>
Date: Mon, 3 Mar 2014 12:10:24 -0800
Subject: [PATCH 047/178] Live leaderboard updating

---
 app/views/play/ladder/my_matches_tab.coffee |  4 ++++
 app/views/play/ladder_view.coffee           | 13 ++++++++++++-
 2 files changed, 16 insertions(+), 1 deletion(-)

diff --git a/app/views/play/ladder/my_matches_tab.coffee b/app/views/play/ladder/my_matches_tab.coffee
index f4994d519..00aec7c8c 100644
--- a/app/views/play/ladder/my_matches_tab.coffee
+++ b/app/views/play/ladder/my_matches_tab.coffee
@@ -14,6 +14,9 @@ module.exports = class MyMatchesTabView extends CocoView
 
   constructor: (options, @level, @sessions) ->
     super(options)
+    @refreshMatches()
+    
+  refreshMatches: ->
     @teams = teamDataFromLevel @level
     @nameMap = {}
     @loadNames()
@@ -82,6 +85,7 @@ module.exports = class MyMatchesTabView extends CocoView
     c1 and not _.isEqual(c1, c2)
 
   rankSession: (e) ->
+    console.log "Clicked"
     button = $(e.target).closest('.rank-button')
     sessionID = button.data('session-id')
     session = _.find @sessions.models, { id: sessionID }
diff --git a/app/views/play/ladder_view.coffee b/app/views/play/ladder_view.coffee
index b1d74b182..bc4f0fd6d 100644
--- a/app/views/play/ladder_view.coffee
+++ b/app/views/play/ladder_view.coffee
@@ -64,7 +64,18 @@ module.exports = class LadderView extends RootView
     return if @startsLoading
     @insertSubView(@ladderTab = new LadderTabView({}, @level, @sessions))
     @insertSubView(@myMatchesTab = new MyMatchesTabView({}, @level, @sessions))
-
+    setInterval(@fetchSessionsAndRefreshViews.bind(@), 10000)
+  
+  fetchSessionsAndRefreshViews: ->
+    @sessions.fetch({"success": @refreshViews})
+    
+  refreshViews: =>
+    @ladderTab.constructor({}, @level, @sessions)
+    @myMatchesTab.refreshMatches()
+    console.log "refreshed views!"
+    
+    
+    
   # Simulations
 
   onSimulateAllButtonClick: (e) ->

From ee920afea9cdb78e933550de89c8cf4730221381 Mon Sep 17 00:00:00 2001
From: Ruben Vereecken <ruben.vereecken@student.uantwerpen.be>
Date: Mon, 3 Mar 2014 21:13:02 +0100
Subject: [PATCH 048/178] Disabled buttons for non-owners in Article Editor

---
 app/models/CocoModel.coffee                    | 4 ++--
 app/templates/editor/article/edit.jade         | 6 +++---
 app/views/editor/article/edit.coffee           | 1 +
 app/views/editor/level/scripts_tab_view.coffee | 1 +
 4 files changed, 7 insertions(+), 5 deletions(-)

diff --git a/app/models/CocoModel.coffee b/app/models/CocoModel.coffee
index 74dc336bf..ba4ff850c 100644
--- a/app/models/CocoModel.coffee
+++ b/app/models/CocoModel.coffee
@@ -194,8 +194,8 @@ class CocoModel extends Backbone.Model
 
     return false
 
-    hasWriteAccess: (actor) ->
-      # actor is a User object
+  hasWriteAccess: (actor) ->
+    # actor is a User object
 
     if @get('permissions')?
       for permission in @get('permissions')
diff --git a/app/templates/editor/article/edit.jade b/app/templates/editor/article/edit.jade
index a729bf67a..e7dff7cd7 100644
--- a/app/templates/editor/article/edit.jade
+++ b/app/templates/editor/article/edit.jade
@@ -10,9 +10,9 @@ block content
       li.active
         | #{article.attributes.name}
 
-  button(data-toggle="coco-modal", data-target="modal/revert", data-i18n="revert.revert").btn.btn-primary#revert-button Revert
-  button(data-i18n="article.edit_btn_preview").btn.btn-primary#preview-button Preview
-  button(data-toggle="coco-modal", data-target="modal/save_version", data-i18n="common.save").btn.btn-primary#save-button Save
+  button(data-toggle="coco-modal", data-target="modal/revert", data-i18n="revert.revert", disabled=authorized === true ? undefined : "true").btn.btn-primary#revert-button Revert
+  button(data-i18n="article.edit_btn_preview", disabled=authorized === true ? undefined : "true").btn.btn-primary#preview-button Preview
+  button(data-toggle="coco-modal", data-target="modal/save_version", data-i18n="common.save", disabled=authorized === true ? undefined : "true").btn.btn-primary#save-button Save
 
   h3(data-i18n="article.edit_article_title") Edit Article
     span
diff --git a/app/views/editor/article/edit.coffee b/app/views/editor/article/edit.coffee
index 05df5291d..d3c2d4c1c 100644
--- a/app/views/editor/article/edit.coffee
+++ b/app/views/editor/article/edit.coffee
@@ -56,6 +56,7 @@ module.exports = class ArticleEditView extends View
   getRenderData: (context={}) ->
     context = super(context)
     context.article = @article
+    context.authorized = me.isAdmin() or @article.hasWriteAccess(me)
     context
 
   openPreview: =>
diff --git a/app/views/editor/level/scripts_tab_view.coffee b/app/views/editor/level/scripts_tab_view.coffee
index 45b5c210d..9da71e31c 100644
--- a/app/views/editor/level/scripts_tab_view.coffee
+++ b/app/views/editor/level/scripts_tab_view.coffee
@@ -59,6 +59,7 @@ module.exports = class ScriptsTabView extends View
       thangIDs: thangIDs
       dimensions: @dimensions
       supermodel: @supermodel
+      readOnly: not me.isAdmin() and not @level.hasWriteAccess(me)
       callbacks:
         change: @onScriptChanged
       nodeClasses:

From 0c45b6c0e3fca4c6f6152f4dcaaf05fb605afa4c Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 3 Mar 2014 12:21:59 -0800
Subject: [PATCH 049/178] Fixes for ladder refreshing.

---
 app/views/play/ladder/ladder_tab.coffee |  7 +++++--
 app/views/play/ladder_view.coffee       | 13 ++++++-------
 2 files changed, 11 insertions(+), 9 deletions(-)

diff --git a/app/views/play/ladder/ladder_tab.coffee b/app/views/play/ladder/ladder_tab.coffee
index 96e268e3c..4d236fe9e 100644
--- a/app/views/play/ladder/ladder_tab.coffee
+++ b/app/views/play/ladder/ladder_tab.coffee
@@ -24,14 +24,17 @@ module.exports = class LadderView extends CocoView
     super(options)
     @teams = teamDataFromLevel @level
     @leaderboards = {}
+    @refreshLadder()
+
+  refreshLadder: ->
     for team in @teams
+      @leaderboards[team.id]?.off 'sync'
 #      teamSession = _.find @sessions.models, (session) -> session.get('team') is team.id
       teamSession = null
 #      console.log "Team session: #{JSON.stringify teamSession}"
       @leaderboards[team.id] = new LeaderboardData(@level, team.id, teamSession)
       @leaderboards[team.id].once 'sync', @onLeaderboardLoaded, @
 
-  onChallengersLoaded: -> @renderMaybe()
   onLeaderboardLoaded: -> @renderMaybe()
 
   renderMaybe: ->
@@ -48,7 +51,7 @@ module.exports = class LadderView extends CocoView
     team.leaderboard = @leaderboards[team.id] for team in @teams
     ctx.levelID = @levelID
     ctx
-    
+
 class LeaderboardData
   constructor: (@level, @team, @session) ->
     _.extend @, Backbone.Events
diff --git a/app/views/play/ladder_view.coffee b/app/views/play/ladder_view.coffee
index bc4f0fd6d..5bc430eaa 100644
--- a/app/views/play/ladder_view.coffee
+++ b/app/views/play/ladder_view.coffee
@@ -65,17 +65,16 @@ module.exports = class LadderView extends RootView
     @insertSubView(@ladderTab = new LadderTabView({}, @level, @sessions))
     @insertSubView(@myMatchesTab = new MyMatchesTabView({}, @level, @sessions))
     setInterval(@fetchSessionsAndRefreshViews.bind(@), 10000)
-  
+
   fetchSessionsAndRefreshViews: ->
     @sessions.fetch({"success": @refreshViews})
-    
+
   refreshViews: =>
-    @ladderTab.constructor({}, @level, @sessions)
+    @ladderTab.refreshLadder()
     @myMatchesTab.refreshMatches()
     console.log "refreshed views!"
-    
-    
-    
+
+
   # Simulations
 
   onSimulateAllButtonClick: (e) ->
@@ -116,4 +115,4 @@ module.exports = class LadderView extends RootView
     teamID = button.data('team')
     session = (s for s in @sessions.models when s.get('team') is teamID)[0]
     modal = new LadderPlayModal({}, @level, session, teamID)
-    @openModalView modal
\ No newline at end of file
+    @openModalView modal

From 05b4f536c10fc2811ece438e825a85b569ebc1ec Mon Sep 17 00:00:00 2001
From: Ruben Vereecken <ruben.vereecken@student.uantwerpen.be>
Date: Mon, 3 Mar 2014 21:22:04 +0100
Subject: [PATCH 050/178] Fixed a link to point correctly

---
 app/templates/editor/article/edit.jade | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/templates/editor/article/edit.jade b/app/templates/editor/article/edit.jade
index e7dff7cd7..7c6a7136a 100644
--- a/app/templates/editor/article/edit.jade
+++ b/app/templates/editor/article/edit.jade
@@ -6,7 +6,7 @@ block content
       li
         a(href="/editor", data-i18n="editor.main_title") CodeCombat Editors
       li
-        a(href="/editor/thang", data-i18n="editor.article_title") Article Editor
+        a(href="/editor/article", data-i18n="editor.article_title") Article Editor
       li.active
         | #{article.attributes.name}
 

From 0c4a2a2f2b8b58be64889d6bb371840f6b4eadf0 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 3 Mar 2014 12:27:32 -0800
Subject: [PATCH 051/178] Multiplied visible scores by 100 for more awesome.

---
 app/templates/play/ladder/ladder_tab.jade | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/templates/play/ladder/ladder_tab.jade b/app/templates/play/ladder/ladder_tab.jade
index eed7e9d2c..c3090caa3 100644
--- a/app/templates/play/ladder/ladder_tab.jade
+++ b/app/templates/play/ladder/ladder_tab.jade
@@ -14,7 +14,7 @@ div#columns.row
         for session in team.leaderboard.topPlayers.models
           - var myRow = session.get('creator') == me.id
           tr(class=myRow ? "success" : "")
-            td.score-cell= session.get('totalScore').toFixed(2)
+            td.score-cell= Math.round(session.get('totalScore') * 100)
             td.name-col-cell= session.get('creatorName') || "Anonymous"
             td
               a(href="/play/level/#{level.get('slug') || level.id}/?team=#{team.otherTeam}&opponent=#{session.id}")

From 461d66c795e0fb38852bcdf7815b9502cec6798a Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 3 Mar 2014 13:02:34 -0800
Subject: [PATCH 052/178] Typo in my matches team ranking.

---
 app/templates/play/ladder/my_matches_tab.jade | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/templates/play/ladder/my_matches_tab.jade b/app/templates/play/ladder/my_matches_tab.jade
index 310beca34..d2b543478 100644
--- a/app/templates/play/ladder/my_matches_tab.jade
+++ b/app/templates/play/ladder/my_matches_tab.jade
@@ -43,7 +43,7 @@ div#columns.row
             td.time-cell= match.when
             td.battle-cell
               - var text = match.state === 'win' ? 'Watch your victory' : 'Defeat the ' + team.otherTeam
-              a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{match.sessionID}")= text
+              a(href="/play/level/#{levelID}?team=#{team.id}&opponent=#{match.sessionID}")= text
 
         if !team.matches.length
           tr

From d66a9d6be03d42f3730b8e47c0f567f63dd0fe87 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Mon, 3 Mar 2014 13:21:05 -0800
Subject: [PATCH 053/178] Set up a generic choose tab for root views so
 hash->tab behavior is baked in.

---
 app/views/kinds/RootView.coffee         | 9 ++++++++-
 app/views/play/ladder/ladder_tab.coffee | 2 +-
 2 files changed, 9 insertions(+), 2 deletions(-)

diff --git a/app/views/kinds/RootView.coffee b/app/views/kinds/RootView.coffee
index 6dbc36be0..b27ebb92f 100644
--- a/app/views/kinds/RootView.coffee
+++ b/app/views/kinds/RootView.coffee
@@ -38,8 +38,15 @@ module.exports = class RootView extends CocoView
     location.hash = ''
     location.hash = hash
     @buildLanguages()
+    
+  afterRender: ->
+    super(arguments...)
+    @chooseTab(location.hash.replace('#','')) if location.hash
 
-    # TODO: automate tabs to put in hashes and navigate to them here
+  chooseTab: (category) ->
+    $("a[href='##{category}']", @$el).tab('show')
+
+  # TODO: automate tabs to put in hashes when they are clicked
 
   buildLanguages: ->
     $select = @$el.find(".language-dropdown").empty()
diff --git a/app/views/play/ladder/ladder_tab.coffee b/app/views/play/ladder/ladder_tab.coffee
index 4d236fe9e..323ca86fe 100644
--- a/app/views/play/ladder/ladder_tab.coffee
+++ b/app/views/play/ladder/ladder_tab.coffee
@@ -15,7 +15,7 @@ class LevelSessionsCollection extends CocoCollection
     super()
     @url = "/db/level/#{levelID}/all_sessions"
 
-module.exports = class LadderView extends CocoView
+module.exports = class LadderTabView extends CocoView
   id: 'ladder-tab-view'
   template: require 'templates/play/ladder/ladder_tab'
   startsLoading: true

From 0aa8e7bb7d4d4219eb5ad2242dc4ee06f0bc6b39 Mon Sep 17 00:00:00 2001
From: Ruben Vereecken <ruben.vereecken@student.uantwerpen.be>
Date: Mon, 3 Mar 2014 22:21:56 +0100
Subject: [PATCH 054/178] Added owner-based save enabling on Thang Types. NOTE
 server-sided doesn't work yet

---
 app/templates/editor/thang/edit.jade            | 9 ++++-----
 app/views/editor/article/edit.coffee            | 2 +-
 app/views/editor/level/scripts_tab_view.coffee  | 2 +-
 app/views/editor/level/settings_tab_view.coffee | 1 +
 app/views/editor/level/systems_tab_view.coffee  | 1 +
 app/views/editor/thang/colors_tab_view.coffee   | 1 +
 app/views/editor/thang/edit.coffee              | 2 ++
 7 files changed, 11 insertions(+), 7 deletions(-)

diff --git a/app/templates/editor/thang/edit.jade b/app/templates/editor/thang/edit.jade
index c7b80fea0..af30eb2c1 100644
--- a/app/templates/editor/thang/edit.jade
+++ b/app/templates/editor/thang/edit.jade
@@ -12,9 +12,8 @@ block content
 
   img#portrait.img-thumbnail
 
-  button.btn.btn-primary#save-button(data-toggle="coco-modal", data-target="modal/save_version")
-    | Save
-  button.btn.btn-primary#revert-button(data-toggle="coco-modal", data-target="modal/revert", data-i18n="revert.revert") Revert
+  button.btn.btn-primary#save-button(data-toggle="coco-modal", data-target="modal/save_version", disabled=authorized === true ? undefined : "true") Save
+  button.btn.btn-primary#revert-button(data-toggle="coco-modal", data-target="modal/revert", data-i18n="revert.revert", disabled=authorized === true ? undefined : "true") Revert
   
   h3 Edit Thang Type: "#{thangType.attributes.name}"
 
@@ -38,9 +37,9 @@ block content
           select#animations-select
             for animation in animations
               option #{animation}
-          button.btn.btn-small.btn-primary#upload-button
+          button(disabled=authorized === true ? undefined : "true").btn.btn-small.btn-primary#upload-button
             i.icon-upload
-          button.btn.btn-small.btn-primary#clear-button
+          button(disabled=authorized === true ? undefined : "true").btn.btn-small.btn-primary#clear-button
             i.icon-remove
           input#real-upload-button(type="file")
 
diff --git a/app/views/editor/article/edit.coffee b/app/views/editor/article/edit.coffee
index d3c2d4c1c..77f95ac40 100644
--- a/app/views/editor/article/edit.coffee
+++ b/app/views/editor/article/edit.coffee
@@ -37,9 +37,9 @@ module.exports = class ArticleEditView extends View
       data: data
       filePath: "db/thang.type/#{@article.get('original')}"
       schema: Article.schema.attributes
+      readOnly: true unless me.isAdmin() or @article.hasWriteAccess(me)
       callbacks:
         change: @pushChangesToPreview
-    options.readOnly = true unless me.isAdmin()
     @treema = @$el.find('#article-treema').treema(options)
 
     @treema.build()
diff --git a/app/views/editor/level/scripts_tab_view.coffee b/app/views/editor/level/scripts_tab_view.coffee
index 9da71e31c..f0088ad21 100644
--- a/app/views/editor/level/scripts_tab_view.coffee
+++ b/app/views/editor/level/scripts_tab_view.coffee
@@ -59,7 +59,7 @@ module.exports = class ScriptsTabView extends View
       thangIDs: thangIDs
       dimensions: @dimensions
       supermodel: @supermodel
-      readOnly: not me.isAdmin() and not @level.hasWriteAccess(me)
+      readOnly: true unless me.isAdmin() or @level.hasWriteAccess(me)
       callbacks:
         change: @onScriptChanged
       nodeClasses:
diff --git a/app/views/editor/level/settings_tab_view.coffee b/app/views/editor/level/settings_tab_view.coffee
index 2dbfcf165..4e2518be6 100644
--- a/app/views/editor/level/settings_tab_view.coffee
+++ b/app/views/editor/level/settings_tab_view.coffee
@@ -29,6 +29,7 @@ module.exports = class SettingsTabView extends View
       supermodel: @supermodel
       schema: schema
       data: data
+      readOnly: true unless me.isAdmin() or @level.hasWriteAccess(me)
       callbacks: {change: @onSettingsChanged}
       thangIDs: thangIDs
       nodeClasses:
diff --git a/app/views/editor/level/systems_tab_view.coffee b/app/views/editor/level/systems_tab_view.coffee
index 23a58617a..a4b481754 100644
--- a/app/views/editor/level/systems_tab_view.coffee
+++ b/app/views/editor/level/systems_tab_view.coffee
@@ -69,6 +69,7 @@ module.exports = class SystemsTabView extends View
       supermodel: @supermodel
       schema: Level.schema.get('properties').systems
       data: systems
+      readOnly: true unless me.isAdmin() or @level.hasWriteAccess(me)
       callbacks:
         change: @onSystemsChanged
         select: @onSystemSelected
diff --git a/app/views/editor/thang/colors_tab_view.coffee b/app/views/editor/thang/colors_tab_view.coffee
index 4f42a15fc..b1ba229dc 100644
--- a/app/views/editor/thang/colors_tab_view.coffee
+++ b/app/views/editor/thang/colors_tab_view.coffee
@@ -115,6 +115,7 @@ module.exports = class ColorsTabView extends CocoView
     treemaOptions =
       data: data
       schema: schema
+      readOnly: true unless me.isAdmin() or @thangType.hasWriteAccess(me)
       callbacks:
         change: @onColorGroupsChanged
         select: @onColorGroupSelected
diff --git a/app/views/editor/thang/edit.coffee b/app/views/editor/thang/edit.coffee
index de0cc062e..a8310cdf2 100644
--- a/app/views/editor/thang/edit.coffee
+++ b/app/views/editor/thang/edit.coffee
@@ -57,6 +57,7 @@ module.exports = class ThangTypeEditView extends View
     context = super(context)
     context.thangType = @thangType
     context.animations = @getAnimationNames()
+    context.authorized = me.isAdmin() or @thangType.hasWriteAccess(me)
     context
 
   getAnimationNames: ->
@@ -328,6 +329,7 @@ module.exports = class ThangTypeEditView extends View
       schema: schema
       files: @files
       filePath: "db/thang.type/#{@thangType.get('original')}"
+      readOnly: true unless me.isAdmin() or @thangType.hasWriteAccess(me)
       callbacks:
         change: @pushChangesToPreview
         select: @onSelectNode

From b7fbf67e8946f7bb193986d4c823b1d7b01a9719 Mon Sep 17 00:00:00 2001
From: Ruben Vereecken <ruben.vereecken@student.uantwerpen.be>
Date: Mon, 3 Mar 2014 22:52:37 +0100
Subject: [PATCH 055/178] Enabled Fork button again. Previous commits solve
 issue #455

---
 app/templates/editor/level/edit.jade | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/templates/editor/level/edit.jade b/app/templates/editor/level/edit.jade
index e63cdd2b9..bbf0026d5 100644
--- a/app/templates/editor/level/edit.jade
+++ b/app/templates/editor/level/edit.jade
@@ -31,7 +31,7 @@ block outer_content
           ul.nav.navbar-nav.navbar-right
             li(data-toggle="coco-modal", data-target="modal/revert", data-i18n="revert.revert", disabled=authorized === true ? undefined : "true").btn.btn-primary.navbar-btn#revert-button Revert
             li(data-i18n="common.save", disabled=authorized === true ? undefined : "true").btn.btn-primary.navbar-btn#commit-level-start-button Save
-            li(data-i18n="common.fork", disabled=authorized === true ? undefined : "true").btn.btn-primary.navbar-btn#fork-level-start-button Fork
+            li(data-i18n="common.fork").btn.btn-primary.navbar-btn#fork-level-start-button Fork
             li(title="⌃↩ or ⌘↩: Play preview of current level", data-i18n="common.play")#play-button.btn.btn-inverse.banner.navbar-btn Play!
 
             li.divider

From e231952fcf1e8dab32eaf8d64c935c13a2a060bf Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Mon, 3 Mar 2014 14:17:36 -0800
Subject: [PATCH 056/178] Added the jquery minicolors image.

---
 app/assets/images/jquery.minicolors.png | Bin 0 -> 77459 bytes
 1 file changed, 0 insertions(+), 0 deletions(-)
 create mode 100644 app/assets/images/jquery.minicolors.png

diff --git a/app/assets/images/jquery.minicolors.png b/app/assets/images/jquery.minicolors.png
new file mode 100644
index 0000000000000000000000000000000000000000..8fa1e9d9062696b96641450a0b0dcc8669835782
GIT binary patch
literal 77459
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
JPDHLkV1o8UQl|g_

literal 0
HcmV?d00001


From c944e78271b60d1f60891d7548d02e618334d149 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 3 Mar 2014 14:59:39 -0800
Subject: [PATCH 057/178] Ladder views now show level descvription content if
 present.

---
 app/templates/play/ladder.jade    | 5 ++++-
 app/views/play/ladder_view.coffee | 1 +
 2 files changed, 5 insertions(+), 1 deletion(-)

diff --git a/app/templates/play/ladder.jade b/app/templates/play/ladder.jade
index 41360c85b..b7de9c539 100644
--- a/app/templates/play/ladder.jade
+++ b/app/templates/play/ladder.jade
@@ -2,7 +2,10 @@ extends /templates/base
 block content
 
   div#level-column
-    h1= level.get('name')
+    if levelDescription
+      div!= levelDescription
+    else
+      h1= level.get('name')
 
   if me.get('anonymous')
     div#must-log-in
diff --git a/app/views/play/ladder_view.coffee b/app/views/play/ladder_view.coffee
index 5bc430eaa..8f6a046de 100644
--- a/app/views/play/ladder_view.coffee
+++ b/app/views/play/ladder_view.coffee
@@ -57,6 +57,7 @@ module.exports = class LadderView extends RootView
     ctx.simulationStatus = @simulationStatus
     ctx.teams = @teams
     ctx.levelID = @levelID
+    ctx.levelDescription = marked(@level.get('description')) if @level.get('description')
     ctx
 
   afterRender: ->

From df95e8c784422c7f53689793e58987d98f79cd8a Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 3 Mar 2014 16:19:35 -0800
Subject: [PATCH 058/178] Experimenting with using compress() in production.

---
 server/commons/Handler.coffee | 2 +-
 server_setup.coffee           | 4 ++++
 2 files changed, 5 insertions(+), 1 deletion(-)

diff --git a/server/commons/Handler.coffee b/server/commons/Handler.coffee
index 2f81d0acd..ada7acad3 100644
--- a/server/commons/Handler.coffee
+++ b/server/commons/Handler.coffee
@@ -261,7 +261,7 @@ module.exports = class Handler
     tv4 = require('tv4').tv4
     res = tv4.validateMultiple(input, @jsonSchema)
     res
-    
+
   @isID: (id) -> _.isString(id) and id.length is 24 and id.match(/[a-z0-9]/gi)?.length is 24
 
   getDocumentForIdOrSlug: (idOrSlug, done) ->
diff --git a/server_setup.coffee b/server_setup.coffee
index ad23a6bf1..ff04b7b15 100644
--- a/server_setup.coffee
+++ b/server_setup.coffee
@@ -47,6 +47,8 @@ setupExpressMiddleware = (app) ->
   app.use(express.bodyParser())
   app.use(express.methodOverride())
   app.use(express.cookieSession({secret:'defenestrate'}))
+  if config.isProduction
+    app.use(express.compress())
 
 setupPassportMiddleware = (app) ->
   app.use(authentication.initialize())
@@ -115,3 +117,5 @@ exports.setExpressConfigurationOptions = (app) ->
   app.set('views', __dirname + '/app/views')
   app.set('view engine', 'jade')
   app.set('view options', { layout: false })
+  app.set('env', if config.isProduction then 'production' else 'development')
+  app.set('json spaces', 0)

From 873c560a064090b171ab9cee4e739923832aa4da Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 3 Mar 2014 16:22:50 -0800
Subject: [PATCH 059/178] Added Alexandru's Bubble Sort Bootcamp Battle to
 /play.

---
 app/views/play_view.coffee | 7 +++++++
 1 file changed, 7 insertions(+)

diff --git a/app/views/play_view.coffee b/app/views/play_view.coffee
index 32e6d3bf1..9088f1cde 100644
--- a/app/views/play_view.coffee
+++ b/app/views/play_view.coffee
@@ -154,6 +154,13 @@ module.exports = class PlayView extends View
         image: '/file/db/level/526fd3043c637ece50001bb2/the_herd_icon.png'
         description: "Strike at the weak point in an array of enemies. - by Aftermath"
       }
+      {
+        name: 'Bubble Sort Bootcamp Battle'
+        difficulty: 3
+        id: 'bubble-sort-bootcamp-battle'
+        image: '/file/db/level/525ef8ef06e1ab0962000003/commanding_followers_icon.png'
+        description: "Write a bubble sort to organize your soldiers. - by Alexandru"
+      }
       {
         name: 'Enemy Artillery'
         difficulty: 1

From 619b6286619a10018ad3dddf091122a536e9b593 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Mon, 3 Mar 2014 16:39:56 -0800
Subject: [PATCH 060/178] Reworked the victory modal styling a bit.

---
 app/styles/play/level/modal/victory.sass    | 51 ++++++++++-----------
 app/templates/play/level/modal/victory.jade |  6 +--
 2 files changed, 27 insertions(+), 30 deletions(-)

diff --git a/app/styles/play/level/modal/victory.sass b/app/styles/play/level/modal/victory.sass
index 5e2b9e727..1dc9221f5 100644
--- a/app/styles/play/level/modal/victory.sass
+++ b/app/styles/play/level/modal/victory.sass
@@ -1,25 +1,35 @@
 #level-victory-modal
+  .victory-banner
+    float: right
+    width: 150px
+    position: relative
+    
+  .modal-footer
+    clear: both
+    padding-top: 15px
+  
   p.sign-up-poke
     text-align: left
     margin-bottom: 10px
-
-  .sign-up-button
-    margin-left: 20px
-
-  .next-level-button
-    margin-bottom: 10px
-    margin-right: 30px
-    float: right
     
+  .sign-up-button
+    float: right
+    margin-left: 10px
+    
+  .next-level-button
+    float: right
+    margin-left: 10px
+
   .rating
-    float: center
-    margin-right: 22%
+    float: left
+    position: relative
+    top: 5px
     span
       margin-right: 5px
     i
       cursor: pointer
       padding: 2px
-      
+
   .review
     margin-top: 5px
     width: 100%
@@ -31,23 +41,10 @@
       width: 100%
       height: 80px
       box-sizing: border-box
-    
+
   .share-buttons
+    padding-top: 15px
     clear: both
 
   .modal-header
-    text-align: left
-
-  .victory-banner
-    width: 450px
-    position: absolute
-    left: -150px
-
-  .modal-dialog
-    margin-left: 30%
-    padding-left: 300px
-    width: 700px
-
-  .modal-footer
-    margin: 0px
-    padding: 0px
+    text-align: center
diff --git a/app/templates/play/level/modal/victory.jade b/app/templates/play/level/modal/victory.jade
index 7a0637f8a..5dca8f3f6 100644
--- a/app/templates/play/level/modal/victory.jade
+++ b/app/templates/play/level/modal/victory.jade
@@ -2,8 +2,6 @@
 
 .modal-dialog
 
-  img.victory-banner(src="/images/level/victory.png", alt="")
-
   .modal-header
     button(type='button', data-dismiss="modal", aria-hidden="true").close &times;
     h3
@@ -11,7 +9,9 @@
       span= levelName
       span(data-i18n="play_level.victory_title_suffix") Complete
 
-  .modal-body!= body
+  .modal-body
+    img.victory-banner(src="/images/level/victory.png", alt="")
+    div!= body
   
   .modal-footer
     if hasNextLevel

From 7b25eeec47bea9bbaf81ed99d95d8d958be21020 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 3 Mar 2014 16:45:03 -0800
Subject: [PATCH 061/178] HUD properties now flow into columns more evenly,
 have max styling.

---
 app/templates/play/level/hud_prop.jade |  1 +
 app/views/play/level/hud_view.coffee   | 11 ++++++-----
 2 files changed, 7 insertions(+), 5 deletions(-)

diff --git a/app/templates/play/level/hud_prop.jade b/app/templates/play/level/hud_prop.jade
index 017129bb7..fcccab38e 100644
--- a/app/templates/play/level/hud_prop.jade
+++ b/app/templates/play/level/hud_prop.jade
@@ -7,5 +7,6 @@
   if hasBar
     span.prop-value.bar-prop
       .bar
+    span.prop-value.bar-prop-value
   else
     span.prop-value
diff --git a/app/views/play/level/hud_view.coffee b/app/views/play/level/hud_view.coffee
index 79aad4e7c..c3c35d34a 100644
--- a/app/views/play/level/hud_view.coffee
+++ b/app/views/play/level/hud_view.coffee
@@ -135,17 +135,17 @@ module.exports = class HUDView extends View
     props = @$el.find('.thang-props')
     props.find(":not(.thang-name)").remove()
     props.find('.thang-name').text(if @thang.type then "#{@thang.id} - #{@thang.type}" else @thang.id)
-    column = null
-    for prop in @thang.hudProperties ? []
+    propNames = @thang.hudProperties ? []
+    nColumns = Math.ceil propNames.length / 5
+    columns = ($('<div class="thang-props-column"></div>').appendTo(props) for i in [0 ... nColumns])
+    for prop, i in propNames
       continue if prop is 'action'
       pel = @createPropElement prop
       continue unless pel?
       if pel.find('.bar').is('*') and props.find('.bar').is('*')
         props.find('.bar-prop').last().after pel  # Keep bars together
       else
-        column ?= $('<div class="thang-props-column"></div>').appendTo props
-        column.append pel
-        column = null if column.find('.prop').length is 5
+        columns[i % nColumns].append pel
     null
 
   createActions: ->
@@ -263,6 +263,7 @@ module.exports = class HUDView extends View
       labelText = prop + ": " + @formatValue(prop, val) + " / " + @formatValue(prop, max)
       if regen
         labelText += " (+" + @formatValue(prop, regen) + "/s)"
+      pel.find('.bar-prop-value').text(Math.round(max)) if max
     else
       s = @formatValue(prop, val)
       labelText = "#{prop}: #{s}"

From 65d9186ef11a32b1eede4f2d03af51549543b08c Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 3 Mar 2014 16:55:36 -0800
Subject: [PATCH 062/178] Fixed actions showing up under dialogue bubbles on
 repeated toggling.

---
 app/views/play/level/hud_view.coffee | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/views/play/level/hud_view.coffee b/app/views/play/level/hud_view.coffee
index c3c35d34a..f4580660b 100644
--- a/app/views/play/level/hud_view.coffee
+++ b/app/views/play/level/hud_view.coffee
@@ -151,7 +151,7 @@ module.exports = class HUDView extends View
   createActions: ->
     actions = @$el.find('.thang-actions tbody').empty()
     showActions = @thang.world and not _.isEmpty(@thang.actions) and 'action' in @thang.hudProperties ? []
-    @$el.find('.thang-actions').toggle showActions
+    @$el.find('.thang-actions').toggleClass 'secret', showActions
     return unless showActions
     @buildActionTimespans()
     for actionName, action of @thang.actions

From 881ec1a4b344c3c8427a712486300c404faff2b9 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Mon, 3 Mar 2014 17:03:02 -0800
Subject: [PATCH 063/178] Added a fix for files with spaces.

---
 server/routes/file.coffee | 1 +
 1 file changed, 1 insertion(+)

diff --git a/server/routes/file.coffee b/server/routes/file.coffee
index ce9f4ca7c..7a16c3709 100644
--- a/server/routes/file.coffee
+++ b/server/routes/file.coffee
@@ -13,6 +13,7 @@ module.exports.setup = (app) ->
 
 fileGet = (req, res) ->
   path = req.path[6..]
+  path = decodeURI path
   isFolder = false
   try
     objectId = mongoose.Types.ObjectId(path)

From 246bbefc912e0ba56ae13741fd3f2e8605851503 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Mon, 3 Mar 2014 17:14:13 -0800
Subject: [PATCH 064/178] Made the file uploading just overwrite files with the
 same name.

---
 app/treema-ext.coffee | 7 +++----
 1 file changed, 3 insertions(+), 4 deletions(-)

diff --git a/app/treema-ext.coffee b/app/treema-ext.coffee
index 9112abaa6..674e6215a 100644
--- a/app/treema-ext.coffee
+++ b/app/treema-ext.coffee
@@ -39,6 +39,7 @@ class LiveEditingMarkup extends TreemaNode.nodeMap.ace
       filename: InkBlob.filename
       mimetype: InkBlob.mimetype
       path: @settings.filePath
+      force: true
 
     @uploadingPath = [@settings.filePath, InkBlob.filename].join('/')
     $.ajax('/file', { type: 'POST', data: body, success: @onFileUploaded })
@@ -147,9 +148,8 @@ class SoundFileTreema extends TreemaNode.nodeMap.string
       filename: InkBlob.filename
       mimetype: InkBlob.mimetype
       path: @settings.filePath
+      force: true
       
-    # Automatically overwrite if the same path was put in here before
-    body.force = true # if InkBlob.filename is @data
     @uploadingPath = [@settings.filePath, InkBlob.filename].join('/')
     $.ajax('/file', { type: 'POST', data: body, success: @onFileUploaded })
 
@@ -185,9 +185,8 @@ class ImageFileTreema extends TreemaNode.nodeMap.string
       filename: InkBlob.filename
       mimetype: InkBlob.mimetype
       path: @settings.filePath
+      force: true
 
-    # Automatically overwrite if the same path was put in here before
-    body.force = true # if InkBlob.filename is @data
     @uploadingPath = [@settings.filePath, InkBlob.filename].join('/')
     $.ajax('/file', { type: 'POST', data: body, success: @onFileUploaded })
 

From 113c71e141be7d8debbc6d63445f2eda298b2472 Mon Sep 17 00:00:00 2001
From: George Saines <gsaines@gmail.com>
Date: Mon, 3 Mar 2014 18:03:54 -0800
Subject: [PATCH 065/178] adding new hud icons

---
 app/assets/images/level/hud_info_icons.png | Bin 0 -> 8571 bytes
 1 file changed, 0 insertions(+), 0 deletions(-)
 create mode 100644 app/assets/images/level/hud_info_icons.png

diff --git a/app/assets/images/level/hud_info_icons.png b/app/assets/images/level/hud_info_icons.png
new file mode 100644
index 0000000000000000000000000000000000000000..bff08711922f2cee437293e67171e84d08edb0db
GIT binary patch
literal 8571
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
BqW=H@

literal 0
HcmV?d00001


From 9aa1d5a26ccad8c192d3722e7e36fc7d6407cf69 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 3 Mar 2014 18:50:59 -0800
Subject: [PATCH 066/178] Okay, we'll just let Cloudflare compress things.

---
 server_setup.coffee | 3 +--
 1 file changed, 1 insertion(+), 2 deletions(-)

diff --git a/server_setup.coffee b/server_setup.coffee
index ff04b7b15..d15035c53 100644
--- a/server_setup.coffee
+++ b/server_setup.coffee
@@ -47,8 +47,7 @@ setupExpressMiddleware = (app) ->
   app.use(express.bodyParser())
   app.use(express.methodOverride())
   app.use(express.cookieSession({secret:'defenestrate'}))
-  if config.isProduction
-    app.use(express.compress())
+  #app.use(express.compress()) if config.isProduction  # just let Cloudflare do it
 
 setupPassportMiddleware = (app) ->
   app.use(authentication.initialize())

From d43ff5d21729a85cf33dcd966e6d5915b78962a2 Mon Sep 17 00:00:00 2001
From: Ruben Vereecken <ruben.vereecken@student.uantwerpen.be>
Date: Tue, 4 Mar 2014 11:24:38 +0100
Subject: [PATCH 067/178] Added an anonymous check on the Fork button

---
 app/templates/editor/level/edit.jade | 2 +-
 app/views/editor/level/edit.coffee   | 1 +
 2 files changed, 2 insertions(+), 1 deletion(-)

diff --git a/app/templates/editor/level/edit.jade b/app/templates/editor/level/edit.jade
index bbf0026d5..7d2c7ac66 100644
--- a/app/templates/editor/level/edit.jade
+++ b/app/templates/editor/level/edit.jade
@@ -31,7 +31,7 @@ block outer_content
           ul.nav.navbar-nav.navbar-right
             li(data-toggle="coco-modal", data-target="modal/revert", data-i18n="revert.revert", disabled=authorized === true ? undefined : "true").btn.btn-primary.navbar-btn#revert-button Revert
             li(data-i18n="common.save", disabled=authorized === true ? undefined : "true").btn.btn-primary.navbar-btn#commit-level-start-button Save
-            li(data-i18n="common.fork").btn.btn-primary.navbar-btn#fork-level-start-button Fork
+            li(data-i18n="common.fork", disabled=anonymous ? "true": undefined).btn.btn-primary.navbar-btn#fork-level-start-button Fork
             li(title="⌃↩ or ⌘↩: Play preview of current level", data-i18n="common.play")#play-button.btn.btn-inverse.banner.navbar-btn Play!
 
             li.divider
diff --git a/app/views/editor/level/edit.coffee b/app/views/editor/level/edit.coffee
index c79439800..962db3fac 100644
--- a/app/views/editor/level/edit.coffee
+++ b/app/views/editor/level/edit.coffee
@@ -64,6 +64,7 @@ module.exports = class EditorLevelView extends View
     context = super(context)
     context.level = @level
     context.authorized = me.isAdmin() or @level.hasWriteAccess(me)
+    context.anonymous = me.get('anonymous')
     context
 
   afterRender: ->

From 2f653c966e7775648fba3efc57100d3b7cff4762 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Tue, 4 Mar 2014 08:30:50 -0800
Subject: [PATCH 068/178] Simulator now swaps in empty methods when code
 doesn't compile.

---
 app/lib/simulator/Simulator.coffee | 7 ++++++-
 1 file changed, 6 insertions(+), 1 deletion(-)

diff --git a/app/lib/simulator/Simulator.coffee b/app/lib/simulator/Simulator.coffee
index 521e810c7..f92bccb98 100644
--- a/app/lib/simulator/Simulator.coffee
+++ b/app/lib/simulator/Simulator.coffee
@@ -207,7 +207,12 @@ module.exports = class Simulator
   transpileSpell: (thang, spellKey, methodName) ->
     slugifiedThangID = _.string.slugify thang.id
     source = @currentUserCodeMap[[slugifiedThangID,methodName].join '/'] ? ""
-    @spells[spellKey].thangs[thang.id].aether.transpile source
+    aether = @spells[spellKey].thangs[thang.id].aether
+    try
+      aether.transpile source
+    catch e
+      console.log "Couldn't transpile #{spellKey}:\n#{source}\n", e
+      aether.transpile ''
 
   createAether: (methodName, method) ->
     aetherOptions =

From 195b1941eabe33186e720d6deab92918d436306f Mon Sep 17 00:00:00 2001
From: Yos Riady <yosriady@gmail.com>
Date: Wed, 5 Mar 2014 02:53:54 +0800
Subject: [PATCH 069/178] Added even more names for #53

---
 app/lib/world/names.coffee | 9 ++++++++-
 1 file changed, 8 insertions(+), 1 deletion(-)

diff --git a/app/lib/world/names.coffee b/app/lib/world/names.coffee
index 7ba332f60..a7a3f256e 100644
--- a/app/lib/world/names.coffee
+++ b/app/lib/world/names.coffee
@@ -58,6 +58,9 @@ module.exports.thangNames = thangNames =
     "Nikita"
     "Alana"
     "Lana"
+    "Joan"
+    "Helga"
+    "Annie"
   ]
   "Peasant": [
     "Yorik"
@@ -112,6 +115,8 @@ module.exports.thangNames = thangNames =
     "Roman"
     "Hunter"
     "Simon"
+    "Robin"
+    "Quinn"
   ]
   "Ogre Munchkin M": [
     "Brack"
@@ -151,13 +156,15 @@ module.exports.thangNames = thangNames =
     "Trung"
     "Axe Ox"
     "Vargutt"
+    "Grumus"
+    "Gug"
   ]
   "Ogre F": [
     "Nareng"
     "Morthrug"
     "Glonc"
     "Marghurk"
-    
+    "Martha"
   ]
   "Ogre Brawler": [
     "Grul'thock"

From 4556536df4d2f36052cc718a6313e6547aafb7b1 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Tue, 4 Mar 2014 11:15:31 -0800
Subject: [PATCH 070/178] Hooked up new HUD icons.

---
 app/assets/images/level/info_icons.png     | Bin 646 -> 0 bytes
 app/assets/images/level/prop_health.png    | Bin 513 -> 0 bytes
 app/assets/images/level/prop_inventory.png | Bin 571 -> 0 bytes
 app/assets/images/level/prop_pos.png       | Bin 576 -> 0 bytes
 app/assets/images/level/prop_target.png    | Bin 581 -> 0 bytes
 app/styles/play/level/hud.sass             |  33 +++++++++++++++++----
 app/templates/play/level/hud_prop.jade     |   2 +-
 app/views/play/level/hud_view.coffee       |   2 +-
 8 files changed, 29 insertions(+), 8 deletions(-)
 delete mode 100644 app/assets/images/level/info_icons.png
 delete mode 100644 app/assets/images/level/prop_health.png
 delete mode 100644 app/assets/images/level/prop_inventory.png
 delete mode 100644 app/assets/images/level/prop_pos.png
 delete mode 100644 app/assets/images/level/prop_target.png

diff --git a/app/assets/images/level/info_icons.png b/app/assets/images/level/info_icons.png
deleted file mode 100644
index de9ecfb6f43c4cad12d76ee4f32e8a54781d776c..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001

literal 646
zcmV;10(t$3P)<h;3K|Lk000e1NJLTq001}u000gM1^@s6dZpj40006_Nkl<Zc-pm=
zziSjh6vy{Zu(6Tb%lRWQml$$h62p;$Y_O1sh-e@o7A^!*1(Y-*SS+!y5w^%5a7Eg{
zVPRvll^|H0l^`ewYUQsXf5FT9E;I0)+sPyh4t%(GZ)f-RbMG-T5k^s;X)y&ungTgj
zKB^eYJ8v>ZF{Dw*MJJ3uwJQnJXUGD%M9zKnXsN+uKIA<{uED7H^=bMRPm~}=?u2~9
zk{dAOVBnDJG2|f^f)JqRYxDH<UfW5PVDDZ18vU$|n}?P$a^P3HOGlgACBGL}${g;o
zDo3EZ{hH?Q9{|KsGPC4zAu(i9@jcZ!()6?-fxFkosB`JOi%LQDCB8cH@BOGvfHMBu
zY}3)!<C3p)%MesTP#L-A(mp+V@fIMKl1jFve0k=a&J#e4oK@i@M6hf>U%ME=@7-Gv
zBq2xuRfCurh><hbg<#)Q&dAw<Mp9I=Zv<CoYX3)2CAUc}CYT!}C|fb|4AHuf%eGb~
zAraD|elQn;0WC^4=9+Jgmhh1?DbtdR%xMJ!7MU=?!_~QTZ~X>sbS{rb-JH5W4;C&G
zD!hAVmM{<P{LeP=DQ<e@KE_i4+eC1SnuP`ekjk(vynNU}t`ga%PZ+eHuQfsH+~`N;
zaxo=zkxA^A*Rhk4{m>=(Ii7UM*^-5j8_H{;+eEOSltjDAwv=)n!Bhx3=UXrD+vsyG
z(_+*a1dB)xyInujr^V4s$oVCQuIf2YGj4hn=h4C;h<wH*zTY;7?;$gA9Iic8w5u0>
g*tSGd{pl4t24w6<%WBZ`qyPW_07*qoM6N<$g3yRBAOHXW

diff --git a/app/assets/images/level/prop_health.png b/app/assets/images/level/prop_health.png
deleted file mode 100644
index 87f3c840753e0f8d73b7a4c2fbe736c41918965a..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001

literal 513
zcmV+c0{;DpP)<h;3K|Lk000e1NJLTq000gE000gM0ssI2kSe-{0005VNkl<ZC{vBm
z%T5zf0D$5D+}aMEUWS&op@3n*g1F%cOmyX2=p*=4>dK7^7Z?|4V%SJbG`hjS0wf9)
zr*rAdIdhH+U%>AbKC!<$<Ns&?B_FJ>KmML?r?sa0X4^a;-0YR+*=e(l`GI|TF#NDs
zsEs7PZ{nw?dYUagVuo5v&0x}dSVUWC1#Sn~s?E)`uQrnSi5oroy=+qkq@tGionLM*
zEeu*<6Xdw`W@9XY{k(NS25L^I2o+&M)}?lYKI#lxksq8_61a?9PA8)Yc!jL+0-Ylz
zC}f5c=<8)Jfx}iNTm71F4FgP@)e3!slaSKPQJUV{xs||)&EC(k^gtQRh{&Qx7>i%h
zNY)NF<x#(pfOWiHUlsQfO>;$MiI&u3REl?mpX2z|q(1YN8V_SF{e>&O#cYx6=8Yil
zz>2z}f4(_=k(6KZI}w5!4`c6xY@5>CTje4hjSAIfwTApONsiT@&mm29K*)bvI<{)z
zi;r#Bo^>ME*<8?Ij%LltU3wX6VOrV-4d4iQ)oF=R%dfr|Cf(_zchzl<%iBBM1g%>^
z3FkD>F**n$qjd1MW%*9kT4#$z>4T73MM6m%Oi+IT@y66CmB$XB00000NkvXXu0mjf
Dg0S#3

diff --git a/app/assets/images/level/prop_inventory.png b/app/assets/images/level/prop_inventory.png
deleted file mode 100644
index 37f147521f63e34e9723f8d9b8fa7a0a2d97ed25..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001

literal 571
zcmV-B0>u4^P)<h;3K|Lk000e1NJLTq000gE000gM0ssI2kSe-{00064Nkl<ZC{qQ&
z&ubGw6ae7&=DnFsHtE(hT579Fq!mkR(VoOZFDhacL_PK5A0vYQizmUGg5IQfGKfVg
z2ue+9n>I-|+0D+6o$>o{^GbdD-j<Xyn-<=K5Uh>U+9+X0x#{lz5tZ1Lp)3t=-Pu%1
ze)#-lm`-YOtc~*lm<WOk#2i(|`IMLekdICV-a}p(B1R;DQc2dfWXS=^I<KT8W?MQD
z5ROj=K6qj(v<3jFRDj@uKtTcoc%Nk%A&EvZTND}vF>wfJ%aG-TWXZ%vTcsF7*_f$Y
zYsicTd0~yQA%KjOl&UZW5uLM@NbWq_{?`4D5F`ug%qN|zivlS(MjMMLhy(yuBIT^v
z`>}WNLT%&5a__K@8;diW8>@n4nr6R#AAo?{H`h9?w(y3aT$*p593KxS*;4BodsQ>p
zs6pDh5Z7S)@q=>MgQH!oqYC4{Ua76E)f4IFgzzANf`KYaJJI>+IPIU423d`>5-l=S
zV;wLvULZoJ)h-<Pk|L7hv$RIFx~z|#@G71Pm3bZ%A(>yGB&i;r96f(>fB$c9l-qW@
zrKkB+Ygcb3jk(qJ_4l8456;fNc7Ni^^~>aq^zP-(i?{DSK7I7??ya>~uiuOeMoLPi
zbdm!J6;*I~Wx*MJsX5#451F_**GNa_QUC<$y%!_`05NBI{vWs`3)qrO;z$4h002ov
JPDHLkV1l3N1Tp{s

diff --git a/app/assets/images/level/prop_pos.png b/app/assets/images/level/prop_pos.png
deleted file mode 100644
index d3461a8e259b684e07591c84d4366294e1039667..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001

literal 576
zcmV-G0>Ax<P)<h;3K|Lk000e1NJLTq000gE000gM0ssI2kSe-{00069Nkl<ZC{qQ&
z%T5zf7y#h^|Ie9QJB1FWg$h=YG-zPuLUf_7jSFMq3-|&)k6ZU7x;HT~M2U$pEJV2(
z0#;$L6ldtnoHOU<_u+EC>p8Nan2^1(@|4))Fc^fvP{>awTOVXGBqPJ1Ro03Vp{;X+
zx8`oX$jpOUb&0~uJocyW^p{*U`jY+mu8q(J2od8b?hPL4D-Y9xhwTaCW_wUvPMwMN
zy|A}xiGeX5AxA_3S<+oeM{{#a?(M7D2q1;Q^OqZ`TD(4Usu_<=%{U_fsKtWeeHW|o
zaHDUaQ|VO=-qEpO^L%>2Ay-B@qscnQtt#{1TIy!-tR!%g+ZP;6t4Bo|Nc5c#&H@1k
z0Na#)W2aH96?W_Xlgt)hgZaCcYGrn6uO9CiRWd~FfOV3%cA3m&dm63lxMWQ<m&<y|
zsPVOMqe?$lqy4~x$^h#maqMxO7W(hER=a<$uEI>0<wuLj`)qeTKi$T9;`yF1NB~et
zzr|!*N>9F{3CMhN%8uB{>q)w;iwn+}whR#o0Axs}!Cfeh4|k&GK_q%fl>gfO5ulEI
zpRrIIAaDc#%F4>oxIT}Yo$B8k{~{V5u1C$t0$9hD@r~t-hybABd~zZVXGQTdl%s?n
z=yG4EGv_>QxdWe&gSD)a#N3iHtVWLL#gs?*(TS`s!$2!Si&Q%@7VtlXr5Ox_NvCiC
O0000<MNUMnLSTYC`4A8Q

diff --git a/app/assets/images/level/prop_target.png b/app/assets/images/level/prop_target.png
deleted file mode 100644
index 0f2b411ac11223b0122e32b7d380d7f1fa58ff1d..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001

literal 581
zcmV-L0=oT)P)<h;3K|Lk000e1NJLTq000gE000gM0ssI2kSe-{0006ENkl<ZC{qQ&
zIZqT}00!XqyXM%Lft^_(Yajtk)<rOwXlaE}3MH|#ATerZ?JrPz8f$-mM>H|9z(_)c
z6@`V3ny71xi?A%q%+CECJ`cIL*jA__#b<J}BHd1IwF|UBY82bEJRx#KRl;;l4GX8d
zK-E`L{$Sv27NU^FD(VHWpHLf#b1P;?R`JCvCzQ+1``)T)JUR@T!1he``do2+XiJdC
zbEjoSLQR%3UIBlE=JPqfdu&%QFPg=hV9YFDp)msa#nii5`8$^nkDU7_?vQ45DHj*O
zcN^LJ>9Vw}cIi^s1;iZj1{@zJ@>i|1Nyz1b^6r@TQ0z??BO9F)F%<gV{4t1;wQ?;_
z(dt`J_fO<P!pIIfmtbN@!2Jxe)lCvvHYEh3PkywzsE3G`!S}&v11dm0Mm0cPKrso*
zxS@mrr;vc=AU;?GM)t5!BY)OTzDD*$<{a5%I)#E9`kE7jSsB*V)4#0*4vS~QwZx=^
z^=vsbWeenaFT4?3``%fmY(m-XQP3&*hbW^7kq*RK;$KTWK~S1%2D%ICYuX_C=0CO`
z+bPH)1&f+~X2?ouT}_2FN(*y27H3eqe!4O8?u=an`f{o|@HFWzwvE7gUu-5K0{T#^
z;10+>>HsDMc{?R{LzQd8CR?7XdCw`oJN3Xas?dGN5ujl0MCm6_9(Xz?n5_AK&WZk~
Tn9!vg00000NkvXXu0mjf*)9cL

diff --git a/app/styles/play/level/hud.sass b/app/styles/play/level/hud.sass
index 7858f0365..4f6a5fea3 100644
--- a/app/styles/play/level/hud.sass
+++ b/app/styles/play/level/hud.sass
@@ -95,14 +95,35 @@
         height: 100px
         @include user-select(text)
   
-        .prop
-          img
-            margin-right: 5px
-            width: 16px
-            height: 16px
-  
         .text-prop
           width: 50%
+
+        .prop-label-icon
+          $iconSize: 16px
+          display: inline-block
+          width: $iconSize
+          height: $iconSize
+          margin-right: 5px
+          background: transparent url(/images/level/hud_info_icons.png) no-repeat
+          background-size: auto $iconSize
+          float: left
+
+          &.prop-label-icon-pos
+            background-position-x: -1 * $iconSize
+          &.prop-label-icon-target
+            background-position-x: -2 * $iconSize
+          &.prop-label-icon-inventory
+            background-position-x: -3 * $iconSize
+          &.prop-label-icon-visualRange
+            background-position-x: -4 * $iconSize
+          &.prop-label-icon-attackDamage
+            background-position-x: -5 * $iconSize
+          &.prop-label-icon-attackRange
+            background-position-x: -6 * $iconSize
+          &.prop-label-icon-maxSpeed
+            background-position-x: -7 * $iconSize
+          &.prop-label-icon-gold
+            background-position-x: -8 * $iconSize
   
         .prop-value.bar-prop
           width: 100px
diff --git a/app/templates/play/level/hud_prop.jade b/app/templates/play/level/hud_prop.jade
index fcccab38e..705377d92 100644
--- a/app/templates/play/level/hud_prop.jade
+++ b/app/templates/play/level/hud_prop.jade
@@ -1,6 +1,6 @@
 .prop(name="#{prop}")
   if hasIcon
-    img.prop-label(src="/images/level/prop_#{prop}.png", alt="#{prop}")
+    span(class="prop-label prop-label-icon prop-label-icon-#{prop}")
   else
     span.prop-label #{prop}: 
     
diff --git a/app/views/play/level/hud_view.coffee b/app/views/play/level/hud_view.coffee
index f4580660b..7b036c433 100644
--- a/app/views/play/level/hud_view.coffee
+++ b/app/views/play/level/hud_view.coffee
@@ -247,7 +247,7 @@ module.exports = class HUDView extends View
       return null  # included in the bar
     context =
       prop: prop
-      hasIcon: prop in ["health", "pos", "target", "inventory", "gold"]
+      hasIcon: prop in ["health", "pos", "target", "inventory", "gold", "visualRange", "attackDamage", "attackRange", "maxSpeed"]
       hasBar: prop in ["health"]
     $(prop_template(context))
 

From e02780e4415de899f0d2ccdc5ea1b649d16e3ec7 Mon Sep 17 00:00:00 2001
From: Darredevil <alex.darredevil@gmail.com>
Date: Tue, 4 Mar 2014 21:26:52 +0200
Subject: [PATCH 071/178] Update for ro.coffee

-first 80 lines done
---
 app/locale/ro.coffee | 136 +++++++++++++++++++++----------------------
 1 file changed, 68 insertions(+), 68 deletions(-)

diff --git a/app/locale/ro.coffee b/app/locale/ro.coffee
index 454278097..6a07f8345 100644
--- a/app/locale/ro.coffee
+++ b/app/locale/ro.coffee
@@ -1,83 +1,83 @@
 module.exports = nativeDescription: "limba română", englishDescription: "Romanian", translation:
   common:
     loading: "Loading..."
-#    saving: "Saving..."
-#    sending: "Sending..."
-#    cancel: "Cancel"
-#    save: "Save"
-#    delay_1_sec: "1 second"
-#    delay_3_sec: "3 seconds"
-#    delay_5_sec: "5 seconds"
-#    manual: "Manual"
-#    fork: "Fork"
-#    play: "Play"
+    saving: "Se salvează..."
+    sending: "Se trimite..."
+    cancel: "Anulează"
+    save: "Salvează"
+    delay_1_sec: "1 secundă"
+    delay_3_sec: "3 secunde"
+    delay_5_sec: "5 secunde"
+    manual: "Manual"
+    fork: "Fork"
+    play: "Joaca"
 
-#  modal:
-#    close: "Close"
-#    okay: "Okay"
+  modal:
+    close: "Inchide"
+    okay: "Okay"
 
-#  not_found:
-#    page_not_found: "Page not found"
+  not_found:
+    page_not_found: "Pagina nu a fost gasită"
 
-#  nav:
-#    play: "Levels"
-#    editor: "Editor"
-#    blog: "Blog"
-#    forum: "Forum"
-#    admin: "Admin"
-#    home: "Home"
-#    contribute: "Contribute"
-#    legal: "Legal"
-#    about: "About"
-#    contact: "Contact"
-#    twitter_follow: "Follow"
-#    employers: "Employers"
+  nav:
+    play: "Nivele"
+    editor: "Editor"
+    blog: "Blog"
+    forum: "Forum"
+    admin: "Admin"
+    home: "Acasa"
+    contribute: "Contribuie"
+    legal: "Confidențialitate și termeni"
+    about: "Despre"
+    contact: "Contact"
+    twitter_follow: "Urmărește"
+    employers: "Angajați"
 
-#  versions:
-#    save_version_title: "Save New Version"
-#    new_major_version: "New Major Version"
-#    cla_prefix: "To save changes, first you must agree to our"
-#    cla_url: "CLA"
-#    cla_suffix: "."
-#    cla_agree: "I AGREE"
+  versions:
+    save_version_title: "Salvează noua versiune"
+    new_major_version: "Versiune nouă majoră"
+    cla_prefix: "Pentru a salva modificările mai intâi trebuie sa fiți de acord cu"
+    cla_url: "CLA"
+    cla_suffix: "."
+    cla_agree: "SUNT DE ACORD"
 
-#  login:
-#    sign_up: "Create Account"
-#    log_in: "Log In"
-#    log_out: "Log Out"
-#    recover: "recover account"
+  login:
+    sign_up: "Crează cont"
+    log_in: "Log In"
+    log_out: "Log Out"
+    recover: "recuperează cont"
 
-#  recover:
-#    recover_account_title: "Recover Account"
-#    send_password: "Send Recovery Password"
+  recover:
+    recover_account_title: "Recuperează Cont"
+    send_password: "Trimite parolă de recuperare"
 
-#  signup:
-#    create_account_title: "Create Account to Save Progress"
-#    description: "It's free. Just need a couple things and you'll be good to go:"
-#    email_announcements: "Receive announcements by email"
-#    coppa: "13+ or non-USA "
-#    coppa_why: "(Why?)"
-#    creating: "Creating Account..."
-#    sign_up: "Sign Up"
-#    log_in: "log in with password"
+  signup:
+    create_account_title: "Crează cont pentru a salva progresul"
+    description: "Este gratis. Doar câte ceva inainte si poți continua:"#contextual translation a bit off, could be better# It's free. Just need a couple things and you'll be good to go:"
+    email_announcements: "Receive announcements by email"
+    coppa: "13+ sau non-USA "
+    coppa_why: "(De ce?)"
+    creating: "Se crează contul..."
+    sign_up: "Înscrie-te"
+    log_in: "loghează-te cu parola"
 
-#  home:
-#    slogan: "Learn to Code JavaScript by Playing a Game"
-#    no_ie: "CodeCombat does not run in Internet Explorer 9 or older. Sorry!"
-#    no_mobile: "CodeCombat wasn't designed for mobile devices and may not work!"
-#    play: "Play"
+  home:
+    slogan: "Învață sa scri JavaScript jucându-te"# again sounds funny# Learn to Code JavaScript by Playing a Game"
+    no_ie: "CodeCombat does not run in Internet Explorer 9 or older. Sorry!"
+    no_mobile: "CodeCombat nu a fost proiectat pentru dispozitive mobile si s-ar putea sa nu meargâ!"
+    play: "Joacâ"
 
-#  play:
-#    choose_your_level: "Choose Your Level"
-#    adventurer_prefix: "You can jump to any level below, or discuss the levels on "
-#    adventurer_forum: "the Adventurer forum"
-#    adventurer_suffix: "."
-#    campaign_beginner: "Beginner Campaign"
-#    campaign_beginner_description: "... in which you learn the wizardry of programming."
-#    campaign_dev: "Random Harder Levels"
-#    campaign_dev_description: "... in which you learn the interface while doing something a little harder."
-#    campaign_multiplayer: "Multiplayer Arenas"
-#    campaign_multiplayer_description: "... in which you code head-to-head against other players."
+  play:
+    choose_your_level: "Alege nivelul"
+    adventurer_prefix: "Poți să sari la orice nivel de mai jos"#what do you mean by discuss?# You can jump to any level below, or discuss the levels on "
+    adventurer_forum: "forumul Aventurierului"#sonds waaaaaay too funny#the Adventurer forum"
+    adventurer_suffix: "."
+    campaign_beginner: "Campanie pentru Începători"
+    campaign_beginner_description: "... în care se învață tainele programării."
+    campaign_dev: "Nivele aleatoare mai grele"
+    campaign_dev_description: "... în care se învață interfața, cu o dificultate puțin mai mare."
+    campaign_multiplayer: "Arene Multiplayer"
+    campaign_multiplayer_description: "... în care te lupți cap-la-cap contra alti jucători."
 #    campaign_player_created: "Player-Created"
 #    campaign_player_created_description: "... in which you battle against the creativity of your fellow <a href=\"/contribute#artisan\">Artisan Wizards</a>."
 #    level_difficulty: "Difficulty: "

From 3680d628e7730221c9ee606a69fe6a7e451552aa Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Tue, 4 Mar 2014 12:06:31 -0800
Subject: [PATCH 072/178] Fixed showing generic errors.

---
 app/lib/errors.coffee | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/lib/errors.coffee b/app/lib/errors.coffee
index 372507d0c..8e7fe4203 100644
--- a/app/lib/errors.coffee
+++ b/app/lib/errors.coffee
@@ -21,7 +21,7 @@ module.exports.genericFailure = (jqxhr) ->
   if existingForm[0]
     missingErrors = applyErrorsToForm(existingForm, [error])
     for error in missingErrors
-      existingForm.append($('<div class="alert"></div>').text(error.message))
+      existingForm.append($('<div class="alert alert-danger"></div>').text(error.message))
   else
     res = errorModalTemplate(
       status:jqxhr.status

From 9726840d8a4d2142989d38035639eadb85f96d67 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Tue, 4 Mar 2014 12:21:02 -0800
Subject: [PATCH 073/178] Trying to handle a case where a user did not have
 their user account created.

---
 app/lib/auth.coffee     | 5 +++++
 server/routes/db.coffee | 2 +-
 2 files changed, 6 insertions(+), 1 deletion(-)

diff --git a/app/lib/auth.coffee b/app/lib/auth.coffee
index 2af810b37..df2b2b6c3 100644
--- a/app/lib/auth.coffee
+++ b/app/lib/auth.coffee
@@ -43,7 +43,12 @@ init = ->
   module.exports.me = window.me = new User(storedUser)
   me.url = -> '/auth/whoami'
   me.fetch()
+  
+  retry = -> me.fetch() # blindly try again
+  error = -> setTimeout(retry, 1000) # blindly try again
+  me.on 'error', error, @
   me.on 'sync', ->
+    me.off 'error', error, @ if firstTime
     me.url = -> "/db/user/#{me.id}"
     trackFirstArrival() if firstTime
     if me and not me.get('testGroupNumber')?
diff --git a/server/routes/db.coffee b/server/routes/db.coffee
index 8bf2ef7eb..fec290cfa 100644
--- a/server/routes/db.coffee
+++ b/server/routes/db.coffee
@@ -11,7 +11,7 @@ module.exports.setup = (app) ->
     parts = module.split('/')
     module = parts[0]
     return getSchema(req, res, module) if parts[1] is 'schema'
-    return errors.unauthorized(res, 'Must have an identity to do anything with the db.') unless req.user
+    return errors.unauthorized(res, 'Must have an identity to do anything with the db. Do you have cookies enabled?') unless req.user
 
     try
       moduleName = module.replace '.', '_'

From a0e39480e331d69325ceb7a5d1bcf5b59ce589f4 Mon Sep 17 00:00:00 2001
From: Dominik Kundel <dominik.kundel@gmail.com>
Date: Wed, 5 Mar 2014 02:00:38 +0100
Subject: [PATCH 074/178] new UI design for executing and executed lines in ACE
 editor

---
 app/styles/play/level/tome/spell.sass       | 14 ++++++++++++--
 app/views/play/level/tome/spell_view.coffee |  4 ++--
 2 files changed, 14 insertions(+), 4 deletions(-)

diff --git a/app/styles/play/level/tome/spell.sass b/app/styles/play/level/tome/spell.sass
index 9066b4af5..5fdfa7c61 100644
--- a/app/styles/play/level/tome/spell.sass
+++ b/app/styles/play/level/tome/spell.sass
@@ -73,9 +73,10 @@
       .executing, .executed, .problem-marker-info, .problem-marker-warning, .problem-marker-error
         position: absolute
     .executing
-      background-color: rgba(216, 255, 255, 0.85)
+      background-color: rgba(0, 255, 0, 0.15)
+      @include gradient-striped()
     .executed
-      background-color: rgba(245, 255, 6, 0.18)
+      background-color: rgba(40, 40, 40, 0.12)
     .problem-marker-info
       background-color: rgba(196, 163, 184, 0.25)
     .problem-marker-warning
@@ -83,11 +84,20 @@
     .problem-marker-error
       background-color: rgba(110, 45, 27, 0.25)
 
+    .executing:not(.ace_gutter-cell)
+      background-size: 40px 40px
+      @include animation(progress-bar-stripes 2s linear infinite)
+
     .ace_gutter-cell.executing:not(.ace_error)
       margin-left: 1px
       background-image: url()
       background-position: 0px center
 
+    .ace_gutter-cell.executed:not(.ace_error)
+      margin-left: 1px
+      background-image: url()
+      background-position: 0px center
+
     .ace_gutter-cell.comment-line
       background-position: 0px center
       background-image: url()
diff --git a/app/views/play/level/tome/spell_view.coffee b/app/views/play/level/tome/spell_view.coffee
index c44cfa48e..679ef7d87 100644
--- a/app/views/play/level/tome/spell_view.coffee
+++ b/app/views/play/level/tome/spell_view.coffee
@@ -478,13 +478,13 @@ module.exports = class SpellView extends View
       else
         @debugView.setVariableStates state.variables
         gotVariableStates = true
-        markerType = "text"
+        markerType = "fullLine"
       markerRange = new Range start.row, start.col, end.row, end.col
       markerRange.start = @aceDoc.createAnchor markerRange.start
       markerRange.end = @aceDoc.createAnchor markerRange.end
       markerRange.id = @aceSession.addMarker markerRange, clazz, markerType
       @markerRanges.push markerRange
-      @aceSession.addGutterDecoration start.row, clazz if clazz is 'executing'
+      @aceSession.addGutterDecoration start.row, clazz 
     @debugView.setVariableStates {} unless gotVariableStates
     null
 

From 0ec431f90ac137c382b022309bd753721e2e5111 Mon Sep 17 00:00:00 2001
From: Shrihari <gfxindia@gmail.com>
Date: Wed, 5 Mar 2014 13:46:21 +0530
Subject: [PATCH 075/178] Moved 'skip tutorial' text to locales.

---
 app/locale/ar.coffee                 | 1 +
 app/locale/bg.coffee                 | 1 +
 app/locale/cs.coffee                 | 1 +
 app/locale/da.coffee                 | 1 +
 app/locale/de.coffee                 | 1 +
 app/locale/el.coffee                 | 1 +
 app/locale/en-AU.coffee              | 1 +
 app/locale/en-GB.coffee              | 1 +
 app/locale/en-US.coffee              | 1 +
 app/locale/en.coffee                 | 1 +
 app/locale/es-419.coffee             | 1 +
 app/locale/es-ES.coffee              | 1 +
 app/locale/es.coffee                 | 1 +
 app/locale/fa.coffee                 | 1 +
 app/locale/fi.coffee                 | 1 +
 app/locale/fr.coffee                 | 1 +
 app/locale/he.coffee                 | 1 +
 app/locale/hi.coffee                 | 1 +
 app/locale/hu.coffee                 | 1 +
 app/locale/id.coffee                 | 1 +
 app/locale/it.coffee                 | 1 +
 app/locale/ja.coffee                 | 1 +
 app/locale/ko.coffee                 | 1 +
 app/locale/lt.coffee                 | 1 +
 app/locale/ms-BA.coffee              | 1 +
 app/locale/nb.coffee                 | 1 +
 app/locale/nl.coffee                 | 1 +
 app/locale/nn.coffee                 | 1 +
 app/locale/no.coffee                 | 1 +
 app/locale/pl.coffee                 | 1 +
 app/locale/pt-BR.coffee              | 1 +
 app/locale/pt-PT.coffee              | 1 +
 app/locale/pt.coffee                 | 1 +
 app/locale/ro.coffee                 | 1 +
 app/locale/ru.coffee                 | 1 +
 app/locale/sk.coffee                 | 1 +
 app/locale/sl.coffee                 | 1 +
 app/locale/sr.coffee                 | 1 +
 app/locale/sv.coffee                 | 1 +
 app/locale/th.coffee                 | 1 +
 app/locale/tr.coffee                 | 1 +
 app/locale/uk.coffee                 | 1 +
 app/locale/ur.coffee                 | 1 +
 app/locale/vi.coffee                 | 1 +
 app/locale/zh-HANS.coffee            | 1 +
 app/locale/zh-HANT.coffee            | 1 +
 app/locale/zh.coffee                 | 1 +
 app/views/play/level/hud_view.coffee | 3 ++-
 48 files changed, 49 insertions(+), 1 deletion(-)

diff --git a/app/locale/ar.coffee b/app/locale/ar.coffee
index b1d899a65..cb4c293df 100644
--- a/app/locale/ar.coffee
+++ b/app/locale/ar.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "العربية", englishDescription: "Arabi
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/bg.coffee b/app/locale/bg.coffee
index 72ca47fca..6829f58fa 100644
--- a/app/locale/bg.coffee
+++ b/app/locale/bg.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "български език", englishDescri
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/cs.coffee b/app/locale/cs.coffee
index aa32f0aa2..15896ec83 100644
--- a/app/locale/cs.coffee
+++ b/app/locale/cs.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "čeština", englishDescription: "Czech", tr
     tome_available_spells: "Dostupná kouzla"
     hud_continue: "Pokračovat (stiskněte shift-mezera)"
     spell_saved: "Kouzlo uloženo"
+#    skip_tutorial: "skip: esc"
 
   admin:
     av_title: "Administrátorský pohled"
diff --git a/app/locale/da.coffee b/app/locale/da.coffee
index 727e8d713..c825de14d 100644
--- a/app/locale/da.coffee
+++ b/app/locale/da.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "dansk", englishDescription: "Danish", trans
     tome_available_spells: "Tilgængelige trylleformularer"
     hud_continue: "Fortsæt (tryk skift-mellemrum)"
     spell_saved: "Trylleformularen er gemt"
+#    skip_tutorial: "skip: esc"
 
   admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/de.coffee b/app/locale/de.coffee
index aebfcccbb..6d6bb469d 100644
--- a/app/locale/de.coffee
+++ b/app/locale/de.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "Deutsch", englishDescription: "German", tra
     tome_available_spells: "Verfügbare Zauber"
     hud_continue: "Weiter (drücke Shift + Leertaste)"
     spell_saved: "Zauber gespeichert"
+#    skip_tutorial: "skip: esc"
 
   admin:
     av_title: "Administrator Übersicht"
diff --git a/app/locale/el.coffee b/app/locale/el.coffee
index 5425164b7..a03bff546 100644
--- a/app/locale/el.coffee
+++ b/app/locale/el.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "ελληνικά", englishDescription: "Gre
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/en-AU.coffee b/app/locale/en-AU.coffee
index 9c115d83d..70ec9b81e 100644
--- a/app/locale/en-AU.coffee
+++ b/app/locale/en-AU.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "English (AU)", englishDescription: "English
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/en-GB.coffee b/app/locale/en-GB.coffee
index 12bac5e6f..214cf5715 100644
--- a/app/locale/en-GB.coffee
+++ b/app/locale/en-GB.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "English (UK)", englishDescription: "English
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/en-US.coffee b/app/locale/en-US.coffee
index e34074395..9446407bf 100644
--- a/app/locale/en-US.coffee
+++ b/app/locale/en-US.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "English (US)", englishDescription: "English
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/en.coffee b/app/locale/en.coffee
index 2afc376cd..dbfdb35d1 100644
--- a/app/locale/en.coffee
+++ b/app/locale/en.coffee
@@ -201,6 +201,7 @@ module.exports = nativeDescription: "English", englishDescription: "English", tr
     tome_available_spells: "Available Spells"
     hud_continue: "Continue (press shift-space)"
     spell_saved: "Spell Saved"
+    skip_tutorial: "skip: esc"
 
   admin:
     av_title: "Admin Views"
diff --git a/app/locale/es-419.coffee b/app/locale/es-419.coffee
index 476b9fa7d..3ed36701e 100644
--- a/app/locale/es-419.coffee
+++ b/app/locale/es-419.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "español (América Latina)", englishDescrip
     tome_available_spells: "Hechizos Disponibles"
     hud_continue: "Continuar (presionar shift+space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/es-ES.coffee b/app/locale/es-ES.coffee
index 5b911e4b1..344426f6a 100644
--- a/app/locale/es-ES.coffee
+++ b/app/locale/es-ES.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "español (ES)", englishDescription: "Spanis
     tome_available_spells: "Hechizos disponibles"
     hud_continue: "Continuar (pulsa Shift+Space)"
     spell_saved: "Hechizo guardado"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/es.coffee b/app/locale/es.coffee
index fabd7b970..502c1f335 100644
--- a/app/locale/es.coffee
+++ b/app/locale/es.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "español", englishDescription: "Spanish", t
     tome_available_spells: "Hechizos Disponibles"
     hud_continue: "Continuar (presionar shift+space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/fa.coffee b/app/locale/fa.coffee
index 36d6b4991..543cf7df3 100644
--- a/app/locale/fa.coffee
+++ b/app/locale/fa.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "فارسی", englishDescription: "Persian",
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/fi.coffee b/app/locale/fi.coffee
index 35af0bb25..0182c8112 100644
--- a/app/locale/fi.coffee
+++ b/app/locale/fi.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "suomi", englishDescription: "Finnish", tran
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/fr.coffee b/app/locale/fr.coffee
index c869fbe8b..2a70fe98c 100644
--- a/app/locale/fr.coffee
+++ b/app/locale/fr.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "français", englishDescription: "French", t
     tome_available_spells: "Sorts diponibles"
     hud_continue: "Continuer (appuie sur shift ou espace)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
   admin:
     av_title: "Vues d'administrateurs"
diff --git a/app/locale/he.coffee b/app/locale/he.coffee
index 5d39d1b87..54c89b01f 100644
--- a/app/locale/he.coffee
+++ b/app/locale/he.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "עברית", englishDescription: "Hebrew",
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/hi.coffee b/app/locale/hi.coffee
index abaae44c6..b0a6f9161 100644
--- a/app/locale/hi.coffee
+++ b/app/locale/hi.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "मानक हिन्दी", englishDe
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/hu.coffee b/app/locale/hu.coffee
index 39fb5c38d..cb652101c 100644
--- a/app/locale/hu.coffee
+++ b/app/locale/hu.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "magyar", englishDescription: "Hungarian", t
     tome_available_spells: "Elérhető varázslatok"
     hud_continue: "Folytatás (shift+space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/id.coffee b/app/locale/id.coffee
index 4c21ab17c..c2d897c22 100644
--- a/app/locale/id.coffee
+++ b/app/locale/id.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "Bahasa Indonesia", englishDescription: "Ind
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/it.coffee b/app/locale/it.coffee
index e1ecb580f..cfd2a89a9 100644
--- a/app/locale/it.coffee
+++ b/app/locale/it.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "italiano", englishDescription: "Italian", t
     tome_available_spells: "Incantesimi disponibili"
     hud_continue: "Continua (premi Maiusc-Spazio)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
   admin:
     av_title: "Vista amministratore"
diff --git a/app/locale/ja.coffee b/app/locale/ja.coffee
index 47b10f118..827987e69 100644
--- a/app/locale/ja.coffee
+++ b/app/locale/ja.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "日本語", englishDescription: "Japanese",
     tome_available_spells: "利用できる呪文"
     hud_continue: "続く (Shift+Spaceキー)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
   admin:
     av_title: "管理画面"
diff --git a/app/locale/ko.coffee b/app/locale/ko.coffee
index 9b0f4c9c4..87254c99e 100644
--- a/app/locale/ko.coffee
+++ b/app/locale/ko.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "한국어", englishDescription: "Korean", t
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/lt.coffee b/app/locale/lt.coffee
index a562a051c..ca7436fe2 100644
--- a/app/locale/lt.coffee
+++ b/app/locale/lt.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "lietuvių kalba", englishDescription: "Lith
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/ms-BA.coffee b/app/locale/ms-BA.coffee
index 861bce18e..2605603e6 100644
--- a/app/locale/ms-BA.coffee
+++ b/app/locale/ms-BA.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "Bahasa Melayu", englishDescription: "Bahasa
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/nb.coffee b/app/locale/nb.coffee
index 92ac6f932..95fa9862f 100644
--- a/app/locale/nb.coffee
+++ b/app/locale/nb.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "Norsk Bokmål", englishDescription: "Norweg
     tome_available_spells: "Tilgjenglige Trylleformularer"
     hud_continue: "Fortsett (trykk shift-mellomrom)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/nl.coffee b/app/locale/nl.coffee
index 929741d23..c48fe57a5 100644
--- a/app/locale/nl.coffee
+++ b/app/locale/nl.coffee
@@ -201,6 +201,7 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     tome_available_spells: "Beschikbare spreuken"
     hud_continue: "Ga verder (druk shift-space)"
     spell_saved: "Spreuk Opgeslagen"
+#    skip_tutorial: "skip: esc"
 
   admin:
     av_title: "Administrator panels"
diff --git a/app/locale/nn.coffee b/app/locale/nn.coffee
index ceea3a0ea..fc1db1657 100644
--- a/app/locale/nn.coffee
+++ b/app/locale/nn.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "Norwegian Nynorsk", englishDescription: "No
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/no.coffee b/app/locale/no.coffee
index 4dcd77714..91d8b88b9 100644
--- a/app/locale/no.coffee
+++ b/app/locale/no.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "Norsk", englishDescription: "Norwegian", tr
     tome_available_spells: "Tilgjenglige Trylleformularer"
     hud_continue: "Fortsett (trykk shift-mellomrom)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/pl.coffee b/app/locale/pl.coffee
index 826e5669c..4b0a37ec0 100644
--- a/app/locale/pl.coffee
+++ b/app/locale/pl.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "język polski", englishDescription: "Polish
     tome_available_spells: "Dostępne czary"
     hud_continue: "Kontynuuj (naciśnij enter)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/pt-BR.coffee b/app/locale/pt-BR.coffee
index e6a285655..84dae2d58 100644
--- a/app/locale/pt-BR.coffee
+++ b/app/locale/pt-BR.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "português do Brasil", englishDescription:
     tome_available_spells: "Feitiços Disponíveis"
     hud_continue: "Continue (tecle Shift+Space)"
     spell_saved: "Feitiço Salvo"
+#    skip_tutorial: "skip: esc"
 
   admin:
     av_title: "Visualização de Administrador"
diff --git a/app/locale/pt-PT.coffee b/app/locale/pt-PT.coffee
index 1ce48a42f..d3ab714c1 100644
--- a/app/locale/pt-PT.coffee
+++ b/app/locale/pt-PT.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "Português europeu", englishDescription: "P
     tome_available_spells: "Feitiços disponíveis"
     hud_continue: "Continuar (pressiona shift-space)"
     spell_saved: "Feitiço Guardado"
+#    skip_tutorial: "skip: esc"
 
   admin:
     av_title: "Visualizações de Admin"
diff --git a/app/locale/pt.coffee b/app/locale/pt.coffee
index 9aedeae92..76b378b71 100644
--- a/app/locale/pt.coffee
+++ b/app/locale/pt.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "português", englishDescription: "Portugues
     tome_available_spells: "Feitiços Disponíveis"
     hud_continue: "Continue (tecle Shift+Space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/ro.coffee b/app/locale/ro.coffee
index 454278097..6cb6dfcb3 100644
--- a/app/locale/ro.coffee
+++ b/app/locale/ro.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/ru.coffee b/app/locale/ru.coffee
index 985282a6c..d1c7e7db2 100644
--- a/app/locale/ru.coffee
+++ b/app/locale/ru.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     tome_available_spells: "Доступные заклинания"
     hud_continue: "Продолжить (нажмите Shift+Пробел)"
     spell_saved: "Заклинание сохранено"
+#    skip_tutorial: "skip: esc"
 
   admin:
     av_title: "Админ панель"
diff --git a/app/locale/sk.coffee b/app/locale/sk.coffee
index 9050f2181..7c6dd325f 100644
--- a/app/locale/sk.coffee
+++ b/app/locale/sk.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "slovenčina", englishDescription: "Slovak",
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/sl.coffee b/app/locale/sl.coffee
index 97559521a..8ebcde87e 100644
--- a/app/locale/sl.coffee
+++ b/app/locale/sl.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "slovenščina", englishDescription: "Sloven
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/sr.coffee b/app/locale/sr.coffee
index 179b74906..77f9513dd 100644
--- a/app/locale/sr.coffee
+++ b/app/locale/sr.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "српски", englishDescription: "Serbian
     tome_available_spells: "Доступне чини"
     hud_continue: "Настави (притисни ентер)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/sv.coffee b/app/locale/sv.coffee
index b3868e937..075c1adbf 100644
--- a/app/locale/sv.coffee
+++ b/app/locale/sv.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "Svenska", englishDescription: "Swedish", tr
     tome_available_spells: "Tillgängliga Förmågor"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/th.coffee b/app/locale/th.coffee
index 9fd74d3a0..8a00dddc9 100644
--- a/app/locale/th.coffee
+++ b/app/locale/th.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "ไทย", englishDescription: "Thai", tra
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/tr.coffee b/app/locale/tr.coffee
index 297c9f50d..d5f698aed 100644
--- a/app/locale/tr.coffee
+++ b/app/locale/tr.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "Türkçe", englishDescription: "Turkish", t
     tome_available_spells: "Kullanılabilir Büyüler"
     hud_continue: "Devam (ÜstKarakter+Boşluk)"
     spell_saved: "Büyü Kaydedildi"
+#    skip_tutorial: "skip: esc"
 
   admin:
     av_title: "Yönetici Görünümleri"
diff --git a/app/locale/uk.coffee b/app/locale/uk.coffee
index 3f3a375c7..d0f9f2614 100644
--- a/app/locale/uk.coffee
+++ b/app/locale/uk.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "українська мова", englishDesc
     tome_available_spells: "Доступні закляття"
     hud_continue: "Продовжити (натисніть shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/ur.coffee b/app/locale/ur.coffee
index d36ce12ec..6090f23bc 100644
--- a/app/locale/ur.coffee
+++ b/app/locale/ur.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "اُردُو", englishDescription: "Urdu",
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/vi.coffee b/app/locale/vi.coffee
index 99992f8c5..6e424c0d9 100644
--- a/app/locale/vi.coffee
+++ b/app/locale/vi.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "Tiếng Việt", englishDescription: "Vietn
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/zh-HANS.coffee b/app/locale/zh-HANS.coffee
index 13c3d3161..c8d3a7d26 100644
--- a/app/locale/zh-HANS.coffee
+++ b/app/locale/zh-HANS.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
     tome_available_spells: "可用的法术"
     hud_continue: "继续(按 Shift-空格)"
     spell_saved: "咒语已保存"
+#    skip_tutorial: "skip: esc"
 
   admin:
     av_title: "管理员视图"
diff --git a/app/locale/zh-HANT.coffee b/app/locale/zh-HANT.coffee
index aad8bbbf9..2e869ad1e 100644
--- a/app/locale/zh-HANT.coffee
+++ b/app/locale/zh-HANT.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "繁体中文", englishDescription: "Chinese
     tome_available_spells: "可用的法術"
     hud_continue: "繼續 (按 shift-空格)"
     spell_saved: "咒語已儲存"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/zh.coffee b/app/locale/zh.coffee
index f0afbd64b..a06033aab 100644
--- a/app/locale/zh.coffee
+++ b/app/locale/zh.coffee
@@ -200,6 +200,7 @@ module.exports = nativeDescription: "中文", englishDescription: "Chinese", tra
 #    tome_available_spells: "Available Spells"
 #    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
+#    skip_tutorial: "skip: esc"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/views/play/level/hud_view.coffee b/app/views/play/level/hud_view.coffee
index f4580660b..c2ea8cc4c 100644
--- a/app/views/play/level/hud_view.coffee
+++ b/app/views/play/level/hud_view.coffee
@@ -176,8 +176,9 @@ module.exports = class HUDView extends View
         response.button = $('button:last', group)
     else
       s = $.i18n.t('play_level.hud_continue', defaultValue: "Continue (press shift-space)")
+      sk = $.i18n.t('play_level.skip_tutorial', defaultValue: "skip: esc")
       if @shiftSpacePressed > 4 and not @escapePressed
-        @bubble.append('<span class="hud-hint">skip: esc</span>')
+        @bubble.append('<span class="hud-hint">' + sk + '</span>')
       group.append($('<button class="btn btn-small banner with-dot">' + s + ' <div class="dot"></div></button>'))
       @lastResponses = null
     @bubble.append($("<h3>#{@speaker ? 'Captain Anya'}</h3>"))

From d16d02478254ca92169e485145119d878921a887 Mon Sep 17 00:00:00 2001
From: Dominik Kundel <dominik.kundel@gmail.com>
Date: Wed, 5 Mar 2014 17:06:46 +0100
Subject: [PATCH 076/178] further changes in design of highlighting lines in
 code

---
 app/styles/play/level/tome/spell.sass       | 4 ++--
 app/views/play/level/tome/spell_view.coffee | 2 ++
 2 files changed, 4 insertions(+), 2 deletions(-)

diff --git a/app/styles/play/level/tome/spell.sass b/app/styles/play/level/tome/spell.sass
index 5fdfa7c61..5990e7885 100644
--- a/app/styles/play/level/tome/spell.sass
+++ b/app/styles/play/level/tome/spell.sass
@@ -76,7 +76,7 @@
       background-color: rgba(0, 255, 0, 0.15)
       @include gradient-striped()
     .executed
-      background-color: rgba(40, 40, 40, 0.12)
+      background-color: rgba(110, 110, 110, 0.12)
     .problem-marker-info
       background-color: rgba(196, 163, 184, 0.25)
     .problem-marker-warning
@@ -95,7 +95,7 @@
 
     .ace_gutter-cell.executed:not(.ace_error)
       margin-left: 1px
-      background-image: url()
+      background-image: url()
       background-position: 0px center
 
     .ace_gutter-cell.comment-line
diff --git a/app/views/play/level/tome/spell_view.coffee b/app/views/play/level/tome/spell_view.coffee
index 679ef7d87..af9d2ad7f 100644
--- a/app/views/play/level/tome/spell_view.coffee
+++ b/app/views/play/level/tome/spell_view.coffee
@@ -456,7 +456,9 @@ module.exports = class SpellView extends View
       @aceSession.removeMarker markerRange.id
     @markerRanges = []
     @aceSession.removeGutterDecoration row, 'executing' for row in [0 ... @aceSession.getLength()]
+    @aceSession.removeGutterDecoration row, 'executed' for row in [0 ... @aceSession.getLength()]
     $(@ace.container).find('.ace_gutter-cell.executing').removeClass('executing')
+    $(@ace.container).find('.ace_gutter-cell.executed').removeClass('executed')
     if not executed.length or (@spell.name is "plan" and @spellThang.castAether.metrics.statementsExecuted < 20)
       @toolbarView?.toggleFlow false
       @debugView.setVariableStates {}

From f6324807b399d7cf888315b66bccc54d74b4aad0 Mon Sep 17 00:00:00 2001
From: Shrihari <gfxindia@gmail.com>
Date: Wed, 5 Mar 2014 23:36:18 +0530
Subject: [PATCH 077/178] Moved "Skip (esc)" next to Continue button

---
 app/locale/ar.coffee                 | 4 ++--
 app/locale/bg.coffee                 | 4 ++--
 app/locale/cs.coffee                 | 2 +-
 app/locale/da.coffee                 | 2 +-
 app/locale/de.coffee                 | 2 +-
 app/locale/el.coffee                 | 4 ++--
 app/locale/en-AU.coffee              | 4 ++--
 app/locale/en-GB.coffee              | 4 ++--
 app/locale/en-US.coffee              | 4 ++--
 app/locale/en.coffee                 | 4 ++--
 app/locale/es-419.coffee             | 2 +-
 app/locale/es-ES.coffee              | 2 +-
 app/locale/es.coffee                 | 2 +-
 app/locale/fa.coffee                 | 4 ++--
 app/locale/fi.coffee                 | 4 ++--
 app/locale/fr.coffee                 | 2 +-
 app/locale/he.coffee                 | 4 ++--
 app/locale/hi.coffee                 | 4 ++--
 app/locale/hu.coffee                 | 2 +-
 app/locale/id.coffee                 | 4 ++--
 app/locale/it.coffee                 | 2 +-
 app/locale/ja.coffee                 | 2 +-
 app/locale/ko.coffee                 | 4 ++--
 app/locale/lt.coffee                 | 4 ++--
 app/locale/ms-BA.coffee              | 4 ++--
 app/locale/nb.coffee                 | 2 +-
 app/locale/nl.coffee                 | 2 +-
 app/locale/nn.coffee                 | 4 ++--
 app/locale/no.coffee                 | 2 +-
 app/locale/pl.coffee                 | 2 +-
 app/locale/pt-BR.coffee              | 2 +-
 app/locale/pt-PT.coffee              | 2 +-
 app/locale/pt.coffee                 | 2 +-
 app/locale/ro.coffee                 | 4 ++--
 app/locale/ru.coffee                 | 2 +-
 app/locale/sk.coffee                 | 4 ++--
 app/locale/sl.coffee                 | 4 ++--
 app/locale/sr.coffee                 | 2 +-
 app/locale/sv.coffee                 | 4 ++--
 app/locale/th.coffee                 | 4 ++--
 app/locale/tr.coffee                 | 2 +-
 app/locale/uk.coffee                 | 2 +-
 app/locale/ur.coffee                 | 4 ++--
 app/locale/vi.coffee                 | 4 ++--
 app/locale/zh-HANS.coffee            | 2 +-
 app/locale/zh-HANT.coffee            | 2 +-
 app/locale/zh.coffee                 | 4 ++--
 app/styles/play/level/hud.sass       | 5 +----
 app/views/play/level/hud_view.coffee | 6 +++---
 49 files changed, 75 insertions(+), 78 deletions(-)

diff --git a/app/locale/ar.coffee b/app/locale/ar.coffee
index cb4c293df..1f0584c85 100644
--- a/app/locale/ar.coffee
+++ b/app/locale/ar.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "العربية", englishDescription: "Arabi
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/bg.coffee b/app/locale/bg.coffee
index 6829f58fa..fd038c56c 100644
--- a/app/locale/bg.coffee
+++ b/app/locale/bg.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "български език", englishDescri
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/cs.coffee b/app/locale/cs.coffee
index 15896ec83..121f0da4f 100644
--- a/app/locale/cs.coffee
+++ b/app/locale/cs.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "čeština", englishDescription: "Czech", tr
     tome_available_spells: "Dostupná kouzla"
     hud_continue: "Pokračovat (stiskněte shift-mezera)"
     spell_saved: "Kouzlo uloženo"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
   admin:
     av_title: "Administrátorský pohled"
diff --git a/app/locale/da.coffee b/app/locale/da.coffee
index c825de14d..ad47c71fc 100644
--- a/app/locale/da.coffee
+++ b/app/locale/da.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "dansk", englishDescription: "Danish", trans
     tome_available_spells: "Tilgængelige trylleformularer"
     hud_continue: "Fortsæt (tryk skift-mellemrum)"
     spell_saved: "Trylleformularen er gemt"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
   admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/de.coffee b/app/locale/de.coffee
index 6d6bb469d..04b640ce8 100644
--- a/app/locale/de.coffee
+++ b/app/locale/de.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "Deutsch", englishDescription: "German", tra
     tome_available_spells: "Verfügbare Zauber"
     hud_continue: "Weiter (drücke Shift + Leertaste)"
     spell_saved: "Zauber gespeichert"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
   admin:
     av_title: "Administrator Übersicht"
diff --git a/app/locale/el.coffee b/app/locale/el.coffee
index a03bff546..91fd4be36 100644
--- a/app/locale/el.coffee
+++ b/app/locale/el.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "ελληνικά", englishDescription: "Gre
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/en-AU.coffee b/app/locale/en-AU.coffee
index 70ec9b81e..8c36e3ef7 100644
--- a/app/locale/en-AU.coffee
+++ b/app/locale/en-AU.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "English (AU)", englishDescription: "English
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/en-GB.coffee b/app/locale/en-GB.coffee
index 214cf5715..1da9d218a 100644
--- a/app/locale/en-GB.coffee
+++ b/app/locale/en-GB.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "English (UK)", englishDescription: "English
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/en-US.coffee b/app/locale/en-US.coffee
index 9446407bf..9d3b2dc5a 100644
--- a/app/locale/en-US.coffee
+++ b/app/locale/en-US.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "English (US)", englishDescription: "English
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/en.coffee b/app/locale/en.coffee
index dbfdb35d1..5ca5cf902 100644
--- a/app/locale/en.coffee
+++ b/app/locale/en.coffee
@@ -199,9 +199,9 @@ module.exports = nativeDescription: "English", englishDescription: "English", tr
     tome_select_spell: "Select a Spell"
     tome_select_a_thang: "Select Someone for "
     tome_available_spells: "Available Spells"
-    hud_continue: "Continue (press shift-space)"
+    hud_continue: "Continue (shift+space)"
     spell_saved: "Spell Saved"
-    skip_tutorial: "skip: esc"
+    skip_tutorial: "Skip (esc)"
 
   admin:
     av_title: "Admin Views"
diff --git a/app/locale/es-419.coffee b/app/locale/es-419.coffee
index 3ed36701e..3c2614713 100644
--- a/app/locale/es-419.coffee
+++ b/app/locale/es-419.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "español (América Latina)", englishDescrip
     tome_available_spells: "Hechizos Disponibles"
     hud_continue: "Continuar (presionar shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/es-ES.coffee b/app/locale/es-ES.coffee
index 344426f6a..e45550e08 100644
--- a/app/locale/es-ES.coffee
+++ b/app/locale/es-ES.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "español (ES)", englishDescription: "Spanis
     tome_available_spells: "Hechizos disponibles"
     hud_continue: "Continuar (pulsa Shift+Space)"
     spell_saved: "Hechizo guardado"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/es.coffee b/app/locale/es.coffee
index 502c1f335..b4009323b 100644
--- a/app/locale/es.coffee
+++ b/app/locale/es.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "español", englishDescription: "Spanish", t
     tome_available_spells: "Hechizos Disponibles"
     hud_continue: "Continuar (presionar shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/fa.coffee b/app/locale/fa.coffee
index 543cf7df3..28a4bd760 100644
--- a/app/locale/fa.coffee
+++ b/app/locale/fa.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "فارسی", englishDescription: "Persian",
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/fi.coffee b/app/locale/fi.coffee
index 0182c8112..57e5151e4 100644
--- a/app/locale/fi.coffee
+++ b/app/locale/fi.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "suomi", englishDescription: "Finnish", tran
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/fr.coffee b/app/locale/fr.coffee
index 2a70fe98c..1469c7269 100644
--- a/app/locale/fr.coffee
+++ b/app/locale/fr.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "français", englishDescription: "French", t
     tome_available_spells: "Sorts diponibles"
     hud_continue: "Continuer (appuie sur shift ou espace)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
   admin:
     av_title: "Vues d'administrateurs"
diff --git a/app/locale/he.coffee b/app/locale/he.coffee
index 54c89b01f..ddcb7484c 100644
--- a/app/locale/he.coffee
+++ b/app/locale/he.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "עברית", englishDescription: "Hebrew",
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/hi.coffee b/app/locale/hi.coffee
index b0a6f9161..596ecd1ce 100644
--- a/app/locale/hi.coffee
+++ b/app/locale/hi.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "मानक हिन्दी", englishDe
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/hu.coffee b/app/locale/hu.coffee
index cb652101c..ad6cd0273 100644
--- a/app/locale/hu.coffee
+++ b/app/locale/hu.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "magyar", englishDescription: "Hungarian", t
     tome_available_spells: "Elérhető varázslatok"
     hud_continue: "Folytatás (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/id.coffee b/app/locale/id.coffee
index c2d897c22..20d16335f 100644
--- a/app/locale/id.coffee
+++ b/app/locale/id.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "Bahasa Indonesia", englishDescription: "Ind
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/it.coffee b/app/locale/it.coffee
index cfd2a89a9..a95f98916 100644
--- a/app/locale/it.coffee
+++ b/app/locale/it.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "italiano", englishDescription: "Italian", t
     tome_available_spells: "Incantesimi disponibili"
     hud_continue: "Continua (premi Maiusc-Spazio)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
   admin:
     av_title: "Vista amministratore"
diff --git a/app/locale/ja.coffee b/app/locale/ja.coffee
index 827987e69..4c9e4dfb9 100644
--- a/app/locale/ja.coffee
+++ b/app/locale/ja.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "日本語", englishDescription: "Japanese",
     tome_available_spells: "利用できる呪文"
     hud_continue: "続く (Shift+Spaceキー)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
   admin:
     av_title: "管理画面"
diff --git a/app/locale/ko.coffee b/app/locale/ko.coffee
index 87254c99e..882f3d4ac 100644
--- a/app/locale/ko.coffee
+++ b/app/locale/ko.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "한국어", englishDescription: "Korean", t
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/lt.coffee b/app/locale/lt.coffee
index ca7436fe2..1008a8bfb 100644
--- a/app/locale/lt.coffee
+++ b/app/locale/lt.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "lietuvių kalba", englishDescription: "Lith
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/ms-BA.coffee b/app/locale/ms-BA.coffee
index 2605603e6..29c914f23 100644
--- a/app/locale/ms-BA.coffee
+++ b/app/locale/ms-BA.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "Bahasa Melayu", englishDescription: "Bahasa
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/nb.coffee b/app/locale/nb.coffee
index 95fa9862f..db74e540d 100644
--- a/app/locale/nb.coffee
+++ b/app/locale/nb.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "Norsk Bokmål", englishDescription: "Norweg
     tome_available_spells: "Tilgjenglige Trylleformularer"
     hud_continue: "Fortsett (trykk shift-mellomrom)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/nl.coffee b/app/locale/nl.coffee
index c48fe57a5..508d6e384 100644
--- a/app/locale/nl.coffee
+++ b/app/locale/nl.coffee
@@ -201,7 +201,7 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     tome_available_spells: "Beschikbare spreuken"
     hud_continue: "Ga verder (druk shift-space)"
     spell_saved: "Spreuk Opgeslagen"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
   admin:
     av_title: "Administrator panels"
diff --git a/app/locale/nn.coffee b/app/locale/nn.coffee
index fc1db1657..c9e770bfa 100644
--- a/app/locale/nn.coffee
+++ b/app/locale/nn.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "Norwegian Nynorsk", englishDescription: "No
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/no.coffee b/app/locale/no.coffee
index 91d8b88b9..bc0e69e0c 100644
--- a/app/locale/no.coffee
+++ b/app/locale/no.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "Norsk", englishDescription: "Norwegian", tr
     tome_available_spells: "Tilgjenglige Trylleformularer"
     hud_continue: "Fortsett (trykk shift-mellomrom)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/pl.coffee b/app/locale/pl.coffee
index 4b0a37ec0..9988be45e 100644
--- a/app/locale/pl.coffee
+++ b/app/locale/pl.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "język polski", englishDescription: "Polish
     tome_available_spells: "Dostępne czary"
     hud_continue: "Kontynuuj (naciśnij enter)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/pt-BR.coffee b/app/locale/pt-BR.coffee
index 84dae2d58..40f374f81 100644
--- a/app/locale/pt-BR.coffee
+++ b/app/locale/pt-BR.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "português do Brasil", englishDescription:
     tome_available_spells: "Feitiços Disponíveis"
     hud_continue: "Continue (tecle Shift+Space)"
     spell_saved: "Feitiço Salvo"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
   admin:
     av_title: "Visualização de Administrador"
diff --git a/app/locale/pt-PT.coffee b/app/locale/pt-PT.coffee
index d3ab714c1..77fb7afad 100644
--- a/app/locale/pt-PT.coffee
+++ b/app/locale/pt-PT.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "Português europeu", englishDescription: "P
     tome_available_spells: "Feitiços disponíveis"
     hud_continue: "Continuar (pressiona shift-space)"
     spell_saved: "Feitiço Guardado"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
   admin:
     av_title: "Visualizações de Admin"
diff --git a/app/locale/pt.coffee b/app/locale/pt.coffee
index 76b378b71..38a21790c 100644
--- a/app/locale/pt.coffee
+++ b/app/locale/pt.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "português", englishDescription: "Portugues
     tome_available_spells: "Feitiços Disponíveis"
     hud_continue: "Continue (tecle Shift+Space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/ro.coffee b/app/locale/ro.coffee
index 6cb6dfcb3..3dfc71d1e 100644
--- a/app/locale/ro.coffee
+++ b/app/locale/ro.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/ru.coffee b/app/locale/ru.coffee
index d1c7e7db2..57cfc1847 100644
--- a/app/locale/ru.coffee
+++ b/app/locale/ru.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     tome_available_spells: "Доступные заклинания"
     hud_continue: "Продолжить (нажмите Shift+Пробел)"
     spell_saved: "Заклинание сохранено"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
   admin:
     av_title: "Админ панель"
diff --git a/app/locale/sk.coffee b/app/locale/sk.coffee
index 7c6dd325f..c9e3e4023 100644
--- a/app/locale/sk.coffee
+++ b/app/locale/sk.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "slovenčina", englishDescription: "Slovak",
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/sl.coffee b/app/locale/sl.coffee
index 8ebcde87e..b0cbdd3f5 100644
--- a/app/locale/sl.coffee
+++ b/app/locale/sl.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "slovenščina", englishDescription: "Sloven
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/sr.coffee b/app/locale/sr.coffee
index 77f9513dd..c9676fa65 100644
--- a/app/locale/sr.coffee
+++ b/app/locale/sr.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "српски", englishDescription: "Serbian
     tome_available_spells: "Доступне чини"
     hud_continue: "Настави (притисни ентер)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/sv.coffee b/app/locale/sv.coffee
index 075c1adbf..d8b41ec50 100644
--- a/app/locale/sv.coffee
+++ b/app/locale/sv.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "Svenska", englishDescription: "Swedish", tr
     tome_select_spell: "Välj en Förmåga"
     tome_select_a_thang: "Välj Någon för "
     tome_available_spells: "Tillgängliga Förmågor"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/th.coffee b/app/locale/th.coffee
index 8a00dddc9..9b2134eb8 100644
--- a/app/locale/th.coffee
+++ b/app/locale/th.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "ไทย", englishDescription: "Thai", tra
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/tr.coffee b/app/locale/tr.coffee
index d5f698aed..4a256defe 100644
--- a/app/locale/tr.coffee
+++ b/app/locale/tr.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "Türkçe", englishDescription: "Turkish", t
     tome_available_spells: "Kullanılabilir Büyüler"
     hud_continue: "Devam (ÜstKarakter+Boşluk)"
     spell_saved: "Büyü Kaydedildi"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
   admin:
     av_title: "Yönetici Görünümleri"
diff --git a/app/locale/uk.coffee b/app/locale/uk.coffee
index d0f9f2614..bf67c2138 100644
--- a/app/locale/uk.coffee
+++ b/app/locale/uk.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "українська мова", englishDesc
     tome_available_spells: "Доступні закляття"
     hud_continue: "Продовжити (натисніть shift-space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/ur.coffee b/app/locale/ur.coffee
index 6090f23bc..8237971a3 100644
--- a/app/locale/ur.coffee
+++ b/app/locale/ur.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "اُردُو", englishDescription: "Urdu",
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/vi.coffee b/app/locale/vi.coffee
index 6e424c0d9..c3a00a167 100644
--- a/app/locale/vi.coffee
+++ b/app/locale/vi.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "Tiếng Việt", englishDescription: "Vietn
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/zh-HANS.coffee b/app/locale/zh-HANS.coffee
index c8d3a7d26..f6eebce5d 100644
--- a/app/locale/zh-HANS.coffee
+++ b/app/locale/zh-HANS.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
     tome_available_spells: "可用的法术"
     hud_continue: "继续(按 Shift-空格)"
     spell_saved: "咒语已保存"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
   admin:
     av_title: "管理员视图"
diff --git a/app/locale/zh-HANT.coffee b/app/locale/zh-HANT.coffee
index 2e869ad1e..d38532973 100644
--- a/app/locale/zh-HANT.coffee
+++ b/app/locale/zh-HANT.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "繁体中文", englishDescription: "Chinese
     tome_available_spells: "可用的法術"
     hud_continue: "繼續 (按 shift-空格)"
     spell_saved: "咒語已儲存"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/locale/zh.coffee b/app/locale/zh.coffee
index a06033aab..aacf8fe4d 100644
--- a/app/locale/zh.coffee
+++ b/app/locale/zh.coffee
@@ -198,9 +198,9 @@ module.exports = nativeDescription: "中文", englishDescription: "Chinese", tra
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
+#    hud_continue: "Continue (shift+space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "skip: esc"
+#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
diff --git a/app/styles/play/level/hud.sass b/app/styles/play/level/hud.sass
index 7858f0365..9cccb690d 100644
--- a/app/styles/play/level/hud.sass
+++ b/app/styles/play/level/hud.sass
@@ -230,10 +230,7 @@
 
         .hud-hint
           font-weight: normal
-          color: #aaa
-          position: absolute
-          top: 0
-          right: 4px
+          color: #999
 
         .enter
           position: absolute
diff --git a/app/views/play/level/hud_view.coffee b/app/views/play/level/hud_view.coffee
index c2ea8cc4c..7a5c42af9 100644
--- a/app/views/play/level/hud_view.coffee
+++ b/app/views/play/level/hud_view.coffee
@@ -175,10 +175,10 @@ module.exports = class HUDView extends View
         group.append(button)
         response.button = $('button:last', group)
     else
-      s = $.i18n.t('play_level.hud_continue', defaultValue: "Continue (press shift-space)")
+      s = $.i18n.t('play_level.hud_continue', defaultValue: "Continue (shift+space)")
       sk = $.i18n.t('play_level.skip_tutorial', defaultValue: "skip: esc")
-      if @shiftSpacePressed > 4 and not @escapePressed
-        @bubble.append('<span class="hud-hint">' + sk + '</span>')
+      if not @escapePressed
+        group.append('<span class="hud-hint">' + sk + '</span>')
       group.append($('<button class="btn btn-small banner with-dot">' + s + ' <div class="dot"></div></button>'))
       @lastResponses = null
     @bubble.append($("<h3>#{@speaker ? 'Captain Anya'}</h3>"))

From ea98a9917d4993b1d066f2a949e072e693f9f80a Mon Sep 17 00:00:00 2001
From: Darredevil <alex.darredevil@gmail.com>
Date: Wed, 5 Mar 2014 20:40:14 +0200
Subject: [PATCH 078/178] Update ro

---
 app/locale/ro.coffee | 107 +++++++++++++++++++++----------------------
 1 file changed, 53 insertions(+), 54 deletions(-)

diff --git a/app/locale/ro.coffee b/app/locale/ro.coffee
index f8dd54668..6bdb8e75a 100644
--- a/app/locale/ro.coffee
+++ b/app/locale/ro.coffee
@@ -1,6 +1,6 @@
 module.exports = nativeDescription: "limba română", englishDescription: "Romanian", translation:
   common:
-    loading: "Loading..."
+    loading: "Se incarcă..."
     saving: "Se salvează..."
     sending: "Se trimite..."
     cancel: "Anulează"
@@ -53,8 +53,8 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
 
   signup:
     create_account_title: "Crează cont pentru a salva progresul"
-    description: "Este gratis. Doar câte ceva inainte si poți continua:"#contextual translation a bit off, could be better# It's free. Just need a couple things and you'll be good to go:"
-    email_announcements: "Receive announcements by email"
+    description: "Este gratis. Doar un scurt formular inainte si poți continua:"#contextual translation a bit off, could be better# It's free. Just need a couple things and you'll be good to go:"
+    email_announcements: "Primește notificări prin emaill"
     coppa: "13+ sau non-USA "
     coppa_why: "(De ce?)"
     creating: "Se crează contul..."
@@ -63,7 +63,7 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
 
   home:
     slogan: "Învață sa scri JavaScript jucându-te"# again sounds funny# Learn to Code JavaScript by Playing a Game"
-    no_ie: "CodeCombat does not run in Internet Explorer 9 or older. Sorry!"
+    no_ie: "CodeCombat nu merge pe Internet Explorer 9 sau mai vechi. Scuze!"
     no_mobile: "CodeCombat nu a fost proiectat pentru dispozitive mobile si s-ar putea sa nu meargâ!"
     play: "Joacâ"
 
@@ -78,59 +78,59 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
     campaign_dev_description: "... în care se învață interfața, cu o dificultate puțin mai mare."
     campaign_multiplayer: "Arene Multiplayer"
     campaign_multiplayer_description: "... în care te lupți cap-la-cap contra alti jucători."
-#    campaign_player_created: "Player-Created"
-#    campaign_player_created_description: "... in which you battle against the creativity of your fellow <a href=\"/contribute#artisan\">Artisan Wizards</a>."
-#    level_difficulty: "Difficulty: "
+    campaign_player_created: "Create de jucători"
+    campaign_player_created_description: "... în care ai ocazia să testezi creativitatea colegilor tai <a href=\"/contribute#artisan\">Artisan Wizards</a>."
+    level_difficulty: "Dificultate: "
 
-#  contact:
-#    contact_us: "Contact CodeCombat"
-#    welcome: "Good to hear from you! Use this form to send us email. "
-#    contribute_prefix: "If you're interested in contributing, check out our "
-#    contribute_page: "contribute page"
-#    contribute_suffix: "!"
-#    forum_prefix: "For anything public, please try "
-#    forum_page: "our forum"
-#    forum_suffix: " instead."
-#    send: "Send Feedback"
+  contact:
+    contact_us: "Contact CodeCombat"
+    welcome: "Folosiți acest formular pentru a ne trimite email. "
+    contribute_prefix: "Dacă sunteți interesați in a contribui uitați-vă pe "
+    contribute_page: "pagina de contribuție"
+    contribute_suffix: "!"
+    forum_prefix: "Pentru orice altceva vă rugăm sa incercați "
+    forum_page: "forumul nostru"
+    forum_suffix: " în schimb."
+    send: "Trimite Feedback"
 
   diplomat_suggestion:
-#    title: "Help translate CodeCombat!"
-#    sub_heading: "We need your language skills."
-    pitch_body: "We develop CodeCombat in English, but we already have players all over the world. Many of them want to play in Romanian but don't speak English, so if you can speak both, please consider signing up to be a Diplomat and help translate both the CodeCombat website and all the levels into Romanian."
-    missing_translations: "Until we can translate everything into Romanian, you'll see English when Romanian isn't available."
-#    learn_more: "Learn more about being a Diplomat"
-#    subscribe_as_diplomat: "Subscribe as a Diplomat"
+    title: "Ajută-ne  să traducem CodeCombat!"
+    sub_heading: "Avem nevoie de abilitățile tale lingvistice."
+    pitch_body: "We develop CodeCombat in English, but we already have players all over the world. Many of them want to play in Romanian but don't speak English, so if you can speak both, please consider signing up to be a Diplomat and help translate both the CodeCombat website and all the levels into Romanian." #are these still needed??
+    missing_translations: "Until we can translate everything into Romanian, you'll see English when Romanian isn't available." # is this still needed?
+    learn_more: "Află mai multe despre cum să fi un Diplomat"
+    subscribe_as_diplomat: "Înscrie-te ca Diplomat"
 
-#  wizard_settings:
-#    title: "Wizard Settings"
-#    customize_avatar: "Customize Your Avatar"
-#    clothes: "Clothes"
-#    trim: "Trim"
-#    cloud: "Cloud"
-#    spell: "Spell"
-#    boots: "Boots"
-#    hue: "Hue"
-#    saturation: "Saturation"
-#    lightness: "Lightness"
+  wizard_settings:
+    title: "Setări Wizard"
+    customize_avatar: "Personalizează-ți Avatarul"
+    clothes: "Haine"
+    trim: "Margine" 
+    cloud: "Nor"
+    spell: "Vrajă"
+    boots: "Încălțăminte"
+    hue: "Culoare"
+    saturation: "Saturație"
+    lightness: "Luminozitate"
 
-#  account_settings:
-#    title: "Account Settings"
-#    not_logged_in: "Log in or create an account to change your settings."
-#    autosave: "Changes Save Automatically"
-#    me_tab: "Me"
-#    picture_tab: "Picture"
-#    wizard_tab: "Wizard"
-#    password_tab: "Password"
-#    emails_tab: "Emails"
-#    gravatar_select: "Select which Gravatar photo to use"
-#    gravatar_add_photos: "Add thumbnails and photos to a Gravatar account for your email to choose an image."
-#    gravatar_add_more_photos: "Add more photos to your Gravatar account to access them here."
-#    wizard_color: "Wizard Clothes Color"
-#    new_password: "New Password"
-#    new_password_verify: "Verify"
-#    email_subscriptions: "Email Subscriptions"
-#    email_announcements: "Announcements"
-#    email_notifications_description: "Get periodic notifications for your account."
+  account_settings:
+    title: "Setări Cont"
+    not_logged_in: "Loghează-te sau crează un cont nou pentru a schimba setările."
+    autosave: "Modificările se salvează automat"
+    me_tab: "Eu"
+    picture_tab: "Poză"
+    wizard_tab: "Wizard"
+    password_tab: "Parolă"
+    emails_tab: "Email-uri"
+    gravatar_select: "Selectează ce poză Gravatar vrei să foloșesti"
+    gravatar_add_photos: "Adaugă thumbnails și poze la un cont Gravatar pentru email-ul tău pentru a alege o imagine."
+    gravatar_add_more_photos: "Adaugă mai multe poze la contul tău Gravatar pentru a le accesa aici."
+    wizard_color: "Culoare haine pentru Wizard"
+    new_password: "Parolă nouă"
+    new_password_verify: "Verifică"
+    email_subscriptions: "Subscripție Email"
+    email_announcements: "Anunțuri"
+    email_notifications_description: "Get periodic notifications for your account."
 #    email_announcements_description: "Get emails on the latest news and developments at CodeCombat."
 #    contributor_emails: "Contributor Class Emails"
 #    contribute_prefix: "We're looking for people to join our party! Check out the "
@@ -198,9 +198,8 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
 #    tome_select_spell: "Select a Spell"
 #    tome_select_a_thang: "Select Someone for "
 #    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (shift+space)"
+#    hud_continue: "Continue (press shift-space)"
 #    spell_saved: "Spell Saved"
-#    skip_tutorial: "Skip (esc)"
 
 #  admin:
 #    av_title: "Admin Views"

From 5ec06159a6cf5c6c726485941986df34ae85e1f2 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Wed, 5 Mar 2014 12:00:29 -0800
Subject: [PATCH 079/178] Fixed ChallengePost utm link. Tweaked God worker
 timeout for more reliable performance logging.

---
 README.md          | 2 +-
 app/lib/God.coffee | 2 +-
 2 files changed, 2 insertions(+), 2 deletions(-)

diff --git a/README.md b/README.md
index 7ec226f6e..4d23640e3 100644
--- a/README.md
+++ b/README.md
@@ -52,6 +52,6 @@ Whether you're novice or pro, the CodeCombat team is ready to help you implement
 
 ----------
 
-[![](https://dl.dropboxusercontent.com/u/138899/GitHub%20Wikis/challengepost.png)](http://codecombat.challengepost.com/?utm_source-github&utm_medium-oswidget&utm_campaign-codecombat)
+[![](https://dl.dropboxusercontent.com/u/138899/GitHub%20Wikis/challengepost.png)](http://codecombat.challengepost.com/?utm_source=github&utm_medium=oswidget&utm_campaign=codecombat)
 
 [![](http://1-ps.googleusercontent.com/x/s.google-melange.appspot.com/www.google-melange.com/soc/content/2-1-20140225/images/gsoc/logo/920x156xbanner-gsoc2014.png.pagespeed.ic.gdr4t3Igca.png)](http://www.google-melange.com/gsoc/homepage/google/gsoc2014)
\ No newline at end of file
diff --git a/app/lib/God.coffee b/app/lib/God.coffee
index ff92f7e31..f7563517e 100644
--- a/app/lib/God.coffee
+++ b/app/lib/God.coffee
@@ -227,7 +227,7 @@ class Angel
       _.delay ->
         worker.terminate()
         worker.removeEventListener 'message', onWorkerMessage
-      , 2000
+      , 3000
       @worker = null
     @
 

From 7885b21b4b516a873cf288f7f9e32e4f91377530 Mon Sep 17 00:00:00 2001
From: iraladson <aladso@saic.edu>
Date: Wed, 5 Mar 2014 14:03:57 -0600
Subject: [PATCH 080/178] Update names.coffee

---
 app/lib/world/names.coffee | 6 ++++++
 1 file changed, 6 insertions(+)

diff --git a/app/lib/world/names.coffee b/app/lib/world/names.coffee
index a7a3f256e..789c9b0f8 100644
--- a/app/lib/world/names.coffee
+++ b/app/lib/world/names.coffee
@@ -61,6 +61,8 @@ module.exports.thangNames = thangNames =
     "Joan"
     "Helga"
     "Annie"
+    "Lukaz"
+    "Gorgin"
   ]
   "Peasant": [
     "Yorik"
@@ -83,6 +85,8 @@ module.exports.thangNames = thangNames =
     "Bernadette"
     "Hershell"
     "Gawain"
+    "Durfkor"
+    "Paps"
   ]
   "Archer F": [
     "Phoebe"
@@ -117,6 +121,8 @@ module.exports.thangNames = thangNames =
     "Simon"
     "Robin"
     "Quinn"
+    "Arty"
+    "Gimsley"
   ]
   "Ogre Munchkin M": [
     "Brack"

From 8d046c31b990fbfdeef066422fa95954aa43253b Mon Sep 17 00:00:00 2001
From: iraladson <aladso@saic.edu>
Date: Wed, 5 Mar 2014 14:06:29 -0600
Subject: [PATCH 081/178] Update names.coffee

"Added more names for #53
---
 app/lib/world/names.coffee | 2 ++
 1 file changed, 2 insertions(+)

diff --git a/app/lib/world/names.coffee b/app/lib/world/names.coffee
index a7a3f256e..b23f8b862 100644
--- a/app/lib/world/names.coffee
+++ b/app/lib/world/names.coffee
@@ -117,6 +117,7 @@ module.exports.thangNames = thangNames =
     "Simon"
     "Robin"
     "Quinn"
+    "Fidsdale"
   ]
   "Ogre Munchkin M": [
     "Brack"
@@ -179,6 +180,7 @@ module.exports.thangNames = thangNames =
     "Borgag"
     "Grognar"
     "Ironjaw"
+    "Tuguro"
   ]
   "Ogre Fangrider": [
     "Dreek"

From db0fc82eb08afd2243c56d9b4d8c7eb8c4e0454b Mon Sep 17 00:00:00 2001
From: Jayant Jain <jayantjain1992@gmail.com>
Date: Thu, 6 Mar 2014 01:54:09 +0530
Subject: [PATCH 082/178] Minor bugfix for freezing editor due to out of bounds
 frame index

---
 app/lib/surface/Surface.coffee | 1 +
 1 file changed, 1 insertion(+)

diff --git a/app/lib/surface/Surface.coffee b/app/lib/surface/Surface.coffee
index 5ce3bf29c..ec006d449 100644
--- a/app/lib/surface/Surface.coffee
+++ b/app/lib/surface/Surface.coffee
@@ -502,6 +502,7 @@ module.exports = Surface = class Surface extends CocoClass
       # Skip some frame updates unless we're playing and not at end (or we haven't drawn much yet)
       frameAdvanced = (@playing and @currentFrame < @world.totalFrames) or @totalFramesDrawn < 2
       @currentFrame += @world.frameRate / @options.frameRate if frameAdvanced and @playing
+      @currentFrame = Math.min(@currentFrame, @world.totalFrames - 1)
       newWorldFrame = Math.floor @currentFrame
       worldFrameAdvanced = newWorldFrame isnt oldWorldFrame
       if worldFrameAdvanced

From af265ebee851c116c362256ab7529057305ab7d0 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Wed, 5 Mar 2014 12:53:11 -0800
Subject: [PATCH 083/178] Tweaked an error message to be more generally useful.

---
 app/lib/surface/CocoSprite.coffee | 4 +++-
 1 file changed, 3 insertions(+), 1 deletion(-)

diff --git a/app/lib/surface/CocoSprite.coffee b/app/lib/surface/CocoSprite.coffee
index 008519599..3a56f77e7 100644
--- a/app/lib/surface/CocoSprite.coffee
+++ b/app/lib/surface/CocoSprite.coffee
@@ -252,7 +252,9 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
   updateAction: ->
     action = @determineAction()
     isDifferent = action isnt @currentRootAction
-    console.error "action is", action, "for", @thang?.id, "from", @currentRootAction, @thang.action, @thang.getActionName?() if not action and @thang?.actionActivated and @thang.id is 'Artillery'
+    if not action and @thang?.actionActivated and not @stopLogging
+      console.error "action is", action, "for", @thang?.id, "from", @currentRootAction, @thang.action, @thang.getActionName?()
+      @stopLogging = true
     @queueAction(action) if isDifferent or (@thang?.actionActivated and action.name isnt 'move')
     @updateActionDirection()
 

From fc42043cb83edb9f241921a14b49c1502717fb9d Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Wed, 5 Mar 2014 12:53:48 -0800
Subject: [PATCH 084/178] Fixed #480

---
 app/models/ThangType.coffee | 3 ++-
 1 file changed, 2 insertions(+), 1 deletion(-)

diff --git a/app/models/ThangType.coffee b/app/models/ThangType.coffee
index 62f9eaa4e..b28bea773 100644
--- a/app/models/ThangType.coffee
+++ b/app/models/ThangType.coffee
@@ -157,7 +157,8 @@ module.exports = class ThangType extends CocoModel
     for groupName, config of options.colorConfig or {}
       colorConfigs.push "#{groupName}:#{config.hue}|#{config.saturation}|#{config.lightness}"
     colorConfigs = colorConfigs.join ','
-    "#{@get('name')} - #{options.resolutionFactor} - #{colorConfigs}"
+    portraitOnly = !!options.portraitOnly
+    "#{@get('name')} - #{options.resolutionFactor} - #{colorConfigs} - #{portraitOnly}"
 
   getPortraitImage: (spriteOptionsOrKey, size=100) ->
     src = @getPortraitSource(spriteOptionsOrKey, size)

From b0d571c7adb431bc10b198bcada1a4ef3cd4a0ab Mon Sep 17 00:00:00 2001
From: Dominik Kundel <dominik.kundel@gmail.com>
Date: Wed, 5 Mar 2014 22:17:34 +0100
Subject: [PATCH 085/178] highlight only the executing statement

---
 app/styles/play/level/tome/spell.sass       | 2 +-
 app/views/play/level/tome/spell_view.coffee | 2 +-
 2 files changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/styles/play/level/tome/spell.sass b/app/styles/play/level/tome/spell.sass
index 5990e7885..16473b06d 100644
--- a/app/styles/play/level/tome/spell.sass
+++ b/app/styles/play/level/tome/spell.sass
@@ -73,7 +73,7 @@
       .executing, .executed, .problem-marker-info, .problem-marker-warning, .problem-marker-error
         position: absolute
     .executing
-      background-color: rgba(0, 255, 0, 0.15)
+      background-color: rgba(0, 255, 0, 0.20)
       @include gradient-striped()
     .executed
       background-color: rgba(110, 110, 110, 0.12)
diff --git a/app/views/play/level/tome/spell_view.coffee b/app/views/play/level/tome/spell_view.coffee
index af9d2ad7f..8c98db854 100644
--- a/app/views/play/level/tome/spell_view.coffee
+++ b/app/views/play/level/tome/spell_view.coffee
@@ -480,7 +480,7 @@ module.exports = class SpellView extends View
       else
         @debugView.setVariableStates state.variables
         gotVariableStates = true
-        markerType = "fullLine"
+        markerType = "text"
       markerRange = new Range start.row, start.col, end.row, end.col
       markerRange.start = @aceDoc.createAnchor markerRange.start
       markerRange.end = @aceDoc.createAnchor markerRange.end

From 398fa3c426fba3c14fb8dcad581241ef337e58c0 Mon Sep 17 00:00:00 2001
From: Darredevil <alex.darredevil@gmail.com>
Date: Thu, 6 Mar 2014 00:12:58 +0200
Subject: [PATCH 086/178] Update ro.coffee

so much to translate....
---
 app/locale/ro.coffee | 174 +++++++++++++++++++++----------------------
 1 file changed, 87 insertions(+), 87 deletions(-)

diff --git a/app/locale/ro.coffee b/app/locale/ro.coffee
index 6bdb8e75a..3551d730b 100644
--- a/app/locale/ro.coffee
+++ b/app/locale/ro.coffee
@@ -96,8 +96,8 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
   diplomat_suggestion:
     title: "Ajută-ne  să traducem CodeCombat!"
     sub_heading: "Avem nevoie de abilitățile tale lingvistice."
-    pitch_body: "We develop CodeCombat in English, but we already have players all over the world. Many of them want to play in Romanian but don't speak English, so if you can speak both, please consider signing up to be a Diplomat and help translate both the CodeCombat website and all the levels into Romanian." #are these still needed??
-    missing_translations: "Until we can translate everything into Romanian, you'll see English when Romanian isn't available." # is this still needed?
+    pitch_body: "CodeCombat este dezvoltat in limba engleza , dar deja avem jucatări din toate colțurile lumii.Mulți dintre ei vor să joace in română și nu vorbesc engleză.Dacă poți vorbi ambele te rugăm să te gândești dacă ai dori să devi un Diplomat și să ne ajuți sa traducem atât jocul cât și site-ul." 
+    missing_translations: "Until we can translate everything into Romanian, you'll see English when Romanian isn't available." 
     learn_more: "Află mai multe despre cum să fi un Diplomat"
     subscribe_as_diplomat: "Înscrie-te ca Diplomat"
 
@@ -130,95 +130,95 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
     new_password_verify: "Verifică"
     email_subscriptions: "Subscripție Email"
     email_announcements: "Anunțuri"
-    email_notifications_description: "Get periodic notifications for your account."
-#    email_announcements_description: "Get emails on the latest news and developments at CodeCombat."
-#    contributor_emails: "Contributor Class Emails"
-#    contribute_prefix: "We're looking for people to join our party! Check out the "
-#    contribute_page: "contribute page"
-#    contribute_suffix: " to find out more."
-#    email_toggle: "Toggle All"
-#    error_saving: "Error Saving"
-#    saved: "Changes Saved"
-#    password_mismatch: "Password does not match."
+    email_notifications_description: "Primește notificări periodic pentru contul tău."
+    email_announcements_description: "Primește email-uri cu ultimele știri despre CodeCombat."
+    contributor_emails: "Contributor Class Emails"
+    contribute_prefix: "Căutăm oameni să se alăture distracției! Intră pe "
+    contribute_page: "pagina de contribuție"
+    contribute_suffix: " pentru a afla mai multe."
+    email_toggle: "Alege tot"
+    error_saving: "Salvare erori"
+    saved: "Modificări salvate"
+    password_mismatch: "Parola nu se potrivește."
 
-#  account_profile:
-#    edit_settings: "Edit Settings"
-#    profile_for_prefix: "Profile for "
-#    profile_for_suffix: ""
-#    profile: "Profile"
-#    user_not_found: "No user found. Check the URL?"
-#    gravatar_not_found_mine: "We couldn't find your profile associated with:"
-#    gravatar_not_found_email_suffix: "."
-#    gravatar_signup_prefix: "Sign up at "
-#    gravatar_signup_suffix: " to get set up!"
-#    gravatar_not_found_other: "Alas, there's no profile associated with this person's email address."
-#    gravatar_contact: "Contact"
-#    gravatar_websites: "Websites"
-#    gravatar_accounts: "As Seen On"
-#    gravatar_profile_link: "Full Gravatar Profile"
+  account_profile:
+    edit_settings: "Modifică setările"
+    profile_for_prefix: "Profil pentru "
+    profile_for_suffix: ""
+    profile: "Profil"
+    user_not_found: "Utilizator negăsit. Verifică URL-ul??"
+    gravatar_not_found_mine: "N-am putut găsi profilul asociat cu:"
+    gravatar_not_found_email_suffix: "."
+    gravatar_signup_prefix: "Înscrie-te la "
+    gravatar_signup_suffix: " pentru a fi gata!" #sounds funny # to get set up!"
+    gravatar_not_found_other: "Din păcate nu este asociat nici un profil cu această adresă de email."
+    gravatar_contact: "Contact"
+    gravatar_websites: "Website-uri"
+    gravatar_accounts: "Așa cum apare la"
+    gravatar_profile_link: "Full Gravatar Profile" #better leave this one as it is
 
-#  play_level:
-#    level_load_error: "Level could not be loaded: "
-#    done: "Done"
-#    grid: "Grid"
-#    customize_wizard: "Customize Wizard"
-#    home: "Home"
-#    guide: "Guide"
-#    multiplayer: "Multiplayer"
-#    restart: "Restart"
-#    goals: "Goals"
-#    action_timeline: "Action Timeline"
-#    click_to_select: "Click on a unit to select it."
-#    reload_title: "Reload All Code?"
-#    reload_really: "Are you sure you want to reload this level back to the beginning?"
-#    reload_confirm: "Reload All"
-#    victory_title_prefix: ""
-#    victory_title_suffix: " Complete"
-#    victory_sign_up: "Sign Up to Save Progress"
-#    victory_sign_up_poke: "Want to save your code? Create a free account!"
-#    victory_rate_the_level: "Rate the level: "
-#    victory_play_next_level: "Play Next Level"
-#    victory_go_home: "Go Home"
-#    victory_review: "Tell us more!"
-#    victory_hour_of_code_done: "Are You Done?"
-#    victory_hour_of_code_done_yes: "Yes, I'm finished with my Hour of Code™!"
-#    multiplayer_title: "Multiplayer Settings"
-#    multiplayer_link_description: "Give this link to anyone to have them join you."
-#    multiplayer_hint_label: "Hint:"
-#    multiplayer_hint: " Click the link to select all, then press ⌘-C or Ctrl-C to copy the link."
-#    multiplayer_coming_soon: "More multiplayer features to come!"
-#    guide_title: "Guide"
-#    tome_minion_spells: "Your Minions' Spells"
-#    tome_read_only_spells: "Read-Only Spells"
-#    tome_other_units: "Other Units"
-#    tome_cast_button_castable: "Cast Spell"
-#    tome_cast_button_casting: "Casting"
-#    tome_cast_button_cast: "Spell Cast"
-#    tome_autocast_delay: "Autocast Delay"
-#    tome_select_spell: "Select a Spell"
-#    tome_select_a_thang: "Select Someone for "
-#    tome_available_spells: "Available Spells"
-#    hud_continue: "Continue (press shift-space)"
-#    spell_saved: "Spell Saved"
+  play_level:
+    level_load_error: "Nivelul nu a putut fi încărcat: "
+    done: "Gata"
+    grid: "Grilă"
+    customize_wizard: "Personalizează Wizard-ul"
+    home: "Acasă"
+    guide: "Ghid"
+    multiplayer: "Multiplayer"
+    restart: "Restart"
+    goals: "Obiective"
+    action_timeline: "Timeline-ul acțiunii"
+    click_to_select: "Apasă pe o unitate pentru a o selecta."
+    reload_title: "Reîncarcă tot Codul?"
+    reload_really: "Ești sigur că vrei să reîncarci nivelul de la început?"
+    reload_confirm: "Reload All"
+    victory_title_prefix: ""
+    victory_title_suffix: " Terminat"
+    victory_sign_up: "Înscrie-te pentru a salva progresul"
+    victory_sign_up_poke: "Vrei să-ți salvezi codul? Crează un cont gratis!"
+    victory_rate_the_level: "Rate the level: "
+    victory_play_next_level: "Joacă nivelul următor"
+    victory_go_home: "Acasă"
+    victory_review: "Spune-ne mai multe!"
+    victory_hour_of_code_done: "Ai terminat?"
+    victory_hour_of_code_done_yes: "Da, am terminat Hour of Code™!"
+    multiplayer_title: "Setări Multiplayer"
+    multiplayer_link_description: "Împărtășește acest link cu cei care vor să ți se alăture."
+    multiplayer_hint_label: "Hint:"
+    multiplayer_hint: " Apasă pe link pentru a selecta tot, apoi apasă ⌘-C sau Ctrl-C pentru a copia link-ul."
+    multiplayer_coming_soon: "Mai multe feature-uri multiplayer în curând!"
+    guide_title: "Ghid"
+    tome_minion_spells: "Vrăjile Minion-ilor tăi"
+    tome_read_only_spells: "Vrăji Read-Only"
+    tome_other_units: "Alte unități"
+    tome_cast_button_castable: "Aplică Vraja"
+    tome_cast_button_casting: "Se încarcă"
+    tome_cast_button_cast: "Aplică Vraja"
+    tome_autocast_delay: "Întârziere Autocast"
+    tome_select_spell: "Alege o vrajă"
+    tome_select_a_thang: "Alege pe cineva pentru "
+    tome_available_spells: "Vrăjile disponibile"
+    hud_continue: "Continuă (apasă shift-space)"
+    spell_saved: "Vrajă salvată"
 
-#  admin:
-#    av_title: "Admin Views"
-#    av_entities_sub_title: "Entities"
-#    av_entities_users_url: "Users"
-#    av_entities_active_instances_url: "Active Instances"
-#    av_other_sub_title: "Other"
-#    av_other_debug_base_url: "Base (for debugging base.jade)"
-#    u_title: "User List"
-#    lg_title: "Latest Games"
+  admin:
+    av_title: "Admin vede"
+    av_entities_sub_title: "Entități"
+    av_entities_users_url: "Utilizatori"
+    av_entities_active_instances_url: "Instanțe active"
+    av_other_sub_title: "Altele"
+    av_other_debug_base_url: "Base (pentru debugging base.jade)"
+    u_title: "Listă utilizatori"
+    lg_title: "Ultimele jocuri"
 
-#  editor:
-#    main_title: "CodeCombat Editors"
-#    main_description: "Build your own levels, campaigns, units and educational content. We provide all the tools you need!"
-#    article_title: "Article Editor"
-#    article_description: "Write articles that give players overviews of programming concepts which can be used across a variety of levels and campaigns."
-#    thang_title: "Thang Editor"
-#    thang_description: "Build units, defining their default logic, graphics and audio. Currently only supports importing Flash exported vector graphics."
-#    level_title: "Level Editor"
+  editor:
+    main_title: "Editori CodeCombat"
+    main_description: "Construiește propriile nivele,campanii,unități și conținut educațional.Noi îți furnizăm toate uneltele necesare!"
+    article_title: "Editor Articol"
+    article_description: "Scrie articole care oferă jucătorilor cunoștințe despre conceptele de programare care pot fi folosite pe o varietate de nivele și campanii."
+    thang_title: "Editor Thang"
+    thang_description: "Construiește unități ,definește logica lor,grafica și sunetul.Momentan suportă numai importare de grafică vectorială exportată din Flash."
+    level_title: "Editor Nivele"
 #    level_description: "Includes the tools for scripting, uploading audio, and constructing custom logic to create all sorts of levels. Everything we use ourselves!"
 #    security_notice: "Many major features in these editors are not currently enabled by default. As we improve the security of these systems, they will be made generally available. If you'd like to use these features sooner, "
 #    contact_us: "contact us!"

From aa3167f30344c811d459aa21a4225d93febb2aa2 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Wed, 5 Mar 2014 15:06:20 -0800
Subject: [PATCH 087/178] Temporary hack fix while mongodb text search is
 broken.

---
 app/treema-ext.coffee | 17 ++++++++++++++++-
 1 file changed, 16 insertions(+), 1 deletion(-)

diff --git a/app/treema-ext.coffee b/app/treema-ext.coffee
index 674e6215a..1a8bdbe40 100644
--- a/app/treema-ext.coffee
+++ b/app/treema-ext.coffee
@@ -279,12 +279,23 @@ class LatestVersionReferenceNode extends TreemaNode
   search: =>
     term = @getValEl().find('input').val()
     return if term is @lastTerm
+    
+    # HACK while search is broken
+    if @collection
+      @lastTerm = term
+      @searchCallback()
+      return
+      
     @getSearchResultsEl().empty() if @lastTerm and not term
     return unless term
     @lastTerm = term
     @getSearchResultsEl().empty().append('Searching')
     @collection = new LatestVersionCollection()
-    @collection.url = "#{@url}?term=#{term}&project=true"
+
+    # HACK while search is broken
+#    @collection.url = "#{@url}?term=#{term}&project=true"
+    @collection.url = "#{@url}?term=#{''}&project=true"
+    
     @collection.fetch()
     @collection.on 'sync', @searchCallback
 
@@ -295,6 +306,10 @@ class LatestVersionReferenceNode extends TreemaNode
       row = $('<div></div>').addClass('treema-search-result-row')
       text = @formatDocument(model)
       continue unless text?
+      
+      # HACK while search is broken
+      continue unless text.toLowerCase().indexOf(@lastTerm.toLowerCase()) >= 0
+      
       row.addClass('treema-search-selected') if first
       first = false
       row.text(text)

From 99f43028a02e77473124340745ef68531351f542 Mon Sep 17 00:00:00 2001
From: Akaza Akari <tt@a-kar.in>
Date: Wed, 5 Mar 2014 19:39:14 -0800
Subject: [PATCH 088/178] Add thang sound panning for #454

---
 app/lib/AudioPlayer.coffee        | 15 +++++++++++++--
 app/lib/surface/CocoSprite.coffee |  9 ++++++++-
 app/lib/surface/Surface.coffee    |  1 +
 3 files changed, 22 insertions(+), 3 deletions(-)

diff --git a/app/lib/AudioPlayer.coffee b/app/lib/AudioPlayer.coffee
index b5eb4f70d..9c55282f2 100644
--- a/app/lib/AudioPlayer.coffee
+++ b/app/lib/AudioPlayer.coffee
@@ -38,6 +38,7 @@ class AudioPlayer extends CocoClass
   constructor: () ->
     super()
     @ext = if createjs.Sound.getCapability('mp3') then '.mp3' else '.ogg'
+    @camera = null
     @listenToSound()
     @createNewManifest()
     @soundsToPlayWhenLoaded = {}
@@ -51,6 +52,13 @@ class AudioPlayer extends CocoClass
     # So for now, we'll just load through SoundJS instead.
     createjs.Sound.on 'fileload', @onSoundLoaded
 
+  applyPanning: (options, pos) ->
+    sup = @camera.worldToSurface pos
+    svp = @camera.surfaceViewport
+    pan = Math.max -1, Math.min 1, ((sup.x - svp.x) - svp.width / 2) / svp.width * 2
+    # TODO: derive new volume from old one and distance ratio
+    volume: options.volume, delay: options.delay, pan: pan
+
   # PUBLIC LOADING METHODS
 
   soundForDialogue: (message, soundTriggers) ->
@@ -78,8 +86,11 @@ class AudioPlayer extends CocoClass
       @preloadInterfaceSounds [name] unless filename of cache
       @soundsToPlayWhenLoaded[name] = volume
 
-  playSound: (name, volume=1, delay=0) ->
-    instance = createjs.Sound.play name, {volume: (me.get('volume') ? 1) * volume, delay: delay}
+  playSound: (name, volume=1, delay=0, pos=null) ->
+    audioOptions = {volume: (me.get('volume') ? 1) * volume, delay: delay}
+    unless @camera is null or pos is null
+      audioOptions = @applyPanning audioOptions, pos
+    instance = createjs.Sound.play name, audioOptions
     instance
 
 #  # TODO: load Interface sounds somehow, somewhere, somewhen
diff --git a/app/lib/surface/CocoSprite.coffee b/app/lib/surface/CocoSprite.coffee
index 3a56f77e7..9104f1b37 100644
--- a/app/lib/surface/CocoSprite.coffee
+++ b/app/lib/surface/CocoSprite.coffee
@@ -188,6 +188,13 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
     return 0 unless @thang.bobHeight
     @thang.bobHeight * (1 + Math.sin(@age * Math.PI / @thang.bobTime))
 
+  getWorldPosition: ->
+    p1 = @thang.pos
+    if bobOffset = @getBobOffset()
+      p1 = p1.copy?() or _.clone(p1)
+      p1.z += bobOffset
+    x: p1.x, y: p1.y, z: if @thang.isLand then 0 else p1.z - @thang.depth / 2
+
   updatePosition: ->
     return unless @thang?.pos and @options.camera?
     [p0, p1] = [@lastPos, @thang.pos]
@@ -477,6 +484,6 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
     return null unless sound
     delay = if withDelay and sound.delay then 1000 * sound.delay / createjs.Ticker.getFPS() else 0
     name = AudioPlayer.nameForSoundReference sound
-    instance = AudioPlayer.playSound name, volume, delay
+    instance = AudioPlayer.playSound name, volume, delay, @getWorldPosition()
 #    console.log @thang?.id, "played sound", name, "with delay", delay, "volume", volume, "and got sound instance", instance
     instance
diff --git a/app/lib/surface/Surface.coffee b/app/lib/surface/Surface.coffee
index ec006d449..c5e15813d 100644
--- a/app/lib/surface/Surface.coffee
+++ b/app/lib/surface/Surface.coffee
@@ -360,6 +360,7 @@ module.exports = Surface = class Surface extends CocoClass
     canvasHeight = parseInt(@canvas.attr('height'), 10)
     @camera?.destroy()
     @camera = new Camera canvasWidth, canvasHeight
+    AudioPlayer.camera = @camera
     @layers.push @surfaceLayer = new Layer name: "Surface", layerPriority: 0, transform: Layer.TRANSFORM_SURFACE, camera: @camera
     @layers.push @surfaceTextLayer = new Layer name: "Surface Text", layerPriority: 1, transform: Layer.TRANSFORM_SURFACE_TEXT, camera: @camera
     @layers.push @screenLayer = new Layer name: "Screen", layerPriority: 2, transform: Layer.TRANSFORM_SCREEN, camera: @camera

From 51d7bd656d1b17e35c1ff7e484e10fadf37b4ed5 Mon Sep 17 00:00:00 2001
From: Akaza Akari <tt@a-kar.in>
Date: Wed, 5 Mar 2014 19:54:58 -0800
Subject: [PATCH 089/178] Made updatePosition use getWorldPosition

---
 app/lib/surface/CocoSprite.coffee | 5 +----
 1 file changed, 1 insertion(+), 4 deletions(-)

diff --git a/app/lib/surface/CocoSprite.coffee b/app/lib/surface/CocoSprite.coffee
index 9104f1b37..57146866a 100644
--- a/app/lib/surface/CocoSprite.coffee
+++ b/app/lib/surface/CocoSprite.coffee
@@ -197,12 +197,9 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
 
   updatePosition: ->
     return unless @thang?.pos and @options.camera?
+    wop = @getWorldPosition()
     [p0, p1] = [@lastPos, @thang.pos]
-    if bobOffset = @getBobOffset()
-      p1 = p1.copy?() or _.clone(p1)
-      p1.z += bobOffset
     return if p0 and p0.x is p1.x and p0.y is p1.y and p0.z is p1.z and not @options.camera.tweeningZoomTo
-    wop = x: p1.x, y: p1.y, z: if @thang.isLand then 0 else p1.z - @thang.depth / 2
     sup = @options.camera.worldToSurface wop
     [@displayObject.x, @displayObject.y] = [sup.x, sup.y]
     @lastPos = p1.copy?() or _.clone(p1)

From 030da44c23ae66668405f5f7cce0fc3a5bcf778e Mon Sep 17 00:00:00 2001
From: Akaza Akari <tt@a-kar.in>
Date: Wed, 5 Mar 2014 20:12:42 -0800
Subject: [PATCH 090/178] Added volume manipulation, adjusted panning
 parameters

---
 app/lib/AudioPlayer.coffee | 5 +++--
 1 file changed, 3 insertions(+), 2 deletions(-)

diff --git a/app/lib/AudioPlayer.coffee b/app/lib/AudioPlayer.coffee
index 9c55282f2..40b07974d 100644
--- a/app/lib/AudioPlayer.coffee
+++ b/app/lib/AudioPlayer.coffee
@@ -55,8 +55,9 @@ class AudioPlayer extends CocoClass
   applyPanning: (options, pos) ->
     sup = @camera.worldToSurface pos
     svp = @camera.surfaceViewport
-    pan = Math.max -1, Math.min 1, ((sup.x - svp.x) - svp.width / 2) / svp.width * 2
-    # TODO: derive new volume from old one and distance ratio
+    pan = Math.max -1, Math.min 1, ((sup.x - svp.x) - svp.width / 2) / svp.width
+    dst = @camera.distanceRatioTo pos
+    vol = Math.min 1, options.volume / Math.pow (dst + 0.2), 2
     volume: options.volume, delay: options.delay, pan: pan
 
   # PUBLIC LOADING METHODS

From 929d10588e70dfd2eb5a45169b19947256ad6b5e Mon Sep 17 00:00:00 2001
From: Akaza Akari <tt@a-kar.in>
Date: Thu, 6 Mar 2014 16:02:42 +0800
Subject: [PATCH 091/178] Translate passages on Contribute and About page

---
 app/locale/zh-HANS.coffee | 86 +++++++++++++++++++--------------------
 1 file changed, 43 insertions(+), 43 deletions(-)

diff --git a/app/locale/zh-HANS.coffee b/app/locale/zh-HANS.coffee
index f6eebce5d..4f999712c 100644
--- a/app/locale/zh-HANS.coffee
+++ b/app/locale/zh-HANS.coffee
@@ -271,16 +271,16 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
     who_description_prefix: "在2013年开始一起编写 CodeCombat。在2008年时,我们还创造"
     who_description_suffix: "并且发展出了首选的学习如何写中文和日文的Web和IOS应用"
     who_description_ending: "现在是时候教人们如何写代码了。"
-#    why_paragraph_1: "When making Skritter, George didn't know how to program and was constantly frustrated by his inability to implement his ideas. Afterwards, he tried learning, but the lessons were too slow. His housemate, wanting to reskill and stop teaching, tried Codecademy, but \"got bored.\" Each week another friend started Codecademy, then dropped off. We realized it was the same problem we'd solved with Skritter: people learning a skill via slow, intensive lessons when what they need is fast, extensive practice. We know how to fix that."
-#    why_paragraph_2: "Need to learn to code? You don't need lessons. You need to write a lot of code and have a great time doing it."
-#    why_paragraph_3_prefix: "That's what programming is about. It's gotta be fun. Not fun like"
-#    why_paragraph_3_italic: "yay a badge"
-#    why_paragraph_3_center: "but fun like"
-#    why_paragraph_3_italic_caps: "NO MOM I HAVE TO FINISH THE LEVEL!"
-#    why_paragraph_3_suffix: "That's why CodeCombat is a multiplayer game, not a gamified lesson course. We won't stop until you can't stop--but this time, that's a good thing."
-#    why_paragraph_4: "If you're going to get addicted to some game, get addicted to this one and become one of the wizards of the tech age."
-#    why_ending: "And hey, it's free. "
-#    why_ending_url: "Start wizarding now!"
+    why_paragraph_1: "当我们制作 Skritter 的时候,George 不会写程序,对于不能实现他的灵感这一点很苦恼。他试着学了学,但是那些课程都太慢了。他的室友想不再教书学习新技能,试了试 CodeAcademy,但是觉得“太无聊了。”每个星期都会有个熟人尝试 CodeAcademy,然后无一例外地放弃掉。我们发现这和 Skritter 想要解决的是一个问题:人们想要的是高速学习、充分练习,得到的却是缓慢、冗长的课程。我们知道该怎么办了。"
+    why_paragraph_2: "你想学编程?你不用上课。你需要的是写好多代码,并且享受这个过程。"
+    why_paragraph_3_prefix: "这才是编程的要义。编程必须要好玩。不是"
+    why_paragraph_3_italic: "哇又一个奖章诶"
+    why_paragraph_3_center: "那种“好玩”,而是"
+    why_paragraph_3_italic_caps: "不老妈,我德先把这关打完!"
+    why_paragraph_3_suffix: "这就是为什么 CodeCombat 是个多人游戏,而不是一个游戏化的编程课。你不停,我们就不停——但这次这是件好事。"
+    why_paragraph_4: "如果你一定要对游戏上瘾,那就对这个游戏上瘾,然后成为科技时代的法师吧。"
+    why_ending: "再说,这游戏还是免费的。"
+    why_ending_url: "开始学习法术吧!"
 #    george_description: "CEO, business guy, web designer, game designer, and champion of beginning programmers everywhere."
 #    scott_description: "Programmer extraordinaire, software architect, kitchen wizard, and master of finances. Scott is the reasonable one."
 #    nick_description: "Programming wizard, eccentric motivation mage, and upside-down experimenter. Nick can do anything and chooses to build CodeCombat."
@@ -350,18 +350,18 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
 
   contribute:
     page_title: "贡献"
-#    character_classes_title: "Character Classes"
-#    introduction_desc_intro: "We have high hopes for CodeCombat."
-#    introduction_desc_pref: "We want to be where programmers of all stripes come to learn and play together, introduce others to the wonderful world of coding, and reflect the best parts of the community. We can't and don't want to do that alone; what makes projects like GitHub, Stack Overflow and Linux great are the people who use them and build on them. To that end, "
-#    introduction_desc_github_url: "CodeCombat is totally open source"
-#    introduction_desc_suf: ", and we aim to provide as many ways as possible for you to take part and make this project as much yours as ours."
-#    introduction_desc_ending: "We hope you'll join our party!"
+    character_classes_title: "贡献者职业"
+    introduction_desc_intro: "我们对 CodeCombat 有很高的期望。"
+    introduction_desc_pref: "我们希望所有的程序员一起来学习和游戏,让其他人也见识到代码的美妙,并且展现出社区的最好一面。我们不能也不想独自完成这个目标:让 GitHub、Stack Overflow 和 Linux 真正伟大的是它们的用户。为了完成这个目标,"
+    introduction_desc_github_url: "我们把 CodeCombat 完全开源"
+    introduction_desc_suf: ",而且我们希望提供尽可能多的方法让你来参加这个项目,与我们一起创造。"
+    introduction_desc_ending: "我们希望你也会加入进来!"
 #    introduction_desc_signature: "- Nick, George, Scott, Michael, and Jeremy"
-#    alert_account_message_intro: "Hey there!"
-#    alert_account_message_pref: "To subscribe for class emails, you'll need to "
-#    alert_account_message_suf: "first."
-#    alert_account_message_create_url: "create an account"
-#    archmage_summary: "Interested in working on game graphics, user interface design, database and server organization, multiplayer networking, physics, sound, or game engine performance? Want to help build a game to help other people learn what you are good at? We have a lot to do and if you are an experienced programmer and want to develop for CodeCombat, this class is for you. We would love your help building the best programming game ever."
+    alert_account_message_intro: "你好!"
+    alert_account_message_pref: "要订阅贡献者邮件,你得先"
+    alert_account_message_suf: "。"
+    alert_account_message_create_url: "创建账号"
+    archmage_summary: "你对游戏图像、界面设计、数据库和服务器运营、多人在线、物理、声音、游戏引擎性能感兴趣吗?想做一个教别人编程的游戏吗?如果你有编程经验,想要开发 CodeCombat ,那就选择这个职业吧。我们会非常高兴在制作史上最好的编程游戏的过程中得到你的帮助。"
 #    archmage_introduction: "One of the best parts about building games is they synthesize so many different things. Graphics, sound, real-time networking, social networking, and of course many of the more common aspects of programming, from low-level database management, and server administration to user facing design and interface building. There's a lot to do, and if you're an experienced programmer with a hankering to really dive into the nitty-gritty of CodeCombat, this class might be for you. We would love to have your help building the best programming game ever."
 #    class_attributes: "Class Attributes"
 #    archmage_attribute_1_pref: "Knowledge in "
@@ -374,10 +374,10 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
 #    join_desc_4: "and we'll go from there!"
 #    join_url_email: "Email us"
 #    join_url_hipchat: "public HipChat room"
-#    more_about_archmage: "Learn More About Becoming an Archmage"
+    more_about_archmage: "了解成为大法师的方法"
 #    archmage_subscribe_desc: "Get emails on new coding opportunities and announcements."
-#    artisan_summary_pref: "Want to design levels and expand CodeCombat's arsenal? People are playing through our content at a pace faster than we can build! Right now, our level editor is barebone, so be wary. Making levels will be a little challenging and buggy. If you have visions of campaigns spanning for-loops to"
-#    artisan_summary_suf: "then this class is for you."
+    artisan_summary_pref: "想要设计 CodeCombat 的关卡吗?人们玩的比我们做的快多了!现在我们的关卡编辑器还很基本,所以做起关卡来会有点麻烦,还会有bug。只要你有制作关卡的灵感,不管是简单的for循环还是"
+    artisan_summary_suf: "这种东西,这个职业都很适合你。"
 #    artisan_introduction_pref: "We must construct additional levels! People be clamoring for more content, and we can only build so many ourselves. Right now your workstation is level one; our level editor is barely usable even by its creators, so be wary. If you have visions of campaigns spanning for-loops to"
 #    artisan_introduction_suf: "then this class might be for you."
 #    artisan_attribute_1: "Any experience in building content like this would be nice, such as using Blizzard's level editors. But not required!"
@@ -388,52 +388,52 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
 #    artisan_join_step2: "Create a new level and explore existing levels."
 #    artisan_join_step3: "Find us in our public HipChat room for help."
 #    artisan_join_step4: "Post your levels on the forum for feedback."
-#    more_about_artisan: "Learn More About Becoming an Artisan"
+    more_about_artisan: "了解成为工匠的方法"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+    adventurer_sumamry: "丑话说在前面,你就是那个挡枪子的,而且你会伤得很重。我们需要人手来测试崭新的关卡,并且提出改进意见。做一个好游戏是一个漫长的过程,没人第一次就能搞对。如果你能忍得了这些,而且身体健壮,那这个职业就是你的了。"
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
 #    adventurer_join_pref: "Either get together with (or recruit!) an Artisan and work with them, or check the box below to receive emails when there are new levels to test. We'll also be posting about levels to review on our networks like"
 #    adventurer_forum_url: "our forum"
 #    adventurer_join_suf: "so if you prefer to be notified those ways, sign up there!"
-#    more_about_adventurer: "Learn More About Becoming an Adventurer"
+    more_about_adventurer: "了解成为冒险家的方法"
 #    adventurer_subscribe_desc: "Get emails when there are new levels to test."
-#    scribe_summary_pref: "CodeCombat is not just going to be a bunch of levels. It will also be a resource of programming knowledge that players can hook into. That way, each Artisan can link to a detailed article that for the player's edification: documentation akin to what the "
-#    scribe_summary_sufx: " has built. If you enjoy explaining programming concepts, then this class is for you."
+    scribe_summary_pref: "CodeCombat 不只是一堆关卡的集合,它还是玩家们编程知识的来源。这样的话,每个工匠都能链接详尽的文档,以供玩家们学习,类似于"
+    scribe_summary_sufx: "那些。如果你喜欢解释编程概念,那这个职业适合你。"
 #    scribe_introduction_pref: "CodeCombat isn't just going to be a bunch of levels. It will also include a resource for knowledge, a wiki of programming concepts that levels can hook into. That way rather than each Artisan having to describe in detail what a comparison operator is, they can simply link their level to the Article describing them that is already written for the player's edification. Something along the lines of what the "
 #    scribe_introduction_url_mozilla: "Mozilla Developer Network"
 #    scribe_introduction_suf: " has built. If your idea of fun is articulating the concepts of programming in Markdown form, then this class might be for you."
 #    scribe_attribute_1: "Skill in words is pretty much all you need. Not only grammar and spelling, but able to convey complicated ideas to others."
 #    contact_us_url: "Contact us"
 #    scribe_join_description: "tell us a little about yourself, your experience with programming and what sort of things you'd like to write about. We'll go from there!"
-#    more_about_scribe: "Learn More About Becoming a Scribe"
+    more_about_scribe: "了解成为文书的方法"
 #    scribe_subscribe_desc: "Get emails about article writing announcements."
-    diplomat_summary: "在其他国家不讲英语,很多人对于CodeCombat有很大的兴趣。我们正在寻找愿意花时间翻译网站语料库的词语的译者,这样 CodeCombat 就能尽快地遍及世界各地。如果你想帮助 CodeCombat 的国际化,那么这个类就是给你的。"
-    diplomat_introduction_pref: "如果有一件事情是从 "
-    diplomat_launch_url: "launch in October"
-    diplomat_introduction_suf: "学来的,Combat有相当大的兴趣在其他国家发展。我们正在构建一个译者兵团把单词一个一个的翻译,让CodeCombat尽可能地访问到世界各地。如果你喜欢偷偷地瞄一眼即将到来的内容,并让你的国民尽快学习到CodeCombat,那么这个类可能适合你。"
-    diplomat_attribute_1: "会流利的英语和能翻译的语言。当传递复杂思想,难得的是能很好的同时掌握这两个。"
-    diplomat_join_pref_github: "找到你自己的语言文件 "
-    diplomat_github_url: "在GitHub网站"
-    diplomat_join_suf_github: "在线编辑它,然后提交一个合并请求。同时,选中下面这个复选框来关注最新的国际化开发!"
-    more_about_diplomat: "了解更多“如何成为一名外交官(翻译者)”"
+    diplomat_summary: "很多国家不说英文,但是人们对 CodeCombat 兴致很高!我们需要具有热情的翻译者,来把这个网站上的文字尽快带向全世界。如果你想帮我们走向全球,那这个职业适合你。"
+    diplomat_introduction_pref: "如果说我们从"
+    diplomat_launch_url: "十月的发布"
+    diplomat_introduction_suf: "中得到了什么启发:那就是全球的人对 CodeCombat 都很感兴趣。我们召集了一群翻译者,尽快地把网站上的信息翻译成各国文字。如果你对即将发布的新内容感兴趣,想让你的国家的人们玩上,就快来成为外交官吧。"
+    diplomat_attribute_1: "既会说流利的英语,也熟悉自己的语言。编程是一件很复杂的事情,而要翻译复杂的概念,你必须对两种语言都在行!"
+    diplomat_join_pref_github: "在"
+    diplomat_github_url: "GitHub"
+    diplomat_join_suf_github: "找到你的语言文件,在线编辑它,然后提交一个合并请求。同时,选中下面这个复选框来关注最新的国际化开发!"
+    more_about_diplomat: "了解成为外交官的方法"
     diplomat_subscribe_desc: "接受有关国际化开发和翻译情况的邮件"
-#    ambassador_summary: "We are trying to build a community, and every community needs a support team when there are troubles. We have got chats, emails, and social networks so that our users can get acquainted with the game. If you want to help people get involved, have fun, and learn some programming, then this class is for you."
+    ambassador_summary: "我们要建立一个社区,而当社区遇到麻烦的时候,就要支持人员出场了。我们运用 IRC、电邮、社交网站等多种平台帮助玩家熟悉游戏。如果你想帮人们参与进来,学习编程,然后玩的开心,那这个职业属于你。"
 #    ambassador_introduction: "This is a community we're building, and you are the connections. We've got Olark chats, emails, and social networks with lots of people to talk with and help get acquainted with the game and learn from. If you want to help people get involved and have fun, and get a good feel of the pulse of CodeCombat and where we're going, then this class might be for you."
 #    ambassador_attribute_1: "Communication skills. Be able to identify the problems players are having and help them solve them. Also, keep the rest of us informed about what players are saying, what they like and don't like and want more of!"
 #    ambassador_join_desc: "tell us a little about yourself, what you've done and what you'd be interested in doing. We'll go from there!"
 #    ambassador_join_note_strong: "Note"
 #    ambassador_join_note_desc: "One of our top priorities is to build multiplayer where players having difficulty solving levels can summon higher level wizards to help them. This will be a great way for ambassadors to do their thing. We'll keep you posted!"
-#    more_about_ambassador: "Learn More About Becoming an Ambassador"
+    more_about_ambassador: "了解成为使节的方法"
 #    ambassador_subscribe_desc: "Get emails on support updates and multiplayer developments."
-#    counselor_summary: "None of the above roles fit what you are interested in? Do not worry, we are on the lookout for anybody who wants a hand in the development of CodeCombat! If you are interested in teaching, game development, open source management, or anything else that you think will be relevant to us, then this class is for you."
+    counselor_summary: "以上的职业都不适合你?没关系,我们欢迎每一个想参与 CodeCombat 开发的人!如果你熟悉教学、游戏开发、开源管理,或者任何你觉得和我们有关的方面,那这个职业属于你。"
 #    counselor_introduction_1: "Do you have life experience? A different perspective on things that can help us decide how to shape CodeCombat? Of all these roles, this will probably take the least time, but individually you may make the most difference. We're on the lookout for wisened sages, particularly in areas like: teaching, game development, open source project management, technical recruiting, entrepreneurship, or design."
 #    counselor_introduction_2: "Or really anything that is relevant to the development of CodeCombat. If you have knowledge and want to share it to help grow this project, then this class might be for you."
 #    counselor_attribute_1: "Experience, in any of the areas above or something you think might be helpful."
 #    counselor_attribute_2: "A little bit of free time!"
 #    counselor_join_desc: "tell us a little about yourself, what you've done and what you'd be interested in doing. We'll put you in our contact list and be in touch when we could use advice (not too often)."
-#    more_about_counselor: "Learn More About Becoming a Counselor"
+    more_about_counselor: "了解成为顾问的方式"
     changes_auto_save: "在您切换复选框时,更改将自动保存。"
     diligent_scribes: "我们勤奋的文书:"
     powerful_archmages: "我们强力的大法师:"

From ad3019a6cbefe573e7b8d1e88cbd900ffedd3ff1 Mon Sep 17 00:00:00 2001
From: Akaza Akari <tt@a-kar.in>
Date: Thu, 6 Mar 2014 16:51:21 +0800
Subject: [PATCH 092/178] Translated more strings

---
 app/locale/zh-HANS.coffee | 80 +++++++++++++++++++--------------------
 1 file changed, 40 insertions(+), 40 deletions(-)

diff --git a/app/locale/zh-HANS.coffee b/app/locale/zh-HANS.coffee
index 4f999712c..caa2b9802 100644
--- a/app/locale/zh-HANS.coffee
+++ b/app/locale/zh-HANS.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
     tome_available_spells: "可用的法术"
     hud_continue: "继续(按 Shift-空格)"
     spell_saved: "咒语已保存"
-#    skip_tutorial: "Skip (esc)"
+    skip_tutorial: "跳过(esc)"
 
   admin:
     av_title: "管理员视图"
@@ -321,32 +321,32 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
     code_title: "代码 - MIT"
     code_description_prefix: "所有由 CodeCombat 拥有或者托管在 codecombat.com 的代码,在 GitHub 版本库或者 codecombat.com 数据库,以上许可协议都依照"
     mit_license_url: "MIT 许可证"
-#    code_description_suffix: "This includes all code in Systems and Components that are made available by CodeCombat for the purpose of creating levels."
-#    art_title: "Art/Music - Creative Commons "
-#    art_description_prefix: "All common content is available under the"
+    code_description_suffix: "这包括所有 CodeCombat 公开的制作关卡用的系统和组件代码。"
+    art_title: "美术和音乐 - Creative Commons"
+    art_description_prefix: "所有共通的内容都在"
 #    cc_license_url: "Creative Commons Attribution 4.0 International License"
-#    art_description_suffix: "Common content is anything made generally available by CodeCombat for the purpose of creating Levels. This includes:"
+    art_description_suffix: "条款下公开。共通内容是指所有 CodeCombat 发布出来用于制作关卡的内容。这包括:"
     art_music: "音乐"
     art_sound: "声效"
-    art_artwork: "艺术品"
-    art_sprites: "小妖精"
-#    art_other: "Any and all other non-code creative works that are made available when creating Levels."
-#    art_access: "Currently there is no universal, easy system for fetching these assets. In general, fetch them from the URLs as used by the site, contact us for assistance, or help us in extending the site to make these assets more easily accessible."
-#    art_paragraph_1: "For attribution, please name and link to codecombat.com near where the source is used or where appropriate for the medium. For example:"
-#    use_list_1: "If used in a movie or another game, include codecombat.com in the credits."
-#    use_list_2: "If used on a website, include a link near the usage, for example underneath an image, or in a general attributions page where you might also mention other Creative Commons works and open source software being used on the site. Something that's already clearly referencing CodeCombat, such as a blog post mentioning CodeCombat, does not need some separate attribution."
-#    art_paragraph_2: "If the content being used is created not by CodeCombat but instead by a user of codecombat.com, attribute them instead, and follow attribution directions provided in that resource's description if there are any."
+    art_artwork: "图画"
+    art_sprites: "精灵"
+    art_other: "所有制作关卡时公开的,不是代码的创造性产品。"
+    art_access: "目前还没有简便通用的下载素材的方式。一般来讲,从网站上使用的URL下载,或者联系我们寻找帮助。当然你也可以帮我们扩展网站,让这些资源更容易下载。"
+    art_paragraph_1: "关于署名,请说明并在使用处附近,或对媒体形式来说合适的地方提供一个 codecombat.com 的链接。举例:"
+    use_list_1: "如果是用在电影里或者其他游戏里,请在制作人员表中加入 codecombat.com 。"
+    use_list_2: "如果用在网站上,将链接在使用的地方附近,比如图片下面,或者一个你放置其他 Creative Commons 署名和开源软件协议的专门页面。如果你的内容明确提到关于 CodeCombat,那你就不需要额外署名。"
+    art_paragraph_2: "如果你使用的内容不是由 CodeCombat 制作,而是由 codecombat.com 上其他的用户制作的,那你应该给他们署名。如果相应资源的页面上有署名指示,那你应该遵循那些指示。"
     rights_title: "版权所有"
     rights_desc: "所有关卡由他们自己版权所有。这包括"
     rights_scripts: "脚本"
     rights_unit: "单元配置"
     rights_description: "描述"
     rights_writings: "作品"
-#    rights_media: "Media (sounds, music) and any other creative content made specifically for that Level and not made generally available when creating Levels."
-#    rights_clarification: "To clarify, anything that is made available in the Level Editor for the purpose of making levels is under CC, whereas the content created with the Level Editor or uploaded in the course of creation of Levels is not."
-#    nutshell_title: "In a Nutshell"
-#    nutshell_description: "Any resources we provide in the Level Editor are free to use as you like for creating Levels. But we reserve the right to restrict distribution of the Levels themselves (that are created on codecombat.com) so that they may be charged for in the future, if that's what ends up happening."
-#    canonical: "The English version of this document is the definitive, canonical version. If there are any discrepencies between translations, the English document takes precedence."
+    rights_media: "声音、音乐以及其他专门为某个关卡制作,而不对其他关卡开放的创造性内容"
+    rights_clarification: "澄清:所有在关卡编辑器里公开用于制作关卡的资源都是在CC协议下发布的,而使用关卡编辑器制作,或者在关卡制作过程中上传的内容则不是。"
+    nutshell_title: "简而言之"
+    nutshell_description: "我们在关卡编辑器里公开的任何资源,你都可以在制作关卡时随意使用,但我们保留限制在 codecombat.com 之上创建的关卡本身传播的权利,因为我们以后可能决定为它们收费。"
+    canonical: "这篇说明的英文版本是权威版本。如果各个翻译版本之间有任何冲突,以英文版为准。"
 
   contribute:
     page_title: "贡献"
@@ -362,20 +362,20 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
     alert_account_message_suf: "。"
     alert_account_message_create_url: "创建账号"
     archmage_summary: "你对游戏图像、界面设计、数据库和服务器运营、多人在线、物理、声音、游戏引擎性能感兴趣吗?想做一个教别人编程的游戏吗?如果你有编程经验,想要开发 CodeCombat ,那就选择这个职业吧。我们会非常高兴在制作史上最好的编程游戏的过程中得到你的帮助。"
-#    archmage_introduction: "One of the best parts about building games is they synthesize so many different things. Graphics, sound, real-time networking, social networking, and of course many of the more common aspects of programming, from low-level database management, and server administration to user facing design and interface building. There's a lot to do, and if you're an experienced programmer with a hankering to really dive into the nitty-gritty of CodeCombat, this class might be for you. We would love to have your help building the best programming game ever."
-#    class_attributes: "Class Attributes"
-#    archmage_attribute_1_pref: "Knowledge in "
-#    archmage_attribute_1_suf: ", or a desire to learn. Most of our code is in this language. If you're a fan of Ruby or Python, you'll feel right at home. It's JavaScript, but with a nicer syntax."
-#    archmage_attribute_2: "Some experience in programming and personal initiative. We'll help you get oriented, but we can't spend much time training you."
-#    how_to_join: "How To Join"
-#    join_desc_1: "Anyone can help out! Just check out our "
-#    join_desc_2: "to get started, and check the box below to mark yourself as a brave Archmage and get the latest news by email. Want to chat about what to do or how to get more deeply involved? "
-#    join_desc_3: ", or find us in our "
-#    join_desc_4: "and we'll go from there!"
-#    join_url_email: "Email us"
-#    join_url_hipchat: "public HipChat room"
+    archmage_introduction: "制作游戏的时候,最令人激动人心的事情莫过于整合诸多的要素。图像、音响、实事网络交流、社交网络,以及从底层数据库管理到服务器运维,再到用户界面的设计和实现的各种编程方面。制作游戏有很多事情要做,因此如果你有编程经验,还有深入 CodeCombat 的细节中的干劲,你可能应该选择这个职业。我们会非常高兴在制作史上最好的编程游戏的过程中得到你的帮助。"
+    class_attributes: "职业特性"
+    archmage_attribute_1_pref: "了解"
+    archmage_attribute_1_suf: ",或者想要学习。我们的多数代码都是用它写就的。如果你喜欢 Ruby 或者 Python,那你肯定会感到很熟悉。它就是 JavaScript,但它的语法更友好。"
+    archmage_attribute_2: "编程经验和干劲。我们可以帮你走上正规,但恐怕没多少时间培训你。"
+    how_to_join: "如何加入"
+    join_desc_1: "谁都可以帮忙!先看看我们的"
+    join_desc_2: ",然后勾上下面的复选框,这样你就会作为勇敢的大法师收到我们的电邮。如果你想和开发人员聊天或者更深入地参与,可以"
+    join_desc_3: "或者去我们的"
+    join_desc_4: ",然后我们有话好说!"
+    join_url_email: "给我们发邮件"
+    join_url_hipchat: " HipChat 聊天室"
     more_about_archmage: "了解成为大法师的方法"
-#    archmage_subscribe_desc: "Get emails on new coding opportunities and announcements."
+    archmage_subscribe_desc: "通过电子邮件获得新的编码机会和公告。"
     artisan_summary_pref: "想要设计 CodeCombat 的关卡吗?人们玩的比我们做的快多了!现在我们的关卡编辑器还很基本,所以做起关卡来会有点麻烦,还会有bug。只要你有制作关卡的灵感,不管是简单的for循环还是"
     artisan_summary_suf: "这种东西,这个职业都很适合你。"
 #    artisan_introduction_pref: "We must construct additional levels! People be clamoring for more content, and we can only build so many ourselves. Right now your workstation is level one; our level editor is barely usable even by its creators, so be wary. If you have visions of campaigns spanning for-loops to"
@@ -389,7 +389,7 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
 #    artisan_join_step3: "Find us in our public HipChat room for help."
 #    artisan_join_step4: "Post your levels on the forum for feedback."
     more_about_artisan: "了解成为工匠的方法"
-#    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
+    artisan_subscribe_desc: "通过电子邮件获得关卡编辑器更新和公告。"
     adventurer_sumamry: "丑话说在前面,你就是那个挡枪子的,而且你会伤得很重。我们需要人手来测试崭新的关卡,并且提出改进意见。做一个好游戏是一个漫长的过程,没人第一次就能搞对。如果你能忍得了这些,而且身体健壮,那这个职业就是你的了。"
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
@@ -398,7 +398,7 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
 #    adventurer_forum_url: "our forum"
 #    adventurer_join_suf: "so if you prefer to be notified those ways, sign up there!"
     more_about_adventurer: "了解成为冒险家的方法"
-#    adventurer_subscribe_desc: "Get emails when there are new levels to test."
+    adventurer_subscribe_desc: "通过电子邮件获得新关卡通知。"
     scribe_summary_pref: "CodeCombat 不只是一堆关卡的集合,它还是玩家们编程知识的来源。这样的话,每个工匠都能链接详尽的文档,以供玩家们学习,类似于"
     scribe_summary_sufx: "那些。如果你喜欢解释编程概念,那这个职业适合你。"
 #    scribe_introduction_pref: "CodeCombat isn't just going to be a bunch of levels. It will also include a resource for knowledge, a wiki of programming concepts that levels can hook into. That way rather than each Artisan having to describe in detail what a comparison operator is, they can simply link their level to the Article describing them that is already written for the player's edification. Something along the lines of what the "
@@ -408,7 +408,7 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
 #    contact_us_url: "Contact us"
 #    scribe_join_description: "tell us a little about yourself, your experience with programming and what sort of things you'd like to write about. We'll go from there!"
     more_about_scribe: "了解成为文书的方法"
-#    scribe_subscribe_desc: "Get emails about article writing announcements."
+    scribe_subscribe_desc: "通过电子邮件获得写作新文档的通知。"
     diplomat_summary: "很多国家不说英文,但是人们对 CodeCombat 兴致很高!我们需要具有热情的翻译者,来把这个网站上的文字尽快带向全世界。如果你想帮我们走向全球,那这个职业适合你。"
     diplomat_introduction_pref: "如果说我们从"
     diplomat_launch_url: "十月的发布"
@@ -426,13 +426,13 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
 #    ambassador_join_note_strong: "Note"
 #    ambassador_join_note_desc: "One of our top priorities is to build multiplayer where players having difficulty solving levels can summon higher level wizards to help them. This will be a great way for ambassadors to do their thing. We'll keep you posted!"
     more_about_ambassador: "了解成为使节的方法"
-#    ambassador_subscribe_desc: "Get emails on support updates and multiplayer developments."
+    ambassador_subscribe_desc: "通过电子邮件获得支持系统的现状,以及多人游戏方面的新进展。"
     counselor_summary: "以上的职业都不适合你?没关系,我们欢迎每一个想参与 CodeCombat 开发的人!如果你熟悉教学、游戏开发、开源管理,或者任何你觉得和我们有关的方面,那这个职业属于你。"
-#    counselor_introduction_1: "Do you have life experience? A different perspective on things that can help us decide how to shape CodeCombat? Of all these roles, this will probably take the least time, but individually you may make the most difference. We're on the lookout for wisened sages, particularly in areas like: teaching, game development, open source project management, technical recruiting, entrepreneurship, or design."
-#    counselor_introduction_2: "Or really anything that is relevant to the development of CodeCombat. If you have knowledge and want to share it to help grow this project, then this class might be for you."
-#    counselor_attribute_1: "Experience, in any of the areas above or something you think might be helpful."
-#    counselor_attribute_2: "A little bit of free time!"
-#    counselor_join_desc: "tell us a little about yourself, what you've done and what you'd be interested in doing. We'll put you in our contact list and be in touch when we could use advice (not too often)."
+    counselor_introduction_1: "也许你有人生的经验,也许你对 CodeCombat 的发展有独特的观点。在所有这些角色中,这个角色花费的时间可能最少,但作为个人你的价值却最高。我们在寻找各方面的贤人,尤其是在教学、游戏开发、开源软件管理、技术企业招聘、创业或者设计方面的。"
+    counselor_introduction_2: "任何和 CodeCombat 的开发有关系的又可以。如果你有知识,并且希望分享给我们,帮这个项目成长,那这个职业属于你。"
+    counselor_attribute_1: "经验。上述的任何领域,或者你认为对我们有帮助的领域。"
+    counselor_attribute_2: "一点用来谈笑风生的时间!"
+    counselor_join_desc: ",向我们介绍以下你自己:你做过什么、对什么有兴趣。当我们需要你的建议的时候,我们会联系你的(不会很经常)。"
     more_about_counselor: "了解成为顾问的方式"
     changes_auto_save: "在您切换复选框时,更改将自动保存。"
     diligent_scribes: "我们勤奋的文书:"

From d3dd0eb32d6ff703622be27c970d78331117265f Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Thu, 6 Mar 2014 09:25:16 -0800
Subject: [PATCH 093/178] Fixed i18n tag typo: adventurer_sumamry ->
 adventurer_summary.

---
 app/locale/ar.coffee      |  2 +-
 app/locale/bg.coffee      |  2 +-
 app/locale/cs.coffee      |  2 +-
 app/locale/da.coffee      |  2 +-
 app/locale/de.coffee      |  2 +-
 app/locale/el.coffee      |  2 +-
 app/locale/en-AU.coffee   |  2 +-
 app/locale/en-GB.coffee   |  2 +-
 app/locale/en-US.coffee   |  2 +-
 app/locale/es-419.coffee  |  2 +-
 app/locale/es-ES.coffee   |  2 +-
 app/locale/es.coffee      |  2 +-
 app/locale/fa.coffee      |  2 +-
 app/locale/fi.coffee      |  2 +-
 app/locale/fr.coffee      |  2 +-
 app/locale/he.coffee      |  2 +-
 app/locale/hi.coffee      |  2 +-
 app/locale/hu.coffee      |  2 +-
 app/locale/id.coffee      |  2 +-
 app/locale/it.coffee      |  2 +-
 app/locale/ja.coffee      |  2 +-
 app/locale/ko.coffee      |  2 +-
 app/locale/lt.coffee      |  2 +-
 app/locale/ms-BA.coffee   |  2 +-
 app/locale/nb.coffee      |  2 +-
 app/locale/nl.coffee      | 28 ++++++++++++++--------------
 app/locale/nn.coffee      |  2 +-
 app/locale/no.coffee      |  2 +-
 app/locale/pl.coffee      |  2 +-
 app/locale/pt-BR.coffee   |  2 +-
 app/locale/pt-PT.coffee   |  2 +-
 app/locale/pt.coffee      |  2 +-
 app/locale/ro.coffee      |  8 ++++----
 app/locale/ru.coffee      |  2 +-
 app/locale/sk.coffee      |  2 +-
 app/locale/sl.coffee      |  2 +-
 app/locale/sr.coffee      |  2 +-
 app/locale/sv.coffee      |  2 +-
 app/locale/th.coffee      |  2 +-
 app/locale/tr.coffee      |  2 +-
 app/locale/uk.coffee      |  2 +-
 app/locale/ur.coffee      |  2 +-
 app/locale/vi.coffee      |  2 +-
 app/locale/zh-HANS.coffee |  2 +-
 app/locale/zh-HANT.coffee |  2 +-
 app/locale/zh.coffee      |  2 +-
 46 files changed, 62 insertions(+), 62 deletions(-)

diff --git a/app/locale/ar.coffee b/app/locale/ar.coffee
index 1f0584c85..32716882e 100644
--- a/app/locale/ar.coffee
+++ b/app/locale/ar.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "العربية", englishDescription: "Arabi
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/bg.coffee b/app/locale/bg.coffee
index fd038c56c..2396688fe 100644
--- a/app/locale/bg.coffee
+++ b/app/locale/bg.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "български език", englishDescri
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/cs.coffee b/app/locale/cs.coffee
index 121f0da4f..ee87db33d 100644
--- a/app/locale/cs.coffee
+++ b/app/locale/cs.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "čeština", englishDescription: "Czech", tr
     artisan_join_step4: "Zveřejněte vaši úroveň na fóru pro připomínkování."
     more_about_artisan: "Dozvědět se více o tom, jak se stát kreativním Řemeslníkem"
     artisan_subscribe_desc: "Dostávat emailem oznámení a informace o aktualizacích editoru úrovní."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
     adventurer_introduction: "Ujasněme si dopředu jednu věc o vaší roli: budete jako tank. Projdete ohněm. Potřebujeme někoho, kdo odzkouší zbrusu nové úrovně a pomůže identifikovat kde je možno je zlepšit. Ten boj bude ohromný - tvorba her je dlouhý proces, který nikdo nezvládne na první pokus. Máte-li na to a vydržíte-li to, pak toto je vaše skupina."
     adventurer_attribute_1: "Touha po učení se. Vy se chcete naučit programovat a my vás to chceme naučit. Jenom, v tomto případě to budete vy, kdo bude vyučovat."
     adventurer_attribute_2: "Charismatický. Buďte mírný a pečlivě artikulujte co a jak je potřeba zlepšit."
diff --git a/app/locale/da.coffee b/app/locale/da.coffee
index ad47c71fc..d3a40c86f 100644
--- a/app/locale/da.coffee
+++ b/app/locale/da.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "dansk", englishDescription: "Danish", trans
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/de.coffee b/app/locale/de.coffee
index 04b640ce8..7e9f4b9ee 100644
--- a/app/locale/de.coffee
+++ b/app/locale/de.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "Deutsch", englishDescription: "German", tra
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/el.coffee b/app/locale/el.coffee
index 91fd4be36..831c4eee9 100644
--- a/app/locale/el.coffee
+++ b/app/locale/el.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "ελληνικά", englishDescription: "Gre
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/en-AU.coffee b/app/locale/en-AU.coffee
index 8c36e3ef7..cce71adb9 100644
--- a/app/locale/en-AU.coffee
+++ b/app/locale/en-AU.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "English (AU)", englishDescription: "English
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/en-GB.coffee b/app/locale/en-GB.coffee
index 1da9d218a..7dcad9c36 100644
--- a/app/locale/en-GB.coffee
+++ b/app/locale/en-GB.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "English (UK)", englishDescription: "English
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/en-US.coffee b/app/locale/en-US.coffee
index 9d3b2dc5a..1bcf71a71 100644
--- a/app/locale/en-US.coffee
+++ b/app/locale/en-US.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "English (US)", englishDescription: "English
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/es-419.coffee b/app/locale/es-419.coffee
index 3c2614713..237fb3a69 100644
--- a/app/locale/es-419.coffee
+++ b/app/locale/es-419.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "español (América Latina)", englishDescrip
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/es-ES.coffee b/app/locale/es-ES.coffee
index e45550e08..be9c83a7d 100644
--- a/app/locale/es-ES.coffee
+++ b/app/locale/es-ES.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "español (ES)", englishDescription: "Spanis
     artisan_join_step4: "Publica tus niveles en el foro para recibir comentarios críticos."
     more_about_artisan: "Aprende más sobre convertirte en un Artesano creativo"
     artisan_subscribe_desc: "Recibe correos sobre actualizaciones del editor de niveles y anuncios."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
     adventurer_introduction: "Hablemos claro sobre tu papel: eres el tanque. Vas a recibir fuertes daños. Necesitamos gente para probar nuestros flamantes niveles y ayudar a mejorarlos. El dolor será enorme; hacer buenos juegos es un proceso largo y nadie lo consigue a la primera. Si puedes resistir y tener una puntuación alta en Resistencia, entonces esta Clase es para ti."
     adventurer_attribute_1: "Estar sediento de conocimientos. Quieres aprender a programar y nosotros queremos enseñarte cómo hacerlo. Aunque en este caso es más probable que seas tú el que esté haciendo la mayor parte de la enseñanza."
     adventurer_attribute_2: "Carismático. Se amable pero claro a la hora de desglosar qué necesita ser mejorado y sugiere de qué formas podría hacerse."
diff --git a/app/locale/es.coffee b/app/locale/es.coffee
index b4009323b..f576a0c85 100644
--- a/app/locale/es.coffee
+++ b/app/locale/es.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "español", englishDescription: "Spanish", t
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/fa.coffee b/app/locale/fa.coffee
index 28a4bd760..a0e1c046b 100644
--- a/app/locale/fa.coffee
+++ b/app/locale/fa.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "فارسی", englishDescription: "Persian",
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/fi.coffee b/app/locale/fi.coffee
index 57e5151e4..5c78e1914 100644
--- a/app/locale/fi.coffee
+++ b/app/locale/fi.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "suomi", englishDescription: "Finnish", tran
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/fr.coffee b/app/locale/fr.coffee
index 1469c7269..330b029f8 100644
--- a/app/locale/fr.coffee
+++ b/app/locale/fr.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "français", englishDescription: "French", t
     artisan_join_step4: "Postez vos niveaux dans le forum pour avoir des retours."
     more_about_artisan: "En apprendre plus sur comment devenir un Artisan créatif"
     artisan_subscribe_desc: "Recevoir un email sur les annonces et mises à jour de l'éditeur de niveaux."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
     adventurer_introduction: "Soyons clair à propos de votre rôle : vous êtes le tank. Vous allez subir beaucoup de dommages. Nous avons besoin de gens pour essayer les nouveaux niveaux et aider à identifier comment améliorer les choses. La douleur sera énorme; faire de bons jeux est une longue tâche et personne n'y arrive du premier coup. Si vous pouvez résister et avez un gros score de constitution, alors cette classe est faite pour vous."
     adventurer_attribute_1: "Une soif d'apprendre. Vous voulez apprendre à développer et nous voulons vous apprendre. Vous allez toutefois faire la plupart de l'apprentissage."
     adventurer_attribute_2: "Charismatique. Soyez doux mais exprimez-vous sur ce qui a besoin d'être amélioré, et faites des propositions sur comment l'améliorer."
diff --git a/app/locale/he.coffee b/app/locale/he.coffee
index ddcb7484c..f1e34dc8e 100644
--- a/app/locale/he.coffee
+++ b/app/locale/he.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "עברית", englishDescription: "Hebrew",
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/hi.coffee b/app/locale/hi.coffee
index 596ecd1ce..c144127a5 100644
--- a/app/locale/hi.coffee
+++ b/app/locale/hi.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "मानक हिन्दी", englishDe
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/hu.coffee b/app/locale/hu.coffee
index ad6cd0273..bed717d2c 100644
--- a/app/locale/hu.coffee
+++ b/app/locale/hu.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "magyar", englishDescription: "Hungarian", t
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/id.coffee b/app/locale/id.coffee
index 20d16335f..9b938c230 100644
--- a/app/locale/id.coffee
+++ b/app/locale/id.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "Bahasa Indonesia", englishDescription: "Ind
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/it.coffee b/app/locale/it.coffee
index a95f98916..883ba4321 100644
--- a/app/locale/it.coffee
+++ b/app/locale/it.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "italiano", englishDescription: "Italian", t
 #    artisan_join_step4: "Post your levels on the forum for feedback."
     more_about_artisan: "Leggi di più su cosa vuol dire diventare un creativo Artigiano"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/ja.coffee b/app/locale/ja.coffee
index 4c9e4dfb9..195269c4d 100644
--- a/app/locale/ja.coffee
+++ b/app/locale/ja.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "日本語", englishDescription: "Japanese",
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/ko.coffee b/app/locale/ko.coffee
index 882f3d4ac..55fdd32f1 100644
--- a/app/locale/ko.coffee
+++ b/app/locale/ko.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "한국어", englishDescription: "Korean", t
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/lt.coffee b/app/locale/lt.coffee
index 1008a8bfb..64cfe04f2 100644
--- a/app/locale/lt.coffee
+++ b/app/locale/lt.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "lietuvių kalba", englishDescription: "Lith
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/ms-BA.coffee b/app/locale/ms-BA.coffee
index 29c914f23..e551e3e0f 100644
--- a/app/locale/ms-BA.coffee
+++ b/app/locale/ms-BA.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "Bahasa Melayu", englishDescription: "Bahasa
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/nb.coffee b/app/locale/nb.coffee
index db74e540d..68569c841 100644
--- a/app/locale/nb.coffee
+++ b/app/locale/nb.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "Norsk Bokmål", englishDescription: "Norweg
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/nl.coffee b/app/locale/nl.coffee
index 508d6e384..f4c45edc5 100644
--- a/app/locale/nl.coffee
+++ b/app/locale/nl.coffee
@@ -397,28 +397,28 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     artisan_join_step4: "Maak een bericht over jouw level op ons forum voor feedback."
     more_about_artisan: "Leer meer over hoe je een Creatieve Ambachtsman kan worden."
     artisan_subscribe_desc: "Ontvang e-mails met nieuws over de Level Editor."
-    adventurer_sumamry: "Laten we duidelijk zijn over je rol: jij bent de tank. Jij krijgt de zware klappen te verduren. We hebben mensen nodig om spiksplinternieuwe levels te proberen en te kijken hoe deze beter kunnen. De pijn zal groot zijn, het maken van een goede game is een lang proces en niemand doet het de eerste keer goed. Als jij dit kan verduren en een hoge constitution score hebt, dan is dit de klasse voor jou."
+    adventurer_summary: "Laten we duidelijk zijn over je rol: jij bent de tank. Jij krijgt de zware klappen te verduren. We hebben mensen nodig om spiksplinternieuwe levels te proberen en te kijken hoe deze beter kunnen. De pijn zal groot zijn, het maken van een goede game is een lang proces en niemand doet het de eerste keer goed. Als jij dit kan verduren en een hoge constitution score hebt, dan is dit de klasse voor jou."
     adventurer_introduction: "Laten we duidelijk zijn over je rol: jij bent de tank. Jij krijgt de zware klappen te verduren. We hebben mensen nodig om spiksplinternieuwe levels te proberen en te kijken hoe deze beter kunnen. De pijn zal groot zijn, het maken van een goede game is een lang proces en niemand doet het de eerste keer goed. Als jij dit kan verduren en een hoge constitution score hebt, dan is dit de klasse voor jou."
     adventurer_attribute_1: "Een wil om te leren. Jij wilt leren hoe je programmeert en wij willen het jou leren. Je zal overigens zelf het meeste leren doen."
     adventurer_attribute_2: "Charismatisch. Wees netjes maar duidelijk over wat er beter kan en geef suggesties over hoe het beter kan."
-    adventurer_join_pref: "Werk samen met een Ambachtsman of recruteer er een, of tik het veld hieronder aan om e-mails te ontvangen wanneer er nieuwe levels zijn om te testen. We zullen ook posten over levels die beoordeeld moeten worden op onze netwerken zoals"    
+    adventurer_join_pref: "Werk samen met een Ambachtsman of recruteer er een, of tik het veld hieronder aan om e-mails te ontvangen wanneer er nieuwe levels zijn om te testen. We zullen ook posten over levels die beoordeeld moeten worden op onze netwerken zoals"
     adventurer_forum_url: "ons forum"
-    adventurer_join_suf: "dus als je liever op deze manier wordt geïnformeerd, schrijf je daar in!"    
+    adventurer_join_suf: "dus als je liever op deze manier wordt geïnformeerd, schrijf je daar in!"
     more_about_adventurer: "Leer meer over hoe je een dappere avonturier kunt worden."
     adventurer_subscribe_desc: "Ontvang e-mails wanneer er nieuwe levels zijn die getest moeten worden."
     scribe_summary_pref: "CodeCombat is meer dan slechts een aantal levels, het zal ook een bron van kennis kennis zijn en een wiki met programmeerconcepten waar levels op in kunnen gaan. Op die manier zal een Ambachtslied  een link kunnen geven naar een artikel wat past bij een level. Net zoiets als het  "
     scribe_summary_sufx: " heeft gebouwd. Als jij het leuk vindt programmeerconcepten uit te leggen, dan is deze klasse iets voor jou."
-    scribe_introduction_pref: "CodeCombat is meer dan slechts een aantal levels, het zal ook een bron van kennis kennis zijn en een wiki met programmeerconcepten waar levels op in kunnen gaan. Op die manier zal elk Ambachtslied niet in detail hoeven uit te leggen wat een vergelijkingsoperator is, maar een link kunnen geven naar een artikel wat deze informatie bevat voor de speler. Net zoiets als het  "    
+    scribe_introduction_pref: "CodeCombat is meer dan slechts een aantal levels, het zal ook een bron van kennis kennis zijn en een wiki met programmeerconcepten waar levels op in kunnen gaan. Op die manier zal elk Ambachtslied niet in detail hoeven uit te leggen wat een vergelijkingsoperator is, maar een link kunnen geven naar een artikel wat deze informatie bevat voor de speler. Net zoiets als het  "
     scribe_introduction_url_mozilla: "Mozilla Developer Network"
     scribe_introduction_suf: " heeft gebouwd. Als jij het leuk vindt om programmeerconcepten uit te leggen in Markdown-vorm, dan is deze klasse wellicht iets voor jou."
     scribe_attribute_1: "Taal-skills zijn praktisch alles wat je nodig hebt. Niet alleen grammatica of spelling, maar ook moeilijke ideeën overbrengen aan anderen."
     contact_us_url: "Contacteer ons"
-    scribe_join_description: "vertel ons wat over jezelf, je ervaring met programmeren en over wat voor soort dingen je graag zou schrijven. Verder zien we wel!"    
-    more_about_scribe: "Leer meer over het worden van een ijverige Klerk."    
-    
-    scribe_subscribe_desc: "Ontvang e-mails met aankondigingen over het schrijven van artikelen."    
+    scribe_join_description: "vertel ons wat over jezelf, je ervaring met programmeren en over wat voor soort dingen je graag zou schrijven. Verder zien we wel!"
+    more_about_scribe: "Leer meer over het worden van een ijverige Klerk."
+
+    scribe_subscribe_desc: "Ontvang e-mails met aankondigingen over het schrijven van artikelen."
     diplomat_summary: "Er is grote interesse in CodeCombat in landen waar geen Engels wordt gesproken! We zijn op zoek naar vertalers wie tijd willen spenderen aan het vertalen van de site's corpus aan woorden zodat CodeCombat zo snel mogelijk toegankelijk wordt voor heel de wereld. Als jij wilt helpen met CodeCombat internationaal maken, dan is dit de klasse voor jou."
-    diplomat_introduction_pref: "Dus, als er iets is wat we geleerd hebben van de "    
+    diplomat_introduction_pref: "Dus, als er iets is wat we geleerd hebben van de "
     diplomat_launch_url: "release in oktober"
     diplomat_introduction_suf: "dan is het wel dat er een significante interesse is in CodeCombat in andere landen, vooral Brazilië! We zijn een corps aan vertalers aan het creëren dat ijverig de ene set woorden in een andere omzet om CodeCombat zo toegankelijk te maken als mogelijk in heel de wereld. Als jij het leuk vindt glimpsen op te vangen van aankomende content en deze levels zo snel mogelijk naar je landgenoten te krijgen, dan is dit de klasse voor jou."
     diplomat_attribute_1: "Vloeiend Engels en de taal waar naar je wilt vertalen kunnen spreken. Wanneer je moeilijke ideeën wilt overbrengen, is het belangrijk beide goed te kunnen!"
@@ -429,17 +429,17 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     diplomat_subscribe_desc: "Ontvang e-mails over i18n ontwikkelingen en levels om te vertalen."
     ambassador_summary: "We proberen een gemeenschap te bouwen en elke gemeenschap heeft een supportteam nodig wanneer er problemen zijn. We hebben chats, e-mails en sociale netwerken zodat onze gebruikers het spel kunnen leren kennen. Als jij mensen wilt helpen betrokken te raken, plezier te hebben en wat te leren programmeren, dan is dit wellicht de klasse voor jou."
     ambassador_attribute_1: "Communicatieskills. Problemen die spelers hebben kunnen identificeren en ze helpen deze op te lossen. Verder zul je ook de rest van ons geïnformeerd houden over wat de spelers zeggen, wat ze leuk vinden, wat ze minder vinden en waar er meer van moet zijn!"
-    ambassador_join_desc: "vertel ons wat over jezelf, wat je hebt gedaan en wat je graag zou doen. We zien verder wel!"    
+    ambassador_join_desc: "vertel ons wat over jezelf, wat je hebt gedaan en wat je graag zou doen. We zien verder wel!"
     ambassador_join_note_strong: "Opmerking"
-    ambassador_join_note_desc: "Een van onze topprioriteiten is om een multiplayer te bouwen waar spelers die moeite hebben een level op te lossen een wizard met een hoger level kunnen oproepen om te helpen. Dit zal een goede manier zijn voor ambassadeurs om hun ding te doen. We houden je op de hoogte!"    
+    ambassador_join_note_desc: "Een van onze topprioriteiten is om een multiplayer te bouwen waar spelers die moeite hebben een level op te lossen een wizard met een hoger level kunnen oproepen om te helpen. Dit zal een goede manier zijn voor ambassadeurs om hun ding te doen. We houden je op de hoogte!"
     more_about_ambassador: "Leer meer over het worden van een behulpzame Ambassadeur"
-    ambassador_subscribe_desc: "Ontvang e-mails met updates over ondersteuning en multiplayer-ontwikkelingen."    
+    ambassador_subscribe_desc: "Ontvang e-mails met updates over ondersteuning en multiplayer-ontwikkelingen."
     counselor_summary: "Geen van de rollen hierboven in jouw interessegebied? Maak je geen zorgen, we zijn op zoek naar iedereen die wil helpen met het ontwikkelen van CodeCombat! Als je geïnteresseerd bent in lesgeven, gameontwikkeling, open source management of iets anders waarvan je denkt dat het relevant voor ons is, dan is dit de klasse voor jou."
     counselor_introduction_1: "Heb jij levenservaring? Een afwijkend perspectief op zaken die ons kunnen helpen CodeCombat te vormen? Van alle rollen neemt deze wellicht de minste tijd in, maar individueel maak je misschien het grootste verschil. We zijn op zoek naar wijze tovenaars, vooral in het gebied van lesgeven, gameontwikkeling, open source projectmanagement, technische recrutering, ondernemerschap of design."
-    counselor_introduction_2: "Of eigenlijk alles wat relevant is voor de ontwikkeling van CodeCombat. Als jij kennis hebt en deze wilt dezen om dit project te laten groeien, dan is dit misschien de klasse voor jou."    
+    counselor_introduction_2: "Of eigenlijk alles wat relevant is voor de ontwikkeling van CodeCombat. Als jij kennis hebt en deze wilt dezen om dit project te laten groeien, dan is dit misschien de klasse voor jou."
     counselor_attribute_1: "Ervaring, in enig van de bovenstaande gebieden of iets anders waarvan je denkt dat het behulpzaam zal zijn."
     counselor_attribute_2: "Een beetje vrije tijd!"
-    counselor_join_desc: "vertel ons wat over jezelf, wat je hebt gedaan en wat je graag wilt doen. We zullen je in onze contactlijst zetten en je benaderen wanneer we je advies kunnen gebruiken (niet te vaak)."    
+    counselor_join_desc: "vertel ons wat over jezelf, wat je hebt gedaan en wat je graag wilt doen. We zullen je in onze contactlijst zetten en je benaderen wanneer we je advies kunnen gebruiken (niet te vaak)."
     more_about_counselor: "Leer meer over het worden van een waardevolle Raadgever"
     changes_auto_save: "Veranderingen worden automatisch opgeslagen wanneer je het vierkantje aan- of afvinkt."
     diligent_scribes: "Onze ijverige Klerks:"
diff --git a/app/locale/nn.coffee b/app/locale/nn.coffee
index c9e770bfa..7bd5b0307 100644
--- a/app/locale/nn.coffee
+++ b/app/locale/nn.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "Norwegian Nynorsk", englishDescription: "No
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/no.coffee b/app/locale/no.coffee
index bc0e69e0c..78fc15832 100644
--- a/app/locale/no.coffee
+++ b/app/locale/no.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "Norsk", englishDescription: "Norwegian", tr
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/pl.coffee b/app/locale/pl.coffee
index 9988be45e..32f394f1c 100644
--- a/app/locale/pl.coffee
+++ b/app/locale/pl.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "język polski", englishDescription: "Polish
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/pt-BR.coffee b/app/locale/pt-BR.coffee
index 40f374f81..be8e8079f 100644
--- a/app/locale/pt-BR.coffee
+++ b/app/locale/pt-BR.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "português do Brasil", englishDescription:
     artisan_join_step4: "Publique seus níveis no fórum para avaliação."
     more_about_artisan: "Saiba Mais Sobre Como Se Tornar Um Artesão Criativo"
     artisan_subscribe_desc: "Receba emails com novidades sobre o editor de níveis e anúncios."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
     adventurer_introduction: "Vamos ser claros sobre o seu papel: você é o tanque. Você vai tomar dano pesado. Precisamos de pessoas para experimentar níveis inéditos e ajudar a identificar como fazer as coisas melhorarem. A dor será enorme, fazer bons jogos é um processo longo e ninguém acerta na primeira vez. Se você pode suportar e ter uma alta pontuação de constituição, então esta classe pode ser para você."
     adventurer_attribute_1: "Sede de aprendizado. Você quer aprender a codificar e nós queremos ensiná-lo a codificar. Você provavelmente vai fazer a maior parte do ensino neste caso."
     adventurer_attribute_2: "Carismático. Seja gentil, mas articulado sobre o que precisa melhorar, e ofereça sugestões sobre como melhorar."
diff --git a/app/locale/pt-PT.coffee b/app/locale/pt-PT.coffee
index 77fb7afad..57278318b 100644
--- a/app/locale/pt-PT.coffee
+++ b/app/locale/pt-PT.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "Português europeu", englishDescription: "P
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/pt.coffee b/app/locale/pt.coffee
index 38a21790c..ee3395e7f 100644
--- a/app/locale/pt.coffee
+++ b/app/locale/pt.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "português", englishDescription: "Portugues
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/ro.coffee b/app/locale/ro.coffee
index 3551d730b..1fee5fe47 100644
--- a/app/locale/ro.coffee
+++ b/app/locale/ro.coffee
@@ -96,8 +96,8 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
   diplomat_suggestion:
     title: "Ajută-ne  să traducem CodeCombat!"
     sub_heading: "Avem nevoie de abilitățile tale lingvistice."
-    pitch_body: "CodeCombat este dezvoltat in limba engleza , dar deja avem jucatări din toate colțurile lumii.Mulți dintre ei vor să joace in română și nu vorbesc engleză.Dacă poți vorbi ambele te rugăm să te gândești dacă ai dori să devi un Diplomat și să ne ajuți sa traducem atât jocul cât și site-ul." 
-    missing_translations: "Until we can translate everything into Romanian, you'll see English when Romanian isn't available." 
+    pitch_body: "CodeCombat este dezvoltat in limba engleza , dar deja avem jucatări din toate colțurile lumii.Mulți dintre ei vor să joace in română și nu vorbesc engleză.Dacă poți vorbi ambele te rugăm să te gândești dacă ai dori să devi un Diplomat și să ne ajuți sa traducem atât jocul cât și site-ul."
+    missing_translations: "Until we can translate everything into Romanian, you'll see English when Romanian isn't available."
     learn_more: "Află mai multe despre cum să fi un Diplomat"
     subscribe_as_diplomat: "Înscrie-te ca Diplomat"
 
@@ -105,7 +105,7 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
     title: "Setări Wizard"
     customize_avatar: "Personalizează-ți Avatarul"
     clothes: "Haine"
-    trim: "Margine" 
+    trim: "Margine"
     cloud: "Nor"
     spell: "Vrajă"
     boots: "Încălțăminte"
@@ -389,7 +389,7 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/ru.coffee b/app/locale/ru.coffee
index 57cfc1847..51f6d1a7e 100644
--- a/app/locale/ru.coffee
+++ b/app/locale/ru.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     artisan_join_step4: "Разместите свои уровни на форуме для обратной связи."
     more_about_artisan: "Узнать больше о том, как стать Ремесленником"
     artisan_subscribe_desc: "Получать email-ы об обновлениях редактора уровней и объявления."
-    adventurer_sumamry: "Позвольте внести ясность о вашей роли: вы танк. Вы собираетесь принять тяжелые повреждения. Нам нужны люди, чтобы испытать совершенно новые уровни и помочь определить, как сделать лучше. Боль будет огромной; создание хороших игр - длительный процесс и никто не делает это правильно в первый раз. Если вы можете выдержать и имеете высокий балл конституции (D&D), этот класс для вас."
+    adventurer_summary: "Позвольте внести ясность о вашей роли: вы танк. Вы собираетесь принять тяжелые повреждения. Нам нужны люди, чтобы испытать совершенно новые уровни и помочь определить, как сделать лучше. Боль будет огромной; создание хороших игр - длительный процесс и никто не делает это правильно в первый раз. Если вы можете выдержать и имеете высокий балл конституции (D&D), этот класс для вас."
     adventurer_introduction: "Позвольте внести ясность о вашей роли: вы танк. Вы собираетесь принять тяжелые повреждения. Нам нужны люди, чтобы испытать совершенно новые уровни и помочь определить, как сделать лучше. Боль будет огромной; создание хороших игр - длительный процесс и никто не делает это правильно в первый раз. Если вы можете выдержать и имеете высокий балл конституции (D&D), этот класс для вас."
     adventurer_attribute_1: "Жажда обучения. Вы хотите научиться программировать и мы хотим научить вас программировать. Вы, вероятно, проведёте большую часть обучения в процессе."
     adventurer_attribute_2: "Харизматичность. Будьте нежны, но ясно формулируйте, что нуждается в улучшении и вносите свои предложения по улучшению."
diff --git a/app/locale/sk.coffee b/app/locale/sk.coffee
index c9e3e4023..ae22ef336 100644
--- a/app/locale/sk.coffee
+++ b/app/locale/sk.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "slovenčina", englishDescription: "Slovak",
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/sl.coffee b/app/locale/sl.coffee
index b0cbdd3f5..89b759c19 100644
--- a/app/locale/sl.coffee
+++ b/app/locale/sl.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "slovenščina", englishDescription: "Sloven
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/sr.coffee b/app/locale/sr.coffee
index c9676fa65..d3c351f39 100644
--- a/app/locale/sr.coffee
+++ b/app/locale/sr.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "српски", englishDescription: "Serbian
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/sv.coffee b/app/locale/sv.coffee
index d8b41ec50..001ce3eca 100644
--- a/app/locale/sv.coffee
+++ b/app/locale/sv.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "Svenska", englishDescription: "Swedish", tr
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/th.coffee b/app/locale/th.coffee
index 9b2134eb8..76f1aa37d 100644
--- a/app/locale/th.coffee
+++ b/app/locale/th.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "ไทย", englishDescription: "Thai", tra
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/tr.coffee b/app/locale/tr.coffee
index 4a256defe..39e268908 100644
--- a/app/locale/tr.coffee
+++ b/app/locale/tr.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "Türkçe", englishDescription: "Turkish", t
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/uk.coffee b/app/locale/uk.coffee
index bf67c2138..52d4fd3b4 100644
--- a/app/locale/uk.coffee
+++ b/app/locale/uk.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "українська мова", englishDesc
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/ur.coffee b/app/locale/ur.coffee
index 8237971a3..75a2c2e38 100644
--- a/app/locale/ur.coffee
+++ b/app/locale/ur.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "اُردُو", englishDescription: "Urdu",
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/vi.coffee b/app/locale/vi.coffee
index c3a00a167..bb7ac3b89 100644
--- a/app/locale/vi.coffee
+++ b/app/locale/vi.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "Tiếng Việt", englishDescription: "Vietn
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/zh-HANS.coffee b/app/locale/zh-HANS.coffee
index caa2b9802..fd0e42cbf 100644
--- a/app/locale/zh-HANS.coffee
+++ b/app/locale/zh-HANS.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
 #    artisan_join_step4: "Post your levels on the forum for feedback."
     more_about_artisan: "了解成为工匠的方法"
     artisan_subscribe_desc: "通过电子邮件获得关卡编辑器更新和公告。"
-    adventurer_sumamry: "丑话说在前面,你就是那个挡枪子的,而且你会伤得很重。我们需要人手来测试崭新的关卡,并且提出改进意见。做一个好游戏是一个漫长的过程,没人第一次就能搞对。如果你能忍得了这些,而且身体健壮,那这个职业就是你的了。"
+    adventurer_summary: "丑话说在前面,你就是那个挡枪子的,而且你会伤得很重。我们需要人手来测试崭新的关卡,并且提出改进意见。做一个好游戏是一个漫长的过程,没人第一次就能搞对。如果你能忍得了这些,而且身体健壮,那这个职业就是你的了。"
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/zh-HANT.coffee b/app/locale/zh-HANT.coffee
index d38532973..b6cf92b26 100644
--- a/app/locale/zh-HANT.coffee
+++ b/app/locale/zh-HANT.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "繁体中文", englishDescription: "Chinese
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."
diff --git a/app/locale/zh.coffee b/app/locale/zh.coffee
index aacf8fe4d..452df3628 100644
--- a/app/locale/zh.coffee
+++ b/app/locale/zh.coffee
@@ -390,7 +390,7 @@ module.exports = nativeDescription: "中文", englishDescription: "Chinese", tra
 #    artisan_join_step4: "Post your levels on the forum for feedback."
 #    more_about_artisan: "Learn More About Becoming an Artisan"
 #    artisan_subscribe_desc: "Get emails on level editor updates and announcements."
-#    adventurer_sumamry: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
+#    adventurer_summary: "Let us be clear about your role: you are the tank. You are going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class is for you."
 #    adventurer_introduction: "Let's be clear about your role: you are the tank. You're going to take heavy damage. We need people to try out brand-new levels and help identify how to make things better. The pain will be enormous; making good games is a long process and no one gets it right the first time. If you can endure and have a high constitution score, then this class might be for you."
 #    adventurer_attribute_1: "A thirst for learning. You want to learn how to code and we want to teach you how to code. You'll probably be doing most of the teaching in this case, though."
 #    adventurer_attribute_2: "Charismatic. Be gentle but articulate about what needs improving, and offer suggestions on how to improve."

From a03403de944d7e2f7a24de7e42c488c3f2646fa8 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Thu, 6 Mar 2014 09:35:58 -0800
Subject: [PATCH 094/178] Started some more name categories.

---
 app/lib/world/names.coffee | 19 +++++++++++++++++++
 1 file changed, 19 insertions(+)

diff --git a/app/lib/world/names.coffee b/app/lib/world/names.coffee
index 44c446583..3b1cd0080 100644
--- a/app/lib/world/names.coffee
+++ b/app/lib/world/names.coffee
@@ -88,6 +88,9 @@ module.exports.thangNames = thangNames =
     "Durfkor"
     "Paps"
   ]
+  "Peasant F": [
+    "Hilda"
+  ]
   "Archer F": [
     "Phoebe"
     "Mira"
@@ -238,3 +241,19 @@ module.exports.thangNames = thangNames =
     "Rakash"
     "Drumbaa"
   ]
+  "Burl": [
+    "Borlit"
+    "Burlosh"
+  ]
+  "Griffin Rider": [
+    "Aeoldan"
+  ]
+  "Potion Master": [
+    "Snake"
+  ]
+  "Librarian": [
+    "Hushbaum"
+  ]
+  "Equestrian": [
+    "Reynaldo"
+  ]

From 40a32d7b43e582efaa91ccfaafde56580d023889 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Thu, 6 Mar 2014 14:53:37 -0800
Subject: [PATCH 095/178] Fixed array / string ThangState key serialization
 conflicts. Fixed #333.

---
 app/lib/world/thang_state.coffee | 6 ++++--
 1 file changed, 4 insertions(+), 2 deletions(-)

diff --git a/app/lib/world/thang_state.coffee b/app/lib/world/thang_state.coffee
index 964f25eda..7db97aca6 100644
--- a/app/lib/world/thang_state.coffee
+++ b/app/lib/world/thang_state.coffee
@@ -51,7 +51,8 @@ module.exports = class ThangState
     else if type is 'array'
       specialKey = storage[@frameIndex]
       value = @specialKeysToValues[specialKey]
-      value = value.split('\x1E')  # Record Separator
+      # Remove leading and trailing Group Separators and split by any Record Separators to restore the array of strings
+      value = value.substring(1, value.length - 1).split('\x1E') if value
     else
       value = storage[@frameIndex]
     value
@@ -133,7 +134,8 @@ module.exports = class ThangState
             storage[frameIndex] = specialKey
           storage[frameIndex] = specialKey
         else if type is 'array'
-          value = value.join '\x1E'  # Record Separator
+          # We make sure the array keys won't collide with any string keys by using some unprintable characters.
+          value = '\x1D' + value.join('\x1E') + '\x1D'  # Group Separator, Record Separator(s), Group Separator
           specialKey = specialValuesToKeys[value]
           unless specialKey
             specialKey = specialKeysToValues.length

From 041f4512e8c02387d3aa0ec788a7f69da9293d18 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Thu, 6 Mar 2014 15:52:09 -0800
Subject: [PATCH 096/178] Set up marks to all load dynamically, and set up
 effect marks, which appear above a thang's head and rotate between them if
 there are multiple.

---
 app/lib/surface/CocoSprite.coffee | 42 +++++++++++++++++++++++++++++--
 app/lib/surface/Mark.coffee       | 35 +++++++++++++++++++++++---
 app/lib/surface/SpriteBoss.coffee | 10 +++-----
 app/models/Level.coffee           | 10 --------
 4 files changed, 74 insertions(+), 23 deletions(-)

diff --git a/app/lib/surface/CocoSprite.coffee b/app/lib/surface/CocoSprite.coffee
index 57146866a..279ef087b 100644
--- a/app/lib/surface/CocoSprite.coffee
+++ b/app/lib/surface/CocoSprite.coffee
@@ -82,6 +82,7 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
     @imageObject?.off 'animationend', @playNextAction
     @playNextAction = null
     @displayObject?.off()
+    clearInterval @effectInterval if @effectInterval
     super()
 
   toString: -> "<CocoSprite: #{@thang?.id}>"
@@ -375,18 +376,55 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
     scale *= @options.resolutionFactor if prop is 'registration'
     pos.x *= scale
     pos.y *= scale
+    if @thang
+      scaleFactor = @thang.scaleFactor ? 1
+      pos.x *= @thang.scaleFactorX ? scaleFactor
+      pos.y *= @thang.scaleFactorY ? scaleFactor
     pos
 
   updateMarks: ->
     return unless @options.camera
-    @addMark 'repair', null, @options.markThangTypes.repair if @thang?.errorsOut
+    @addMark 'repair', null, 'repair' if @thang?.errorsOut
     @marks.repair?.toggle @thang?.errorsOut
     @addMark('bounds').toggle true if @thang?.drawsBounds
     @addMark('shadow').toggle true unless @thangType.get('shadow') is 0
     mark.update() for name, mark of @marks
+#    @thang.effectNames = ['berserk', 'confused', 'controlled',  'cursed', 'fear', 'poison', 'paralyzed', 'regeneration', 'sleep', 'slowed', 'speed']
+    @updateEffectMarks() if @thang?.effectNames?.length
+    
+  updateEffectMarks: ->
+    return if _.isEqual @thang.effectNames, @previousEffectNames
+    for effect in @thang.effectNames
+      mark = @addMark effect, @options.floatingLayer, effect
+      mark.statusEffect = true
+      mark.toggle 'on'
+      mark.show()
+      
+    if @previousEffectNames
+      for effect in @previousEffectNames
+        mark = @marks[effect]
+        mark.toggle 'off'
+      
+    if @thang.effectNames.length > 1 and not @effectInterval
+      @rotateEffect()
+      @effectInterval = setInterval @rotateEffect, 1500
+      
+    else if @effectInterval and @thang.effectNames.length <= 1
+      @clearInterval @effectInterval
+      @effectInterval = null
+      
+    @previousEffectNames = @thang.effectNames
+
+  rotateEffect: =>
+    effects = (m.name for m in _.values(@marks) when m.on and m.statusEffect and m.mark)
+    effects.sort()
+    @effectIndex ?= 0
+    @effectIndex = (@effectIndex + 1) % effects.length
+    @marks[effect].hide() for effect in effects
+    @marks[effects[@effectIndex]].show()
 
   setHighlight: (to, delay) ->
-    @addMark 'highlight', @options.floatingLayer, @options.markThangTypes.highlight if to
+    @addMark 'highlight', @options.floatingLayer, 'highlight' if to
     @marks.highlight?.highlightDelay = delay
     @marks.highlight?.toggle to and not @dimmed
 
diff --git a/app/lib/surface/Mark.coffee b/app/lib/surface/Mark.coffee
index c012b537c..5a2e7689f 100644
--- a/app/lib/surface/Mark.coffee
+++ b/app/lib/surface/Mark.coffee
@@ -1,5 +1,7 @@
 CocoClass = require 'lib/CocoClass'
 Camera = require './Camera'
+ThangType = require 'models/ThangType'
+markThangTypes = {}
 
 module.exports = class Mark extends CocoClass
   subscriptions: {}
@@ -20,6 +22,7 @@ module.exports = class Mark extends CocoClass
   destroy: ->
     @mark?.parent?.removeChild @mark
     @markSprite?.destroy()
+    @thangType?.off 'sync', @onLoadedThangType, @
     @sprite = null
     super()
 
@@ -27,7 +30,9 @@ module.exports = class Mark extends CocoClass
 
   toggle: (to) ->
     return @ if to is @on
+    return @toggleTo = to unless @mark
     @on = to
+    delete @toggleTo
     if @on
       @layer.addChild @mark
       @layer.updateLayerOrder()
@@ -52,7 +57,7 @@ module.exports = class Mark extends CocoClass
       else if @name is 'debug' then @buildDebug()
       else if @thangType then @buildSprite()
       else console.error "Don't know how to build mark for", @name
-      @mark.mouseEnabled = false
+      @mark?.mouseEnabled = false
     @
 
   buildBounds: ->
@@ -126,15 +131,34 @@ module.exports = class Mark extends CocoClass
     @mark.graphics.endFill()
 
   buildSprite: ->
-    #console.log "building", @name, "with thangtype", @thangType
+    if _.isString @thangType
+      thangType = markThangTypes[@thangType]
+      return @loadThangType() if not thangType
+      @thangType = thangType
+
+    return @thangType.once 'sync', @onLoadedThangType, @ if not @thangType.loaded
     CocoSprite = require './CocoSprite'
     markSprite = new CocoSprite @thangType, @thangType.spriteOptions
     markSprite.queueAction 'idle'
     @mark = markSprite.displayObject
     @markSprite = markSprite
 
+  loadThangType: ->
+    name = @thangType
+    @thangType = new ThangType()
+    @thangType.url = -> "/db/thang.type/#{name}"
+    @thangType.once 'sync', @onLoadedThangType, @
+    @thangType.fetch()
+    markThangTypes[name] = @thangType
+    window.mtt = markThangTypes
+    
+  onLoadedThangType: ->
+    @build()
+    @toggle(@toggleTo) if @toggleTo?
+
   update: (pos=null) ->
-    return false unless @on
+    return false unless @on and @mark
+    @mark.alpha = if @hidden then 0 else 1
     @updatePosition pos
     @updateRotation()
     @updateScale()
@@ -156,10 +180,11 @@ module.exports = class Mark extends CocoClass
       pos ?= @sprite?.displayObject
     @mark.x = pos.x
     @mark.y = pos.y
-    if @name is 'highlight'
+    if @statusEffect or @name is 'highlight'
       offset = @sprite.getOffset 'aboveHead'
       @mark.x += offset.x
       @mark.y += offset.y
+      @mark.y -= 3 if @statusEffect
 
   updateRotation: ->
     if @name is 'debug' or (@name is 'shadow' and @sprite.thang?.shape in ["rectangle", "box"])
@@ -187,3 +212,5 @@ module.exports = class Mark extends CocoClass
 
   stop: -> @markSprite?.stop()
   play: -> @markSprite?.play()
+  hide: -> @hidden = true
+  show: -> @hidden = false
diff --git a/app/lib/surface/SpriteBoss.coffee b/app/lib/surface/SpriteBoss.coffee
index 130d017b7..f84d52a28 100644
--- a/app/lib/surface/SpriteBoss.coffee
+++ b/app/lib/surface/SpriteBoss.coffee
@@ -48,10 +48,6 @@ module.exports = class SpriteBoss extends CocoClass
   thangTypeFor: (type) ->
     _.find @options.thangTypes, (m) -> m.get('original') is type or m.get('name') is type
 
-  markThangTypes: ->
-    highlight: @thangTypeFor "Highlight"
-    repair: @thangTypeFor "Repair"
-
   createLayers: ->
     @spriteLayers = {}
     for [name, priority] in [
@@ -87,11 +83,11 @@ module.exports = class SpriteBoss extends CocoClass
     sprite
 
   createMarks: ->
-    @targetMark = new Mark name: 'target', camera: @camera, layer: @spriteLayers["Ground"], thangType: @thangTypeFor("Target")
-    @selectionMark = new Mark name: 'selection', camera: @camera, layer: @spriteLayers["Ground"], thangType: @thangTypeFor("Selection")
+    @targetMark = new Mark name: 'target', camera: @camera, layer: @spriteLayers["Ground"], thangType: 'target'
+    @selectionMark = new Mark name: 'selection', camera: @camera, layer: @spriteLayers["Ground"], thangType: 'selection'
 
   createSpriteOptions: (options) ->
-    _.extend options, camera: @camera, resolutionFactor: 4, groundLayer: @spriteLayers["Ground"], textLayer: @surfaceTextLayer, floatingLayer: @spriteLayers["Floating"], markThangTypes: @markThangTypes(), spriteSheetCache: @spriteSheetCache, showInvisible: @options.showInvisible
+    _.extend options, camera: @camera, resolutionFactor: 4, groundLayer: @spriteLayers["Ground"], textLayer: @surfaceTextLayer, floatingLayer: @spriteLayers["Floating"], spriteSheetCache: @spriteSheetCache, showInvisible: @options.showInvisible
 
   createIndieSprites: (indieSprites, withWizards) ->
     unless @indieSprites
diff --git a/app/models/Level.coffee b/app/models/Level.coffee
index fca1f8249..176b09aa6 100644
--- a/app/models/Level.coffee
+++ b/app/models/Level.coffee
@@ -115,15 +115,5 @@ module.exports = class Level extends CocoModel
         model = CocoModel.getOrMakeModelFromLink link, shouldLoadProjection
         models.push model if model
     else if path is '/'
-      # We also we need to make sure we grab the Wizard ThangType and the Marks. Hackitrooooid!
-      for [type, original] in [
-        ["Highlight", "529f8fdbdacd325127000003"]
-        ["Selection", "52aa5f7520fccb0000000002"]
-        ["Target", "52b32ad97385ec3d03000001"]
-        ["Repair", "52bcc4591f766a891c000003"]
-      ]
-        link = "/db/thang_type/#{original}/version"
-        model = CocoModel.getOrMakeModelFromLink link, shouldLoadProjection
-        models.push model if model
       models.push ThangType.loadUniversalWizard()
     models

From ea406bd3c7e2d9adf19e9b2a80aa91ec2fa7d849 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Thu, 6 Mar 2014 15:52:51 -0800
Subject: [PATCH 097/178] Fixed some event listeners on destroyed loaders.

---
 app/lib/God.coffee         | 2 +-
 app/lib/LevelLoader.coffee | 2 ++
 2 files changed, 3 insertions(+), 1 deletion(-)

diff --git a/app/lib/God.coffee b/app/lib/God.coffee
index f7563517e..3839013a6 100644
--- a/app/lib/God.coffee
+++ b/app/lib/God.coffee
@@ -30,7 +30,7 @@ module.exports = class God
     @createWorld()
 
   fillWorkerPool: =>
-    return unless Worker
+    return unless Worker and not @dead
     @workerPool ?= []
     if @workerPool.length < @maxWorkerPoolSize
       @workerPool.push @createWorker()
diff --git a/app/lib/LevelLoader.coffee b/app/lib/LevelLoader.coffee
index 52865a8b1..18ae9f82f 100644
--- a/app/lib/LevelLoader.coffee
+++ b/app/lib/LevelLoader.coffee
@@ -70,6 +70,7 @@ module.exports = class LevelLoader extends CocoClass
     @session.loaded and ((not @opponentSession) or @opponentSession.loaded)
 
   onSessionLoaded: ->
+    return if @destroyed
     # TODO: maybe have all non versioned models do this? Or make it work to PUT/PATCH to relative urls
     if @session.loaded
       @session.url = -> '/db/level.session/' + @id
@@ -171,6 +172,7 @@ module.exports = class LevelLoader extends CocoClass
     t0 = new Date()
     @spriteSheetsToBuild += 1
     thangType.once 'build-complete', =>
+      return if @destroyed
       @spriteSheetsBuilt += 1
       @notifyProgress()
       console.log "Built", thangType.get('name'), 'after', ((new Date()) - t0), 'ms'

From 1b61b928b0f4945173d41ec17dff01f13647ebbf Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Thu, 6 Mar 2014 16:21:06 -0800
Subject: [PATCH 098/178] Couple tweaks to the cocosprite for effect marks.

---
 app/lib/surface/CocoSprite.coffee | 4 +++-
 1 file changed, 3 insertions(+), 1 deletion(-)

diff --git a/app/lib/surface/CocoSprite.coffee b/app/lib/surface/CocoSprite.coffee
index 279ef087b..998fdccda 100644
--- a/app/lib/surface/CocoSprite.coffee
+++ b/app/lib/surface/CocoSprite.coffee
@@ -389,10 +389,11 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
     @addMark('bounds').toggle true if @thang?.drawsBounds
     @addMark('shadow').toggle true unless @thangType.get('shadow') is 0
     mark.update() for name, mark of @marks
-#    @thang.effectNames = ['berserk', 'confused', 'controlled',  'cursed', 'fear', 'poison', 'paralyzed', 'regeneration', 'sleep', 'slowed', 'speed']
+#    @thang.effectNames = ['berserk', 'confuse', 'control', 'curse', 'fear', 'poison', 'paralyze', 'regen', 'sleep', 'slow', 'speed']
     @updateEffectMarks() if @thang?.effectNames?.length
     
   updateEffectMarks: ->
+    @thang.effectNames = (e for e in @thang.effectNames when e) # hack because empty strings appear in the array
     return if _.isEqual @thang.effectNames, @previousEffectNames
     for effect in @thang.effectNames
       mark = @addMark effect, @options.floatingLayer, effect
@@ -449,6 +450,7 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
     @labels[name]
 
   addMark: (name, layer, thangType=null) ->
+    console.log 'what are my effects?', @thang.effectNames unless name
     @marks[name] ?= new Mark name: name, sprite: @, camera: @options.camera, layer: layer ? @options.groundLayer, thangType: thangType
     @marks[name]
 

From b6335a30ae89fc70c8508c1db7c6908a42358635 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Thu, 6 Mar 2014 16:32:13 -0800
Subject: [PATCH 099/178] More fixes for empty string array serialization.

---
 app/lib/surface/CocoSprite.coffee | 14 ++++++--------
 app/lib/world/thang_state.coffee  | 14 ++++++++++----
 2 files changed, 16 insertions(+), 12 deletions(-)

diff --git a/app/lib/surface/CocoSprite.coffee b/app/lib/surface/CocoSprite.coffee
index 998fdccda..daac71ec5 100644
--- a/app/lib/surface/CocoSprite.coffee
+++ b/app/lib/surface/CocoSprite.coffee
@@ -389,31 +389,30 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
     @addMark('bounds').toggle true if @thang?.drawsBounds
     @addMark('shadow').toggle true unless @thangType.get('shadow') is 0
     mark.update() for name, mark of @marks
-#    @thang.effectNames = ['berserk', 'confuse', 'control', 'curse', 'fear', 'poison', 'paralyze', 'regen', 'sleep', 'slow', 'speed']
+    #@thang.effectNames = ['berserk', 'confuse', 'control', 'curse', 'fear', 'poison', 'paralyze', 'regen', 'sleep', 'slow', 'speed']
     @updateEffectMarks() if @thang?.effectNames?.length
-    
+
   updateEffectMarks: ->
-    @thang.effectNames = (e for e in @thang.effectNames when e) # hack because empty strings appear in the array
     return if _.isEqual @thang.effectNames, @previousEffectNames
     for effect in @thang.effectNames
       mark = @addMark effect, @options.floatingLayer, effect
       mark.statusEffect = true
       mark.toggle 'on'
       mark.show()
-      
+
     if @previousEffectNames
       for effect in @previousEffectNames
         mark = @marks[effect]
         mark.toggle 'off'
-      
+
     if @thang.effectNames.length > 1 and not @effectInterval
       @rotateEffect()
       @effectInterval = setInterval @rotateEffect, 1500
-      
+
     else if @effectInterval and @thang.effectNames.length <= 1
       @clearInterval @effectInterval
       @effectInterval = null
-      
+
     @previousEffectNames = @thang.effectNames
 
   rotateEffect: =>
@@ -450,7 +449,6 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
     @labels[name]
 
   addMark: (name, layer, thangType=null) ->
-    console.log 'what are my effects?', @thang.effectNames unless name
     @marks[name] ?= new Mark name: name, sprite: @, camera: @options.camera, layer: layer ? @options.groundLayer, thangType: thangType
     @marks[name]
 
diff --git a/app/lib/world/thang_state.coffee b/app/lib/world/thang_state.coffee
index 7db97aca6..7ea6a9687 100644
--- a/app/lib/world/thang_state.coffee
+++ b/app/lib/world/thang_state.coffee
@@ -50,9 +50,12 @@ module.exports = class ThangState
       value = @thang.world.getThangByID @specialKeysToValues[specialKey]
     else if type is 'array'
       specialKey = storage[@frameIndex]
-      value = @specialKeysToValues[specialKey]
-      # Remove leading and trailing Group Separators and split by any Record Separators to restore the array of strings
-      value = value.substring(1, value.length - 1).split('\x1E') if value
+      valueString = @specialKeysToValues[specialKey]
+      if valueString and valueString.length > 1
+        # Trim leading Group Separator and trailing Record Separator, split by Record Separators, restore string array.
+        value = valueString.substring(1, valueString.length - 1).split '\x1E'
+      else
+        value = []
     else
       value = storage[@frameIndex]
     value
@@ -135,7 +138,10 @@ module.exports = class ThangState
           storage[frameIndex] = specialKey
         else if type is 'array'
           # We make sure the array keys won't collide with any string keys by using some unprintable characters.
-          value = '\x1D' + value.join('\x1E') + '\x1D'  # Group Separator, Record Separator(s), Group Separator
+          stringPieces = ['\x1D']  # Group Separator
+          for element in value
+            stringPieces.push element, '\x1E'  # Record Separator(s)
+          value = stringPieces.join('')
           specialKey = specialValuesToKeys[value]
           unless specialKey
             specialKey = specialKeysToValues.length

From 5b11e132b2e1ec57c44555f1f0511ab3318cef50 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Thu, 6 Mar 2014 17:13:16 -0800
Subject: [PATCH 100/178] Fixed the effect marks so they properly go away.

---
 app/lib/surface/CocoSprite.coffee | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/lib/surface/CocoSprite.coffee b/app/lib/surface/CocoSprite.coffee
index daac71ec5..5c082320b 100644
--- a/app/lib/surface/CocoSprite.coffee
+++ b/app/lib/surface/CocoSprite.coffee
@@ -390,7 +390,7 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
     @addMark('shadow').toggle true unless @thangType.get('shadow') is 0
     mark.update() for name, mark of @marks
     #@thang.effectNames = ['berserk', 'confuse', 'control', 'curse', 'fear', 'poison', 'paralyze', 'regen', 'sleep', 'slow', 'speed']
-    @updateEffectMarks() if @thang?.effectNames?.length
+    @updateEffectMarks() if @thang?.effectNames?.length or @previousEffectNames?.length
 
   updateEffectMarks: ->
     return if _.isEqual @thang.effectNames, @previousEffectNames
@@ -403,7 +403,7 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
     if @previousEffectNames
       for effect in @previousEffectNames
         mark = @marks[effect]
-        mark.toggle 'off'
+        mark.toggle false
 
     if @thang.effectNames.length > 1 and not @effectInterval
       @rotateEffect()

From 40e896868be316ee8e607db5c9a370c6a85c36c6 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Thu, 6 Mar 2014 17:21:24 -0800
Subject: [PATCH 101/178] Fixed CocoSprite getOffset to not apply scale to
 registration points.

---
 app/lib/surface/CocoSprite.coffee | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/lib/surface/CocoSprite.coffee b/app/lib/surface/CocoSprite.coffee
index 5c082320b..5173c5141 100644
--- a/app/lib/surface/CocoSprite.coffee
+++ b/app/lib/surface/CocoSprite.coffee
@@ -376,7 +376,7 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
     scale *= @options.resolutionFactor if prop is 'registration'
     pos.x *= scale
     pos.y *= scale
-    if @thang
+    if @thang and prop isnt 'registration'
       scaleFactor = @thang.scaleFactor ? 1
       pos.x *= @thang.scaleFactorX ? scaleFactor
       pos.y *= @thang.scaleFactorY ? scaleFactor

From a4edb76c982bf7eeecb3b03616132feefe6a7f72 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Thu, 6 Mar 2014 17:25:40 -0800
Subject: [PATCH 102/178] Fixed a couple of the bugs with marks.

---
 app/lib/surface/CocoSprite.coffee | 3 ++-
 1 file changed, 2 insertions(+), 1 deletion(-)

diff --git a/app/lib/surface/CocoSprite.coffee b/app/lib/surface/CocoSprite.coffee
index 5c082320b..545addaa6 100644
--- a/app/lib/surface/CocoSprite.coffee
+++ b/app/lib/surface/CocoSprite.coffee
@@ -410,13 +410,14 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
       @effectInterval = setInterval @rotateEffect, 1500
 
     else if @effectInterval and @thang.effectNames.length <= 1
-      @clearInterval @effectInterval
+      clearInterval @effectInterval
       @effectInterval = null
 
     @previousEffectNames = @thang.effectNames
 
   rotateEffect: =>
     effects = (m.name for m in _.values(@marks) when m.on and m.statusEffect and m.mark)
+    return unless effects.length
     effects.sort()
     @effectIndex ?= 0
     @effectIndex = (@effectIndex + 1) % effects.length

From 45cd7e31cae2fee22c0870b4b622545f3f0c0d92 Mon Sep 17 00:00:00 2001
From: Michael Schmatz <schmatz@umich.edu>
Date: Thu, 6 Mar 2014 18:48:41 -0800
Subject: [PATCH 103/178] Made simulator level-independent

---
 app/views/play/ladder/my_matches_tab.coffee |  5 +++--
 server/queues/scoring.coffee                | 24 +++++++++++++--------
 2 files changed, 18 insertions(+), 11 deletions(-)

diff --git a/app/views/play/ladder/my_matches_tab.coffee b/app/views/play/ladder/my_matches_tab.coffee
index 00aec7c8c..6b2cf71be 100644
--- a/app/views/play/ladder/my_matches_tab.coffee
+++ b/app/views/play/ladder/my_matches_tab.coffee
@@ -94,10 +94,11 @@ module.exports = class MyMatchesTabView extends CocoView
     @setRankingButtonText(button, 'ranking')
     success = => @setRankingButtonText(button, 'ranked')
     failure = => @setRankingButtonText(button, 'failed')
-
+    
+    ajaxData = { session: sessionID, levelID: @level.attributes.original, levelMajorVersion: @level.attributes.version.major }
     $.ajax '/queue/scoring', {
       type: 'POST'
-      data: { session: sessionID }
+      data: ajaxData
       success: success
       failure: failure
     }
diff --git a/server/queues/scoring.coffee b/server/queues/scoring.coffee
index c527f921f..dd2a51030 100644
--- a/server/queues/scoring.coffee
+++ b/server/queues/scoring.coffee
@@ -48,6 +48,9 @@ addPairwiseTaskToQueue = (taskPair, cb) ->
 
 module.exports.createNewTask = (req, res) ->
   requestSessionID = req.body.session
+  requestLevelID = req.body.levelID
+  requestLevelMajorVersion = parseInt(req.body.levelMajorVersion)
+  
   validatePermissions req, requestSessionID, (error, permissionsAreValid) ->
     if err? then return errors.serverError res, "There was an error validating permissions"
     unless permissionsAreValid then return errors.forbidden res, "You do not have the permissions to submit that game to the leaderboard"
@@ -60,7 +63,7 @@ module.exports.createNewTask = (req, res) ->
       updateSessionToSubmit sessionToSubmit, (err, data) ->
         if err? then return errors.serverError res, "There was an error updating the session"
         opposingTeam = calculateOpposingTeam(sessionToSubmit.team)
-        fetchInitialSessionsToRankAgainst opposingTeam, (err, sessionsToRankAgainst) ->
+        fetchInitialSessionsToRankAgainst opposingTeam,requestLevelID, requestLevelMajorVersion, (err, sessionsToRankAgainst) ->
           if err? then return errors.serverError res, "There was an error fetching the sessions to rank against"
 
           taskPairs = generateTaskPairs(sessionsToRankAgainst, sessionToSubmit)
@@ -139,7 +142,10 @@ module.exports.processTaskResult = (req, res) ->
                   opponentID = _.pull(_.keys(newScoresObject), originalSessionID)
                   sessionNewScore = newScoresObject[originalSessionID].totalScore
                   opponentNewScore = newScoresObject[opponentID].totalScore
-                  findNearestBetterSessionID originalSessionID, sessionNewScore, opponentNewScore, opponentID ,opposingTeam, (err, opponentSessionID) ->
+                  
+                  levelOriginalID = levelSession.level.original
+                  levelOriginalMajorVersion = levelSession.level.majorVersion
+                  findNearestBetterSessionID levelOriginalID, levelOriginalMajorVersion, originalSessionID, sessionNewScore, opponentNewScore, opponentID ,opposingTeam, (err, opponentSessionID) ->
                     if err? then return errors.serverError res, "There was an error finding the nearest sessionID!"
                     unless opponentSessionID then return sendResponseObject req, res, {"message":"There were no more games to rank(game is at top!"}
   
@@ -181,7 +187,7 @@ determineIfSessionShouldContinueAndUpdateLog = (sessionID, sessionRank, cb) ->
         cb null, true
 
 
-findNearestBetterSessionID = (sessionID, sessionTotalScore, opponentSessionTotalScore, opponentSessionID, opposingTeam, cb) ->
+findNearestBetterSessionID = (levelOriginalID, levelMajorVersion, sessionID, sessionTotalScore, opponentSessionTotalScore, opponentSessionID, opposingTeam, cb) ->
   retrieveAllOpponentSessionIDs sessionID, (err, opponentSessionIDs) ->
     if err? then return cb err, null
 
@@ -190,8 +196,8 @@ findNearestBetterSessionID = (sessionID, sessionTotalScore, opponentSessionTotal
         $gt:opponentSessionTotalScore
       _id:
         $nin: opponentSessionIDs
-      "level.original": "52d97ecd32362bc86e004e87"
-      "level.majorVersion": 0
+      "level.original": levelOriginalID
+      "level.majorVersion": levelMajorVersion
       submitted: true
       submittedCode:
         $exists: true
@@ -298,16 +304,16 @@ updateSessionToSubmit = (sessionToUpdate, callback) ->
     numberOfLosses: 0
   LevelSession.update {_id: sessionToUpdate._id}, sessionUpdateObject, callback
 
-fetchInitialSessionsToRankAgainst = (opposingTeam, callback) ->
+fetchInitialSessionsToRankAgainst = (opposingTeam, levelID, levelMajorVersion, callback) ->
   console.log "Fetching sessions to rank against for opposing team #{opposingTeam}"
   findParameters =
-    "level.original": "52d97ecd32362bc86e004e87"
-    "level.majorVersion": 0
+    "level.original": levelID
+    "level.majorVersion": levelMajorVersion
     submitted: true
     submittedCode:
       $exists: true
     team: opposingTeam
-
+  
   sortParameters =
     totalScore: 1
 

From 6e47416d242d7d050a0f88136a4c4862cd92fb91 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Thu, 6 Mar 2014 19:09:15 -0800
Subject: [PATCH 104/178] Fixed some level IDs.

---
 app/views/editor/components/main.coffee           | 2 ++
 app/views/play/level/control_bar_view.coffee      | 2 +-
 app/views/play/level/tome/cast_button_view.coffee | 2 +-
 app/views/play/level/tome/tome_view.coffee        | 1 +
 4 files changed, 5 insertions(+), 2 deletions(-)

diff --git a/app/views/editor/components/main.coffee b/app/views/editor/components/main.coffee
index 87a4ed3b6..fdcc55a6e 100644
--- a/app/views/editor/components/main.coffee
+++ b/app/views/editor/components/main.coffee
@@ -18,6 +18,7 @@ module.exports = class ThangComponentEditView extends CocoView
     @callback = options.callback
 
   render: =>
+    return if @destroyed
     for model in [Level, LevelComponent]
       (new model()).on 'schema-loaded', @render unless model.schema?.loaded
     if not @componentCollection
@@ -35,6 +36,7 @@ module.exports = class ThangComponentEditView extends CocoView
     @buildAddComponentTreema()
 
   onComponentsSync: =>
+    return if @destroyed
     @supermodel.addCollection @componentCollection
     @render()
 
diff --git a/app/views/play/level/control_bar_view.coffee b/app/views/play/level/control_bar_view.coffee
index 392eb4ebf..06d689f82 100644
--- a/app/views/play/level/control_bar_view.coffee
+++ b/app/views/play/level/control_bar_view.coffee
@@ -54,7 +54,7 @@ module.exports = class ControlBarView extends View
     c.ladderGame = @ladderGame
     c.homeLink = "/"
     levelID = @level.get('slug')
-    if levelID in ["brawlwood", "brawlwood-tutorial"]
+    if levelID in ["brawlwood", "brawlwood-tutorial", "dungeon-arena"]
       levelID = 'brawlwood' if levelID is 'brawlwood-tutorial'
       c.homeLink = "/play/ladder/" + levelID
     c
diff --git a/app/views/play/level/tome/cast_button_view.coffee b/app/views/play/level/tome/cast_button_view.coffee
index 98c8ab3c8..15d59f194 100644
--- a/app/views/play/level/tome/cast_button_view.coffee
+++ b/app/views/play/level/tome/cast_button_view.coffee
@@ -35,7 +35,7 @@ module.exports = class CastButtonView extends View
     # TODO: use a User setting instead of localStorage
     delay = localStorage.getItem 'autocastDelay'
     delay ?= 5000
-    if @levelID in ['brawlwood', 'brawlwood-tutorial']
+    if @levelID in ['brawlwood', 'brawlwood-tutorial', 'dungeon-arena']
       delay = 90019001
     @setAutocastDelay delay
 
diff --git a/app/views/play/level/tome/tome_view.coffee b/app/views/play/level/tome/tome_view.coffee
index f7be355f6..0d18a9f1d 100644
--- a/app/views/play/level/tome/tome_view.coffee
+++ b/app/views/play/level/tome/tome_view.coffee
@@ -130,6 +130,7 @@ module.exports = class TomeView extends View
         unless method.cloneOf
           skipProtectAPI = @getQueryVariable("skip_protect_api") is "true" or @options.levelID isnt 'brawlwood'
           skipProtectAPI = true  # gah, it's so slow :( and somehow still affects simulation
+          #skipProtectAPI = false if @options.levelID is 'dungeon-arena'
           skipFlow = @getQueryVariable("skip_flow") is "true" or @options.levelID is 'brawlwood'
           spell = @spells[spellKey] = new Spell programmableMethod: method, spellKey: spellKey, pathComponents: pathPrefixComponents.concat(pathComponents), session: @options.session, supermodel: @supermodel, skipFlow: skipFlow, skipProtectAPI: skipProtectAPI, worker: @worker
     for thangID, spellKeys of @thangSpells

From d14f8623595a93c036fc4b4231b56016e84a2ec3 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Fri, 7 Mar 2014 11:23:06 -0800
Subject: [PATCH 105/178] Ladder view can now be linked to to play against a
 specific team.

---
 app/views/play/ladder_view.coffee | 6 ++++--
 1 file changed, 4 insertions(+), 2 deletions(-)

diff --git a/app/views/play/ladder_view.coffee b/app/views/play/ladder_view.coffee
index 8f6a046de..723a7910d 100644
--- a/app/views/play/ladder_view.coffee
+++ b/app/views/play/ladder_view.coffee
@@ -66,6 +66,7 @@ module.exports = class LadderView extends RootView
     @insertSubView(@ladderTab = new LadderTabView({}, @level, @sessions))
     @insertSubView(@myMatchesTab = new MyMatchesTabView({}, @level, @sessions))
     setInterval(@fetchSessionsAndRefreshViews.bind(@), 10000)
+    @showPlayModal(document.location.hash[1..]) if document.location.hash and @sessions.loaded
 
   fetchSessionsAndRefreshViews: ->
     @sessions.fetch({"success": @refreshViews})
@@ -112,8 +113,9 @@ module.exports = class LadderView extends RootView
     $("#simulation-status-text").text @simulationStatus
 
   onClickPlayButton: (e) ->
-    button = $(e.target).closest('.play-button')
-    teamID = button.data('team')
+    @showPlayModal($(e.target).closest('.play-button').data('team'))
+    
+  showPlayModal: (teamID) ->
     session = (s for s in @sessions.models when s.get('team') is teamID)[0]
     modal = new LadderPlayModal({}, @level, session, teamID)
     @openModalView modal

From 5274f22467293101147095bed8d512f18d6dab50 Mon Sep 17 00:00:00 2001
From: Shrihari <gfxindia@gmail.com>
Date: Sat, 8 Mar 2014 01:10:12 +0530
Subject: [PATCH 106/178] Fixed Spell Debug View width bug

---
 app/styles/play/level/tome/spell_debug.sass       | 1 +
 app/views/play/level/tome/spell_debug_view.coffee | 4 +++-
 2 files changed, 4 insertions(+), 1 deletion(-)

diff --git a/app/styles/play/level/tome/spell_debug.sass b/app/styles/play/level/tome/spell_debug.sass
index d0f97a28f..e02326b55 100644
--- a/app/styles/play/level/tome/spell_debug.sass
+++ b/app/styles/play/level/tome/spell_debug.sass
@@ -3,6 +3,7 @@
 .spell-debug-view
   position: absolute
   z-index: 9001
+  min-width: 250px
   max-width: 400px
   padding: 10px
   background: transparent url(/images/level/popover_background.png)
diff --git a/app/views/play/level/tome/spell_debug_view.coffee b/app/views/play/level/tome/spell_debug_view.coffee
index 5f4cfdd30..c35314932 100644
--- a/app/views/play/level/tome/spell_debug_view.coffee
+++ b/app/views/play/level/tome/spell_debug_view.coffee
@@ -61,7 +61,9 @@ module.exports = class DebugView extends View
       @variableChain = chain
       offsetX = e.domEvent.offsetX ? e.clientX - $(e.domEvent.target).offset().left
       offsetY = e.domEvent.offsetY ? e.clientY - $(e.domEvent.target).offset().top
-      @pos = {left: offsetX + 50, top: offsetY + 50}
+      w = $(document).width()
+      offsetX = w - $(e.domEvent.target).offset().left - 300 if e.clientX + 300 > w
+      @pos = {left: offsetX + 50, top: offsetY + 20}
       @markerRange = new Range pos.row, start, pos.row, end
     else
       @variableChain = @markerRange = null

From 610fc0694bbf6b2750d64e10d7284b52dde8c111 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Fri, 7 Mar 2014 11:57:15 -0800
Subject: [PATCH 107/178] Turning off view caching by default now. Should help
 lower the memory footprint.

---
 app/lib/Router.coffee           | 2 +-
 app/views/kinds/CocoView.coffee | 2 +-
 2 files changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/lib/Router.coffee b/app/lib/Router.coffee
index 435fdb7d4..db70b8c74 100644
--- a/app/lib/Router.coffee
+++ b/app/lib/Router.coffee
@@ -90,7 +90,7 @@ module.exports = class CocoRouter extends Backbone.Router
       @cache[route].fromCache = true
       return @cache[route]
     view = @getView(route)
-    @cache[route] = view unless view and view.cache is false
+    @cache[route] = view if view?.cache
     return view
     
   routeDirectly: (path, args) ->
diff --git a/app/views/kinds/CocoView.coffee b/app/views/kinds/CocoView.coffee
index 2811318c6..dbdb14146 100644
--- a/app/views/kinds/CocoView.coffee
+++ b/app/views/kinds/CocoView.coffee
@@ -10,7 +10,7 @@ makeScopeName = -> "view-scope-#{classCount++}"
 
 module.exports = class CocoView extends Backbone.View
   startsLoading: false
-  cache: true # signals to the router to keep this view around
+  cache: false # signals to the router to keep this view around
   template: -> ''
 
   events:

From cf2a0a999e63e8e45ea00d03d454d9440ab4e7da Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Fri, 7 Mar 2014 11:57:31 -0800
Subject: [PATCH 108/178] Set up the ladder play modal to point to the tutorial
 level dynamically.

---
 app/templates/play/ladder/play_modal.jade | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/templates/play/ladder/play_modal.jade b/app/templates/play/ladder/play_modal.jade
index 6c1cd22d6..5e7953896 100644
--- a/app/templates/play/ladder/play_modal.jade
+++ b/app/templates/play/ladder/play_modal.jade
@@ -8,7 +8,7 @@ block modal-body-content
   p.tutorial-suggestion
     span Not sure what's going on?
     |  
-    a(href="/play/level/brawlwood-tutorial") Play the tutorial first.
+    a(href="/play/level/#{levelID}-tutorial") Play the tutorial first.
 
   a(href="/play/level/#{levelID}?team=#{teamID}")
     div.play-option

From 4e24e2a2442ef04a3c8b9ddfbf844727b93cf874 Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Fri, 7 Mar 2014 12:10:25 -0800
Subject: [PATCH 109/178] Update ru.coffee

---
 app/locale/ru.coffee | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/locale/ru.coffee b/app/locale/ru.coffee
index 51f6d1a7e..dcc68a5a2 100644
--- a/app/locale/ru.coffee
+++ b/app/locale/ru.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     tome_available_spells: "Доступные заклинания"
     hud_continue: "Продолжить (нажмите Shift+Пробел)"
     spell_saved: "Заклинание сохранено"
-#    skip_tutorial: "Skip (esc)"
+    skip_tutorial: "Пропуск (Esc)"y
 
   admin:
     av_title: "Админ панель"
@@ -298,7 +298,7 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     practices_title: "Лучшие уважаемые практики"
     practices_description: "Это наши обещания тебе, игрок, менее юридическим языком."
     privacy_title: "Конфиденциальность"
-    privacy_description: "Мы не будем продавать какой-либо личной информации. Мы намерены заработать деньги с помощью рекрутинга в конечном счете, но будьте уверены, мы не будем распространять вашу личную информацию заинтересованным компаниям без вашего явного согласия."
+    privacy_description: "Мы не будем продавать какой-либо личной информации. Мы намерены заработать деньги с помощью рекрутинга в конечном счёте, но будьте уверены, мы не будем распространять вашу личную информацию заинтересованным компаниям без вашего явного согласия."
     security_title: "Безопасность"
     security_description: "Мы стремимся сохранить вашу личную информацию в безопасности. Как проект с открытым исходным кодом, наш сайт в свободном доступе для всех для пересмотра и совершенствования систем безопасности."
     email_title: "Email"

From 517ef8611bc4eabb7d2e833534cb5286c785d2a1 Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Fri, 7 Mar 2014 12:13:15 -0800
Subject: [PATCH 110/178] Update ru.coffee

---
 app/locale/ru.coffee | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/locale/ru.coffee b/app/locale/ru.coffee
index dcc68a5a2..0ee54cb73 100644
--- a/app/locale/ru.coffee
+++ b/app/locale/ru.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     tome_available_spells: "Доступные заклинания"
     hud_continue: "Продолжить (нажмите Shift+Пробел)"
     spell_saved: "Заклинание сохранено"
-    skip_tutorial: "Пропуск (Esc)"y
+    skip_tutorial: "Пропуск (Esc)"
 
   admin:
     av_title: "Админ панель"

From d8d917ecf9e52b742ee66ed2576ec8353fb0c6a0 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Fri, 7 Mar 2014 12:15:16 -0800
Subject: [PATCH 111/178] Set up better teardown for the ladder view.

---
 app/lib/CocoClass.coffee           | 1 +
 app/lib/simulator/Simulator.coffee | 7 ++++++-
 app/views/kinds/CocoView.coffee    | 1 +
 app/views/play/ladder_view.coffee  | 7 ++++++-
 4 files changed, 14 insertions(+), 2 deletions(-)

diff --git a/app/lib/CocoClass.coffee b/app/lib/CocoClass.coffee
index 22dc25f64..67a8e4cac 100644
--- a/app/lib/CocoClass.coffee
+++ b/app/lib/CocoClass.coffee
@@ -20,6 +20,7 @@ module.exports = class CocoClass
   destroy: ->
     # teardown subscriptions, prevent new ones
     @stopListening?()
+    @off()
     @unsubscribeAll()
     @stopListeningToShortcuts()
     @[key] = undefined for key of @
diff --git a/app/lib/simulator/Simulator.coffee b/app/lib/simulator/Simulator.coffee
index f92bccb98..f41c0acb3 100644
--- a/app/lib/simulator/Simulator.coffee
+++ b/app/lib/simulator/Simulator.coffee
@@ -10,6 +10,11 @@ module.exports = class Simulator
     @trigger 'statusUpdate', 'Starting simulation!'
     @retryDelayInSeconds = 10
     @taskURL = '/queue/scoring'
+    
+  destroy: ->
+    @off()
+    @cleanupSimulation()
+    # TODO: More teardown?
 
   fetchAndSimulateTask: =>
     @trigger 'statusUpdate', 'Fetching simulation data!'
@@ -99,7 +104,7 @@ module.exports = class Simulator
     @fetchAndSimulateTask()
 
   cleanupSimulation: ->
-    @god.destroy()
+    @god?.destroy()
     @god = null
     @world = null
     @level = null
diff --git a/app/views/kinds/CocoView.coffee b/app/views/kinds/CocoView.coffee
index dbdb14146..806a2d25c 100644
--- a/app/views/kinds/CocoView.coffee
+++ b/app/views/kinds/CocoView.coffee
@@ -37,6 +37,7 @@ module.exports = class CocoView extends Backbone.View
 
   destroy: ->
     @stopListening()
+    @off()
     @stopListeningToShortcuts()
     @undelegateEvents() # removes both events and subs
     view.destroy() for id, view of @subviews
diff --git a/app/views/play/ladder_view.coffee b/app/views/play/ladder_view.coffee
index 723a7910d..81e2f6558 100644
--- a/app/views/play/ladder_view.coffee
+++ b/app/views/play/ladder_view.coffee
@@ -65,7 +65,7 @@ module.exports = class LadderView extends RootView
     return if @startsLoading
     @insertSubView(@ladderTab = new LadderTabView({}, @level, @sessions))
     @insertSubView(@myMatchesTab = new MyMatchesTabView({}, @level, @sessions))
-    setInterval(@fetchSessionsAndRefreshViews.bind(@), 10000)
+    @refreshInterval = setInterval(@fetchSessionsAndRefreshViews.bind(@), 10000)
     @showPlayModal(document.location.hash[1..]) if document.location.hash and @sessions.loaded
 
   fetchSessionsAndRefreshViews: ->
@@ -119,3 +119,8 @@ module.exports = class LadderView extends RootView
     session = (s for s in @sessions.models when s.get('team') is teamID)[0]
     modal = new LadderPlayModal({}, @level, session, teamID)
     @openModalView modal
+    
+  destroy: ->
+    clearInterval @refreshInterval
+    @simulator.destroy()
+    super()

From 317ae5cc329a1318874468eddebd2d99f6debd23 Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Fri, 7 Mar 2014 12:17:32 -0800
Subject: [PATCH 112/178] Update scribe.jade

---
 app/templates/contribute/scribe.jade | 6 +++---
 1 file changed, 3 insertions(+), 3 deletions(-)

diff --git a/app/templates/contribute/scribe.jade b/app/templates/contribute/scribe.jade
index 9bd404ad2..eafab49b6 100644
--- a/app/templates/contribute/scribe.jade
+++ b/app/templates/contribute/scribe.jade
@@ -16,7 +16,7 @@ block content
         span  
         span(data-i18n="classes.scribe_title_description") (Article Editor)
       p
-        span(data-i18n="account_settings.scribe_introduction_pref")
+        span(data-i18n="contribute.scribe_introduction_pref")
           | CodeCombat isn't just going to be a bunch of levels.
           | It will also include a resource for knowledge, a wiki of programming concepts that levels can hook into.
           | That way rather than each Artisan having to describe in detail what a comparison operator is, they
@@ -24,7 +24,7 @@ block content
           | Something along the lines of what the 
         a(href="https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide", data-i18n="contribute.scribe_introduction_url_mozilla")
           | Mozilla Developer Network 
-        span(data-i18n="account_settings.scribe_introduction_suf")  
+        span(data-i18n="contribute.scribe_introduction_suf")  
           | has built. If your idea of fun is articulating the concepts of programming in Markdown form,
           | then this class might be for you.
   
@@ -77,4 +77,4 @@ block content
           li mattinsler
           
 
-    div.clearfix
\ No newline at end of file
+    div.clearfix

From b002ae376b06f9e91cf59a5f7c912b73e3640a20 Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Fri, 7 Mar 2014 12:19:14 -0800
Subject: [PATCH 113/178] Update counselor.jade

---
 app/templates/contribute/counselor.jade | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/templates/contribute/counselor.jade b/app/templates/contribute/counselor.jade
index 821a86694..36a79d8cc 100644
--- a/app/templates/contribute/counselor.jade
+++ b/app/templates/contribute/counselor.jade
@@ -38,7 +38,7 @@ block content
   
       h4(data-i18n="contribute.how_to_join") How to Join 
       p 
-        a(title='Contact', tabindex=-1, data-toggle="coco-modal", data-target="modal/contact", data-i18n="contact_us_url")
+        a(title='Contact', tabindex=-1, data-toggle="coco-modal", data-target="modal/contact", data-i18n="contribute.contact_us_url")
           | Contact us
         span , 
         span(data-i18n="contribute.counselor_join_desc")
@@ -46,4 +46,4 @@ block content
           | be interested in doing. We'll put you in our contact list and be in touch
           | when we could use advice (not too often).
 
-    div.clearfix
\ No newline at end of file
+    div.clearfix

From 535881b6373f02e8f22dfa083a8da7196de50a25 Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Fri, 7 Mar 2014 12:20:17 -0800
Subject: [PATCH 114/178] Update ambassador.jade

---
 app/templates/contribute/ambassador.jade | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/templates/contribute/ambassador.jade b/app/templates/contribute/ambassador.jade
index ef17d2735..dc1048ac6 100644
--- a/app/templates/contribute/ambassador.jade
+++ b/app/templates/contribute/ambassador.jade
@@ -31,7 +31,7 @@ block content
   
       h4(data-i18n="contribute.how_to_join") How to Join
       p
-        a(title='Contact', tabindex=-1, data-toggle="coco-modal", data-target="modal/contact", data-i18n="contact_us_url")
+        a(title='Contact', tabindex=-1, data-toggle="coco-modal", data-target="modal/contact", data-i18n="contribute.contact_us_url")
           | Contact us
         span , 
         span(data-i18n="contribute.ambassador_join_desc")

From 16f2379af9afce26e287c9baa6b36a26acd5c29e Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Fri, 7 Mar 2014 12:27:12 -0800
Subject: [PATCH 115/178] Update ru.coffee

---
 app/locale/ru.coffee | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/locale/ru.coffee b/app/locale/ru.coffee
index 0ee54cb73..1fdd72d87 100644
--- a/app/locale/ru.coffee
+++ b/app/locale/ru.coffee
@@ -400,7 +400,7 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     more_about_adventurer: "Узнать больше о том, как стать Искателем приключений"
     adventurer_subscribe_desc: "Получать email-ы при появлении новых уровней для тестирования."
     scribe_summary_pref: "CodeCombat будет не просто кучей уровней. Он также будет ресурсом знаний в области программирования, к которому игроки могут присоединиться. Таким образом, каждый Ремесленник может ссылаться на подробную статью для назидания игрока: документация сродни тому, что создана "
-    scribe_summary_sufx: ". Если вам нравится объяснять концепции программирования, этот класс для вас."
+    scribe_summary_suf: ". Если вам нравится объяснять концепции программирования, этот класс для вас."
     scribe_introduction_pref: "CodeCombat будет не просто кучей уровней. Он также включает в себя ресурс для познания, вики концепций программирования, которые уровни могут включать. Таким образом, вместо того, чтобы каждому Ремесленнику необходимо было подробно описывать, что такое оператор сравнения, они могут просто связать их уровень с уже написанной в назидание игрокам статьёй, описывающей их. Что-то по аналогии с "
     scribe_introduction_url_mozilla: "Mozilla Developer Network"
     scribe_introduction_suf: ". Если ваше представление о веселье это формулирование концепций программирования в форме Markdown, этот класс для вас."

From a3afa802d9eb5068af4e9b3239b74663d4f69ea9 Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Fri, 7 Mar 2014 12:33:18 -0800
Subject: [PATCH 116/178] Update ru.coffee

---
 app/locale/ru.coffee | 30 ++----------------------------
 1 file changed, 2 insertions(+), 28 deletions(-)

diff --git a/app/locale/ru.coffee b/app/locale/ru.coffee
index 1fdd72d87..8f2189174 100644
--- a/app/locale/ru.coffee
+++ b/app/locale/ru.coffee
@@ -129,6 +129,7 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     new_password: "Новый пароль"
     new_password_verify: "Подтверждение пароля"
     email_subscriptions: "Email-подписки"
+    email_notifications: "Уведомления"
     email_announcements: "Оповещения"
     email_notifications_description: "Получать периодические уведомления для вашего аккаунта."
     email_announcements_description: "Получать email-оповещения о последних новостях CodeCombat."
@@ -429,31 +430,4 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     ambassador_subscribe_desc: "Получать email-ы о разработке мультиплеера и обновлениях в системе поддержки."
     counselor_summary: "Ни одна из вышеупомянутых ролей не соответствует тому, в чём вы заинтересованы? Не волнуйтесь, мы в поисках тех, кто хочет приложить руку к разработке CodeCombat! Если вы заинтересованы в обучении, разработке игр, управлением проектами с открытым исходным кодом, или в чём-нибудь ещё, что, как вы думаете, будет актуально для нас, то этот класс для вас."
     counselor_introduction_1: "У вас есть жизненный опыт? Другая точка зрения на вещи, которые могут помочь нам решить, как формировать CodeCombat? Из всех этих ролей, эта, возможно, займёт меньше всего времени, но по отдельности, вы можете сделать наибольшие изменения. Мы в поисках морщинистых мудрецов, особенно в таких областях, как: обучение, разработка игр, управление проектами с открытым исходным кодом, технической рекрутинг, предпринимательство или дизайн."
-    counselor_introduction_2: "Или действительно всё, что имеет отношение к развитию CodeCombat. Если у вас есть знания и вы хотите поделиться ими, чтобы помочь вырастить этот проект, то этот класс для вас."
-    counselor_attribute_1: "Опыт, в любой из областей выше, или в том, что, как вы думаете, может быть полезным."
-    counselor_attribute_2: "Немного свободного времени!"
-    counselor_join_desc: "расскажите нам немного о себе, чем вы занимались и чем хотели бы заниматься. Мы поместим вас в наш список контактов и выйдем на связь, когда нам понадобится совет(не слишком часто)."
-    more_about_counselor: "Узнать больше о том, как стать Советником"
-    changes_auto_save: "Изменения сохраняются автоматически при переключении флажков."
-    diligent_scribes: "Наши старательные Писари:"
-    powerful_archmages: "Наши могущественные Архимаги:"
-    creative_artisans: "Наши творческие Ремесленники:"
-    brave_adventurers: "Наши отважные Искатели приключений:"
-    translating_diplomats: "Наши переводящие Дипломаты:"
-    helpful_ambassadors: "Наши полезные Послы:"
-
-  classes:
-    archmage_title: "Архимаг"
-    archmage_title_description: "(программист)"
-    artisan_title: "Ремесленник"
-    artisan_title_description: "(создатель уровней)"
-    adventurer_title: "Искатель приключений"
-    adventurer_title_description: "(тестировщик уровней)"
-    scribe_title: "Писарь"
-    scribe_title_description: "(редактор статей)"
-    diplomat_title: "Дипломат"
-    diplomat_title_description: "(переводчик)"
-    ambassador_title: "Посол"
-    ambassador_title_description: "(поддержка)"
-    counselor_title: "Советник"
-    counselor_title_description: "(эксперт/учитель)"
+    counselor_introduction_2: "Или действительно всё, что имеет отношение к развитию CodeCombat

From 683c578dcbc0e5bcffff0d38b1c313b09a514e85 Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Fri, 7 Mar 2014 12:36:17 -0800
Subject: [PATCH 117/178] Fix lolwtf deleting

---
 app/locale/ru.coffee | 29 ++++++++++++++++++++++++++++-
 1 file changed, 28 insertions(+), 1 deletion(-)

diff --git a/app/locale/ru.coffee b/app/locale/ru.coffee
index 8f2189174..36bcf1ac1 100644
--- a/app/locale/ru.coffee
+++ b/app/locale/ru.coffee
@@ -430,4 +430,31 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     ambassador_subscribe_desc: "Получать email-ы о разработке мультиплеера и обновлениях в системе поддержки."
     counselor_summary: "Ни одна из вышеупомянутых ролей не соответствует тому, в чём вы заинтересованы? Не волнуйтесь, мы в поисках тех, кто хочет приложить руку к разработке CodeCombat! Если вы заинтересованы в обучении, разработке игр, управлением проектами с открытым исходным кодом, или в чём-нибудь ещё, что, как вы думаете, будет актуально для нас, то этот класс для вас."
     counselor_introduction_1: "У вас есть жизненный опыт? Другая точка зрения на вещи, которые могут помочь нам решить, как формировать CodeCombat? Из всех этих ролей, эта, возможно, займёт меньше всего времени, но по отдельности, вы можете сделать наибольшие изменения. Мы в поисках морщинистых мудрецов, особенно в таких областях, как: обучение, разработка игр, управление проектами с открытым исходным кодом, технической рекрутинг, предпринимательство или дизайн."
-    counselor_introduction_2: "Или действительно всё, что имеет отношение к развитию CodeCombat
+    counselor_introduction_2: "Или действительно всё, что имеет отношение к развитию CodeCombat. Если у вас есть знания и вы хотите поделиться ими, чтобы помочь вырастить этот проект, то этот класс для вас."
+    counselor_attribute_1: "Опыт, в любой из областей выше, или в том, что, как вы думаете, может быть полезным."
+    counselor_attribute_2: "Немного свободного времени!"
+    counselor_join_desc: "расскажите нам немного о себе, чем вы занимались и чем хотели бы заниматься. Мы поместим вас в наш список контактов и выйдем на связь, когда нам понадобится совет(не слишком часто)."
+    more_about_counselor: "Узнать больше о том, как стать Советником"
+    changes_auto_save: "Изменения сохраняются автоматически при переключении флажков."
+    diligent_scribes: "Наши старательные Писари:"
+    powerful_archmages: "Наши могущественные Архимаги:"
+    creative_artisans: "Наши творческие Ремесленники:"
+    brave_adventurers: "Наши отважные Искатели приключений:"
+    translating_diplomats: "Наши переводящие Дипломаты:"
+    helpful_ambassadors: "Наши полезные Послы:"
+
+  classes:
+    archmage_title: "Архимаг"
+    archmage_title_description: "(программист)"
+    artisan_title: "Ремесленник"
+    artisan_title_description: "(создатель уровней)"
+    adventurer_title: "Искатель приключений"
+    adventurer_title_description: "(тестировщик уровней)"
+    scribe_title: "Писарь"
+    scribe_title_description: "(редактор статей)"
+    diplomat_title: "Дипломат"
+    diplomat_title_description: "(переводчик)"
+    ambassador_title: "Посол"
+    ambassador_title_description: "(поддержка)"
+    counselor_title: "Советник"
+    counselor_title_description: "(эксперт/учитель)"

From be034855385fe83b706385547a9755a5e04fa32b Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Fri, 7 Mar 2014 12:58:17 -0800
Subject: [PATCH 118/178] Set up the ladder play modal to be more forthright
 getting new players to play the tutorial.

---
 app/styles/play/ladder/play_modal.sass    |   7 ++
 app/templates/play/ladder/play_modal.jade | 119 ++++++++++++----------
 app/views/play/ladder/play_modal.coffee   |  17 ++++
 3 files changed, 88 insertions(+), 55 deletions(-)

diff --git a/app/styles/play/ladder/play_modal.sass b/app/styles/play/ladder/play_modal.sass
index 7d87d475b..465052114 100644
--- a/app/styles/play/ladder/play_modal.sass
+++ b/app/styles/play/ladder/play_modal.sass
@@ -1,7 +1,14 @@
 #ladder-play-modal
+  #noob-view p
+    font-size: 30px
+  
+  #skip-tutorial-button
+    font-size: 16px
+  
   .tutorial-suggestion
     text-align: center
     font-size: 18px
+    margin: 10px 0 30px
   
   .play-option
     margin-bottom: 15px
diff --git a/app/templates/play/ladder/play_modal.jade b/app/templates/play/ladder/play_modal.jade
index 5e7953896..25d492b6f 100644
--- a/app/templates/play/ladder/play_modal.jade
+++ b/app/templates/play/ladder/play_modal.jade
@@ -5,68 +5,77 @@ block modal-header-content
 
 block modal-body-content
   
-  p.tutorial-suggestion
-    span Not sure what's going on?
-    |  
-    a(href="/play/level/#{levelID}-tutorial") Play the tutorial first.
+  div#noob-view.secret
+    a(href="/play/level/#{levelID}-tutorial").btn.btn-success.btn-block.btn-lg
+      p
+        strong Play Tutorial
+      span Recommended if you've never played before
+    span.btn.btn-primary.btn-block.btn-lg#skip-tutorial-button Skip Tutorial
 
-  a(href="/play/level/#{levelID}?team=#{teamID}")
-    div.play-option
-      img(src=myPortrait).my-icon.only-one
-      img(src="/images/pages/play/ladder/"+teamID+"_ladder_tutorial.png", style="border: 1px solid #{teamColor}; background: #{teamBackgroundColor}").my-team-icon.img-circle.only-one
-      img(src=genericPortrait).opponent-icon
-      img(src="/images/pages/play/ladder/"+otherTeamID+"_ladder_tutorial.png", style="border: 1px solid #{opponentTeamColor}; background: #{opponentTeamBackgroundColor}").opponent-team-icon.img-circle
-      div.my-name.name-label.only-one
-        span= myName
-      div.opponent-name.name-label
-        span Simple AI
-      div.difficulty
-        span Warmup
-      div.vs VS
+  div#normal-view
   
-  if challengers.easy
-    a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.easy.sessionID}")
-      div.play-option.easy-option
+    p.tutorial-suggestion
+      strong Not sure what's going on?
+      |  
+      a(href="/play/level/#{levelID}-tutorial") Play the tutorial first.
+  
+    a(href="/play/level/#{levelID}?team=#{teamID}")
+      div.play-option
         img(src=myPortrait).my-icon.only-one
-        img(src="/images/pages/play/ladder/"+teamID+"_ladder_easy.png", style="border: 1px solid #{teamColor}; background: #{teamBackgroundColor}").my-team-icon.img-circle.only-one
-        img(src=challengers.easy.opponentImageSource||genericPortrait).opponent-icon
-        img(src="/images/pages/play/ladder/"+otherTeamID+"_ladder_easy.png", style="border: 1px solid #{opponentTeamColor}; background: #{opponentTeamBackgroundColor}").opponent-team-icon.img-circle
+        img(src="/images/pages/play/ladder/"+teamID+"_ladder_tutorial.png", style="border: 1px solid #{teamColor}; background: #{teamBackgroundColor}").my-team-icon.img-circle.only-one
+        img(src=genericPortrait).opponent-icon
+        img(src="/images/pages/play/ladder/"+otherTeamID+"_ladder_tutorial.png", style="border: 1px solid #{opponentTeamColor}; background: #{opponentTeamBackgroundColor}").opponent-team-icon.img-circle
         div.my-name.name-label.only-one
           span= myName
         div.opponent-name.name-label
-          span= challengers.easy.opponentName
+          span Simple AI
         div.difficulty
-          span Easy
-        div.vs VS
-
-  if challengers.medium
-    a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.medium.sessionID}")
-      div.play-option.medium-option
-        img(src=myPortrait).my-icon.only-one
-        img(src="/images/pages/play/ladder/"+teamID+"_ladder_medium.png", style="border: 1px solid #{teamColor}; background: #{teamBackgroundColor}").my-team-icon.img-circle.only-one
-        img(src=challengers.medium.opponentImageSource||genericPortrait).opponent-icon
-        img(src="/images/pages/play/ladder/"+otherTeamID+"_ladder_medium.png", style="border: 1px solid #{opponentTeamColor}; background: #{opponentTeamBackgroundColor}").opponent-team-icon.img-circle
-        div.my-name.name-label.only-one
-          span= myName
-        div.opponent-name.name-label
-          span= challengers.medium.opponentName
-        div.difficulty
-          span Medium
-        div.vs VS
-
-  if challengers.hard
-    a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.hard.sessionID}")
-      div.play-option.hard-option
-        img(src=myPortrait).my-icon.only-one
-        img(src="/images/pages/play/ladder/"+teamID+"_ladder_hard.png", style="border: 1px solid #{teamColor}; background: #{teamBackgroundColor}").my-team-icon.img-circle.only-one
-        img(src=challengers.hard.opponentImageSource||genericPortrait).opponent-icon
-        img(src="/images/pages/play/ladder/"+otherTeamID+"_ladder_hard.png", style="border: 1px solid #{opponentTeamColor}; background: #{opponentTeamBackgroundColor}").opponent-team-icon.img-circle
-        div.my-name.name-label.only-one
-          span= myName
-        div.opponent-name.name-label
-          span= challengers.hard.opponentName
-        div.difficulty
-          span Hard
+          span Warmup
         div.vs VS
+    
+    if challengers.easy
+      a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.easy.sessionID}")
+        div.play-option.easy-option
+          img(src=myPortrait).my-icon.only-one
+          img(src="/images/pages/play/ladder/"+teamID+"_ladder_easy.png", style="border: 1px solid #{teamColor}; background: #{teamBackgroundColor}").my-team-icon.img-circle.only-one
+          img(src=challengers.easy.opponentImageSource||genericPortrait).opponent-icon
+          img(src="/images/pages/play/ladder/"+otherTeamID+"_ladder_easy.png", style="border: 1px solid #{opponentTeamColor}; background: #{opponentTeamBackgroundColor}").opponent-team-icon.img-circle
+          div.my-name.name-label.only-one
+            span= myName
+          div.opponent-name.name-label
+            span= challengers.easy.opponentName
+          div.difficulty
+            span Easy
+          div.vs VS
+  
+    if challengers.medium
+      a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.medium.sessionID}")
+        div.play-option.medium-option
+          img(src=myPortrait).my-icon.only-one
+          img(src="/images/pages/play/ladder/"+teamID+"_ladder_medium.png", style="border: 1px solid #{teamColor}; background: #{teamBackgroundColor}").my-team-icon.img-circle.only-one
+          img(src=challengers.medium.opponentImageSource||genericPortrait).opponent-icon
+          img(src="/images/pages/play/ladder/"+otherTeamID+"_ladder_medium.png", style="border: 1px solid #{opponentTeamColor}; background: #{opponentTeamBackgroundColor}").opponent-team-icon.img-circle
+          div.my-name.name-label.only-one
+            span= myName
+          div.opponent-name.name-label
+            span= challengers.medium.opponentName
+          div.difficulty
+            span Medium
+          div.vs VS
+  
+    if challengers.hard
+      a(href="/play/level/#{levelID}?team=#{teamID}&opponent=#{challengers.hard.sessionID}")
+        div.play-option.hard-option
+          img(src=myPortrait).my-icon.only-one
+          img(src="/images/pages/play/ladder/"+teamID+"_ladder_hard.png", style="border: 1px solid #{teamColor}; background: #{teamBackgroundColor}").my-team-icon.img-circle.only-one
+          img(src=challengers.hard.opponentImageSource||genericPortrait).opponent-icon
+          img(src="/images/pages/play/ladder/"+otherTeamID+"_ladder_hard.png", style="border: 1px solid #{opponentTeamColor}; background: #{opponentTeamBackgroundColor}").opponent-team-icon.img-circle
+          div.my-name.name-label.only-one
+            span= myName
+          div.opponent-name.name-label
+            span= challengers.hard.opponentName
+          div.difficulty
+            span Hard
+          div.vs VS
 
 block modal-footer
\ No newline at end of file
diff --git a/app/views/play/ladder/play_modal.coffee b/app/views/play/ladder/play_modal.coffee
index 28f689b14..e5d83fc3b 100644
--- a/app/views/play/ladder/play_modal.coffee
+++ b/app/views/play/ladder/play_modal.coffee
@@ -10,6 +10,10 @@ module.exports = class LadderPlayModal extends View
   template: template
   closeButton: true
   startsLoading: true
+  @shownTutorialButton: false
+  
+  events:
+    'click #skip-tutorial-button': 'hideTutorialButtons'
 
   constructor: (options, @level, @session, @team) ->
     super(options)
@@ -56,6 +60,7 @@ module.exports = class LadderPlayModal extends View
   finishRendering: ->
     @startsLoading = false
     @render()
+    @maybeShowTutorialButtons()
 
   getRenderData: ->
     ctx = super()
@@ -87,6 +92,18 @@ module.exports = class LadderPlayModal extends View
 
     ctx.myName = me.get('name') || 'Newcomer'
     ctx
+
+  maybeShowTutorialButtons: ->
+    return if @session or LadderPlayModal.shownTutorialButton
+    @$el.find('#normal-view').addClass('secret')
+    @$el.find('.modal-header').addClass('secret')
+    @$el.find('#noob-view').removeClass('secret')
+    LadderPlayModal.shownTutorialButton = true
+
+  hideTutorialButtons: ->
+    @$el.find('#normal-view').removeClass('secret')
+    @$el.find('.modal-header').removeClass('secret')
+    @$el.find('#noob-view').addClass('secret')
     
   # Choosing challengers
 

From 17231bce06f7c37adfc89f462ea9a9d75c1241b8 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Fri, 7 Mar 2014 13:13:58 -0800
Subject: [PATCH 119/178] Added showsGuide and type properties to level.

---
 app/views/editor/level/settings_tab_view.coffee | 7 ++++++-
 server/levels/level_schema.coffee               | 3 ++-
 2 files changed, 8 insertions(+), 2 deletions(-)

diff --git a/app/views/editor/level/settings_tab_view.coffee b/app/views/editor/level/settings_tab_view.coffee
index 4e2518be6..7a1290db1 100644
--- a/app/views/editor/level/settings_tab_view.coffee
+++ b/app/views/editor/level/settings_tab_view.coffee
@@ -8,7 +8,12 @@ module.exports = class SettingsTabView extends View
   id: 'editor-level-settings-tab-view'
   className: 'tab-pane'
   template: template
-  editableSettings: ['name', 'description', 'documentation', 'nextLevel', 'background', 'victory', 'i18n', 'icon', 'goals']  # not thangs or scripts or the backend stuff
+  
+  # not thangs or scripts or the backend stuff
+  editableSettings: [
+    'name', 'description', 'documentation', 'nextLevel', 'background', 'victory', 'i18n', 'icon', 'goals',
+    'type', 'showsGuide'
+  ]
 
   subscriptions:
     'level-loaded': 'onLevelLoaded'
diff --git a/server/levels/level_schema.coffee b/server/levels/level_schema.coffee
index d0d448267..cbfb84497 100644
--- a/server/levels/level_schema.coffee
+++ b/server/levels/level_schema.coffee
@@ -226,7 +226,8 @@ _.extend LevelSchema.properties,
   i18n: {type: "object", format: 'i18n', props: ['name', 'description'], description: "Help translate this level"}
   icon: { type: 'string', format: 'image-file', title: 'Icon' }
   goals: c.array {title: 'Goals', description: 'An array of goals which are visible to the player and can trigger scripts.'}, GoalSchema
-
+  type: c.shortString(title: "Type", description: "What kind of level this is.", "enum": ['campaign', 'ladder'])
+  showsGuide: c.shortString(title: "Shows Guide", description: "If the guide is shown at the beginning of the level.", "enum": ['first-time', 'always'])
 
 c.extendBasicProperties LevelSchema, 'level'
 c.extendSearchableProperties LevelSchema

From 2f68e64dc5d728ea96203f0fc9fa3c7474c6f090 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Fri, 7 Mar 2014 13:14:27 -0800
Subject: [PATCH 120/178] Almost forgot to add to the level handler the new
 properties.

---
 server/levels/level_handler.coffee | 2 ++
 1 file changed, 2 insertions(+)

diff --git a/server/levels/level_handler.coffee b/server/levels/level_handler.coffee
index 0cf0a2066..da79be8d6 100644
--- a/server/levels/level_handler.coffee
+++ b/server/levels/level_handler.coffee
@@ -20,6 +20,8 @@ LevelHandler = class LevelHandler extends Handler
     'i18n'
     'icon'
     'goals'
+    'type'
+    'showsGuide'
   ]
 
   postEditableProperties: ['name']

From 990a2e34e56394ee93f6d5238226b6899ccc0d7f Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Fri, 7 Mar 2014 15:15:49 -0800
Subject: [PATCH 121/178] Utilizing the new level type property in the level
 handler.

---
 server/levels/level_handler.coffee | 5 +++--
 1 file changed, 3 insertions(+), 2 deletions(-)

diff --git a/server/levels/level_handler.coffee b/server/levels/level_handler.coffee
index da79be8d6..94bf33dd8 100644
--- a/server/levels/level_handler.coffee
+++ b/server/levels/level_handler.coffee
@@ -49,10 +49,11 @@ LevelHandler = class LevelHandler extends Handler
           majorVersion: level.version.major
         creator: req.user.id
 
-      # TODO: generalize this for levels that need teams
       if req.query.team?
         sessionQuery.team = req.query.team
-      else if level.name is 'Project DotA'
+
+      # TODO: generalize this for levels based on their teams
+      else if level.get('type') is 'ladder'
         sessionQuery.team = 'humans'
       
       Session.findOne(sessionQuery).exec (err, doc) =>

From 13b9a9cb5853bbe256c35d9d94ea8f8906a5bc36 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Fri, 7 Mar 2014 15:16:13 -0800
Subject: [PATCH 122/178] Added dungeon arena tutorial to a few hardcoded
 places.

---
 app/views/play/level/control_bar_view.coffee      | 2 +-
 app/views/play/level/tome/cast_button_view.coffee | 2 +-
 2 files changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/views/play/level/control_bar_view.coffee b/app/views/play/level/control_bar_view.coffee
index 06d689f82..bae8753b6 100644
--- a/app/views/play/level/control_bar_view.coffee
+++ b/app/views/play/level/control_bar_view.coffee
@@ -54,7 +54,7 @@ module.exports = class ControlBarView extends View
     c.ladderGame = @ladderGame
     c.homeLink = "/"
     levelID = @level.get('slug')
-    if levelID in ["brawlwood", "brawlwood-tutorial", "dungeon-arena"]
+    if levelID in ["brawlwood", "brawlwood-tutorial", "dungeon-arena", "dungeon-arena-tutorial"]
       levelID = 'brawlwood' if levelID is 'brawlwood-tutorial'
       c.homeLink = "/play/ladder/" + levelID
     c
diff --git a/app/views/play/level/tome/cast_button_view.coffee b/app/views/play/level/tome/cast_button_view.coffee
index 15d59f194..af45ca57c 100644
--- a/app/views/play/level/tome/cast_button_view.coffee
+++ b/app/views/play/level/tome/cast_button_view.coffee
@@ -35,7 +35,7 @@ module.exports = class CastButtonView extends View
     # TODO: use a User setting instead of localStorage
     delay = localStorage.getItem 'autocastDelay'
     delay ?= 5000
-    if @levelID in ['brawlwood', 'brawlwood-tutorial', 'dungeon-arena']
+    if @levelID in ['brawlwood', 'brawlwood-tutorial', 'dungeon-arena', 'dungeon-arena-tutorial']
       delay = 90019001
     @setAutocastDelay delay
 

From 8fa9c9c41052eed0048c3dd7cbbbeecc8f171fe7 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Fri, 7 Mar 2014 15:18:56 -0800
Subject: [PATCH 123/178] Set up the level play view to show the guide on
 startup based on the showGuide property.

---
 app/lib/LevelLoader.coffee       |  1 +
 app/views/kinds/CocoView.coffee  |  5 +++--
 app/views/play/level_view.coffee | 27 +++++++++++++++++++++++++--
 3 files changed, 29 insertions(+), 4 deletions(-)

diff --git a/app/lib/LevelLoader.coffee b/app/lib/LevelLoader.coffee
index 18ae9f82f..a0e1d78b7 100644
--- a/app/lib/LevelLoader.coffee
+++ b/app/lib/LevelLoader.coffee
@@ -229,6 +229,7 @@ module.exports = class LevelLoader extends CocoClass
   notifyProgress: ->
     Backbone.Mediator.publish 'level-loader:progress-changed', progress: @progress()
     @initWorld() if @allDone()
+    @trigger 'progress'
     @trigger 'loaded-all' if @progress() is 1
 
   destroy: ->
diff --git a/app/views/kinds/CocoView.coffee b/app/views/kinds/CocoView.coffee
index 806a2d25c..b95abd12e 100644
--- a/app/views/kinds/CocoView.coffee
+++ b/app/views/kinds/CocoView.coffee
@@ -99,10 +99,11 @@ module.exports = class CocoView extends Backbone.View
     view = application.router.getView(target, '_modal') # could set up a system for loading cached modals, if told to
     @openModalView(view)
 
-  openModalView: (modalView) ->
-    return if @waitingModal # can only have one waiting at once
+  openModalView: (modalView, softly=false) ->
+    return if waitingModal # can only have one waiting at once
     if visibleModal
       waitingModal = modalView
+      return if softly
       return visibleModal.hide() if visibleModal.$el.is(':visible') # close, then this will get called again
       return @modalClosed(visibleModal) # was closed, but modalClosed was not called somehow
     modalView.render()
diff --git a/app/views/play/level_view.coffee b/app/views/play/level_view.coffee
index 11fa59ba3..a7faa74cb 100644
--- a/app/views/play/level_view.coffee
+++ b/app/views/play/level_view.coffee
@@ -16,6 +16,7 @@ LevelLoader = require 'lib/LevelLoader'
 LevelSession = require 'models/LevelSession'
 Level = require 'models/Level'
 LevelComponent = require 'models/LevelComponent'
+Article = require 'models/Article'
 Camera = require 'lib/surface/Camera'
 AudioPlayer = require 'lib/AudioPlayer'
 
@@ -105,7 +106,8 @@ module.exports = class PlayLevelView extends View
 
   load: ->
     @levelLoader = new LevelLoader supermodel: @supermodel, levelID: @levelID, sessionID: @sessionID, opponentSessionID: @getQueryVariable('opponent'), team: @getQueryVariable("team")
-    @levelLoader.once 'loaded-all', @onLevelLoaderLoaded
+    @levelLoader.once 'loaded-all', @onLevelLoaderLoaded, @
+    @levelLoader.on 'progress', @onLevelLoaderProgressChanged, @
     @god = new God()
 
   getRenderData: ->
@@ -124,7 +126,28 @@ module.exports = class PlayLevelView extends View
     @$el.find('#level-done-button').hide()
     super()
 
-  onLevelLoaderLoaded: =>
+  onLevelLoaderProgressChanged: ->
+    return if @seenDocs
+    return unless showFrequency = @levelLoader.level.get('showGuide')
+    session = @levelLoader.session
+    diff = new Date().getTime() - new Date(session.get('created')).getTime()
+    return if showFrequency is 'first-time' and diff > (5 * 60 * 1000)
+    return unless @levelLoader.level.loaded
+    articles = @levelLoader.supermodel.getModels Article
+    for article in articles
+      return unless article.loaded
+    @showGuide()
+
+  showGuide: ->
+    @seenDocs = true
+    DocsModal = require './level/modal/docs_modal'
+    options = {docs: @levelLoader.level.get('documentation'), supermodel: @supermodel}
+    @openModalView(new DocsModal(options), true)
+    Backbone.Mediator.subscribeOnce 'modal-closed', @onLevelLoaderLoaded, @
+    return true
+
+  onLevelLoaderLoaded: ->
+    return unless @levelLoader.progress() is 1 # double check, since closing the guide may trigger this early
     # Save latest level played in local storage
     if window.currentModal and not window.currentModal.destroyed
       @loadingScreen.showReady()

From ec86a07906f3facf802a5ddb5ce8be8b2befc347 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Fri, 7 Mar 2014 15:20:54 -0800
Subject: [PATCH 124/178] Changed Camera min/max zoom.

---
 app/lib/surface/Camera.coffee | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/lib/surface/Camera.coffee b/app/lib/surface/Camera.coffee
index 76323e5e8..07c4fbbcc 100644
--- a/app/lib/surface/Camera.coffee
+++ b/app/lib/surface/Camera.coffee
@@ -5,8 +5,8 @@ CocoClass = require 'lib/CocoClass'
 r2d = (radians) -> radians * 180 / Math.PI
 d2r = (degrees) -> degrees / 180 * Math.PI
 
-MAX_ZOOM = 8
-MIN_ZOOM = 0.1
+MAX_ZOOM = 4
+MIN_ZOOM = 0.05
 DEFAULT_ZOOM = 2.0
 DEFAULT_TARGET = {x:0, y:0}
 DEFAULT_TIME = 1000

From 98b36d299ca09793794a4dca1cdddeec8c354c7f Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Sat, 8 Mar 2014 05:05:35 +0400
Subject: [PATCH 125/178] Update article model

---
 app/views/play/level/modal/docs_modal.coffee | 3 ++-
 server/articles/article_handler.coffee       | 2 +-
 server/articles/article_schema.coffee        | 1 +
 3 files changed, 4 insertions(+), 2 deletions(-)

diff --git a/app/views/play/level/modal/docs_modal.coffee b/app/views/play/level/modal/docs_modal.coffee
index c7b77a287..3e9ab5cf8 100644
--- a/app/views/play/level/modal/docs_modal.coffee
+++ b/app/views/play/level/modal/docs_modal.coffee
@@ -25,7 +25,8 @@ module.exports = class DocsModal extends View
     @docs = specific.concat(general)
     marked.setOptions {gfm: true, sanitize: false, smartLists: true, breaks: false}
     @docs = _.cloneDeep(@docs)
-    doc.html = marked(doc.body) for doc in @docs
+    doc.html = marked(doc.i18n?[me.lang()]?.body or doc.body) for doc in @docs
+    doc.name = (doc.i18n?[me.lang()]?.name or doc.name) for doc in @docs
     doc.slug = _.string.slugify(doc.name) for doc in @docs
     super()
 
diff --git a/server/articles/article_handler.coffee b/server/articles/article_handler.coffee
index ac4fb4b97..b519b8b9f 100644
--- a/server/articles/article_handler.coffee
+++ b/server/articles/article_handler.coffee
@@ -3,7 +3,7 @@ Handler = require('../commons/Handler')
 
 ArticleHandler = class ArticleHandler extends Handler
   modelClass: Article
-  editableProperties: ['body', 'name']
+  editableProperties: ['body', 'name', 'i18n']
 
   hasAccess: (req) ->
     req.method is 'GET' or req.user?.isAdmin()
diff --git a/server/articles/article_schema.coffee b/server/articles/article_schema.coffee
index 08226d183..48ae42821 100644
--- a/server/articles/article_schema.coffee
+++ b/server/articles/article_schema.coffee
@@ -4,6 +4,7 @@ ArticleSchema = c.object()
 c.extendNamedProperties ArticleSchema  # name first
 
 ArticleSchema.properties.body = { type: 'string', title: 'Content', format: 'markdown' }
+ArticleSchema.properties.i18n = { type: 'object', title: 'i18n', format: 'i18n', props: ['body'] }
 
 c.extendBasicProperties(ArticleSchema, 'article')
 c.extendSearchableProperties(ArticleSchema)

From fe2ced1a67fbc7288541cc1ff02ec8491d3438cf Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Fri, 7 Mar 2014 17:37:11 -0800
Subject: [PATCH 126/178] Gave the health bar a border, made it a little more
 high res.

---
 app/lib/surface/CocoSprite.coffee   |  4 ++--
 app/lib/surface/sprite_utils.coffee | 14 +++++++++++++-
 2 files changed, 15 insertions(+), 3 deletions(-)

diff --git a/app/lib/surface/CocoSprite.coffee b/app/lib/surface/CocoSprite.coffee
index 7c7748121..cdbde389e 100644
--- a/app/lib/surface/CocoSprite.coffee
+++ b/app/lib/surface/CocoSprite.coffee
@@ -328,7 +328,7 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
       return if @thang.health is @lastHealth
       @lastHealth = @thang.health
       healthPct = Math.max(@thang.health / @thang.maxHealth, 0)
-      bar.scaleX = healthPct
+      bar.scaleX = healthPct / bar.baseScale
       healthOffset = @getOffset 'aboveHead'
       [bar.x, bar.y] = [healthOffset.x - bar.width / 2, healthOffset.y]
 
@@ -357,7 +357,7 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
     bar = @healthBar = createProgressBar(healthColor, healthOffset.y)
     bar.x = healthOffset.x - bar.width / 2
     bar.name = 'health bar'
-    bar.cache 0, -bar.height / 2, bar.width, bar.height
+    bar.cache 0, -bar.height * bar.baseScale / 2, bar.width * bar.baseScale, bar.height * bar.baseScale
     @displayObject.addChild bar
 
   getActionProp: (prop, subProp, def=null) ->
diff --git a/app/lib/surface/sprite_utils.coffee b/app/lib/surface/sprite_utils.coffee
index 6d106e0aa..68984f370 100644
--- a/app/lib/surface/sprite_utils.coffee
+++ b/app/lib/surface/sprite_utils.coffee
@@ -1,16 +1,28 @@
 PROG_BAR_WIDTH = 20
 PROG_BAR_HEIGHT = 2
+PROG_BAR_SCALE = 2.5
+EDGE_SIZE = 0.3
 
 module.exports.createProgressBar = createProgressBar = (color, y, width=PROG_BAR_WIDTH, height=PROG_BAR_HEIGHT) ->
   g = new createjs.Graphics()
   g.setStrokeStyle(1)
+
+  sWidth = width * PROG_BAR_SCALE
+  sHeight = height * PROG_BAR_SCALE
+  sEdge = EDGE_SIZE * PROG_BAR_SCALE
+
+  g.beginFill(createjs.Graphics.getRGB(0, 0, 0))
+  g.drawRect(0, -sHeight/2, sWidth, sHeight, sHeight)
   g.beginFill(createjs.Graphics.getRGB(color...))
-  g.drawRoundRect(0, -1, width, height, height)
+  g.drawRoundRect(sEdge, sEdge - sHeight/2, sWidth-sEdge*2, sHeight-sEdge*2, sHeight-sEdge*2)
 
   s = new createjs.Shape(g)
   s.x = -width / 2
   s.y = y
   s.z = 100
+  s.baseScale = PROG_BAR_SCALE
+  s.scaleX = 1 / PROG_BAR_SCALE
+  s.scaleY = 1 / PROG_BAR_SCALE
   s.width = width
   s.height = height
   return s

From 2f378106caa2251c675cd4bd28576ec83a0c5fdf Mon Sep 17 00:00:00 2001
From: Michael Schmatz <schmatz@umich.edu>
Date: Fri, 7 Mar 2014 18:16:48 -0800
Subject: [PATCH 127/178] Added checking level ladder status before submitting

---
 app/views/play/ladder/my_matches_tab.coffee |  2 +-
 server/queues/scoring.coffee                | 40 ++++++++++++---------
 2 files changed, 25 insertions(+), 17 deletions(-)

diff --git a/app/views/play/ladder/my_matches_tab.coffee b/app/views/play/ladder/my_matches_tab.coffee
index 6b2cf71be..b08a15083 100644
--- a/app/views/play/ladder/my_matches_tab.coffee
+++ b/app/views/play/ladder/my_matches_tab.coffee
@@ -95,7 +95,7 @@ module.exports = class MyMatchesTabView extends CocoView
     success = => @setRankingButtonText(button, 'ranked')
     failure = => @setRankingButtonText(button, 'failed')
     
-    ajaxData = { session: sessionID, levelID: @level.attributes.original, levelMajorVersion: @level.attributes.version.major }
+    ajaxData = { session: sessionID, levelID: @level.id, originalLevelID: @level.attributes.original, levelMajorVersion: @level.attributes.version.major }
     $.ajax '/queue/scoring', {
       type: 'POST'
       data: ajaxData
diff --git a/server/queues/scoring.coffee b/server/queues/scoring.coffee
index dd2a51030..1e2cd3f29 100644
--- a/server/queues/scoring.coffee
+++ b/server/queues/scoring.coffee
@@ -8,6 +8,7 @@ db = require './../routes/db'
 mongoose = require 'mongoose'
 queues = require '../commons/queue'
 LevelSession = require '../levels/sessions/LevelSession'
+Level = require '../levels/Level'
 TaskLog = require './task/ScoringTask'
 bayes = new (require 'bayesian-battle')()
 
@@ -48,7 +49,8 @@ addPairwiseTaskToQueue = (taskPair, cb) ->
 
 module.exports.createNewTask = (req, res) ->
   requestSessionID = req.body.session
-  requestLevelID = req.body.levelID
+  requestLevelID = req.body.originalLevelID
+  requestCurrentLevelID = req.body.levelID
   requestLevelMajorVersion = parseInt(req.body.levelMajorVersion)
   
   validatePermissions req, requestSessionID, (error, permissionsAreValid) ->
@@ -56,21 +58,27 @@ module.exports.createNewTask = (req, res) ->
     unless permissionsAreValid then return errors.forbidden res, "You do not have the permissions to submit that game to the leaderboard"
 
     return errors.badInput res, "The session ID is invalid" unless typeof requestSessionID is "string"
-
-    fetchSessionToSubmit requestSessionID, (err, sessionToSubmit) ->
-      if err? then return errors.serverError res, "There was an error finding the given session."
-
-      updateSessionToSubmit sessionToSubmit, (err, data) ->
-        if err? then return errors.serverError res, "There was an error updating the session"
-        opposingTeam = calculateOpposingTeam(sessionToSubmit.team)
-        fetchInitialSessionsToRankAgainst opposingTeam,requestLevelID, requestLevelMajorVersion, (err, sessionsToRankAgainst) ->
-          if err? then return errors.serverError res, "There was an error fetching the sessions to rank against"
-
-          taskPairs = generateTaskPairs(sessionsToRankAgainst, sessionToSubmit)
-          sendEachTaskPairToTheQueue taskPairs, (taskPairError) ->
-            if taskPairError? then return errors.serverError res, "There was an error sending the task pairs to the queue"
-
-            sendResponseObject req, res, {"message":"All task pairs were succesfully sent to the queue"}
+    Level.findOne({_id: requestCurrentLevelID}).lean().select('type').exec (err, levelWithType) ->
+      if err? then return errors.serverError res, "There was an error finding the level type"
+        
+      if not levelWithType.type or levelWithType.type isnt "ladder" 
+        console.log "The level type of level with ID #{requestLevelID} is #{levelWithType.type}"
+        return errors.badInput res, "That level isn't a ladder level"
+  
+      fetchSessionToSubmit requestSessionID, (err, sessionToSubmit) ->
+        if err? then return errors.serverError res, "There was an error finding the given session."
+  
+        updateSessionToSubmit sessionToSubmit, (err, data) ->
+          if err? then return errors.serverError res, "There was an error updating the session"
+          opposingTeam = calculateOpposingTeam(sessionToSubmit.team)
+          fetchInitialSessionsToRankAgainst opposingTeam,requestLevelID, requestLevelMajorVersion, (err, sessionsToRankAgainst) ->
+            if err? then return errors.serverError res, "There was an error fetching the sessions to rank against"
+  
+            taskPairs = generateTaskPairs(sessionsToRankAgainst, sessionToSubmit)
+            sendEachTaskPairToTheQueue taskPairs, (taskPairError) ->
+              if taskPairError? then return errors.serverError res, "There was an error sending the task pairs to the queue"
+  
+              sendResponseObject req, res, {"message":"All task pairs were succesfully sent to the queue"}
 
 module.exports.dispatchTaskToConsumer = (req, res) ->
   if isUserAnonymous(req) then return errors.forbidden res, "You need to be logged in to simulate games"

From 085fb82cc999aa7e7961434b6adbcc3df5563db9 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Fri, 7 Mar 2014 21:02:10 -0800
Subject: [PATCH 128/178] Fixed a multiplayer link and debug mark alpha.

---
 app/lib/surface/CocoSprite.coffee               | 2 +-
 app/lib/surface/Mark.coffee                     | 4 ++--
 app/lib/surface/Surface.coffee                  | 2 +-
 app/templates/play/level/modal/multiplayer.jade | 2 +-
 server/articles/article_schema.coffee           | 2 +-
 5 files changed, 6 insertions(+), 6 deletions(-)

diff --git a/app/lib/surface/CocoSprite.coffee b/app/lib/surface/CocoSprite.coffee
index cdbde389e..b454031e1 100644
--- a/app/lib/surface/CocoSprite.coffee
+++ b/app/lib/surface/CocoSprite.coffee
@@ -389,7 +389,7 @@ module.exports = CocoSprite = class CocoSprite extends CocoClass
     @addMark('bounds').toggle true if @thang?.drawsBounds
     @addMark('shadow').toggle true unless @thangType.get('shadow') is 0
     mark.update() for name, mark of @marks
-    #@thang.effectNames = ['berserk', 'confuse', 'control', 'curse', 'fear', 'poison', 'paralyze', 'regen', 'sleep', 'slow', 'speed']
+    #@thang.effectNames = ['berserk', 'confuse', 'control', 'curse', 'fear', 'poison', 'paralyze', 'regen', 'sleep', 'slow', 'haste']
     @updateEffectMarks() if @thang?.effectNames?.length or @previousEffectNames?.length
 
   updateEffectMarks: ->
diff --git a/app/lib/surface/Mark.coffee b/app/lib/surface/Mark.coffee
index 5a2e7689f..62801a60f 100644
--- a/app/lib/surface/Mark.coffee
+++ b/app/lib/surface/Mark.coffee
@@ -151,14 +151,14 @@ module.exports = class Mark extends CocoClass
     @thangType.fetch()
     markThangTypes[name] = @thangType
     window.mtt = markThangTypes
-    
+
   onLoadedThangType: ->
     @build()
     @toggle(@toggleTo) if @toggleTo?
 
   update: (pos=null) ->
     return false unless @on and @mark
-    @mark.alpha = if @hidden then 0 else 1
+    @mark.visible = not @hidden
     @updatePosition pos
     @updateRotation()
     @updateScale()
diff --git a/app/lib/surface/Surface.coffee b/app/lib/surface/Surface.coffee
index c5e15813d..feb87fc93 100644
--- a/app/lib/surface/Surface.coffee
+++ b/app/lib/surface/Surface.coffee
@@ -302,7 +302,7 @@ module.exports = Surface = class Surface extends CocoClass
       world: @world
     )
 
-    if @lastFrame < @world.totalFrames and @currentFrame >= @world.totalFrames
+    if @lastFrame < @world.totalFrames and @currentFrame >= @world.totalFrames - 1
       @spriteBoss.stop()
       @playbackOverScreen.show()
       @ended = true
diff --git a/app/templates/play/level/modal/multiplayer.jade b/app/templates/play/level/modal/multiplayer.jade
index bc8cc39b5..2135e1977 100644
--- a/app/templates/play/level/modal/multiplayer.jade
+++ b/app/templates/play/level/modal/multiplayer.jade
@@ -30,7 +30,7 @@
       if me.get('anonymous')
         p Sign in or create an account and get your solution on the leaderboard!
       else
-        a#go-to-leaderboard-button.btn.btn-primary(href="/play/ladder/#{levelSlug}/team/#{team}") Go to the leaderboard!
+        a#go-to-leaderboard-button.btn.btn-primary(href="/play/ladder/#{levelSlug}#my-matches") Go to the leaderboard!
         p You can submit your game to be ranked from the leaderboard page.
   
   .modal-footer
diff --git a/server/articles/article_schema.coffee b/server/articles/article_schema.coffee
index 48ae42821..1fd4769f7 100644
--- a/server/articles/article_schema.coffee
+++ b/server/articles/article_schema.coffee
@@ -10,4 +10,4 @@ c.extendBasicProperties(ArticleSchema, 'article')
 c.extendSearchableProperties(ArticleSchema)
 c.extendVersionedProperties(ArticleSchema, 'article')
 
-module.exports = ArticleSchema
\ No newline at end of file
+module.exports = ArticleSchema

From 68099bbd7cc505721ceddf142069a3065275fc8d Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Fri, 7 Mar 2014 21:20:09 -0800
Subject: [PATCH 129/178] Victory modal for ladder games takes you back to the
 ladder matches.

---
 app/templates/play/level/modal/victory.jade     | 4 +++-
 app/views/play/ladder_view.coffee               | 5 +++--
 app/views/play/level/modal/victory_modal.coffee | 1 +
 3 files changed, 7 insertions(+), 3 deletions(-)

diff --git a/app/templates/play/level/modal/victory.jade b/app/templates/play/level/modal/victory.jade
index 5dca8f3f6..45c7f567f 100644
--- a/app/templates/play/level/modal/victory.jade
+++ b/app/templates/play/level/modal/victory.jade
@@ -14,7 +14,9 @@
     div!= body
   
   .modal-footer
-    if hasNextLevel
+    if level.get('type') === 'ladder'
+      a.btn.btn-primary(href="/play/ladder/#{level.get('slug')}#my-matches", data-dismiss="modal", data-i18n="play_level.victory_go_home") Go Home
+    else if hasNextLevel
       button.btn.btn-primary.next-level-button(data-dismiss="modal", data-i18n="play_level.victory_play_next_level") Play Next Level
     else
       a.btn.btn-primary(href="/", data-dismiss="modal", data-i18n="play_level.victory_go_home") Go Home
diff --git a/app/views/play/ladder_view.coffee b/app/views/play/ladder_view.coffee
index 81e2f6558..7b6597380 100644
--- a/app/views/play/ladder_view.coffee
+++ b/app/views/play/ladder_view.coffee
@@ -72,6 +72,7 @@ module.exports = class LadderView extends RootView
     @sessions.fetch({"success": @refreshViews})
 
   refreshViews: =>
+    return if @destroyed
     @ladderTab.refreshLadder()
     @myMatchesTab.refreshMatches()
     console.log "refreshed views!"
@@ -114,12 +115,12 @@ module.exports = class LadderView extends RootView
 
   onClickPlayButton: (e) ->
     @showPlayModal($(e.target).closest('.play-button').data('team'))
-    
+
   showPlayModal: (teamID) ->
     session = (s for s in @sessions.models when s.get('team') is teamID)[0]
     modal = new LadderPlayModal({}, @level, session, teamID)
     @openModalView modal
-    
+
   destroy: ->
     clearInterval @refreshInterval
     @simulator.destroy()
diff --git a/app/views/play/level/modal/victory_modal.coffee b/app/views/play/level/modal/victory_modal.coffee
index e2fa6b607..6773982ad 100644
--- a/app/views/play/level/modal/victory_modal.coffee
+++ b/app/views/play/level/modal/victory_modal.coffee
@@ -64,6 +64,7 @@ module.exports = class VictoryModal extends View
     c.me = me
     c.hasNextLevel = _.isObject(@level.get('nextLevel')) and (@level.get('name') isnt "Mobile Artillery")
     c.levelName = @level.get('i18n')?[me.lang()]?.name ? @level.get('name')
+    c.level = @level
     if me.get 'hourOfCode'
       # Show the Hour of Code "I'm Done" tracking pixel after they played for 30 minutes
       elapsed = (new Date() - new Date(me.get('dateCreated')))

From bcd8c837b4461d841aebca2669ba66b58d489a20 Mon Sep 17 00:00:00 2001
From: Yinkan Li <liyinkan.biz@gmail.com>
Date: Sat, 8 Mar 2014 23:53:47 +0800
Subject: [PATCH 130/178] =?UTF-8?q?Update=20=EF=BC=9A=20employers=20for=20?=
 =?UTF-8?q?Chinese=20simplified?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Update : employers for Chinese simplified
---
 app/locale/zh-HANS.coffee | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/locale/zh-HANS.coffee b/app/locale/zh-HANS.coffee
index fd0e42cbf..274caf775 100644
--- a/app/locale/zh-HANS.coffee
+++ b/app/locale/zh-HANS.coffee
@@ -31,7 +31,7 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
     about: "关于"
     contact: "联系"
     twitter_follow: "关注"
-    employers: "雇佣我们"
+    employers: "招募信息"
 
   versions:
     save_version_title: "保存新版本"

From 452857e66cf7b215d5bd44c260f5681bf55dc283 Mon Sep 17 00:00:00 2001
From: Yinkan Li <liyinkan.biz@gmail.com>
Date: Sat, 8 Mar 2014 23:56:05 +0800
Subject: [PATCH 131/178] update Chinese Simplified

update
legal
concat
---
 app/locale/zh-HANS.coffee | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/locale/zh-HANS.coffee b/app/locale/zh-HANS.coffee
index 274caf775..6611b070d 100644
--- a/app/locale/zh-HANS.coffee
+++ b/app/locale/zh-HANS.coffee
@@ -27,9 +27,9 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
     admin: "管理"
     home: "首页"
     contribute: "贡献"
-    legal: "法律"
+    legal: "版权声明"
     about: "关于"
-    contact: "联系"
+    contact: "联系我们"
     twitter_follow: "关注"
     employers: "招募信息"
 

From bfad68b03b0ac3212e5b76d4ea933fbad890cd13 Mon Sep 17 00:00:00 2001
From: Yinkan Li <liyinkan.biz@gmail.com>
Date: Sun, 9 Mar 2014 00:01:13 +0800
Subject: [PATCH 132/178] refine skip_tutorial
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

refine skip_tutorial
using Chinese ()
---
 app/locale/zh-HANS.coffee | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/locale/zh-HANS.coffee b/app/locale/zh-HANS.coffee
index 6611b070d..21f328154 100644
--- a/app/locale/zh-HANS.coffee
+++ b/app/locale/zh-HANS.coffee
@@ -200,7 +200,7 @@ module.exports = nativeDescription: "简体中文", englishDescription: "Chinese
     tome_available_spells: "可用的法术"
     hud_continue: "继续(按 Shift-空格)"
     spell_saved: "咒语已保存"
-    skip_tutorial: "跳过(esc)"
+    skip_tutorial: "跳过(esc)"
 
   admin:
     av_title: "管理员视图"

From 1c44e3df1eaaa22fe253ec0210e7ee4d48e76055 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sat, 8 Mar 2014 11:37:33 -0800
Subject: [PATCH 133/178] Fixed bug with linking to #my-matches tab.

---
 app/views/play/ladder/play_modal.coffee | 16 ++++++++--------
 app/views/play/ladder_view.coffee       |  4 +++-
 2 files changed, 11 insertions(+), 9 deletions(-)

diff --git a/app/views/play/ladder/play_modal.coffee b/app/views/play/ladder/play_modal.coffee
index e5d83fc3b..ebe51e68d 100644
--- a/app/views/play/ladder/play_modal.coffee
+++ b/app/views/play/ladder/play_modal.coffee
@@ -11,7 +11,7 @@ module.exports = class LadderPlayModal extends View
   closeButton: true
   startsLoading: true
   @shownTutorialButton: false
-  
+
   events:
     'click #skip-tutorial-button': 'hideTutorialButtons'
 
@@ -21,7 +21,7 @@ module.exports = class LadderPlayModal extends View
     @otherTeam = if team is 'ogres' then 'humans' else 'ogres'
     @startLoadingChallengersMaybe()
     @wizardType = ThangType.loadUniversalWizard()
-    
+
   # PART 1: Load challengers from the db unless some are in the matches
 
   startLoadingChallengersMaybe: ->
@@ -49,12 +49,12 @@ module.exports = class LadderPlayModal extends View
       type: 'POST'
       success: success
     })
-    
+
   # PART 3: Make sure wizard is loaded
-  
+
   checkWizardLoaded: ->
     if @wizardType.loaded then @finishRendering() else @wizardType.once 'sync', @finishRendering, @
-    
+
   # PART 4: Render
 
   finishRendering: ->
@@ -69,7 +69,7 @@ module.exports = class LadderPlayModal extends View
     ctx.teamName = _.string.titleize @team
     ctx.teamID = @team
     ctx.otherTeamID = @otherTeam
-    
+
     teamsList = teamDataFromLevel @level
     teams = {}
     teams[team.id] = team for team in teamsList
@@ -104,7 +104,7 @@ module.exports = class LadderPlayModal extends View
     @$el.find('#normal-view').removeClass('secret')
     @$el.find('.modal-header').removeClass('secret')
     @$el.find('#noob-view').addClass('secret')
-    
+
   # Choosing challengers
 
   getChallengers: ->
@@ -165,7 +165,7 @@ class ChallengersData
     @hardPlayer = new LeaderboardCollection(@level, {order:-1, scoreOffset: score + 5, limit: 1, team: @otherTeam})
     @hardPlayer.fetch()
     @hardPlayer.once 'sync', @challengerLoaded, @
-    
+
   challengerLoaded: ->
     if @allLoaded()
       @loaded = true
diff --git a/app/views/play/ladder_view.coffee b/app/views/play/ladder_view.coffee
index 7b6597380..7ed43660e 100644
--- a/app/views/play/ladder_view.coffee
+++ b/app/views/play/ladder_view.coffee
@@ -66,7 +66,9 @@ module.exports = class LadderView extends RootView
     @insertSubView(@ladderTab = new LadderTabView({}, @level, @sessions))
     @insertSubView(@myMatchesTab = new MyMatchesTabView({}, @level, @sessions))
     @refreshInterval = setInterval(@fetchSessionsAndRefreshViews.bind(@), 10000)
-    @showPlayModal(document.location.hash[1..]) if document.location.hash and @sessions.loaded
+    hash = document.location.hash[1..] if document.location.hash
+    unless hash in ['my-matches', 'simulate', 'ladder']
+      @showPlayModal(hash) if @sessions.loaded
 
   fetchSessionsAndRefreshViews: ->
     @sessions.fetch({"success": @refreshViews})

From b0238d74a40693d616cbd40101dcf700a94372b1 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sat, 8 Mar 2014 11:50:10 -0800
Subject: [PATCH 134/178] Let's just show 200 leaderboard entries for now until
 we have paging.

---
 app/views/play/ladder/ladder_tab.coffee | 3 ++-
 1 file changed, 2 insertions(+), 1 deletion(-)

diff --git a/app/views/play/ladder/ladder_tab.coffee b/app/views/play/ladder/ladder_tab.coffee
index 323ca86fe..b8b2d3615 100644
--- a/app/views/play/ladder/ladder_tab.coffee
+++ b/app/views/play/ladder/ladder_tab.coffee
@@ -55,7 +55,8 @@ module.exports = class LadderTabView extends CocoView
 class LeaderboardData
   constructor: (@level, @team, @session) ->
     _.extend @, Backbone.Events
-    @topPlayers = new LeaderboardCollection(@level, {order:-1, scoreOffset: HIGHEST_SCORE, team: @team, limit: if @session then 10 else 20})
+    limit = 200 # if @session then 10 else 20  # We need to figure out paging.
+    @topPlayers = new LeaderboardCollection(@level, {order:-1, scoreOffset: HIGHEST_SCORE, team: @team, limit: limit})
     @topPlayers.fetch()
     @topPlayers.comparator = (model) ->
       return -model.get('totalScore')

From 2861f62248483687088a473e5046f8988cfc9ec3 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Sat, 8 Mar 2014 13:06:04 -0800
Subject: [PATCH 135/178] Simple fix for a race condition where the user schema
 might load after /auth/whoami, overwriting /auth/whoami's cookie.

---
 app/models/CocoModel.coffee | 3 ++-
 1 file changed, 2 insertions(+), 1 deletion(-)

diff --git a/app/models/CocoModel.coffee b/app/models/CocoModel.coffee
index ba4ff850c..919d9a6b9 100644
--- a/app/models/CocoModel.coffee
+++ b/app/models/CocoModel.coffee
@@ -22,7 +22,8 @@ class CocoModel extends Backbone.Model
     if @constructor.schema?.loaded
       @addSchemaDefaults()
     else
-      @loadSchema()
+      {me} = require 'lib/auth'
+      @loadSchema() if me?.loaded
     @once 'sync', @onLoaded, @
     @saveBackup = _.debounce(@saveBackup, 500)
 

From afbe2200a68d84dfcfd39c53046cde74f8f2fe72 Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Sat, 8 Mar 2014 13:52:34 -0800
Subject: [PATCH 136/178] Fixed a bug with the new race condition fix. Moving
 the schemas to the application would be good.

---
 app/models/CocoModel.coffee        | 7 +++----
 app/views/editor/thang/edit.coffee | 1 +
 2 files changed, 4 insertions(+), 4 deletions(-)

diff --git a/app/models/CocoModel.coffee b/app/models/CocoModel.coffee
index 919d9a6b9..3f4c9455c 100644
--- a/app/models/CocoModel.coffee
+++ b/app/models/CocoModel.coffee
@@ -16,6 +16,7 @@ class CocoModel extends Backbone.Model
 
   initialize: ->
     super()
+    @constructor.schema ?= new CocoSchema(@urlRoot)
     if not @constructor.className
       console.error("#{@} needs a className set.")
     @markToRevert()
@@ -52,10 +53,8 @@ class CocoModel extends Backbone.Model
   @backedUp = {}
 
   loadSchema: ->
-    unless @constructor.schema
-      @constructor.schema = new CocoSchema(@urlRoot)
-      @constructor.schema.fetch()
-
+    return if @constructor.schema.loading
+    @constructor.schema.fetch()
     @constructor.schema.once 'sync', =>
       @constructor.schema.loaded = true
       @addSchemaDefaults()
diff --git a/app/views/editor/thang/edit.coffee b/app/views/editor/thang/edit.coffee
index a8310cdf2..833ce3e0e 100644
--- a/app/views/editor/thang/edit.coffee
+++ b/app/views/editor/thang/edit.coffee
@@ -42,6 +42,7 @@ module.exports = class ThangTypeEditView extends View
     @thangType = new ThangType(_id: @thangTypeID)
     @thangType.saveBackups = true
     @thangType.fetch()
+    @thangType.loadSchema()
     @thangType.schema().once 'sync', @onThangTypeSync, @
     @thangType.once 'sync', @onThangTypeSync, @
     @refreshAnimation = _.debounce @refreshAnimation, 500

From 57f1588dea9c12ff869baa9222f202ebb7b6a72f Mon Sep 17 00:00:00 2001
From: Scott Erickson <sderickson@gmail.com>
Date: Sat, 8 Mar 2014 14:34:25 -0800
Subject: [PATCH 137/178] Fixed another bug related to the race condition fix.

---
 app/models/SuperModel.coffee | 11 +++++++++--
 1 file changed, 9 insertions(+), 2 deletions(-)

diff --git a/app/models/SuperModel.coffee b/app/models/SuperModel.coffee
index 36a7fab72..9ce14c75d 100644
--- a/app/models/SuperModel.coffee
+++ b/app/models/SuperModel.coffee
@@ -2,6 +2,7 @@ class SuperModel
   constructor: ->
     @models = {}
     @collections = {}
+    @schemas = {}
     _.extend(@, Backbone.Events)
 
   populateModel: (model) ->
@@ -25,8 +26,11 @@ class SuperModel
     @removeEventsFromModel(model)
 
   modelLoaded: (model) ->
+    model.loadSchema()
     schema = model.schema()
-    return schema.once('sync', => @modelLoaded(model)) unless schema.loaded
+    unless schema.loaded
+      @schemas[schema.urlRoot] = schema
+      return schema.once('sync', => @modelLoaded(model))
     refs = model.getReferencedModels(model.attributes, schema.attributes, '/', @shouldLoadProjection)
     refs = [] unless @mustPopulate is model or @shouldPopulate(model)
 #    console.log 'Loaded', model.get('name')
@@ -96,9 +100,12 @@ class SuperModel
     total = 0
     loaded = 0
 
-    for key, model of @models
+    for model in _.values @models
       total += 1
       loaded += 1 if model.loaded
+    for schema in _.values @schemas
+      total += 1
+      loaded += 1 if schema.loaded
 
     return 1.0 unless total
     return loaded / total

From be12d38e34063496b798830fead91929293c679b Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sat, 8 Mar 2014 14:38:35 -0800
Subject: [PATCH 138/178] Taking into account null rects and library
 width/height subtraction on sprite parsing.

---
 app/lib/sprites/SpriteParser.coffee | 12 ++++++++++++
 1 file changed, 12 insertions(+)

diff --git a/app/lib/sprites/SpriteParser.coffee b/app/lib/sprites/SpriteParser.coffee
index 75288c5fa..a5e1f9472 100644
--- a/app/lib/sprites/SpriteParser.coffee
+++ b/app/lib/sprites/SpriteParser.coffee
@@ -24,6 +24,11 @@ module.exports = class SpriteParser
       @animationLongKeys[longKey] = shortKey
 
   parse: (source) ->
+    # Grab the library properties' width/height so we can subtract half of each from frame bounds
+    properties = source.match(/.*lib\.properties = \{\n.*?width: (\d+),\n.*?height: (\d+)/im)
+    @width = parseInt(properties[1] ? "0", 10)
+    @height = parseInt(properties[2] ? "0", 10)
+
     options = {loc: false, range: true}
     ast = esprima.parse source, options
     blocks = @findBlocks ast, source
@@ -178,11 +183,18 @@ module.exports = class SpriteParser
                 else if arg.type is 'AssignmentExpression'
                   bounds = @grabFunctionArguments argSource.replace('rect=', ''), true
                   lastRect = bounds
+                else if arg.type is 'Literal' and arg.value is null
+                  bounds = [0, 0, 1, 1]  # Let's try this.
                 frameBounds.push bounds
           else
             console.log "Didn't have multiframe bounds for this movie clip!"
             frameBounds = [nominalBounds]
 
+          # Subtract half of width/height parsed from lib.properties
+          for bounds in frameBounds
+            bounds[0] -= @width / 2
+            bounds[1] -= @height / 2
+
           functionExpressions.push {name: name, bounds: nominalBounds, frameBounds: frameBounds, expression: node.parent.parent, kind: kind}
     @walk ast, null, gatherFunctionExpressions
     functionExpressions

From 5bde5347571950730e23f9f1b5b9bbe4a1cb2e34 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sat, 8 Mar 2014 14:53:17 -0800
Subject: [PATCH 139/178] Fixed parsing for ThangTypes that don't have
 lib.properties.

---
 app/lib/sprites/SpriteParser.coffee | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/lib/sprites/SpriteParser.coffee b/app/lib/sprites/SpriteParser.coffee
index a5e1f9472..9dd45dbcc 100644
--- a/app/lib/sprites/SpriteParser.coffee
+++ b/app/lib/sprites/SpriteParser.coffee
@@ -26,8 +26,8 @@ module.exports = class SpriteParser
   parse: (source) ->
     # Grab the library properties' width/height so we can subtract half of each from frame bounds
     properties = source.match(/.*lib\.properties = \{\n.*?width: (\d+),\n.*?height: (\d+)/im)
-    @width = parseInt(properties[1] ? "0", 10)
-    @height = parseInt(properties[2] ? "0", 10)
+    @width = parseInt(properties?[1] ? "0", 10)
+    @height = parseInt(properties?[2] ? "0", 10)
 
     options = {loc: false, range: true}
     ast = esprima.parse source, options

From b3964571e1dd752dd467c3f99f67da27acfad550 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sat, 8 Mar 2014 15:04:11 -0800
Subject: [PATCH 140/178] Fixed bug importing animations into existing
 ThangTypes that had only containers.

---
 app/lib/sprites/SpriteParser.coffee | 5 ++++-
 app/views/editor/thang/edit.coffee  | 2 +-
 2 files changed, 5 insertions(+), 2 deletions(-)

diff --git a/app/lib/sprites/SpriteParser.coffee b/app/lib/sprites/SpriteParser.coffee
index 9dd45dbcc..15fe749cd 100644
--- a/app/lib/sprites/SpriteParser.coffee
+++ b/app/lib/sprites/SpriteParser.coffee
@@ -2,7 +2,10 @@ module.exports = class SpriteParser
   constructor: (@thangTypeModel) ->
     # Create a new ThangType, or work with one we've been building
     @thangType = _.cloneDeep(@thangTypeModel.attributes.raw)
-    @thangType ?= {shapes: {}, containers: {}, animations: {}}
+    @thangType ?= {}
+    @thangType.shapes ?= {}
+    @thangType.containers ?= {}
+    @thangType.animations ?= {}
 
     # Internal parser state
     @shapeLongKeys = {}
diff --git a/app/views/editor/thang/edit.coffee b/app/views/editor/thang/edit.coffee
index 833ce3e0e..5a5d188f2 100644
--- a/app/views/editor/thang/edit.coffee
+++ b/app/views/editor/thang/edit.coffee
@@ -316,7 +316,7 @@ module.exports = class ThangTypeEditView extends View
     @thangType.set 'actions', undefined
     @clearDisplayObject()
     @treema.set('/', @getThangData())
-    
+
   getThangData: ->
     data = _.cloneDeep(@thangType.attributes)
     data = _.pick data, (value, key) => not (key in ['components'])

From 39c465720433f3c6f991c2ec362e69c8f60c7cae Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sat, 8 Mar 2014 15:43:56 -0800
Subject: [PATCH 141/178] Fixed HUD showing actions backwardly. Added rank
 numbering to leaderboard.

---
 app/templates/play/ladder/ladder_tab.jade | 6 ++++--
 app/views/play/level/hud_view.coffee      | 2 +-
 2 files changed, 5 insertions(+), 3 deletions(-)

diff --git a/app/templates/play/ladder/ladder_tab.jade b/app/templates/play/ladder/ladder_tab.jade
index c3090caa3..9fc48c340 100644
--- a/app/templates/play/ladder/ladder_tab.jade
+++ b/app/templates/play/ladder/ladder_tab.jade
@@ -3,17 +3,19 @@ div#columns.row
     div.column.col-md-6
       table.table.table-bordered.table-condensed.table-hover
         tr
-          th(colspan=3, style="color: #{team.primaryColor}")
+          th(colspan=4, style="color: #{team.primaryColor}")
             span= team.name
             span  Leaderboard
         tr
+          th Rank
           th Score
           th.name-col-cell Name
           th
 
-        for session in team.leaderboard.topPlayers.models
+        for session, rank in team.leaderboard.topPlayers.models
           - var myRow = session.get('creator') == me.id
           tr(class=myRow ? "success" : "")
+            td.rank-cell= rank + 1
             td.score-cell= Math.round(session.get('totalScore') * 100)
             td.name-col-cell= session.get('creatorName') || "Anonymous"
             td
diff --git a/app/views/play/level/hud_view.coffee b/app/views/play/level/hud_view.coffee
index 1951e3539..662c9fc8f 100644
--- a/app/views/play/level/hud_view.coffee
+++ b/app/views/play/level/hud_view.coffee
@@ -151,7 +151,7 @@ module.exports = class HUDView extends View
   createActions: ->
     actions = @$el.find('.thang-actions tbody').empty()
     showActions = @thang.world and not _.isEmpty(@thang.actions) and 'action' in @thang.hudProperties ? []
-    @$el.find('.thang-actions').toggleClass 'secret', showActions
+    @$el.find('.thang-actions').toggleClass 'secret', not showActions
     return unless showActions
     @buildActionTimespans()
     for actionName, action of @thang.actions

From bfebd0b47ac4cf7747e710207514d27f6c5e628a Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Sun, 9 Mar 2014 04:13:30 +0400
Subject: [PATCH 142/178] Add revert i18n

---
 app/locale/ru.coffee                   | 2 ++
 app/templates/editor/article/edit.jade | 2 +-
 app/templates/editor/level/edit.jade   | 2 +-
 app/templates/editor/thang/edit.jade   | 2 +-
 app/templates/modal/revert.jade        | 2 +-
 5 files changed, 6 insertions(+), 4 deletions(-)

diff --git a/app/locale/ru.coffee b/app/locale/ru.coffee
index 36bcf1ac1..1cb8f0c53 100644
--- a/app/locale/ru.coffee
+++ b/app/locale/ru.coffee
@@ -226,6 +226,8 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     contact_us: "свяжитесь с нами!"
     hipchat_prefix: "Также вы можете найти нас в нашей"
     hipchat_url: "комнате HipChat."
+    revert: "Откатить"
+    revert_models: "Откатить Модели"
     level_some_options: "Ещё опции"
     level_tab_thangs: "Объекты"
     level_tab_scripts: "Скрипты"
diff --git a/app/templates/editor/article/edit.jade b/app/templates/editor/article/edit.jade
index 7c6a7136a..3bd861bd9 100644
--- a/app/templates/editor/article/edit.jade
+++ b/app/templates/editor/article/edit.jade
@@ -10,7 +10,7 @@ block content
       li.active
         | #{article.attributes.name}
 
-  button(data-toggle="coco-modal", data-target="modal/revert", data-i18n="revert.revert", disabled=authorized === true ? undefined : "true").btn.btn-primary#revert-button Revert
+  button(data-toggle="coco-modal", data-target="modal/revert", data-i18n="editor.revert", disabled=authorized === true ? undefined : "true").btn.btn-primary#revert-button Revert
   button(data-i18n="article.edit_btn_preview", disabled=authorized === true ? undefined : "true").btn.btn-primary#preview-button Preview
   button(data-toggle="coco-modal", data-target="modal/save_version", data-i18n="common.save", disabled=authorized === true ? undefined : "true").btn.btn-primary#save-button Save
 
diff --git a/app/templates/editor/level/edit.jade b/app/templates/editor/level/edit.jade
index 7d2c7ac66..4166c81a1 100644
--- a/app/templates/editor/level/edit.jade
+++ b/app/templates/editor/level/edit.jade
@@ -29,7 +29,7 @@ block outer_content
             
             
           ul.nav.navbar-nav.navbar-right
-            li(data-toggle="coco-modal", data-target="modal/revert", data-i18n="revert.revert", disabled=authorized === true ? undefined : "true").btn.btn-primary.navbar-btn#revert-button Revert
+            li(data-toggle="coco-modal", data-target="modal/revert", data-i18n="editor.revert", disabled=authorized === true ? undefined : "true").btn.btn-primary.navbar-btn#revert-button Revert
             li(data-i18n="common.save", disabled=authorized === true ? undefined : "true").btn.btn-primary.navbar-btn#commit-level-start-button Save
             li(data-i18n="common.fork", disabled=anonymous ? "true": undefined).btn.btn-primary.navbar-btn#fork-level-start-button Fork
             li(title="⌃↩ or ⌘↩: Play preview of current level", data-i18n="common.play")#play-button.btn.btn-inverse.banner.navbar-btn Play!
diff --git a/app/templates/editor/thang/edit.jade b/app/templates/editor/thang/edit.jade
index af30eb2c1..52422fe9a 100644
--- a/app/templates/editor/thang/edit.jade
+++ b/app/templates/editor/thang/edit.jade
@@ -13,7 +13,7 @@ block content
   img#portrait.img-thumbnail
 
   button.btn.btn-primary#save-button(data-toggle="coco-modal", data-target="modal/save_version", disabled=authorized === true ? undefined : "true") Save
-  button.btn.btn-primary#revert-button(data-toggle="coco-modal", data-target="modal/revert", data-i18n="revert.revert", disabled=authorized === true ? undefined : "true") Revert
+  button.btn.btn-primary#revert-button(data-toggle="coco-modal", data-target="modal/revert", data-i18n="editor.revert", disabled=authorized === true ? undefined : "true") Revert
   
   h3 Edit Thang Type: "#{thangType.attributes.name}"
 
diff --git a/app/templates/modal/revert.jade b/app/templates/modal/revert.jade
index adfd7688a..f20edd7d2 100644
--- a/app/templates/modal/revert.jade
+++ b/app/templates/modal/revert.jade
@@ -1,7 +1,7 @@
 extends /templates/modal/modal_base
 
 block modal-header-content
-  h3(data-i18n="revert.revert_models") Revert Models
+  h3(data-i18n="editor.revert_models") Revert Models
 
 block modal-body-content
   table.table.table-striped#changed-models

From f6d68055345f3e2abb2f8eb2920b49a418987814 Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Sun, 9 Mar 2014 04:31:15 +0400
Subject: [PATCH 143/178] Add password i18n

---
 app/locale/ru.coffee            | 1 +
 app/templates/modal/login.jade  | 2 +-
 app/templates/modal/signup.jade | 2 +-
 3 files changed, 3 insertions(+), 2 deletions(-)

diff --git a/app/locale/ru.coffee b/app/locale/ru.coffee
index 1cb8f0c53..55c0f2fbf 100644
--- a/app/locale/ru.coffee
+++ b/app/locale/ru.coffee
@@ -266,6 +266,7 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     description: "Описание"
     or: "или"
     email: "Email"
+    password: "Пароль"
     message: "Сообщение"
 
   about:
diff --git a/app/templates/modal/login.jade b/app/templates/modal/login.jade
index ae6b8b236..bd0307824 100644
--- a/app/templates/modal/login.jade
+++ b/app/templates/modal/login.jade
@@ -9,7 +9,7 @@ block modal-body-content
       label.control-label(for="login-email", data-i18n="general.email") Email      
       input#login-email.input-large.form-control(name="email", type="email")
     .form-group
-      label.control-label(for="login-password", data-i18n="forms.password") Password      
+      label.control-label(for="login-password", data-i18n="general.password") Password      
       input#login-password.input-large.form-control(name="password", type="password")
 
 block modal-body-wait-content
diff --git a/app/templates/modal/signup.jade b/app/templates/modal/signup.jade
index d99fed9d6..2b27577d2 100644
--- a/app/templates/modal/signup.jade
+++ b/app/templates/modal/signup.jade
@@ -12,7 +12,7 @@ block modal-body-content
           label.control-label(for="signup-email", data-i18n="general.email") Email              
           input#signup-email.form-control.input-large(name="email", type="email")
         .form-group
-          label.control-label(for="signup-password", data-i18n="forms.password") Password              
+          label.control-label(for="signup-password", data-i18n="general.password") Password              
           input#signup-password.input-large.form-control(name="password", type="password")
         hr
         .form-group.checkbox

From 81084a24c63a8e8dbc6ebd837f60c2d57087333f Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Sun, 9 Mar 2014 04:36:48 +0400
Subject: [PATCH 144/178] Add admin i18n

---
 app/locale/ru.coffee                | 1 +
 app/templates/account/settings.jade | 2 +-
 2 files changed, 2 insertions(+), 1 deletion(-)

diff --git a/app/locale/ru.coffee b/app/locale/ru.coffee
index 55c0f2fbf..6c52002e5 100644
--- a/app/locale/ru.coffee
+++ b/app/locale/ru.coffee
@@ -122,6 +122,7 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     wizard_tab: "Волшебник"
     password_tab: "Пароль"
     emails_tab: "Email-адреса"
+    admin: "Админ"
     gravatar_select: "Выберите, какое фото с Gravatar использовать"
     gravatar_add_photos: "Чтобы выбрать изображение, добавьте фото и уменьшенные изображения в ваш Gravatar-аккаунт."
     gravatar_add_more_photos: "Добавьте больше фото к вашему аккаунту в Gravatar, чтобы использовать их здесь."
diff --git a/app/templates/account/settings.jade b/app/templates/account/settings.jade
index d64936645..91b533b1b 100644
--- a/app/templates/account/settings.jade
+++ b/app/templates/account/settings.jade
@@ -34,7 +34,7 @@ block content
               input#email.form-control(name="email", type="text", value="#{me.get('email')}")
             if !isProduction
               .form-group.checkbox
-                label(for="email", data-i18n="forms.admin") Admin
+                label(for="email", data-i18n="account_settings.admin") Admin
                 input#admin(name="admin", type="checkbox", checked=me.get('permissions').indexOf('admin')>-1))
               
   

From bcaa801f880cb7004fdbb27f8e33873aced842f0 Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Sun, 9 Mar 2014 05:04:11 +0400
Subject: [PATCH 145/178] Sync en i18n

---
 app/locale/en.coffee | 6 +++++-
 app/locale/ru.coffee | 6 ++++++
 2 files changed, 11 insertions(+), 1 deletion(-)

diff --git a/app/locale/en.coffee b/app/locale/en.coffee
index 5ca5cf902..e3c694f82 100644
--- a/app/locale/en.coffee
+++ b/app/locale/en.coffee
@@ -122,6 +122,7 @@ module.exports = nativeDescription: "English", englishDescription: "English", tr
     wizard_tab: "Wizard"
     password_tab: "Password"
     emails_tab: "Emails"
+    admin: "Admin"
     gravatar_select: "Select which Gravatar photo to use"
     gravatar_add_photos: "Add thumbnails and photos to a Gravatar account for your email to choose an image."
     gravatar_add_more_photos: "Add more photos to your Gravatar account to access them here."
@@ -226,6 +227,8 @@ module.exports = nativeDescription: "English", englishDescription: "English", tr
     contact_us: "contact us!"
     hipchat_prefix: "You can also find us in our"
     hipchat_url: "HipChat room."
+    revert: "Revert"
+    revert_models: "Revert Models"
     level_some_options: "Some Options?"
     level_tab_thangs: "Thangs"
     level_tab_scripts: "Scripts"
@@ -270,6 +273,7 @@ module.exports = nativeDescription: "English", englishDescription: "English", tr
     description: "Description"
     or: "or"
     email: "Email"
+    password: "Password"
     message: "Message"
 
   about:
@@ -407,7 +411,7 @@ module.exports = nativeDescription: "English", englishDescription: "English", tr
     more_about_adventurer: "Learn More About Becoming an Adventurer"
     adventurer_subscribe_desc: "Get emails when there are new levels to test."
     scribe_summary_pref: "CodeCombat is not just going to be a bunch of levels. It will also be a resource of programming knowledge that players can hook into. That way, each Artisan can link to a detailed article that for the player's edification: documentation akin to what the "
-    scribe_summary_sufx: " has built. If you enjoy explaining programming concepts, then this class is for you."
+    scribe_summary_suf: " has built. If you enjoy explaining programming concepts, then this class is for you."
     scribe_introduction_pref: "CodeCombat isn't just going to be a bunch of levels. It will also include a resource for knowledge, a wiki of programming concepts that levels can hook into. That way rather than each Artisan having to describe in detail what a comparison operator is, they can simply link their level to the Article describing them that is already written for the player's edification. Something along the lines of what the "
     scribe_introduction_url_mozilla: "Mozilla Developer Network"
     scribe_introduction_suf: " has built. If your idea of fun is articulating the concepts of programming in Markdown form, then this class might be for you."
diff --git a/app/locale/ru.coffee b/app/locale/ru.coffee
index 6c52002e5..24c129bf9 100644
--- a/app/locale/ru.coffee
+++ b/app/locale/ru.coffee
@@ -251,6 +251,12 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     create_system_title: "Создать новую систему"
     new_component_title: "Создать новый компонент"
     new_component_field_system: "Система"
+    new_article_title: "Создать новую статью"
+    new_thang_title: "Создать новый объект"
+    new_level_title: "Создать новый уровень"
+    article_search_title: "Искать статьи"
+    thang_search_title: "Искать типы объектов"
+    level_search_title: "Искать уровни"
 
   article:
     edit_btn_preview: "Предпросмотр"

From 59318678bac69b6d5b276e7bc0c61cbd58513df2 Mon Sep 17 00:00:00 2001
From: Alexei Nikitin <mr-a1@yandex.ru>
Date: Sun, 9 Mar 2014 05:06:53 +0400
Subject: [PATCH 146/178] Update ru

---
 app/locale/ru.coffee | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/locale/ru.coffee b/app/locale/ru.coffee
index 24c129bf9..0296d95d4 100644
--- a/app/locale/ru.coffee
+++ b/app/locale/ru.coffee
@@ -252,7 +252,7 @@ module.exports = nativeDescription: "русский", englishDescription: "Russi
     new_component_title: "Создать новый компонент"
     new_component_field_system: "Система"
     new_article_title: "Создать новую статью"
-    new_thang_title: "Создать новый объект"
+    new_thang_title: "Создать новый тип объектов"
     new_level_title: "Создать новый уровень"
     article_search_title: "Искать статьи"
     thang_search_title: "Искать типы объектов"

From f05e8432a9eb53bfbeb8ddc62e8910438714783a Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sat, 8 Mar 2014 18:49:09 -0800
Subject: [PATCH 147/178] Starting work on ladder update emails.

---
 server/routes/mail.coffee | 149 +++++++++++++++++++++++++++++++-------
 server/sendwithus.coffee  |   3 +-
 2 files changed, 123 insertions(+), 29 deletions(-)

diff --git a/server/routes/mail.coffee b/server/routes/mail.coffee
index d5306e16a..2e693da66 100644
--- a/server/routes/mail.coffee
+++ b/server/routes/mail.coffee
@@ -4,37 +4,130 @@ User = require '../users/User.coffee'
 errors = require '../commons/errors'
 #request = require 'request'
 config = require '../../server_config'
+LevelSession = require '../levels/sessions/LevelSession.coffee'
+log = require 'winston'
+sendwithus = require '../sendwithus'
 
 #badLog = (text) ->
 #  console.log text
 #  request.post 'http://requestb.in/1brdpaz1', { form: {log: text} }
-  
+
 module.exports.setup = (app) ->
-  app.all config.mail.mailchimpWebhook, (req, res) ->
-    post = req.body
-#    badLog("Got post data: #{JSON.stringify(post, null, '\t')}")
-    
-    unless post.type in ['unsubscribe', 'profile']
-      res.send 'Bad post type'
-      return res.end()
+  app.all config.mail.mailchimpWebhook, handleMailchimpWebHook
+  app.get '/mail/cron/ladder-update', handleLadderUpdate
 
-    unless post.data.email
-      res.send 'No email provided'
-      return res.end()
+handleLadderUpdate = (req, res) ->
+  emailDays = [1, 2, 4, 7, 30]
+  now = new Date()
+  getTimeFromDaysAgo = (daysAgo) ->
+    # 2 hours before the date
+    t = now - (86400 * daysAgo + 2 * 3600) * 1000
+  for daysAgo in emailDays
+    startTime = getTimeFromDaysAgo daysAgo
+    endTime = startTime + 5 * 60 * 1000
+    # Get every session that was submitted in a 5-minute window after the time.
+    findParameters = {submitted: true, submitDate: {$gt: startTime, $lte: endTime}}
+    # TODO: think about putting screenshots in the email
+    selectString = "creator team levelID totalScore matches"
+    query = LevelSession.find(findParameters)
+      .select(selectString)
+      .lean()
+    mongoose = require 'mongoose'
+    mongoose.set 'debug', true
+    query.exec (err, results) ->
+      log.info "Yooooo got results: #{results.length}"
+      if err
+        log.error "Couldn't fetch ladder updates for", findParameters, "\nError: ", err
+        return errors.serverError res, "Ladder update email query failed: #{JSON.stringify(err)}"
+      sendLadderUpdateEmail result, daysAgo for result in results
+      res.send('')
+      res.end()
 
-    query = {'mailChimp.leid':post.data.web_id}
-    User.findOne query, (err, user) ->
+sendLadderUpdateEmail = (session, daysAgo) ->
+  User.findOne({_id: session.creator}).select("name email firstName lastName emailSubscriptions preferredLanguage").lean().exec (err, user) ->
+    if err
+      log.error "Couldn't find user for", session.creator, "from session", session._id
+      return
+    return unless user.email and 'notification' in user.emailSubscriptions
+    name = if user.firstName and user.lastName then "#{user.firstName} #{user.lastName}" else user.name
+    name = "Wizard" if not name or name is "Anoner"
+
+    sendEmail = (defeatContext, victoryContext) ->
+      # TODO: do something with the preferredLanguage?
+      context =
+        email_id: sendwithus.templates.ladder_update_email
+        #recipient:
+        #  address: user.email
+        recipient:
+           address: 'nick@codecombat.com'
+        days_ago: daysAgo
+        name: name
+        wins: session.numberOfWinsAndTies
+        losses: session.numberOfLosses
+        total_score: session.totalScore
+        team: session.team
+        level: id
+        defeat: defeatContext
+        victory: victoryContext
+      sendwithus.api.send context, (err, result) ->
+        log.error "Error sending ladder update email:", err, 'result', result
+
+    defeats = _.filter session.matches, (match) -> match.metrics.rank is 1 and match.opponents[0].metrics.rank is 0
+    victories = _.filter session.matches, (match) -> match.metrics.rank is 0
+    defeat = _.sample defeats
+    victory = _.sample victories
+    urlForMatch = (match) ->
+      "http://codecombat.com/play/ladder/#{session.levelID}?team=#{session.team}&session=#{session._id}&opponent=#{match.opponents[0].sessionID}"
+
+    onFetchedDefeatedOpponent = (err, defeatedOpponent) ->
+      if err
+        log.error "Couldn't find defeateded opponent: #{err}"
+        defeatedOpponent = null
+      victoryContext = {opponent_name: defeatedOpponent?.name ? "Anoner", url: urlForMatch(victory)} if victory
+
+      onFetchedVictoriousOpponent = (err, victoriousOpponent) ->
+        if err
+          log.error "Couldn't find victorious opponent: #{err}"
+          victoriousOpponent = null
+        defeatContext = {opponent_name: victoriousOpponent?.name ? "Anoner", url: urlForMatch(defeat)} if defeat
+        sendEmail defeatContext, victoryContext
+
+      if defeat
+        User.findOne({_id: defeat.opponents[0].userID}).select("name").lean().exec onFetchedVictoriousOpponent
+      else
+        onFetchedVictoriousOpponent null, null
+
+    if victory
+      User.findOne({_id: victory.opponents[0].userID}).select("name").lean().exec onFetchedDefeatedOpponent
+    else
+      onFetchedDefeatedOpponent null, null
+
+
+handleMailchimpWebHook = (req, res) ->
+  post = req.body
+  #badLog("Got post data: #{JSON.stringify(post, null, '\t')}")
+
+  unless post.type in ['unsubscribe', 'profile']
+    res.send 'Bad post type'
+    return res.end()
+
+  unless post.data.email
+    res.send 'No email provided'
+    return res.end()
+
+  query = {'mailChimp.leid':post.data.web_id}
+  User.findOne query, (err, user) ->
+    return errors.serverError(res) if err
+    if not user
+      return errors.notFound(res)
+
+    handleProfileUpdate(user, post) if post.type is 'profile'
+    handleUnsubscribe(user) if post.type is 'unsubscribe'
+
+    user.updatedMailChimp = true # so as not to echo back to mailchimp
+    user.save (err) ->
       return errors.serverError(res) if err
-      if not user
-        return errors.notFound(res)
-
-      handleProfileUpdate(user, post) if post.type is 'profile'
-      handleUnsubscribe(user) if post.type is 'unsubscribe'
-
-      user.updatedMailChimp = true # so as not to echo back to mailchimp
-      user.save (err) ->
-        return errors.serverError(res) if err
-        res.end('Success')
+      res.end('Success')
 
 
 handleProfileUpdate = (user, post) ->
@@ -43,19 +136,19 @@ handleProfileUpdate = (user, post) ->
   otherSubscriptions = (g for g in user.get('emailSubscriptions') when not mail.MAILCHIMP_GROUP_MAP[g])
   groups = groups.concat otherSubscriptions
   user.set 'emailSubscriptions', groups
-  
+
   fname = post.data.merges.FNAME
   user.set('firstName', fname) if fname
 
   lname = post.data.merges.LNAME
   user.set('lastName', lname) if lname
-  
+
   user.set 'mailChimp.email', post.data.email
   user.set 'mailChimp.euid', post.data.id
-  
+
 #  badLog("Updating user object to: #{JSON.stringify(user.toObject(), null, '\t')}")
-    
+
 handleUnsubscribe = (user) ->
   user.set 'emailSubscriptions', []
 
-#  badLog("Unsubscribing user object to: #{JSON.stringify(user.toObject(), null, '\t')}") 
\ No newline at end of file
+#  badLog("Unsubscribing user object to: #{JSON.stringify(user.toObject(), null, '\t')}")
diff --git a/server/sendwithus.coffee b/server/sendwithus.coffee
index a9bb41bf4..c481893bf 100644
--- a/server/sendwithus.coffee
+++ b/server/sendwithus.coffee
@@ -10,4 +10,5 @@ module.exports.setupRoutes = (app) ->
 options = { DEBUG: not config.isProduction }
 module.exports.api = new sendwithusAPI swuAPIKey, options
 module.exports.templates =
-  welcome_email: 'utnGaBHuSU4Hmsi7qrAypU'
\ No newline at end of file
+  welcome_email: 'utnGaBHuSU4Hmsi7qrAypU'
+  ladder_update_email: 'Xq3vSbDHXcjXfje7n2e7Eb'

From ed93b2bbe56ceab2ba2747fe228bd2a862de352f Mon Sep 17 00:00:00 2001
From: Darredevil <alex.darredevil@gmail.com>
Date: Sun, 9 Mar 2014 05:53:22 +0200
Subject: [PATCH 148/178] More than half done

---
 app/locale/ro.coffee | 110 +++++++++++++++++++++----------------------
 1 file changed, 55 insertions(+), 55 deletions(-)

diff --git a/app/locale/ro.coffee b/app/locale/ro.coffee
index 1fee5fe47..e9d7e3db6 100644
--- a/app/locale/ro.coffee
+++ b/app/locale/ro.coffee
@@ -219,64 +219,64 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
     thang_title: "Editor Thang"
     thang_description: "Construiește unități ,definește logica lor,grafica și sunetul.Momentan suportă numai importare de grafică vectorială exportată din Flash."
     level_title: "Editor Nivele"
-#    level_description: "Includes the tools for scripting, uploading audio, and constructing custom logic to create all sorts of levels. Everything we use ourselves!"
-#    security_notice: "Many major features in these editors are not currently enabled by default. As we improve the security of these systems, they will be made generally available. If you'd like to use these features sooner, "
-#    contact_us: "contact us!"
-#    hipchat_prefix: "You can also find us in our"
-#    hipchat_url: "HipChat room."
-#    level_some_options: "Some Options?"
-#    level_tab_thangs: "Thangs"
-#    level_tab_scripts: "Scripts"
-#    level_tab_settings: "Settings"
-#    level_tab_components: "Components"
-#    level_tab_systems: "Systems"
-#    level_tab_thangs_title: "Current Thangs"
-#    level_tab_thangs_conditions: "Starting Conditions"
-#    level_tab_thangs_add: "Add Thangs"
-#    level_settings_title: "Settings"
-#    level_component_tab_title: "Current Components"
-#    level_component_btn_new: "Create New Component"
-#    level_systems_tab_title: "Current Systems"
-#    level_systems_btn_new: "Create New System"
-#    level_systems_btn_add: "Add System"
-#    level_components_title: "Back to All Thangs"
-#    level_components_type: "Type"
-#    level_component_edit_title: "Edit Component"
-#    level_system_edit_title: "Edit System"
-#    create_system_title: "Create New System"
-#    new_component_title: "Create New Component"
-#    new_component_field_system: "System"
+    level_description: "Include uneltele pentru scriptare, upload audio, și construcție de logică costum pentru toate tipurile de nivele.Tot ce folosim noi înșine!"
+    security_notice: "Multe setări majore de securitate în aceste editoare nu sunt momentan disponibile.Pe măsură ce îmbunătățim securitatea acestor sisteme, ele vor deveni disponibile. Dacă doriți să folosiți aceste setări mai devrme, "
+    contact_us: "contactați-ne!"
+    hipchat_prefix: "Ne puteți de asemenea găsi la"
+    hipchat_url: "HipChat."
+    level_some_options: "Opțiuni?"
+    level_tab_thangs: "Thangs"
+    level_tab_scripts: "Script-uri"
+    level_tab_settings: "Setări"
+    level_tab_components: "Componente"
+    level_tab_systems: "Sisteme"
+    level_tab_thangs_title: "Thangs actuali"
+    level_tab_thangs_conditions: "Condiți inițiale"
+    level_tab_thangs_add: "Adaugă Thangs"
+    level_settings_title: "Setări"
+    level_component_tab_title: "Componente actuale"
+    level_component_btn_new: "Crează componentă nouă"
+    level_systems_tab_title: "Sisteme actuale"
+    level_systems_btn_new: "Crează sistem nou"
+    level_systems_btn_add: "Adaugă Sistem"
+    level_components_title: "Înapoi la toți Thangs"
+    level_components_type: "Tip"
+    level_component_edit_title: "Editează Componenta"
+    level_system_edit_title: "Editează Sistem"
+    create_system_title: "Crează sistem nou"
+    new_component_title: "Crează componentă nouă"
+    new_component_field_system: "Sistem"
 
-#  article:
-#    edit_btn_preview: "Preview"
-#    edit_article_title: "Edit Article"
+  article:
+    edit_btn_preview: "Preview"
+    edit_article_title: "Editează Articol"
 
-#  general:
-#    and: "and"
-#    name: "Name"
-#    body: "Body"
-#    version: "Version"
-#    commit_msg: "Commit Message"
-#    version_history_for: "Version History for: "
-#    results: "Results"
-#    description: "Description"
-#    or: "or"
-#    email: "Email"
-#    message: "Message"
+  general:
+    and: "și"
+    name: "Nume"
+    body: "Corp"
+    version: "Versiune"
+    commit_msg: "Înregistrează Mesajul"
+    version_history_for: "Versiune istorie pentru: "
+    results: "Resultate"
+    description: "Descriere"
+    or: "sau"
+    email: "Email"
+    message: "Mesaj"
 
-#  about:
-#    who_is_codecombat: "Who is CodeCombat?"
-#    why_codecombat: "Why CodeCombat?"
-#    who_description_prefix: "together started CodeCombat in 2013. We also created "
-#    who_description_suffix: "in 2008, growing it to the #1 web and iOS application for learning to write Chinese and Japanese characters."
-#    who_description_ending: "Now it's time to teach people to write code."
-#    why_paragraph_1: "When making Skritter, George didn't know how to program and was constantly frustrated by his inability to implement his ideas. Afterwards, he tried learning, but the lessons were too slow. His housemate, wanting to reskill and stop teaching, tried Codecademy, but \"got bored.\" Each week another friend started Codecademy, then dropped off. We realized it was the same problem we'd solved with Skritter: people learning a skill via slow, intensive lessons when what they need is fast, extensive practice. We know how to fix that."
-#    why_paragraph_2: "Need to learn to code? You don't need lessons. You need to write a lot of code and have a great time doing it."
-#    why_paragraph_3_prefix: "That's what programming is about. It's gotta be fun. Not fun like"
-#    why_paragraph_3_italic: "yay a badge"
-#    why_paragraph_3_center: "but fun like"
-#    why_paragraph_3_italic_caps: "NO MOM I HAVE TO FINISH THE LEVEL!"
-#    why_paragraph_3_suffix: "That's why CodeCombat is a multiplayer game, not a gamified lesson course. We won't stop until you can't stop--but this time, that's a good thing."
+  about:
+    who_is_codecombat: "Cine este CodeCombat?"  # I assume you meant (what)
+    why_codecombat: "De ce CodeCombat?"
+    who_description_prefix: "au pornit împreuna CodeCombat în 2013. Tot noi am creat "
+    who_description_suffix: "în 2008, dezvoltând aplicația web si iOS #1 de învățat cum să scri caractere Japoneze si Chinezești."
+    who_description_ending: "Acum este timpul să învățăm oamenii să scrie cod."
+    why_paragraph_1: "Când am dezolvat Skritter, George nu știa cum să programeze și era mereu frustat de inabilitatea sa de a putea implementa ideile sale. După aceea, a încercat să învețe, dar lecțiile erau prea lente. Colegul său , vrând să se reprofilze și să se lase de predat,a încercat Codecademy, dar \"s-a plictisit.\" În fiecare săptămână un alt prieten a început Codecademy, iar apoi s-a lăsat. Am realizat că este aceeași problemă care am rezolvat-u cu Skritter: oameni încercând să învețe ceva nou prin lecții lente și intensive când defapt ceea ce le trebuie sunt lecții rapide și multă practică. Noi știm cum să rezolvăm asta."
+    why_paragraph_2: "Trebuie să înveți să programezi? Nu-ți trebuie lecții. Trebuie să scri mult cod și să te distrezi făcând asta."
+    why_paragraph_3_prefix: "Despre asta este programarea. Trebuie să fie distractiv. Nu precum"
+    why_paragraph_3_italic: "wow o insignă"
+    why_paragraph_3_center: "ci"
+    why_paragraph_3_italic_caps: "TREBUIE SĂ TERMIN ACEST NIVEL!"
+    why_paragraph_3_suffix: "De aceea CodeCombat este un joc multiplayer, nu un curs transfigurat în joc. Nu ne vom opri până când tu nu te poți opri--și de data asta, e de bine."
 #    why_paragraph_4: "If you're going to get addicted to some game, get addicted to this one and become one of the wizards of the tech age."
 #    why_ending: "And hey, it's free. "
 #    why_ending_url: "Start wizarding now!"

From 82857fd0809f4201b42d66678ea2b1d3967b9a60 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sat, 8 Mar 2014 21:29:41 -0800
Subject: [PATCH 149/178] Was able to send out some emails.

---
 server/routes/mail.coffee | 60 +++++++++++++++++++++------------------
 1 file changed, 32 insertions(+), 28 deletions(-)

diff --git a/server/routes/mail.coffee b/server/routes/mail.coffee
index 2e693da66..ed6b58795 100644
--- a/server/routes/mail.coffee
+++ b/server/routes/mail.coffee
@@ -17,31 +17,31 @@ module.exports.setup = (app) ->
   app.get '/mail/cron/ladder-update', handleLadderUpdate
 
 handleLadderUpdate = (req, res) ->
+  res.send('Great work, Captain Cron! I can take it from here.')
+  res.end()
   emailDays = [1, 2, 4, 7, 30]
   now = new Date()
   getTimeFromDaysAgo = (daysAgo) ->
     # 2 hours before the date
     t = now - (86400 * daysAgo + 2 * 3600) * 1000
   for daysAgo in emailDays
+    # Get every session that was submitted in a 5-minute window after the time.
     startTime = getTimeFromDaysAgo daysAgo
     endTime = startTime + 5 * 60 * 1000
-    # Get every session that was submitted in a 5-minute window after the time.
-    findParameters = {submitted: true, submitDate: {$gt: startTime, $lte: endTime}}
+    #endTime = startTime + 1 * 60 * 60 * 1000
+    findParameters = {submitted: true, submitDate: {$gt: new Date(startTime), $lte: new Date(endTime)}}
     # TODO: think about putting screenshots in the email
-    selectString = "creator team levelID totalScore matches"
+    selectString = "creator team levelName levelID totalScore matches submitted submitDate numberOfWinsAndTies numberOfLosses"
     query = LevelSession.find(findParameters)
       .select(selectString)
       .lean()
     mongoose = require 'mongoose'
-    mongoose.set 'debug', true
-    query.exec (err, results) ->
-      log.info "Yooooo got results: #{results.length}"
-      if err
-        log.error "Couldn't fetch ladder updates for", findParameters, "\nError: ", err
-        return errors.serverError res, "Ladder update email query failed: #{JSON.stringify(err)}"
-      sendLadderUpdateEmail result, daysAgo for result in results
-      res.send('')
-      res.end()
+    do (daysAgo) ->
+      query.exec (err, results) ->
+        if err
+          log.error "Couldn't fetch ladder updates for", findParameters, "\nError: ", err
+          return errors.serverError res, "Ladder update email query failed: #{JSON.stringify(err)}"
+        sendLadderUpdateEmail result, daysAgo for result in results
 
 sendLadderUpdateEmail = (session, daysAgo) ->
   User.findOne({_id: session.creator}).select("name email firstName lastName emailSubscriptions preferredLanguage").lean().exec (err, user) ->
@@ -56,28 +56,32 @@ sendLadderUpdateEmail = (session, daysAgo) ->
       # TODO: do something with the preferredLanguage?
       context =
         email_id: sendwithus.templates.ladder_update_email
-        #recipient:
-        #  address: user.email
         recipient:
-           address: 'nick@codecombat.com'
-        days_ago: daysAgo
-        name: name
-        wins: session.numberOfWinsAndTies
-        losses: session.numberOfLosses
-        total_score: session.totalScore
-        team: session.team
-        level: id
-        defeat: defeatContext
-        victory: victoryContext
+          #address: user.email
+          address: 'nick@codecombat.com'
+          name: name
+        email_data:
+          name: name
+          days_ago: daysAgo
+          wins: session.numberOfWinsAndTies
+          losses: session.numberOfLosses
+          total_score: Math.round(session.totalScore * 100)
+          team: session.team
+          level_name: session.levelName
+          ladder_url: "http://codecombat.com/play/ladder/#{session.levelID}#my-matches"
+          defeat: defeatContext
+          victory: victoryContext
       sendwithus.api.send context, (err, result) ->
-        log.error "Error sending ladder update email:", err, 'result', result
+        log.error "Error sending ladder update email:", err, 'result', result if err
 
+    # Fetch the most recent defeat and victory, if there are any.
+    # (We could look at strongest/weakest, but we'd have to fetch everyone, or denormalize more.)
     defeats = _.filter session.matches, (match) -> match.metrics.rank is 1 and match.opponents[0].metrics.rank is 0
     victories = _.filter session.matches, (match) -> match.metrics.rank is 0
-    defeat = _.sample defeats
-    victory = _.sample victories
+    defeat = _.last defeats
+    victory = _.last victories
     urlForMatch = (match) ->
-      "http://codecombat.com/play/ladder/#{session.levelID}?team=#{session.team}&session=#{session._id}&opponent=#{match.opponents[0].sessionID}"
+      "http://codecombat.com/play/level/#{session.levelID}?team=#{session.team}&session=#{session._id}&opponent=#{match.opponents[0].sessionID}"
 
     onFetchedDefeatedOpponent = (err, defeatedOpponent) ->
       if err

From af2b43a3097514241a73a67b418f793c1d3c2f55 Mon Sep 17 00:00:00 2001
From: Darredevil <alex.darredevil@gmail.com>
Date: Sun, 9 Mar 2014 15:09:22 +0200
Subject: [PATCH 150/178] Changed description of "Get to Locations"

---
 server/levels/level_schema.coffee | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/server/levels/level_schema.coffee b/server/levels/level_schema.coffee
index cbfb84497..86f774ad0 100644
--- a/server/levels/level_schema.coffee
+++ b/server/levels/level_schema.coffee
@@ -33,7 +33,7 @@ GoalSchema = c.object {title: "Goal", description: "A goal that the player can a
   team: c.shortString(title: 'Team', description: 'Name of the team this goal is for, if it is not for all of the playable teams.')
   killThangs: c.array {title: "Kill Thangs", description: "A list of Thang IDs the player should kill, or team names.", uniqueItems: true, minItems: 1, "default": ["ogres"]}, thang
   saveThangs: c.array {title: "Save Thangs", description: "A list of Thang IDs the player should save, or team names", uniqueItems: true, minItems: 1, "default": ["humans"]}, thang
-  getToLocations: c.object {title: "Get To Locations", description: "TODO: explain", required: ["who", "targets"]},
+  getToLocations: c.object {title: "Get To Locations", description: "Will be set off when any of the \"who\" touch any of the \"targets\" ", required: ["who", "targets"]},
     who: c.array {title: "Who", description: "The Thangs who must get to the target locations.", minItems: 1}, thang
     targets: c.array {title: "Targets", description: "The target locations to which the Thangs must get.", minItems: 1}, thang
   keepFromLocations: c.object {title: "Keep From Locations", description: "TODO: explain", required: ["who", "targets"]},

From adc8c8eddf72711ede377be47c8e636feaeba122 Mon Sep 17 00:00:00 2001
From: gorodsb <gorodsb@gmail.com>
Date: Sun, 9 Mar 2014 16:47:48 +0200
Subject: [PATCH 151/178] Update uk.coffee

---
 app/locale/uk.coffee | 92 ++++++++++++++++++++++----------------------
 1 file changed, 46 insertions(+), 46 deletions(-)

diff --git a/app/locale/uk.coffee b/app/locale/uk.coffee
index 52d4fd3b4..4dede7181 100644
--- a/app/locale/uk.coffee
+++ b/app/locale/uk.coffee
@@ -1,16 +1,16 @@
 module.exports = nativeDescription: "українська мова", englishDescription: "Ukranian", translation:
   common:
     loading: "Завантаження..."
-#    saving: "Saving..."
-#    sending: "Sending..."
-#    cancel: "Cancel"
-#    save: "Save"
-#    delay_1_sec: "1 second"
-#    delay_3_sec: "3 seconds"
-#    delay_5_sec: "5 seconds"
-#    manual: "Manual"
-#    fork: "Fork"
-#    play: "Play"
+#    saving: "Збереження..."
+#    sending: "Відправлення..."
+#    cancel: "Відміна"
+#    save: "Зберегти"
+#    delay_1_sec: "1 секунда"
+#    delay_3_sec: "3 секунди"
+#    delay_5_sec: "5 секунд"
+#    manual: "Інструкція"
+#    fork: "Форк"
+#    play: "Грати"
 
   modal:
     close: "Закрити"
@@ -31,28 +31,28 @@ module.exports = nativeDescription: "українська мова", englishDesc
     about: "Про нас"
     contact: "Контакти"
     twitter_follow: "Фоловити"
-#    employers: "Employers"
+#    employers: "Зайняті"
 
 #  versions:
-#    save_version_title: "Save New Version"
-#    new_major_version: "New Major Version"
-#    cla_prefix: "To save changes, first you must agree to our"
+#    save_version_title: "Зберегти нову версію"
+#    new_major_version: "Зберегти основну версію"
+#    cla_prefix: "Для збереження змін, спочатку треба погодитись з нашим"
 #    cla_url: "CLA"
 #    cla_suffix: "."
-#    cla_agree: "I AGREE"
+#    cla_agree: "Я Згоден"
 
   login:
     sign_up: "створити акаунт"
     log_in: "Увійти"
-#    log_out: "Log Out"
+#    log_out: "Вийти"
     recover: "відновити акаунт"
 
 #  recover:
-#    recover_account_title: "Recover Account"
-#    send_password: "Send Recovery Password"
+#    recover_account_title: "Відновити акаунт"
+#    send_password: "Вислати пароль відновлення"
 
   signup:
-#    create_account_title: "Create Account to Save Progress"
+#    create_account_title: "Створити акаунт, щоб зберегти прогрес"
     description: "Це безкоштовно. Просто зробіть кілька простих кроків, щоб бути готовим до гри:"
     email_announcements: "Отримувати анонси на email"
     coppa: "Ви старші 13 років або живете не в США"
@@ -102,13 +102,13 @@ module.exports = nativeDescription: "українська мова", englishDesc
     subscribe_as_diplomat: "Записатися у Дипломати"
 
 #  wizard_settings:
-#    title: "Wizard Settings"
-#    customize_avatar: "Customize Your Avatar"
-#    clothes: "Clothes"
+#    title: "Налаштування"
+#    customize_avatar: "Налаштувати аватар"
+#    clothes: "Одяг"
 #    trim: "Trim"
 #    cloud: "Cloud"
 #    spell: "Spell"
-#    boots: "Boots"
+#    boots: "Черевики"
 #    hue: "Hue"
 #    saturation: "Saturation"
 #    lightness: "Lightness"
@@ -130,7 +130,7 @@ module.exports = nativeDescription: "українська мова", englishDesc
     new_password_verify: "Підтвердження паролю"
     email_subscriptions: "Email-підписки"
     email_announcements: "Оголошення"
-#    email_notifications_description: "Get periodic notifications for your account."
+#    email_notifications_description: "Отримувати періодичні нагадування для Вашого акаунта."
     email_announcements_description: "Отримувати електронні листи про останні новини CodeCombat."
     contributor_emails: "Підписки за класами учасників"
     contribute_prefix: "Нам потрібні люди, які приєднаються до нашої команди! Зайдіть на "
@@ -199,18 +199,18 @@ module.exports = nativeDescription: "українська мова", englishDesc
     tome_select_a_thang: "Оберіть когось для "
     tome_available_spells: "Доступні закляття"
     hud_continue: "Продовжити (натисніть shift-space)"
-#    spell_saved: "Spell Saved"
-#    skip_tutorial: "Skip (esc)"
+#    spell_saved: "Заклинання збережено"
+#    skip_tutorial: "Пропустити (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
 #    av_entities_sub_title: "Entities"
-#    av_entities_users_url: "Users"
+#    av_entities_users_url: "користувачы"
 #    av_entities_active_instances_url: "Active Instances"
-#    av_other_sub_title: "Other"
+#    av_other_sub_title: "Ынше"
 #    av_other_debug_base_url: "Base (for debugging base.jade)"
-#    u_title: "User List"
-#    lg_title: "Latest Games"
+#    u_title: "Список користувачів"
+#    lg_title: "Останні ігри"
 
 #  editor:
 #    main_title: "CodeCombat Editors"
@@ -222,8 +222,8 @@ module.exports = nativeDescription: "українська мова", englishDesc
 #    level_title: "Level Editor"
 #    level_description: "Includes the tools for scripting, uploading audio, and constructing custom logic to create all sorts of levels. Everything we use ourselves!"
 #    security_notice: "Many major features in these editors are not currently enabled by default. As we improve the security of these systems, they will be made generally available. If you'd like to use these features sooner, "
-#    contact_us: "contact us!"
-#    hipchat_prefix: "You can also find us in our"
+#    contact_us: "Зв’язатися з нами!"
+#    hipchat_prefix: "Ви можете також знайти нас в нашому"
 #    hipchat_url: "HipChat room."
 #    level_some_options: "Some Options?"
 #    level_tab_thangs: "Thangs"
@@ -253,30 +253,30 @@ module.exports = nativeDescription: "українська мова", englishDesc
 #    edit_article_title: "Edit Article"
 
 #  general:
-#    and: "and"
-#    name: "Name"
+#    and: "та"
+#    name: "Ім’я"
 #    body: "Body"
-#    version: "Version"
+#    version: "Версія"
 #    commit_msg: "Commit Message"
 #    version_history_for: "Version History for: "
-#    results: "Results"
-#    description: "Description"
-#    or: "or"
+#    results: "Результати"
+#    description: "Опис"
+#    or: "чи"
 #    email: "Email"
-#    message: "Message"
+#    message: "Повідомлення"
 
 #  about:
-#    who_is_codecombat: "Who is CodeCombat?"
-#    why_codecombat: "Why CodeCombat?"
-#    who_description_prefix: "together started CodeCombat in 2013. We also created "
+#    who_is_codecombat: "Хто є CodeCombat?"
+#    why_codecombat: "Чому CodeCombat?"
+#    who_description_prefix: "Взагалом розпочався CodeCombat у 2013. Ми також створили "
 #    who_description_suffix: "in 2008, growing it to the #1 web and iOS application for learning to write Chinese and Japanese characters."
-#    who_description_ending: "Now it's time to teach people to write code."
+#    who_description_ending: "Зараз час вчити людей аисати код."
 #    why_paragraph_1: "When making Skritter, George didn't know how to program and was constantly frustrated by his inability to implement his ideas. Afterwards, he tried learning, but the lessons were too slow. His housemate, wanting to reskill and stop teaching, tried Codecademy, but \"got bored.\" Each week another friend started Codecademy, then dropped off. We realized it was the same problem we'd solved with Skritter: people learning a skill via slow, intensive lessons when what they need is fast, extensive practice. We know how to fix that."
 #    why_paragraph_2: "Need to learn to code? You don't need lessons. You need to write a lot of code and have a great time doing it."
 #    why_paragraph_3_prefix: "That's what programming is about. It's gotta be fun. Not fun like"
 #    why_paragraph_3_italic: "yay a badge"
 #    why_paragraph_3_center: "but fun like"
-#    why_paragraph_3_italic_caps: "NO MOM I HAVE TO FINISH THE LEVEL!"
+#    why_paragraph_3_italic_caps: "НІ, МАМО, Я МУШУ ПРОЙТИ РІВЕНЬ!"
 #    why_paragraph_3_suffix: "That's why CodeCombat is a multiplayer game, not a gamified lesson course. We won't stop until you can't stop--but this time, that's a good thing."
 #    why_paragraph_4: "If you're going to get addicted to some game, get addicted to this one and become one of the wizards of the tech age."
 #    why_ending: "And hey, it's free. "
@@ -288,10 +288,10 @@ module.exports = nativeDescription: "українська мова", englishDesc
 #    michael_description: "Programmer, sys-admin, and undergrad technical wunderkind, Michael is the person keeping our servers online."
 
 #  legal:
-#    page_title: "Legal"
+#    page_title: "Юридичні нотатки"
 #    opensource_intro: "CodeCombat is free to play and completely open source."
 #    opensource_description_prefix: "Check out "
-#    github_url: "our GitHub"
+#    github_url: "наш GitHub"
 #    opensource_description_center: "and help out if you like! CodeCombat is built on dozens of open source projects, and we love them. See "
 #    archmage_wiki_url: "our Archmage wiki"
 #    opensource_description_suffix: "for a list of the software that makes this game possible."

From 6be9ff2bb7933ecae7800c497f5704f718368d25 Mon Sep 17 00:00:00 2001
From: gorodsb <gorodsb@users.noreply.github.com>
Date: Sun, 9 Mar 2014 17:54:08 +0200
Subject: [PATCH 152/178] Update uk.coffee

---
 app/locale/uk.coffee | 94 ++++++++++++++++++++++----------------------
 1 file changed, 47 insertions(+), 47 deletions(-)

diff --git a/app/locale/uk.coffee b/app/locale/uk.coffee
index 4dede7181..b7e1bcf27 100644
--- a/app/locale/uk.coffee
+++ b/app/locale/uk.coffee
@@ -1,16 +1,16 @@
 module.exports = nativeDescription: "українська мова", englishDescription: "Ukranian", translation:
   common:
     loading: "Завантаження..."
-#    saving: "Збереження..."
-#    sending: "Відправлення..."
-#    cancel: "Відміна"
-#    save: "Зберегти"
-#    delay_1_sec: "1 секунда"
-#    delay_3_sec: "3 секунди"
-#    delay_5_sec: "5 секунд"
-#    manual: "Інструкція"
-#    fork: "Форк"
-#    play: "Грати"
+    saving: "Збереження..."
+    sending: "Відправлення..."
+    cancel: "Відміна"
+    save: "Зберегти"
+    delay_1_sec: "1 секунда"
+    delay_3_sec: "3 секунди"
+    delay_5_sec: "5 секунд"
+    manual: "Інструкція"
+    fork: "Форк"
+    play: "Грати"
 
   modal:
     close: "Закрити"
@@ -31,28 +31,28 @@ module.exports = nativeDescription: "українська мова", englishDesc
     about: "Про нас"
     contact: "Контакти"
     twitter_follow: "Фоловити"
-#    employers: "Зайняті"
+    employers: "Зайняті"
 
 #  versions:
-#    save_version_title: "Зберегти нову версію"
-#    new_major_version: "Зберегти основну версію"
-#    cla_prefix: "Для збереження змін, спочатку треба погодитись з нашим"
+    save_version_title: "Зберегти нову версію"
+    new_major_version: "Зберегти основну версію"
+    cla_prefix: "Для збереження змін, спочатку треба погодитись з нашим"
 #    cla_url: "CLA"
 #    cla_suffix: "."
-#    cla_agree: "Я Згоден"
+    cla_agree: "Я Згоден"
 
   login:
     sign_up: "створити акаунт"
     log_in: "Увійти"
-#    log_out: "Вийти"
+    log_out: "Вийти"
     recover: "відновити акаунт"
 
 #  recover:
-#    recover_account_title: "Відновити акаунт"
-#    send_password: "Вислати пароль відновлення"
+    recover_account_title: "Відновити акаунт"
+    send_password: "Вислати пароль відновлення"
 
   signup:
-#    create_account_title: "Створити акаунт, щоб зберегти прогрес"
+    create_account_title: "Створити акаунт, щоб зберегти прогрес"
     description: "Це безкоштовно. Просто зробіть кілька простих кроків, щоб бути готовим до гри:"
     email_announcements: "Отримувати анонси на email"
     coppa: "Ви старші 13 років або живете не в США"
@@ -102,13 +102,13 @@ module.exports = nativeDescription: "українська мова", englishDesc
     subscribe_as_diplomat: "Записатися у Дипломати"
 
 #  wizard_settings:
-#    title: "Налаштування"
-#    customize_avatar: "Налаштувати аватар"
-#    clothes: "Одяг"
+    title: "Налаштування"
+    customize_avatar: "Налаштувати аватар"
+    clothes: "Одяг"
 #    trim: "Trim"
 #    cloud: "Cloud"
-#    spell: "Spell"
-#    boots: "Черевики"
+    spell: "аклинанняЗ"
+    boots: "Черевики"
 #    hue: "Hue"
 #    saturation: "Saturation"
 #    lightness: "Lightness"
@@ -130,7 +130,7 @@ module.exports = nativeDescription: "українська мова", englishDesc
     new_password_verify: "Підтвердження паролю"
     email_subscriptions: "Email-підписки"
     email_announcements: "Оголошення"
-#    email_notifications_description: "Отримувати періодичні нагадування для Вашого акаунта."
+    email_notifications_description: "Отримувати періодичні нагадування для Вашого акаунта."
     email_announcements_description: "Отримувати електронні листи про останні новини CodeCombat."
     contributor_emails: "Підписки за класами учасників"
     contribute_prefix: "Нам потрібні люди, які приєднаються до нашої команди! Зайдіть на "
@@ -199,18 +199,18 @@ module.exports = nativeDescription: "українська мова", englishDesc
     tome_select_a_thang: "Оберіть когось для "
     tome_available_spells: "Доступні закляття"
     hud_continue: "Продовжити (натисніть shift-space)"
-#    spell_saved: "Заклинання збережено"
-#    skip_tutorial: "Пропустити (esc)"
+    spell_saved: "Заклинання збережено"
+    skip_tutorial: "Пропустити (esc)"
 
 #  admin:
 #    av_title: "Admin Views"
 #    av_entities_sub_title: "Entities"
-#    av_entities_users_url: "користувачы"
+    av_entities_users_url: "користувачі"
 #    av_entities_active_instances_url: "Active Instances"
 #    av_other_sub_title: "Ынше"
 #    av_other_debug_base_url: "Base (for debugging base.jade)"
-#    u_title: "Список користувачів"
-#    lg_title: "Останні ігри"
+    u_title: "Список користувачів"
+    lg_title: "Останні ігри"
 
 #  editor:
 #    main_title: "CodeCombat Editors"
@@ -222,8 +222,8 @@ module.exports = nativeDescription: "українська мова", englishDesc
 #    level_title: "Level Editor"
 #    level_description: "Includes the tools for scripting, uploading audio, and constructing custom logic to create all sorts of levels. Everything we use ourselves!"
 #    security_notice: "Many major features in these editors are not currently enabled by default. As we improve the security of these systems, they will be made generally available. If you'd like to use these features sooner, "
-#    contact_us: "Зв’язатися з нами!"
-#    hipchat_prefix: "Ви можете також знайти нас в нашому"
+    contact_us: "Зв’язатися з нами!"
+    hipchat_prefix: "Ви можете також знайти нас в нашому"
 #    hipchat_url: "HipChat room."
 #    level_some_options: "Some Options?"
 #    level_tab_thangs: "Thangs"
@@ -253,30 +253,30 @@ module.exports = nativeDescription: "українська мова", englishDesc
 #    edit_article_title: "Edit Article"
 
 #  general:
-#    and: "та"
-#    name: "Ім’я"
+    and: "та"
+    name: "Ім’я"
 #    body: "Body"
-#    version: "Версія"
+    version: "Версія"
 #    commit_msg: "Commit Message"
 #    version_history_for: "Version History for: "
-#    results: "Результати"
-#    description: "Опис"
-#    or: "чи"
-#    email: "Email"
-#    message: "Повідомлення"
+    results: "Результати"
+    description: "Опис"
+    or: "чи"
+    email: "Email"
+    message: "Повідомлення"
 
 #  about:
-#    who_is_codecombat: "Хто є CodeCombat?"
-#    why_codecombat: "Чому CodeCombat?"
-#    who_description_prefix: "Взагалом розпочався CodeCombat у 2013. Ми також створили "
+    who_is_codecombat: "Хто є CodeCombat?"
+    why_codecombat: "Чому CodeCombat?"
+    who_description_prefix: "Взагалом розпочався CodeCombat у 2013. Ми також створили "
 #    who_description_suffix: "in 2008, growing it to the #1 web and iOS application for learning to write Chinese and Japanese characters."
-#    who_description_ending: "Зараз час вчити людей аисати код."
+    who_description_ending: "Зараз час вчити людей писати код."
 #    why_paragraph_1: "When making Skritter, George didn't know how to program and was constantly frustrated by his inability to implement his ideas. Afterwards, he tried learning, but the lessons were too slow. His housemate, wanting to reskill and stop teaching, tried Codecademy, but \"got bored.\" Each week another friend started Codecademy, then dropped off. We realized it was the same problem we'd solved with Skritter: people learning a skill via slow, intensive lessons when what they need is fast, extensive practice. We know how to fix that."
 #    why_paragraph_2: "Need to learn to code? You don't need lessons. You need to write a lot of code and have a great time doing it."
 #    why_paragraph_3_prefix: "That's what programming is about. It's gotta be fun. Not fun like"
 #    why_paragraph_3_italic: "yay a badge"
 #    why_paragraph_3_center: "but fun like"
-#    why_paragraph_3_italic_caps: "НІ, МАМО, Я МУШУ ПРОЙТИ РІВЕНЬ!"
+    why_paragraph_3_italic_caps: "НІ, МАМО, Я МАЮ ПРОЙТИ РІВЕНЬ!"
 #    why_paragraph_3_suffix: "That's why CodeCombat is a multiplayer game, not a gamified lesson course. We won't stop until you can't stop--but this time, that's a good thing."
 #    why_paragraph_4: "If you're going to get addicted to some game, get addicted to this one and become one of the wizards of the tech age."
 #    why_ending: "And hey, it's free. "
@@ -288,10 +288,10 @@ module.exports = nativeDescription: "українська мова", englishDesc
 #    michael_description: "Programmer, sys-admin, and undergrad technical wunderkind, Michael is the person keeping our servers online."
 
 #  legal:
-#    page_title: "Юридичні нотатки"
+    page_title: "Юридичні нотатки"
 #    opensource_intro: "CodeCombat is free to play and completely open source."
 #    opensource_description_prefix: "Check out "
-#    github_url: "наш GitHub"
+    github_url: "наш GitHub"
 #    opensource_description_center: "and help out if you like! CodeCombat is built on dozens of open source projects, and we love them. See "
 #    archmage_wiki_url: "our Archmage wiki"
 #    opensource_description_suffix: "for a list of the software that makes this game possible."

From 9e56540cee71a041dbc8ef25800ec54d27e40f49 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sun, 9 Mar 2014 11:03:21 -0700
Subject: [PATCH 153/178] Uncommented out some headers in uk.coffee
 localization.

---
 app/locale/uk.coffee | 28 ++++++++++++++--------------
 1 file changed, 14 insertions(+), 14 deletions(-)

diff --git a/app/locale/uk.coffee b/app/locale/uk.coffee
index b7e1bcf27..e93dae8d9 100644
--- a/app/locale/uk.coffee
+++ b/app/locale/uk.coffee
@@ -33,12 +33,12 @@ module.exports = nativeDescription: "українська мова", englishDesc
     twitter_follow: "Фоловити"
     employers: "Зайняті"
 
-#  versions:
+  versions:
     save_version_title: "Зберегти нову версію"
     new_major_version: "Зберегти основну версію"
     cla_prefix: "Для збереження змін, спочатку треба погодитись з нашим"
-#    cla_url: "CLA"
-#    cla_suffix: "."
+    cla_url: "CLA"
+    cla_suffix: "."
     cla_agree: "Я Згоден"
 
   login:
@@ -47,7 +47,7 @@ module.exports = nativeDescription: "українська мова", englishDesc
     log_out: "Вийти"
     recover: "відновити акаунт"
 
-#  recover:
+  recover:
     recover_account_title: "Відновити акаунт"
     send_password: "Вислати пароль відновлення"
 
@@ -101,7 +101,7 @@ module.exports = nativeDescription: "українська мова", englishDesc
     learn_more: "Узнати, як стати Дипломатом"
     subscribe_as_diplomat: "Записатися у Дипломати"
 
-#  wizard_settings:
+  wizard_settings:
     title: "Налаштування"
     customize_avatar: "Налаштувати аватар"
     clothes: "Одяг"
@@ -144,11 +144,11 @@ module.exports = nativeDescription: "українська мова", englishDesc
   account_profile:
     edit_settings: "Змінити налаштування"
     profile_for_prefix: "Профіль для "
-#    profile_for_suffix: ""
+    profile_for_suffix: ""
     profile: "Профіль"
     user_not_found: "Користувача не знайдено. Будь ласка, перевірте URL."
     gravatar_not_found_mine: "Ми не можемо знайти ваш профіль, пов'язаний з:"
-#    gravatar_not_found_email_suffix: "."
+    gravatar_not_found_email_suffix: "."
     gravatar_signup_prefix: "Зареєструйтеся на "
     gravatar_signup_suffix: " щоб продовжувати"
     gravatar_not_found_other: "Нажаль, немає профіля, що пов'язаний з електронною адресою цієї людини."
@@ -172,7 +172,7 @@ module.exports = nativeDescription: "українська мова", englishDesc
     reload_title: "Перезавантажити весь код?"
     reload_really: "Ви впевнені, що хочете перезавантажити цей рівень і почати спочатку?"
     reload_confirm: "Перезавантажити все"
-#    victory_title_prefix: ""
+    victory_title_prefix: ""
     victory_title_suffix: " закінчено"
     victory_sign_up: "Підписатися на оновлення"
     victory_sign_up_poke: "Хочете отримувати останні новини на email? Створіть безкоштовний акаунт, і ми будемо тримати вас в курсі!"
@@ -202,17 +202,17 @@ module.exports = nativeDescription: "українська мова", englishDesc
     spell_saved: "Заклинання збережено"
     skip_tutorial: "Пропустити (esc)"
 
-#  admin:
+  admin:
 #    av_title: "Admin Views"
 #    av_entities_sub_title: "Entities"
     av_entities_users_url: "користувачі"
 #    av_entities_active_instances_url: "Active Instances"
-#    av_other_sub_title: "Ынше"
+    av_other_sub_title: "Ынше"
 #    av_other_debug_base_url: "Base (for debugging base.jade)"
     u_title: "Список користувачів"
     lg_title: "Останні ігри"
 
-#  editor:
+  editor:
 #    main_title: "CodeCombat Editors"
 #    main_description: "Build your own levels, campaigns, units and educational content. We provide all the tools you need!"
 #    article_title: "Article Editor"
@@ -252,7 +252,7 @@ module.exports = nativeDescription: "українська мова", englishDesc
 #    edit_btn_preview: "Preview"
 #    edit_article_title: "Edit Article"
 
-#  general:
+  general:
     and: "та"
     name: "Ім’я"
 #    body: "Body"
@@ -265,7 +265,7 @@ module.exports = nativeDescription: "українська мова", englishDesc
     email: "Email"
     message: "Повідомлення"
 
-#  about:
+  about:
     who_is_codecombat: "Хто є CodeCombat?"
     why_codecombat: "Чому CodeCombat?"
     who_description_prefix: "Взагалом розпочався CodeCombat у 2013. Ми також створили "
@@ -287,7 +287,7 @@ module.exports = nativeDescription: "українська мова", englishDesc
 #    jeremy_description: "Customer support mage, usability tester, and community organizer; you've probably already spoken with Jeremy."
 #    michael_description: "Programmer, sys-admin, and undergrad technical wunderkind, Michael is the person keeping our servers online."
 
-#  legal:
+  legal:
     page_title: "Юридичні нотатки"
 #    opensource_intro: "CodeCombat is free to play and completely open source."
 #    opensource_description_prefix: "Check out "

From 03abbc44cb913a6398482e204e83933242fbdd7c Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sun, 9 Mar 2014 11:46:53 -0700
Subject: [PATCH 154/178] Better error messages for missing Component
 dependencies.

---
 app/models/Level.coffee | 3 ++-
 1 file changed, 2 insertions(+), 1 deletion(-)

diff --git a/app/models/Level.coffee b/app/models/Level.coffee
index 176b09aa6..c260a8178 100644
--- a/app/models/Level.coffee
+++ b/app/models/Level.coffee
@@ -52,13 +52,14 @@ module.exports = class Level extends CocoModel
       visit = (c) ->
         return if c in sorted
         lc = _.find levelComponents, {original: c.original}
-        console.error "Couldn't find lc for", c unless lc
+        console.error thang.id, "couldn't find lc for", c unless lc
         if lc.name is "Programmable"
           # Programmable always comes last
           visit c2 for c2 in _.without thang.components, c
         else
           for d in lc.dependencies or []
             c2 = _.find thang.components, {original: d.original}
+            console.error thang.id, "couldn't find dependent Component", d.original, "from", lc.name unless c2
             visit c2
           if lc.name is "Collides"
             allied = _.find levelComponents, {name: "Allied"}

From 34983059aaa9a3aa5fc7ce5394163abc6a11891c Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sun, 9 Mar 2014 11:52:05 -0700
Subject: [PATCH 155/178] Fixed bug with trying to show challenger modals with
 no hash to indicate we should.

---
 app/views/play/ladder_view.coffee | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/views/play/ladder_view.coffee b/app/views/play/ladder_view.coffee
index 7ed43660e..11283af13 100644
--- a/app/views/play/ladder_view.coffee
+++ b/app/views/play/ladder_view.coffee
@@ -67,7 +67,7 @@ module.exports = class LadderView extends RootView
     @insertSubView(@myMatchesTab = new MyMatchesTabView({}, @level, @sessions))
     @refreshInterval = setInterval(@fetchSessionsAndRefreshViews.bind(@), 10000)
     hash = document.location.hash[1..] if document.location.hash
-    unless hash in ['my-matches', 'simulate', 'ladder']
+    if hash and not (hash in ['my-matches', 'simulate', 'ladder'])
       @showPlayModal(hash) if @sessions.loaded
 
   fetchSessionsAndRefreshViews: ->

From 3e82e0b599305ee8e99c0dbbb1a45243ac39a4ef Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sun, 9 Mar 2014 12:53:11 -0700
Subject: [PATCH 156/178] Disallow submission of non-denormalized sessions to
 ladders.

---
 app/views/play/ladder/my_matches_tab.coffee | 7 ++++---
 1 file changed, 4 insertions(+), 3 deletions(-)

diff --git a/app/views/play/ladder/my_matches_tab.coffee b/app/views/play/ladder/my_matches_tab.coffee
index b08a15083..f0f8ecc79 100644
--- a/app/views/play/ladder/my_matches_tab.coffee
+++ b/app/views/play/ladder/my_matches_tab.coffee
@@ -15,7 +15,7 @@ module.exports = class MyMatchesTabView extends CocoView
   constructor: (options, @level, @sessions) ->
     super(options)
     @refreshMatches()
-    
+
   refreshMatches: ->
     @teams = teamDataFromLevel @level
     @nameMap = {}
@@ -68,7 +68,7 @@ module.exports = class MyMatchesTabView extends CocoView
       team.matches = (convertMatch(match) for match in team.session?.get('matches') or [])
       team.matches.reverse()
       team.score = (team.session?.get('totalScore') or 10).toFixed(2)
-      
+
     ctx
 
   afterRender: ->
@@ -80,6 +80,7 @@ module.exports = class MyMatchesTabView extends CocoView
       @setRankingButtonText button, if @readyToRank(session) then 'rank' else 'unavailable'
 
   readyToRank: (session) ->
+    return false unless session?.get('levelID')  # If it hasn't been denormalized, then it's not ready.
     c1 = session?.get('code')
     c2 = session?.get('submittedCode')
     c1 and not _.isEqual(c1, c2)
@@ -94,7 +95,7 @@ module.exports = class MyMatchesTabView extends CocoView
     @setRankingButtonText(button, 'ranking')
     success = => @setRankingButtonText(button, 'ranked')
     failure = => @setRankingButtonText(button, 'failed')
-    
+
     ajaxData = { session: sessionID, levelID: @level.id, originalLevelID: @level.attributes.original, levelMajorVersion: @level.attributes.version.major }
     $.ajax '/queue/scoring', {
       type: 'POST'

From 2d32bc1ac42ca23dd28ed7614305d955ab8046b0 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sun, 9 Mar 2014 13:22:22 -0700
Subject: [PATCH 157/178] Reduced unnecessary name fetching.

---
 app/views/play/ladder/ladder_tab.coffee     |  4 ++++
 app/views/play/ladder/my_matches_tab.coffee | 13 +++++++++----
 2 files changed, 13 insertions(+), 4 deletions(-)

diff --git a/app/views/play/ladder/ladder_tab.coffee b/app/views/play/ladder/ladder_tab.coffee
index b8b2d3615..466b0a402 100644
--- a/app/views/play/ladder/ladder_tab.coffee
+++ b/app/views/play/ladder/ladder_tab.coffee
@@ -74,6 +74,10 @@ class LeaderboardData
 #      @playersBelow.once 'sync', @leaderboardPartLoaded, @
 
   leaderboardPartLoaded: ->
+    # Forget loading the up-to-date names, that's way too slow for something that refreshes all the time, we learned.
+    @loaded = true
+    @trigger 'sync'
+    return
     if @session
       if @topPlayers.loaded # and @playersAbove.loaded and @playersBelow.loaded
         @loaded = true
diff --git a/app/views/play/ladder/my_matches_tab.coffee b/app/views/play/ladder/my_matches_tab.coffee
index f0f8ecc79..998b63bfa 100644
--- a/app/views/play/ladder/my_matches_tab.coffee
+++ b/app/views/play/ladder/my_matches_tab.coffee
@@ -14,23 +14,28 @@ module.exports = class MyMatchesTabView extends CocoView
 
   constructor: (options, @level, @sessions) ->
     super(options)
+    @nameMap = {}
     @refreshMatches()
 
   refreshMatches: ->
     @teams = teamDataFromLevel @level
-    @nameMap = {}
     @loadNames()
 
   loadNames: ->
+    # Only fetch the names for the userIDs we don't already have in @nameMap
     ids = []
     for session in @sessions.models
-      ids.push match.opponents[0].userID for match in session.get('matches') or []
+      for match in (session.get('matches') or [])
+        id = match.opponents[0].userID
+        ids.push id unless @nameMap[id]
 
-    success = (@nameMap) =>
+    return @finishRendering() unless ids.length
+
+    success = (nameMap) =>
       for session in @sessions.models
         for match in session.get('matches') or []
           opponent = match.opponents[0]
-          opponent.userName = @nameMap[opponent.userID]
+          @nameMap[opponent.userID] = nameMap[opponent.userID]
       @finishRendering()
 
     $.ajax('/db/user/-/names', {

From e55a43b396afc8e489ac8c77273bc0f1750ee367 Mon Sep 17 00:00:00 2001
From: Rahazan <guido.reaver@gmail.com>
Date: Sun, 9 Mar 2014 23:37:25 +0100
Subject: [PATCH 158/178] Fixed some formatting of sentences

---
 app/locale/nl.coffee | 6 +++---
 1 file changed, 3 insertions(+), 3 deletions(-)

diff --git a/app/locale/nl.coffee b/app/locale/nl.coffee
index f4c45edc5..eacff6de7 100644
--- a/app/locale/nl.coffee
+++ b/app/locale/nl.coffee
@@ -103,7 +103,7 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
 
   wizard_settings:
     title: "Tovenaar instellingen"
-    customize_avatar: "Bewerk jouw avatar"
+    customize_avatar: "Bewerk je avatar"
     clothes: "Kleren"
     trim: "Trim"
     cloud: "Wolk"
@@ -357,7 +357,7 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
 
   contribute:
     page_title: "Bijdragen"
-    character_classes_title: "Karakter Klassen"
+    character_classes_title: "Karakterklassen"
     introduction_desc_intro: "We hebben hoge verwachtingen over CodeCombat."
     introduction_desc_pref: "We willen zijn waar programmeurs van alle niveaus komen om te leren en samen te spelen, anderen introduceren aan de wondere wereld van code, en de beste delen van de gemeenschap te reflecteren. We kunnen en willen dit niet alleen doen; wat projecten zoals GitHub, Stack Overflow en Linux groots en succesvol maken, zijn de mensen die deze software gebruiken en verbeteren. Daartoe, "
     introduction_desc_github_url: "CodeCombat is volledig open source"
@@ -422,7 +422,7 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     diplomat_launch_url: "release in oktober"
     diplomat_introduction_suf: "dan is het wel dat er een significante interesse is in CodeCombat in andere landen, vooral Brazilië! We zijn een corps aan vertalers aan het creëren dat ijverig de ene set woorden in een andere omzet om CodeCombat zo toegankelijk te maken als mogelijk in heel de wereld. Als jij het leuk vindt glimpsen op te vangen van aankomende content en deze levels zo snel mogelijk naar je landgenoten te krijgen, dan is dit de klasse voor jou."
     diplomat_attribute_1: "Vloeiend Engels en de taal waar naar je wilt vertalen kunnen spreken. Wanneer je moeilijke ideeën wilt overbrengen, is het belangrijk beide goed te kunnen!"
-    diplomat_join_pref_github: "Vind jouw taal haar locale bestand "
+    diplomat_join_pref_github: "Vind van jouw taal het locale bestand "
     diplomat_github_url: "op GitHub"
     diplomat_join_suf_github: ", edit het online, en submit een pull request. Daarnaast kun je hieronder aanvinken als je up-to-date wilt worden gehouden met nieuwe internationalisatie-ontwikkelingen."
     more_about_diplomat: "Leer meer over het worden van een geweldige Diplomaat"

From 2daa70bef4f5bd9db264f10e87b1dfa5a18d5a89 Mon Sep 17 00:00:00 2001
From: Rahazan <guido.reaver@gmail.com>
Date: Sun, 9 Mar 2014 23:45:04 +0100
Subject: [PATCH 159/178] Changed manual

Manual is ambiguous, is it manual as in "manual instead of automatic" or manual as in a handbook/guide. I thought it was the latter, so I changed it here.
---
 app/locale/nl.coffee | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/locale/nl.coffee b/app/locale/nl.coffee
index eacff6de7..99a0a54dc 100644
--- a/app/locale/nl.coffee
+++ b/app/locale/nl.coffee
@@ -8,7 +8,7 @@ module.exports = nativeDescription: "Nederlands", englishDescription: "Dutch", t
     delay_1_sec: "1 seconde"
     delay_3_sec: "3 secondes"
     delay_5_sec: "5 secondes"
-    manual: "Handmatig"
+    manual: "Handleiding"
     fork: "Fork"
     play: "Spelen"
 

From 66b7ba40e84550e81553c0c46904f08349d49e72 Mon Sep 17 00:00:00 2001
From: Rahazan <guido.reaver@gmail.com>
Date: Sun, 9 Mar 2014 23:54:06 +0100
Subject: [PATCH 160/178] Added missing Dutch diplomats

---
 app/templates/contribute/diplomat.jade | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/templates/contribute/diplomat.jade b/app/templates/contribute/diplomat.jade
index 06743761c..9f75161de 100644
--- a/app/templates/contribute/diplomat.jade
+++ b/app/templates/contribute/diplomat.jade
@@ -73,7 +73,7 @@ block content
           li German - Dirk, faabsen, HiroP0, Anon
           li Thai - Kamolchanok Jittrepit
           li Vietnamese - An Nguyen Hoang Thien
-          li Dutch - Glen De Cauwsemaecker
+          li Dutch - Glen De Cauwsemaecker, Guido Zuidhof, Ruben Vereecken
           li Greek - Stergios
           li Latin American Spanish - Jesús Ruppel, Matthew Burt, Mariano Luzza
           li Spain Spanish - Matthew Burt, DanielRodriguezRivero, Anon

From 3d71fcbc708bee9fd0d62530ab491f79f865307e Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Sun, 9 Mar 2014 18:46:11 -0700
Subject: [PATCH 161/178] Wins, losses, and realizing that I'm not doing server
 code right.

---
 app/templates/play/ladder/my_matches_tab.jade |  6 +-----
 app/views/play/ladder/my_matches_tab.coffee   |  3 +++
 server/routes/mail.coffee                     | 16 +++++++++++++++-
 3 files changed, 19 insertions(+), 6 deletions(-)

diff --git a/app/templates/play/ladder/my_matches_tab.jade b/app/templates/play/ladder/my_matches_tab.jade
index d2b543478..c70192221 100644
--- a/app/templates/play/ladder/my_matches_tab.jade
+++ b/app/templates/play/ladder/my_matches_tab.jade
@@ -11,11 +11,7 @@ div#columns.row
 
         tr
           th(colspan=4, style="color: #{team.primaryColor}")
-            span Your
-            span  
-            span= team.name
-            span  
-            span Matches
+            span Your #{team.name} Matches - #{team.wins} Wins, #{team.losses} Losses
 
             if team.session
               button.btn.btn-sm.btn-warning.pull-right.rank-button(data-session-id=team.session.id)
diff --git a/app/views/play/ladder/my_matches_tab.coffee b/app/views/play/ladder/my_matches_tab.coffee
index 998b63bfa..a5e679f1b 100644
--- a/app/views/play/ladder/my_matches_tab.coffee
+++ b/app/views/play/ladder/my_matches_tab.coffee
@@ -73,6 +73,9 @@ module.exports = class MyMatchesTabView extends CocoView
       team.matches = (convertMatch(match) for match in team.session?.get('matches') or [])
       team.matches.reverse()
       team.score = (team.session?.get('totalScore') or 10).toFixed(2)
+      team.wins = _.filter(team.matches, {state: 'win'}).length
+      team.ties = _.filter(team.matches, {state: 'tie'}).length
+      team.losses = _.filter(team.matches, {state: 'loss'}).length
 
     ctx
 
diff --git a/server/routes/mail.coffee b/server/routes/mail.coffee
index ed6b58795..d629883d7 100644
--- a/server/routes/mail.coffee
+++ b/server/routes/mail.coffee
@@ -5,6 +5,7 @@ errors = require '../commons/errors'
 #request = require 'request'
 config = require '../../server_config'
 LevelSession = require '../levels/sessions/LevelSession.coffee'
+Level = require '../levels/Level.coffee'
 log = require 'winston'
 sendwithus = require '../sendwithus'
 
@@ -16,9 +17,23 @@ module.exports.setup = (app) ->
   app.all config.mail.mailchimpWebhook, handleMailchimpWebHook
   app.get '/mail/cron/ladder-update', handleLadderUpdate
 
+getAllLadderScores = (next) ->
+  query = Level.find({type: 'ladder'})
+    .select('levelID')
+    .lean()
+  query.exec (err, levels) ->
+    if err
+      log.error "Couldn't fetch ladder levels. Error: ", err
+      return next []
+    for level in levels
+      for team in ['humans', 'ogres']
+        'I ... am not doing this.'
+
 handleLadderUpdate = (req, res) ->
+  log.info("Going to see about sending ladder update emails.")
   res.send('Great work, Captain Cron! I can take it from here.')
   res.end()
+  # TODO: somehow fetch the histograms
   emailDays = [1, 2, 4, 7, 30]
   now = new Date()
   getTimeFromDaysAgo = (daysAgo) ->
@@ -35,7 +50,6 @@ handleLadderUpdate = (req, res) ->
     query = LevelSession.find(findParameters)
       .select(selectString)
       .lean()
-    mongoose = require 'mongoose'
     do (daysAgo) ->
       query.exec (err, results) ->
         if err

From 5e634f9915bfc980a59bcd4d824ef45f37ca0448 Mon Sep 17 00:00:00 2001
From: Darredevil <alex.darredevil@gmail.com>
Date: Mon, 10 Mar 2014 04:10:29 +0200
Subject: [PATCH 162/178] Update ro.coffee

Almost done.
The legal stuff is the most boring thing i ever had to translate...
---
 app/locale/ro.coffee | 140 +++++++++++++++++++++----------------------
 1 file changed, 70 insertions(+), 70 deletions(-)

diff --git a/app/locale/ro.coffee b/app/locale/ro.coffee
index e9d7e3db6..531e3825f 100644
--- a/app/locale/ro.coffee
+++ b/app/locale/ro.coffee
@@ -265,7 +265,7 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
     message: "Mesaj"
 
   about:
-    who_is_codecombat: "Cine este CodeCombat?"  # I assume you meant (what)
+    who_is_codecombat: "Cine este CodeCombat?"  
     why_codecombat: "De ce CodeCombat?"
     who_description_prefix: "au pornit împreuna CodeCombat în 2013. Tot noi am creat "
     who_description_suffix: "în 2008, dezvoltând aplicația web si iOS #1 de învățat cum să scri caractere Japoneze si Chinezești."
@@ -277,77 +277,77 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
     why_paragraph_3_center: "ci"
     why_paragraph_3_italic_caps: "TREBUIE SĂ TERMIN ACEST NIVEL!"
     why_paragraph_3_suffix: "De aceea CodeCombat este un joc multiplayer, nu un curs transfigurat în joc. Nu ne vom opri până când tu nu te poți opri--și de data asta, e de bine."
-#    why_paragraph_4: "If you're going to get addicted to some game, get addicted to this one and become one of the wizards of the tech age."
-#    why_ending: "And hey, it's free. "
-#    why_ending_url: "Start wizarding now!"
-#    george_description: "CEO, business guy, web designer, game designer, and champion of beginning programmers everywhere."
-#    scott_description: "Programmer extraordinaire, software architect, kitchen wizard, and master of finances. Scott is the reasonable one."
-#    nick_description: "Programming wizard, eccentric motivation mage, and upside-down experimenter. Nick can do anything and chooses to build CodeCombat."
-#    jeremy_description: "Customer support mage, usability tester, and community organizer; you've probably already spoken with Jeremy."
-#    michael_description: "Programmer, sys-admin, and undergrad technical wunderkind, Michael is the person keeping our servers online."
+    why_paragraph_4: "Dacă e să devi dependent de vreun joc, devino dependent de acesta și fi un vrăjitor al noii ere tehnologice."
+    why_ending: "Nu uita, este totul gratis. "
+    why_ending_url: "Devino un vrăjitor acum!"
+    george_description: "CEO, business guy, web designer, game designer, și campion al programatorilor începători."
+    scott_description: "Programmer extraordinaire, software architect, kitchen wizard, și  maestru al finanțelor. Scott este cel rezonabil."
+    nick_description: "Programming wizard, eccentric motivation mage, and upside-down experimenter. Nick poate să facă orice si a ales să dezvolte CodeCombat."
+    jeremy_description: "Customer support mage, usability tester, and community organizer; probabil ca ați vorbit deja cu Jeremy."
+    michael_description: "Programmer, sys-admin, and undergrad technical wunderkind, Michael este cel care ține serverele in picioare."
 
-#  legal:
-#    page_title: "Legal"
-#    opensource_intro: "CodeCombat is free to play and completely open source."
-#    opensource_description_prefix: "Check out "
-#    github_url: "our GitHub"
-#    opensource_description_center: "and help out if you like! CodeCombat is built on dozens of open source projects, and we love them. See "
-#    archmage_wiki_url: "our Archmage wiki"
-#    opensource_description_suffix: "for a list of the software that makes this game possible."
-#    practices_title: "Respectful Best Practices"
-#    practices_description: "These are our promises to you, the player, in slightly less legalese."
-#    privacy_title: "Privacy"
-#    privacy_description: "We will not sell any of your personal information. We intend to make money through recruitment eventually, but rest assured we will not distribute your personal information to interested companies without your explicit consent."
-#    security_title: "Security"
-#    security_description: "We strive to keep your personal information safe. As an open source project, our site is freely open to anyone to review and improve our security systems."
-#    email_title: "Email"
-#    email_description_prefix: "We will not inundate you with spam. Through"
-#    email_settings_url: "your email settings"
-#    email_description_suffix: "or through links in the emails we send, you can change your preferences and easily unsubscribe at any time."
-#    cost_title: "Cost"
-#    cost_description: "Currently, CodeCombat is 100% free! One of our main goals is to keep it that way, so that as many people can play as possible, regardless of place in life. If the sky darkens, we might have to charge subscriptions or for some content, but we'd rather not. With any luck, we'll be able to sustain the company with:"
-#    recruitment_title: "Recruitment"
-#    recruitment_description_prefix: "Here on CodeCombat, you're going to become a powerful wizard–not just in the game, but also in real life."
-#    url_hire_programmers: "No one can hire programmers fast enough"
-#    recruitment_description_suffix: "so once you've sharpened your skills and if you agree, we will demo your best coding accomplishments to the thousands of employers who are drooling for the chance to hire you. They pay us a little, they pay you"
-#    recruitment_description_italic: "a lot"
-#    recruitment_description_ending: "the site remains free and everybody's happy. That's the plan."
-#    copyrights_title: "Copyrights and Licenses"
-#    contributor_title: "Contributor License Agreement"
-#    contributor_description_prefix: "All contributions, both on the site and on our GitHub repository, are subject to our"
-#    cla_url: "CLA"
-#    contributor_description_suffix: "to which you should agree before contributing."
-#    code_title: "Code - MIT"
-#    code_description_prefix: "All code owned by CodeCombat or hosted on codecombat.com, both in the GitHub repository or in the codecombat.com database, is licensed under the"
-#    mit_license_url: "MIT license"
-#    code_description_suffix: "This includes all code in Systems and Components that are made available by CodeCombat for the purpose of creating levels."
-#    art_title: "Art/Music - Creative Commons "
-#    art_description_prefix: "All common content is available under the"
-#    cc_license_url: "Creative Commons Attribution 4.0 International License"
-#    art_description_suffix: "Common content is anything made generally available by CodeCombat for the purpose of creating Levels. This includes:"
-#    art_music: "Music"
-#    art_sound: "Sound"
-#    art_artwork: "Artwork"
-#    art_sprites: "Sprites"
-#    art_other: "Any and all other non-code creative works that are made available when creating Levels."
-#    art_access: "Currently there is no universal, easy system for fetching these assets. In general, fetch them from the URLs as used by the site, contact us for assistance, or help us in extending the site to make these assets more easily accessible."
-#    art_paragraph_1: "For attribution, please name and link to codecombat.com near where the source is used or where appropriate for the medium. For example:"
-#    use_list_1: "If used in a movie or another game, include codecombat.com in the credits."
-#    use_list_2: "If used on a website, include a link near the usage, for example underneath an image, or in a general attributions page where you might also mention other Creative Commons works and open source software being used on the site. Something that's already clearly referencing CodeCombat, such as a blog post mentioning CodeCombat, does not need some separate attribution."
-#    art_paragraph_2: "If the content being used is created not by CodeCombat but instead by a user of codecombat.com, attribute them instead, and follow attribution directions provided in that resource's description if there are any."
-#    rights_title: "Rights Reserved"
-#    rights_desc: "All rights are reserved for Levels themselves. This includes"
-#    rights_scripts: "Scripts"
-#    rights_unit: "Unit configuration"
-#    rights_description: "Description"
-#    rights_writings: "Writings"
-#    rights_media: "Media (sounds, music) and any other creative content made specifically for that Level and not made generally available when creating Levels."
-#    rights_clarification: "To clarify, anything that is made available in the Level Editor for the purpose of making levels is under CC, whereas the content created with the Level Editor or uploaded in the course of creation of Levels is not."
-#    nutshell_title: "In a Nutshell"
-#    nutshell_description: "Any resources we provide in the Level Editor are free to use as you like for creating Levels. But we reserve the right to restrict distribution of the Levels themselves (that are created on codecombat.com) so that they may be charged for in the future, if that's what ends up happening."
-#    canonical: "The English version of this document is the definitive, canonical version. If there are any discrepencies between translations, the English document takes precedence."
+  legal:
+    page_title: "Aspecte Legale"
+    opensource_intro: "CodeCombat este free-to-play și complet open source."
+    opensource_description_prefix: "Vizitează "
+    github_url: "pagina noastră de GitHub"
+    opensource_description_center: "și ajută-ne dacă îți place! CodeCombat este construit peste o mulțime de proiecte open source, care noi le iubim. Vizitați"
+    archmage_wiki_url: "Archmage wiki"
+    opensource_description_suffix: "pentru o listă cu software-ul care face acest joc posibil."
+#    practices_title: "Respectful Best Practices"   #not sure what you mean here? other word for /practices/?
+    practices_description: "Acestea sunt promisiunile noastre către tine, jucătorul, fără așa mulți termeni legali."
+    privacy_title: "Confidenţialitate şi termeni"
+    privacy_description: "Noi nu vom vinde nici o informație personală. Intenționăm să obținem profit prin recrutare eventual, dar stați liniștiți , nu vă vom vinde informațiile personale companiilor interesate fără consimțământul vostru explicit."
+    security_title: "Securitate"
+    security_description: "Ne străduim să vă protejăm informațiile personale. Fiind un proiect open-source, site-ul nostru oferă oricui posibilitatea de a ne revizui și îmbunătăți sistemul de securitate."
+    email_title: "Email"
+    email_description_prefix: "Noi nu vă vom inunda cu spam. Prin"
+    email_settings_url: "setările tale de email"
+    email_description_suffix: " sau prin link-urile din email-urile care vi le trimitem, puteți să schimbați preferințele și să vâ dezabonați oricând."
+    cost_title: "Cost"
+    cost_description: "Momentan, CodeCombat este 100% gratis! Unul dintre obiectele noastre principale este să îl menținem așa, astfel încât să poată juca cât mai mulți oameni. Dacă va fi nevoie , s-ar putea să percepem o plată pentru o pentru anumite servici,dar am prefera să nu o facem. Cu puțin noroc, vom putea susține compania cu:"
+    recruitment_title: "Recrutare"
+    recruitment_description_prefix: "Aici la CodeCombat, vei deveni un vrăjitor puternic nu doar în joc , ci și în viața reală."
+    url_hire_programmers: "Nimeni nu poate angaja programatori destul de rapid"
+    recruitment_description_suffix: "așa că odată ce ți-ai dezvoltat abilitățile și esti de acord, noi vom trimite un demo cu cele mai bune realizări ale tale către miile de angajatori care se omoară să pună mâna pe tine. Pe noi ne plătesc puțin, pe tine te vor plăti"
+    recruitment_description_italic: "mult"
+    recruitment_description_ending: "site-ul rămâne gratis și toată lumea este fericită. Acesta este planul."
+    copyrights_title: "Drepturi de autor și licențe"
+    contributor_title: "Acord de licență Contributor"
+    contributor_description_prefix: "Toți contribuitorii, atât pe site cât și pe GitHub-ul nostru, sunt supuși la"
+    cla_url: "ALC"
+    contributor_description_suffix: "la care trebuie să fi de accord înainte să poți contribui."
+    code_title: "Code - MIT"
+    code_description_prefix: "Tot codul deținut de CodeCombat sau hostat pe codecombat.com, atât pe GitHub cât și în baza de date codecombat.com, este licențiată sub"
+    mit_license_url: "MIT license"
+    code_description_suffix: "Asta include tot codul din Systems și Components care este oferit de către CodeCombat cu scopul de a crea nivele."
+    art_title: "Artă/Muzică - Conținut Comun "
+    art_description_prefix: "Tot conținutul creativ/artistic este valabil sub"
+    cc_license_url: "Creative Commons Attribution 4.0 International License"
+    art_description_suffix: "Conținut comun este orice făcut general valabil de către CodeCombat cu scopul de a crea nivele. Asta include:"
+    art_music: "Muzică"
+    art_sound: "Sunet"
+    art_artwork: "Artwork"
+    art_sprites: "Sprites"  #can t be translated, either suggest alternative name or must be left like this
+    art_other: "Orice si toate celelalte creații non-cod care sunt disponibile când se crează nivele."
+    art_access: "Momentan nu există nici un sistem universal,ușor pentru preluarea acestor bunuri. În general, preluați-le precum site-ul din URL-urile folosite, contactați-ne pentru asistență, sau ajutați-ne sa extindem site-ul pentru a face aceste bunuri mai ușor accesibile."
+    art_paragraph_1: "Pentru atribuire, vă rugăm numiți și lăsați referire link la codecombat.com unde este folosită sursa  sau unde este adecvat pentru mediu. De exemplu:"
+    use_list_1: "Dacă este folosit într-un film sau alt joc, includeți codecombat.com la credite."
+    use_list_2: "Dacă este folosit pe un site, includeți un link in apropiere, de exemplu sub o imagine, sau in pagina generală de atribuiri unde menționați și alte Bunuri Creative și software open source folosit pe site. Ceva care face referință explicit la CodeCombat, precum o postare pe un blog care menționează CodeCombat, nu trebuie să facă o atribuire separată."
+    art_paragraph_2: "Dacă conținutul folosit nu este creat de către CodeCombat ci de către un utilizator al codecombat.com,atunci faceți referință către ei, și urmăriți indicațiile de atribuire prevăzute în descrierea resursei dacă există."
+    rights_title: "Drepturi rezervate"
+    rights_desc: "Toate drepturile sunt rezervate pentru Nivele în sine. Asta include"
+    rights_scripts: "Script-uri"
+    rights_unit: "Configurații de unități"
+    rights_description: "Descriere"
+    rights_writings: "Scrieri"
+    rights_media: "Media (sunete, muzică) și orice alt conținut creativ dezvoltat special pentru acel nivel care nu este valabil în mod normal pentru creat nivele."
+    rights_clarification: "Pentru a clarifica, orice este valabil in Editorul de Nivele pentru scopul de a crea nivele se află sub CC,pe când conținutul creat cu Editorul de Nivele sau încărcat pentru a face nivelul nu se află." #CC stands for...?
+    nutshell_title: "Pe scurt"
+    nutshell_description: "Orice resurse vă punem la dispoziție în Editorul de Nivele puteți folosi liber cum vreți pentru a crea nivele. Dar ne rezervăm dreptul de a rezerva distribuția de nivele în sine (care sunt create pe codecombat.com) astfel încât să se poată percepe o taxă pentru ele pe vitor, dacă se va ajunge la așa ceva."
+    canonical: "The English version of this document is the definitive, canonical version. If there are any discrepencies between translations, the English document takes precedence."
 
-#  contribute:
+  contribute:
 #    page_title: "Contributing"
 #    character_classes_title: "Character Classes"
 #    introduction_desc_intro: "We have high hopes for CodeCombat."

From 69b902f62ac8a7ad6cb2cfb01a3cacad559175b4 Mon Sep 17 00:00:00 2001
From: Michael Schmatz <schmatz@umich.edu>
Date: Mon, 10 Mar 2014 08:14:28 -0700
Subject: [PATCH 163/178] Added support for score history

---
 server/levels/sessions/level_session_schema.coffee | 12 ++++++++++++
 server/queues/scoring.coffee                       |  7 ++++++-
 2 files changed, 18 insertions(+), 1 deletion(-)

diff --git a/server/levels/sessions/level_session_schema.coffee b/server/levels/sessions/level_session_schema.coffee
index 290422c10..ca8df4c05 100644
--- a/server/levels/sessions/level_session_schema.coffee
+++ b/server/levels/sessions/level_session_schema.coffee
@@ -146,6 +146,18 @@ _.extend LevelSessionSchema.properties,
   numberOfLosses:
     type: 'number'
     default: 0
+    
+  scoreHistory:
+    type: 'array'
+    title: 'Score History'
+    description: 'A list of objects representing the score history of a session'
+    items: 
+      title: 'Score History Point'
+      description: 'An array with the format [unix timestamp, totalScore]'
+      type: 'array'
+      items:
+        type: 'number'
+      
 
   matches:
     type: 'array'
diff --git a/server/queues/scoring.coffee b/server/queues/scoring.coffee
index 1e2cd3f29..0462cd503 100644
--- a/server/queues/scoring.coffee
+++ b/server/queues/scoring.coffee
@@ -443,10 +443,15 @@ updateScoreInSession = (scoreObject,callback) ->
     if err? then return callback err, null
 
     session = session.toObject()
+    newTotalScore = scoreObject.meanStrength - 1.8 * scoreObject.standardDeviation
+    scoreHistoryAddition = [Date.now(), newTotalScore]
     updateObject =
       meanStrength: scoreObject.meanStrength
       standardDeviation: scoreObject.standardDeviation
-      totalScore: scoreObject.meanStrength - 1.8 * scoreObject.standardDeviation
+      totalScore: newTotalScore
+      $push:
+        scoreHistory: scoreHistoryAddition        
+
     LevelSession.update {"_id": scoreObject.id}, updateObject, callback
     log.info "New total score for session #{scoreObject.id} is #{updateObject.totalScore}"
 

From c6b1d9089cd5b7802291757168adfa823030603a Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 10 Mar 2014 08:45:36 -0700
Subject: [PATCH 164/178] Fixed sounds spamming on last frame.

---
 app/lib/surface/Surface.coffee | 9 ++++++---
 1 file changed, 6 insertions(+), 3 deletions(-)

diff --git a/app/lib/surface/Surface.coffee b/app/lib/surface/Surface.coffee
index feb87fc93..5f8ae5f49 100644
--- a/app/lib/surface/Surface.coffee
+++ b/app/lib/surface/Surface.coffee
@@ -497,13 +497,15 @@ module.exports = Surface = class Surface extends CocoClass
     # seems to be a bug where only one object can register with the Ticker...
     oldFrame = @currentFrame
     oldWorldFrame = Math.floor oldFrame
+    lastFrame = @world.totalFrames - 1
     while true
       Dropper.tick()
       @trailmaster.tick() if @trailmaster
       # Skip some frame updates unless we're playing and not at end (or we haven't drawn much yet)
-      frameAdvanced = (@playing and @currentFrame < @world.totalFrames) or @totalFramesDrawn < 2
-      @currentFrame += @world.frameRate / @options.frameRate if frameAdvanced and @playing
-      @currentFrame = Math.min(@currentFrame, @world.totalFrames - 1)
+      frameAdvanced = (@playing and @currentFrame < lastFrame) or @totalFramesDrawn < 2
+      if frameAdvanced and @playing
+        @currentFrame += @world.frameRate / @options.frameRate
+        @currentFrame = Math.min @currentFrame, lastFrame
       newWorldFrame = Math.floor @currentFrame
       worldFrameAdvanced = newWorldFrame isnt oldWorldFrame
       if worldFrameAdvanced
@@ -513,6 +515,7 @@ module.exports = Surface = class Surface extends CocoClass
       break unless Dropper.drop()
     if frameAdvanced and not worldFrameAdvanced
       # We didn't end the above loop on an integer frame, so do the world state update.
+      console.log "Restore world state"
       @restoreWorldState()
 
     # these are skipped for dropped frames

From 6a71e97204743c279a8e0b8fb672659a99871688 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 10 Mar 2014 09:37:05 -0700
Subject: [PATCH 165/178] Improvements to HUD actions.

---
 app/styles/play/level/hud.sass       | 6 +++---
 app/views/play/level/hud_view.coffee | 7 ++++---
 2 files changed, 7 insertions(+), 6 deletions(-)

diff --git a/app/styles/play/level/hud.sass b/app/styles/play/level/hud.sass
index 0379fc151..c83b20933 100644
--- a/app/styles/play/level/hud.sass
+++ b/app/styles/play/level/hud.sass
@@ -208,11 +208,11 @@
               height: 19px
 
               div
-                @include box-sizing(border-box)
+                border-radius: 1px
                 background-color: #6BA1C8
                 height: 100%
-                border-bottom: 2px groove #201B15
-                border-right: 1px solid #201B15
+                border-bottom: 2px groove darken(#6BA1C8, 30%)
+                border-right: 1px solid darken(#6BA1C8, 10%)
                 position: absolute
                 top: 0
 
diff --git a/app/views/play/level/hud_view.coffee b/app/views/play/level/hud_view.coffee
index 662c9fc8f..9abd6d726 100644
--- a/app/views/play/level/hud_view.coffee
+++ b/app/views/play/level/hud_view.coffee
@@ -135,7 +135,7 @@ module.exports = class HUDView extends View
     props = @$el.find('.thang-props')
     props.find(":not(.thang-name)").remove()
     props.find('.thang-name').text(if @thang.type then "#{@thang.id} - #{@thang.type}" else @thang.id)
-    propNames = @thang.hudProperties ? []
+    propNames = _.without @thang.hudProperties ? [], 'action'
     nColumns = Math.ceil propNames.length / 5
     columns = ($('<div class="thang-props-column"></div>').appendTo(props) for i in [0 ... nColumns])
     for prop, i in propNames
@@ -316,11 +316,12 @@ module.exports = class HUDView extends View
     @timespans = {}
     dt = @thang.world.dt
     actionHistory = @thang.world.actionsForThang @thang.id, true
-    [lastFrame, lastAction] = [0, 'idle']
+    console.log "got actionHistory", actionHistory
+    [lastFrame, lastAction] = [0, null]
     for hist in actionHistory.concat {frame: @thang.world.totalFrames, name: 'END'}
       [newFrame, newAction] = [hist.frame, hist.name]
       continue if newAction is lastAction
-      if newFrame > lastFrame
+      if newFrame > lastFrame and lastAction
         # TODO: don't push it if it didn't exist until then
         (@timespans[lastAction] ?= []).push [lastFrame * dt, newFrame * dt]
       [lastFrame, lastAction] = [newFrame, newAction]

From 234b3f105d8ea2f74fad3fa83b8db7d360575baf Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 10 Mar 2014 10:36:28 -0700
Subject: [PATCH 166/178] Reworked ladder game victory modal main call to
 action to actually submit game for ranking.

---
 app/lib/surface/Surface.coffee                |  1 -
 app/locale/en.coffee                          |  3 +++
 app/templates/play/level/modal/victory.jade   |  6 ++++--
 app/views/play/ladder/my_matches_tab.coffee   |  1 -
 app/views/play/level/hud_view.coffee          |  1 -
 .../play/level/modal/victory_modal.coffee     | 20 +++++++++++++++++++
 6 files changed, 27 insertions(+), 5 deletions(-)

diff --git a/app/lib/surface/Surface.coffee b/app/lib/surface/Surface.coffee
index 5f8ae5f49..fd278b8cc 100644
--- a/app/lib/surface/Surface.coffee
+++ b/app/lib/surface/Surface.coffee
@@ -515,7 +515,6 @@ module.exports = Surface = class Surface extends CocoClass
       break unless Dropper.drop()
     if frameAdvanced and not worldFrameAdvanced
       # We didn't end the above loop on an integer frame, so do the world state update.
-      console.log "Restore world state"
       @restoreWorldState()
 
     # these are skipped for dropped frames
diff --git a/app/locale/en.coffee b/app/locale/en.coffee
index e3c694f82..2766dda9c 100644
--- a/app/locale/en.coffee
+++ b/app/locale/en.coffee
@@ -179,6 +179,9 @@ module.exports = nativeDescription: "English", englishDescription: "English", tr
     victory_sign_up: "Sign Up to Save Progress"
     victory_sign_up_poke: "Want to save your code? Create a free account!"
     victory_rate_the_level: "Rate the level: "
+    victory_rank_my_game: "Rank My Game"
+    victory_ranking_game: "Submitting..."
+    victory_return_to_ladder: "Return to Ladder"
     victory_play_next_level: "Play Next Level"
     victory_go_home: "Go Home"
     victory_review: "Tell us more!"
diff --git a/app/templates/play/level/modal/victory.jade b/app/templates/play/level/modal/victory.jade
index 45c7f567f..6329f0bbe 100644
--- a/app/templates/play/level/modal/victory.jade
+++ b/app/templates/play/level/modal/victory.jade
@@ -14,8 +14,10 @@
     div!= body
   
   .modal-footer
-    if level.get('type') === 'ladder'
-      a.btn.btn-primary(href="/play/ladder/#{level.get('slug')}#my-matches", data-dismiss="modal", data-i18n="play_level.victory_go_home") Go Home
+    if readyToRank
+      button.btn.btn-success.rank-game-button(data-i18n="play_level.victory_rank_my_game") Rank My Game
+    else if level.get('type') === 'ladder'
+      a.btn.btn-primary(href="/play/ladder/#{level.get('slug')}#my-matches", data-dismiss="modal", data-i18n="play_level.victory_go_ladder") Return to Ladder
     else if hasNextLevel
       button.btn.btn-primary.next-level-button(data-dismiss="modal", data-i18n="play_level.victory_play_next_level") Play Next Level
     else
diff --git a/app/views/play/ladder/my_matches_tab.coffee b/app/views/play/ladder/my_matches_tab.coffee
index a5e679f1b..a094aef82 100644
--- a/app/views/play/ladder/my_matches_tab.coffee
+++ b/app/views/play/ladder/my_matches_tab.coffee
@@ -94,7 +94,6 @@ module.exports = class MyMatchesTabView extends CocoView
     c1 and not _.isEqual(c1, c2)
 
   rankSession: (e) ->
-    console.log "Clicked"
     button = $(e.target).closest('.rank-button')
     sessionID = button.data('session-id')
     session = _.find @sessions.models, { id: sessionID }
diff --git a/app/views/play/level/hud_view.coffee b/app/views/play/level/hud_view.coffee
index 9abd6d726..324c5e68b 100644
--- a/app/views/play/level/hud_view.coffee
+++ b/app/views/play/level/hud_view.coffee
@@ -316,7 +316,6 @@ module.exports = class HUDView extends View
     @timespans = {}
     dt = @thang.world.dt
     actionHistory = @thang.world.actionsForThang @thang.id, true
-    console.log "got actionHistory", actionHistory
     [lastFrame, lastAction] = [0, null]
     for hist in actionHistory.concat {frame: @thang.world.totalFrames, name: 'END'}
       [newFrame, newAction] = [hist.frame, hist.name]
diff --git a/app/views/play/level/modal/victory_modal.coffee b/app/views/play/level/modal/victory_modal.coffee
index 6773982ad..c71b2bcac 100644
--- a/app/views/play/level/modal/victory_modal.coffee
+++ b/app/views/play/level/modal/victory_modal.coffee
@@ -11,6 +11,7 @@ module.exports = class VictoryModal extends View
 
   events:
     'click .next-level-button': 'onPlayNextLevel'
+    'click .rank-game-button': 'onRankGame'
 
     # review events
     'mouseover .rating i': (e) -> @showStars(@starNum($(e.target)))
@@ -58,6 +59,21 @@ module.exports = class VictoryModal extends View
     @saveReview() if @$el.find('.review textarea').val()
     Backbone.Mediator.publish('play-next-level')
 
+  onRankGame: (e) ->
+    button = @$el.find('.rank-game-button')
+    button.text($.i18n.t('play_level.victory_ranking_game', defaultValue: 'Submitting...'))
+    button.prop 'disabled', true
+    ajaxData = session: @session.id, levelID: @level.id, originalLevelID: @level.get('original'), levelMajorVersion: @level.get('version').major
+    ladderURL = "/play/ladder/#{@level.get('slug')}#my-matches"
+    goToLadder = -> Backbone.Mediator.publish 'router:navigate', route: ladderURL
+    $.ajax '/queue/scoring',
+      type: 'POST'
+      data: ajaxData
+      success: goToLadder
+      failure: (response) ->
+        console.error "Couldn't submit game for ranking:", response
+        goToLadder()
+
   getRenderData: ->
     c = super()
     c.body = @body
@@ -65,6 +81,10 @@ module.exports = class VictoryModal extends View
     c.hasNextLevel = _.isObject(@level.get('nextLevel')) and (@level.get('name') isnt "Mobile Artillery")
     c.levelName = @level.get('i18n')?[me.lang()]?.name ? @level.get('name')
     c.level = @level
+    if c.level.get('type') is 'ladder'
+      c1 = @session?.get('code')
+      c2 = @session?.get('submittedCode')
+      c.readyToRank = @session.get('levelID') and c1 and not _.isEqual(c1, c2)
     if me.get 'hourOfCode'
       # Show the Hour of Code "I'm Done" tracking pixel after they played for 30 minutes
       elapsed = (new Date() - new Date(me.get('dateCreated')))

From 2ed83f4b40023edff4fd189915f521df331b521b Mon Sep 17 00:00:00 2001
From: Michael Schmatz <schmatz@umich.edu>
Date: Mon, 10 Mar 2014 10:56:33 -0700
Subject: [PATCH 167/178] Sped up my_sessions using projection, bypassing auth

---
 server/levels/level_handler.coffee | 20 ++++++++++++++++----
 1 file changed, 16 insertions(+), 4 deletions(-)

diff --git a/server/levels/level_handler.coffee b/server/levels/level_handler.coffee
index 94bf33dd8..01b7eec40 100644
--- a/server/levels/level_handler.coffee
+++ b/server/levels/level_handler.coffee
@@ -89,15 +89,27 @@ LevelHandler = class LevelHandler extends Handler
       # associated with the handler, because the handler might return a different type
       # of model, like in this case. Refactor to move that logic to the model instead.
 
-  getMySessions: (req, res, id) ->
-    @fetchLevelByIDAndHandleErrors id, req, res, (err, level) =>
+  getMySessions: (req, res, slugOrID) ->
+    findParameters = {}
+    if Handler.isID slugOrID
+      findParameters["_id"] = slugOrID
+    else
+      findParameters["slug"] = slugOrID
+    selectString = 'original version.major permissions'
+    query = Level.findOne(findParameters)
+      .select(selectString)
+      .lean()
+    
+    query.exec (err, level) =>
+      return @sendDatabaseError(res, err) if err
+      return @sendNotFoundError(res) unless level?
       sessionQuery =
         level:
           original: level.original.toString()
           majorVersion: level.version.major
         creator: req.user._id+''
-
-      query = Session.find(sessionQuery)
+      
+      query = Session.find(sessionQuery).select('-screenshot')
       query.exec (err, results) =>
         if err then @sendDatabaseError(res, err) else @sendSuccess res, results
 

From d4abad88a2f479525bdf124bc8ea78f12ac713f2 Mon Sep 17 00:00:00 2001
From: Michael Schmatz <schmatz@umich.edu>
Date: Mon, 10 Mar 2014 12:57:25 -0700
Subject: [PATCH 168/178] Removed leaderboard mongo sort

---
 server/levels/level_handler.coffee | 1 -
 1 file changed, 1 deletion(-)

diff --git a/server/levels/level_handler.coffee b/server/levels/level_handler.coffee
index 01b7eec40..4a2bdd24a 100644
--- a/server/levels/level_handler.coffee
+++ b/server/levels/level_handler.coffee
@@ -140,7 +140,6 @@ LevelHandler = class LevelHandler extends Handler
     query = Session
       .find(sessionsQueryParameters)
       .limit(req.query.limit)
-      .sort(sortParameters)
       .select(selectProperties.join ' ')
 
     query.exec (err, resultSessions) =>

From 6055512ac7bc57be4add9e633e0f90656a6555c2 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 10 Mar 2014 13:20:00 -0700
Subject: [PATCH 169/178] Working on ladder updates.

---
 app/views/account/settings_view.coffee         |  4 ++--
 app/views/modal/signup_modal.coffee            |  3 ++-
 .../sessions/level_session_handler.coffee      |  2 +-
 .../sessions/level_session_schema.coffee       | 14 +++++++++-----
 server/routes/mail.coffee                      | 18 +++++++++++-------
 server/sendwithus.coffee                       |  2 +-
 server/users/user_schema.coffee                |  6 +++---
 7 files changed, 29 insertions(+), 20 deletions(-)

diff --git a/app/views/account/settings_view.coffee b/app/views/account/settings_view.coffee
index 2f609eebd..27edf27e1 100644
--- a/app/views/account/settings_view.coffee
+++ b/app/views/account/settings_view.coffee
@@ -66,7 +66,7 @@ module.exports = class SettingsView extends View
     c.photos = me.gravatarPhotoURLs()
     c.chosenPhoto = me.getPhotoURL()
     c.subs = {}
-    c.subs[sub] = 1 for sub in c.me.get('emailSubscriptions') or ['announcement', 'tester', 'level_creator', 'developer']
+    c.subs[sub] = 1 for sub in c.me.get('emailSubscriptions') or ['announcement', 'notification', 'tester', 'level_creator', 'developer']
     c
 
   getSubscriptions: ->
@@ -88,7 +88,7 @@ module.exports = class SettingsView extends View
     if res?
       forms.applyErrorsToForm(@$el, res)
       return
-      
+
     return unless me.hasLocalChanges()
 
     res = me.save()
diff --git a/app/views/modal/signup_modal.coffee b/app/views/modal/signup_modal.coffee
index 579a938cb..d1c76e3a1 100644
--- a/app/views/modal/signup_modal.coffee
+++ b/app/views/modal/signup_modal.coffee
@@ -49,8 +49,9 @@ module.exports = class SignupModalView extends View
     userObject.emailSubscriptions ?= []
     if subscribe
       userObject.emailSubscriptions.push 'announcement' unless 'announcement' in userObject.emailSubscriptions
+      userObject.emailSubscriptions.push 'notification' unless 'notification' in userObject.emailSubscriptions
     else
-      userObject.emailSubscriptions = _.without (userObject.emailSubscriptions ? []), 'announcement'
+      userObject.emailSubscriptions = _.without (userObject.emailSubscriptions ? []), 'announcement', 'notification'
     res = tv4.validateMultiple userObject, User.schema.attributes
     return forms.applyErrorsToForm(@$el, res.errors) unless res.valid
     window.tracker?.trackEvent 'Finished Signup'
diff --git a/server/levels/sessions/level_session_handler.coffee b/server/levels/sessions/level_session_handler.coffee
index dbb3324f4..d3ab07830 100644
--- a/server/levels/sessions/level_session_handler.coffee
+++ b/server/levels/sessions/level_session_handler.coffee
@@ -7,7 +7,7 @@ class LevelSessionHandler extends Handler
   modelClass: LevelSession
   editableProperties: ['multiplayer', 'players', 'code', 'completed', 'state',
                        'levelName', 'creatorName', 'levelID', 'screenshot',
-                       'chat', 'teamSpells','submitted']
+                       'chat', 'teamSpells', 'submitted', 'unsubscribed']
 
   getByRelationship: (req, res, args...) ->
     return @sendNotFoundError(res) unless args.length is 2 and args[1] is 'active'
diff --git a/server/levels/sessions/level_session_schema.coffee b/server/levels/sessions/level_session_schema.coffee
index ca8df4c05..da4395dec 100644
--- a/server/levels/sessions/level_session_schema.coffee
+++ b/server/levels/sessions/level_session_schema.coffee
@@ -122,7 +122,7 @@ _.extend LevelSessionSchema.properties,
 
   standardDeviation:
     type:'number'
-    default:25/3
+    default: 25/3
     minimum: 0
 
   totalScore:
@@ -139,25 +139,29 @@ _.extend LevelSessionSchema.properties,
 
   submittedCode:
     type: 'object'
-    
+
+  unsubscribed:
+    type: 'boolean'
+    description: 'Whether the player has opted out of receiving email updates about ladder rankings for this session.'
+
   numberOfWinsAndTies:
     type: 'number'
     default: 0
   numberOfLosses:
     type: 'number'
     default: 0
-    
+
   scoreHistory:
     type: 'array'
     title: 'Score History'
     description: 'A list of objects representing the score history of a session'
-    items: 
+    items:
       title: 'Score History Point'
       description: 'An array with the format [unix timestamp, totalScore]'
       type: 'array'
       items:
         type: 'number'
-      
+
 
   matches:
     type: 'array'
diff --git a/server/routes/mail.coffee b/server/routes/mail.coffee
index d629883d7..4ef488b67 100644
--- a/server/routes/mail.coffee
+++ b/server/routes/mail.coffee
@@ -42,8 +42,8 @@ handleLadderUpdate = (req, res) ->
   for daysAgo in emailDays
     # Get every session that was submitted in a 5-minute window after the time.
     startTime = getTimeFromDaysAgo daysAgo
-    endTime = startTime + 5 * 60 * 1000
-    #endTime = startTime + 1 * 60 * 60 * 1000
+    #endTime = startTime + 5 * 60 * 1000
+    endTime = startTime + 1 * 60 * 60 * 1000  # Debugging: make sure there's something to send
     findParameters = {submitted: true, submitDate: {$gt: new Date(startTime), $lte: new Date(endTime)}}
     # TODO: think about putting screenshots in the email
     selectString = "creator team levelName levelID totalScore matches submitted submitDate numberOfWinsAndTies numberOfLosses"
@@ -53,16 +53,19 @@ handleLadderUpdate = (req, res) ->
     do (daysAgo) ->
       query.exec (err, results) ->
         if err
-          log.error "Couldn't fetch ladder updates for", findParameters, "\nError: ", err
+          log.error "Couldn't fetch ladder updates for #{findParameters}\nError: #{err}"
           return errors.serverError res, "Ladder update email query failed: #{JSON.stringify(err)}"
+        log.info "Found #{results.length} ladder sessions to email updates about for #{daysAgo} day(s) ago."
         sendLadderUpdateEmail result, daysAgo for result in results
 
 sendLadderUpdateEmail = (session, daysAgo) ->
   User.findOne({_id: session.creator}).select("name email firstName lastName emailSubscriptions preferredLanguage").lean().exec (err, user) ->
     if err
-      log.error "Couldn't find user for", session.creator, "from session", session._id
+      log.error "Couldn't find user for #{session.creator} from session #{session._id}"
+      return
+    if not user.email or not ('notification' in user.emailSubscriptions)
+      log.info "Not sending email to #{user.email} #{user.name} because they only want emails about #{user.emailSubscriptions}"
       return
-    return unless user.email and 'notification' in user.emailSubscriptions
     name = if user.firstName and user.lastName then "#{user.firstName} #{user.lastName}" else user.name
     name = "Wizard" if not name or name is "Anoner"
 
@@ -72,7 +75,7 @@ sendLadderUpdateEmail = (session, daysAgo) ->
         email_id: sendwithus.templates.ladder_update_email
         recipient:
           #address: user.email
-          address: 'nick@codecombat.com'
+          address: 'nick@codecombat.com'  # Debugging
           name: name
         email_data:
           name: name
@@ -85,8 +88,9 @@ sendLadderUpdateEmail = (session, daysAgo) ->
           ladder_url: "http://codecombat.com/play/ladder/#{session.levelID}#my-matches"
           defeat: defeatContext
           victory: victoryContext
+      log.info "Sending ladder update email to #{context.recipient.address} with #{context.email_data.wins} wins and #{context.email_data.losses} since #{daysAgo} day(s) ago."
       sendwithus.api.send context, (err, result) ->
-        log.error "Error sending ladder update email:", err, 'result', result if err
+        log.error "Error sending ladder update email: #{err} with result #{result}" if err
 
     # Fetch the most recent defeat and victory, if there are any.
     # (We could look at strongest/weakest, but we'd have to fetch everyone, or denormalize more.)
diff --git a/server/sendwithus.coffee b/server/sendwithus.coffee
index c481893bf..659ce5ec8 100644
--- a/server/sendwithus.coffee
+++ b/server/sendwithus.coffee
@@ -11,4 +11,4 @@ options = { DEBUG: not config.isProduction }
 module.exports.api = new sendwithusAPI swuAPIKey, options
 module.exports.templates =
   welcome_email: 'utnGaBHuSU4Hmsi7qrAypU'
-  ladder_update_email: 'Xq3vSbDHXcjXfje7n2e7Eb'
+  ladder_update_email: 'JzaZxf39A4cKMxpPZUfWy4'
diff --git a/server/users/user_schema.coffee b/server/users/user_schema.coffee
index 7ae34c4af..d43a8d6b6 100644
--- a/server/users/user_schema.coffee
+++ b/server/users/user_schema.coffee
@@ -19,7 +19,7 @@ UserSchema = c.object {},
   music: {type: 'boolean', default: true}
   #autocastDelay, or more complex autocast options? I guess I'll see what I need when trying to hook up Scott's suggested autocast behavior
 
-  emailSubscriptions: c.array {uniqueItems: true, 'default': ['announcement']}, {'enum': emailSubscriptions}
+  emailSubscriptions: c.array {uniqueItems: true, 'default': ['announcement', 'notification']}, {'enum': emailSubscriptions}
 
   # server controlled
   permissions: c.array {'default': []}, c.shortString()
@@ -29,7 +29,7 @@ UserSchema = c.object {},
   mailChimp: {type: 'object'}
   hourOfCode: {type: 'boolean'}
   hourOfCodeComplete: {type: 'boolean'}
-  
+
   emailLower: c.shortString()
   nameLower: c.shortString()
   passwordHash: {type: 'string', maxLength: 256}
@@ -40,7 +40,7 @@ UserSchema = c.object {},
 
   #Internationalization stuff
   preferredLanguage: {type: 'string', default: 'en', 'enum': c.getLanguageCodeArray()}
-  
+
   signedCLA: c.date({title: 'Date Signed the CLA'})
   wizard: c.object {},
     colorConfig: c.object {additionalProperties: c.colorConfig()}

From c88c973fa714b14c51764e5669ea2c95c2aad34d Mon Sep 17 00:00:00 2001
From: Michael Schmatz <michaelschmatz@gmail.com>
Date: Mon, 10 Mar 2014 14:18:34 -0700
Subject: [PATCH 170/178] Added 2.6.0-rc1 to allowed Mongo versions

---
 bin/coco-mongodb | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/bin/coco-mongodb b/bin/coco-mongodb
index be4f285ed..4ff889493 100755
--- a/bin/coco-mongodb
+++ b/bin/coco-mongodb
@@ -71,7 +71,7 @@ def which(cmd, mode=os.F_OK | os.X_OK, path=None):
 
 
 current_directory = os.path.dirname(os.path.realpath(sys.argv[0]))
-allowedMongoVersions = ["v2.5.4","v2.5.5"]
+allowedMongoVersions = ["v2.5.4","v2.5.5","v2.6.0-rc1"]
 if which("mongod") and any(i in subprocess.check_output("mongod --version",shell=True) for i in allowedMongoVersions):
     mongo_executable = "mongod"
 else:

From 6b48577e56aed5eb3a71f6f80b781d9522c6a827 Mon Sep 17 00:00:00 2001
From: Darredevil <alex.darredevil@gmail.com>
Date: Tue, 11 Mar 2014 01:03:20 +0200
Subject: [PATCH 171/178] Update ro.coffee

---
 app/locale/ro.coffee | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/app/locale/ro.coffee b/app/locale/ro.coffee
index 531e3825f..5038a79d8 100644
--- a/app/locale/ro.coffee
+++ b/app/locale/ro.coffee
@@ -347,7 +347,7 @@ module.exports = nativeDescription: "limba română", englishDescription: "Roman
     nutshell_description: "Orice resurse vă punem la dispoziție în Editorul de Nivele puteți folosi liber cum vreți pentru a crea nivele. Dar ne rezervăm dreptul de a rezerva distribuția de nivele în sine (care sunt create pe codecombat.com) astfel încât să se poată percepe o taxă pentru ele pe vitor, dacă se va ajunge la așa ceva."
     canonical: "The English version of this document is the definitive, canonical version. If there are any discrepencies between translations, the English document takes precedence."
 
-  contribute:
+#  contribute:
 #    page_title: "Contributing"
 #    character_classes_title: "Character Classes"
 #    introduction_desc_intro: "We have high hopes for CodeCombat."

From 6ce65488aa727899336cff7585a893b37cee97f5 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 10 Mar 2014 20:22:25 -0700
Subject: [PATCH 172/178] Added simple rank history graph. Improving ladder
 update mail.

---
 app/templates/play/ladder/my_matches_tab.jade |  4 ++++
 app/views/play/ladder/my_matches_tab.coffee   |  9 +++++++++
 app/views/play/ladder_view.coffee             |  2 +-
 server/commons/Handler.coffee                 |  4 +++-
 server/routes/mail.coffee                     | 10 +++++++---
 5 files changed, 24 insertions(+), 5 deletions(-)

diff --git a/app/templates/play/ladder/my_matches_tab.jade b/app/templates/play/ladder/my_matches_tab.jade
index c70192221..e9bc0dd21 100644
--- a/app/templates/play/ladder/my_matches_tab.jade
+++ b/app/templates/play/ladder/my_matches_tab.jade
@@ -21,6 +21,10 @@ div#columns.row
                 span.ranked.hidden Submitted for Ranking
                 span.failed.hidden Failed to Rank
 
+        tr
+          th(colspan=4, style="color: #{team.primaryColor}")
+            img(src="https://chart.googleapis.com/chart?chs=450x125&cht=lxy&chco=#{team.chartColor}&chtt=Score+History&chts=#{team.chartColor},16,c&chf=a,s,000000FF&chls=2&chm=o,#{team.chartColor},0,4&chd=t:#{team.chartData}")
+
         tr
           th Result
           th Opponent
diff --git a/app/views/play/ladder/my_matches_tab.coffee b/app/views/play/ladder/my_matches_tab.coffee
index a094aef82..b7708c8df 100644
--- a/app/views/play/ladder/my_matches_tab.coffee
+++ b/app/views/play/ladder/my_matches_tab.coffee
@@ -76,6 +76,15 @@ module.exports = class MyMatchesTabView extends CocoView
       team.wins = _.filter(team.matches, {state: 'win'}).length
       team.ties = _.filter(team.matches, {state: 'tie'}).length
       team.losses = _.filter(team.matches, {state: 'loss'}).length
+      team.scoreHistory = team.session.get('scoreHistory')
+      team.chartColor = team.primaryColor.replace '#', ''
+      times = (s[0] for s in team.scoreHistory)
+      times = (100 * (t - times[0]) / (times[times.length - 1] - times[0]) for t in times)
+      scores = (s[1] for s in team.scoreHistory)
+      lowest = _.min scores
+      highest = _.max scores
+      scores = (100 * (s - lowest) / highest for s in scores)
+      team.chartData = times.join(',') + '|' + scores.join(',')
 
     ctx
 
diff --git a/app/views/play/ladder_view.coffee b/app/views/play/ladder_view.coffee
index 11283af13..b7cf49599 100644
--- a/app/views/play/ladder_view.coffee
+++ b/app/views/play/ladder_view.coffee
@@ -65,7 +65,7 @@ module.exports = class LadderView extends RootView
     return if @startsLoading
     @insertSubView(@ladderTab = new LadderTabView({}, @level, @sessions))
     @insertSubView(@myMatchesTab = new MyMatchesTabView({}, @level, @sessions))
-    @refreshInterval = setInterval(@fetchSessionsAndRefreshViews.bind(@), 10000)
+    @refreshInterval = setInterval(@fetchSessionsAndRefreshViews.bind(@), 10 * 1000)
     hash = document.location.hash[1..] if document.location.hash
     if hash and not (hash in ['my-matches', 'simulate', 'ladder'])
       @showPlayModal(hash) if @sessions.loaded
diff --git a/server/commons/Handler.coffee b/server/commons/Handler.coffee
index ada7acad3..4460b2d22 100644
--- a/server/commons/Handler.coffee
+++ b/server/commons/Handler.coffee
@@ -123,7 +123,9 @@ module.exports = class Handler
     # Keeping it simple for now and just allowing access to the first FETCH_LIMIT results.
     query = {'original': mongoose.Types.ObjectId(id)}
     sort = {'created': -1}
-    @modelClass.find(query).limit(FETCH_LIMIT).sort(sort).exec (err, results) =>
+    selectString = 'slug name version commitMessage created'  # Is this even working?
+    @modelClass.find(query).select(selectString).lean().limit(FETCH_LIMIT).sort(sort).exec (err, results) =>
+      return @sendDatabaseError(res, err) if err
       for doc in results
         return @sendUnauthorizedError(res) unless @hasAccessToDocument(req, doc)
       res.send(results)
diff --git a/server/routes/mail.coffee b/server/routes/mail.coffee
index 4ef488b67..5ed1fad22 100644
--- a/server/routes/mail.coffee
+++ b/server/routes/mail.coffee
@@ -42,8 +42,8 @@ handleLadderUpdate = (req, res) ->
   for daysAgo in emailDays
     # Get every session that was submitted in a 5-minute window after the time.
     startTime = getTimeFromDaysAgo daysAgo
-    #endTime = startTime + 5 * 60 * 1000
-    endTime = startTime + 1 * 60 * 60 * 1000  # Debugging: make sure there's something to send
+    endTime = startTime + 5 * 60 * 1000
+    #endTime = startTime + 1.5 * 60 * 60 * 1000  # Debugging: make sure there's something to send
     findParameters = {submitted: true, submitDate: {$gt: new Date(startTime), $lte: new Date(endTime)}}
     # TODO: think about putting screenshots in the email
     selectString = "creator team levelName levelID totalScore matches submitted submitDate numberOfWinsAndTies numberOfLosses"
@@ -63,9 +63,12 @@ sendLadderUpdateEmail = (session, daysAgo) ->
     if err
       log.error "Couldn't find user for #{session.creator} from session #{session._id}"
       return
-    if not user.email or not ('notification' in user.emailSubscriptions)
+    unless user.email and ('notification' in user.emailSubscriptions)
       log.info "Not sending email to #{user.email} #{user.name} because they only want emails about #{user.emailSubscriptions}"
       return
+    unless session.levelName
+      log.info "Not sending email to #{user.email} #{user.name} because the session had no levelName in it."
+      return
     name = if user.firstName and user.lastName then "#{user.firstName} #{user.lastName}" else user.name
     name = "Wizard" if not name or name is "Anoner"
 
@@ -84,6 +87,7 @@ sendLadderUpdateEmail = (session, daysAgo) ->
           losses: session.numberOfLosses
           total_score: Math.round(session.totalScore * 100)
           team: session.team
+          team_name: session.team[0].toUpperCase() + session.team.substr(1)
           level_name: session.levelName
           ladder_url: "http://codecombat.com/play/ladder/#{session.levelID}#my-matches"
           defeat: defeatContext

From 32baf2ae7988685baeb3585a5a63b5dc1b186fa6 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 10 Mar 2014 21:30:46 -0700
Subject: [PATCH 173/178] Improved my-matches rank history and name fetching,
 ladder update emails, ladder update unsubscribes.

---
 app/templates/play/ladder/my_matches_tab.jade |  7 ++--
 app/views/play/ladder/my_matches_tab.coffee   | 21 ++++++-----
 server/routes/auth.coffee                     | 37 ++++++++++++-------
 server/routes/mail.coffee                     | 29 ++++++++-------
 4 files changed, 55 insertions(+), 39 deletions(-)

diff --git a/app/templates/play/ladder/my_matches_tab.jade b/app/templates/play/ladder/my_matches_tab.jade
index e9bc0dd21..f40b4a04d 100644
--- a/app/templates/play/ladder/my_matches_tab.jade
+++ b/app/templates/play/ladder/my_matches_tab.jade
@@ -21,9 +21,10 @@ div#columns.row
                 span.ranked.hidden Submitted for Ranking
                 span.failed.hidden Failed to Rank
 
-        tr
-          th(colspan=4, style="color: #{team.primaryColor}")
-            img(src="https://chart.googleapis.com/chart?chs=450x125&cht=lxy&chco=#{team.chartColor}&chtt=Score+History&chts=#{team.chartColor},16,c&chf=a,s,000000FF&chls=2&chm=o,#{team.chartColor},0,4&chd=t:#{team.chartData}")
+        if team.chartData
+          tr
+            th(colspan=4, style="color: #{team.primaryColor}")
+              img(src="https://chart.googleapis.com/chart?chs=450x125&cht=lxy&chco=#{team.chartColor}&chtt=Score+History&chts=#{team.chartColor},16,c&chf=a,s,000000FF&chls=2&chm=o,#{team.chartColor},0,4&chd=t:#{team.chartData}")
 
         tr
           th Result
diff --git a/app/views/play/ladder/my_matches_tab.coffee b/app/views/play/ladder/my_matches_tab.coffee
index b7708c8df..24df7fbf1 100644
--- a/app/views/play/ladder/my_matches_tab.coffee
+++ b/app/views/play/ladder/my_matches_tab.coffee
@@ -35,7 +35,7 @@ module.exports = class MyMatchesTabView extends CocoView
       for session in @sessions.models
         for match in session.get('matches') or []
           opponent = match.opponents[0]
-          @nameMap[opponent.userID] = nameMap[opponent.userID]
+          @nameMap[opponent.userID] ?= nameMap[opponent.userID]
       @finishRendering()
 
     $.ajax('/db/user/-/names', {
@@ -76,15 +76,16 @@ module.exports = class MyMatchesTabView extends CocoView
       team.wins = _.filter(team.matches, {state: 'win'}).length
       team.ties = _.filter(team.matches, {state: 'tie'}).length
       team.losses = _.filter(team.matches, {state: 'loss'}).length
-      team.scoreHistory = team.session.get('scoreHistory')
-      team.chartColor = team.primaryColor.replace '#', ''
-      times = (s[0] for s in team.scoreHistory)
-      times = (100 * (t - times[0]) / (times[times.length - 1] - times[0]) for t in times)
-      scores = (s[1] for s in team.scoreHistory)
-      lowest = _.min scores
-      highest = _.max scores
-      scores = (100 * (s - lowest) / highest for s in scores)
-      team.chartData = times.join(',') + '|' + scores.join(',')
+      team.scoreHistory = team.session?.get('scoreHistory')
+      if team.scoreHistory?.length > 1
+        team.chartColor = team.primaryColor.replace '#', ''
+        times = (s[0] for s in team.scoreHistory)
+        times = ((100 * (t - times[0]) / (times[times.length - 1] - times[0])).toFixed(1) for t in times)
+        scores = (s[1] for s in team.scoreHistory)
+        lowest = _.min scores
+        highest = _.max scores
+        scores = (Math.round(100 * (s - lowest) / (highest - lowest)) for s in scores)
+        team.chartData = times.join(',') + '|' + scores.join(',')
 
     ctx
 
diff --git a/server/routes/auth.coffee b/server/routes/auth.coffee
index 2e6dbf72d..c845b28c2 100644
--- a/server/routes/auth.coffee
+++ b/server/routes/auth.coffee
@@ -2,6 +2,7 @@ authentication = require('passport')
 LocalStrategy = require('passport-local').Strategy
 User = require('../users/User')
 UserHandler = require('../users/user_handler')
+LevelSession = require '../levels/sessions/LevelSession'
 config = require '../../server_config'
 errors = require '../commons/errors'
 mail = require '../commons/mail'
@@ -21,16 +22,16 @@ module.exports.setup = (app) ->
         if passwordReset and password.toLowerCase() is passwordReset
           User.update {_id: user.get('_id')}, {passwordReset: ''}, {}, ->
           return done(null, user)
-          
+
         hash = User.hashPassword(password)
         unless user.get('passwordHash') is hash
-          return done(null, false, {message:'is wrong, wrong, wrong', property:'password'}) 
+          return done(null, false, {message:'is wrong, wrong, wrong', property:'password'})
         return done(null, user)
       )
   ))
   app.post '/auth/spy', (req, res, next) ->
     if req?.user?.isAdmin()
-      
+
       username = req.body.usernameLower
       emailLower = req.body.emailLower
       if emailLower
@@ -39,19 +40,19 @@ module.exports.setup = (app) ->
         query = {"nameLower":username}
       else
         return errors.badInput res, "You need to supply one of emailLower or username"
-        
+
       User.findOne query, (err, user) ->
         if err? then return errors.serverError res, "There was an error finding the specified user"
-        
+
         unless user then return errors.badInput res, "The specified user couldn't be found"
-          
+
         req.logIn user, (err) ->
           if err? then return errors.serverError res, "There was an error logging in with the specified"
           res.send(UserHandler.formatEntity(req, user))
           return res.end()
     else
       return errors.unauthorized res, "You must be an admin to enter espionage mode"
-      
+
   app.post('/auth/login', (req, res, next) ->
     authentication.authenticate('local', (err, user, info) ->
       return next(err) if err
@@ -87,11 +88,11 @@ module.exports.setup = (app) ->
     user.save((err) ->
       if err
         return @sendDatabaseError(res, err)
-  
+
       req.logIn(user, (err) ->
         if err
           return @sendDatabaseError(res, err)
-  
+
         if send
           return @sendSuccess(res, user)
         next() if next
@@ -110,7 +111,7 @@ module.exports.setup = (app) ->
     User.findOne({emailLower:req.body.email.toLowerCase()}).exec((err, user) ->
       if not user
         return errors.notFound(res, [{message:'not found.', property:'email'}])
-        
+
       user.set('passwordReset', Math.random().toString(36).slice(2,7).toUpperCase())
       user.save (err) =>
         return errors.serverError(res) if err
@@ -127,12 +128,22 @@ module.exports.setup = (app) ->
           return res.end()
     )
   )
-  
+
   app.get '/auth/unsubscribe', (req, res) ->
     email = req.query.email
     unless req.query.email
       return errors.badInput res, 'No email provided to unsubscribe.'
-      
+
+    if req.query.session
+      # Unsubscribe from just one session's notifications instead.
+      return LevelSession.findOne({_id: req.query.session}).exec (err, session) ->
+        return errors.serverError res, 'Could not unsubscribe: #{req.query.session}, #{req.query.email}: #{err}' if err
+        session.set 'unsubscribed', true
+        session.save (err) ->
+          return errors.serverError res, 'Database failure.' if err
+          res.send "Unsubscribed #{req.query.email} from CodeCombat emails for #{session.levelName} #{session.team} ladder updates. Sorry to see you go! <p><a href='/play/ladder/#{session.levelID}#my-matches'>Ladder preferences</a></p>"
+          res.end()
+
     User.findOne({emailLower:req.query.email.toLowerCase()}).exec (err, user) ->
       if not user
         return errors.notFound res, "No user found with email '#{req.query.email}'"
@@ -152,4 +163,4 @@ createMailOptions = (receiver, password) ->
     replyTo: config.mail.username
     subject: "[CodeCombat] Password Reset"
     text: "You can log into your account with: #{password}"
-#
\ No newline at end of file
+#
diff --git a/server/routes/mail.coffee b/server/routes/mail.coffee
index 5ed1fad22..a9430115b 100644
--- a/server/routes/mail.coffee
+++ b/server/routes/mail.coffee
@@ -46,7 +46,7 @@ handleLadderUpdate = (req, res) ->
     #endTime = startTime + 1.5 * 60 * 60 * 1000  # Debugging: make sure there's something to send
     findParameters = {submitted: true, submitDate: {$gt: new Date(startTime), $lte: new Date(endTime)}}
     # TODO: think about putting screenshots in the email
-    selectString = "creator team levelName levelID totalScore matches submitted submitDate numberOfWinsAndTies numberOfLosses"
+    selectString = "creator team levelName levelID totalScore matches submitted submitDate"
     query = LevelSession.find(findParameters)
       .select(selectString)
       .lean()
@@ -63,8 +63,8 @@ sendLadderUpdateEmail = (session, daysAgo) ->
     if err
       log.error "Couldn't find user for #{session.creator} from session #{session._id}"
       return
-    unless user.email and ('notification' in user.emailSubscriptions)
-      log.info "Not sending email to #{user.email} #{user.name} because they only want emails about #{user.emailSubscriptions}"
+    unless user.email and ('notification' in user.emailSubscriptions) and not session.unsubscribed
+      log.info "Not sending email to #{user.email} #{user.name} because they only want emails about #{user.emailSubscriptions} - session unsubscribed: #{session.unsubscribed}"
       return
     unless session.levelName
       log.info "Not sending email to #{user.email} #{user.name} because the session had no levelName in it."
@@ -72,23 +72,32 @@ sendLadderUpdateEmail = (session, daysAgo) ->
     name = if user.firstName and user.lastName then "#{user.firstName} #{user.lastName}" else user.name
     name = "Wizard" if not name or name is "Anoner"
 
+    # Fetch the most recent defeat and victory, if there are any.
+    # (We could look at strongest/weakest, but we'd have to fetch everyone, or denormalize more.)
+    matches = _.filter session.matches, (match) -> match.date >= (new Date() - 86400 * 1000 * daysAgo)
+    defeats = _.filter matches, (match) -> match.metrics.rank is 1 and match.opponents[0].metrics.rank is 0
+    victories = _.filter matches, (match) -> match.metrics.rank is 0
+    defeat = _.last defeats
+    victory = _.last victories
+
     sendEmail = (defeatContext, victoryContext) ->
       # TODO: do something with the preferredLanguage?
       context =
         email_id: sendwithus.templates.ladder_update_email
         recipient:
-          #address: user.email
-          address: 'nick@codecombat.com'  # Debugging
+          address: user.email
+          #address: 'nick@codecombat.com'  # Debugging
           name: name
         email_data:
           name: name
           days_ago: daysAgo
-          wins: session.numberOfWinsAndTies
-          losses: session.numberOfLosses
+          wins: victories.length
+          losses: defeats.length
           total_score: Math.round(session.totalScore * 100)
           team: session.team
           team_name: session.team[0].toUpperCase() + session.team.substr(1)
           level_name: session.levelName
+          session_id: session._id
           ladder_url: "http://codecombat.com/play/ladder/#{session.levelID}#my-matches"
           defeat: defeatContext
           victory: victoryContext
@@ -96,12 +105,6 @@ sendLadderUpdateEmail = (session, daysAgo) ->
       sendwithus.api.send context, (err, result) ->
         log.error "Error sending ladder update email: #{err} with result #{result}" if err
 
-    # Fetch the most recent defeat and victory, if there are any.
-    # (We could look at strongest/weakest, but we'd have to fetch everyone, or denormalize more.)
-    defeats = _.filter session.matches, (match) -> match.metrics.rank is 1 and match.opponents[0].metrics.rank is 0
-    victories = _.filter session.matches, (match) -> match.metrics.rank is 0
-    defeat = _.last defeats
-    victory = _.last victories
     urlForMatch = (match) ->
       "http://codecombat.com/play/level/#{session.levelID}?team=#{session.team}&session=#{session._id}&opponent=#{match.opponents[0].sessionID}"
 

From eac219a2be8b7277d2985629d2abad769bfbdb1b Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Mon, 10 Mar 2014 22:03:33 -0700
Subject: [PATCH 174/178] Inserted score history graph into ladder update
 emails.

---
 app/templates/play/ladder/my_matches_tab.jade |  2 +-
 app/views/play/ladder/my_matches_tab.coffee   |  1 +
 server/routes/mail.coffee                     | 19 +++++++++++++++++--
 3 files changed, 19 insertions(+), 3 deletions(-)

diff --git a/app/templates/play/ladder/my_matches_tab.jade b/app/templates/play/ladder/my_matches_tab.jade
index f40b4a04d..0eb83dddb 100644
--- a/app/templates/play/ladder/my_matches_tab.jade
+++ b/app/templates/play/ladder/my_matches_tab.jade
@@ -24,7 +24,7 @@ div#columns.row
         if team.chartData
           tr
             th(colspan=4, style="color: #{team.primaryColor}")
-              img(src="https://chart.googleapis.com/chart?chs=450x125&cht=lxy&chco=#{team.chartColor}&chtt=Score+History&chts=#{team.chartColor},16,c&chf=a,s,000000FF&chls=2&chm=o,#{team.chartColor},0,4&chd=t:#{team.chartData}")
+              img(src="https://chart.googleapis.com/chart?chs=450x125&cht=lxy&chco=#{team.chartColor}&chtt=Score%3A+#{team.currentScore}&chts=#{team.chartColor},16,c&chf=a,s,000000FF&chls=2&chm=o,#{team.chartColor},0,4&chd=t:#{team.chartData}")
 
         tr
           th Result
diff --git a/app/views/play/ladder/my_matches_tab.coffee b/app/views/play/ladder/my_matches_tab.coffee
index 24df7fbf1..660114111 100644
--- a/app/views/play/ladder/my_matches_tab.coffee
+++ b/app/views/play/ladder/my_matches_tab.coffee
@@ -78,6 +78,7 @@ module.exports = class MyMatchesTabView extends CocoView
       team.losses = _.filter(team.matches, {state: 'loss'}).length
       team.scoreHistory = team.session?.get('scoreHistory')
       if team.scoreHistory?.length > 1
+        team.currentScore = Math.round team.scoreHistory[team.scoreHistory.length - 1][1] * 100
         team.chartColor = team.primaryColor.replace '#', ''
         times = (s[0] for s in team.scoreHistory)
         times = ((100 * (t - times[0]) / (times[times.length - 1] - times[0])).toFixed(1) for t in times)
diff --git a/server/routes/mail.coffee b/server/routes/mail.coffee
index a9430115b..5ad2dfcd4 100644
--- a/server/routes/mail.coffee
+++ b/server/routes/mail.coffee
@@ -46,7 +46,7 @@ handleLadderUpdate = (req, res) ->
     #endTime = startTime + 1.5 * 60 * 60 * 1000  # Debugging: make sure there's something to send
     findParameters = {submitted: true, submitDate: {$gt: new Date(startTime), $lte: new Date(endTime)}}
     # TODO: think about putting screenshots in the email
-    selectString = "creator team levelName levelID totalScore matches submitted submitDate"
+    selectString = "creator team levelName levelID totalScore matches submitted submitDate scoreHistory"
     query = LevelSession.find(findParameters)
       .select(selectString)
       .lean()
@@ -69,7 +69,7 @@ sendLadderUpdateEmail = (session, daysAgo) ->
     unless session.levelName
       log.info "Not sending email to #{user.email} #{user.name} because the session had no levelName in it."
       return
-    name = if user.firstName and user.lastName then "#{user.firstName} #{user.lastName}" else user.name
+    name = if user.firstName and user.lastName then "#{user.firstName}" else user.name
     name = "Wizard" if not name or name is "Anoner"
 
     # Fetch the most recent defeat and victory, if there are any.
@@ -99,6 +99,7 @@ sendLadderUpdateEmail = (session, daysAgo) ->
           level_name: session.levelName
           session_id: session._id
           ladder_url: "http://codecombat.com/play/ladder/#{session.levelID}#my-matches"
+          score_history_graph_url: getScoreHistoryGraphURL session, daysAgo
           defeat: defeatContext
           victory: victoryContext
       log.info "Sending ladder update email to #{context.recipient.address} with #{context.email_data.wins} wins and #{context.email_data.losses} since #{daysAgo} day(s) ago."
@@ -131,6 +132,20 @@ sendLadderUpdateEmail = (session, daysAgo) ->
     else
       onFetchedDefeatedOpponent null, null
 
+getScoreHistoryGraphURL = (session, daysAgo) ->
+  # Totally duplicated in My Matches tab for now until we figure out what we're doing.
+  since = new Date() - 86400 * 1000 * daysAgo
+  scoreHistory = (s for s in session.scoreHistory ? [] when s[0] >= since)
+  return '' unless scoreHistory.length > 1
+  times = (s[0] for s in scoreHistory)
+  times = ((100 * (t - times[0]) / (times[times.length - 1] - times[0])).toFixed(1) for t in times)
+  scores = (s[1] for s in scoreHistory)
+  lowest = _.min scores
+  highest = _.max scores
+  scores = (Math.round(100 * (s - lowest) / (highest - lowest)) for s in scores)
+  currentScore = Math.round scoreHistory[scoreHistory.length - 1][1] * 100
+  chartData = times.join(',') + '|' + scores.join(',')
+  "https://chart.googleapis.com/chart?chs=600x75&cht=lxy&chtt=Score%3A+#{currentScore}&chts=222222,12,r&chf=a,s,000000FF&chls=2&chd=t:#{chartData}"
 
 handleMailchimpWebHook = (req, res) ->
   post = req.body

From 4062bba245829b185827eda47d2835bcbaa11e35 Mon Sep 17 00:00:00 2001
From: Michael Schmatz <schmatz@umich.edu>
Date: Tue, 11 Mar 2014 13:12:40 -0700
Subject: [PATCH 175/178] Created initial spectate level view

---
 app/styles/play/spectate.sass       | 152 +++++++++++++++++++++
 app/templates/play/spectate.jade    |  16 ++-
 app/views/play/spectate_view.coffee | 199 ++++++++++++++++++++++------
 3 files changed, 323 insertions(+), 44 deletions(-)
 create mode 100644 app/styles/play/spectate.sass

diff --git a/app/styles/play/spectate.sass b/app/styles/play/spectate.sass
new file mode 100644
index 000000000..9de885caa
--- /dev/null
+++ b/app/styles/play/spectate.sass
@@ -0,0 +1,152 @@
+@import "app/styles/bootstrap/mixins"
+@import "app/styles/mixins"
+
+#spectate-level-view
+  margin: 0 auto
+  @include user-select(none)
+
+  .level-content
+    position: relative
+
+  #canvas-wrapper
+    width: 55%
+    position: relative
+
+  canvas#surface
+    background-color: #ddd
+    width: 100%
+    display: block
+    z-index: 1
+
+
+  //max-width: 1680px   // guideline, but for now let's let it stretch out
+  min-width: 1024px
+  position: relative
+
+  #code-area
+    @include box-sizing(border-box)
+    padding: 10px 1%
+    width: 45%
+    background: transparent url(/images/level/wood_texture.png)
+    background-size: 100% 100%
+    position: absolute
+    right: 0
+    top: 0px
+    bottom: 0
+
+  #pointer
+    position: absolute
+    left: 0
+    top: 0
+    height: 100px
+    opacity: 0.0
+    pointer-events: none
+    z-index: 10
+
+  // Level Docs
+  .ui-effects-transfer
+    border: 2px dotted gray
+
+  .modal
+    img
+      float: right
+
+    img.diagram
+      float: none
+
+  #multiplayer-join-link
+    font-size: 12px
+
+  #level-done-button
+    position: absolute
+    right: 46%
+    top: 43px
+    @include box-shadow(4px 4px 15px black)
+
+  // Custom Buttons
+  .btn.banner
+    @include banner-button(#FFF, #333)
+    @include box-shadow(2px 2px 2px rgba(0, 0, 0, 0.5))
+    border: 1px solid black
+    text-shadow: none
+
+    $buttonConfig: 'primary' #6CA8EA, 'info' #71AACC, 'success' #90B236, 'warning' #CD6800, 'danger' #B43C20, 'inverse' #3A537F
+    @each $tuple in $buttonConfig
+      &.btn-#{nth($tuple, 1)}
+        @include banner-button(nth($tuple, 2), #FFF)
+
+  .footer .footer-link-text a
+    @include opacity(0.75)
+    @include transition(opacity .10s linear)
+
+    &:hover, &:active
+      @include opacity(1)
+
+  $GI: 0.5  // gradient intensity; can tweak this 0-1
+
+  .gradient
+    position: absolute
+    z-index: 10
+
+  #code-area-gradient
+    top: 0px
+    width: 3px
+    background: linear-gradient(to right, rgba(0,0,0,0) 0%,rgba(0,0,0,$GI) 100%)
+    left: -3px
+    bottom: 0
+
+  #hud-top-gradient
+    top: -32px
+    background: linear-gradient(to bottom, rgba(0,0,0,0) 0%,rgba(0,0,0,0.8*$GI) 100%)
+    left: 0
+    right: 0
+    bottom: 0
+    height: 3px
+
+  #canvas-left-gradient
+    left: 0px
+    width: 5px
+    background: linear-gradient(to left, rgba(0,0,0,0) 0%,rgba(0,0,0,0.8*$GI) 100%)
+    bottom: -30px
+    top: 0
+
+  #canvas-top-gradient
+    top: 0
+    height: 5px
+    left: 0
+    right: 0
+    background: linear-gradient(to top, rgba(0,0,0,0) 0%,rgba(0,0,0,0.8*$GI) 100%)
+
+  #hud-left-gradient
+    background: linear-gradient(to right, rgba(0,0,0,$GI) 0%,rgba(0,0,0,0) 100%)
+    left: 0
+    top: 0
+    height: 100%
+    width: 2%
+
+  #hud-right-gradient
+    background: linear-gradient(to right, rgba(0,0,0,0) 0%,rgba(0,0,0,$GI) 100%)
+    right: 0
+    position: absolute
+    top: 0
+    height: 100%
+    width: 2%
+
+  .footer
+    @media screen and (min-aspect-ratio: 17/10)
+      display: none
+
+    &:not(:hover)
+      @include opacity(0.6)
+
+    .hour-of-code-explanation
+      margin-top: 5px
+      color: white
+      font-size: 12px
+
+      &:not(:hover)
+        @include opacity(0.75)
+
+      a
+        color: white
+        text-decoration: underline
diff --git a/app/templates/play/spectate.jade b/app/templates/play/spectate.jade
index 9be37b46a..cfaba9234 100644
--- a/app/templates/play/spectate.jade
+++ b/app/templates/play/spectate.jade
@@ -1,16 +1,22 @@
 .level-content
   #control-bar-view
-
   #canvas-wrapper
-    canvas(width=924, height=589)#surface
+    canvas(width=1848, height=1178)#surface
     #canvas-left-gradient.gradient
     #canvas-top-gradient.gradient
-  #goals-view.hide
-  #gold-view.hide.expanded
+  #gold-view.secret.expanded
   #level-chat-view
   #playback-view
   #thang-hud
 .footer
   .content
     p(class='footer-link-text')
-      a(title='Contact', tabindex=-1, data-toggle="coco-modal", data-target="modal/contact", data-i18n="nav.contact") Contact
+      a(title='Send CodeCombat a message', tabindex=-1, data-toggle="coco-modal", data-target="modal/contact", data-i18n="nav.contact") Contact
+    if explainHourOfCode
+      // Does not show up unless lang is en-US.
+      div.hour-of-code-explanation
+        | The 'Hour of Code' is a nationwide initiative by 
+        a(href="http://csedweek.org") Computer Science Education Week
+        | and 
+        a(href="http://code.org") Code.org
+        | to introduce millions of students to one hour of computer science and computer programming.
\ No newline at end of file
diff --git a/app/views/play/spectate_view.coffee b/app/views/play/spectate_view.coffee
index e2686e90d..17dbcf6d3 100644
--- a/app/views/play/spectate_view.coffee
+++ b/app/views/play/spectate_view.coffee
@@ -5,7 +5,6 @@ ThangType = require 'models/ThangType'
 
 # temp hard coded data
 World = require 'lib/world/world'
-docs = require 'lib/world/docs'
 
 # tools
 Surface = require 'lib/surface/Surface'
@@ -17,7 +16,9 @@ LevelLoader = require 'lib/LevelLoader'
 LevelSession = require 'models/LevelSession'
 Level = require 'models/Level'
 LevelComponent = require 'models/LevelComponent'
+Article = require 'models/Article'
 Camera = require 'lib/surface/Camera'
+AudioPlayer = require 'lib/AudioPlayer'
 
 # subviews
 TomeView = require './level/tome/tome_view'
@@ -34,8 +35,6 @@ LoadingScreen = require 'lib/LoadingScreen'
 
 PROFILE_ME = false
 
-PlayLevelView = require './level_view'
-
 module.exports = class SpectateLevelView extends View
   id: 'spectate-level-view'
   template: template
@@ -46,6 +45,8 @@ module.exports = class SpectateLevelView extends View
 
   subscriptions:
     'level-set-volume': (e) -> createjs.Sound.setVolume(e.volume)
+    'level-show-victory': 'onShowVictory'
+    'restart-level': 'onRestartLevel'
     'level-highlight-dom': 'onHighlightDom'
     'end-level-highlight-dom': 'onEndHighlight'
     'level-focus-dom': 'onFocusDom'
@@ -53,33 +54,33 @@ module.exports = class SpectateLevelView extends View
     'level-enable-controls': 'onEnableControls'
     'god:new-world-created': 'onNewWorld'
     'god:infinite-loop': 'onInfiniteLoop'
+    'level-reload-from-data': 'onLevelReloadFromData'
+    'play-next-level': 'onPlayNextLevel'
     'edit-wizard-settings': 'showWizardSettingsModal'
     'surface:world-set-up': 'onSurfaceSetUpNewWorld'
     'level:session-will-save': 'onSessionWillSave'
     'level:set-team': 'setTeam'
+    'god:new-world-created': 'loadSoundsForWorld'
 
   events:
     'click #level-done-button': 'onDonePressed'
 
+  shortcuts:
+    'ctrl+s': 'onCtrlS'
 
   constructor: (options, @levelID) ->
     console.profile?() if PROFILE_ME
     super options
-    console.log @levelID
-
-    @ogreSessionID = @getQueryVariable 'ogres'
-    @humanSessionID = @getQueryVariable 'humans'
 
+    @sessionID = @getQueryVariable 'session'
 
     $(window).on('resize', @onWindowResize)
-    @supermodel.once 'error', =>
-      msg = $.i18n.t('play_level.level_load_error', defaultValue: "Level could not be loaded.")
-      @$el.html('<div class="alert">' + msg + '</div>')
-
+    @supermodel.once 'error', @onLevelLoadError
 
     @load()
 
-
+  onLevelLoadError: (e) =>
+    application.router.navigate "/play?not_found=#{@levelID}", {trigger: true}
 
   setLevel: (@level, @supermodel) ->
     @god?.level = @level.serialize @supermodel
@@ -91,7 +92,8 @@ module.exports = class SpectateLevelView extends View
 
   load: ->
     @levelLoader = new LevelLoader supermodel: @supermodel, levelID: @levelID, sessionID: @sessionID, opponentSessionID: @getQueryVariable('opponent'), team: @getQueryVariable("team")
-    @levelLoader.once 'loaded-all', @onLevelLoaderLoaded
+    @levelLoader.once 'loaded-all', @onLevelLoaderLoaded, @
+    @levelLoader.on 'progress', @onLevelLoaderProgressChanged, @
     @god = new God()
 
   getRenderData: ->
@@ -103,30 +105,83 @@ module.exports = class SpectateLevelView extends View
     window.onPlayLevelViewLoaded? @  # still a hack
     @loadingScreen = new LoadingScreen(@$el.find('canvas')[0])
     @loadingScreen.show()
+    @$el.find('#level-done-button').hide()
     super()
 
-  onLevelLoaderLoaded: =>
-    #needs editing
-    @session = @levelLoader.session
-    @world = @levelLoader.world
-    @level = @levelLoader.level
-    @levelLoader.destroy()
-    @levelLoader = null
+  onLevelLoaderProgressChanged: ->
+    return if @seenDocs
+    return unless showFrequency = @levelLoader.level.get('showGuide')
+    session = @levelLoader.session
+    diff = new Date().getTime() - new Date(session.get('created')).getTime()
+    return if showFrequency is 'first-time' and diff > (5 * 60 * 1000)
+    return unless @levelLoader.level.loaded
+    articles = @levelLoader.supermodel.getModels Article
+    for article in articles
+      return unless article.loaded
+    @showGuide()
+
+  showGuide: ->
+    @seenDocs = true
+    DocsModal = require './level/modal/docs_modal'
+    options = {docs: @levelLoader.level.get('documentation'), supermodel: @supermodel}
+    @openModalView(new DocsModal(options), true)
+    Backbone.Mediator.subscribeOnce 'modal-closed', @onLevelLoaderLoaded, @
+    return true
+
+  onLevelLoaderLoaded: ->
+    return unless @levelLoader.progress() is 1 # double check, since closing the guide may trigger this early
+    # Save latest level played in local storage
+    if window.currentModal and not window.currentModal.destroyed
+      @loadingScreen.showReady()
+      return Backbone.Mediator.subscribeOnce 'modal-closed', @onLevelLoaderLoaded, @
+
+    localStorage["lastLevel"] = @levelID if localStorage?
+    @grabLevelLoaderData()
+    team = @getQueryVariable("team") ? @world.teamForPlayer(0)
+    @loadOpponentTeam(team)
     @loadingScreen.destroy()
     @god.level = @level.serialize @supermodel
     @god.worldClassMap = @world.classMap
-    #@setTeam @world.teamForPlayer _.size @session.get 'players'   # TODO: players aren't initialized yet?
-    @setTeam @getQueryVariable("team") ? @world.teamForPlayer(0)
+    @setTeam team
     @initSurface()
     @initGoalManager()
     @initScriptManager()
-    @insertSubviews()
+    @insertSubviews ladderGame: @otherSession?
     @initVolume()
-    @session.on 'change:multiplayer', @onMultiplayerChanged, @
     @originalSessionState = _.cloneDeep(@session.get('state'))
     @register()
     @controlBar.setBus(@bus)
     @surface.showLevel()
+    if @otherSession
+      # TODO: colorize name and cloud by team, colorize wizard by user's color config
+      @surface.createOpponentWizard id: @otherSession.get('creator'), name: @otherSession.get('creatorName'), team: @otherSession.get('team')
+
+  grabLevelLoaderData: ->
+    @session = @levelLoader.session
+    @world = @levelLoader.world
+    @level = @levelLoader.level
+    @otherSession = @levelLoader.opponentSession
+    @levelLoader.destroy()
+    @levelLoader = null
+
+  loadOpponentTeam: (myTeam) ->
+    opponentSpells = []
+    for spellTeam, spells of @session.get('teamSpells') ? @otherSession?.get('teamSpells') ? {}
+      continue if spellTeam is myTeam or not myTeam
+      opponentSpells = opponentSpells.concat spells
+
+    opponentCode = @otherSession?.get('submittedCode') or {}
+    myCode = @session.get('code') or {}
+    for spell in opponentSpells
+      [thang, spell] = spell.split '/'
+      c = opponentCode[thang]?[spell]
+      myCode[thang] ?= {}
+      if c then myCode[thang][spell] = c else delete myCode[thang][spell]
+    @session.set('code', myCode)
+    if @session.get('multiplayer') and @otherSession?
+      # For now, ladderGame will disallow multiplayer, because session code combining doesn't play nice yet.
+      @session.set 'multiplayer', false
+
 
   onSupermodelLoadedOne: =>
     @modelsLoaded ?= 0
@@ -142,38 +197,66 @@ module.exports = class SpectateLevelView extends View
     ctx.clearRect(0, 0, canvas.width, canvas.height)
     ctx.fillText("Loaded #{@modelsLoaded} thingies",50,50)
 
-  insertSubviews: ->
-    #needs editing
-    @insertSubView @tome = new TomeView levelID: @levelID, session: @session, thangs: @world.thangs, supermodel: @supermodel
+  insertSubviews: (subviewOptions) ->
+    @insertSubView @tome = new TomeView levelID: @levelID, session: @session, thangs: @world.thangs, supermodel: @supermodel, ladderGame: subviewOptions.ladderGame
     @insertSubView new PlaybackView {}
     @insertSubView new GoalsView {}
     @insertSubView new GoldView {}
     @insertSubView new HUDView {}
     @insertSubView new ChatView levelID: @levelID, sessionID: @session.id, session: @session
     worldName = @level.get('i18n')?[me.lang()]?.name ? @level.get('name')
-    @controlBar = @insertSubView new ControlBarView {worldName: worldName, session: @session, level: @level, supermodel: @supermodel, playableTeams: @world.playableTeams}
+    @controlBar = @insertSubView new ControlBarView {worldName: worldName, session: @session, level: @level, supermodel: @supermodel, playableTeams: @world.playableTeams, ladderGame: subviewOptions.ladderGame}
   #Backbone.Mediator.publish('level-set-debug', debug: true) if me.displayName() is 'Nick!'
 
   afterInsert: ->
     super()
+    @showWizardSettingsModal() if not me.get('name')
 
+  # callbacks
+
+  onCtrlS: (e) ->
+    e.preventDefault()
+
+  onLevelReloadFromData: (e) ->
+    isReload = Boolean @world
+    @setLevel e.level, e.supermodel
+    if isReload
+      @scriptManager.setScripts(e.level.get('scripts'))
+      Backbone.Mediator.publish 'tome:cast-spell'  # a bit hacky
 
   onWindowResize: (s...) ->
     $('#pointer').css('opacity', 0.0)
 
-  onDisableControls: (e) =>
+  onDisableControls: (e) ->
     return if e.controls and not ('level' in e.controls)
     @shortcutsEnabled = false
     @wasFocusedOn = document.activeElement
     $('body').focus()
 
-  onEnableControls: (e) =>
+  onEnableControls: (e) ->
     return if e.controls? and not ('level' in e.controls)
     @shortcutsEnabled = true
     $(@wasFocusedOn).focus() if @wasFocusedOn
     @wasFocusedOn = null
 
-  onDonePressed: => @showVictory()
+  onDonePressed: -> @showVictory()
+
+  onShowVictory: (e) ->
+    $('#level-done-button').show()
+    @showVictory() if e.showModal
+    setTimeout(@preloadNextLevel, 3000)
+
+  showVictory: ->
+    options = {level: @level, supermodel: @supermodel, session:@session}
+    docs = new VictoryModal(options)
+    @openModalView(docs)
+    window.tracker?.trackEvent 'Saw Victory', level: @world.name, label: @world.name
+
+  onRestartLevel: ->
+    @tome.reloadAllCode()
+    Backbone.Mediator.publish 'level:restarted'
+    $('#level-done-button', @$el).hide()
+    window.tracker?.trackEvent 'Confirmed Restart', level: @world.name, label: @world.name
 
   onNewWorld: (e) ->
     @world = e.world
@@ -183,13 +266,21 @@ module.exports = class SpectateLevelView extends View
     @openModalView new InfiniteLoopModal()
     window.tracker?.trackEvent 'Saw Initial Infinite Loop', level: @world.name, label: @world.name
 
+  onPlayNextLevel: ->
+    nextLevel = @getNextLevel()
+    nextLevelID = nextLevel.get('slug') or nextLevel.id
+    url = "/play/level/#{nextLevelID}"
+    Backbone.Mediator.publish 'router:navigate', {
+      route: url,
+      viewClass: PlayLevelView,
+      viewArgs: [{supermodel:@supermodel}, nextLevelID]}
 
   getNextLevel: ->
     nextLevelOriginal = @level.get('nextLevel')?.original
     levels = @supermodel.getModels(Level)
     return l for l in levels when l.get('original') is nextLevelOriginal
 
-  onHighlightDom: (e) =>
+  onHighlightDom: (e) ->
     if e.delay
       delay = e.delay
       delete e.delay
@@ -243,19 +334,25 @@ module.exports = class SpectateLevelView extends View
     ), 1)
 
 
-  animatePointer: =>
+  animatePointer: ->
     pointer = $('#pointer')
     pointer.css('transition', 'all 0.6s ease-out')
     pointer.css('transform', "rotate(#{@pointerRotation}rad) translate(-3px, #{@pointerRadialDistance-50}px)")
     setTimeout((=>
       pointer.css('transform', "rotate(#{@pointerRotation}rad) translate(-3px, #{@pointerRadialDistance}px)").css('transition', 'all 0.4s ease-in')), 800)
 
-  onFocusDom: (e) => $(e.selector).focus()
+  onFocusDom: (e) -> $(e.selector).focus()
 
-  onEndHighlight: =>
+  onEndHighlight: ->
     $('#pointer').css('opacity', 0.0)
     clearInterval(@pointerInterval)
 
+  onMultiplayerChanged: (e) ->
+    if @session.get('multiplayer')
+      @bus.connect()
+    else
+      @bus.removeFirebaseData =>
+        @bus.disconnect()
 
   # initialization
 
@@ -273,7 +370,7 @@ module.exports = class SpectateLevelView extends View
     @surface.camera.zoomTo({x:0, y:0}, 0.1, 0)
 
   initGoalManager: ->
-    @goalManager = new GoalManager(@world)
+    @goalManager = new GoalManager(@world, @level.get('goals'))
     @god.goalManager = @goalManager
 
   initScriptManager: ->
@@ -297,11 +394,18 @@ module.exports = class SpectateLevelView extends View
     if state.playing?
       Backbone.Mediator.publish 'level-set-playing', { playing: state.playing }
 
+  preloadNextLevel: =>
+    # TODO: Loading models in the middle of gameplay causes stuttering. Most of the improvement in loading time is simply from passing the supermodel from this level to the next, but if we can find a way to populate the level early without it being noticeable, that would be even better.
+#    return if @destroyed
+#    return if @preloaded
+#    nextLevel = @getNextLevel()
+#    @supermodel.populateModel nextLevel
+#    @preloaded = true
 
   register: ->
     @bus = LevelBus.get(@levelID, @session.id)
     @bus.setSession(@session)
-    @bus.setTeamSpellMap @tome.teamSpellMap
+    @bus.setSpells @tome.spells
     @bus.connect() if @session.get('multiplayer')
 
   onSessionWillSave: (e) ->
@@ -319,7 +423,20 @@ module.exports = class SpectateLevelView extends View
     me.team = team
     Backbone.Mediator.publish 'level:team-set', team: team
 
+  # Dynamic sound loading
+
+  loadSoundsForWorld: (e) ->
+    return if @headless
+    world = e.world
+    thangTypes = @supermodel.getModels(ThangType)
+    for [spriteName, message] in world.thangDialogueSounds()
+      continue unless thangType = _.find thangTypes, (m) -> m.get('name') is spriteName
+      continue unless sound = AudioPlayer.soundForDialogue message, thangType.get('soundTriggers')
+      AudioPlayer.preloadSoundReference sound
+
   destroy: ->
+    @supermodel?.off 'error', @onLevelLoadError
+    @levelLoader?.off 'loaded-all', @onLevelLoaderLoaded
     @levelLoader?.destroy()
     @surface?.destroy()
     @god?.destroy()
@@ -327,10 +444,14 @@ module.exports = class SpectateLevelView extends View
     @scriptManager?.destroy()
     $(window).off('resize', @onWindowResize)
     delete window.world # not sure where this is set, but this is one way to clean it up
-
     clearInterval(@pointerInterval)
     @bus?.destroy()
     #@instance.save() unless @instance.loading
     console.profileEnd?() if PROFILE_ME
-    @session.off 'change:multiplayer', @onMultiplayerChanged, @
+    @session?.off 'change:multiplayer', @onMultiplayerChanged, @
+    @onLevelLoadError = null
+    @onLevelLoaderLoaded = null
+    @onSupermodelLoadedOne = null
+    @preloadNextLevel = null
+    @saveScreenshot = null
     super()

From bba3c78107c7bc09633d013b95ee4aaae833e882 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Tue, 11 Mar 2014 13:20:52 -0700
Subject: [PATCH 176/178] Ties are not victories in the emails.

---
 app/templates/play/ladder/my_matches_tab.jade | 2 +-
 server/routes/mail.coffee                     | 2 +-
 2 files changed, 2 insertions(+), 2 deletions(-)

diff --git a/app/templates/play/ladder/my_matches_tab.jade b/app/templates/play/ladder/my_matches_tab.jade
index 0eb83dddb..3827bd3b1 100644
--- a/app/templates/play/ladder/my_matches_tab.jade
+++ b/app/templates/play/ladder/my_matches_tab.jade
@@ -24,7 +24,7 @@ div#columns.row
         if team.chartData
           tr
             th(colspan=4, style="color: #{team.primaryColor}")
-              img(src="https://chart.googleapis.com/chart?chs=450x125&cht=lxy&chco=#{team.chartColor}&chtt=Score%3A+#{team.currentScore}&chts=#{team.chartColor},16,c&chf=a,s,000000FF&chls=2&chm=o,#{team.chartColor},0,4&chd=t:#{team.chartData}")
+              img(src="https://chart.googleapis.com/chart?chs=450x125&cht=lxy&chco=#{team.chartColor}&chtt=Score%3A+#{team.currentScore}&chts=#{team.chartColor},16,r&chf=a,s,000000FF&chls=2&chm=o,#{team.chartColor},0,4&chd=t:#{team.chartData}")
 
         tr
           th Result
diff --git a/server/routes/mail.coffee b/server/routes/mail.coffee
index 5ad2dfcd4..5a46a9006 100644
--- a/server/routes/mail.coffee
+++ b/server/routes/mail.coffee
@@ -76,7 +76,7 @@ sendLadderUpdateEmail = (session, daysAgo) ->
     # (We could look at strongest/weakest, but we'd have to fetch everyone, or denormalize more.)
     matches = _.filter session.matches, (match) -> match.date >= (new Date() - 86400 * 1000 * daysAgo)
     defeats = _.filter matches, (match) -> match.metrics.rank is 1 and match.opponents[0].metrics.rank is 0
-    victories = _.filter matches, (match) -> match.metrics.rank is 0
+    victories = _.filter matches, (match) -> match.metrics.rank is 0 and match.opponents[0].metrics.rank is 1
     defeat = _.last defeats
     victory = _.last victories
 

From 852b1c97aca6655f24799695200617f6345da280 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Tue, 11 Mar 2014 13:59:12 -0700
Subject: [PATCH 177/178] Added idle.js so we can stop refreshing the page
 forever on ladder views when they're not looking.

---
 app/application.coffee            |   9 ++-
 app/views/play/ladder_view.coffee |   4 +-
 vendor/scripts/idle.js            | 126 ++++++++++++++++++++++++++++++
 3 files changed, 135 insertions(+), 4 deletions(-)
 create mode 100644 vendor/scripts/idle.js

diff --git a/app/application.coffee b/app/application.coffee
index 6b0c7f80b..860aee96e 100644
--- a/app/application.coffee
+++ b/app/application.coffee
@@ -44,8 +44,13 @@ Application = initialize: ->
   }, (t) =>
     @router = new Router()
     @router.subscribe()
-    Object.freeze this if typeof Object.freeze is 'function'
-    @router = Router
+    @idleTracker = new Idle
+      onAway: => @userIsIdle = true
+      onAwayBack: => @userIsIdle = false
+      onHidden: => @userIsIdle = true
+      onVisible: => @userIsIdle = false
+      awayTimeout: 5 * 60 * 1000
+    @idleTracker.start()
 
 module.exports = Application
 window.application = Application
diff --git a/app/views/play/ladder_view.coffee b/app/views/play/ladder_view.coffee
index b7cf49599..e12304f96 100644
--- a/app/views/play/ladder_view.coffee
+++ b/app/views/play/ladder_view.coffee
@@ -4,6 +4,7 @@ Simulator = require 'lib/simulator/Simulator'
 LevelSession = require 'models/LevelSession'
 CocoCollection = require 'models/CocoCollection'
 {teamDataFromLevel} = require './ladder/utils'
+application = require 'application'
 
 LadderTabView = require './ladder/ladder_tab'
 MyMatchesTabView = require './ladder/my_matches_tab'
@@ -74,12 +75,11 @@ module.exports = class LadderView extends RootView
     @sessions.fetch({"success": @refreshViews})
 
   refreshViews: =>
-    return if @destroyed
+    return if @destroyed or application.userIsIdle
     @ladderTab.refreshLadder()
     @myMatchesTab.refreshMatches()
     console.log "refreshed views!"
 
-
   # Simulations
 
   onSimulateAllButtonClick: (e) ->
diff --git a/vendor/scripts/idle.js b/vendor/scripts/idle.js
new file mode 100644
index 000000000..702c8ecd2
--- /dev/null
+++ b/vendor/scripts/idle.js
@@ -0,0 +1,126 @@
+// https://github.com/shawnmclean/Idle.js
+(function() {
+  "use strict";
+  var Idle;
+
+  Idle = {};
+
+  Idle = (function() {
+    Idle.isAway = false;
+
+    Idle.awayTimeout = 3000;
+
+    Idle.awayTimestamp = 0;
+
+    Idle.awayTimer = null;
+
+    Idle.onAway = null;
+
+    Idle.onAwayBack = null;
+
+    Idle.onVisible = null;
+
+    Idle.onHidden = null;
+
+    function Idle(options) {
+      var activeMethod, activity;
+
+      if (options) {
+        this.awayTimeout = parseInt(options.awayTimeout, 10);
+        this.onAway = options.onAway;
+        this.onAwayBack = options.onAwayBack;
+        this.onVisible = options.onVisible;
+        this.onHidden = options.onHidden;
+      }
+      activity = this;
+      activeMethod = function() {
+        return activity.onActive();
+      };
+      window.onclick = activeMethod;
+      window.onmousemove = activeMethod;
+      window.onmouseenter = activeMethod;
+      window.onkeydown = activeMethod;
+      window.onscroll = activeMethod;
+      window.onmousewheel = activeMethod;
+      document.addEventListener("visibilitychange", (function() {
+        return activity.handleVisibilityChange();
+      }), false);
+      document.addEventListener("webkitvisibilitychange", (function() {
+        return activity.handleVisibilityChange();
+      }), false);
+      document.addEventListener("msvisibilitychange", (function() {
+        return activity.handleVisibilityChange();
+      }), false);
+    }
+
+    Idle.prototype.onActive = function() {
+      this.awayTimestamp = new Date().getTime() + this.awayTimeout;
+      if (this.isAway) {
+        if (this.onAwayBack) {
+          this.onAwayBack();
+        }
+        this.start();
+      }
+      this.isAway = false;
+      return true;
+    };
+
+    Idle.prototype.start = function() {
+      var activity;
+
+      this.awayTimestamp = new Date().getTime() + this.awayTimeout;
+      if (this.awayTimer !== null) {
+        clearTimeout(this.awayTimer);
+      }
+      activity = this;
+      this.awayTimer = setTimeout((function() {
+        return activity.checkAway();
+      }), this.awayTimeout + 100);
+      return this;
+    };
+
+    Idle.prototype.setAwayTimeout = function(ms) {
+      this.awayTimeout = parseInt(ms, 10);
+      return this;
+    };
+
+    Idle.prototype.checkAway = function() {
+      var activity, t;
+
+      t = new Date().getTime();
+      if (t < this.awayTimestamp) {
+        this.isAway = false;
+        activity = this;
+        this.awayTimer = setTimeout((function() {
+          return activity.checkAway();
+        }), this.awayTimestamp - t + 100);
+        return;
+      }
+      if (this.awayTimer !== null) {
+        clearTimeout(this.awayTimer);
+      }
+      this.isAway = true;
+      if (this.onAway) {
+        return this.onAway();
+      }
+    };
+
+    Idle.prototype.handleVisibilityChange = function() {
+      if (document.hidden || document.msHidden || document.webkitHidden) {
+        if (this.onHidden) {
+          return this.onHidden();
+        }
+      } else {
+        if (this.onVisible) {
+          return this.onVisible();
+        }
+      }
+    };
+
+    return Idle;
+
+  })();
+
+  window.Idle = Idle;
+
+}).call(this);

From 392534878a2c3f6aa9f48a703d7219ad9e6751f3 Mon Sep 17 00:00:00 2001
From: Nick Winter <livelily@gmail.com>
Date: Tue, 11 Mar 2014 16:31:39 -0700
Subject: [PATCH 178/178] Keeping old matches, up to 200, and showing them as
 stale. Added LevelSession.isRanking for better display of when you're still
 in the initial ranking phase.

---
 app/styles/play/ladder.sass                   | 10 +++-
 app/templates/play/ladder/my_matches_tab.jade | 17 ++++---
 app/views/play/ladder/my_matches_tab.coffee   | 17 +++++--
 .../sessions/level_session_schema.coffee      |  6 ++-
 server/queues/scoring.coffee                  | 51 ++++++++++---------
 5 files changed, 63 insertions(+), 38 deletions(-)

diff --git a/app/styles/play/ladder.sass b/app/styles/play/ladder.sass
index 6904c5739..fc9bc6abc 100644
--- a/app/styles/play/ladder.sass
+++ b/app/styles/play/ladder.sass
@@ -18,5 +18,13 @@
     white-space: nowrap
     overflow: hidden
     
+  tr.stale
+    opacity: 0.5
+
+  tr.win .state-cell
+    color: #172
+  tr.loss .state-cell
+    color: #712
+
   #must-log-in button
-    margin-right: 10px
\ No newline at end of file
+    margin-right: 10px
diff --git a/app/templates/play/ladder/my_matches_tab.jade b/app/templates/play/ladder/my_matches_tab.jade
index 3827bd3b1..8cacdb49f 100644
--- a/app/templates/play/ladder/my_matches_tab.jade
+++ b/app/templates/play/ladder/my_matches_tab.jade
@@ -17,9 +17,10 @@ div#columns.row
               button.btn.btn-sm.btn-warning.pull-right.rank-button(data-session-id=team.session.id)
                 span.unavailable.hidden No New Code to Rank
                 span.rank.hidden Rank My Game!
-                span.ranking.hidden Submitting...
-                span.ranked.hidden Submitted for Ranking
+                span.submitting.hidden Submitting...
+                span.submitted.hidden Submitted for Ranking
                 span.failed.hidden Failed to Rank
+                span.ranking.hidden Game Being Ranked
 
         if team.chartData
           tr
@@ -32,7 +33,7 @@ div#columns.row
           th When
           th
         for match in team.matches
-          tr
+          tr(class=(match.stale ? "stale " : "") + match.state)
             td.state-cell
               if match.state === 'win'
                 span.win Win
@@ -48,7 +49,11 @@ div#columns.row
 
         if !team.matches.length
           tr
-            td(colspan=4).alert.alert-warning
-              | No ranked matches for this team!
-              | Play against some competitors and then come back here to get your game ranked.
+            if team.isRanking
+              td(colspan=4).alert.alert-info
+                | Your new code is being simulated by other players for ranking.
+            else
+              td(colspan=4).alert.alert-warning
+                | No ranked matches for this team!
+                | Play against some competitors and then come back here to get your game ranked.
       
diff --git a/app/views/play/ladder/my_matches_tab.coffee b/app/views/play/ladder/my_matches_tab.coffee
index 660114111..278bcb9b0 100644
--- a/app/views/play/ladder/my_matches_tab.coffee
+++ b/app/views/play/ladder/my_matches_tab.coffee
@@ -54,7 +54,7 @@ module.exports = class MyMatchesTabView extends CocoView
     ctx.levelID = @level.get('slug') or @level.id
     ctx.teams = @teams
 
-    convertMatch = (match) =>
+    convertMatch = (match, submitDate) =>
       opponent = match.opponents[0]
       state = 'win'
       state = 'loss' if match.metrics.rank > opponent.metrics.rank
@@ -65,12 +65,14 @@ module.exports = class MyMatchesTabView extends CocoView
         opponentID: opponent.userID
         when: moment(match.date).fromNow()
         sessionID: opponent.sessionID
+        stale: match.date < submitDate
       }
 
     for team in @teams
       team.session = (s for s in @sessions.models when s.get('team') is team.id)[0]
       team.readyToRank = @readyToRank(team.session)
-      team.matches = (convertMatch(match) for match in team.session?.get('matches') or [])
+      team.isRanking = team.session.get('isRanking')
+      team.matches = (convertMatch(match, team.session.get('submitDate')) for match in team.session?.get('matches') or [])
       team.matches.reverse()
       team.score = (team.session?.get('totalScore') or 10).toFixed(2)
       team.wins = _.filter(team.matches, {state: 'win'}).length
@@ -96,7 +98,12 @@ module.exports = class MyMatchesTabView extends CocoView
       button = $(el)
       sessionID = button.data('session-id')
       session = _.find @sessions.models, { id: sessionID }
-      @setRankingButtonText button, if @readyToRank(session) then 'rank' else 'unavailable'
+      rankingState = 'unavailable'
+      if @readyToRank session
+        rankingState = 'rank'
+      else if session.get 'isRanking'
+        rankingState = 'ranking'
+      @setRankingButtonText button, rankingState
 
   readyToRank: (session) ->
     return false unless session?.get('levelID')  # If it hasn't been denormalized, then it's not ready.
@@ -110,8 +117,8 @@ module.exports = class MyMatchesTabView extends CocoView
     session = _.find @sessions.models, { id: sessionID }
     return unless @readyToRank(session)
 
-    @setRankingButtonText(button, 'ranking')
-    success = => @setRankingButtonText(button, 'ranked')
+    @setRankingButtonText(button, 'submitting')
+    success = => @setRankingButtonText(button, 'submitted')
     failure = => @setRankingButtonText(button, 'failed')
 
     ajaxData = { session: sessionID, levelID: @level.id, originalLevelID: @level.attributes.original, levelMajorVersion: @level.attributes.version.major }
diff --git a/server/levels/sessions/level_session_schema.coffee b/server/levels/sessions/level_session_schema.coffee
index da4395dec..a7d742001 100644
--- a/server/levels/sessions/level_session_schema.coffee
+++ b/server/levels/sessions/level_session_schema.coffee
@@ -140,6 +140,10 @@ _.extend LevelSessionSchema.properties,
   submittedCode:
     type: 'object'
 
+  isRanking:
+    type: 'boolean'
+    description: 'Whether this session is still in the first ranking chain after being submitted.'
+
   unsubscribed:
     type: 'boolean'
     description: 'Whether the player has opted out of receiving email updates about ladder rankings for this session.'
@@ -147,6 +151,7 @@ _.extend LevelSessionSchema.properties,
   numberOfWinsAndTies:
     type: 'number'
     default: 0
+
   numberOfLosses:
     type: 'number'
     default: 0
@@ -162,7 +167,6 @@ _.extend LevelSessionSchema.properties,
       items:
         type: 'number'
 
-
   matches:
     type: 'array'
     title: 'Matches'
diff --git a/server/queues/scoring.coffee b/server/queues/scoring.coffee
index 0462cd503..9577df844 100644
--- a/server/queues/scoring.coffee
+++ b/server/queues/scoring.coffee
@@ -52,7 +52,7 @@ module.exports.createNewTask = (req, res) ->
   requestLevelID = req.body.originalLevelID
   requestCurrentLevelID = req.body.levelID
   requestLevelMajorVersion = parseInt(req.body.levelMajorVersion)
-  
+
   validatePermissions req, requestSessionID, (error, permissionsAreValid) ->
     if err? then return errors.serverError res, "There was an error validating permissions"
     unless permissionsAreValid then return errors.forbidden res, "You do not have the permissions to submit that game to the leaderboard"
@@ -60,24 +60,24 @@ module.exports.createNewTask = (req, res) ->
     return errors.badInput res, "The session ID is invalid" unless typeof requestSessionID is "string"
     Level.findOne({_id: requestCurrentLevelID}).lean().select('type').exec (err, levelWithType) ->
       if err? then return errors.serverError res, "There was an error finding the level type"
-        
-      if not levelWithType.type or levelWithType.type isnt "ladder" 
+
+      if not levelWithType.type or levelWithType.type isnt "ladder"
         console.log "The level type of level with ID #{requestLevelID} is #{levelWithType.type}"
         return errors.badInput res, "That level isn't a ladder level"
-  
+
       fetchSessionToSubmit requestSessionID, (err, sessionToSubmit) ->
         if err? then return errors.serverError res, "There was an error finding the given session."
-  
+
         updateSessionToSubmit sessionToSubmit, (err, data) ->
           if err? then return errors.serverError res, "There was an error updating the session"
           opposingTeam = calculateOpposingTeam(sessionToSubmit.team)
           fetchInitialSessionsToRankAgainst opposingTeam,requestLevelID, requestLevelMajorVersion, (err, sessionsToRankAgainst) ->
             if err? then return errors.serverError res, "There was an error fetching the sessions to rank against"
-  
+
             taskPairs = generateTaskPairs(sessionsToRankAgainst, sessionToSubmit)
             sendEachTaskPairToTheQueue taskPairs, (taskPairError) ->
               if taskPairError? then return errors.serverError res, "There was an error sending the task pairs to the queue"
-  
+
               sendResponseObject req, res, {"message":"All task pairs were succesfully sent to the queue"}
 
 module.exports.dispatchTaskToConsumer = (req, res) ->
@@ -118,51 +118,53 @@ module.exports.processTaskResult = (req, res) ->
     scoringTaskQueue.deleteMessage clientResponseObject.receiptHandle, (err) ->
       console.log "Deleted message."
       if err? then return errors.badInput res, "The queue message is already back in the queue, rejecting results."
-        
+
       LevelSession.findOne(_id: clientResponseObject.originalSessionID).lean().exec (err, levelSession) ->
         if err? then return errors.serverError res, "There was a problem finding the level session:#{err}"
-          
+
         supposedSubmissionDate = new Date(clientResponseObject.sessions[0].submitDate)
-          
+
         if Number(supposedSubmissionDate) isnt Number(levelSession.submitDate)
           return sendResponseObject req, res, {"message":"The game has been resubmitted. Removing from queue..."}
-  
+
         logTaskComputation clientResponseObject, taskLog, (logErr) ->
           if logErr? then return errors.serverError res, "There as a problem logging the task computation: #{logErr}"
-  
+
           updateSessions clientResponseObject, (updateError, newScoreArray) ->
             if updateError? then return errors.serverError res, "There was an error updating the scores.#{updateError}"
-  
+
             newScoresObject = _.indexBy newScoreArray, 'id'
-  
+
             addMatchToSessions clientResponseObject, newScoresObject, (err, data) ->
               if err? then return errors.serverError res, "There was an error updating the sessions with the match! #{JSON.stringify err}"
-  
+
               originalSessionID = clientResponseObject.originalSessionID
               originalSessionTeam = clientResponseObject.originalSessionTeam
               originalSessionRank = parseInt clientResponseObject.originalSessionRank
-  
+
               determineIfSessionShouldContinueAndUpdateLog originalSessionID, originalSessionRank, (err, sessionShouldContinue) ->
                 if err? then return errors.serverError res, "There was an error determining if the session should continue, #{err}"
-  
+
                 if sessionShouldContinue
                   opposingTeam = calculateOpposingTeam(originalSessionTeam)
                   opponentID = _.pull(_.keys(newScoresObject), originalSessionID)
                   sessionNewScore = newScoresObject[originalSessionID].totalScore
                   opponentNewScore = newScoresObject[opponentID].totalScore
-                  
+
                   levelOriginalID = levelSession.level.original
                   levelOriginalMajorVersion = levelSession.level.majorVersion
                   findNearestBetterSessionID levelOriginalID, levelOriginalMajorVersion, originalSessionID, sessionNewScore, opponentNewScore, opponentID ,opposingTeam, (err, opponentSessionID) ->
                     if err? then return errors.serverError res, "There was an error finding the nearest sessionID!"
                     unless opponentSessionID then return sendResponseObject req, res, {"message":"There were no more games to rank(game is at top!"}
-  
+
                     addPairwiseTaskToQueue [originalSessionID, opponentSessionID], (err, success) ->
                       if err? then return errors.serverError res, "There was an error sending the pairwise tasks to the queue!"
                       sendResponseObject req, res, {"message":"The scores were updated successfully and more games were sent to the queue!"}
                 else
                   console.log "Player lost, achieved rank #{originalSessionRank}"
-                  sendResponseObject req, res, {"message":"The scores were updated successfully, person lost so no more games are being inserted!"}
+                  LevelSession.update {_id: originalSessionID}, {isRanking: false}, {multi: false}, (err, affected) ->
+                    if err? then return errors.serverError res, "There was an error marking the completed session as not being ranked."
+                    sendResponseObject req, res, {"message":"The scores were updated successfully, person lost so no more games are being inserted!"}
 
 
 determineIfSessionShouldContinueAndUpdateLog = (sessionID, sessionRank, cb) ->
@@ -285,7 +287,7 @@ updateMatchesInSession = (matchObject, sessionID, callback) ->
   currentMatchObject.opponents = opponentsArray
 
   sessionUpdateObject =
-    $push: {matches: currentMatchObject}
+    $push: {matches: {$each: [currentMatchObject], $slice: -200}}
   log.info "Updating session #{sessionID}"
   LevelSession.update {"_id":sessionID}, sessionUpdateObject, callback
 
@@ -304,12 +306,12 @@ updateSessionToSubmit = (sessionToUpdate, callback) ->
     submitted: true
     submittedCode: sessionToUpdate.code
     submitDate: new Date()
-    matches: []
     meanStrength: 25
     standardDeviation: 25/3
     totalScore: 10
     numberOfWinsAndTies: 0
     numberOfLosses: 0
+    isRanking: true
   LevelSession.update {_id: sessionToUpdate._id}, sessionUpdateObject, callback
 
 fetchInitialSessionsToRankAgainst = (opposingTeam, levelID, levelMajorVersion, callback) ->
@@ -321,7 +323,7 @@ fetchInitialSessionsToRankAgainst = (opposingTeam, levelID, levelMajorVersion, c
     submittedCode:
       $exists: true
     team: opposingTeam
-  
+
   sortParameters =
     totalScore: 1
 
@@ -449,8 +451,7 @@ updateScoreInSession = (scoreObject,callback) ->
       meanStrength: scoreObject.meanStrength
       standardDeviation: scoreObject.standardDeviation
       totalScore: newTotalScore
-      $push:
-        scoreHistory: scoreHistoryAddition        
+      $push: {scoreHistory: {$each: [scoreHistoryAddition], $slice: -1000}}
 
     LevelSession.update {"_id": scoreObject.id}, updateObject, callback
     log.info "New total score for session #{scoreObject.id} is #{updateObject.totalScore}"