karel
Version:
Compilador y evaluador de Karel en javascript
1,566 lines (1,497 loc) • 57.2 kB
JavaScript
$(document).ready(function () {
var codeMirrorThemes = [
'karel',
'3024-day',
'3024-night',
'abcdef',
'ambiance',
'ambiance-mobile',
'base16-dark',
'base16-light',
'bespin',
'blackboard',
'cobalt',
'colorforth',
'dracula',
'eclipse',
'elegant',
'erlang-dark',
'hopscotch',
'icecoder',
'isotope',
'lesser-dark',
'liquibyte',
'material',
'mbo',
'mdn-like',
'midnight',
'monokai',
'neat',
'neo',
'night',
'paraiso-dark',
'paraiso-light',
'pastel-on-dark',
'railscasts',
'rubyblue',
'seti',
'solarized',
'the-matrix',
'tomorrow-night-bright',
'tomorrow-night-eighties',
'ttcn',
'twilight',
'vibrant-ink',
'xq-dark',
'xq-light',
'yeti',
'zenburn',
];
function getTheme() {
return (
(typeof sessionStorage !== 'undefined' &&
sessionStorage.getItem('karel.js:theme')) ||
'karel'
);
}
function setTheme(theme) {
if (codeMirrorThemes.indexOf(theme) === -1) return;
editor.setOption('theme', theme);
if (typeof sessionStorage !== 'undefined') {
sessionStorage.setItem('karel.js:theme', theme);
}
}
function getParser(str) {
language = detectLanguage(str);
switch (language) {
case 'pascal':
return { parser: new karelpascal.Parser(), name: 'pascal' };
break;
case 'java':
return { parser: new kareljava.Parser(), name: 'java' };
break;
case 'ruby':
return { parser: new karelruby.Parser(), name: 'ruby' };
break;
default:
return { parser: new kareljava.Parser(), name: 'pascal' };
break;
}
}
function parseError(str, hash) {
if (hash.recoverable) {
this.trace(str);
} else {
var e = new Error(str);
for (var n in hash) {
if (hash.hasOwnProperty(n)) {
e[n] = hash[n];
}
}
e.text = e.text;
var line = editor.getLine(e.line);
var i = line.indexOf(e.text, hash.loc ? hash.loc.first_column : 0);
if (i == -1) {
i = line.indexOf(e.text);
}
if (i != -1) {
e.loc = {
first_line: e.line,
last_line: e.line,
first_column: i,
last_column: i + e.text.length,
};
} else {
e.loc = {
first_line: e.line,
last_line: e.line + 1,
first_column: 0,
last_column: 0,
};
}
throw e;
}
}
function addEventListeners(world) {
world.runtime.addEventListener('debug', function (evt) {
console.log(evt);
});
world.runtime.addEventListener('call', function (evt) {
$('#pila').prepend(
'<div class="well well-small">' +
evt.function +
'(' +
evt.param +
') Línea <span class="badge badge-info">' +
(evt.line + 1) +
'</span></div>',
);
});
world.runtime.addEventListener('return', function (evt) {
var arreglo = $('#pila > div:first-child').remove();
});
world.runtime.addEventListener('start', function (evt) {
var arreglo = $('#pila > div:first-child').remove();
});
}
function getSyntax(str) {
var parser = getParser(str);
parser.parser.yy.parseError = parseError;
return parser;
}
function parseWorld(xml) {
// Parses the xml and returns a document object.
return new DOMParser().parseFromString(xml, 'text/xml');
}
function htmlEscape(s) {
return s.replace(/</g, '<').replace(/</g, '>');
}
function modalPrompt(label, value, options) {
var dfd = jQuery.Deferred();
$('#prompt_modal h4').html(label);
$('#prompt_modal label').html(label);
$('#prompt_modal .form-group').hide();
if (options) {
$('#prompt_modal .group-select').show();
$('#prompt_modal select').empty();
for (var i = 0; i < options.length; i++) {
$('#prompt_modal select').append(
$('<option/>').text(options[i]).val(options[i]),
);
}
$('#prompt_modal select').focus().val(value);
} else {
$('#prompt_modal .group-value').show();
$('#prompt_modal input[type="text"]').focus().val(value);
}
$('#prompt_modal').modal('show');
function resolve(e) {
if (options) {
dfd.resolve($('#prompt-select').val());
} else {
dfd.resolve($('#prompt-value').val());
}
$('#prompt_modal').modal('hide');
return false;
}
$('#prompt_modal form').on('submit', resolve);
$('#prompt_modal button.btn-primary').on('click', resolve);
$('#prompt_modal').on('hidden.bs.modal', function () {
$('#prompt_modal form').off('submit', resolve);
$('#prompt_modal button.btn-primary').off('click', resolve);
if (dfd.state() == 'pending') {
dfd.reject();
}
});
return dfd;
}
var ERROR_CODES = {
WALL: 'Karel ha chocado con un muro!',
WORLDUNDERFLOW:
'Karel intentó tomar zumbadores en una posición donde no había!',
BAGUNDERFLOW:
'Karel intentó dejar un zumbador pero su mochila estaba vacía!',
INSTRUCTION: 'Karel ha superado el límite de instrucciones!',
STACK: 'La pila de karel se ha desbordado!',
};
var COMPILATION_ERROR = 'Error de compilación';
var ERROR_TOKENS = {
pascal: {
BEGINPROG: '"iniciar-programa"',
BEGINEXEC: '"inicia-ejecución"',
ENDEXEC: '"termina-ejecución"',
ENDPROG: '"finalizar-programa"',
DEF: '"define-nueva-instrucción"',
PROTO: '"define-prototipo-instrucción"',
RET: '"sal-de-instrucción"',
AS: '"como"',
HALT: '"apágate"',
LEFT: '"gira-izquierda"',
FORWARD: '"avanza"',
PICKBUZZER: '"coge-zumbador"',
LEAVEBUZZER: '"deja-zumbador"',
BEGIN: '"inicio"',
END: '"fin"',
THEN: '"entonces"',
WHILE: '"mientras"',
DO: '"hacer"',
REPEAT: '"repetir"',
TIMES: '"veces"',
DEC: '"precede"',
INC: '"sucede"',
IFZ: '"si-es-cero"',
IFNFWALL: '"frente-libre"',
IFFWALL: '"frente-bloqueado"',
IFNLWALL: '"izquierda-libre"',
IFLWALL: '"izquierda-bloqueada"',
IFNRWALL: '"derecha-libre"',
IFRWALL: '"derecha-bloqueada"',
IFWBUZZER: '"junto-a-zumbador"',
IFNWBUZZER: '"no-junto-a-zumbador"',
IFBBUZZER: '"algún-zumbador-en-la-mochila"',
IFNBBUZZER: '"ningún-zumbador-en-la-mochila"',
IFN: '"orientado-al-norte"',
IFS: '"orientado-al-sur"',
IFE: '"orientado-al-este"',
IFW: '"orientado-al-oeste"',
IFNN: '"no-orientado-al-norte"',
IFNS: '"no-orientado-al-sur"',
IFNE: '"no-orientado-al-este"',
IFNW: '"no-orientado-al-oeste"',
ELSE: '"si-no"',
IF: '"si"',
NOT: '"no"',
OR: '"o"',
AND: '"y"',
'(': '"("',
')': '")"',
';': '";"',
NUM: 'un número',
VAR: 'un nombre',
EOF: 'el final del programa',
},
java: {
CLASS: '"class"',
PROG: '"program"',
DEF: '"define"',
DEF: '"void"',
RET: '"return"',
HALT: '"turnoff"',
LEFT: '"turnleft"',
FORWARD: '"move"',
PICKBUZZER: '"pickbeeper"',
LEAVEBUZZER: '"putbeeper"',
WHILE: '"while"',
REPEAT: '"iterate"',
DEC: '"pred"',
INC: '"succ"',
IFZ: '"iszero"',
IFNFWALL: '"frontIsClear"',
IFFWALL: '"frontIsBlocked"',
IFNLWALL: '"leftIsClear"',
IFLWALL: '"leftIsBlocked"',
IFNRWALL: '"rightIsClear"',
IFRWALL: '"rightIsBlocked"',
IFWBUZZER: '"nextToABeeper"',
IFNWBUZZER: '"notNextToABeeper"',
IFBBUZZER: '"anyBeepersInBeeperBag"',
IFNBBUZZER: '"noBeepersInBeeperBag"',
IFN: '"facingNorth"',
IFS: '"facingSouth"',
IFE: '"facingEast"',
IFW: '"facingWest"',
IFNN: '"notFacingNorth"',
IFNS: '"notFacingSouth"',
IFNE: '"notFacingEast"',
IFNW: '"notFacingWest"',
ELSE: '"else"',
IF: '"if"',
NOT: '"!"',
OR: '"||"',
AND: '"&&"',
AND: '"&"',
'(': '"("',
')': '")"',
BEGIN: '"{"',
END: '"}"',
';': '";"',
NUM: 'un número',
VAR: 'un nombre',
EOF: 'el final del programa',
},
};
// Preparación del editor
var editor = CodeMirror.fromTextArea(document.getElementById('editor'), {
lineNumbers: true,
firstLineNumber: 1,
styleActiveLine: true,
viewportMargin: Infinity,
mode: 'karelpascal',
theme: getTheme(),
foldGutter: {
rangeFinder: CodeMirror.fold.indent,
},
gutters: [
'CodeMirror-foldgutter',
'errors',
'breakpoints',
'CodeMirror-linenumbers',
],
});
editor.numBreakpoints = 0;
editor.on('gutterClick', function (instance, line, gutter, clickEvent) {
if (gutter == 'CodeMirror-foldgutter') return;
function makeBreakpoint() {
var marker = document.createElement('div');
marker.style.color = '#822';
marker.innerHTML = '●';
return marker;
}
var markers = instance.lineInfo(line).gutterMarkers;
if (markers && markers.breakpoints) {
instance.setGutterMarker(line, 'breakpoints', null);
editor.numBreakpoints--;
} else {
instance.setGutterMarker(line, 'breakpoints', makeBreakpoint());
editor.numBreakpoints++;
}
});
function validatorCallbacks(message) {
if (message.type == 'error') {
$('#mensajes').trigger('error', { mensaje: message.message });
} else if (message.type == 'info') {
$('#mensajes').trigger('info', { mensaje: message.message });
} else if (message.type == 'invalidCell') {
$('#mensajes').trigger('error', {
mensaje: 'La celda (' + message.x + ', ' + message.y + ') es inválida',
});
} else {
console.error('Mensaje no reconocido', message);
}
}
var world = $('#world')[0];
var h = 100;
var w = 100;
var context = world.getContext('2d');
var wRender = new WorldRender(context, h, w);
var borrar_zumbadores = false;
var zumbadores_anterior = 0;
var fila_evento;
var columna_evento;
var mundo = new World(w, h);
if (location.hash == '#debug') {
mundo.runtime.debug = true;
}
addEventListeners(mundo);
var mundo_editable = true;
var linea_actual = null;
var tab_actual = 'mensajes';
var mensajes_no_leidos = 0;
var currentCell = undefined;
$('a[data-toggle="tab"]').on('shown', function (e) {
tab_actual = e.target.firstChild.nodeValue.toLowerCase().trim();
if (tab_actual == 'mensajes') {
mensajes_no_leidos = 0;
$('#mensajes_cuenta').html('');
}
});
var src = null;
if (sessionStorage) {
var restoredSource = sessionStorage.getItem('karel.js:karelsource');
if (restoredSource) {
editor.setValue(restoredSource);
}
var restoredWorld = sessionStorage.getItem('karel.js:karelworld');
if (restoredWorld) {
src = restoredWorld;
}
}
if (document.location.hash.indexOf('#mundo:') === 0) {
src = decodeURIComponent(document.location.hash.substring(7));
} else if (!src) {
src = $('script#xmlMundo').html();
}
mundo.load(parseWorld(src));
$('#world').attr('width', $('#world').width());
wRender.paint(mundo, world.width, world.height, { editable: mundo_editable });
var interval = null;
function highlightCurrentLine() {
if (linea_actual != null) {
editor.removeLineClass(linea_actual, 'background', 'karel-current-line');
}
if (mundo.runtime.state.line >= 0) {
linea_actual = mundo.runtime.state.line;
editor.addLineClass(linea_actual, 'background', 'karel-current-line');
editor.setCursor({ line: linea_actual, ch: 0 });
}
}
function step() {
// Avanza un paso en la ejecución del código
mundo.runtime.step();
highlightCurrentLine();
var markers = editor.lineInfo(linea_actual).gutterMarkers;
if (markers && markers.breakpoints && interval) {
$('#mensajes').trigger('info', {
mensaje: 'Breakpoint en la línea ' + (linea_actual + 1),
});
clearInterval(interval);
interval = null;
$('#ejecutar em').removeClass('icon-pause').addClass('icon-play');
$('#worldclean').removeAttr('disabled');
$('#paso').removeAttr('disabled');
$('#futuro').removeAttr('disabled');
$('#evaluacion').removeAttr('disabled');
}
if (mundo.dirty) {
mundo.dirty = false;
wRender.paint(mundo, world.width, world.height, { track_karel: true });
}
if (!mundo.runtime.state.running) {
clearInterval(interval);
interval = null;
mensaje_final();
highlightCurrentLine();
// Aún no se permite editar el mundo, pero se podrá si se regresa a su
// estado original.
$('#ejecutar').attr('disabled', 'disabled');
$('#paso').attr('disabled', 'disabled');
$('#futuro').attr('disabled', 'disabled');
$('#evaluacion').attr('disabled', 'disabled');
$('#worldclean').removeAttr('disabled');
}
}
function compile() {
if (sessionStorage) {
sessionStorage.setItem('karel.js:karelsource', editor.getValue());
sessionStorage.setItem('karel.js:karelworld', mundo.save());
}
var syntax = getSyntax(editor.getValue());
$('#pila').html('');
try {
editor.clearGutter('errors');
var allMarks = editor.getAllMarks();
for (var i = 0; i < allMarks.length; i++) {
allMarks[i].clear();
}
var compiled = syntax.parser.parse(editor.getValue());
if (location.hash == '#debug') {
console.log(JSON.stringify(compiled));
}
$('#mensajes').trigger('info', {
mensaje: 'Programa compilado (sintaxis ' + syntax.name + ')',
});
return compiled;
} catch (e) {
if (e.expected) {
e.message =
'Error de compilación en la línea ' +
(e.line + 1) +
': "' +
e.text +
'"\n' +
'Se encontró ' +
ERROR_TOKENS[syntax.name][e.token] +
', se esperaba uno de:';
for (var i = 0; i < e.expected.length; i++) {
e.message +=
' ' +
ERROR_TOKENS[syntax.name][
e.expected[i].substring(1, e.expected[i].length - 1)
];
}
} else {
var translations = {
'Prototype redefinition': 'El prototipo ya había sido declarado',
'Function redefinition': 'La función ya había sido definida',
'Function parameter mismatch':
'El número de parámetros de la llamada a función no coincide con el de su declaración',
'Prototype parameter mismatch':
'El número de parámetros de la función no coincide con el de su prototipo',
'Undefined function': 'Función no definida',
'Unknown variable': 'Variable desconocida',
};
if (typeof e === 'string') {
var message = e;
e = new Error(e);
e.text = message.split(':')[1];
e.line = 0;
e.loc = { first_line: 0, last_line: 0 };
console.log(e);
}
var message = e.message.split(':')[0];
e.message =
'Error de compilación en la línea ' +
(e.line + 1) +
'\n' +
translations[message] +
': "' +
e.text +
'"';
}
var marker = document.createElement('div');
marker.className = 'error';
marker.style.position = 'relative';
marker.innerHTML =
'<img src="" width="16" height="16"/><pre class="error-popup">' +
e.message +
'</pre>';
editor.setGutterMarker(e.line, 'errors', marker);
var first = { line: e.loc.first_line, ch: e.loc.first_column };
var last = { line: e.loc.last_line, ch: e.loc.last_column };
var options = { title: e.message, className: 'parse-error' };
var mark = editor.markText(first, last, options);
$('#mensajes').trigger('error', {
mensaje: '<pre>' + e + '</pre> (sintaxis ' + syntax.name + ')',
});
return null;
}
}
$(window).resize(function (event) {
$('#world').attr('width', $('#world').width());
wRender.paint(mundo, world.width, world.height, {
editable: mundo_editable,
});
});
$('#mensajes').bind('error', function (event, data) {
var d = new Date();
$('#mensajes').prepend(
'<p class="text-error"><strong>[' +
d.toLocaleString() +
']</strong> ' +
data['mensaje'] +
'</p>',
);
if (tab_actual != 'mensajes') {
mensajes_no_leidos++;
$('#mensajes_cuenta').html(
'<span class="badge badge-info">' + mensajes_no_leidos + '</span>',
);
}
});
$('#mensajes').bind('info', function (event, data) {
var d = new Date();
$('#mensajes').prepend(
'<p class="text-info"><strong>[' +
d.toLocaleString() +
']</strong> ' +
data['mensaje'] +
'</p>',
);
if (tab_actual != 'mensajes') {
mensajes_no_leidos++;
$('#mensajes_cuenta').html(
'<span class="badge badge-info">' + mensajes_no_leidos + '</span>',
);
}
});
$('#mensajes').bind('success', function (event, data) {
var d = new Date();
$('#mensajes').prepend(
'<p class="text-success"><strong>[' +
d.toLocaleString() +
']</strong> ' +
data['mensaje'] +
'</p>',
);
if (tab_actual != 'mensajes') {
mensajes_no_leidos++;
$('#mensajes_cuenta').html(
'<span class="badge badge-info">' + mensajes_no_leidos + '</span>',
);
}
});
$('#compilar').click(function (event) {
compile();
editor.focus();
});
function futuro() {
$('#worldclean').attr('disabled', 'disabled');
$('#ejecutar').attr('disabled', 'disabled');
$('#paso').attr('disabled', 'disabled');
$('#futuro').attr('disabled', 'disabled');
$('#evaluacion').attr('disabled', 'disabled');
if (editor.numBreakpoints) {
while (mundo.runtime.step()) {
var markers = editor.lineInfo(mundo.runtime.state.line).gutterMarkers;
if (markers && markers.breakpoints) {
$('#mensajes').trigger('info', {
mensaje: 'Breakpoint en la línea ' + (mundo.runtime.state.line + 1),
});
$('#ejecutar em').removeClass('icon-pause').addClass('icon-play');
$('#worldclean').removeAttr('disabled');
$('#ejecutar').removeAttr('disabled');
$('#paso').removeAttr('disabled');
$('#futuro').removeAttr('disabled');
$('#evaluacion').removeAttr('disabled');
break;
}
}
} else {
mundo.runtime.disableStackEvents = true;
while (mundo.runtime.step());
mundo.runtime.disableStackEvents = false;
}
highlightCurrentLine();
wRender.paint(mundo, world.width, world.height, { track_karel: true });
mensaje_final();
// Aún no se permite editar el mundo, pero se podrá si se regresa a su
// estado original.
editor.focus();
$('#worldclean').removeAttr('disabled');
}
function mensaje_final() {
if (mundo.runtime.state.error) {
$('#mensajes').trigger('error', {
mensaje: ERROR_CODES[mundo.runtime.state.error],
});
return;
}
conteo = '';
if (mundo.dumps[World.DUMP_MOVE])
conteo =
conteo + ' avanza..........' + mundo.runtime.state.moveCount + '\n';
if (mundo.dumps[World.DUMP_LEFT])
conteo =
conteo + ' gira-izquierda..' + mundo.runtime.state.turnLeftCount + '\n';
if (mundo.dumps[World.DUMP_PICK_BUZZER])
conteo =
conteo +
' coge-zumbador...' +
mundo.runtime.state.pickBuzzerCount +
'\n';
if (mundo.dumps[World.DUMP_LEAVE_BUZZER])
conteo =
conteo + ' deja-zumbador...' + mundo.runtime.state.leaveBuzzerCount;
if (conteo != '')
conteo = '<pre> Instrucciones ejecutadas:\n' + conteo + '</pre>';
$('#mensajes').trigger('success', {
mensaje: 'Ejecución terminada!' + conteo,
});
mensaje_validacion();
}
function mensaje_validacion() {
mundo.postValidate(validatorCallbacks).then(
function (didValidation) {
if (didValidation)
$('#mensajes').trigger('success', {
mensaje: 'La solución es correcta',
});
},
function (message) {
$('#mensajes').trigger('error', {
mensaje:
'La solución es incorrecta' + (message ? ': ' + message : ''),
});
},
);
}
$('#futuro').click(function (event) {
if (!mundo_editable) {
futuro();
return;
}
var compiled = compile();
if (compiled != null) {
$('#ejecutar').trigger('lock');
mundo.reset();
mundo.runtime.load(compiled);
mundo.preValidate(validatorCallbacks).then(
function (didValidation) {
if (didValidation) {
$('#mensajes').trigger('success', {
mensaje: 'La validación fue exitosa',
});
}
mundo.runtime.start();
futuro();
},
function (message) {
$('#mensajes').trigger('error', {
mensaje: 'La validación falló' + (message ? ': ' + message : ''),
});
compiled = null;
$('#ejecutar').trigger('unlock');
},
);
}
});
$('#save_out').click(function (event) {
var compiled = compile();
if (compiled == null) {
$('#guardar_salida').html(COMPILATION_ERROR);
return;
}
$('#ejecutar').trigger('lock');
mundo.reset();
mundo.runtime.load(compiled);
mundo.runtime.start();
mundo.runtime.disableStackEvents = true;
while (mundo.runtime.step());
mundo.runtime.disableStackEvents = false;
wRender.paint(mundo, world.width, world.height, { track_karel: true });
if (mundo.runtime.state.error) {
$('#mensajes').trigger('error', {
mensaje: ERROR_CODES[mundo.runtime.state.error],
});
$('#guardar_salida').html(
htmlEscape(ERROR_CODES[mundo.runtime.state.error]),
);
} else {
$('#mensajes').trigger('success', { mensaje: 'Ejecución terminada!' });
mensaje_validacion();
$('#guardar_salida').html(htmlEscape(mundo.output()));
}
highlightCurrentLine();
// Aún no se permite editar el mundo, pero se podrá si se regresa a
// su estado original.
editor.focus();
$('#ejecutar').attr('disabled', 'disabled');
$('#worldclean').removeAttr('disabled');
var blob = new Blob([mundo.output()], { type: 'text/xml' });
$('#guardar_descargar')
.attr('href', window.URL.createObjectURL(blob))
.attr('download', 'mundo.out');
});
$('#ejecutar').bind('lock', function (evt) {
// Bloquea los controles de ejecución y edición
mundo_editable = false; // Previene ediciones del mundo
$('#compilar').attr('disabled', 'disabled');
$('#futuro').attr('disabled', 'disabled');
$('#evaluacion').attr('disabled', 'disabled');
$('#quitar_zumbadores').attr('disabled', 'disabled');
$('#mochila').attr('disabled', 'disabled');
$('#filas').attr('disabled', 'disabled');
$('#columnas').attr('disabled', 'disabled');
$('#inf_zumbadores').attr('disabled', 'disabled');
$('#paso').attr('disabled', 'disabled');
$('#worldclean').attr('disabled', 'disabled');
$('#ejecutar em').removeClass('icon-play').addClass('icon-pause');
$('#posicion_karel').attr('disabled', 'disabled');
$('#orientacion_karel').attr('disabled', 'disabled');
$('#mochila_karel').attr('disabled', 'disabled');
$('#universo').attr('disabled', 'disabled');
$('#dump_avanza').attr('disabled', 'disabled');
$('#dump_gira_izquierda').attr('disabled', 'disabled');
$('#dump_coge_zumbador').attr('disabled', 'disabled');
$('#dump_deja_zumbador').attr('disabled', 'disabled');
editor.setOption('readOnly', true);
});
$('#ejecutar').bind('unlock', function (evt) {
// Desbloquea los controles de ejecución
mundo_editable = true; // Previene ediciones del mundo
$('#compilar').removeAttr('disabled');
$('#ejecutar').removeAttr('disabled');
$('#futuro').removeAttr('disabled');
$('#evaluacion').removeAttr('disabled');
$('#quitar_zumbadores').removeAttr('disabled');
$('#mochila').removeAttr('disabled');
$('#filas').removeAttr('disabled');
$('#columnas').removeAttr('disabled');
$('#inf_zumbadores').removeAttr('disabled');
$('#paso').removeAttr('disabled');
$('#worldclean').removeAttr('disabled');
$('#ejecutar').removeAttr('disabled');
$('#ejecutar em').removeClass('icon-pause').addClass('icon-play');
$('#posicion_karel').removeAttr('disabled');
$('#orientacion_karel').removeAttr('disabled');
$('#mochila_karel').removeAttr('disabled');
$('#universo').removeAttr('disabled');
$('#dump_avanza').removeAttr('disabled');
$('#dump_gira_izquierda').removeAttr('disabled');
$('#dump_coge_zumbador').removeAttr('disabled');
$('#dump_deja_zumbador').removeAttr('disabled');
editor.setOption('readOnly', false);
});
$('#ejecutar').click(function (event) {
if ($('#ejecutar em').hasClass('icon-play')) {
if (mundo_editable) {
var compiled = compile();
if (compiled != null) {
$('#ejecutar').trigger('lock');
mundo.reset();
mundo.runtime.load(compiled);
mundo.preValidate(validatorCallbacks).then(
function (didValidation) {
if (didValidation) {
$('#mensajes').trigger('success', {
mensaje: 'La validación fue exitosa',
});
}
mundo.runtime.start();
interval = setInterval(step, $('#retraso_txt').val());
},
function (message) {
$('#mensajes').trigger('error', {
mensaje:
'La validación falló' + (message ? ': ' + message : ''),
});
$('#ejecutar').trigger('unlock');
compiled = null;
},
);
}
} else {
$('#ejecutar').trigger('lock');
interval = setInterval(step, $('#retraso_txt').val());
}
} else {
clearInterval(interval);
$('#ejecutar em').removeClass('icon-pause').addClass('icon-play');
$('#worldclean').removeAttr('disabled');
$('#paso').removeAttr('disabled');
$('#futuro').removeAttr('disabled');
$('#evaluacion').removeAttr('disabled');
}
});
$('#paso').click(function (event) {
if (!mundo_editable) {
step();
return;
}
var compiled = compile();
if (compiled != null) {
$('#ejecutar').trigger('lock');
mundo.reset();
mundo.runtime.load(compiled);
mundo.preValidate(validatorCallbacks).then(
function (didValidation) {
if (didValidation) {
$('#mensajes').trigger('success', {
mensaje: 'La validación fue exitosa',
});
}
mundo.runtime.start();
$('#paso').removeAttr('disabled');
$('#worldclean').removeAttr('disabled');
$('#futuro').removeAttr('disabled');
$('#evaluacion').removeAttr('disabled');
$('#ejecutar em').removeClass('icon-pause').addClass('icon-play');
step();
},
function (message) {
$('#mensajes').trigger('error', {
mensaje: 'La validación falló' + (message ? ': ' + message : ''),
});
$('#ejecutar').trigger('unlock');
compiled = null;
},
);
}
});
$('#rubysyntax').click(function (event) {
editor.setValue('#TODO poner codigo aqui');
editor.focus();
});
$('#pascalsyntax').click(function (event) {
// editor.getSession().setMode("ace/mode/karelpascal");
editor.setValue(
'iniciar-programa\n inicia-ejecucion\n { TODO poner codigo aquí }\n apagate;\n termina-ejecucion\nfinalizar-programa',
1,
);
editor.focus();
});
$('#javasyntax').click(function (event) {
// editor.getSession().setMode("ace/mode/kareljava");
editor.setValue(
'class program {\n program () {\n // TODO poner codigo aqui\n turnoff();\n }\n}',
1,
);
editor.focus();
});
$('#retraso_minus').click(function () {
var valor = $('#retraso_txt').val() * 1;
if (valor >= 50) {
valor -= 50;
$('#retraso_txt').val(valor);
if (interval) {
clearInterval(interval);
interval = setInterval(step, $('#retraso_txt').val());
}
}
});
$('#retraso_plus').click(function () {
var valor = $('#retraso_txt').val() * 1;
if (valor < 1000) {
valor += 50;
$('#retraso_txt').val(valor);
if (interval) {
clearInterval(interval);
interval = setInterval(step, $('#retraso_txt').val());
}
}
});
$('#retraso_txt').blur(function (event) {
var valor = $('#retraso_txt').val() * 1;
if (valor < 0 || valor > 1000) {
$('#retraso_txt').val(500);
if (interval) {
clearInterval(interval);
interval = setInterval(step, $('#retraso_txt').val());
}
}
});
function guardarMochila() {
var valor = parseInt($('#mochila').val(), 10);
if (Number.isNaN(valor) || valor < 0 || valor > 65534) {
return;
}
mundo.setBagBuzzers(valor);
$('#xmlMundo').html(mundo.save());
}
$('#mochila')
.keyup(guardarMochila)
.blur(function () {
guardarMochila();
$('#mochila').val(mundo.bagBuzzers);
});
$('#codeload').click(function (event) {
var file = document.createElement('input');
file.type = 'file';
file.addEventListener('change', function (evt) {
var files = evt.target.files; // FileList object
// Loop through the FileList and render image files as thumbnails.
for (var i = 0, f; (f = files[i]); i++) {
// Only process text files.
if (
f.type &&
!(f.type.match('text.*') || f.type == 'application/javascript')
) {
continue;
}
var reader = new FileReader();
// Closure to capture the file information.
reader.onload = (function (theFile) {
return function (e) {
editor.setValue(reader.result);
};
})(f);
// Read in the file as a data URL.
reader.readAsText(f);
}
});
file.click();
});
$('#codesave').click(function (event) {
$('#guardar_modal').modal('show');
$('#guardar_salida').html(htmlEscape(editor.getValue()));
var blob = new Blob([editor.getValue()], { type: 'text/xml' });
$('#guardar_descargar')
.attr('href', window.URL.createObjectURL(blob))
.attr('download', 'karel.txt');
});
$('#worldload').click(function (event) {
var file = document.createElement('input');
file.type = 'file';
file.addEventListener('change', function (evt) {
var files = evt.target.files; // FileList object
// Loop through the FileList and render image files as thumbnails.
for (var i = 0, f; (f = files[i]); i++) {
// Only process text files.
if (f.type && !f.type.match('text.*')) {
continue;
}
var reader = new FileReader();
// Closure to capture the file information.
reader.onload = (function (theFile) {
return function (e) {
$('script#xmlMundo').html(reader.result);
$('#worldclean').click();
};
})(f);
// Read in the file as a data URL.
reader.readAsText(f);
}
});
file.click();
});
$('#worldsave').click(function (event) {
$('#guardar_modal').modal('show');
$('#guardar_salida').html(htmlEscape($('script#xmlMundo').html()));
var blob = new Blob([$('script#xmlMundo').html()], { type: 'text/xml' });
$('#guardar_descargar')
.attr('href', window.URL.createObjectURL(blob))
.attr('download', 'mundo.in');
});
$('#worldclean').click(function (event) {
if (linea_actual != null) {
editor.removeLineClass(linea_actual, 'background', 'karel-current-line');
}
$('#ejecutar').trigger('unlock');
$('#pila').html('');
var src = null;
if (document.location.hash.indexOf('#mundo:') === 0) {
src = decodeURIComponent(document.location.hash.substring(7));
} else {
src = $('script#xmlMundo').html();
}
mundo.load(parseWorld(src));
if ($('#filas').val() != mundo.h || $('#columnas').val() != mundo.w) {
$('#filas').val(mundo.h);
h = mundo.h;
$('#columnas').val(mundo.w);
w = mundo.w;
wRender = new WorldRender(context, h, w);
}
if ((mundo.bagBuzzers == -1) != $('#inf_zumbadores').hasClass('active')) {
$('#inf_zumbadores').toggleClass('active');
}
if (mundo.bagBuzzers == -1) {
$('#mochila').val('').attr('disabled', 'disabled');
} else {
$('#mochila').val(mundo.bagBuzzers).removeAttr('disabled');
}
wRender.paint(mundo, world.width, world.height, {
editable: true,
track_karel: true,
});
if (
$('#posicion_karel').hasClass('active') !=
mundo.getDumps(World.DUMP_POSITION)
) {
$('#posicion_karel').button('toggle');
}
if (
$('#orientacion_karel').hasClass('active') !=
mundo.getDumps(World.DUMP_ORIENTATION)
) {
$('#orientacion_karel').button('toggle');
}
if (
$('#mochila_karel').hasClass('active') != mundo.getDumps(World.DUMP_BAG)
) {
$('#mochila_karel').button('toggle');
}
if (
$('#universo').hasClass('active') !=
mundo.getDumps(World.DUMP_ALL_BUZZERS)
) {
$('#universo').button('toggle');
}
if (
$('#dump_avanza').hasClass('active') != mundo.getDumps(World.DUMP_MOVE)
) {
$('#dump_avanza').button('toggle');
}
if (
$('#dump_gira_izquierda').hasClass('active') !=
mundo.getDumps(World.DUMP_LEFT)
) {
$('#dump_gira_izquierda').button('toggle');
}
if (
$('#dump_coge_zumbador').hasClass('active') !=
mundo.getDumps(World.DUMP_PICK_BUZZER)
) {
$('#dump_coge_zumbador').button('toggle');
}
if (
$('#dump_deja_zumbador').hasClass('active') !=
mundo.getDumps(World.DUMP_LEAVE_BUZZER)
) {
$('#dump_deja_zumbador').button('toggle');
}
});
$('#newworld').click(function (event) {
if (linea_actual != null) {
editor.removeLineClass(linea_actual, 'background', 'karel-current-line');
}
mundo = new World(w, h);
$('#ejecutar').trigger('unlock');
mundo.reset();
if (location.hash == '#debug') {
mundo.runtime.debug = true;
}
addEventListeners(mundo);
wRender.paint(mundo, world.width, world.height, {
editable: true,
track_karel: true,
});
$('#xmlMundo').html(mundo.save());
if ($('#posicion_karel').hasClass('active')) {
$('#posicion_karel').button('toggle');
}
if ($('#orientacion_karel').hasClass('active')) {
$('#orientacion_karel').button('toggle');
}
if ($('#inf_zumbadores').hasClass('active')) {
$('#inf_zumbadores').button('toggle');
}
if ($('#mochila_karel').hasClass('active')) {
$('#mochila_karel').button('toggle');
}
if ($('#universo').hasClass('active')) {
$('#universo').button('toggle');
}
if ($('#dump_avanza').hasClass('active')) {
$('#dump_avanza').button('toggle');
}
if ($('#dump_gira_izquierda').hasClass('active')) {
$('#dump_gira_izquierda').button('toggle');
}
if ($('#dump_coge_zumbador').hasClass('active')) {
$('#dump_coge_zumbador').button('toggle');
}
if ($('#dump_deja_zumbador').hasClass('active')) {
$('#dump_deja_zumbador').button('toggle');
}
});
$('#theme').click(function () {
modalPrompt('tema', getTheme(), codeMirrorThemes).then(function (response) {
setTheme(response);
});
});
$('#evaluacion').click(function (event) {
$('#evaluacion_modal').modal('show');
});
$('body').keyup(function (event) {
var repaint = false;
var saveWorld = false;
var tag = event.target.tagName.toLowerCase();
// globals
if (event.which == 27) {
$('#wcontext_menu').css('display', 'none');
}
// not in the editor
if (tag != 'input' && tag != 'textarea' && mundo_editable) {
repaint = true;
saveWorld = true;
if (event.which == 37) {
wRender.moveWest();
saveWorld = false;
} else if (event.which == 38) {
wRender.moveNorth();
saveWorld = false;
} else if (event.which == 39) {
wRender.moveEast();
saveWorld = false;
} else if (event.which == 40) {
wRender.moveSouth();
saveWorld = false;
} else if (currentCell && event.which >= 96 && event.which <= 105) {
mundo.setBuzzers(currentCell.row, currentCell.column, event.which - 96);
} else if (currentCell && event.which >= 48 && event.which <= 57) {
mundo.setBuzzers(currentCell.row, currentCell.column, event.which - 48);
} else if (currentCell && event.which == 73) {
// I
mundo.setBuzzers(currentCell.row, currentCell.column, -1);
} else if (currentCell && event.which == 82) {
// R
mundo.setBuzzers(
currentCell.row,
currentCell.column,
Math.random() * 100,
);
} else if (currentCell && event.which == 78) {
// N
modalPrompt('¿Cuántos zumbadores?', '0').then(function (response) {
mundo.setBuzzers(currentCell.row, currentCell.column, response);
wRender.paint(mundo, world.width, world.height, {
editable: mundo_editable,
});
$('#xmlMundo').html(mundo.save());
});
} else if (currentCell && event.which == 68) {
// D
mundo.toggleDumpCell(currentCell.row, currentCell.column);
} else if (currentCell && (event.which == 75 || event.which == 105)) {
// K
mundo.move(currentCell.row, currentCell.column);
mundo.rotate();
} else {
repaint = false;
}
}
if (repaint) {
wRender.paint(mundo, world.width, world.height, {
editable: mundo_editable,
});
}
if (saveWorld) {
$('#xmlMundo').html(mundo.save());
}
});
$('#world').click(function (event) {
$('#wcontext_menu').css('display', 'none');
// Maneja los clicks en el mundo
if (mundo_editable && currentCell) {
if (currentCell.kind == Kind.Corner) {
if (wRender.polygon) {
var result = wRender.polygonFinish(
currentCell.row,
currentCell.column,
);
if (result) {
for (var i = result.start_row; i < result.finish_row; i++) {
mundo.toggleWall(i, result.start_column, 0); // oeste
}
for (var i = result.start_column; i < result.finish_column; i++) {
mundo.toggleWall(result.start_row, i, 3); // sur
}
}
}
wRender.polygonStart(currentCell.row, currentCell.column);
} else {
wRender.polygonClear();
if (currentCell.kind == Kind.WestWall) {
mundo.toggleWall(currentCell.row, currentCell.column, 0); // oeste
} else if (currentCell.kind == Kind.SouthWall) {
mundo.toggleWall(currentCell.row, currentCell.column, 3); // sur
} else if (currentCell.kind == Kind.Beeper) {
if (borrar_zumbadores) {
mundo.setBuzzers(currentCell.row, currentCell.column, 0);
}
if (event.shiftKey) {
mundo.toggleDumpCell(currentCell.row, currentCell.column);
} else {
zumbadores = mundo.buzzers(currentCell.row, currentCell.column);
if (zumbadores >= 0 && !event.ctrlKey)
mundo.setBuzzers(
currentCell.row,
currentCell.column,
zumbadores + 1,
);
else if (zumbadores > 0 && event.ctrlKey)
mundo.setBuzzers(
currentCell.row,
currentCell.column,
zumbadores - 1,
);
}
}
}
$('#xmlMundo').html(mundo.save());
}
// Volvemos a pintar el canvas
wRender.paint(mundo, world.width, world.height, {
editable: mundo_editable,
});
});
$('#world').mousemove(function (event) {
var x =
event.offsetX ||
event.clientX +
document.body.scrollLeft +
document.documentElement.scrollLeft -
$('#world').offset().left;
var y =
event.offsetY ||
event.clientY +
document.body.scrollTop +
document.documentElement.scrollTop -
$('#world').offset().top;
var cellInfo = wRender.calculateCell(x, world.height - y);
changed = !(
currentCell &&
cellInfo.row == currentCell.row &&
cellInfo.column == currentCell.column &&
cellInfo.kind == currentCell.kind
);
currentCell = cellInfo;
if (mundo_editable && changed) {
wRender.paint(mundo, world.width, world.height, {
editable: mundo_editable,
});
if (cellInfo.kind == Kind.Corner) {
if (wRender.polygon) {
wRender.polygonUpdate(cellInfo.row, cellInfo.column);
wRender.paint(mundo, world.width, world.height, {
editable: mundo_editable,
});
}
wRender.hoverCorner(
cellInfo.row,
cellInfo.column,
world.width,
world.height,
);
} else {
if (cellInfo.kind == Kind.WestWall) {
wRender.hoverWall(
cellInfo.row,
cellInfo.column,
0,
world.width,
world.height,
); // oeste
} else if (cellInfo.kind == Kind.SouthWall) {
wRender.hoverWall(
cellInfo.row,
cellInfo.column,
3,
world.width,
world.height,
); // sur
} else if (cellInfo.kind == Kind.Beeper) {
wRender.hoverBuzzer(
cellInfo.row,
cellInfo.column,
world.width,
world.height,
);
}
}
}
});
$('#world').bind('contextmenu', function (event) {
// Maneja el click derecho sobre el mundo
if (mundo_editable) {
var x = event.pageX;
var y = event.pageY;
columna_evento =
Math.floor(event.offsetX / wRender.tamano_celda) +
wRender.primera_columna -
1;
fila_evento =
Math.floor((world.height - event.offsetY) / wRender.tamano_celda) +
wRender.primera_fila -
1;
if (y + $('#wcontext_menu').height() > $(window).height()) {
y -= $('#wcontext_menu').height();
}
if (x + $('#wcontext_menu').width() > $(window).width()) {
x -= $('#wcontext_menu').width();
}
$('#wcontext_menu').css({
top: y + 'px',
left: x + 'px',
display: 'block',
});
return false;
}
});
$('#world');
[0].onmousewheel = function (event) {
if (
event.wheelDeltaX < 0 &&
wRender.primera_columna + wRender.num_columnas < w + 2
) {
wRender.primera_columna += 1;
} else if (event.wheelDeltaX > 0 && wRender.primera_columna > 1) {
wRender.primera_columna -= 1;
}
if (
event.wheelDeltaY > 0 &&
wRender.primera_fila + wRender.num_filas < h + 2
) {
wRender.primera_fila += 1;
} else if (event.wheelDeltaY < 0 && wRender.primera_fila > 1) {
wRender.primera_fila -= 1;
}
wRender.paint(mundo, world.width, world.height, {
editable: mundo_editable,
});
return false;
};
$('#world')
.hammer()
.on('drag', function (event) {
var x = event.gesture.deltaX % 2;
var y = event.gesture.deltaY % 2;
if (
event.gesture.deltaX < 0 &&
wRender.primera_columna + wRender.num_columnas < w + 2 &&
x == 0
) {
wRender.primera_columna += 1;
} else if (
event.gesture.deltaX > 0 &&
wRender.primera_columna > 1 &&
x == 0
) {
wRender.primera_columna -= 1;
}
if (
event.gesture.deltaY > 0 &&
wRender.primera_fila + wRender.num_filas < h + 2 &&
y == 0
) {
wRender.primera_fila += 1;
} else if (
event.gesture.deltaY < 0 &&
wRender.primera_fila > 1 &&
y == 0
) {
wRender.primera_fila -= 1;
}
wRender.paint(mundo, world.width, world.height, {
editable: mundo_editable,
});
});
$('#world')
.hammer()
.on('release', function (event) {
event.gesture.preventDefault();
});
$('#world').on('mouseup', function (event) {
event.preventDefault();
});
Hammer($('#world')).on('dragstart', function (event) {
$(document.body).css('cursor', 'move');
});
Hammer($('#world')).on('dragend', function (event) {
$(document.body).css('cursor', 'auto');
currentCell = undefined;
});
$('#filas').blur(function (event) {
var valor = parseInt($(this).val());
if (0 > valor || valor > 10000 || valor == h || Number.isNaN(valor)) {
$(this).val(h);
return;
}
h = valor;
mundo.resize(w, h);
addEventListeners(mundo);
wRender = new WorldRender(context, h, w);
wRender.paint(mundo, world.width, world.height, {
editable: true,
track_karel: true,
});
$('#xmlMundo').html(mundo.save());
});
$('#columnas').blur(function (event) {
var valor = parseInt($(this).val());
if (0 > valor || valor > 10000 || valor == w || Number.isNaN(valor)) {
$(this).val(w);
return;
}
w = valor;
mundo.resize(w, h);
addEventListeners(mundo);
wRender = new WorldRender(context, h, w);
wRender.paint(mundo, world.width, world.height, {
editable: true,
track_karel: true,
});
$('#xmlMundo').html(mundo.save());
});
$('#inf_zumbadores').click(function (event) {
if ($(this).hasClass('active')) {
// ya hay infinitos
mundo.setBagBuzzers(zumbadores_anterior);
$('#mochila').val(zumbadores_anterior);
$(this).removeClass('active');
$('#mochila').removeAttr('disabled');
} else {
// hay finitos
zumbadores_anterior = $('#mochila').val();
mundo.setBagBuzzers(-1);
$(this).addClass('active');
$('#mochila').attr('disabled', 1);
}
$('#xmlMundo').html(mundo.save());
});
$('#quitar_zumbadores').click(function (event) {
if ($(this).hasClass('active')) {
// ya hay infinitos
borrar_zumbadores = false;
$(this).removeClass('active');
} else {
// hay finitos
borrar_zumbadores = true;
$(this).addClass('active');
}
$('#xmlMundo').html(mundo.save());
});
$('#go_home').click(function (event) {
wRender.primera_fila = 1;
wRender.primera_columna = 1;
wRender.paint(mundo, world.width, world.height, {
editable: mundo_editable,
});
});
$('#follow_karel').click(function (event) {
wRender.primera_fila = mundo.i;
wRender.primera_columna = mundo.j;
wRender.paint(mundo, world.width, world.height, {
editable: mundo_editable,
});
});
$('#posicion_karel').click(function (event) {
mundo.toggleDumps(World.DUMP_POSITION);
$('#xmlMundo').html(mundo.save());
});
$('#orientacion_karel').click(function (event) {
mundo.toggleDumps(World.DUMP_ORIENTATION);
$('#xmlMundo').html(mundo.save());
});
$('#mochil