RootView = require 'views/kinds/RootView' Level = require 'models/Level' Simulator = require 'lib/simulator/Simulator' LevelSession = require 'models/LevelSession' CocoCollection = require 'models/CocoCollection' {teamDataFromLevel} = require './ladder/utils' {me} = require 'lib/auth' application = require 'application' LadderTabView = require './ladder/ladder_tab' MyMatchesTabView = require './ladder/my_matches_tab' LadderPlayModal = require './ladder/play_modal' SimulatorsLeaderboardCollection = require 'collections/SimulatorsLeaderboardCollection' CocoClass = require 'lib/CocoClass' HIGHEST_SCORE = 1000000 class LevelSessionsCollection extends CocoCollection url: '' model: LevelSession constructor: (levelID) -> super() @url = "/db/level/#{levelID}/my_sessions" module.exports = class LadderView extends RootView id: 'ladder-view' template: require 'templates/play/ladder' subscriptions: 'application:idle-changed': 'onIdleChanged' events: 'click #simulate-button': 'onSimulateButtonClick' 'click #simulate-all-button': 'onSimulateAllButtonClick' 'click .play-button': 'onClickPlayButton' 'click a': 'onClickedLink' constructor: (options, @levelID) -> super(options) @level = new Level(_id:@levelID) @level.fetch() @sessions = new LevelSessionsCollection(levelID) @sessions.fetch({}) @addResourceToLoad(@sessions, 'your_sessions') @addResourceToLoad(@level, 'level') @simulatorsLeaderboardData = new SimulatorsLeaderboardData(me) @addResourceToLoad(@simulatorsLeaderboardData, 'top_simulators') @simulator = new Simulator() @listenTo(@simulator, 'statusUpdate', @updateSimulationStatus) @teams = [] onLoaded: -> @teams = teamDataFromLevel @level super() getRenderData: -> ctx = super() ctx.level = @level ctx.link = "/play/level/#{@level.get('name')}" ctx.simulationStatus = @simulationStatus ctx.teams = @teams ctx.levelID = @levelID ctx.levelDescription = marked(@level.get('description')) if @level.get('description') ctx.simulatorsLeaderboardData = @simulatorsLeaderboardData ctx._ = _ ctx afterRender: -> super() return if @loading() @insertSubView(@ladderTab = new LadderTabView({}, @level, @sessions)) @insertSubView(@myMatchesTab = new MyMatchesTabView({}, @level, @sessions)) @refreshInterval = setInterval(@fetchSessionsAndRefreshViews.bind(@), 20 * 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 fetchSessionsAndRefreshViews: -> return if @destroyed or application.userIsIdle or @$el.find('#simulate.active').length or (new Date() - 2000 < @lastRefreshTime) or @loading() @sessions.fetch({"success": @refreshViews}) refreshViews: => return if @destroyed or application.userIsIdle @lastRefreshTime = new Date() @ladderTab.refreshLadder() @myMatchesTab.refreshMatches() console.log "Refreshed sessions for ladder and matches." onIdleChanged: (e) -> @fetchSessionsAndRefreshViews() unless e.idle # Simulations onSimulateAllButtonClick: (e) -> submitIDs = _.pluck @leaderboards[@teams[0].id].topPlayers.models, "id" for ID in submitIDs $.ajax url: '/queue/scoring' method: 'POST' data: session: ID $("#simulate-all-button").prop "disabled", true $("#simulate-all-button").text "Submitted all!" onSimulateButtonClick: (e) -> $("#simulate-button").prop "disabled",true $("#simulate-button").text "Simulating..." @simulator.fetchAndSimulateTask() updateSimulationStatus: (simulationStatus, sessions) -> @simulationStatus = simulationStatus try if sessions? #TODO: Fetch names from Redis, the creatorName is denormalized creatorNames = (session.creatorName for session in sessions) @simulationStatus = "Simulating game between " for index in [0...creatorNames.length] unless creatorNames[index] creatorNames[index] = "Anonymous" @simulationStatus += (if index != 0 then " and " else "") + creatorNames[index] @simulationStatus += "..." catch e console.log "There was a problem with the named simulation status: #{e}" $("#simulation-status-text").text @simulationStatus onClickPlayButton: (e) -> @showPlayModal($(e.target).closest('.play-button').data('team')) resimulateAllSessions: -> postData = originalLevelID: @level.get('original') levelMajorVersion: @level.get('version').major console.log postData $.ajax url: '/queue/scoring/resimulateAllSessions' method: 'POST' data: postData complete: (jqxhr) -> console.log jqxhr.responseText showPlayModal: (teamID) -> return @showApologeticSignupModal() if me.get('anonymous') session = (s for s in @sessions.models when s.get('team') is teamID)[0] modal = new LadderPlayModal({}, @level, session, teamID) @openModalView modal showApologeticSignupModal: -> SignupModal = require 'views/modal/signup_modal' @openModalView(new SignupModal({showRequiredError:true})) onClickedLink: (e) -> link = $(e.target).closest('a').attr('href') if link?.startsWith('/play/level') and me.get('anonymous') e.stopPropagation() e.preventDefault() @showApologeticSignupModal() destroy: -> clearInterval @refreshInterval @simulator.destroy() super() class SimulatorsLeaderboardData extends CocoClass ### Consolidates what you need to load for a leaderboard into a single Backbone Model-like object. ### constructor: (@me) -> super() @fetch() fetch: -> @topSimulators = new SimulatorsLeaderboardCollection({order:-1, scoreOffset: -1, limit: 20}) promises = [] promises.push @topSimulators.fetch() unless @me.get('anonymous') score = @me.get('simulatedBy') or 0 @playersAbove = new SimulatorsLeaderboardCollection({order:1, scoreOffset: score, limit: 4}) promises.push @playersAbove.fetch() if score @playersBelow = new SimulatorsLeaderboardCollection({order:-1, scoreOffset: score, limit: 4}) promises.push @playersBelow.fetch() @promise = $.when(promises...) @promise.then @onLoad @promise.fail @onFail @promise onLoad: => return if @destroyed @loaded = true @trigger 'sync', @ onFail: (resource, jqxhr) => return if @destroyed @trigger 'error', @, jqxhr inTopSimulators: -> return me.id in (user.id for user in @topSimulators.models) nearbySimulators: -> l = [] above = @playersAbove.models above.reverse() l = l.concat(above) l.push @me l = l.concat(@playersBelow.models) if @playersBelow l allResources: -> resources = [@topSimulators, @playersAbove, @playersBelow] return (r for r in resources when r)