Remove bonfire and courseware controller, relying on all challenge related logic in challenge.js. Remove bonfire folder. Move necessary JS to coursewares folder and rename.
This commit is contained in:
9
app.js
9
app.js
@ -40,8 +40,6 @@ var express = require('express'),
|
|||||||
resourcesController = require('./controllers/resources'),
|
resourcesController = require('./controllers/resources'),
|
||||||
userController = require('./controllers/user'),
|
userController = require('./controllers/user'),
|
||||||
nonprofitController = require('./controllers/nonprofits'),
|
nonprofitController = require('./controllers/nonprofits'),
|
||||||
bonfireController = require('./controllers/bonfire'),
|
|
||||||
coursewareController = require('./controllers/courseware'),
|
|
||||||
fieldGuideController = require('./controllers/fieldGuide'),
|
fieldGuideController = require('./controllers/fieldGuide'),
|
||||||
challengeMapController = require('./controllers/challengeMap'),
|
challengeMapController = require('./controllers/challengeMap'),
|
||||||
challengeController = require('./controllers/challenge'),
|
challengeController = require('./controllers/challenge'),
|
||||||
@ -491,11 +489,6 @@ app.get('/api/trello', resourcesController.trelloCalls);
|
|||||||
|
|
||||||
app.get('/api/codepen/twitter/:screenName', resourcesController.codepenResources.twitter);
|
app.get('/api/codepen/twitter/:screenName', resourcesController.codepenResources.twitter);
|
||||||
|
|
||||||
/**
|
|
||||||
* Bonfire related routes
|
|
||||||
*/
|
|
||||||
app.post('/completed-bonfire/', challengeController.completedBonfire);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Field Guide related routes
|
* Field Guide related routes
|
||||||
*/
|
*/
|
||||||
@ -512,7 +505,7 @@ app.post('/completed-field-guide/', fieldGuideController.completedFieldGuide);
|
|||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Courseware related routes
|
* Challenge related routes
|
||||||
*/
|
*/
|
||||||
|
|
||||||
app.get('/getstuff', challengeController.getStuff);
|
app.get('/getstuff', challengeController.getStuff);
|
||||||
|
@ -1,333 +0,0 @@
|
|||||||
var _ = require('lodash'),
|
|
||||||
debug = require('debug')('freecc:cntr:bonfires'),
|
|
||||||
Bonfire = require('./../models/Bonfire'),
|
|
||||||
User = require('./../models/User'),
|
|
||||||
resources = require('./resources'),
|
|
||||||
R = require('ramda');
|
|
||||||
MDNlinks = require('./../seed_data/bonfireMDNlinks');
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Bonfire controller
|
|
||||||
*/
|
|
||||||
|
|
||||||
exports.showAllBonfires = function(req, res) {
|
|
||||||
var completedBonfires = [];
|
|
||||||
if(req.user) {
|
|
||||||
completedBonfires = req.user.completedBonfires.map(function (elem) {
|
|
||||||
return elem._id;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
var noDuplicateBonfires = R.uniq(completedBonfires);
|
|
||||||
var data = {};
|
|
||||||
data.bonfireList = resources.allBonfireNames();
|
|
||||||
data.completedList = noDuplicateBonfires;
|
|
||||||
res.send(data);
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.index = function(req, res) {
|
|
||||||
res.render('bonfire/show.jade', {
|
|
||||||
completedWith: null,
|
|
||||||
title: 'Bonfire Playground',
|
|
||||||
name: 'Bonfire Playground',
|
|
||||||
difficulty: 0,
|
|
||||||
brief: 'Feel free to play around!',
|
|
||||||
details: '',
|
|
||||||
tests: [],
|
|
||||||
challengeSeed: '',
|
|
||||||
cc: req.user ? req.user.bonfiresHash : undefined,
|
|
||||||
progressTimestamps: req.user ? req.user.progressTimestamps : undefined,
|
|
||||||
verb: resources.randomVerb(),
|
|
||||||
phrase: resources.randomPhrase(),
|
|
||||||
compliments: resources.randomCompliment(),
|
|
||||||
bonfires: [],
|
|
||||||
bonfireHash: 'test'
|
|
||||||
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.returnNextBonfire = function(req, res, next) {
|
|
||||||
if (!req.user) {
|
|
||||||
return res.redirect('../bonfires/meet-bonfire');
|
|
||||||
}
|
|
||||||
var completed = req.user.completedBonfires.map(function (elem) {
|
|
||||||
return elem._id;
|
|
||||||
});
|
|
||||||
|
|
||||||
req.user.uncompletedBonfires = resources.allBonfireIds().filter(function (elem) {
|
|
||||||
if (completed.indexOf(elem) === -1) {
|
|
||||||
return elem;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
req.user.save();
|
|
||||||
|
|
||||||
var uncompletedBonfires = req.user.uncompletedBonfires;
|
|
||||||
|
|
||||||
var displayedBonfires = Bonfire.find({'_id': uncompletedBonfires[0]});
|
|
||||||
displayedBonfires.exec(function(err, bonfireFromMongo) {
|
|
||||||
if (err) {
|
|
||||||
return next(err);
|
|
||||||
}
|
|
||||||
var bonfire = bonfireFromMongo.pop();
|
|
||||||
if (typeof bonfire === 'undefined') {
|
|
||||||
req.flash('errors', {
|
|
||||||
msg: "It looks like you've completed all the bonfires we have available. Good job!"
|
|
||||||
});
|
|
||||||
return res.redirect('../bonfires/meet-bonfire');
|
|
||||||
}
|
|
||||||
var nameString = bonfire.name.toLowerCase().replace(/\s/g, '-');
|
|
||||||
return res.redirect('../bonfires/' + nameString);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.returnIndividualBonfire = function(req, res, next) {
|
|
||||||
var dashedName = req.params.bonfireName;
|
|
||||||
|
|
||||||
var bonfireName = dashedName.replace(/\-/g, ' ');
|
|
||||||
|
|
||||||
Bonfire.find({'name': new RegExp(bonfireName, 'i')}, function(err, bonfireFromMongo) {
|
|
||||||
if (err) {
|
|
||||||
next(err);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
if (bonfireFromMongo.length < 1) {
|
|
||||||
req.flash('errors', {
|
|
||||||
msg: "404: We couldn't find a bonfire with that name. Please double check the name."
|
|
||||||
});
|
|
||||||
|
|
||||||
return res.redirect('/bonfires');
|
|
||||||
}
|
|
||||||
|
|
||||||
var bonfire = bonfireFromMongo.pop();
|
|
||||||
var dashedNameFull = bonfire.name.toLowerCase().replace(/\s/g, '-');
|
|
||||||
if (dashedNameFull !== dashedName) {
|
|
||||||
return res.redirect('../bonfires/' + dashedNameFull);
|
|
||||||
}
|
|
||||||
res.render('bonfire/show', {
|
|
||||||
completedWith: null,
|
|
||||||
title: bonfire.name,
|
|
||||||
dashedName: dashedName,
|
|
||||||
name: bonfire.name,
|
|
||||||
difficulty: Math.floor(+bonfire.difficulty),
|
|
||||||
brief: bonfire.description.shift(),
|
|
||||||
details: bonfire.description,
|
|
||||||
tests: bonfire.tests,
|
|
||||||
challengeSeed: bonfire.challengeSeed,
|
|
||||||
points: req.user ? req.user.points : undefined,
|
|
||||||
verb: resources.randomVerb(),
|
|
||||||
phrase: resources.randomPhrase(),
|
|
||||||
compliment: resources.randomCompliment(),
|
|
||||||
bonfires: bonfire,
|
|
||||||
bonfireHash: bonfire._id,
|
|
||||||
MDNkeys: bonfire.MDNlinks,
|
|
||||||
MDNlinks: getMDNlinks(bonfire.MDNlinks)
|
|
||||||
});
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Bonfire Generator
|
|
||||||
* @param req Request Object
|
|
||||||
* @param res Response Object
|
|
||||||
* @returns void
|
|
||||||
*/
|
|
||||||
|
|
||||||
exports.returnGenerator = function(req, res) {
|
|
||||||
res.render('bonfire/generator', {
|
|
||||||
title: null,
|
|
||||||
name: null,
|
|
||||||
difficulty: null,
|
|
||||||
brief: null,
|
|
||||||
details: null,
|
|
||||||
tests: null,
|
|
||||||
challengeSeed: null,
|
|
||||||
bonfireHash: randomString()
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Post for bonfire generation
|
|
||||||
*/
|
|
||||||
|
|
||||||
function randomString() {
|
|
||||||
var chars = '0123456789abcdef';
|
|
||||||
var string_length = 23;
|
|
||||||
var randomstring = 'a';
|
|
||||||
for (var i = 0; i < string_length; i++) {
|
|
||||||
var rnum = Math.floor(Math.random() * chars.length);
|
|
||||||
randomstring += chars.substring(rnum, rnum + 1);
|
|
||||||
}
|
|
||||||
return randomstring;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Helper function to populate the MDN links array.
|
|
||||||
*/
|
|
||||||
|
|
||||||
function getMDNlinks(links) {
|
|
||||||
// takes in an array of links, which are strings
|
|
||||||
var populatedLinks = [];
|
|
||||||
|
|
||||||
// for each key value, push the corresponding link from the MDNlinks object into a new array
|
|
||||||
links.forEach(function(value, index) {
|
|
||||||
populatedLinks.push(MDNlinks[value]);
|
|
||||||
});
|
|
||||||
|
|
||||||
return populatedLinks;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
exports.testBonfire = function(req, res) {
|
|
||||||
var bonfireName = req.body.name,
|
|
||||||
bonfireTests = req.body.tests,
|
|
||||||
bonfireDifficulty = req.body.difficulty,
|
|
||||||
bonfireDescription = req.body.description,
|
|
||||||
bonfireChallengeSeed = req.body.challengeSeed;
|
|
||||||
bonfireTests = bonfireTests.split('\r\n');
|
|
||||||
bonfireDescription = bonfireDescription.split('\r\n');
|
|
||||||
bonfireTests.filter(getRidOfEmpties);
|
|
||||||
bonfireDescription.filter(getRidOfEmpties);
|
|
||||||
bonfireChallengeSeed = bonfireChallengeSeed.replace('\r', '');
|
|
||||||
|
|
||||||
res.render('bonfire/show', {
|
|
||||||
completedWith: null,
|
|
||||||
title: bonfireName,
|
|
||||||
name: bonfireName,
|
|
||||||
difficulty: +bonfireDifficulty,
|
|
||||||
brief: bonfireDescription[0],
|
|
||||||
details: bonfireDescription.slice(1),
|
|
||||||
tests: bonfireTests,
|
|
||||||
challengeSeed: bonfireChallengeSeed,
|
|
||||||
cc: req.user ? req.user.bonfiresHash : undefined,
|
|
||||||
progressTimestamps: req.user ? req.user.progressTimestamps : undefined,
|
|
||||||
verb: resources.randomVerb(),
|
|
||||||
phrase: resources.randomPhrase(),
|
|
||||||
compliment: resources.randomCompliment(),
|
|
||||||
bonfires: [],
|
|
||||||
bonfireHash: 'test'
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
function getRidOfEmpties(elem) {
|
|
||||||
if (elem.length > 0) {
|
|
||||||
return elem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.publicGenerator = function(req, res) {
|
|
||||||
res.render('bonfire/public-generator');
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.generateChallenge = function(req, res) {
|
|
||||||
var bonfireName = req.body.name,
|
|
||||||
bonfireTests = req.body.tests,
|
|
||||||
bonfireDifficulty = req.body.difficulty,
|
|
||||||
bonfireDescription = req.body.description,
|
|
||||||
bonfireChallengeSeed = req.body.challengeSeed;
|
|
||||||
bonfireTests = bonfireTests.split('\r\n');
|
|
||||||
bonfireDescription = bonfireDescription.split('\r\n');
|
|
||||||
bonfireTests.filter(getRidOfEmpties);
|
|
||||||
bonfireDescription.filter(getRidOfEmpties);
|
|
||||||
bonfireChallengeSeed = bonfireChallengeSeed.replace('\r', '');
|
|
||||||
|
|
||||||
|
|
||||||
var response = {
|
|
||||||
_id: randomString(),
|
|
||||||
name: bonfireName,
|
|
||||||
difficulty: bonfireDifficulty,
|
|
||||||
description: bonfireDescription,
|
|
||||||
challengeSeed: bonfireChallengeSeed,
|
|
||||||
tests: bonfireTests
|
|
||||||
};
|
|
||||||
res.send(response);
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.completedBonfire = function (req, res, next) {
|
|
||||||
var isCompletedWith = req.body.bonfireInfo.completedWith || '';
|
|
||||||
var isCompletedDate = Math.round(+new Date());
|
|
||||||
var bonfireHash = req.body.bonfireInfo.bonfireHash;
|
|
||||||
var isSolution = req.body.bonfireInfo.solution;
|
|
||||||
var bonfireName = req.body.bonfireInfo.bonfireName;
|
|
||||||
|
|
||||||
if (isCompletedWith) {
|
|
||||||
var paired = User.find({'profile.username': isCompletedWith
|
|
||||||
.toLowerCase()}).limit(1);
|
|
||||||
paired.exec(function (err, pairedWith) {
|
|
||||||
if (err) {
|
|
||||||
return next(err);
|
|
||||||
} else {
|
|
||||||
var index = req.user.uncompletedBonfires.indexOf(bonfireHash);
|
|
||||||
if (index > -1) {
|
|
||||||
req.user.progressTimestamps.push(Date.now() || 0);
|
|
||||||
req.user.uncompletedBonfires.splice(index, 1);
|
|
||||||
}
|
|
||||||
pairedWith = pairedWith.pop();
|
|
||||||
|
|
||||||
index = pairedWith.uncompletedBonfires.indexOf(bonfireHash);
|
|
||||||
if (index > -1) {
|
|
||||||
pairedWith.progressTimestamps.push(Date.now() || 0);
|
|
||||||
pairedWith.uncompletedBonfires.splice(index, 1);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
pairedWith.completedBonfires.push({
|
|
||||||
_id: bonfireHash,
|
|
||||||
name: bonfireName,
|
|
||||||
completedWith: req.user._id,
|
|
||||||
completedDate: isCompletedDate,
|
|
||||||
solution: isSolution
|
|
||||||
});
|
|
||||||
|
|
||||||
req.user.completedBonfires.push({
|
|
||||||
_id: bonfireHash,
|
|
||||||
name: bonfireName,
|
|
||||||
completedWith: pairedWith._id,
|
|
||||||
completedDate: isCompletedDate,
|
|
||||||
solution: isSolution
|
|
||||||
});
|
|
||||||
|
|
||||||
req.user.save(function (err, user) {
|
|
||||||
if (err) {
|
|
||||||
return next(err);
|
|
||||||
}
|
|
||||||
pairedWith.save(function (err, paired) {
|
|
||||||
if (err) {
|
|
||||||
return next(err);
|
|
||||||
}
|
|
||||||
if (user && paired) {
|
|
||||||
res.send(true);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
req.user.completedBonfires.push({
|
|
||||||
_id: bonfireHash,
|
|
||||||
name: bonfireName,
|
|
||||||
completedWith: null,
|
|
||||||
completedDate: isCompletedDate,
|
|
||||||
solution: isSolution
|
|
||||||
});
|
|
||||||
|
|
||||||
var index = req.user.uncompletedBonfires.indexOf(bonfireHash);
|
|
||||||
if (index > -1) {
|
|
||||||
|
|
||||||
req.user.progressTimestamps.push(Date.now() || 0);
|
|
||||||
req.user.uncompletedBonfires.splice(index, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
req.user.save(function (err, user) {
|
|
||||||
if (err) {
|
|
||||||
return next(err);
|
|
||||||
}
|
|
||||||
if (user) {
|
|
||||||
res.send(true);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
@ -1,384 +0,0 @@
|
|||||||
var _ = require('lodash'),
|
|
||||||
debug = require('debug')('freecc:cntr:courseware'),
|
|
||||||
Courseware = require('./../models/Courseware'),
|
|
||||||
User = require('./../models/User'),
|
|
||||||
resources = require('./resources'),
|
|
||||||
R = require('ramda'),
|
|
||||||
moment = require('moment');
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Courseware controller
|
|
||||||
*/
|
|
||||||
|
|
||||||
exports.showAllCoursewares = function(req, res) {
|
|
||||||
var completedList = [];
|
|
||||||
if(req.user) {
|
|
||||||
completedList = req.user.completedCoursewares.map(function (elem) {
|
|
||||||
return elem._id;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
var noDuplicatedCoursewares = R.uniq(completedList);
|
|
||||||
var data = {};
|
|
||||||
data.coursewareList = resources.allCoursewareNames();
|
|
||||||
data.completedList = noDuplicatedCoursewares;
|
|
||||||
res.send(data);
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.returnNextCourseware = function(req, res, next) {
|
|
||||||
if (!req.user) {
|
|
||||||
return res.redirect('../challenges/learn-how-free-code-camp-works');
|
|
||||||
}
|
|
||||||
|
|
||||||
var completed = req.user.completedCoursewares.map(function (elem) {
|
|
||||||
return elem._id;
|
|
||||||
});
|
|
||||||
|
|
||||||
req.user.uncompletedCoursewares = resources.allCoursewareIds()
|
|
||||||
.filter(function (elem) {
|
|
||||||
if (completed.indexOf(elem) === -1) {
|
|
||||||
return elem;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
req.user.save();
|
|
||||||
|
|
||||||
var uncompletedCoursewares = req.user.uncompletedCoursewares[0];
|
|
||||||
|
|
||||||
var displayedCoursewares = Courseware.find({'_id': uncompletedCoursewares});
|
|
||||||
displayedCoursewares.exec(function(err, courseware) {
|
|
||||||
if (err) {
|
|
||||||
return next(err);
|
|
||||||
}
|
|
||||||
|
|
||||||
courseware = courseware.pop();
|
|
||||||
if (typeof courseware === 'undefined') {
|
|
||||||
req.flash('errors', {
|
|
||||||
msg: "It looks like you've completed all the courses we have " +
|
|
||||||
"available. Good job!"
|
|
||||||
});
|
|
||||||
return res.redirect('../challenges/learn-how-free-code-camp-works');
|
|
||||||
}
|
|
||||||
var nameString = courseware.name.toLowerCase().replace(/\s/g, '-');
|
|
||||||
return res.redirect('../challenges/' + nameString);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.returnIndividualCourseware = function(req, res, next) {
|
|
||||||
var dashedName = req.params.coursewareName;
|
|
||||||
|
|
||||||
var coursewareName = dashedName.replace(/\-/g, ' ');
|
|
||||||
|
|
||||||
Courseware.find({'name': new RegExp(coursewareName, 'i')},
|
|
||||||
function(err, coursewareFromMongo) {
|
|
||||||
if (err) {
|
|
||||||
next(err);
|
|
||||||
}
|
|
||||||
// Handle not found
|
|
||||||
if (coursewareFromMongo.length < 1) {
|
|
||||||
req.flash('errors', {
|
|
||||||
msg: "404: We couldn't find a challenge with that name. " +
|
|
||||||
"Please double check the name."
|
|
||||||
});
|
|
||||||
return res.redirect('/challenges');
|
|
||||||
}
|
|
||||||
var courseware = coursewareFromMongo.pop();
|
|
||||||
|
|
||||||
// Redirect to full name if the user only entered a partial
|
|
||||||
var dashedNameFull = courseware.name.toLowerCase().replace(/\s/g, '-');
|
|
||||||
if (dashedNameFull !== dashedName) {
|
|
||||||
return res.redirect('../challenges/' + dashedNameFull);
|
|
||||||
}
|
|
||||||
|
|
||||||
var challengeType = {
|
|
||||||
0: function() {
|
|
||||||
res.render('coursewares/showHTML', {
|
|
||||||
title: courseware.name,
|
|
||||||
dashedName: dashedName,
|
|
||||||
name: courseware.name,
|
|
||||||
brief: courseware.description[0],
|
|
||||||
details: courseware.description.slice(1),
|
|
||||||
tests: courseware.tests,
|
|
||||||
challengeSeed: courseware.challengeSeed,
|
|
||||||
verb: resources.randomVerb(),
|
|
||||||
phrase: resources.randomPhrase(),
|
|
||||||
compliment: resources.randomCompliment(),
|
|
||||||
coursewareHash: courseware._id,
|
|
||||||
environment: resources.whichEnvironment(),
|
|
||||||
challengeType: courseware.challengeType
|
|
||||||
});
|
|
||||||
},
|
|
||||||
|
|
||||||
1: function() {
|
|
||||||
res.render('coursewares/showJS', {
|
|
||||||
title: courseware.name,
|
|
||||||
dashedName: dashedName,
|
|
||||||
name: courseware.name,
|
|
||||||
brief: courseware.description[0],
|
|
||||||
details: courseware.description.slice(1),
|
|
||||||
tests: courseware.tests,
|
|
||||||
challengeSeed: courseware.challengeSeed,
|
|
||||||
verb: resources.randomVerb(),
|
|
||||||
phrase: resources.randomPhrase(),
|
|
||||||
compliment: resources.randomCompliment(),
|
|
||||||
coursewareHash: courseware._id,
|
|
||||||
challengeType: courseware.challengeType
|
|
||||||
});
|
|
||||||
},
|
|
||||||
|
|
||||||
2: function() {
|
|
||||||
res.render('coursewares/showVideo', {
|
|
||||||
title: courseware.name,
|
|
||||||
dashedName: dashedName,
|
|
||||||
name: courseware.name,
|
|
||||||
details: courseware.description,
|
|
||||||
tests: courseware.tests,
|
|
||||||
video: courseware.challengeSeed[0],
|
|
||||||
verb: resources.randomVerb(),
|
|
||||||
phrase: resources.randomPhrase(),
|
|
||||||
compliment: resources.randomCompliment(),
|
|
||||||
coursewareHash: courseware._id,
|
|
||||||
challengeType: courseware.challengeType
|
|
||||||
});
|
|
||||||
},
|
|
||||||
|
|
||||||
3: function() {
|
|
||||||
res.render('coursewares/showZiplineOrBasejump', {
|
|
||||||
title: courseware.name,
|
|
||||||
dashedName: dashedName,
|
|
||||||
name: courseware.name,
|
|
||||||
details: courseware.description,
|
|
||||||
video: courseware.challengeSeed[0],
|
|
||||||
verb: resources.randomVerb(),
|
|
||||||
phrase: resources.randomPhrase(),
|
|
||||||
compliment: resources.randomCompliment(),
|
|
||||||
coursewareHash: courseware._id,
|
|
||||||
challengeType: courseware.challengeType
|
|
||||||
});
|
|
||||||
},
|
|
||||||
|
|
||||||
4: function() {
|
|
||||||
res.render('coursewares/showZiplineOrBasejump', {
|
|
||||||
title: courseware.name,
|
|
||||||
dashedName: dashedName,
|
|
||||||
name: courseware.name,
|
|
||||||
details: courseware.description,
|
|
||||||
video: courseware.challengeSeed[0],
|
|
||||||
verb: resources.randomVerb(),
|
|
||||||
phrase: resources.randomPhrase(),
|
|
||||||
compliment: resources.randomCompliment(),
|
|
||||||
coursewareHash: courseware._id,
|
|
||||||
challengeType: courseware.challengeType
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return challengeType[courseware.challengeType]();
|
|
||||||
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.testCourseware = function(req, res) {
|
|
||||||
var coursewareName = req.body.name,
|
|
||||||
coursewareTests = req.body.tests,
|
|
||||||
coursewareDifficulty = req.body.difficulty,
|
|
||||||
coursewareDescription = req.body.description,
|
|
||||||
coursewareEntryPoint = req.body.challengeEntryPoint,
|
|
||||||
coursewareChallengeSeed = req.body.challengeSeed;
|
|
||||||
coursewareTests = coursewareTests.split('\r\n');
|
|
||||||
coursewareDescription = coursewareDescription.split('\r\n');
|
|
||||||
coursewareTests.filter(getRidOfEmpties);
|
|
||||||
coursewareDescription.filter(getRidOfEmpties);
|
|
||||||
coursewareChallengeSeed = coursewareChallengeSeed.replace('\r', '');
|
|
||||||
res.render('courseware/show', {
|
|
||||||
completedWith: null,
|
|
||||||
title: coursewareName,
|
|
||||||
name: coursewareName,
|
|
||||||
difficulty: +coursewareDifficulty,
|
|
||||||
brief: coursewareDescription[0],
|
|
||||||
details: coursewareDescription.slice(1),
|
|
||||||
tests: coursewareTests,
|
|
||||||
challengeSeed: coursewareChallengeSeed,
|
|
||||||
challengeEntryPoint: coursewareEntryPoint,
|
|
||||||
cc: req.user ? req.user.coursewaresHash : undefined,
|
|
||||||
progressTimestamps: req.user ? req.user.progressTimestamps : undefined,
|
|
||||||
verb: resources.randomVerb(),
|
|
||||||
phrase: resources.randomPhrase(),
|
|
||||||
compliment: resources.randomCompliment(),
|
|
||||||
coursewares: [],
|
|
||||||
coursewareHash: 'test'
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
function getRidOfEmpties(elem) {
|
|
||||||
if (elem.length > 0) {
|
|
||||||
return elem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.publicGenerator = function(req, res) {
|
|
||||||
res.render('courseware/public-generator');
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.generateChallenge = function(req, res) {
|
|
||||||
var coursewareName = req.body.name,
|
|
||||||
coursewareTests = req.body.tests,
|
|
||||||
coursewareDifficulty = req.body.difficulty,
|
|
||||||
coursewareDescription = req.body.description,
|
|
||||||
coursewareEntryPoint = req.body.challengeEntryPoint,
|
|
||||||
coursewareChallengeSeed = req.body.challengeSeed;
|
|
||||||
coursewareTests = coursewareTests.split('\r\n');
|
|
||||||
coursewareDescription = coursewareDescription.split('\r\n');
|
|
||||||
coursewareTests.filter(getRidOfEmpties);
|
|
||||||
coursewareDescription.filter(getRidOfEmpties);
|
|
||||||
coursewareChallengeSeed = coursewareChallengeSeed.replace('\r', '');
|
|
||||||
|
|
||||||
var response = {
|
|
||||||
_id: randomString(),
|
|
||||||
name: coursewareName,
|
|
||||||
difficulty: coursewareDifficulty,
|
|
||||||
description: coursewareDescription,
|
|
||||||
challengeEntryPoint: coursewareEntryPoint,
|
|
||||||
challengeSeed: coursewareChallengeSeed,
|
|
||||||
tests: coursewareTests
|
|
||||||
};
|
|
||||||
res.send(response);
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.completedCourseware = function (req, res, next) {
|
|
||||||
|
|
||||||
var isCompletedDate = Math.round(+new Date());
|
|
||||||
var coursewareHash = req.body.coursewareInfo.coursewareHash;
|
|
||||||
if (coursewareHash === "bd7139d8c441eddfaeb5bdef") {
|
|
||||||
req.user.finishedWaypoints = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
req.user.completedCoursewares.push({
|
|
||||||
_id: coursewareHash,
|
|
||||||
completedDate: isCompletedDate,
|
|
||||||
name: req.body.coursewareInfo.coursewareName,
|
|
||||||
solution: null,
|
|
||||||
githubLink: null,
|
|
||||||
verified: true
|
|
||||||
});
|
|
||||||
var index = req.user.uncompletedCoursewares.indexOf(coursewareHash);
|
|
||||||
|
|
||||||
if (index > -1) {
|
|
||||||
req.user.progressTimestamps.push(Date.now() || 0);
|
|
||||||
req.user.uncompletedCoursewares.splice(index, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
req.user.save(function (err, user) {
|
|
||||||
if (err) {
|
|
||||||
return next(err);
|
|
||||||
}
|
|
||||||
if (user) {
|
|
||||||
res.sendStatus(200);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.completedZiplineOrBasejump = function (req, res, next) {
|
|
||||||
debug('Inside controller for completed zipline or basejump with data %s',
|
|
||||||
req.body.coursewareInfo);
|
|
||||||
var isCompletedWith = req.body.coursewareInfo.completedWith || false;
|
|
||||||
var isCompletedDate = Math.round(+new Date());
|
|
||||||
var coursewareHash = req.body.coursewareInfo.coursewareHash;
|
|
||||||
var solutionLink = req.body.coursewareInfo.publicURL;
|
|
||||||
var githubLink = req.body.coursewareInfo.challengeType === '4'
|
|
||||||
? req.body.coursewareInfo.githubURL : true;
|
|
||||||
if (!solutionLink || !githubLink) {
|
|
||||||
req.flash('errors', {
|
|
||||||
msg: 'You haven\'t supplied the necessary URLs for us to inspect ' +
|
|
||||||
'your work.'
|
|
||||||
});
|
|
||||||
return res.sendStatus(403);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isCompletedWith) {
|
|
||||||
var paired = User.find({'profile.username': isCompletedWith.toLowerCase()}).limit(1);
|
|
||||||
paired.exec(function (err, pairedWithFromMongo) {
|
|
||||||
if (err) {
|
|
||||||
return next(err);
|
|
||||||
} else {
|
|
||||||
var index = req.user.uncompletedCoursewares.indexOf(coursewareHash);
|
|
||||||
if (index > -1) {
|
|
||||||
req.user.progressTimestamps.push(Date.now() || 0);
|
|
||||||
req.user.uncompletedCoursewares.splice(index, 1);
|
|
||||||
}
|
|
||||||
var pairedWith = pairedWithFromMongo.pop();
|
|
||||||
|
|
||||||
req.user.completedCoursewares.push({
|
|
||||||
_id: coursewareHash,
|
|
||||||
name: req.body.coursewareInfo.coursewareName,
|
|
||||||
completedWith: pairedWith._id,
|
|
||||||
completedDate: isCompletedDate,
|
|
||||||
solution: solutionLink,
|
|
||||||
githubLink: githubLink,
|
|
||||||
verified: false
|
|
||||||
});
|
|
||||||
|
|
||||||
req.user.save(function (err, user) {
|
|
||||||
if (err) {
|
|
||||||
return next(err);
|
|
||||||
}
|
|
||||||
debug('this is the user object returned %s,' +
|
|
||||||
' this is the req.user._id %s, ' +
|
|
||||||
'this is the pairedWith._id %s', user, req.user._id, pairedWith._id);
|
|
||||||
debug(req.user._id.toString() === pairedWith._id.toString());
|
|
||||||
if (req.user._id.toString() === pairedWith._id.toString()) {
|
|
||||||
return res.sendStatus(200);
|
|
||||||
}
|
|
||||||
index = pairedWith.uncompletedCoursewares.indexOf(coursewareHash);
|
|
||||||
if (index > -1) {
|
|
||||||
pairedWith.progressTimestamps.push(Date.now() || 0);
|
|
||||||
pairedWith.uncompletedCoursewares.splice(index, 1);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
pairedWith.completedCoursewares.push({
|
|
||||||
_id: coursewareHash,
|
|
||||||
name: req.body.coursewareInfo.coursewareName,
|
|
||||||
completedWith: req.user._id,
|
|
||||||
completedDate: isCompletedDate,
|
|
||||||
solution: solutionLink,
|
|
||||||
githubLink: githubLink,
|
|
||||||
verified: false
|
|
||||||
});
|
|
||||||
pairedWith.save(function (err, paired) {
|
|
||||||
if (err) {
|
|
||||||
return next(err);
|
|
||||||
}
|
|
||||||
if (user && paired) {
|
|
||||||
return res.sendStatus(200);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
|
|
||||||
req.user.completedCoursewares.push({
|
|
||||||
_id: coursewareHash,
|
|
||||||
name: req.body.coursewareInfo.coursewareName,
|
|
||||||
completedWith: null,
|
|
||||||
completedDate: isCompletedDate,
|
|
||||||
solution: solutionLink,
|
|
||||||
githubLink: githubLink,
|
|
||||||
verified: false
|
|
||||||
});
|
|
||||||
|
|
||||||
var index = req.user.uncompletedCoursewares.indexOf(coursewareHash);
|
|
||||||
if (index > -1) {
|
|
||||||
req.user.progressTimestamps.push(Date.now() || 0);
|
|
||||||
req.user.uncompletedCoursewares.splice(index, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
req.user.save(function (err, user) {
|
|
||||||
if (err) {
|
|
||||||
return next(err);
|
|
||||||
}
|
|
||||||
if (user) {
|
|
||||||
return res.sendStatus(200);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
@ -1,244 +0,0 @@
|
|||||||
var widgets = [];
|
|
||||||
var myCodeMirror = CodeMirror.fromTextArea(document.getElementById("codeEditor"), {
|
|
||||||
lineNumbers: true,
|
|
||||||
mode: "javascript",
|
|
||||||
theme: 'monokai',
|
|
||||||
runnable: true,
|
|
||||||
lint: true,
|
|
||||||
matchBrackets: true,
|
|
||||||
autoCloseBrackets: true,
|
|
||||||
scrollbarStyle: 'null',
|
|
||||||
lineWrapping: true,
|
|
||||||
gutters: ["CodeMirror-lint-markers"],
|
|
||||||
onKeyEvent: doLinting
|
|
||||||
});
|
|
||||||
var editor = myCodeMirror;
|
|
||||||
editor.setSize("100%", "auto");
|
|
||||||
|
|
||||||
// Hijack tab key to enter two spaces intead
|
|
||||||
editor.setOption("extraKeys", {
|
|
||||||
Tab: function(cm) {
|
|
||||||
if (cm.somethingSelected()){
|
|
||||||
cm.indentSelection("add");
|
|
||||||
} else {
|
|
||||||
var spaces = Array(cm.getOption("indentUnit") + 1).join(" ");
|
|
||||||
cm.replaceSelection(spaces);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"Shift-Tab": function(cm) {
|
|
||||||
if (cm.somethingSelected()){
|
|
||||||
cm.indentSelection("subtract");
|
|
||||||
} else {
|
|
||||||
var spaces = Array(cm.getOption("indentUnit") + 1).join(" ");
|
|
||||||
cm.replaceSelection(spaces);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"Ctrl-Enter": function() {
|
|
||||||
bonfireExecute();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
var attempts = 0;
|
|
||||||
if (attempts) {
|
|
||||||
attempts = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
var codeOutput = CodeMirror.fromTextArea(document.getElementById("codeOutput"), {
|
|
||||||
lineNumbers: false,
|
|
||||||
mode: "text",
|
|
||||||
theme: 'monokai',
|
|
||||||
readOnly: 'nocursor',
|
|
||||||
lineWrapping: true
|
|
||||||
});
|
|
||||||
|
|
||||||
codeOutput.setValue('/**\n' +
|
|
||||||
' * Your output will go here.\n' + ' * Console.log() -type statements\n' +
|
|
||||||
' * will appear in your browser\'s\n' + ' * DevTools JavaScript console.\n' +
|
|
||||||
' */');
|
|
||||||
codeOutput.setSize("100%", "100%");
|
|
||||||
var info = editor.getScrollInfo();
|
|
||||||
var after = editor.charCoords({line: editor.getCursor().line + 1, ch: 0}, "local").top;
|
|
||||||
if (info.top + info.clientHeight < after)
|
|
||||||
editor.scrollTo(null, after - info.clientHeight + 3);
|
|
||||||
|
|
||||||
|
|
||||||
var editorValue;
|
|
||||||
|
|
||||||
|
|
||||||
var challengeSeed = challengeSeed || null;
|
|
||||||
var tests = tests || [];
|
|
||||||
|
|
||||||
var allSeeds = '';
|
|
||||||
(function() {
|
|
||||||
challengeSeed.forEach(function(elem) {
|
|
||||||
allSeeds += elem + '\n';
|
|
||||||
});
|
|
||||||
})();
|
|
||||||
|
|
||||||
|
|
||||||
myCodeMirror.setValue(editorValue);
|
|
||||||
|
|
||||||
function doLinting () {
|
|
||||||
editor.operation(function () {
|
|
||||||
for (var i = 0; i < widgets.length; ++i)
|
|
||||||
editor.removeLineWidget(widgets[i]);
|
|
||||||
widgets.length = 0;
|
|
||||||
JSHINT(editor.getValue());
|
|
||||||
for (var i = 0; i < JSHINT.errors.length; ++i) {
|
|
||||||
var err = JSHINT.errors[i];
|
|
||||||
if (!err) continue;
|
|
||||||
var msg = document.createElement("div");
|
|
||||||
var icon = msg.appendChild(document.createElement("span"));
|
|
||||||
icon.innerHTML = "!!";
|
|
||||||
icon.className = "lint-error-icon";
|
|
||||||
msg.appendChild(document.createTextNode(err.reason));
|
|
||||||
msg.className = "lint-error";
|
|
||||||
widgets.push(editor.addLineWidget(err.line - 1, msg, {
|
|
||||||
coverGutter: false,
|
|
||||||
noHScroll: true
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
$('#submitButton').on('click', function () {
|
|
||||||
bonfireExecute();
|
|
||||||
});
|
|
||||||
|
|
||||||
function bonfireExecute() {
|
|
||||||
attempts++;
|
|
||||||
ga('send', 'event', 'Bonfire', 'ran-code', challengeName);
|
|
||||||
userTests= null;
|
|
||||||
$('#codeOutput').empty();
|
|
||||||
var userJavaScript = myCodeMirror.getValue();
|
|
||||||
userJavaScript = removeComments(userJavaScript);
|
|
||||||
userJavaScript = scrapeTests(userJavaScript);
|
|
||||||
// simple fix in case the user forgets to invoke their function
|
|
||||||
|
|
||||||
submit(userJavaScript, function(cls, message) {
|
|
||||||
if (cls) {
|
|
||||||
codeOutput.setValue(message.error);
|
|
||||||
runTests('Error', null);
|
|
||||||
} else {
|
|
||||||
codeOutput.setValue(message.output);
|
|
||||||
message.input = removeLogs(message.input);
|
|
||||||
runTests(null, message);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
var userTests;
|
|
||||||
var testSalt = Math.random();
|
|
||||||
|
|
||||||
|
|
||||||
var scrapeTests = function(userJavaScript) {
|
|
||||||
|
|
||||||
// insert tests from mongo
|
|
||||||
for (var i = 0; i < tests.length; i++) {
|
|
||||||
userJavaScript += '\n' + tests[i];
|
|
||||||
}
|
|
||||||
|
|
||||||
var counter = 0;
|
|
||||||
var regex = new RegExp(/(expect(\s+)?\(.*\;)|(assert(\s+)?\(.*\;)|(assert\.\w.*\;)|(.*\.should\..*\;)/);
|
|
||||||
var match = regex.exec(userJavaScript);
|
|
||||||
while (match != null) {
|
|
||||||
var replacement = '//' + counter + testSalt;
|
|
||||||
userJavaScript = userJavaScript.substring(0, match.index) + replacement + userJavaScript.substring(match.index + match[0].length);
|
|
||||||
|
|
||||||
if (!userTests) {
|
|
||||||
userTests= [];
|
|
||||||
}
|
|
||||||
userTests.push({"text": match[0], "line": counter, "err": null});
|
|
||||||
counter++;
|
|
||||||
match = regex.exec(userJavaScript);
|
|
||||||
}
|
|
||||||
|
|
||||||
return userJavaScript;
|
|
||||||
};
|
|
||||||
|
|
||||||
function removeComments(userJavaScript) {
|
|
||||||
var regex = new RegExp(/(\/\*[^(\*\/)]*\*\/)|\/\/[^\n]*/g);
|
|
||||||
return userJavaScript.replace(regex, '');
|
|
||||||
}
|
|
||||||
|
|
||||||
function removeLogs(userJavaScript) {
|
|
||||||
return userJavaScript.replace(/(console\.[\w]+\s*\(.*\;)/g, '');
|
|
||||||
}
|
|
||||||
|
|
||||||
var pushed = false;
|
|
||||||
var createTestDisplay = function() {
|
|
||||||
if (pushed) {
|
|
||||||
userTests.pop();
|
|
||||||
}
|
|
||||||
for (var i = 0; i < userTests.length;i++) {
|
|
||||||
var test = userTests[i];
|
|
||||||
var testDoc = document.createElement("div");
|
|
||||||
if (test.err != null) {
|
|
||||||
$(testDoc)
|
|
||||||
.html("<div class='row'><div class='col-xs-1 text-center'><i class='ion-close-circled big-error-icon'></i></div><div class='col-xs-11 test-output wrappable grayed-out-test-output'>" + test.text + "</div><div class='col-xs-11 test-output wrappable'>" + test.err + "</div></div><div class='ten-pixel-break'/>")
|
|
||||||
.prependTo($('#testSuite'))
|
|
||||||
} else {
|
|
||||||
$(testDoc)
|
|
||||||
.html("<div class='row'><div class='col-xs-1 text-center'><i class='ion-checkmark-circled big-success-icon'></i></div><div class='col-xs-11 test-output test-vertical-center wrappable '>" + test.text + "</div></div><div class='ten-pixel-break'/>")
|
|
||||||
.appendTo($('#testSuite'));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
};
|
|
||||||
var assert = chai.assert;
|
|
||||||
var expect = chai.expect;
|
|
||||||
var should = chai.should();
|
|
||||||
chai.config.showDiff = true;
|
|
||||||
|
|
||||||
var reassembleTest = function(test, data) {
|
|
||||||
var lineNum = test.line;
|
|
||||||
var regexp = new RegExp("\/\/" + lineNum + testSalt);
|
|
||||||
return data.input.replace(regexp, test.text);
|
|
||||||
};
|
|
||||||
|
|
||||||
var runTests = function(err, data) {
|
|
||||||
var allTestsPassed = true;
|
|
||||||
pushed = false;
|
|
||||||
$('#testSuite').children().remove();
|
|
||||||
if (err && userTests.length > 0) {
|
|
||||||
userTests= [{text:"Program Execution Failure", err: "No user tests were run."}];
|
|
||||||
createTestDisplay();
|
|
||||||
} else if (userTests) {
|
|
||||||
userTests.push(false);
|
|
||||||
pushed = true;
|
|
||||||
userTests.forEach(function(test, ix, arr){
|
|
||||||
try {
|
|
||||||
if (test) {
|
|
||||||
var output = eval(reassembleTest(test, data));
|
|
||||||
}
|
|
||||||
} catch(error) {
|
|
||||||
allTestsPassed = false;
|
|
||||||
arr[ix].err = error.message;
|
|
||||||
} finally {
|
|
||||||
if (!test) {
|
|
||||||
createTestDisplay();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (allTestsPassed) {
|
|
||||||
allTestsPassed = false;
|
|
||||||
showCompletion();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
function showCompletion() {
|
|
||||||
var time = Math.floor(Date.now()) - started;
|
|
||||||
ga('send', 'event', 'Challenge', 'solved', challengeName + ', Time: ' + time +', Attempts: ' + attempts);
|
|
||||||
$('#complete-bonfire-dialog').modal('show');
|
|
||||||
$('#complete-bonfire-dialog').keydown(function(e) {
|
|
||||||
if (e.ctrlKey && e.keyCode == 13) {
|
|
||||||
$('.next-bonfire-button').click();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
@ -13,7 +13,7 @@ block content
|
|||||||
link(rel="stylesheet", href="http://fonts.googleapis.com/css?family=Ubuntu+Mono")
|
link(rel="stylesheet", href="http://fonts.googleapis.com/css?family=Ubuntu+Mono")
|
||||||
script(src='/js/lib/codemirror/mode/javascript/javascript.js')
|
script(src='/js/lib/codemirror/mode/javascript/javascript.js')
|
||||||
script(src='/js/lib/jailed/jailed.js')
|
script(src='/js/lib/jailed/jailed.js')
|
||||||
script(src='/js/lib/bonfire/bonfireInit.js')
|
script(src='/js/lib/bonfire/sandbox.js')
|
||||||
.row
|
.row
|
||||||
script(type="text/javascript").
|
script(type="text/javascript").
|
||||||
var tests = !{JSON.stringify(tests)};
|
var tests = !{JSON.stringify(tests)};
|
||||||
|
@ -14,7 +14,7 @@ block content
|
|||||||
link(rel="stylesheet", href="http://fonts.googleapis.com/css?family=Ubuntu+Mono")
|
link(rel="stylesheet", href="http://fonts.googleapis.com/css?family=Ubuntu+Mono")
|
||||||
script(type='text/javascript', src='/js/lib/codemirror/mode/javascript/javascript.js')
|
script(type='text/javascript', src='/js/lib/codemirror/mode/javascript/javascript.js')
|
||||||
script(type='text/javascript', src='/js/lib/jailed/jailed.js')
|
script(type='text/javascript', src='/js/lib/jailed/jailed.js')
|
||||||
script(type='text/javascript', src='/js/lib/bonfire/bonfireInit.js')
|
script(type='text/javascript', src='/js/lib/coursewares/sandbox.js')
|
||||||
|
|
||||||
.row
|
.row
|
||||||
.col-xs-12.col-sm-12.col-md-4.bonfire-top
|
.col-xs-12.col-sm-12.col-md-4.bonfire-top
|
||||||
|
@ -14,7 +14,7 @@ block content
|
|||||||
link(rel="stylesheet", href="http://fonts.googleapis.com/css?family=Ubuntu+Mono")
|
link(rel="stylesheet", href="http://fonts.googleapis.com/css?family=Ubuntu+Mono")
|
||||||
script(src='/js/lib/codemirror/mode/javascript/javascript.js')
|
script(src='/js/lib/codemirror/mode/javascript/javascript.js')
|
||||||
script(src='/js/lib/jailed/jailed.js')
|
script(src='/js/lib/jailed/jailed.js')
|
||||||
script(src='/js/lib/bonfire/bonfireInit.js')
|
script(src='/js/lib/bonfire/sandbox.js')
|
||||||
script(src='/js/lib/codemirror/mode/xml/xml.js')
|
script(src='/js/lib/codemirror/mode/xml/xml.js')
|
||||||
script(src='/js/lib/codemirror/mode/css/css.js')
|
script(src='/js/lib/codemirror/mode/css/css.js')
|
||||||
script(src='/js/lib/codemirror/mode/htmlmixed/htmlmixed.js')
|
script(src='/js/lib/codemirror/mode/htmlmixed/htmlmixed.js')
|
||||||
|
@ -13,7 +13,7 @@ block content
|
|||||||
link(rel="stylesheet", href="http://fonts.googleapis.com/css?family=Ubuntu+Mono")
|
link(rel="stylesheet", href="http://fonts.googleapis.com/css?family=Ubuntu+Mono")
|
||||||
script(src='/js/lib/codemirror/mode/javascript/javascript.js')
|
script(src='/js/lib/codemirror/mode/javascript/javascript.js')
|
||||||
script(src='/js/lib/jailed/jailed.js')
|
script(src='/js/lib/jailed/jailed.js')
|
||||||
script(src='/js/lib/bonfire/bonfireInit.js')
|
script(src='/js/lib/coursewares/sandbox.js')
|
||||||
.row
|
.row
|
||||||
.col-xs-12.col-sm-12.col-md-4.bonfire-top
|
.col-xs-12.col-sm-12.col-md-4.bonfire-top
|
||||||
#testCreatePanel
|
#testCreatePanel
|
||||||
|
Reference in New Issue
Block a user