diff --git a/app/assets/images/pages/home/course2.png b/app/assets/images/pages/home/computer-science-2.png similarity index 100% rename from app/assets/images/pages/home/course2.png rename to app/assets/images/pages/home/computer-science-2.png diff --git a/app/assets/images/pages/home/course3.png b/app/assets/images/pages/home/computer-science-3.png similarity index 100% rename from app/assets/images/pages/home/course3.png rename to app/assets/images/pages/home/computer-science-3.png diff --git a/app/assets/images/pages/home/course4.png b/app/assets/images/pages/home/computer-science-4.png similarity index 100% rename from app/assets/images/pages/home/course4.png rename to app/assets/images/pages/home/computer-science-4.png diff --git a/app/assets/images/pages/home/course5.png b/app/assets/images/pages/home/computer-science-5.png similarity index 100% rename from app/assets/images/pages/home/course5.png rename to app/assets/images/pages/home/computer-science-5.png diff --git a/app/assets/images/pages/home/game-dev-1.png b/app/assets/images/pages/home/game-dev-1.png new file mode 100644 index 000000000..4fe0b8f96 Binary files /dev/null and b/app/assets/images/pages/home/game-dev-1.png differ diff --git a/app/assets/images/pages/home/course1.png b/app/assets/images/pages/home/introduction-to-computer-science.png similarity index 100% rename from app/assets/images/pages/home/course1.png rename to app/assets/images/pages/home/introduction-to-computer-science.png diff --git a/app/assets/images/pages/home/web-dev-1.png b/app/assets/images/pages/home/web-dev-1.png new file mode 100644 index 000000000..29a561a73 Binary files /dev/null and b/app/assets/images/pages/home/web-dev-1.png differ diff --git a/app/lib/LevelLoader.coffee b/app/lib/LevelLoader.coffee index e35af2996..2b3dc9249 100644 --- a/app/lib/LevelLoader.coffee +++ b/app/lib/LevelLoader.coffee @@ -79,6 +79,11 @@ module.exports = class LevelLoader extends CocoClass @listenToOnce @level, 'sync', @onLevelLoaded reportLoadError: -> + window.tracker?.trackEvent 'LevelLoadError', + category: 'Error', + levelSlug: @work?.level?.slug, + unloaded: JSON.stringify(@supermodel.report().map (m) -> _.result(m.model, 'url')) + return if me.isAdmin() or /dev=true/.test(window.location?.href ? '') or reportedLoadErrorAlready reportedLoadErrorAlready = true context = email: me.get('email') diff --git a/app/locale/en.coffee b/app/locale/en.coffee index c23b87ee9..0d1469950 100644 --- a/app/locale/en.coffee +++ b/app/locale/en.coffee @@ -71,7 +71,7 @@ curriculum: "Total curriculum hours:" ffa: "Free for all students" lesson_time: "Lesson time:" - coming_soon: "Coming this fall!" + coming_soon: "More coming soon!" # {change} courses_available_in: "Courses are available in JavaScript, Python, and Java (coming soon!)" boast: "Boasts riddles that are complex enough to fascinate gamers and coders alike." winning: "A winning combination of RPG gameplay and programming homework that pulls off making kid-friendly education legitimately enjoyable." diff --git a/app/models/CocoModel.coffee b/app/models/CocoModel.coffee index 90310d5de..8441d2bc0 100644 --- a/app/models/CocoModel.coffee +++ b/app/models/CocoModel.coffee @@ -309,6 +309,8 @@ class CocoModel extends Backbone.Model sum = 0 data ?= $.extend true, {}, @attributes schema ?= @schema() or {} + if schema.oneOf # get populating the Programmable component config to work + schema = _.find(schema.oneOf, {type: 'object'}) addedI18N = false if schema.properties?.i18n and _.isPlainObject(data) and not data.i18n? data.i18n = {'-':{'-':'-'}} # mongoose doesn't work with empty objects @@ -318,7 +320,11 @@ class CocoModel extends Backbone.Model if _.isPlainObject data for key, value of data numChanged = 0 - numChanged = @populateI18N(value, childSchema, path+'/'+key) if childSchema = schema.properties?[key] + childSchema = schema.properties?[key] + if not childSchema and _.isObject(schema.additionalProperties) + childSchema = schema.additionalProperties + if childSchema + numChanged = @populateI18N(value, childSchema, path+'/'+key) if numChanged and not path # should only do this for the root object @set key, value sum += numChanged diff --git a/app/styles/play/level/tome/spell.sass b/app/styles/play/level/tome/spell.sass index cc49b2e07..b3928a4b3 100644 --- a/app/styles/play/level/tome/spell.sass +++ b/app/styles/play/level/tome/spell.sass @@ -164,7 +164,6 @@ content: " " display: inline-block position: relative - left: -49px width: 49px top: -30px height: 38px @@ -172,6 +171,20 @@ background-image: url() + .ace_gutter-cell.entry-point:not(.next-entry-point):after + opacity: 0.5 + + .ace_gutter-cell.entry-point.entry-point-indent-0:after + left: -25px + .ace_gutter-cell.entry-point.entry-point-indent-4:after + left: 5px + .ace_gutter-cell.entry-point.entry-point-indent-8:after + left: 33px + .ace_gutter-cell.entry-point.entry-point-indent-12:after + left: 61px + .ace_gutter-cell.entry-point.entry-point-indent-16:after + left: 89px + .ace_gutter-cell.ace_error background-image: url() @@ -179,9 +192,6 @@ .ace_gutter-cell.ace_info background-image: none - .ace_gutter-cell.entry-point:not(.next-entry-point):after - opacity: 0.5 - .ace_marker-layer .ace_bracket // Override faint gray diff --git a/app/templates/courses/teacher-class-view.jade b/app/templates/courses/teacher-class-view.jade index 37cbb69e0..ad32438aa 100644 --- a/app/templates/courses/teacher-class-view.jade +++ b/app/templates/courses/teacher-class-view.jade @@ -122,12 +122,13 @@ block content li(class=(activeTab === "#enrollment-status-tab" ? 'active' : '')) a.course-progress-tab-btn(href='#enrollment-status-tab') .small-details.text-center(data-i18n='teacher.enrollment_status') - - var courses = view.classroom.get('courses').map(function(c) { return view.courses.get(c._id); }); - if _.find(courses, function(c) { return /dev/.test(c.get('slug')); }) - .tab-spacer - li(class=(activeTab === "#student-projects-tab" ? 'active' : '')) - a.course-progress-tab-btn(href='#student-projects-tab') - .small-details.text-center(data-i18n='teacher.projects') + // TODO: Move projects into course progress tab + //- var courses = view.classroom.get('courses').map(function(c) { return view.courses.get(c._id); }); + //if _.find(courses, function(c) { return /dev/.test(c.get('slug')); }) + // .tab-spacer + // li(class=(activeTab === "#student-projects-tab" ? 'active' : '')) + // a.course-progress-tab-btn(href='#student-projects-tab') + // .small-details.text-center(data-i18n='teacher.projects') .tab-filler .tab-content diff --git a/app/templates/new-home-view.jade b/app/templates/new-home-view.jade index 6f7c77ec8..302fc9d9b 100644 --- a/app/templates/new-home-view.jade +++ b/app/templates/new-home-view.jade @@ -232,7 +232,6 @@ block content span#semester-duration #courses-row.row - var conceptsSeen = {}; - - var lastScreenshot = ""; for course, courseIndex in view.courses.models .col-md-3.col-sm-4 .media.course-details(data-course-slug=course.get('slug')) @@ -256,19 +255,16 @@ block content span.spr , - total += 1; span(data-i18n="concepts." + concept) - img.media-object(src="/images/pages/home/course"+(courseIndex+1)+".png") - - lastScreenshot = course.get('screenshot'); + img.media-object(src="/images/pages/home/" + course.get('slug') + ".png") h6.course-duration span.spr(data-i18n="new_home.lesson_time") span.course-hours= course.get('duration') || 0 span.spl.unit(data-i18n="units.hours") - for upcomingCourse in ['Computer Science 6', 'Computer Science 7', 'Computer Science 8'] - .col-md-3.col-sm-4 - .media.disabled - .media-body - h6.course-name= upcomingCourse + ':' - p.small(data-i18n="new_home.coming_soon") - img.media-object(src="/images/pages/home/inprogress.png") + .col-md-3.col-sm-4 + .media.disabled + .media-body + p.small(data-i18n="new_home.coming_soon") + img.media-object(src="/images/pages/home/inprogress.png") .clearfix .text-center diff --git a/app/views/NewHomeView.coffee b/app/views/NewHomeView.coffee index 3c222881f..68fcd7918 100644 --- a/app/views/NewHomeView.coffee +++ b/app/views/NewHomeView.coffee @@ -131,9 +131,33 @@ module.exports = class NewHomeView extends RootView onChangeSchoolLevelDropdown: (e) -> levels = - elementary: {'introduction-to-computer-science': '2-4', 'computer-science-6': '24-30', 'computer-science-7': '30-40', 'computer-science-8': '30-40', default: '16-25', total: '150-215 hours (about two and a half years)'} - middle: {'introduction-to-computer-science': '1-3', 'computer-science-6': '12-14', 'computer-science-7': '14-16', 'computer-science-8': '14-16', default: '8-12', total: '75-100 hours (about one and a half years)'} - high: {'introduction-to-computer-science': '1', 'computer-science-6': '10-12', 'computer-science-7': '12-16', 'computer-science-8': '12-16', default: '8-10', total: '65-85 hours (about one year)'} + elementary: + 'introduction-to-computer-science': '2-4' + 'game-dev-1': '2-3' + 'web-dev-1': '2-3' + 'computer-science-6': '24-30' + 'computer-science-7': '30-40' + 'computer-science-8': '30-40' + default: '16-25' + total: '150-215 hours (about two and a half years)' + middle: + 'introduction-to-computer-science': '1-3' + 'game-dev-1': '1-3' + 'web-dev-1': '1-3' + 'computer-science-6': '12-14' + 'computer-science-7': '14-16' + 'computer-science-8': '14-16' + default: '8-12' + total: '75-100 hours (about one and a half years)' + high: + 'introduction-to-computer-science': '1' + 'game-dev-1': '1-2' + 'web-dev-1': '1-2' + 'computer-science-6': '10-12' + 'computer-science-7': '12-16' + 'computer-science-8': '12-16' + default: '8-10' + total: '65-85 hours (about one year)' level = if e then $(e.target).val() else 'middle' @$el.find('#courses-row .course-details').each -> slug = $(@).data('course-slug') diff --git a/app/views/editor/level/LevelEditView.coffee b/app/views/editor/level/LevelEditView.coffee index 0f9b0d216..230d18a94 100644 --- a/app/views/editor/level/LevelEditView.coffee +++ b/app/views/editor/level/LevelEditView.coffee @@ -2,6 +2,7 @@ RootView = require 'views/core/RootView' template = require 'templates/editor/level/edit' Level = require 'models/Level' LevelSystem = require 'models/LevelSystem' +LevelComponent = require 'models/LevelComponent' World = require 'lib/world/world' DocumentFiles = require 'collections/DocumentFiles' LevelLoader = require 'lib/LevelLoader' @@ -217,6 +218,19 @@ module.exports = class LevelEditView extends RootView onPopulateI18N: -> @level.populateI18N() + + levelComponentMap = _(currentView.supermodel.getModels(LevelComponent)) + .map((c) -> [c.get('original'), c]) + .object() + .value() + + for thang, thangIndex in @level.get('thangs') + for thangComponent, thangComponentIndex in thang.components + component = levelComponentMap[thangComponent.original] + configSchema = component.get('configSchema') + path = "/thangs/#{thangIndex}/components/#{thangComponentIndex}/config" + @level.populateI18N(thangComponent.config, configSchema, path) + f = -> document.location.reload() setTimeout(f, 2000) diff --git a/app/views/play/level/tome/SpellView.coffee b/app/views/play/level/tome/SpellView.coffee index b4cc1d44b..f3a759558 100644 --- a/app/views/play/level/tome/SpellView.coffee +++ b/app/views/play/level/tome/SpellView.coffee @@ -1113,6 +1113,7 @@ module.exports = class SpellView extends CocoView for line, index in lines session.removeGutterDecoration index, 'entry-point' session.removeGutterDecoration index, 'next-entry-point' + session.removeGutterDecoration index, "entry-point-indent-#{i}" for i in [0, 4, 8, 12, 16] lineHasComment = @singleLineCommentRegex().test line lineHasCode = line.trim()[0] and not @singleLineCommentOnlyRegex().test line @@ -1146,6 +1147,13 @@ module.exports = class SpellView extends CocoView session.addGutterDecoration index, 'next-entry-point' seenAnEntryPoint = true + # Shift pointer right based on current indentation + # TODO: tabs probably need different horizontal offsets than spaces + indent = 0 + indent++ while /\s/.test(line[indent]) + indent = Math.min(16, Math.floor(indent / 4) * 4) + session.addGutterDecoration index, "entry-point-indent-#{indent}" + previousLine = line previousLineHadComment = lineHasComment previousLineHadCode = lineHasCode diff --git a/scripts/mongodb/updateCourses.js b/scripts/mongodb/updateCourses.js index 414b1dfd8..593e301d8 100644 --- a/scripts/mongodb/updateCourses.js +++ b/scripts/mongodb/updateCourses.js @@ -75,7 +75,7 @@ var courses = description: "Learn the basics of web development in this introductory HTML & CSS course.", duration: NumberInt(1), free: false, - releasePhase: 'beta' + releasePhase: 'released' }, { name: "CS: Web Development 2",