$(function() { var ACE_FILES_PATH = '/assets/ace/'; var ADEQUATE_PERCENTAGE = 50; var ALT_1_KEY_CODE = 161; var ALT_2_KEY_CODE = 8220; var ALT_3_KEY_CODE = 182; var ALT_4_KEY_CODE = 162; var ALT_R_KEY_CODE = 174; var ALT_S_KEY_CODE = 8218; var ALT_T_KEY_CODE = 8224; var FILENAME_URL_PLACEHOLDER = '{filename}'; var SUCCESSFULL_PERCENTAGE = 90; var THEME = 'ace/theme/textmate'; var editors = []; var active_file = undefined; var active_frame = undefined; var running = false; var flowrUrl = 'http://vm-teusner-webrtc.eaalab.hpi.uni-potsdam.de:3000/api/exceptioninfo?id=&lang=auto' var flowrResultHtml = '
' var ajaxError = function(response) { $.flash.danger({ text: (response && response.responseJSON && response.responseJSON.message) || $('#flash').data('message-failure') }); }; var clearOutput = function(output) { $('#output pre').remove(); }; var collectFiles = function() { var editable_editors = _.filter(editors, function(editor) { return !editor.getReadOnly(); }); return _.map(editable_editors, function(editor) { return { content: editor.getValue(), file_id: $(editor.container).data('file-id') }; }); }; var configureEditors = function() { _.each(['modePath', 'themePath', 'workerPath'], function(attribute) { ace.config.set(attribute, ACE_FILES_PATH); }); }; var confirmDestroy = function(event) { event.preventDefault(); if (confirm($(this).data('message-confirm'))) { destroyFile(); } }; var confirmReset = function(event) { event.preventDefault(); if (confirm($(this).data('message-confirm'))) { resetCode(); } }; var confirmSubmission = function(event) { event.preventDefault(); if (confirm($(this).data('message-confirm'))) { submitCode(); } }; var createSubmission = function(initiator, filter, callback) { showSpinner(initiator); var jqxhr = $.ajax({ data: { submission: { cause: $(initiator).data('cause') || $(initiator).prop('id'), exercise_id: $('#editor').data('exercise-id'), files_attributes: (filter || _.identity)(collectFiles()) } }, dataType: 'json', method: 'POST', url: $(initiator).data('url') || $('#editor').data('submissions-url') }); jqxhr.always(hideSpinner); jqxhr.done(callback); jqxhr.fail(ajaxError); }; var destroyFile = function() { createSubmission($('#destroy-file'), function(files) { return _.reject(files, function(file) { return file.file_id === active_file.id; }); }, function() { Turbolinks.visit(window.location.pathname); }); }; var downloadCode = function(event) { event.preventDefault(); createSubmission(this, null,function(response) { var url = response.download_url.replace(FILENAME_URL_PLACEHOLDER, active_file.filename); window.location = url; }); }; var evaluateCode = function(url, streamed, callback) { eval('evaluateCode' + (streamed ? 'With' : 'Without') + 'StreamedResponse')(url, callback); }; var evaluateCodeWithStreamedResponse = function(url, callback) { var event_source = new EventSource(url); event_source.addEventListener('close', function(event) { event_source.close(); hideSpinner(); running = false; toggleButtonStates(); if (JSON.parse(event.data).code !== 200) { ajaxError(); showTab(1); } }); event_source.addEventListener('error', ajaxError); event_source.addEventListener('hint', renderHint); event_source.addEventListener('info', storeContainerInformation); event_source.addEventListener('output', callback); event_source.addEventListener('start', callback); event_source.addEventListener('output', handleStderrOutputForFlowr); event_source.addEventListener('close', handleStderrOutputForFlowr); event_source.addEventListener('status', function(event) { showStatus(JSON.parse(event.data)); }); }; var evaluateCodeWithoutStreamedResponse = function(url, callback) { var jqxhr = $.ajax({ dataType: 'json', method: 'GET', url: url }); jqxhr.always(hideSpinner); jqxhr.done(callback); jqxhr.fail(ajaxError); }; var findOrCreateOutputElement = function(index) { if ($('#output-' + index).isPresent()) { return $('#output-' + index); } else { var element = $('').attr('id', 'output-' + index); $('#output').append(element); return element; } }; var handleKeyPress = function(event) { if (event.which === ALT_1_KEY_CODE) { event.preventDefault(); showTab(0); } else if (event.which === ALT_2_KEY_CODE) { showWorkspaceTab(event); } else if (event.which === ALT_3_KEY_CODE) { event.preventDefault(); showTab(2); } else if (event.which === ALT_4_KEY_CODE) { event.preventDefault(); showTab(3); } else if (event.which === ALT_R_KEY_CODE) { event.preventDefault(); $('#run').trigger('click'); } else if (event.which === ALT_S_KEY_CODE) { event.preventDefault(); $('#assess').trigger('click'); } else if (event.which === ALT_T_KEY_CODE) { event.preventDefault(); $('#test').trigger('click'); } }; var handleScoringResponse = function(response) { printScoringResults(response); var score = _.reduce(response, function(sum, result) { return sum + result.score * result.weight; }, 0).toFixed(2); $('#score').data('score', score); renderScore(); showTab(3); }; var handleTestResponse = function(response) { clearOutput(); printOutput(response[0], false, 0); showStatus(response[0]); showTab(2); }; var hideSpinner = function() { $('button i.fa').show(); $('button i.fa-spin').hide(); }; var initializeEditors = function() { $('.editor').each(function(index, element) { var editor = ace.edit(element); editor.setReadOnly($(element).data('read-only') !== undefined); editor.setShowPrintMargin(false); editor.setTheme(THEME); editors.push(editor); var session = editor.getSession(); session.setMode($(element).data('mode')); session.setTabSize($(element).data('indent-size')); session.setUseSoftTabs(true); session.setUseWrapMode(true); }); }; var initializeEventHandlers = function() { $(document).on('click', '#results a', showOutput); $(document).on('keypress', handleKeyPress); $('a[data-toggle="tab"]').on('show.bs.tab', storeTab); $('#assess').on('click', scoreCode); $('#create-file').on('click', showFileDialog); $('#destroy-file').on('click', confirmDestroy); $('#download').on('click', downloadCode); $('#dropdown-render, #render').on('click', renderCode); $('#dropdown-run, #run').on('click', runCode); $('#dropdown-stop, #stop').on('click', stopCode); $('#dropdown-test, #test').on('click', testCode); $('#save').on('click', saveCode); $('#start').on('click', showWorkspaceTab); $('#start-over').on('click', confirmReset); $('#submit').on('click', confirmSubmission); }; var initializeFileTree = function() { $('#files').jstree($('#files').data('entries')); $('#files').on('click', 'li.jstree-leaf', function() { active_file = { filename: $(this).text(), id: parseInt($(this).attr('id')) }; var frame = $('.editor[data-file-id="' + active_file.id + '"]').parent(); showFrame(frame); toggleButtonStates(); }); }; var initializeTooltips = function() { $('[data-tooltip]').tooltip(); }; var printChunk = function(event) { var output = JSON.parse(event.data); if (output) { printOutput(output, true, 0); } else { clearOutput(); $('#hint').fadeOut(); $('#flowrHint').fadeOut(); showTab(2); } }; var printOutput = function(output, colorize, index) { var element = findOrCreateOutputElement(index); if (!colorize) { var stream = _.sortBy([output.stderr || '', output.stdout || ''], function(stream) { return stream.length; })[1]; element.append(stream); } else if (output.stderr) { element.addClass('text-warning').append(output.stderr); } else if (output.stdout) { element.addClass('text-success').append(output.stdout); } else { element.addClass('text-muted').text($('#output').data('message-no-output')); } }; var printScoringResult = function(result, index) { $('#results').show(); var element = $('#dummies').children().first().clone(); element.removeClass('panel-default').addClass(result.stderr ? 'panel-danger' : (result.score === 1 ? 'panel-success' : 'panel-warning')); element.find('.panel-title .number').text(index + 1); element.find('.row .col-sm-9').eq(0).find('.number').eq(0).text(result.passed); element.find('.row .col-sm-9').eq(0).find('.number').eq(1).text(result.count); element.find('.row .col-sm-9').eq(1).find('.number').eq(0).text((result.score * result.weight).toFixed(2)); element.find('.row .col-sm-9').eq(1).find('.number').eq(1).text(result.weight); element.find('.row .col-sm-9').eq(2).text(result.message); element.find('.row .col-sm-9').eq(3).find('a').attr('href', '#output-' + index); $('#results ul').first().append(element); }; var printScoringResults = function(response) { $('#results ul').first().html(''); $('.test-count .number').html(response.length); clearOutput(); _.each(response, function(result, index) { printOutput(result, false, index); printScoringResult(result, index); }); }; var renderCode = function(event) { event.preventDefault(); if ($('#render').is(':visible')) { createSubmission(this, null, function(response) { var url = response.render_url.replace(FILENAME_URL_PLACEHOLDER, active_file.filename); var pop_up_window = window.open(url); if (pop_up_window) { pop_up_window.onerror = function(message) { clearOutput(); printOutput({ stderr: message }, true, 0); sendError(message); showTab(2); }; } }); } }; var renderHint = function(object) { var hint = object.data || object.hint; if (hint) { $('#hint .panel-body').text(hint); $('#hint').fadeIn(); } }; var renderProgressBar = function(score, maximum_score) { var percentage = score / maximum_score * 100; var progress_bar = $('#score .progress-bar'); progress_bar.removeClass(); if (percentage < ADEQUATE_PERCENTAGE) { progress_bar.addClass('progress-bar progress-bar-danger'); } else if (percentage < SUCCESSFULL_PERCENTAGE) { progress_bar.addClass('progress-bar progress-bar-warning'); } else { progress_bar.addClass('progress-bar progress-bar-success'); } progress_bar.attr({ 'aria-valuemax': maximum_score, 'aria-valuemin': 0, 'aria-valuenow': score }); progress_bar.css('width', percentage + '%'); }; var renderScore = function() { var score = $('#score').data('score'); var maxium_score = $('#score').data('maximum-score'); $('.score').html((score || '?') + ' / ' + maxium_score); renderProgressBar(score, maxium_score); }; var resetCode = function() { showSpinner(this); $.ajax({ dataType: 'json', method: 'GET', url: $('#start-over').data('url') }).success(function(response) { hideSpinner(); _.each(editors, function(editor) { var file_id = $(editor.container).data('file-id'); var file = _.find(response.files, function(file) { return file.id === file_id; }); editor.setValue(file.content); }); }); }; var runCode = function(event) { event.preventDefault(); if ($('#run').is(':visible')) { createSubmission(this, null, function(response) { $('#stop').data('url', response.stop_url); running = true; showSpinner($('#run')); toggleButtonStates(); var url = response.run_url.replace(FILENAME_URL_PLACEHOLDER, active_file.filename); evaluateCode(url, true, printChunk); }); } }; var saveCode = function(event) { event.preventDefault(); createSubmission(this, null, function() { $.flash.success({ text: $('#save').data('message-success') }); }); }; var sendError = function(message) { showSpinner($('#render')); var jqxhr = $.ajax({ data: { error: { message: message } }, dataType: 'json', method: 'POST', url: $('#editor').data('errors-url') }); jqxhr.always(hideSpinner); jqxhr.success(renderHint); }; var scoreCode = function(event) { event.preventDefault(); createSubmission(this, null, function(response) { showSpinner($('#assess')); var url = response.score_url; evaluateCode(url, false, handleScoringResponse); }); }; var showFileDialog = function(event) { event.preventDefault(); createSubmission(this, null, function(response) { $('#code_ocean_file_context_id').val(response.id); $('#modal-file').modal('show'); }); }; var showFrame = function(frame) { active_frame = frame; $('.frame').hide(); frame.show(); }; var showMainFile = function() { var frame = $('.frame[data-role="main_file"]'); var file_id = frame.find('.editor').data('file-id'); active_file = { filename: frame.data('filename'), id: file_id }; $('#files').jstree().select_node(file_id); showFrame(frame); toggleButtonStates(); }; var showOutput = function(event) { event.preventDefault(); showTab(2); $('#output').scrollTo($(this).attr('href')); }; var showRequestedTab = function() { var regexp = /tab=(\d+)/; if (regexp.test(window.location.search)) { var index = regexp.exec(window.location.search)[1] - 1; } else { var index = localStorage.tab; } showTab(index); }; var showSpinner = function(initiator) { $(initiator).find('i.fa, i.glyphicon').hide(); $(initiator).find('i.fa-spin').show(); }; var showStatus = function(output) { if (output.status === 'timeout') { $.flash.danger({ icon: ['fa', 'fa-clock-o'], text: $('#editor').data('message-timeout') }); } else if (output.stderr) { $.flash.danger({ icon: ['fa', 'fa-bug'], text: $('#run').data('message-failure') }); } else { $.flash.success({ icon: ['fa', 'fa-check'], text: $('#run').data('message-success') }); } }; var showTab = function(index) { $('a[data-toggle="tab"]').eq(index || 0).tab('show'); }; var showWorkspaceTab = function(event) { event.preventDefault(); showTab(1); }; var stopCode = function(event) { event.preventDefault(); if ($('#stop').is(':visible')) { var jqxhr = $.ajax({ data: { container_id: $('#stop').data('container').id }, dataType: 'json', method: 'POST', url: $('#stop').data('url') }); jqxhr.always(function() { hideSpinner(); running = false; toggleButtonStates(); }); jqxhr.fail(ajaxError); } }; var storeContainerInformation = function(event) { $('#stop').data('container', JSON.parse(event.data)); }; var storeTab = function(event) { localStorage.tab = $(event.target).parent().index(); }; var submitCode = function() { createSubmission($('#submit'), null, function(response) { if (response.redirect) { localStorage.removeItem('tab'); window.location = response.redirect; } }); }; var testCode = function(event) { event.preventDefault(); if ($('#test').is(':visible')) { createSubmission(this, null, function(response) { showSpinner($('#test')); var url = response.test_url.replace(FILENAME_URL_PLACEHOLDER, active_file.filename); evaluateCode(url, false, handleTestResponse); }); } }; var toggleButtonStates = function() { var is_renderable = active_frame.data('renderable') !== undefined; var is_runnable = active_frame.data('executable') !== undefined && _.contains(['main_file', 'user_defined_file'], active_frame.data('role')); var is_testable = active_frame.data('executable') !== undefined && _.contains(['teacher_defined_test', 'user_defined_test'], active_frame.data('role')); $('#destroy-file').prop('disabled', active_frame.data('role') !== 'user_defined_file'); $('#dropdown-render').toggleClass('disabled', !is_renderable); $('#dropdown-run').toggleClass('disabled', !(is_runnable && !running)); $('#dropdown-stop').toggleClass('disabled', !(is_runnable && running)); $('#dropdown-test').toggleClass('disabled', !is_testable); $('#render').toggle(is_renderable); $('#run').toggle(is_runnable && !running); $('#stop').toggle(is_runnable && running); $('#test').toggle(is_testable); }; if ($('#editor').isPresent()) { configureEditors(); initializeEditors(); initializeEventHandlers(); initializeFileTree(); initializeTooltips(); renderScore(); showMainFile(); showRequestedTab(); } var stderrOutput = '' var handleStderrOutputForFlowr = function(event) { var json = JSON.parse(event.data); if (json.stderr) { stderrOutput += json.stderr; } else if (json.code) { var flowrHintBody = $('#flowrHint .panel-body') jQuery.getJSON(flowrUrl + '&query=' + escape(stderrOutput), function(data) { for (var question in data.queryResults) { // replace everything, not only one occurence var collapsibleTileHtml = flowrResultHtml.replace(/{{collapseId}}/g, 'collapse-' + question).replace(/{{headingId}}/g, 'heading-' + question) var resultTile = $(collapsibleTileHtml) resultTile.find('h4 > a').text(data.queryResults[question].title) resultTile.find('.panel-body').append($(data.queryResults[question].body)) flowrHintBody.append(resultTile) } $('#flowrHint').fadeIn() }) stderrOutput = '' } }; });