Merge remote-tracking branch 'refs/remotes/codecombat/master'
Before Width: | Height: | Size: 33 KiB After Width: | Height: | Size: 33 KiB |
Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 24 KiB |
Before Width: | Height: | Size: 27 KiB After Width: | Height: | Size: 27 KiB |
Before Width: | Height: | Size: 38 KiB After Width: | Height: | Size: 38 KiB |
BIN
app/assets/images/pages/home/game-dev-1.png
Normal file
After Width: | Height: | Size: 26 KiB |
Before Width: | Height: | Size: 34 KiB After Width: | Height: | Size: 34 KiB |
BIN
app/assets/images/pages/home/web-dev-1.png
Normal file
After Width: | Height: | Size: 20 KiB |
|
@ -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')
|
||||
|
|
|
@ -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."
|
||||
|
|
|
@ -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
|
||||
|
|
|
@ -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
|
||||
|
|
|
@ -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
|
||||
|
|
|
@ -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
|
||||
|
|
|
@ -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')
|
||||
|
|
|
@ -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)
|
||||
|
||||
|
|
|
@ -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
|
||||
|
|
|
@ -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",
|
||||
|
|