mirror of
https://github.com/codeninjasllc/codecombat.git
synced 2024-12-12 16:51:35 -05:00
d72e4eb750
Update classroom and gameplay Ux to surface practice levels as 3a, 3b, etc. Update next level logic to leverage practice levels based on per level completion playtime thresholds. Patrol buster and patrol buster A are live for testing. Fix a few classroom Ux progress hover bubble info bugs. Closes #3767
180 lines
6.7 KiB
CoffeeScript
180 lines
6.7 KiB
CoffeeScript
ModalView = require 'views/core/ModalView'
|
|
template = require 'templates/play/level/modal/course-victory-modal'
|
|
Achievements = require 'collections/Achievements'
|
|
Level = require 'models/Level'
|
|
Course = require 'models/Course'
|
|
ThangType = require 'models/ThangType'
|
|
ThangTypes = require 'collections/ThangTypes'
|
|
LevelSessions = require 'collections/LevelSessions'
|
|
EarnedAchievement = require 'models/EarnedAchievement'
|
|
LocalMongo = require 'lib/LocalMongo'
|
|
ProgressView = require './ProgressView'
|
|
NewItemView = require './NewItemView'
|
|
Classroom = require 'models/Classroom'
|
|
utils = require 'core/utils'
|
|
|
|
module.exports = class CourseVictoryModal extends ModalView
|
|
id: 'course-victory-modal'
|
|
template: template
|
|
closesOnClickOutside: false
|
|
|
|
|
|
initialize: (options) ->
|
|
@courseID = options.courseID
|
|
@courseInstanceID = options.courseInstanceID
|
|
@views = []
|
|
|
|
@session = options.session
|
|
@level = options.level
|
|
@newItems = new ThangTypes()
|
|
@newHeroes = new ThangTypes()
|
|
|
|
if @courseInstanceID
|
|
@classroom = new Classroom()
|
|
@supermodel.trackRequest(@classroom.fetchForCourseInstance(@courseInstanceID))
|
|
@achievements = options.achievements
|
|
if not @achievements
|
|
@achievements = new Achievements()
|
|
@achievements.fetchRelatedToLevel(@session.get('level').original)
|
|
@achievements = @supermodel.loadCollection(@achievements, 'achievements').model
|
|
@listenToOnce @achievements, 'sync', @onAchievementsLoaded
|
|
else
|
|
@onAchievementsLoaded()
|
|
|
|
@playSound 'victory'
|
|
@nextLevel = new Level()
|
|
@nextLevelRequest = @supermodel.trackRequest(@nextLevel.fetchNextForCourse({
|
|
levelOriginalID: @level.get('original')
|
|
@courseInstanceID
|
|
@courseID
|
|
sessionID: @session.id
|
|
}))
|
|
|
|
@course = options.course
|
|
if @courseID and not @course
|
|
@course = new Course().setURL "/db/course/#{@courseID}"
|
|
@course = @supermodel.loadModel(@course).model
|
|
|
|
if @courseInstanceID
|
|
@levelSessions = new LevelSessions()
|
|
@levelSessions.fetchForCourseInstance(@courseInstanceID)
|
|
@levelSessions = @supermodel.loadCollection(@levelSessions, 'sessions', {
|
|
data: { project: 'state.complete level.original playtime changed' }
|
|
}).model
|
|
window.tracker?.trackEvent 'Play Level Victory Modal Loaded', category: 'Students', levelSlug: @level.get('slug'), ['Mixpanel']
|
|
|
|
onResourceLoadFailed: (e) ->
|
|
if e.resource.jqxhr is @nextLevelRequest
|
|
return
|
|
super(arguments...)
|
|
|
|
|
|
onAchievementsLoaded: ->
|
|
@achievements.models = _.filter @achievements.models, (m) -> not m.get('query')?.ladderAchievementDifficulty # Don't show higher AI difficulty achievements
|
|
itemOriginals = []
|
|
heroOriginals = []
|
|
achievementIDs = []
|
|
for achievement in @achievements.models
|
|
rewards = achievement.get('rewards') or {}
|
|
heroOriginals.push rewards.heroes or []
|
|
itemOriginals.push rewards.items or []
|
|
achievement.completed = LocalMongo.matchesQuery(@session.attributes, achievement.get('query'))
|
|
achievementIDs.push(achievement.id) if achievement.completed
|
|
|
|
itemOriginals = _.uniq _.flatten itemOriginals
|
|
heroOriginals = _.uniq _.flatten heroOriginals
|
|
#project = ['original', 'rasterIcon', 'name', 'soundTriggers', 'i18n'] # This is what we need, but the PlayHeroesModal needs more, and so we load more to fill up the supermodel.
|
|
project = ['original', 'rasterIcon', 'name', 'slug', 'soundTriggers', 'featureImages', 'gems', 'heroClass', 'description', 'components', 'extendedName', 'unlockLevelName', 'i18n']
|
|
for [newThangTypeCollection, originals] in [[@newItems, itemOriginals], [@newHeroes, heroOriginals]]
|
|
for original in originals
|
|
thang= new ThangType()
|
|
thang.url = "/db/thang.type/#{original}/version"
|
|
thang.project = project
|
|
@supermodel.loadModel(thang)
|
|
newThangTypeCollection.add(thang)
|
|
|
|
@newEarnedAchievements = []
|
|
for achievement in @achievements.models
|
|
continue unless achievement.completed
|
|
ea = new EarnedAchievement({
|
|
collection: achievement.get('collection')
|
|
triggeredBy: @session.id
|
|
achievement: achievement.id
|
|
})
|
|
if me.isSessionless()
|
|
@newEarnedAchievements.push ea
|
|
else
|
|
ea.save()
|
|
# Can't just add models to supermodel because each ea has the same url
|
|
ea.sr = @supermodel.addSomethingResource(ea.cid)
|
|
@newEarnedAchievements.push ea
|
|
@listenToOnce ea, 'sync', (model) ->
|
|
model.sr.markLoaded()
|
|
if _.all((ea.id for ea in @newEarnedAchievements))
|
|
unless me.loading
|
|
@supermodel.loadModel(me, {cache: false})
|
|
@newEarnedAchievementsResource.markLoaded()
|
|
|
|
unless me.isSessionless()
|
|
# have to use a something resource because addModelResource doesn't handle models being upserted/fetched via POST like we're doing here
|
|
@newEarnedAchievementsResource = @supermodel.addSomethingResource('earned achievements') if @newEarnedAchievements.length
|
|
|
|
|
|
onLoaded: ->
|
|
super()
|
|
@views = []
|
|
|
|
# TODO: Add main victory view
|
|
# TODO: Add level up view
|
|
# TODO: Add new hero view?
|
|
|
|
for newItem in @newItems.models
|
|
@views.push(new NewItemView({item: newItem}))
|
|
|
|
progressView = new ProgressView({
|
|
level: @level
|
|
nextLevel: @nextLevel
|
|
course: @course
|
|
classroom: @classroom
|
|
levelSessions: @levelSessions
|
|
})
|
|
|
|
progressView.once 'done', @onDone, @
|
|
progressView.once 'next-level', @onNextLevel, @
|
|
for view in @views
|
|
view.on 'continue', @onViewContinue, @
|
|
@views.push(progressView)
|
|
|
|
@showView(_.first(@views))
|
|
|
|
afterRender: ->
|
|
super()
|
|
@showView(@currentView)
|
|
|
|
showView: (view) ->
|
|
return unless view
|
|
view.setElement(@$('.modal-content'))
|
|
view.$el.attr('id', view.id)
|
|
view.$el.addClass(view.className)
|
|
view.render()
|
|
@currentView = view
|
|
|
|
onViewContinue: ->
|
|
index = _.indexOf(@views, @currentView)
|
|
@showView(@views[index+1])
|
|
|
|
onNextLevel: ->
|
|
window.tracker?.trackEvent 'Play Level Victory Modal Next Level', category: 'Students', levelSlug: @level.get('slug'), nextLevelSlug: @nextLevel.get('slug'), ['Mixpanel']
|
|
if me.isSessionless()
|
|
link = "/play/level/#{@nextLevel.get('slug')}?course=#{@courseID}&codeLanguage=#{utils.getQueryVariable('codeLanguage', 'python')}"
|
|
else
|
|
link = "/play/level/#{@nextLevel.get('slug')}?course=#{@courseID}&course-instance=#{@courseInstanceID}"
|
|
application.router.navigate(link, {trigger: true})
|
|
|
|
onDone: ->
|
|
window.tracker?.trackEvent 'Play Level Victory Modal Done', category: 'Students', levelSlug: @level.get('slug'), ['Mixpanel']
|
|
if me.isSessionless()
|
|
link = "/teachers/courses"
|
|
else
|
|
link = "/courses/#{@courseID}/#{@courseInstanceID}"
|
|
application.router.navigate(link, {trigger: true})
|