codecombat/app/views/courses/ClassroomView.coffee

246 lines
10 KiB
CoffeeScript
Raw Normal View History

2015-11-30 14:14:27 -05:00
Campaign = require 'models/Campaign'
CocoCollection = require 'collections/CocoCollection'
Course = require 'models/Course'
CourseInstance = require 'models/CourseInstance'
Classroom = require 'models/Classroom'
Classrooms = require 'collections/Classrooms'
2015-11-30 14:14:27 -05:00
LevelSession = require 'models/LevelSession'
Prepaids = require 'collections/Prepaids'
2016-04-13 12:54:24 -04:00
Levels = require 'collections/Levels'
2015-11-30 14:14:27 -05:00
RootView = require 'views/core/RootView'
template = require 'templates/courses/classroom-view'
User = require 'models/User'
utils = require 'core/utils'
Prepaid = require 'models/Prepaid'
ClassroomSettingsModal = require 'views/courses/ClassroomSettingsModal'
ActivateLicensesModal = require 'views/courses/ActivateLicensesModal'
2015-12-01 16:17:21 -05:00
InviteToClassroomModal = require 'views/courses/InviteToClassroomModal'
RemoveStudentModal = require 'views/courses/RemoveStudentModal'
popoverTemplate = require 'templates/courses/classroom-level-popover'
2015-11-30 14:14:27 -05:00
module.exports = class ClassroomView extends RootView
id: 'classroom-view'
template: template
teacherMode: false
2015-11-30 14:14:27 -05:00
events:
'click #edit-class-details-link': 'onClickEditClassDetailsLink'
'click #activate-licenses-btn': 'onClickActivateLicensesButton'
'click .activate-single-license-btn': 'onClickActivateSingleLicenseButton'
2015-12-01 16:17:21 -05:00
'click #add-students-btn': 'onClickAddStudentsButton'
2015-12-01 16:27:12 -05:00
'click .enable-btn': 'onClickEnableButton'
'click .remove-student-link': 'onClickRemoveStudentLink'
2015-11-30 14:14:27 -05:00
initialize: (options, classroomID) ->
2015-12-04 19:37:13 -05:00
return if me.isAnonymous()
2015-11-30 14:14:27 -05:00
@classroom = new Classroom({_id: classroomID})
@supermodel.loadModel @classroom
2015-11-30 14:14:27 -05:00
@courses = new CocoCollection([], { url: "/db/course", model: Course})
@courses.comparator = '_id'
@supermodel.loadCollection(@courses)
2015-11-30 14:14:27 -05:00
@courses.comparator = '_id'
@courseInstances = new CocoCollection([], { url: "/db/course_instance", model: CourseInstance})
@courseInstances.comparator = 'courseID'
@supermodel.loadCollection(@courseInstances, { data: { classroomID: classroomID } })
@prepaids = new Prepaids()
@prepaids.comparator = '_id'
@prepaids.fetchByCreator(me.id)
@supermodel.loadCollection(@prepaids)
@users = new CocoCollection([], { url: "/db/classroom/#{classroomID}/members?memberLimit=100", model: User })
@users.comparator = (user) => user.broadName().toLowerCase()
@supermodel.loadCollection(@users)
2015-11-30 14:14:27 -05:00
@listenToOnce @courseInstances, 'sync', @onCourseInstancesSync
2015-12-03 17:51:38 -05:00
@sessions = new CocoCollection([], { model: LevelSession })
@ownedClassrooms = new Classrooms()
@ownedClassrooms.fetchMine({data: {project: '_id'}})
@supermodel.trackCollection(@ownedClassrooms)
2016-04-13 12:54:24 -04:00
@levels = new Levels()
@levels.fetchForClassroom(classroomID, {data: {project: 'name,slug,original'}})
@levels.on 'add', (model) -> @_byId[model.get('original')] = model # so you can 'get' them
@supermodel.trackCollection(@levels)
2015-11-30 14:14:27 -05: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(' ') } })
2015-11-30 14:14:27 -05:00
courseInstance.sessions = sessions
sessions.courseInstance = courseInstance
courseInstance.sessionsByUser = {}
2015-11-30 14:14:27 -05:00
@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 = {}
2015-11-30 14:14:27 -05:00
onLoaded: ->
@teacherMode = me.isAdmin() or @classroom.get('ownerID') is me.id
2015-11-30 14:14:27 -05:00
userSessions = @sessions.groupBy('creator')
for user in @users.models
user.sessions = new CocoCollection(userSessions[user.id], { model: LevelSession })
user.sessions.comparator = 'changed'
user.sessions.sort()
for courseInstance in @courseInstances.models
courseID = courseInstance.get('courseID')
course = @courses.get(courseID)
2016-04-13 12:54:24 -04:00
courseInstance.sessions.course = course
2015-11-30 14:14:27 -05:00
super()
afterRender: ->
@$('[data-toggle="popover"]').popover({
html: true
trigger: 'hover'
placement: 'top'
})
super()
2015-11-30 14:14:27 -05:00
onClickActivateLicensesButton: ->
modal = new ActivateLicensesModal({
classroom: @classroom
users: @users
})
@openModalView(modal)
modal.once 'redeem-users', -> document.location.reload()
2015-12-04 17:11:17 -05:00
application.tracker?.trackEvent 'Classroom started enroll students', category: 'Courses'
onClickActivateSingleLicenseButton: (e) ->
userID = $(e.target).closest('.btn').data('user-id')
if @prepaids.totalMaxRedeemers() - @prepaids.totalRedeemers() > 0
# Have an unused enrollment, enroll student immediately instead of opening the enroll modal
Stuff Partially fix ActivateLicensesModal.spec [IN PROGRESS] Don't display deleted users Move userID to classroom.deletedMembers on user delete (not retroactive) Fix PDF links for course guides, remove old PDFs from repo Remove deprecated SalesView Remove underline for not-yet-linked student names Only show class select when there's more than one Ignore case when sorting student names Use student.broadName instead of name for display and sorting Fix initial load not showing progress after joining a course (hacky) Fix text entry for enrollment number input Fix enrollment statistics Fix enrollment stats completely (and add back in per-class unenrolled count) Add deletedMembers to classroom schema More fixes to enrollment stats (don't count nonmember prepaids) Don't use 0 as implicit false for openSpots Update suggested number of credit to buy automatically Fix classroom edit form ignoring cleared values Add alert text when more users selected than enrollments available Alert user when trying to assign course to unenrolled students Alert user when assigning course to nobody Add some tests for TeacherClassView bulk assign alerts Fix TeacherClassView tests failing without demos Use model/collection.fakeRequests :D Remove unused comment Fix handling of improperly sorted deleted users on clientside Add test for moving deleted users to deletedMembers Add script for moving all deleted classroom members to classroom.deletedMembers Completely rewrite tallying up enrollment statistics Fix some tests to not be dependent on logged-in user Address PR comments Fix default number of enrollments to buy Fix i18n for not enough enrollments Use custom error message for classroom name length
2016-04-07 17:55:42 -04:00
prepaid = @prepaids.find((prepaid) -> prepaid.get('properties')?.endDate? and prepaid.openSpots() > 0)
prepaid = @prepaids.find((prepaid) -> prepaid.openSpots() > 0) unless prepaid
$.ajax({
method: 'POST'
url: _.result(prepaid, 'url') + '/redeemers'
data: { userID: userID }
success: =>
application.tracker?.trackEvent 'Classroom finished enroll student', category: 'Courses', userID: userID
# TODO: do a lighter refresh here. @render() did not work out.
document.location.reload()
error: (jqxhr, textStatus, errorThrown) ->
if jqxhr.status is 402
message = arguments[2]
else
message = "#{jqxhr.status}: #{jqxhr.responseText}"
console.err message
})
else
user = @users.get(userID)
modal = new ActivateLicensesModal({
classroom: @classroom
users: @users
user: user
})
@openModalView(modal)
modal.once 'redeem-users', -> document.location.reload()
application.tracker?.trackEvent 'Classroom started enroll student', category: 'Courses', userID: userID
2015-11-30 14:14:27 -05:00
onClickEditClassDetailsLink: ->
modal = new ClassroomSettingsModal({classroom: @classroom})
@openModalView(modal)
@listenToOnce modal, 'hidden', @render
2015-12-03 17:51:38 -05:00
userLastPlayedString: (user) ->
return '' unless user.sessions?
2015-11-30 14:14:27 -05:00
session = user.sessions.last()
return '' unless session
2016-04-13 12:54:24 -04:00
course = session.collection.course
2015-11-30 14:14:27 -05:00
levelOriginal = session.get('level').original
2016-04-13 12:54:24 -04:00
level = @levels.findWhere({original: levelOriginal})
return "#{course.get('name')}, #{level.get('name')}"
2015-12-01 16:17:21 -05:00
2015-12-03 17:51:38 -05:00
userPlaytimeString: (user) ->
return '' unless user.sessions?
playtime = _.reduce user.sessions.pluck('playtime'), (s1, s2) -> (s1 or 0) + (s2 or 0)
return '' unless playtime
return moment.duration(playtime, 'seconds').humanize()
2015-12-03 17:51:38 -05:00
classStats: ->
stats = {}
2015-12-03 17:51:38 -05:00
playtime = 0
total = 0
for session in @sessions.models
pt = session.get('playtime') or 0
playtime += pt
total += 1
stats.averagePlaytime = if playtime and total then moment.duration(playtime / total, "seconds").humanize() else 0
stats.totalPlaytime = if playtime then moment.duration(playtime, "seconds").humanize() else 0
completeSessions = @sessions.filter (s) -> s.get('state')?.complete
stats.averageLevelsComplete = if @users.size() then (_.size(completeSessions) / @users.size()).toFixed(1) else 'N/A' # '
2015-12-03 17:51:38 -05:00
stats.totalLevelsComplete = _.size(completeSessions)
enrolledUsers = @users.filter (user) -> user.get('coursePrepaidID')
stats.enrolledUsers = _.size(enrolledUsers)
2015-12-03 17:51:38 -05:00
return stats
2015-12-01 16:17:21 -05:00
onClickAddStudentsButton: (e) ->
modal = new InviteToClassroomModal({classroom: @classroom})
2015-12-01 16:27:12 -05:00
@openModalView(modal)
2015-12-04 17:11:17 -05:00
application.tracker?.trackEvent 'Classroom started add students', category: 'Courses', classroomID: @classroom.id
2015-12-01 16:27:12 -05:00
onClickEnableButton: (e) ->
$button = $(e.target).closest('.btn')
courseInstance = @courseInstances.get($button.data('course-instance-cid'))
console.log 'looking for course instance', courseInstance, 'for', $button.data('course-instance-cid'), 'out of', @courseInstances
userID = $button.data('user-id')
$button.attr('disabled', true)
2015-12-04 17:11:17 -05:00
application.tracker?.trackEvent 'Course assign student', category: 'Courses', courseInstanceID: courseInstance.id, userID: userID
onCourseInstanceCreated = =>
courseInstance.addMember(userID)
@listenToOnce courseInstance, 'sync', @render
if courseInstance.isNew()
# adding the first student to this course, so generate the course instance for it
if not courseInstance.saving
courseInstance.save(null, {validate: false})
courseInstance.saving = true
courseInstance.once 'sync', onCourseInstanceCreated
else
onCourseInstanceCreated()
2015-12-06 18:00:00 -05:00
# TODO: update newly visible level progress bar (currently all white)
onClickRemoveStudentLink: (e) ->
user = @users.get($(e.target).closest('a').data('user-id'))
modal = new RemoveStudentModal({
classroom: @classroom
user: user
courseInstances: @courseInstances
})
@openModalView(modal)
modal.once 'remove-student', @onStudentRemoved, @
onStudentRemoved: (e) ->
@users.remove(e.user)
@render()
2015-12-06 23:01:52 -05:00
application.tracker?.trackEvent 'Classroom removed student', category: 'Courses', classroomID: @classroom.id, userID: e.user.id
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)
2016-04-13 12:54:24 -04:00
"/play/level/#{level.get('slug')}?course=#{course.id}&course-instance=#{courseInstance.id}&session=#{session.id}&observing=true"