Merge branch 'master' into production

This commit is contained in:
Nick Winter 2015-02-05 15:05:44 -08:00
commit 28d01738f7
22 changed files with 317 additions and 104 deletions

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 516 KiB

View file

@ -1,6 +1,8 @@
// Helper for running tests through Karma. // Helper for running tests through Karma.
// Hooks into the test view logic for running tests. // Hooks into the test view logic for running tests.
window.userObject = {_id:'1'}
initialize = require('core/initialize'); initialize = require('core/initialize');
initialize.init(); initialize.init();
console.debug = function() {}; // Karma conf doesn't seem to work? Debug messages are still emitted when they shouldn't be. console.debug = function() {}; // Karma conf doesn't seem to work? Debug messages are still emitted when they shouldn't be.

View file

@ -43,6 +43,8 @@ init = ->
handleNormalUrls() handleNormalUrls()
setUpMoment() # Set up i18n for moment setUpMoment() # Set up i18n for moment
module.exports.init = init
handleNormalUrls = -> handleNormalUrls = ->
# http://artsy.github.com/blog/2012/06/25/replacing-hashbang-routes-with-pushstate/ # http://artsy.github.com/blog/2012/06/25/replacing-hashbang-routes-with-pushstate/
$(document).on 'click', "a[href^='/']", (event) -> $(document).on 'click', "a[href^='/']", (event) ->

View file

@ -399,10 +399,6 @@
thank_you: "Thank you for supporting CodeCombat." thank_you: "Thank you for supporting CodeCombat."
sorry_to_see_you_go: "Sorry to see you go! Please let us know what we could have done better." sorry_to_see_you_go: "Sorry to see you go! Please let us know what we could have done better."
unsubscribe_feedback_placeholder: "O, what have we done?" unsubscribe_feedback_placeholder: "O, what have we done?"
levels: "Get more practice with bonus levels!"
heroes: "More powerful heroes!"
gems: "3500 bonus gems every month!"
items: "Over 250 bonus items!"
parent_button: "Ask your parent" parent_button: "Ask your parent"
parent_email_description: "We'll email them so they can buy you a CodeCombat subscription." parent_email_description: "We'll email them so they can buy you a CodeCombat subscription."
parent_email_input_invalid: "Email address invalid." parent_email_input_invalid: "Email address invalid."
@ -416,7 +412,6 @@
parents_blurb1: "With CodeCombat, your child learns by writing real code. They start by learning simple commands, and progress to more advanced topics." parents_blurb1: "With CodeCombat, your child learns by writing real code. They start by learning simple commands, and progress to more advanced topics."
parents_blurb2: "For $9.99 USD/mo, they get new challenges every week and personal email support from professional programmers." parents_blurb2: "For $9.99 USD/mo, they get new challenges every week and personal email support from professional programmers."
parents_blurb3: "No Risk: 100% money back guarantee, easy 1-click unsubscribe." parents_blurb3: "No Risk: 100% money back guarantee, easy 1-click unsubscribe."
subscribe_button: "Subscribe"
stripe_description: "Monthly Subscription" stripe_description: "Monthly Subscription"
subscription_required_to_play: "You'll need a subscription to play this level." subscription_required_to_play: "You'll need a subscription to play this level."
unlock_help_videos: "Subscribe to unlock all video tutorials." unlock_help_videos: "Subscribe to unlock all video tutorials."
@ -1003,6 +998,7 @@
play_counts: "Play Counts" play_counts: "Play Counts"
feedback: "Feedback" feedback: "Feedback"
payment_info: "Payment Info" payment_info: "Payment Info"
campaigns: "Campaigns"
delta: delta:
added: "Added" added: "Added"

View file

@ -148,6 +148,14 @@ module.exports = class User extends CocoModel
application.tracker.identify leaderboardsGroup: @leaderboardsGroup unless me.isAdmin() application.tracker.identify leaderboardsGroup: @leaderboardsGroup unless me.isAdmin()
@leaderboardsGroup @leaderboardsGroup
getShowsPortal: ->
return @showsPortal if @showsPortal?
group = me.get('testGroupNumber')
@showsPortal = if group < 128 then true else false
@showsPortal = true if me.isAdmin()
application.tracker.identify showsPortal: @showsPortal unless me.isAdmin()
@showsPortal
getVideoTutorialStylesIndex: (numVideos=0)-> getVideoTutorialStylesIndex: (numVideos=0)->
# A/B Testing video tutorial styles # A/B Testing video tutorial styles
# Not a constant number of videos available (e.g. could be 0, 1, 3, or 4 currently) # Not a constant number of videos available (e.g. could be 0, 1, 3, or 4 currently)

View file

@ -91,5 +91,6 @@ AchievementSchema.definitions = {}
AchievementSchema.definitions['mongoQueryOperator'] = MongoQueryOperatorSchema AchievementSchema.definitions['mongoQueryOperator'] = MongoQueryOperatorSchema
AchievementSchema.definitions['mongoFindQuery'] = MongoFindQuerySchema AchievementSchema.definitions['mongoFindQuery'] = MongoFindQuerySchema
c.extendTranslationCoverageProperties AchievementSchema c.extendTranslationCoverageProperties AchievementSchema
c.extendPatchableProperties AchievementSchema
module.exports = AchievementSchema module.exports = AchievementSchema

