mirror of
https://github.com/yhirose/cpp-peglib.git
synced 2024-12-22 03:55:29 +00:00
223 lines
6.3 KiB
JavaScript
223 lines
6.3 KiB
JavaScript
// Setup editors
|
|
function setupInfoArea(id) {
|
|
const e = ace.edit(id);
|
|
e.setShowPrintMargin(false);
|
|
e.setOptions({
|
|
readOnly: true,
|
|
highlightActiveLine: false,
|
|
highlightGutterLine: false
|
|
})
|
|
e.renderer.$cursorLayer.element.style.opacity=0;
|
|
return e;
|
|
}
|
|
|
|
function setupEditorArea(id, lsKey) {
|
|
const e = ace.edit(id);
|
|
e.setShowPrintMargin(false);
|
|
e.setValue(localStorage.getItem(lsKey) || '');
|
|
e.moveCursorTo(0, 0);
|
|
return e;
|
|
}
|
|
|
|
const grammar = setupEditorArea("grammar-editor", "grammarText");
|
|
const code = setupEditorArea("code-editor", "codeText");
|
|
|
|
const codeAst = setupInfoArea("code-ast");
|
|
const codeAstOptimized = setupInfoArea("code-ast-optimized");
|
|
const codeProfile = setupInfoArea("code-profile");
|
|
|
|
$('#opt-mode').val(localStorage.getItem('optimizationMode') || 'all');
|
|
$('#start-rule').val(localStorage.getItem('startRule') || '');
|
|
$('#packrat').prop('checked', localStorage.getItem('packrat') === 'true');
|
|
$('#auto-refresh').prop('checked', localStorage.getItem('autoRefresh') === 'true');
|
|
$('#parse').prop('disabled', $('#auto-refresh').prop('checked'));
|
|
|
|
// Parse
|
|
function escapeHtml(unsafe) {
|
|
return unsafe
|
|
.replace(/&/g, "&")
|
|
.replace(/</g, "<")
|
|
.replace(/>/g, ">")
|
|
.replace(/"/g, """)
|
|
.replace(/'/g, "'");
|
|
}
|
|
|
|
function generateErrorListHTML(errors) {
|
|
let html = '<ul>';
|
|
|
|
html += $.map(errors, function (x) {
|
|
if (x.gln && x.gcol) {
|
|
return `<li data-ln="${x.ln}" data-col="${x.col}" data-gln="${x.gln}" data-gcol="${x.gcol}"><span>${x.ln}:${x.col}</span> <span>${escapeHtml(x.msg)}</span></li>`;
|
|
} else {
|
|
return `<li data-ln="${x.ln}" data-col="${x.col}"><span>${x.ln}:${x.col}</span> <span>${escapeHtml(x.msg)}</span></li>`;
|
|
}
|
|
}).join('');
|
|
|
|
html += '<ul>';
|
|
|
|
return html;
|
|
}
|
|
|
|
function updateLocalStorage() {
|
|
localStorage.setItem('grammarText', grammar.getValue());
|
|
localStorage.setItem('codeText', code.getValue());
|
|
localStorage.setItem('optimizationMode', $('#opt-mode').val());
|
|
localStorage.setItem('startRule', $('#start-rule').val());
|
|
localStorage.setItem('packrat', $('#packrat').prop('checked'));
|
|
localStorage.setItem('autoRefresh', $('#auto-refresh').prop('checked'));
|
|
}
|
|
|
|
function parse() {
|
|
const $grammarValidation = $('#grammar-validation');
|
|
const $grammarInfo = $('#grammar-info');
|
|
const grammarText = grammar.getValue();
|
|
|
|
const $codeValidation = $('#code-validation');
|
|
const $codeInfo = $('#code-info');
|
|
const codeText = code.getValue();
|
|
|
|
const optimizationMode = $('#opt-mode').val();
|
|
const startRule = $('#start-rule').val();
|
|
const packrat = $('#packrat').prop('checked');
|
|
|
|
$grammarInfo.html('');
|
|
$grammarValidation.hide();
|
|
$codeInfo.html('');
|
|
$codeValidation.hide();
|
|
codeAst.setValue('');
|
|
codeAstOptimized.setValue('');
|
|
codeProfile.setValue('');
|
|
|
|
if (grammarText.length === 0) {
|
|
return;
|
|
}
|
|
|
|
const mode = optimizationMode == 'all';
|
|
|
|
$('#overlay').css({
|
|
'z-index': '1',
|
|
'display': 'block',
|
|
'background-color': 'rgba(0, 0, 0, 0.1)'
|
|
});
|
|
window.setTimeout(() => {
|
|
const data = JSON.parse(Module.lint(grammarText, codeText, mode, packrat, startRule));
|
|
$('#overlay').css({
|
|
'z-index': '-1',
|
|
'display': 'none',
|
|
'background-color': 'rgba(1, 1, 1, 1.0)'
|
|
});
|
|
|
|
if (data.grammar_valid) {
|
|
$grammarValidation.removeClass('validation-invalid').show();
|
|
|
|
codeAst.insert(data.ast);
|
|
codeAstOptimized.insert(data.astOptimized);
|
|
codeProfile.insert(data.profile);
|
|
|
|
if (data.source_valid) {
|
|
$codeValidation.removeClass('validation-invalid').show();
|
|
} else {
|
|
$codeValidation.addClass('validation-invalid').show();
|
|
}
|
|
|
|
if (data.code.length > 0) {
|
|
const html = generateErrorListHTML(data.code);
|
|
$codeInfo.html(html);
|
|
}
|
|
} else {
|
|
$grammarValidation.addClass('validation-invalid').show();
|
|
}
|
|
|
|
if (data.grammar.length > 0) {
|
|
const html = generateErrorListHTML(data.grammar);
|
|
$grammarInfo.html(html);
|
|
}
|
|
}, 0);
|
|
}
|
|
|
|
// Event handing for text editing
|
|
let timer;
|
|
function setupTimer() {
|
|
clearTimeout(timer);
|
|
timer = setTimeout(() => {
|
|
updateLocalStorage();
|
|
if ($('#auto-refresh').prop('checked')) {
|
|
parse();
|
|
}
|
|
}, 750);
|
|
};
|
|
grammar.getSession().on('change', setupTimer);
|
|
code.getSession().on('change', setupTimer);
|
|
|
|
// Event handing in the info area
|
|
function makeOnClickInInfo(editor) {
|
|
return function () {
|
|
const el = $(this);
|
|
editor.navigateTo(el.data('ln') - 1, el.data('col') - 1);
|
|
editor.scrollToLine(el.data('ln') - 1, true, false, null);
|
|
editor.focus();
|
|
|
|
if(el.data('gln') && el.data('gcol')) {
|
|
grammar.navigateTo(el.data('gln') - 1, el.data('gcol') - 1);
|
|
grammar.scrollToLine(el.data('gln') - 1, true, false, null);
|
|
}
|
|
}
|
|
};
|
|
$('#grammar-info').on('click', 'li', makeOnClickInInfo(grammar));
|
|
$('#code-info').on('click', 'li', makeOnClickInInfo(code));
|
|
|
|
// Event handing in the AST optimization
|
|
$('#opt-mode').on('change', setupTimer);
|
|
$('#start-rule').on('keydown', setupTimer);
|
|
$('#packrat').on('change', setupTimer);
|
|
$('#auto-refresh').on('change', () => {
|
|
updateLocalStorage();
|
|
$('#parse').prop('disabled', $('#auto-refresh').prop('checked'));
|
|
setupTimer();
|
|
});
|
|
$('#parse').on('click', parse);
|
|
|
|
// Resize editors to fit their parents
|
|
function resizeEditorsToParent() {
|
|
code.resize();
|
|
code.renderer.updateFull();
|
|
codeAst.resize();
|
|
codeAst.renderer.updateFull();
|
|
codeAstOptimized.resize();
|
|
codeAstOptimized.renderer.updateFull();
|
|
codeProfile.resize();
|
|
codeProfile.renderer.updateFull();
|
|
}
|
|
|
|
// Show windows
|
|
function setupToolWindow(lsKeyName, buttonSel, codeSel) {
|
|
let show = localStorage.getItem(lsKeyName) === 'true';
|
|
$(buttonSel).prop('checked', show);
|
|
$(codeSel).css({ 'display': show ? 'block' : 'none' });
|
|
|
|
$(buttonSel).on('change', () => {
|
|
show = !show;
|
|
localStorage.setItem(lsKeyName, show);
|
|
$(codeSel).css({ 'display': show ? 'block' : 'none' });
|
|
resizeEditorsToParent();
|
|
});
|
|
}
|
|
setupToolWindow('show-ast', '#show-ast', '#code-ast');
|
|
setupToolWindow('show-ast-optimized', '#show-ast-optimized', '#code-ast-optimized');
|
|
setupToolWindow('show-profile', '#show-profile', '#code-profile');
|
|
|
|
// Show page
|
|
$('#main').css({
|
|
'display': 'flex',
|
|
});
|
|
|
|
// WebAssembly
|
|
var Module = {
|
|
onRuntimeInitialized: function() {
|
|
// Initial parse
|
|
if ($('#auto-refresh').prop('checked')) {
|
|
parse();
|
|
}
|
|
}
|
|
};
|