2016-09-14 13:32:11 -04:00
|
|
|
RootView = require 'views/core/RootView'
|
|
|
|
Campaigns = require 'collections/Campaigns'
|
|
|
|
Classroom = require 'models/Classroom'
|
|
|
|
Courses = require 'collections/Courses'
|
|
|
|
Levels = require 'collections/Levels'
|
2016-09-14 22:15:08 -04:00
|
|
|
LevelSession = require 'models/LevelSession'
|
2016-09-14 13:32:11 -04:00
|
|
|
LevelSessions = require 'collections/LevelSessions'
|
|
|
|
User = require 'models/User'
|
|
|
|
Users = require 'collections/Users'
|
|
|
|
CourseInstances = require 'collections/CourseInstances'
|
2016-09-16 19:56:13 -04:00
|
|
|
require 'vendor/d3'
|
2016-09-14 22:15:08 -04:00
|
|
|
# Users = require 'collections/Users'
|
|
|
|
# helper = require 'lib/coursesHelper'
|
|
|
|
# popoverTemplate = require 'templates/courses/classroom-level-popover'
|
|
|
|
# Prepaids = require 'collections/Prepaids'
|
|
|
|
# utils = require 'core/utils'
|
|
|
|
# CocoCollection = require 'collections/CocoCollection'
|
|
|
|
|
2016-09-14 13:32:11 -04:00
|
|
|
|
|
|
|
|
|
|
|
module.exports = class TeacherStudentView extends RootView
|
|
|
|
id: 'teacher-student-view'
|
|
|
|
template: require 'templates/teachers/teacher-student-view'
|
2016-09-14 22:15:08 -04:00
|
|
|
# helper: helper
|
|
|
|
events:
|
|
|
|
'click .assign-student-button': 'onClickAssignStudentButton'
|
2016-09-15 16:48:26 -04:00
|
|
|
'click .enroll-student-button': 'onClickEnrollStudentButton' # this button isn't working yet
|
2016-09-14 13:32:11 -04:00
|
|
|
|
|
|
|
getTitle: -> return @user?.broadName()
|
|
|
|
|
2016-09-15 16:48:26 -04:00
|
|
|
initialize: (options, classroomID, @studentID) ->
|
2016-09-14 13:32:11 -04:00
|
|
|
@classroom = new Classroom({_id: classroomID})
|
|
|
|
@listenToOnce @classroom, 'sync', @onClassroomSync
|
|
|
|
@supermodel.trackRequest(@classroom.fetch())
|
|
|
|
|
|
|
|
@courses = new Courses()
|
2016-09-14 22:15:08 -04:00
|
|
|
# @courses.comparator = '_id'
|
|
|
|
# @courseInstances = new CocoCollection([], { url: "/db/course_instance", model: CourseInstance})
|
|
|
|
# @courseInstances.comparator = 'courseID'
|
|
|
|
# @supermodel.loadCollection(@courseInstances, { data: { classroomID: classroomID } })
|
|
|
|
# @listenToOnce @courses, 'sync', @onCourseInstancesSync
|
2016-09-14 13:32:11 -04:00
|
|
|
@supermodel.trackRequest(@courses.fetch({data: { project: 'name' }}))
|
|
|
|
|
2016-09-14 22:15:08 -04:00
|
|
|
@courseInstances = new CourseInstances()
|
|
|
|
@supermodel.trackRequest @courseInstances.fetchForClassroom(classroomID)
|
|
|
|
|
2016-09-14 13:32:11 -04:00
|
|
|
@levels = new Levels()
|
2016-09-14 22:15:08 -04:00
|
|
|
# @levels.fetchForClassroom(classroomID, {data: {project: 'name,original,practice,slug'}})
|
|
|
|
# @levels.on 'add', (model) -> @_byId[model.get('original')] = model # so you can 'get' them
|
2016-09-14 13:32:11 -04:00
|
|
|
@supermodel.trackRequest(@levels.fetchForClassroom(classroomID, {data: {project: 'name,original'}}))
|
2016-09-15 16:48:26 -04:00
|
|
|
#
|
|
|
|
# @user = new User({_id: studentID})
|
|
|
|
# @supermodel.trackRequest(@user.fetch())
|
2016-09-14 13:32:11 -04:00
|
|
|
|
2016-09-15 16:48:26 -04:00
|
|
|
@urls = require('core/urls')
|
2016-09-14 13:32:11 -04:00
|
|
|
|
2016-09-15 16:48:26 -04:00
|
|
|
|
|
|
|
@singleStudentLevelProgressDotTemplate = require 'templates/teachers/hovers/progress-dot-single-student-level'
|
2016-09-14 22:15:08 -04:00
|
|
|
@levelProgressMap = {}
|
|
|
|
|
2016-09-14 13:32:11 -04:00
|
|
|
super(options)
|
|
|
|
|
2016-09-15 16:48:26 -04:00
|
|
|
onLoaded: ->
|
|
|
|
if @students.loaded and not @destroyed
|
|
|
|
@user = _.find(@students.models, (s)=> s.id is @studentID)
|
|
|
|
@updateLastPlayedString()
|
|
|
|
@updateLevelProgressMap()
|
2016-09-16 19:56:13 -04:00
|
|
|
@updateLevelDataMap()
|
2016-09-15 16:48:26 -04:00
|
|
|
@render()
|
|
|
|
super()
|
|
|
|
|
|
|
|
afterRender: ->
|
|
|
|
super(arguments...)
|
|
|
|
$('.progress-dot, .btn-view-project-level').each (i, el) ->
|
|
|
|
dot = $(el)
|
|
|
|
dot.tooltip({
|
|
|
|
html: true
|
|
|
|
container: dot
|
|
|
|
}).delegate '.tooltip', 'mousemove', ->
|
|
|
|
dot.tooltip('hide')
|
2016-09-14 22:15:08 -04:00
|
|
|
|
2016-09-19 20:50:08 -04:00
|
|
|
# @drawLineGraph()
|
|
|
|
# @drawBarGraph()
|
|
|
|
@sampleBarGraph()
|
2016-09-16 19:56:13 -04:00
|
|
|
|
2016-09-19 20:50:08 -04:00
|
|
|
sampleBarGraph: ->
|
|
|
|
margin = {top: 20, right: 20, bottom: 30, left: 40}
|
|
|
|
width = 960 - margin.left - margin.right
|
|
|
|
height = 500 - margin.top - margin.bottom
|
2016-09-16 19:56:13 -04:00
|
|
|
|
2016-09-19 20:50:08 -04:00
|
|
|
x0 = d3.scale.ordinal()
|
|
|
|
.rangeRoundBands([0, width], .1)
|
|
|
|
|
|
|
|
x1 = d3.scale.ordinal()
|
|
|
|
|
|
|
|
y = d3.scale.linear()
|
|
|
|
.range([height, 0])
|
|
|
|
|
|
|
|
color = d3.scale.ordinal()
|
|
|
|
.range(["#98abc5", "#8a89a6", "#7b6888", "#6b486b", "#a05d56", "#d0743c", "#ff8c00"])
|
|
|
|
|
|
|
|
xAxis = d3.svg.axis()
|
|
|
|
.scale(x0)
|
|
|
|
.orient("bottom")
|
|
|
|
|
|
|
|
yAxis = d3.svg.axis()
|
|
|
|
.scale(y)
|
|
|
|
.orient("left")
|
|
|
|
.tickFormat(d3.format(".2s"))
|
|
|
|
|
|
|
|
svg = d3.select("body").append("svg")
|
|
|
|
.attr("width", width + margin.left + margin.right)
|
|
|
|
.attr("height", height + margin.top + margin.bottom)
|
|
|
|
.append("g")
|
|
|
|
.attr("transform", "translate(" + margin.left + "," + margin.top + ")")
|
|
|
|
|
|
|
|
# missing d3.csv function from example at https://bl.ocks.org/mbostock/3887051 and https://www.dashingd3js.com/lessons/basic-chart-grouped-bar-chart
|
|
|
|
# need to figure out how to not use d3.csv and instead import existing @levelData array
|
|
|
|
|
|
|
|
drawBarGraph: ->
|
|
|
|
return unless @courses.loaded and @levels.loaded and @sessions?.loaded
|
2016-09-16 19:56:13 -04:00
|
|
|
vis = d3.select('#visualisation')
|
|
|
|
WIDTH = 1000
|
|
|
|
HEIGHT = 500
|
|
|
|
MARGINS = {
|
|
|
|
top: 20
|
|
|
|
right: 20
|
|
|
|
bottom: 20
|
|
|
|
left: 50
|
|
|
|
}
|
|
|
|
|
2016-09-19 20:50:08 -04:00
|
|
|
# xRange = d3.scale.linear().range([
|
|
|
|
# MARGINS.left
|
|
|
|
# WIDTH - (MARGINS.right)
|
|
|
|
# ]).domain([
|
|
|
|
# d3.min(@levelData, (d) ->
|
|
|
|
# d.levelIndex
|
|
|
|
# )
|
|
|
|
# d3.max(@levelData, (d) ->
|
|
|
|
# d.levelIndex
|
|
|
|
# )
|
|
|
|
# ])
|
|
|
|
|
|
|
|
xRange = d3.scale.ordinal().rangeRoundBands([MARGINS.left, WIDTH - MARGINS.right], 0.1).domain(@levelData.map( (d) -> d.levelIndex))
|
2016-09-16 19:56:13 -04:00
|
|
|
|
2016-09-19 20:50:08 -04:00
|
|
|
# yRange = d3.scale.linear().range([
|
|
|
|
# HEIGHT - (MARGINS.top)
|
|
|
|
# MARGINS.bottom
|
|
|
|
# ]).domain([
|
|
|
|
# d3.min(@levelData, (d) ->
|
|
|
|
# d.student
|
|
|
|
# )
|
|
|
|
# d3.max(@levelData, (d) ->
|
|
|
|
# d.student
|
|
|
|
# )
|
|
|
|
# ])
|
|
|
|
|
|
|
|
|
|
|
|
yRange = d3.scale.linear().range([HEIGHT - (MARGINS.top), MARGINS.bottom]).domain([0, d3.max(@levelData, (d) -> d.student)])
|
|
|
|
|
|
|
|
xAxis = d3.svg.axis().scale(xRange).tickSize(5).tickSubdivide(true)
|
|
|
|
yAxis = d3.svg.axis().scale(yRange).tickSize(5).orient('left').tickSubdivide(true)
|
|
|
|
|
|
|
|
vis.append('svg:g').attr('class', 'x axis').attr('transform', 'translate(0,' + (HEIGHT - (MARGINS.bottom)) + ')').call xAxis
|
|
|
|
vis.append('svg:g').attr('class', 'y axis').attr('transform', 'translate(' + MARGINS.left + ',0)').call yAxis
|
|
|
|
|
|
|
|
# barGenStudent: ->
|
|
|
|
vis.selectAll('rect').data(@levelData).enter().append('rect').attr('x', (d) ->
|
|
|
|
xRange d.levelIndex
|
|
|
|
).attr('y', (d) ->
|
|
|
|
yRange d.student
|
|
|
|
).attr('width', xRange.rangeBand()).attr('height', (d) ->
|
|
|
|
HEIGHT - (MARGINS.bottom) - yRange(d.student)
|
|
|
|
).attr 'fill', 'grey'
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
drawLineGraph: ->
|
|
|
|
return unless @courses.loaded and @levels.loaded and @sessions?.loaded
|
|
|
|
|
|
|
|
vis = d3.select('#visualisation')
|
|
|
|
WIDTH = 1000
|
|
|
|
HEIGHT = 500
|
|
|
|
MARGINS = {
|
|
|
|
top: 50
|
|
|
|
right: 20
|
|
|
|
bottom: 50
|
|
|
|
left: 50
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
xScale = d3.scale.linear().range([MARGINS.left, WIDTH - MARGINS.right]).domain([
|
|
|
|
d3.min(@levelData, (d) ->
|
|
|
|
return d.levelIndex
|
|
|
|
),
|
|
|
|
d3.max(@levelData, (d) ->
|
|
|
|
return d.levelIndex
|
|
|
|
)])
|
|
|
|
|
|
|
|
yScale = d3.scale.linear().range([HEIGHT - MARGINS.top, MARGINS.bottom]).domain([
|
|
|
|
d3.min(@levelData, (d) ->
|
|
|
|
return d.student
|
|
|
|
),
|
|
|
|
d3.max(@levelData, (d) ->
|
|
|
|
return d.student
|
|
|
|
)])
|
2016-09-16 19:56:13 -04:00
|
|
|
|
|
|
|
xAxis = d3.svg.axis().scale(xScale)
|
|
|
|
|
|
|
|
vis.append("svg:g")
|
|
|
|
.attr("class","axis")
|
|
|
|
.attr("transform", "translate(0," + (HEIGHT - MARGINS.bottom) + ")")
|
|
|
|
.call(xAxis)
|
|
|
|
|
|
|
|
yAxis = d3.svg.axis()
|
|
|
|
.scale(yScale)
|
|
|
|
.orient("left")
|
|
|
|
|
|
|
|
vis.append("svg:g")
|
|
|
|
.attr("class","axis")
|
|
|
|
.attr("transform", "translate(" + (MARGINS.left) + ",0)")
|
|
|
|
.call(yAxis)
|
|
|
|
|
2016-09-19 20:50:08 -04:00
|
|
|
lineGenStudent = d3.svg.line()
|
2016-09-16 19:56:13 -04:00
|
|
|
.x( (d)->
|
2016-09-19 20:50:08 -04:00
|
|
|
return xScale(d.levelIndex)
|
2016-09-16 19:56:13 -04:00
|
|
|
)
|
|
|
|
.y( (d)->
|
2016-09-19 20:50:08 -04:00
|
|
|
return yScale(d.student)
|
2016-09-16 19:56:13 -04:00
|
|
|
)
|
|
|
|
# .interpolate("basis")
|
|
|
|
|
2016-09-19 20:50:08 -04:00
|
|
|
lineGenClass = d3.svg.line()
|
|
|
|
.x( (d)->
|
|
|
|
return xScale(d.levelIndex)
|
|
|
|
)
|
|
|
|
.y( (d)->
|
|
|
|
return yScale(d.class)
|
|
|
|
)
|
|
|
|
|
|
|
|
# vis.append('svg:path')
|
|
|
|
# .attr('d', lineGenStudent(@levelData))
|
|
|
|
# .attr('stroke', 'green')
|
|
|
|
# .attr('stroke-width', 2)
|
|
|
|
# .attr('fill', 'none')
|
|
|
|
|
|
|
|
# vis.append('svg:path')
|
|
|
|
# .attr('d', lineGenClass(@levelData))
|
|
|
|
# .attr('stroke', 'blue')
|
|
|
|
# .attr('stroke-width', 2)
|
|
|
|
# .attr('fill', 'none')
|
|
|
|
|
|
|
|
|
|
|
|
dataGroup = d3.nest()
|
|
|
|
.key( (d)->
|
|
|
|
return d.courseID
|
|
|
|
)
|
|
|
|
.entries(@levelData)
|
|
|
|
|
|
|
|
lSpace = WIDTH/dataGroup.length
|
|
|
|
|
|
|
|
dataGroup.forEach (d, i) ->
|
|
|
|
# console.log (d.values[0].courseName)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
dataGroup.forEach (d, i) ->
|
|
|
|
console.log (lineGenStudent(d.values))
|
|
|
|
vis.append('svg:path').attr('d', lineGenStudent(d.values)).attr('stroke', 'blue').attr('stroke-width', 2).attr('id', 'line_' + d.key).attr 'fill', 'none'
|
|
|
|
# vis.append('svg:path').attr('d', lineGenClass(d.values)).attr('stroke', 'red').attr('stroke-width', 2).attr('id', 'line_' + d.key).attr 'fill', 'none'
|
|
|
|
vis.append('text').attr('x', lSpace / 2 + i * lSpace).attr('y', HEIGHT).style('fill', 'black').attr('class', 'legend').on('click', ->
|
|
|
|
active = if d.active then false else true
|
|
|
|
opacity = if active then 0 else 1
|
|
|
|
d3.select('#line_' + d.key).style 'opacity', opacity
|
|
|
|
d.active = active
|
|
|
|
return
|
|
|
|
).text d.values[0].courseName
|
|
|
|
# console.log (d.values[0].courseName)
|
|
|
|
return
|
|
|
|
|
2016-09-16 19:56:13 -04:00
|
|
|
|
2016-09-19 20:50:08 -04:00
|
|
|
# console.log (@levelData)
|
2016-09-16 19:56:13 -04:00
|
|
|
|
|
|
|
|
2016-09-14 13:32:11 -04:00
|
|
|
onClassroomSync: ->
|
|
|
|
# Now that we have the classroom from db, can request all level sessions for this classroom
|
|
|
|
@sessions = new LevelSessions()
|
2016-09-14 22:15:08 -04:00
|
|
|
@sessions.comparator = 'changed' # Sort level sessions by changed field, ascending
|
2016-09-14 13:32:11 -04:00
|
|
|
@listenTo @sessions, 'sync', @onSessionsSync
|
|
|
|
@supermodel.trackRequests(@sessions.fetchForAllClassroomMembers(@classroom))
|
|
|
|
|
2016-09-15 16:48:26 -04:00
|
|
|
@students = new Users()
|
|
|
|
jqxhrs = @students.fetchForClassroom(@classroom, removeDeleted: true)
|
|
|
|
# @listenTo @students, ->
|
|
|
|
# console.log @students
|
|
|
|
@supermodel.trackRequests jqxhrs
|
|
|
|
|
2016-09-14 13:32:11 -04:00
|
|
|
onSessionsSync: ->
|
|
|
|
# Now we have some level sessions, and enough data to calculate last played string
|
|
|
|
# This may be called multiple times due to paged server API calls via fetchForAllClassroomMembers
|
|
|
|
return if @destroyed # Don't do anything if page was destroyed after db request
|
|
|
|
@updateLastPlayedString()
|
2016-09-14 22:15:08 -04:00
|
|
|
@updateLevelProgressMap()
|
2016-09-16 19:56:13 -04:00
|
|
|
@updateLevelDataMap()
|
2016-09-14 22:15:08 -04:00
|
|
|
|
|
|
|
|
|
|
|
# onCourseInstancesSync: ->
|
|
|
|
# # @sessions = new CocoCollection([], { model: LevelSession })
|
|
|
|
# for courseInstance in @courseInstances.models
|
|
|
|
# sessions = new CocoCollection([], { url: "/db/course_instance/#{courseInstance.id}/level_sessions", model: LevelSession })
|
|
|
|
# @supermodel.loadCollection(sessions, { data: { project: ['level', 'playtime', 'creator', 'changed', 'state.complete'].join(' ') } })
|
|
|
|
# courseInstance.sessions = sessions
|
|
|
|
# sessions.courseInstance = courseInstance
|
|
|
|
# courseInstance.sessionsByUser = {}
|
|
|
|
# @listenToOnce sessions, 'sync', (sessions) ->
|
|
|
|
# @sessions.add(sessions.slice())
|
|
|
|
# for courseInstance in @courseInstances.models
|
|
|
|
# courseInstance.sessionsByUser = courseInstance.sessions.groupBy('creator')
|
|
|
|
#
|
|
|
|
# # Generate course instance JIT, in the meantime have models w/out equivalents in the db
|
|
|
|
# for course in @courses.models
|
|
|
|
# query = {courseID: course.id, classroomID: @classroom.id}
|
|
|
|
# courseInstance = @courseInstances.findWhere(query)
|
|
|
|
# if not courseInstance
|
|
|
|
# courseInstance = new CourseInstance(query)
|
|
|
|
# @courseInstances.add(courseInstance)
|
|
|
|
# courseInstance.sessions = new CocoCollection([], {model: LevelSession})
|
|
|
|
# # sessions.courseInstance = courseInstance
|
|
|
|
# courseInstance.sessionsByUser = {}
|
2016-09-14 13:32:11 -04:00
|
|
|
|
|
|
|
updateLastPlayedString: ->
|
|
|
|
# Make sure all our data is loaded, @sessions may not even be intialized yet
|
2016-09-15 16:48:26 -04:00
|
|
|
return unless @courses.loaded and @levels.loaded and @sessions?.loaded and @user?.loaded
|
2016-09-14 13:32:11 -04:00
|
|
|
|
|
|
|
# Use lodash to find the last session for our user, @sessions already sorted by changed date
|
|
|
|
session = _.findLast @sessions.models, (s) => s.get('creator') is @user.id
|
2016-09-14 22:15:08 -04:00
|
|
|
|
2016-09-14 13:32:11 -04:00
|
|
|
return unless session
|
|
|
|
|
|
|
|
# Find course for this level session, for it's name
|
|
|
|
# Level.original is the original id, used for level versioning, and connects levels to level sessions
|
|
|
|
for versionedCourse in @classroom.get('courses') ? []
|
|
|
|
for level in versionedCourse.levels
|
|
|
|
if level.original is session.get('level').original
|
|
|
|
# Found the level for our level session in the classroom versioned courses
|
|
|
|
# Find the full course so we can get it's name
|
|
|
|
course = _.find @courses.models, (c) => c.id is versionedCourse._id
|
|
|
|
break
|
|
|
|
|
|
|
|
# Find level for this level session, for it's name
|
|
|
|
level = @levels.findWhere({original: session.get('level').original})
|
|
|
|
|
|
|
|
# Update last played string based on what we found
|
|
|
|
@lastPlayedString = ""
|
|
|
|
@lastPlayedString += course.get('name') if course
|
|
|
|
@lastPlayedString += ", " if course and level
|
|
|
|
@lastPlayedString += level.get('name') if level
|
|
|
|
@lastPlayedString += ", " if @lastPlayedString
|
|
|
|
@lastPlayedString += session.get('changed')
|
|
|
|
# Rerun template/jade file to display new last played string
|
|
|
|
@render()
|
2016-09-14 22:15:08 -04:00
|
|
|
|
|
|
|
updateLevelProgressMap: ->
|
2016-09-15 16:48:26 -04:00
|
|
|
return unless @courses.loaded and @levels.loaded and @sessions?.loaded and @user?.loaded
|
2016-09-14 22:15:08 -04:00
|
|
|
|
|
|
|
# Map levels to sessions once, so we don't have to search entire session list multiple times below
|
2016-09-15 16:48:26 -04:00
|
|
|
@levelSessionMap = {}
|
|
|
|
for session in @sessions.models when session.get('creator') is @studentID
|
|
|
|
@levelSessionMap[session.get('level').original] = session
|
2016-09-14 22:15:08 -04:00
|
|
|
|
2016-09-16 13:27:09 -04:00
|
|
|
|
2016-09-14 22:15:08 -04:00
|
|
|
# Create mapping of level to student progress
|
|
|
|
@levelProgressMap = {}
|
|
|
|
for versionedCourse in @classroom.get('courses') ? []
|
|
|
|
for versionedLevel in versionedCourse.levels
|
2016-09-15 16:48:26 -04:00
|
|
|
session = @levelSessionMap[versionedLevel.original]
|
2016-09-14 22:15:08 -04:00
|
|
|
if session
|
|
|
|
if session.get('state')?.complete
|
|
|
|
@levelProgressMap[versionedLevel.original] = 'complete'
|
|
|
|
else
|
|
|
|
@levelProgressMap[versionedLevel.original] = 'started'
|
|
|
|
else
|
|
|
|
@levelProgressMap[versionedLevel.original] = 'not started'
|
|
|
|
|
2016-09-16 19:56:13 -04:00
|
|
|
updateLevelDataMap: ->
|
2016-09-16 13:27:09 -04:00
|
|
|
return unless @courses.loaded and @levels.loaded and @sessions?.loaded
|
2016-09-14 22:15:08 -04:00
|
|
|
|
2016-09-16 19:56:13 -04:00
|
|
|
@levelData = []
|
2016-09-16 13:27:09 -04:00
|
|
|
for versionedCourse in @classroom.get('courses') ? []
|
2016-09-16 19:56:13 -04:00
|
|
|
course = _.find @courses.models, (c) => c.id is versionedCourse._id
|
2016-09-16 13:27:09 -04:00
|
|
|
for versionedLevel in versionedCourse.levels
|
|
|
|
@playTime = 0
|
|
|
|
@timesPlayed = 0
|
2016-09-19 20:50:08 -04:00
|
|
|
@studentTime = 0
|
2016-09-16 19:56:13 -04:00
|
|
|
@levelProgress = 'not started'
|
2016-09-16 13:27:09 -04:00
|
|
|
for session in @sessions.models
|
|
|
|
if session.get('level').original == versionedLevel.original
|
|
|
|
@playTime += session.get('playtime') or 0
|
|
|
|
@timesPlayed += 1
|
2016-09-16 19:56:13 -04:00
|
|
|
if session.get('creator') is @studentID
|
|
|
|
@studentTime = session.get('playtime')
|
|
|
|
if @levelProgressMap[versionedLevel.original] == 'complete'
|
|
|
|
@levelProgress = 'complete'
|
|
|
|
else if @levelProgressMap[versionedLevel.original] == 'started'
|
|
|
|
@levelProgress = 'started'
|
2016-09-19 20:50:08 -04:00
|
|
|
# if @timesPlayed
|
|
|
|
classAvg = if @timesPlayed then Math.round(@playTime / @timesPlayed) else 0
|
|
|
|
@levelData.push {
|
|
|
|
levelID: versionedLevel.original
|
|
|
|
levelIndex: @classroom.getLevelNumber(versionedLevel.original)
|
|
|
|
levelName: versionedLevel.name
|
|
|
|
courseName: course.get('name')
|
|
|
|
courseID: course.get('_id')
|
|
|
|
class: classAvg
|
|
|
|
student: @studentTime
|
|
|
|
levelProgress: @levelProgress
|
|
|
|
}
|
|
|
|
# console.log (@levelData)
|
2016-09-16 13:27:09 -04:00
|
|
|
|
|
|
|
# new map for averages of all classroom playtimes
|
|
|
|
# @averageLevelPlaytimeMap = {}
|
|
|
|
# for versionedCourse in @classroom.get('courses') ? []
|
|
|
|
# for versionedLevel in versionedCourse.levels
|
|
|
|
# var totalPlaytime = null
|
|
|
|
# var totalSessions = 0
|
|
|
|
# session = @levelSessionMap[versionedLevel.original]
|
|
|
|
# # get playtimes for each session in all classroom sessions of this course with this particular level
|
|
|
|
# if totalPlaytime
|
|
|
|
# var averagePlaytime = totalPlaytime / totalSessions
|
|
|
|
# @averagePlaytimeMap[versionedLevel.original] = averagePlaytime
|
2016-09-14 22:15:08 -04:00
|
|
|
|
|
|
|
studentStatusString: () ->
|
|
|
|
status = @user.prepaidStatus()
|
|
|
|
expires = @user.get('coursePrepaid')?.endDate
|
|
|
|
string = switch status
|
|
|
|
when 'not-enrolled' then $.i18n.t('teacher.status_not_enrolled')
|
|
|
|
when 'enrolled' then (if expires then $.i18n.t('teacher.status_enrolled') else '-')
|
|
|
|
when 'expired' then $.i18n.t('teacher.status_expired')
|
|
|
|
return string.replace('{{date}}', moment(expires).utc().format('l'))
|
|
|
|
|
|
|
|
onClickEnrollStudentButton: (e) ->
|
|
|
|
userID = $(e.currentTarget).data('user-id')
|
|
|
|
user = @user.get(userID)
|
|
|
|
selectedUsers = new Users([user])
|
|
|
|
@enrollStudents(selectedUsers)
|
|
|
|
window.tracker?.trackEvent $(e.currentTarget).data('event-action'), category: 'Teachers', classroomID: @classroom.id, userID: userID, ['Mixpanel']
|
|
|
|
|
|
|
|
enrollStudents: (selectedUsers) ->
|
|
|
|
modal = new ActivateLicensesModal { @classroom, selectedUsers, users: @user }
|
|
|
|
@openModalView(modal)
|
|
|
|
modal.once 'redeem-users', (enrolledUsers) =>
|
|
|
|
enrolledUsers.each (newUser) =>
|
|
|
|
user = @user.get(newUser.id)
|
|
|
|
if user
|
|
|
|
user.set(newUser.attributes)
|
|
|
|
null
|
|
|
|
#
|
|
|
|
# onLoaded: ->
|
|
|
|
# console.log("on loaded")
|
|
|
|
# for courseInstance in @courseInstances.models
|
|
|
|
# courseID = courseInstance.get('courseID')
|
|
|
|
# course = @courses.get(courseID)
|
|
|
|
# # courseInstance.sessions.course = course
|
|
|
|
# @updateLastPlayedString()
|
|
|
|
# super()
|
|
|
|
#
|
|
|
|
# afterRender: ->
|
|
|
|
# @$('[data-toggle="popover"]').popover({
|
|
|
|
# html: true
|
|
|
|
# trigger: 'hover'
|
|
|
|
# placement: 'top'
|
|
|
|
# })
|
|
|
|
# super()
|
|
|
|
#
|
|
|
|
# levelPopoverContent: (level, session, i) ->
|
|
|
|
# return null unless level
|
|
|
|
# context = {
|
|
|
|
# moment: moment
|
|
|
|
# level: level
|
|
|
|
# session: session
|
|
|
|
# i: i
|
|
|
|
# canViewSolution: @teacherMode
|
|
|
|
# }
|
|
|
|
# return popoverTemplate(context)
|
|
|
|
#
|
|
|
|
# getLevelURL: (level, course, courseInstance, session) ->
|
|
|
|
# return null unless @teacherMode and _.all(arguments)
|
|
|
|
# "/play/level/#{level.get('slug')}?course=#{course.id}&course-instance=#{courseInstance.id}&session=#{session.id}&observing=true"
|