View file

@ -124,5 +124,6 @@ _.extend CampaignSchema.properties, {
c.extendBasicProperties CampaignSchema, 'campaign' c.extendBasicProperties CampaignSchema, 'campaign'
c.extendTranslationCoverageProperties CampaignSchema c.extendTranslationCoverageProperties CampaignSchema
c.extendPatchableProperties CampaignSchema
module.exports = CampaignSchema module.exports = CampaignSchema

View file

@ -1,6 +1,6 @@
c = require './../schemas' c = require './../schemas'
patchables = ['level', 'thang_type', 'level_system', 'level_component', 'article'] patchables = ['level', 'thang_type', 'level_system', 'level_component', 'article', 'achievement', 'campaign']
PatchSchema = c.object({title: 'Patch', required: ['target', 'delta', 'commitMessage']}, { PatchSchema = c.object({title: 'Patch', required: ['target', 'delta', 'commitMessage']}, {
delta: {title: 'Delta', type: ['array', 'object']} delta: {title: 'Delta', type: ['array', 'object']}

View file

@ -478,6 +478,80 @@ $gameControlMargin: 30px
.particle-man .particle-man
z-index: 2 z-index: 2
.portal
position: relative
width: 100%
height: 100%
background: transparent url(/images/pages/play/portal-background.png)
display: flex
align-items: center
justify-content: center
.portals
$campaignWidth: 317px
$campaignHeight: 634px
$campaignHoverScale: 1.2
width: 6 * $campaignWidth
height: $campaignHeight * $campaignHoverScale
flex-wrap: nowrap
display: flex
overflow: hidden
.campaign
width: $campaignWidth
height: $campaignHeight
margin-top: $campaignHeight * ($campaignHoverScale - 1) / 2
background: transparent url(/images/pages/play/portal-campaigns.png) no-repeat 0 0
display: inline-block
flex-shrink: 0
position: relative
cursor: pointer
// http://easings.net/#easeOutBack plus tweaked a bit: http://cubic-bezier.com/#.11,.67,.08,1.42
@include transition(0.25s cubic-bezier(0.11, 0.67, 0.8, 1.42))
&:hover
@include scale($campaignHoverScale)
&.silhouette
@include filter(contrast(50%) brightness(65%))
pointer-events: none
&.locked
@include filter(contrast(80%) brightness(80%))
pointer-events: none
&.forest
background-position: (-1 * $campaignWidth) 0
&.desert
background-position: (-2 * $campaignWidth) 0
&.mountain
background-position: (-3 * $campaignWidth) 0
&.ice
background-position: (-4 * $campaignWidth) 0
&.volcano
background-position: (-5 * $campaignWidth) 0
.campaign-label
position: absolute
top: 55%
width: 100%
text-align: center
.campaign-name, .levels-completed, .campaign-locked
margin: 0
color: rgb(232, 217, 87)
text-shadow: black 2px 2px 0, black -2px -2px 0, black 2px -2px 0, black -2px 2px 0, black 2px 0px 0, black 0px -2px 0, black -2px 0px 0, black 0px 2px 0
z-index: 30
pointer-events: none
.levels-completed
font-size: 22px
.play-button
margin-top: 30px
min-width: 100px
body.ipad #campaign-view body.ipad #campaign-view
// iPad only supports up to Kithgard Gates for now. // iPad only supports up to Kithgard Gates for now.

View file

@ -65,7 +65,7 @@
span.glyphicon.glyphicon-ok span.glyphicon.glyphicon-ok
#parents-info(data-i18n="subscribe.parents") #parents-info(data-i18n="subscribe.parents")
button.btn.btn-lg.btn-illustrated.purchase-button(data-i18n="subscribe.subscribe_button") button.btn.btn-lg.btn-illustrated.purchase-button(data-i18n="subscribe.subscribe_title")
button.btn.btn-lg.btn-illustrated.parent-button(data-i18n="subscribe.parent_button") button.btn.btn-lg.btn-illustrated.parent-button(data-i18n="subscribe.parent_button")
if state === 'declined' if state === 'declined'

View file

@ -1,54 +1,77 @@
.map if campaign
.gradient.horizontal-gradient.top-gradient .map
.gradient.vertical-gradient.right-gradient .gradient.horizontal-gradient.top-gradient
.gradient.horizontal-gradient.bottom-gradient .gradient.vertical-gradient.right-gradient
.gradient.vertical-gradient.left-gradient .gradient.horizontal-gradient.bottom-gradient
.map-background(alt="", draggable="false") .gradient.vertical-gradient.left-gradient
.map-background(alt="", draggable="false")
each level in levels each level in levels
if !level.hidden if !level.hidden
div(style="left: #{level.position.x}%; bottom: #{level.position.y}%; background-color: #{level.color}", class="level" + (level.next ? " next" : "") + (level.disabled ? " disabled" : "") + (level.locked ? " locked" : "") + " " + (levelStatusMap[level.slug] || ""), data-level-slug=level.slug, data-level-original=level.original, title=i18n(level, 'name') + (level.disabled ? ' (Coming Soon to Adventurers)' : '')) div(style="left: #{level.position.x}%; bottom: #{level.position.y}%; background-color: #{level.color}", class="level" + (level.next ? " next" : "") + (level.disabled ? " disabled" : "") + (level.locked ? " locked" : "") + " " + (levelStatusMap[level.slug] || ""), data-level-slug=level.slug, data-level-original=level.original, title=i18n(level, 'name') + (level.disabled ? ' (Coming Soon to Adventurers)' : ''))
if level.unlocksHero && (!level.purchasedHero || editorMode) if level.unlocksHero && (!level.purchasedHero || editorMode)
img.hero-portrait(src="/file/db/thang.type/#{level.unlocksHero}/portrait.png") img.hero-portrait(src="/file/db/thang.type/#{level.unlocksHero}/portrait.png")
a(href=level.type == 'hero' ? '#' : level.disabled ? "/play" : "/play/#{level.levelPath || 'level'}/#{level.slug}", disabled=level.disabled, data-level-slug=level.slug, data-level-path=level.levelPath || 'level', data-level-name=level.name) a(href=level.type == 'hero' ? '#' : level.disabled ? "/play" : "/play/#{level.levelPath || 'level'}/#{level.slug}", disabled=level.disabled, data-level-slug=level.slug, data-level-path=level.levelPath || 'level', data-level-name=level.name)
if level.requiresSubscription if level.requiresSubscription
img.star(src="/images/pages/play/star.png") img.star(src="/images/pages/play/star.png")
if levelStatusMap[level.slug] === 'complete' if levelStatusMap[level.slug] === 'complete'
img.banner(src="/images/pages/play/level-banner-complete.png") img.banner(src="/images/pages/play/level-banner-complete.png")
if levelStatusMap[level.slug] === 'started' if levelStatusMap[level.slug] === 'started'
img.banner(src="/images/pages/play/level-banner-started.png") img.banner(src="/images/pages/play/level-banner-started.png")
div(style="left: #{level.position.x}%; bottom: #{level.position.y}%", class="level-shadow" + (level.next ? " next" : "") + " " + (levelStatusMap[level.slug] || "")) div(style="left: #{level.position.x}%; bottom: #{level.position.y}%", class="level-shadow" + (level.next ? " next" : "") + " " + (levelStatusMap[level.slug] || ""))
.level-info-container(data-level-slug=level.slug, data-level-path=level.levelPath || 'level', data-level-name=level.name) .level-info-container(data-level-slug=level.slug, data-level-path=level.levelPath || 'level', data-level-name=level.name)
- var playCount = levelPlayCountMap[level.slug] - var playCount = levelPlayCountMap[level.slug]
div(class="level-info " + (levelStatusMap[level.slug] || "") + (level.requiresSubscription ? " premium" : "")) div(class="level-info " + (levelStatusMap[level.slug] || "") + (level.requiresSubscription ? " premium" : ""))
.level-status .level-status
h3= i18n(level, 'name') + (level.disabled ? " (Coming soon!)" : (level.locked ? " (Locked)" : "")) h3= i18n(level, 'name') + (level.disabled ? " (Coming soon!)" : (level.locked ? " (Locked)" : ""))
- var description = i18n(level, 'description') || level.description || "" - var description = i18n(level, 'description') || level.description || ""
.level-description!= marked(description) .level-description!= marked(description)
if level.disabled if level.disabled
p p
span.spr(data-i18n="play.awaiting_levels_adventurer_prefix") We release five levels per week. span.spr(data-i18n="play.awaiting_levels_adventurer_prefix") We release five levels per week.
a.spr(href="/contribute/adventurer") a.spr(href="/contribute/adventurer")
strong(data-i18n="play.awaiting_levels_adventurer") Sign up as an Adventurer strong(data-i18n="play.awaiting_levels_adventurer") Sign up as an Adventurer
span.spl(data-i18n="play.awaiting_levels_adventurer_suffix") to be the first to play new levels. span.spl(data-i18n="play.awaiting_levels_adventurer_suffix") to be the first to play new levels.
if !level.disabled && !level.locked if !level.disabled && !level.locked
if playCount && playCount.sessions if playCount && playCount.sessions
.play-counts.hidden .play-counts.hidden
span.spl.spr= playCount.sessions span.spl.spr= playCount.sessions
span(data-i18n="play.players") players span(data-i18n="play.players") players
span.spr , #{Math.round(playCount.playtime / 3600)} span.spr , #{Math.round(playCount.playtime / 3600)}
span(data-i18n="play.hours_played") hours played span(data-i18n="play.hours_played") hours played
if levelStatusMap[level.slug] === 'complete' if levelStatusMap[level.slug] === 'complete'
button.btn.btn-warning.btn.btn-lg.btn-illustrated.view-solutions(data-level-slug=level.slug) button.btn.btn-warning.btn.btn-lg.btn-illustrated.view-solutions(data-level-slug=level.slug)
span(data-i18n="leaderboard.scores") span(data-i18n="leaderboard.scores")
button.btn.btn-success.btn.btn-lg.btn-illustrated.start-level(data-i18n="common.play") Play button.btn.btn-success.btn.btn-lg.btn-illustrated.start-level(data-i18n="common.play") Play
else if level.unlocksHero && !level.purchasedHero else if level.unlocksHero && !level.purchasedHero
img.hero-portrait(src="/file/db/thang.type/#{level.unlocksHero}/portrait.png", style="left: #{level.position.x}%; bottom: #{level.position.y}%;") img.hero-portrait(src="/file/db/thang.type/#{level.unlocksHero}/portrait.png", style="left: #{level.position.x}%; bottom: #{level.position.y}%;")
for adjacentCampaign in adjacentCampaigns for adjacentCampaign in adjacentCampaigns
a(href=(editorMode ? "/editor/campaign/" : "/play/") + adjacentCampaign.slug) a(href=(editorMode ? "/editor/campaign/" : "/play/") + adjacentCampaign.slug)
span.glyphicon.glyphicon-share-alt.campaign-switch(style=adjacentCampaign.style, title=adjacentCampaign.name, data-campaign-id=adjacentCampaign.id) span.glyphicon.glyphicon-share-alt.campaign-switch(style=adjacentCampaign.style, title=adjacentCampaign.name, data-campaign-id=adjacentCampaign.id)
else
.portal
.portals
for campaignSlug in ['dungeon', 'forest', 'desert', 'mountain', 'ice', 'volcano']
- var campaign = campaigns[campaignSlug];
div(class="campaign #{campaignSlug}" + (campaign ? "" : " silhouette") + (campaign && campaign.locked ? " locked" : ""), data-campaign-slug=campaignSlug)
.campaign-label
h2.campaign-name
if campaign
span= i18n(campaign.attributes, 'fullName')
else
span ???
if campaign && campaign.levelsTotal
h3.levels-completed
span= campaign.levelsCompleted
| /
span= campaign.levelsTotal
if campaign && campaign.locked
h3.campaign-locked(data-i18n="play.locked") Locked
else if campaign
btn(data-i18n="common.play").btn.btn-illustrated.btn-lg.btn-success.play-button
.game-controls.header-font .game-controls.header-font
button.btn.items(data-toggle='coco-modal', data-target='play/modal/PlayItemsModal', data-i18n="[title]play.items") button.btn.items(data-toggle='coco-modal', data-target='play/modal/PlayItemsModal', data-i18n="[title]play.items")
@ -85,7 +108,7 @@ button.btn.btn-lg.btn-inverse#volume-button(data-i18n="[title]play.adjust_volume
.glyphicon.glyphicon-volume-down .glyphicon.glyphicon-volume-down
.glyphicon.glyphicon-volume-up .glyphicon.glyphicon-volume-up
if campaign.loaded if campaign && campaign.loaded
h1#campaign-status h1#campaign-status
.campaign-status-background .campaign-status-background
.campaign-name .campaign-name

View file

@ -241,7 +241,7 @@ module.exports = class CocoView extends Backbone.View
showLoading: ($el=@$el) -> showLoading: ($el=@$el) ->
$el.find('>').addClass('hidden') $el.find('>').addClass('hidden')
$el.append loadingScreenTemplate() $el.append(loadingScreenTemplate()).i18n()
@_lastLoading = $el @_lastLoading = $el
hideLoading: -> hideLoading: ->

View file

@ -140,9 +140,14 @@ module.exports = class I18NEditModelView extends RootView
return _.isArray(delta.o) and delta.o.length is 1 and 'i18n' in delta.dataPath return _.isArray(delta.o) and delta.o.length is 1 and 'i18n' in delta.dataPath
) )
commitMessage = "Diplomat submission for lang #{@selectedLanguage}: #{flattened.length} change(s)."
save = false if @savedBefore
if save if save
modelToSave = @model.cloneNewMinorVersion() modelToSave = @model.cloneNewMinorVersion()
modelToSave.updateI18NCoverage() if modelToSave.get('i18nCoverage') modelToSave.updateI18NCoverage() if modelToSave.get('i18nCoverage')
if @modelClass.schema.properties.commitMessage
modelToSave.set 'commitMessage', commitMessage
else else
modelToSave = new Patch() modelToSave = new Patch()
@ -151,17 +156,21 @@ module.exports = class I18NEditModelView extends RootView
'collection': _.string.underscored @model.constructor.className 'collection': _.string.underscored @model.constructor.className
'id': @model.id 'id': @model.id
} }
if @modelClass.schema.properties.commitMessage
commitMessage = "Diplomat submission for lang #{@selectedLanguage}: #{flattened.length} change(s)."
modelToSave.set 'commitMessage', commitMessage modelToSave.set 'commitMessage', commitMessage
errors = modelToSave.validate() errors = modelToSave.validate()
button = $(e.target) button = $(e.target)
button.attr('disabled', 'disabled') button.attr('disabled', 'disabled')
return button.text('Failed to Submit Changes') if errors return button.text('Failed to Submit Changes') if errors
res = modelToSave.save(null, {type: 'POST'}) # Override PUT so we can trigger postNewVersion logic type = 'PUT'
if @modelClass.schema.properties.version or (not save)
# Override PUT so we can trigger postNewVersion logic
# or you're POSTing a Patch
type = 'POST'
res = modelToSave.save(null, {type: type})
return button.text('Failed to Submit Changes') unless res return button.text('Failed to Submit Changes') unless res
button.text('Submitting...') button.text('Submitting...')
res.error => button.text('Error Submitting Changes') res.error => button.text('Error Submitting Changes')
res.success => button.text('Submit Changes') res.success =>
@savedBefore = true
button.text('Submit Changes')

View file

@ -27,6 +27,11 @@ class LevelSessionsCollection extends CocoCollection
super() super()
@url = "/db/user/#{me.id}/level.sessions?project=state.complete,levelID" @url = "/db/user/#{me.id}/level.sessions?project=state.complete,levelID"
class CampaignsCollection extends CocoCollection
url: '/db/campaign'
model: Campaign
project: ['name', 'fullName', 'i18n']
module.exports = class CampaignView extends RootView module.exports = class CampaignView extends RootView
id: 'campaign-view' id: 'campaign-view'
template: template template: template
@ -41,18 +46,29 @@ module.exports = class CampaignView extends RootView
'click .level-info-container .start-level': 'onClickStartLevel' 'click .level-info-container .start-level': 'onClickStartLevel'
'click .level-info-container .view-solutions': 'onClickViewSolutions' 'click .level-info-container .view-solutions': 'onClickViewSolutions'
'click #volume-button': 'onToggleVolume' 'click #volume-button': 'onToggleVolume'
'click .portal .campaign': 'onClickPortalCampaign'
'mouseenter .portals': 'onMouseEnterPortals'
'mouseleave .portals': 'onMouseLeavePortals'
'mousemove .portals': 'onMouseMovePortals'
constructor: (options, @terrain='dungeon') -> constructor: (options, @terrain) ->
super options super options
options ?= {} @editorMode = options?.editorMode
if @editorMode
@campaign = new Campaign({_id:@terrain}) @terrain ?= 'dungeon'
@campaign = @supermodel.loadModel(@campaign, 'campaign').model else unless me.getShowsPortal()
@terrain ?= 'dungeon'
@editorMode = options.editorMode
@levelStatusMap = {} @levelStatusMap = {}
@levelPlayCountMap = {} @levelPlayCountMap = {}
@sessions = @supermodel.loadCollection(new LevelSessionsCollection(), 'your_sessions', null, 0).model @sessions = @supermodel.loadCollection(new LevelSessionsCollection(), 'your_sessions', null, 0).model
@listenToOnce @sessions, 'sync', @onSessionsLoaded
unless @terrain
@campaigns = @supermodel.loadCollection(new CampaignsCollection(), 'campaigns', null, 0).model
@listenToOnce @campaigns, 'sync', @onCampaignsLoaded
return
@campaign = new Campaign({_id:@terrain})
@campaign = @supermodel.loadModel(@campaign, 'campaign').model
# Temporary attempt to make sure all earned rewards are accounted for. Figure out a better solution... # Temporary attempt to make sure all earned rewards are accounted for. Figure out a better solution...
@earnedAchievements = new CocoCollection([], {url: '/db/earned_achievement', model:EarnedAchievement, project: ['earnedRewards']}) @earnedAchievements = new CocoCollection([], {url: '/db/earned_achievement', model:EarnedAchievement, project: ['earnedRewards']})
@ -69,7 +85,6 @@ module.exports = class CampaignView extends RootView
@supermodel.loadCollection(@earnedAchievements, 'achievements') @supermodel.loadCollection(@earnedAchievements, 'achievements')
@listenToOnce @sessions, 'sync', @onSessionsLoaded
@listenToOnce @campaign, 'sync', @getLevelPlayCounts @listenToOnce @campaign, 'sync', @getLevelPlayCounts
$(window).on 'resize', @onWindowResize $(window).on 'resize', @onWindowResize
@probablyCachedMusic = storage.load("loaded-menu-music") @probablyCachedMusic = storage.load("loaded-menu-music")
@ -99,6 +114,7 @@ module.exports = class CampaignView extends RootView
@musicPlayer?.destroy() @musicPlayer?.destroy()
clearTimeout @playMusicTimeout clearTimeout @playMusicTimeout
@particleMan?.destroy() @particleMan?.destroy()
clearInterval @portalScrollInterval
super() super()
getLevelPlayCounts: -> getLevelPlayCounts: ->
@ -136,31 +152,16 @@ module.exports = class CampaignView extends RootView
getRenderData: (context={}) -> getRenderData: (context={}) ->
context = super(context) context = super(context)
context.campaign = @campaign context.campaign = @campaign
context.levels = _.values($.extend true, {}, @campaign.get('levels')) context.levels = _.values($.extend true, {}, @campaign?.get('levels') ? {})
context.levelsCompleted = context.levelsTotal = 0 @annotateLevel level for level in context.levels
for level in context.levels count = @countLevels context.levels
level.position ?= { x: 10, y: 10 } context.levelsCompleted = count.completed
level.locked = not me.ownsLevel level.original context.levelsTotal = count.total
level.locked = false if @levelStatusMap[level.slug] in ['started', 'complete']
level.locked = false if @editorMode
level.locked = false if @campaign.get('name') is 'Auditions'
level.disabled = true if level.adminOnly and @levelStatusMap[level.slug] not in ['started', 'complete']
level.color = 'rgb(255, 80, 60)'
if level.requiresSubscription
level.color = 'rgb(80, 130, 200)'
if unlocksHero = _.find(level.rewards, 'hero')?.hero
level.unlocksHero = unlocksHero
if level.unlocksHero
level.purchasedHero = level.unlocksHero in (me.get('purchased')?.heroes or [])
level.hidden = level.locked
unless level.disabled
++context.levelsTotal
++context.levelsCompleted if @levelStatusMap[level.slug] is 'complete'
@determineNextLevel context.levels if @sessions.loaded @determineNextLevel context.levels if @sessions?.loaded
# put lower levels in last, so in the world map they layer over one another properly. # put lower levels in last, so in the world map they layer over one another properly.
context.levels = (_.sortBy context.levels, (l) -> l.position.y).reverse() context.levels = (_.sortBy context.levels, (l) -> l.position.y).reverse()
@campaign.renderedLevels = context.levels @campaign.renderedLevels = context.levels if @campaign
context.levelStatusMap = @levelStatusMap context.levelStatusMap = @levelStatusMap
context.levelPlayCountMap = @levelPlayCountMap context.levelPlayCountMap = @levelPlayCountMap
@ -168,7 +169,7 @@ module.exports = class CampaignView extends RootView
context.mapType = _.string.slugify @terrain context.mapType = _.string.slugify @terrain
context.requiresSubscription = @requiresSubscription context.requiresSubscription = @requiresSubscription
context.editorMode = @editorMode context.editorMode = @editorMode
context.adjacentCampaigns = _.filter _.values(_.cloneDeep(@campaign.get('adjacentCampaigns') or {})), (ac) => context.adjacentCampaigns = _.filter _.values(_.cloneDeep(@campaign?.get('adjacentCampaigns') or {})), (ac) =>
return false if ac.showIfUnlocked and (ac.showIfUnlocked not in me.levels()) and not @editorMode return false if ac.showIfUnlocked and (ac.showIfUnlocked not in me.levels()) and not @editorMode
ac.name = utils.i18n ac, 'name' ac.name = utils.i18n ac, 'name'
styles = [] styles = []
@ -181,6 +182,26 @@ module.exports = class CampaignView extends RootView
return true return true
context.marked = marked context.marked = marked
context.i18n = utils.i18n context.i18n = utils.i18n
if @campaigns
context.campaigns = {}
for campaign in @campaigns.models
context.campaigns[campaign.get('slug')] = campaign
if @sessions.loaded
levels = _.values($.extend true, {}, campaign.get('levels') ? {})
count = @countLevels levels
campaign.levelsTotal = count.total
campaign.levelsCompleted = count.completed
if campaign.get('slug') is 'dungeon'
campaign.locked = false
else unless campaign.levelsTotal
campaign.locked = true
else
campaign.locked = true
for campaign in @campaigns.models
for acID, ac of campaign.get('adjacentCampaigns') ? {}
_.find(@campaigns.models, id: acID)?.locked = false if ac.showIfUnlocked in me.levels()
context context
afterRender: -> afterRender: ->
@ -203,7 +224,7 @@ module.exports = class CampaignView extends RootView
unless window.currentModal or not @fullyRendered unless window.currentModal or not @fullyRendered
@highlightElement '.level.next', delay: 500, duration: 60000, rotation: 0, sides: ['top'] @highlightElement '.level.next', delay: 500, duration: 60000, rotation: 0, sides: ['top']
if @editorMode if @editorMode
for level in @campaign.renderedLevels for level in @campaign?.renderedLevels ? []
for nextLevelOriginal in level.nextLevels ? [] for nextLevelOriginal in level.nextLevels ? []
if nextLevel = _.find(@campaign.renderedLevels, original: nextLevelOriginal) if nextLevel = _.find(@campaign.renderedLevels, original: nextLevelOriginal)
@createLine level.position, nextLevel.position @createLine level.position, nextLevel.position
@ -220,6 +241,32 @@ module.exports = class CampaignView extends RootView
authModal.mode = 'signup' authModal.mode = 'signup'
@openModalView authModal @openModalView authModal
annotateLevel: (level) ->
level.position ?= { x: 10, y: 10 }
level.locked = not me.ownsLevel level.original
level.locked = false if @levelStatusMap[level.slug] in ['started', 'complete']
level.locked = false if @editorMode
level.locked = false if @campaign?.get('name') is 'Auditions'
level.disabled = true if level.adminOnly and @levelStatusMap[level.slug] not in ['started', 'complete']
level.color = 'rgb(255, 80, 60)'
if level.requiresSubscription
level.color = 'rgb(80, 130, 200)'
if unlocksHero = _.find(level.rewards, 'hero')?.hero
level.unlocksHero = unlocksHero
if level.unlocksHero
level.purchasedHero = level.unlocksHero in (me.get('purchased')?.heroes or [])
level.hidden = level.locked
level
countLevels: (levels) ->
count = total: 0, completed: 0
for level in levels
@annotateLevel level unless level.locked? # Annotate if we haven't already.
unless level.disabled
++count.total
++count.completed if @levelStatusMap[level.slug] is 'complete'
count
showLeaderboard: (levelSlug) -> showLeaderboard: (levelSlug) ->
#levelSlug ?= 'siege-of-stonehold' # Testing #levelSlug ?= 'siege-of-stonehold' # Testing
leaderboardModal = new LeaderboardModal supermodel: @supermodel, levelSlug: levelSlug leaderboardModal = new LeaderboardModal supermodel: @supermodel, levelSlug: levelSlug
@ -249,7 +296,7 @@ module.exports = class CampaignView extends RootView
line.append($('<div class="line">')).append($('<div class="point">')) line.append($('<div class="line">')).append($('<div class="point">'))
applyCampaignStyles: -> applyCampaignStyles: ->
return unless @campaign.loaded return unless @campaign?.loaded
if (backgrounds = @campaign.get 'backgroundImage') and backgrounds.length if (backgrounds = @campaign.get 'backgroundImage') and backgrounds.length
backgrounds = _.sortBy backgrounds, 'width' backgrounds = _.sortBy backgrounds, 'width'
backgrounds.reverse() backgrounds.reverse()
@ -267,7 +314,7 @@ module.exports = class CampaignView extends RootView
@playAmbientSound() @playAmbientSound()
testParticles: -> testParticles: ->
return unless @campaign.loaded and me.getForeshadowsLevels() return unless @campaign?.loaded and me.getForeshadowsLevels()
@particleMan ?= new ParticleMan() @particleMan ?= new ParticleMan()
@particleMan.removeEmitters() @particleMan.removeEmitters()
@particleMan.attach @$el.find('.map') @particleMan.attach @$el.find('.map')
@ -280,12 +327,42 @@ module.exports = class CampaignView extends RootView
continue if particleKey.length is 2 # Don't show basic levels continue if particleKey.length is 2 # Don't show basic levels
@particleMan.addEmitter level.position.x / 100, level.position.y / 100, particleKey.join('-') @particleMan.addEmitter level.position.x / 100, level.position.y / 100, particleKey.join('-')
onMouseEnterPortals: (e) ->
return unless @campaigns?.loaded and @sessions?.loaded
@portalScrollInterval = setInterval @onMouseMovePortals, 100
@onMouseMovePortals e
onMouseLeavePortals: (e) ->
return unless @portalScrollInterval
clearInterval @portalScrollInterval
@portalScrollInterval = null
onMouseMovePortals: (e) =>
return unless @portalScrollInterval
$portal = @$el.find('.portal')
$portals = @$el.find('.portals')
if e
@portalOffsetX = Math.round Math.max 0, e.clientX - $portal.offset().left
bodyWidth = $('body').innerWidth()
fraction = @portalOffsetX / bodyWidth
return if 0.2 < fraction < 0.8
direction = if fraction < 0.5 then 1 else -1
magnitude = 0.2 * bodyWidth * (if direction is -1 then fraction - 0.8 else 0.2 - fraction) / 0.2
portalsWidth = 1902 # TODO: if we add campaigns or change margins, this will get out of date...
scrollTo = $portals.offset().left + direction * magnitude
scrollTo = Math.max bodyWidth - portalsWidth, scrollTo
scrollTo = Math.min 0, scrollTo
$portals.stop().animate {marginLeft: scrollTo}, 100, 'linear'
onSessionsLoaded: (e) -> onSessionsLoaded: (e) ->
return if @editorMode return if @editorMode
for session in @sessions.models for session in @sessions.models
@levelStatusMap[session.get('levelID')] = if session.get('state')?.complete then 'complete' else 'started' @levelStatusMap[session.get('levelID')] = if session.get('state')?.complete then 'complete' else 'started'
@render() @render()
onCampaignsLoaded: (e) ->
@render()
preloadLevel: (levelSlug) -> preloadLevel: (levelSlug) ->
levelURL = "/db/level/#{levelSlug}" levelURL = "/db/level/#{levelSlug}"
level = new Level().setURL levelURL level = new Level().setURL levelURL
@ -445,3 +522,12 @@ module.exports = class CampaignView extends RootView
@$el.find('.player-hero-icon').removeClass().addClass('player-hero-icon ' + slug) @$el.find('.player-hero-icon').removeClass().addClass('player-hero-icon ' + slug)
return return
console.error "CampaignView hero update couldn't find hero slug for original:", hero console.error "CampaignView hero update couldn't find hero slug for original:", hero
onClickPortalCampaign: (e) ->
campaign = $(e.target).closest('.campaign')
return if campaign.is('.locked') or campaign.is('.silhouette')
campaignSlug = campaign.data('campaign-slug')
Backbone.Mediator.publish 'router:navigate',
route: "/play/#{campaignSlug}"
viewClass: CampaignView
viewArgs: [{supermodel: @supermodel}, campaignSlug]

View file

@ -80,9 +80,8 @@ module.exports = class ControlBarView extends CocoView
@homeLink = c.homeLink = '/play' @homeLink = c.homeLink = '/play'
@homeViewClass = 'views/play/CampaignView' @homeViewClass = 'views/play/CampaignView'
campaign = @level.get 'campaign' campaign = @level.get 'campaign'
if campaign isnt 'dungeon' @homeLink += '/' + campaign
@homeLink += '/' + campaign @homeViewArgs.push campaign
@homeViewArgs.push campaign
else else
@homeLink = c.homeLink = '/' @homeLink = c.homeLink = '/'
@homeViewClass = 'views/HomeView' @homeViewClass = 'views/HomeView'

View file

@ -326,7 +326,7 @@ module.exports = class HeroVictoryModal extends ModalView
getNextLevelLink: -> getNextLevelLink: ->
link = '/play' link = '/play'
nextCampaign = @getNextLevelCampaign() nextCampaign = @getNextLevelCampaign()
link += '/' + nextCampaign unless nextCampaign is 'dungeon' link += '/' + nextCampaign
link link
onClickContinue: (e, extraOptions=null) -> onClickContinue: (e, extraOptions=null) ->

View file

@ -9,6 +9,7 @@ module.exports = function(config) {
// list of files / patterns to load in the browser // list of files / patterns to load in the browser
files : [ files : [
'public/javascripts/vendor.js', // need for jade definition...
'public/javascripts/whole-vendor.js', 'public/javascripts/whole-vendor.js',
'public/lib/ace/ace.js', 'public/lib/ace/ace.js',
'public/javascripts/aether.js', 'public/javascripts/aether.js',

View file

@ -81,6 +81,7 @@ AchievementSchema.post 'save', -> @constructor.loadAchievements()
AchievementSchema.plugin(plugins.NamedPlugin) AchievementSchema.plugin(plugins.NamedPlugin)
AchievementSchema.plugin(plugins.SearchablePlugin, {searchable: ['name']}) AchievementSchema.plugin(plugins.SearchablePlugin, {searchable: ['name']})
AchievementSchema.plugin plugins.TranslationCoveragePlugin AchievementSchema.plugin plugins.TranslationCoveragePlugin
AchievementSchema.plugin plugins.PatchablePlugin
module.exports = Achievement = mongoose.model('Achievement', AchievementSchema, 'achievements') module.exports = Achievement = mongoose.model('Achievement', AchievementSchema, 'achievements')

View file

@ -8,5 +8,6 @@ CampaignSchema.index({slug: 1}, {name: 'slug index', sparse: true, unique: true}
CampaignSchema.plugin(plugins.NamedPlugin) CampaignSchema.plugin(plugins.NamedPlugin)
CampaignSchema.plugin(plugins.TranslationCoveragePlugin) CampaignSchema.plugin(plugins.TranslationCoveragePlugin)
CampaignSchema.plugin plugins.PatchablePlugin
module.exports = mongoose.model('campaign', CampaignSchema) module.exports = mongoose.model('campaign', CampaignSchema)

View file

@ -24,6 +24,15 @@ CampaignHandler = class CampaignHandler extends Handler
hasAccess: (req) -> hasAccess: (req) ->
req.method is 'GET' or req.user?.isAdmin() req.method is 'GET' or req.user?.isAdmin()
get: (req, res) ->
return @sendForbiddenError(res) if not @hasAccess(req)
# We don't have normal text search or anything set up to make /db/campaign work, so we'll just give them all campaigns, no problem.
q = @modelClass.find {}
q.exec (err, documents) =>
return @sendDatabaseError(res, err) if err
documents = (@formatEntity(req, doc) for doc in documents)
@sendSuccess(res, documents)
getByRelationship: (req, res, args...) -> getByRelationship: (req, res, args...) ->
relationship = args[1] relationship = args[1]
if relationship in ['levels', 'achievements'] if relationship in ['levels', 'achievements']

View file

@ -34,7 +34,7 @@ module.exports = class Handler
hasAccessToDocument: (req, document, method=null) -> hasAccessToDocument: (req, document, method=null) ->
return true if req.user?.isAdmin() return true if req.user?.isAdmin()
if @modelClass.schema.uses_coco_translation_coverage and (method or req.method).toLowerCase() is 'post' if @modelClass.schema.uses_coco_translation_coverage and (method or req.method).toLowerCase() in ['post', 'put']
return true if @isJustFillingTranslations(req, document) return true if @isJustFillingTranslations(req, document)
if @modelClass.schema.uses_coco_permissions if @modelClass.schema.uses_coco_permissions