Files
gf-core/src/www/gfse/editor.js

1741 lines
49 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
var editor=element("editor");
var compiler_output=element("compiler_output")
/* -------------------------------------------------------------------------- */
var grammar_cache=[]
function reget_grammar(ix) { return grammar_cache[ix]=local.get(ix) }
function get_grammar(ix) { return grammar_cache[ix] || reget_grammar(ix) }
function put_grammar(ix,g) { grammar_cache[ix]=g; local.put(ix,g) }
/* -------------------------------------------------------------------------- */
function initial_view() {
var current=local.get("current");
if(current>0) open_grammar(current-1);
else draw_grammar_list();
//debug(local.get("dir","no server directory yet"));
}
function draw_grammar_list() {
local.put("current",0);
clear(editor);
var uploaded=local.get("dir") && local.get("json_uploaded");
var cloud_upload=
a(jsurl("upload_json()"),
[node("img",{"class":"cloud",
src:"P/1306856253_weather_06.png",alt:"[Up Cloud]",
title: uploaded
? "Click to upload grammar updates to the cloud"
: "Click to store your grammars in the cloud"},
[])]);
var home=div_class("home",[node("h3",{},
[text("Your grammars"),cloud_upload])]);
if(uploaded) {
var cloud_download=
a(jsurl("download_json()"),
[node("img",{"class":"cloud",
src:"P/1343909216_weather_03.png",alt:"[Down Cloud]",
title:"Click to download grammar updates from the cloud"},
[])]);
insertAfter(cloud_download,cloud_upload);
}
editor.appendChild(home)
function del(i) { return function () { delete_grammar(i); } }
function clone(i) { return function (g,b) { clone_grammar(i); } }
function new_extension(i) { return function (g,b) { new_grammar([g]) }}
function item(i,grammar) {
var link=a(jsurl("open_grammar("+i+")"),[text(grammar.basename)]);
return node("tr",{"class":"extensible deletable"},
[td(delete_button(del(i),"Delete this grammar")),
td(link),
td(more(grammar,clone(i),"Clone this grammar"))])
}
if(local.get("count",null)==null)
home.appendChild(text("You have not created any grammars yet."));
else if(local.count==0)
home.appendChild(text("Your grammar list is empty."));
else {
var rows=[];
for(var i=0;i<local.count;i++) {
var grammar=local.get(i,null);
if(grammar && grammar.basename) rows.push(item(i,grammar))
}
home.appendChild(node("table",{"class":"grammar_list"},rows));
}
home.appendChild(
ul([li([a(jsurl("new_grammar()"),[text("New grammar")])])]));
//editor.appendChild(text(local.count));
home.appendChild(empty_id("div","sharing"));
}
function new_grammar(gs) {
gs=gs || [];
var g={basename:"Unnamed",
extends:gs.map(function(g){return g.basename}),
// check that "Unnamed" is not in exts !!
abstract:{cats:[],funs:[]},
concretes:empty_concretes_extending(gs)}
edit_grammar(g);
}
function empty_concretes_extending(gs) {
var concs=[]
var langs=[];
for(var i in gs) {
var g=gs[i];
for(var ci in g.concretes) {
var conc=g.concretes[ci];
var code=conc.langcode;
if(!langs[code])
langs[code]=true,concs.push(new_concrete(code))
}
}
return concs
}
function remove_local_grammar(i) {
local.remove(i);
while(local.count>0 && !local.get(local.count-1))
local.count--;
}
function delete_grammar(ix) {
var g=get_grammar(ix);
var ok=confirm("Do you really want to delete the grammar "+g.basename+"?")
if(ok) {
remove_local_grammar(ix)
remove_cloud_grammar(g)
initial_view();
}
}
function clone_grammar(i) {
var old=get_grammar(i);
var g={basename:old.basename,extends:old.extends || [],
abstract:old.abstract,concretes:old.concretes}
save_grammar(g); // we rely on the serialization to eliminate sharing
draw_grammar_list();
}
function open_grammar(i) {
var g=get_grammar(i);
g.index=i;
local.put("current",i+1);
edit_grammar(g);
}
function close_grammar(g) {
clear(compiler_output);
save_grammar(g);
draw_grammar_list();
}
function reload_grammar(g) { save_grammar(g); edit_grammar(g); }
function save_grammar(g) {
if(g.index==null) g.index=local.count++;
local.put(g.index,g);
}
function edit_grammar(g) {
replaceChildren(editor,draw_grammar(g));
}
function draw_grammar(g) {
var files=div_class("files",[draw_filebar(g),draw_file(g)]);
return div_class("grammar",[draw_namebar(g,files),files])
}
function draw_namebar(g,files) {
var err_ind=empty("small"); // space for error indicator
var cb=compile_button(g,err_ind);
var mb=minibar_button(g,files,err_ind,cb);
var qb=quiz_button(g,err_ind);
//var pb=draw_plainbutton(g,files);
var mxb=draw_matrixbutton(g,files)
var xb=draw_closebutton(g);
return div_class("namebar",
[table([tr([td(draw_name(g)),
td_right([err_ind,cb,mb,qb/*,pb*/,mxb,xb])])])])
}
function draw_name(g) {
return editable("h3",text(g.basename),g,edit_name,"Rename grammar");
}
function draw_closebutton(g) {
var b=button("X",function(){close_grammar(g);});
b.title="Save and Close this grammar";
return b;
}
function draw_plainbutton(g,files) {
var b2;
function show_editor() { edit_grammar(g); }
function show_plain() {
files.innerHTML="<pre class=plain>"+show_grammar(g)+"</pre>"
b.style.display="none";
if(b2) b2.style.display="";
else {
b2=button("Show editor",show_editor);
insertAfter(b2,b);
}
}
var b=button("Show plain",show_plain);
b.title="Show plain text representaiton of the grammar";
return b;
}
function draw_matrixbutton(g,files) {
var b2;
function show_editor() { edit_grammar(g); }
function show_matrix() {
clear(files)
files.appendChild(draw_matrix(g))
b.style.display="none";
if(b2) b2.style.display="";
else {
b2=button("Show editor",show_editor);
insertAfter(b2,b);
}
}
var b=button("Matrix view",show_matrix);
b.title="Show matrix view of the grammar";
return b;
}
function show_compile_error(res,err_ind) {
var dst=compiler_output
replaceInnerHTML(err_ind,res.errorcode=="OK"
? res.errorcode+" "
: "<span class=error_message>"+res.errorcode+" </span>")
if(dst) {
clear(dst);
//var minibarlink=a(res.minibar_url,[text("Minibar")])
if(res.errorcode=="OK")
dst.appendChild(wrap("h3",text("OK")))
else
appendChildren(dst,
[node("h3",{"class":"error_message"},
[text(res.errorcode)]),
wrap("pre",text(res.command)),
wrap("pre",text(res.output))])
}
}
function compile_grammar(g,err_ind,cont) {
function show_error(res) {
show_compile_error(res,err_ind);
if(cont) cont(res)
}
replaceInnerHTML(err_ind,"Compiling...");
replaceInnerHTML(compiler_output,"<h3>Compiling...</h3>");
upload(g,show_error);
}
function compile_button(g,err_ind) {
function compile() { compile_grammar(g,err_ind) }
var b=button("Compile",compile);
b.title="Upload the grammar to the server to check it in GF for errors";
return b;
}
function minibar_button(g,files,err_ind,comp_btn) {
var b2;
var minibar_div=div_id("minibar");
function page_overlay(inner) {
return wrap_class("table","page_overlay",tr(td(inner)))
}
function extend_grammar(cat0,fun_type0,update_minibar) {
var fname0="New"+cat0;
var fun=parse_fun(fname0+" : " + fun_type0).ok;
var lins=[];
var dc=defined_cats(g),df=inherited_funs(g);
var cs=g.concretes
var ext=div_class("grammar_extension")
var form;
var overlay=page_overlay(ext);
/*
function draw_extension_v1() { // version 1
var cat=fun.type[fun.type.length-1]
ext.innerHTML="<h4>Extending "+cat+"</h4>"
var ef=editable("span",draw_fun(g,fun,dc,df),g,edit_fun,
"Edit this function")
var tbl=empty_class("table","extension");
tbl.appendChild(tr([th(text("Abstract")),td([kw("fun "),ef])]));
var anames=arg_names(fun.type);
for(var i in cs) {
var lc=cs[i].langcode;
var l=[kw("lin "),ident(fun.name)];
for(var j in anames) { l.push(text(" ")); l.push(ident(anames[j]));}
l.push(sep(" = "))
l.push(editable("span",text(lins[lc] || "..."),g,edit_lin(lc),
"Edit this linearization"))
tbl.appendChild(tr([th(text(concname(cs[i].langcode))),td(l)]));
}
ext.appendChild(tbl);
ext.appendChild(button("OK",save_extension))
ext.appendChild(button("Cancel",cancel_extension))
}
function edit_fun(g,el) {
function replace(s) {
var p=parse_fun(s);
if(p.ok) {
fun=p.ok;
draw_extension_v1();
return null;
}
else
return p.error;
}
string_editor(el,show_fun(fun),replace);
}
function edit_lin(lc) {
return function (g,el) {
function check(s,cont) {
function replace() {
lins[lc]=s;
draw_extension_v1();
return null;
}
function check2(msg) { if(msg) cont(msg); else replace(); }
check_exp(s,check2);
}
string_editor(el,lins[lc] || "",check,true);
}
}
*/
function draw_extension() { // version 2
var cat=fun.type[fun.type.length-1]
ext.innerHTML="<h4>Extending "+cat+"</h4>"
var fun_name_els=[];
function update_fun_name() {
//console.log("update_fun_name")
fun.name=fun_name.value;
for(var fi in fun_name_els)
fun_name_els[fi].innerHTML=fun.name
}
var fun_name=node("input",{"class":"string_edit",
name:"Abstract",
value:fun.name})
fun_name.onchange=update_fun_name;
fun_name.onblur=update_fun_name;
var ef=wrap("span",[fun_name,sep(" : "),draw_type(fun.type,dc)])
var tbl=empty_class("table","extension");
tbl.appendChild(tr([th(text("Abstract")),td([kw("fun "),ef])]));
var anames=arg_names(fun.type);
for(var i in cs) {
var lc=cs[i].langcode;
var fn=ident(fun.name);
fun_name_els.push(fn);
var l=[kw("lin "),fn];
for(var j in anames) { l.push(text(" ")); l.push(ident(anames[j]));}
l.push(sep(" = "))
l.push(node("input",{"class":"string_edit",
placeholder:"...",size:30,
name:lc,value:lins[lc] || ""}))
tbl.appendChild(tr([th(text(concname(cs[i].langcode))),td(l)]));
}
form=node("form",{},[tbl])
form.onsubmit=save_extension;
form.appendChild(node("input",{type:"submit",value:"OK"}))
form.appendChild(button("Cancel",cancel_extension))
ext.appendChild(form);
// fun_name.focus(); // can't focus before adding it to the document
return fun_name;
}
function save_extension() {
//console.log("save_extension")
function extend_grammar(newg) {
newg.abstract.funs.push(fun);
var cs=newg.concretes
for(var ci in cs) {
var lc=cs[ci].langcode
if(lins[lc]) {
var lin={fun: fun.name, args: arg_names(fun.type),
lin: lins[lc]}
cs[ci].lins.push(lin)
}
}
}
function save_if_ok(res) {
if(res.errorcode=="OK") {
extend_grammar(g);
save_grammar(g);
files.removeChild(overlay)
minibar_div.style.minHeight="0px";
if(update_minibar) update_minibar();
//goto_minibar();
}
}
// Collect input from the form
fun.name=form["Abstract"].value;
for(var i in cs) {
var lc=cs[i].langcode;
lins[lc]=form[lc].value;
}
// This is not functional programming, so copy the grammar first...
var newg=JSON.parse(JSON.stringify(g));
extend_grammar(newg)
compile_grammar(newg,err_ind,save_if_ok);
return false; // prevent regular form submission
}
function cancel_extension() {
files.removeChild(overlay)
minibar_div.style.minHeight="0px";
//goto_minibar();
}
var fun_name=draw_extension();
files.appendChild(overlay)
fun_name.focus();
// Hack to prevent the overlay from overflowing the minibar div:
minibar_div.style.minHeight=overlay.clientHeight-11+"px"
//overlay.onresize= ... // doesn't work :-(
}
function show_editor() {
clear(minibar_div);
edit_grammar(g);
}
function goto_minibar() {
clear(files);
files.appendChild(minibar_div);
var online_options={grammars_url: local.get("dir")+"/",
grammar_list: [g.basename+".pgf"]}
var pgf_server=pgf_online(online_options)
var minibar_options= {
show_abstract: true,
show_trees: true,
show_grouped_translations: false,
show_brackets: true,
word_replacements: true,
extend_grammar: extend_grammar,
default_source_language: "Eng",
try_google: true
}
var minibar=new Minibar(pgf_server,minibar_options);
b.style.display="none";
comp_btn.disabled=true;
if(b2) b2.style.display="";
else {
b2=button("Show editor",show_editor);
insertAfter(b2,b);
}
}
function goto_minibar_if_ok(res) { if(res.errorcode=="OK") goto_minibar(); }
function compile() { compile_grammar(g,err_ind,goto_minibar_if_ok) }
var b=button("Minibar",compile);
b.title="Upload the grammar and test it in the minibar";
return b;
}
function quiz_button(g,err_ind) {
function goto_quiz(res) {
if(res.errorcode=="OK")
location.href="../TransQuiz/translation_quiz.html?"+local.get("dir")+"/"
}
function compile() { compile_grammar(g,err_ind,goto_quiz) }
var b=button("Quiz",compile);
b.title="Upload the grammar and go to the translation quiz";
return b;
}
function lang(code,name) { return { code:code, name:name} }
function lang1(name) {
var ws=name.split("/");
return ws.length==1 ? lang(name.substr(0,3),name) : lang(ws[0],ws[1]);
}
var languages =
map(lang1,"Amharic Arabic Bulgarian Catalan Danish Dutch English Finnish French German Hindi Ina/Interlingua Italian Jpn/Japanese Latin Norwegian Polish Ron/Romanian Russian Spanish Swedish Thai Turkish Urdu".split(" "));
languages.push(lang("Other","Other"));
var langname={};
for(var i in languages)
langname[languages[i].code]=languages[i].name
function concname(code) { return langname[code] || code; }
function add_concrete(g,el) {
var file=element("file");
clear(file);
var dc={};
for(var i in g.concretes)
dc[g.concretes[i].langcode]=true;
var list=[]
for(var i in languages) {
var l=languages[i], c=l.code;
if(!dc[c])
list.push(li([a(jsurl("add_concrete2("+g.index+",'"+c+"')"),
[text(l.name)])]));
}
var from= g.current>0
? "a copy of "+concname(g.concretes[g.current-1].langcode)
:"scratch";
file.appendChild(p(text("You are about to create a new concrete syntax by starting from "+from+".")));
file.appendChild(p(text("Pick a language for the new concrete syntax:")));
file.appendChild(node("ul",{"class":"languages"},list));
}
function new_concrete(code) {
return { langcode:code,params:[],lincats:[],opers:[],lins:[] };
}
function adjust_opens(cnc,oldcode,code) {
for(var oi in cnc.opens)
for(var li in rgl_modules)
if(cnc.opens[oi]==rgl_modules[li]+oldcode)
cnc.opens[oi]=rgl_modules[li]+code;
}
function add_concrete2(ix,code) {
var g=get_grammar(ix);
var cs=g.concretes;
var ci;
for(var ci=0;ci<cs.length;ci++) if(cs[ci].langcode==code) break;
if(ci==cs.length) {
if(g.current>0) {
cs.push(cs[g.current-1]); // old and new are shared at this point
save_grammar(g); // serialization loses sharing
g=reget_grammar(ix); // old and new are separate now
var oldcode=cs[g.current-1].langcode;
var cnc=g.concretes[ci];
cnc.langcode=code;
adjust_opens(cnc,oldcode,code);
timestamp(cnc)
}
else
cs.push(new_concrete(code))
save_grammar(g);
}
open_concrete(g,ci);
}
function open_abstract(g) { g.current=0; reload_grammar(g); }
function open_concrete(g,i) { g.current=i+1; reload_grammar(g); }
function td_gap(c) {return wrap_class("td","gap",c); }
function gap() { return td_gap(text(" ")); }
function tab(active,link) {
return wrap_class("td",active ? "active" : "inactive",link);
}
function delete_concrete(g,ci) {
var c=g.concretes[ci];
var ok=c.params.length==0 && c.lincats.length==0 && c.opers.length==0
&& c.lins.length==0
|| confirm("Do you really want to delete the concrete syntax for "+
concname(c.langcode)+"?");
if(ok) {
g.concretes=delete_ix(g.concretes,ci)
if(g.current && g.current-1>=ci) g.current--;
reload_grammar(g);
}
}
function draw_filebar(g) {
var cur=(g.current||0)-1;
var filebar = empty_class("tr","extensible")
filebar.appendChild(gap());
filebar.appendChild(
tab(cur== -1,button("Abstract",function(){open_abstract(g);})));
var cs=g.concretes;
function del(ci) { return function() { delete_concrete(g,ci); }}
function open_conc(i) { return function() {open_concrete(g,1*i); }}
for(var i in cs) {
filebar.appendChild(gap());
filebar.appendChild(
tab(i==cur,deletable(del(i),button(concname(cs[i].langcode),open_conc(i)),"Delete this concrete syntax")));
}
filebar.appendChild(td_gap(more(g,add_concrete,"Add a concrete syntax")));
return wrap_class("table","tabs",filebar);
}
function draw_file(g) {
return g.current>0 // && g.current<=g.concretes.length
? draw_concrete(g,g.current-1)
: draw_abstract(g);
}
function draw_startcat(g) {
var abs=g.abstract;
var startcat = abs.startcat || abs.cats[0];
function opt(cat) { return option(cat,cat); }
var opts = g.extends && g.extends.length>0 ? [opt("-")] : [];
var dc=defined_cats(g);
for(var cat in dc) if(dc[cat]!="Predef") opts.push(opt(cat));
var m = node("select",{},opts);
m.value=startcat;
m.onchange=function() {
if(m.value!=abs.startcat) {
abs.startcat=m.value;
timestamp(abs);
save_grammar(g);
}
}
return indent([kw("flags startcat","This is the default start category for parsing, random generation, etc. [C.3.16, C.5.1]"),
sep(" = "),m]);
}
var extends_hint="This grammar is an extension of the grammars listed here. [C.2.1]"
function draw_conc_extends(g,conc) {
var kw_extends=kw("extends ",extends_hint)
var exts=(g.extends || []).map(conc_extends(conc))
return exts.length>0
? indent([kw_extends,ident(exts.join(", "))])
: text("")
}
function draw_extends(g) {
var kw_extends=kw("extends ",extends_hint)
var exts= g.extends || [];
var m1=more(g,add_extends,"Inherit from other grammars");
var m2=more(g,add_extends,"Inherit from more grammars");
var es=[exts.length>0 ? kw_extends : span_class("more",kw_extends)];
function del(i) { return function() { delete_extends(g,i); }}
for(var i=0;i<exts.length;i++) {
if(i>0) es.push(sep(", "))
es.push(deletable(del(i),ident(exts[i]),"Don't inherit from "+exts[i]));
}
es.push(exts.length>0 ? m2 : m1);
return indent([extensible(es)])
}
function delete_extends(g,ix) {
g.extends=delete_ix(g.extends,ix);
//timestamp(g);
reload_grammar(g);
}
function add_extends(g,el) {
var file=element("file");
clear(file)
var gs=cached_grammar_array_byname();
var list=[]
for(var i in gs) {
if(gs[i].basename!=g.basename
&& !elem(gs[i].basename,g.extends || [])
&& !elem(g.basename,gs[i].extends || [])
// also exclude indirectly inherited grammars!!
)
list.push(li([a(jsurl("add_extends2("+g.index+","+gs[i].index+")"),
[text(gs[i].basename)])]));
}
file.appendChild(p(text("Pick a grammar to inherit:")));
file.appendChild(node("ul",{"class":"grammars"},list));
}
function add_extends2(gix,igix) {
var g=get_grammar(gix);
var ig=get_grammar(igix);
if(!g.extends) g.extends=[];
g.extends.push(ig.basename);
//timestamp(g)
reload_grammar(g);
}
function draw_abstract(g) {
var kw_abstract = kw("abstract ","A GF grammar must have one abstract syntax module. [C.2.1]")
var kw_cat = kw("cat","The categories (nonterminals) of the grammar are enumerated here. [C.3.2]");
var kw_fun = kw("fun","The functions (productions) of the grammar are enumerated here. [C.3.4]");
var flags=g.abstract.startcat || g.abstract.cats.length>1
|| g.extends && g.extends.length>0
? draw_startcat(g)
: text("");
function sort_funs() {
g.abstract.funs=sort_list(this,g.abstract.funs,"name");
timestamp(g.abstract);
save_grammar(g);
}
var file=div_id("file",
[kw_abstract,ident(g.basename),sep(" = "),
/*draw_timestamp(g.abstract),*/
draw_extends(g),
flags,
indent([extensible([kw_cat,
indent(draw_cats(g))]),
extensible([kw_fun,
indent_sortable(draw_funs(g),sort_funs)])])]);
if(navigator.onLine) {
var mode_button=text_mode(g,file,0);
insertBefore(mode_button,file.firstChild)
}
return file;
}
function module_name(g,ix) {
return ix==0 ? g.basename : g.basename+g.concretes[ix-1].langcode
}
function show_module(g,ix) {
return ix==0 ? show_abstract(g) : show_concrete(g)(g.concretes[ix-1]);
}
function text_mode(g,file,ix) {
var path=module_name(g,ix)+".gf"
function switch_to_guided_mode() {
clear(compiler_output);
edit_grammar(g); // !!
}
function store_parsed(parse_results) {
var dst=compiler_output;
var msg=parse_results[path];
clear(dst)
//console.log(msg)
if(dst && msg.error)
dst.appendChild(span_class("error_message",
text(msg.location+": "+msg.error)))
else if(dst && msg.parsed)
dst.innerHTML=
"Accepted by GF, but not by this editor ("+msg.parsed+")"
else if(msg.converted) {
if(ix==0) {
var gnew=msg.converted;
g.abstract=gnew.abstract;
g.extends=gnew.extends;
timestamp(g.abstract);
save_grammar(g);
}
else {
var conc=g.concretes[ix-1];
var cnew=msg.converted.concretes[0];
conc.opens=cnew.opens;
conc.params=cnew.params;
conc.lincats=cnew.lincats;
conc.opers=cnew.opers;
conc.lins=cnew.lins;
timestamp(conc);
save_grammar(g);
}
}
else replaceInnerHTML(dst,"unexpected parse result");
}
var last_source=show_abstract(g);
function parse(source) {
if(source!=last_source) {
if(navigator.onLine) {
//clear(compiler_output) // makes the page "jumpy"
last_source=source;
check_module(path,source,store_parsed)
}
else replaceInnerHTML(compiler_output,
"Offline, edits will not be saved")
}
}
function switch_to_text_mode() {
var ta=node("textarea",{class:"text_mode",rows:25,cols:80},
[text(show_module(g,ix))])
var timeout;
ta.onkeyup=function() {
if(timeout) clearTimeout(timeout);
timeout=setTimeout(function(){parse(ta.value)},400)
}
var mode_button=div_class("right",[button("Guided mode",switch_to_guided_mode)])
clear(file)
appendChildren(file,[mode_button,ta])
//ta.style.height=ta.scrollHeight+"px";
ta.focus();
}
var mode_button=div_class("right",[button("Text mode",switch_to_text_mode)])
return mode_button;
}
function add_cat(g,el) {
function add(s) {
var cats=s.split(/\s*(?:\s|[;])\s*/); // allow separating spaces or ";"
if(cats.length>0 && cats[cats.length-1]=="") cats.pop();
for(var i in cats) {
var err=check_name(cats[i],"Category");
if(err) return err;
}
for(var i in cats) g.abstract.cats.push(cats[i]);
timestamp(g.abstract);
reload_grammar(g);
return null;
}
string_editor(el,"",add);
}
function delete_cat(g,ix) {
with(g.abstract) cats=delete_ix(cats,ix);
timestamp(g.abstract);
reload_grammar(g);
}
function rename_cat(g,el,cat) {
function ren(newcat) {
if(newcat!="" && newcat!=cat) {
var err=check_name(newcat,"Category");
if(err) return err;
var dc=defined_cats(g);
if(dc[newcat]) return newcat+" is already in use";
g=rename_category(g,cat,newcat);
timestamp(g.abstract);
reload_grammar(g);
}
return null;
}
string_editor(el,cat,ren);
}
function duplicated(g,kind,orig) {
return orig==g.basename
? "Same "+kind+" defined twice in this module"
: "Same "+kind+" already defined in "+orig
}
function draw_cats(g) {
var cs=g.abstract.cats;
var es=[];
var defined=inherited_cats(g);
function eident(cat) {
function ren(g,el) { rename_cat(g,el,cat); }
return editable("span",ident(cat),g,ren,"Rename category");
}
function check(cat,el) {
return ifError(defined[cat],duplicated(g,"category",defined[cat]),el);
}
function del(i) { return function() { delete_cat(g,i); }}
for(var i in cs) {
es.push(deletable(del(i),check(cs[i],eident(cs[i])),"Delete this category"));
defined[cs[i]]=g.basename;
es.push(sep("; "));
}
es.push(more(g,add_cat,"Add more categories"));
return es;
}
function add_fun(g,el) {
function add(s) {
var p=parse_fun(s);
if(p.ok) {
g.abstract.funs.push(p.ok);
timestamp(g.abstract);
reload_grammar(g);
return null;
}
else
return p.error
}
string_editor(el,"",add);
}
function edit_fun(i) {
return function (g,el) {
function replace(s) {
var p=parse_fun(s);
if(p.ok) {
var old=g.abstract.funs[i];
g.abstract.funs[i]=p.ok;
if(p.ok.name!=old.name) g=rename_function(g,old.name,p.ok.name);
if(show_type(p.ok.type)!=show_type(old.type))
g=change_lin_lhs(g,p.ok);
timestamp(g.abstract);
reload_grammar(g);
return null;
}
else
return p.error;
}
string_editor(el,show_fun(g.abstract.funs[i]),replace);
}
}
function delete_fun(g,ix) {
with(g.abstract) funs=delete_ix(funs,ix);
timestamp(g.abstract);
reload_grammar(g);
}
function draw_funs(g) {
var funs=g.abstract.funs;
var es=[];
var dc=defined_cats(g);
var df=inherited_funs(g);
function del(i) { return function() { delete_fun(g,i); }}
function draw_efun(i,df) {
return editable("span",draw_fun(g,funs[i],dc,df),g,edit_fun(i),"Edit this function");
}
for(var i in funs) {
es.push(node_sortable("fun",funs[i].name,[deletable(del(i),draw_efun(i,df),"Delete this function")]));
df[funs[i].name]=g.basename;
}
es.push(more(g,add_fun,"Add a new function"));
return es;
}
function draw_fun(g,fun,dc,df) {
function check(el) {
return ifError(dc[fun.name],
"Function names must be distinct from category names",
ifError(df[fun.name],duplicated(g,"function",df[fun.name]),el));
}
return node("span",{},
[check(ident(fun.name)),sep(" : "),draw_type(fun.type,dc)]);
}
function draw_type(t,dc) {
var el=empty("span");
function check(t,el) {
if(dc[t]) el.title=dc[t]+"."+t;
return ifError(!dc[t],"Undefined category",el);
}
for(var i in t) {
if(i>0) el.appendChild(sep(" → "));
el.appendChild(check(t[i],ident(t[i])));
}
return el;
}
function edit_name(g,el) {
function change_name(name) {
if(name!=g.basename && name!="") {
var err=check_name(name,"Grammar");
if(err) return err;
g.basename=name
reload_grammar(g);
}
return null;
}
string_editor(el,g.basename,change_name)
}
/* -------------------------------------------------------------------------- */
function draw_concrete(g,i) {
var conc=g.concretes[i];
function edit_langcode(g,el) {
function change_langcode(code) {
var err=check_name(g.basename+code,"Name of concrete syntax");
if(err) return err;
adjust_opens(conc,conc.langcode,code);
conc.langcode=code;
timestamp(conc);
reload_grammar(g);
}
string_editor(el,conc.langcode,change_langcode)
}
var kw_concrete=kw("concrete ", "A GF grammar can have any number of concrete syntax modules matching the abstract syntax [C.2.1]")
var kw_open=kw("open ","Opening makes constants from other modules available in this module. [C.2.8]")
var kw_lincat=kw("lincat","The linearization type for each catagory in the abstract syntax is given here. [C.3.8]")
var kw_lin=kw("lin","The linearization function for each function in the abstract syntax is given here. [C.3.9]")
var kw_param=kw("param","Parameter type definitions can be added here. [C.3.12]")
var kw_oper=kw("oper","Operation definitions can be added here. [C.3.14]")
var file=div_id("file",
[kw_concrete,
ident(g.basename),
editable("span",ident(conc.langcode),g,
edit_langcode,"Change language"),
kw(" of "),ident(g.basename),sep(" = "),
/*draw_timestamp(conc),*/
draw_conc_extends(g,conc),
indent([extensible([kw_open,draw_opens(g,i)])]),
indent([kw_lincat,draw_lincats(g,i)]),
indent([kw_lin,draw_lins(g,i)]),
indent([extensible([kw_param,draw_params(g,i)])]),
indent([extensible([kw_oper,draw_opers(g,i)])])/*,
exb_extra(g,i)*/
])
if(navigator.onLine) {
var mode_button=text_mode(g,file,i+1);
insertBefore(mode_button,file.firstChild)
}
return file;
}
var rgl_modules=["Syntax","Lexicon","Paradigms","Extra","Symbolic"];
var rgl_info = {
Paradigms: "Lexical categories (A, N, V, ...) and smart paradigms (mkA, mkN, mkV, ...) for turning raw strings into new dictionary entries.",
Syntax: "Syntactic categories (Utt, Cl, V, NP, CN, AP, ...), structural words (this_Det, few_Det, ...) and functions for building phrases (mkUtt, mkCl, mkCN, mkVP, mkAP, ...)",
Lexicon: "A multilingual lexicon with ~350 common words.",
Extra: "Language-specific extra constructions not available via the common API.",
Symbolic: "Functions for symbolic expressions (numbers and variables in mathematics)"
}
function add_open(ci) {
return function (g,el) {
var conc=g.concretes[ci];
var os=conc.opens;
var ds={};
for(var i in os) ds[os[i]]=true;
var list=[]
for(var i in rgl_modules) {
var b=rgl_modules[i], m=b+conc.langcode;
if(!ds[m]) {
var info=rgl_info[b];
var infotext=info ? text(" - "+info) : text("");
list.push(li([a(jsurl("add_open2("+g.index+","+ci+",'"+m+"')"),
[text(m)]),infotext]));
}
}
if(list.length>0) {
var file=element("file");
clear(file)
file.appendChild(p(text("Pick a resource library module to open:")));
file.appendChild(node("ul",{},list));
}
}
}
function add_open2(ix,ci,m) {
var g=get_grammar(ix);
var conc=g.concretes[ci];
conc.opens || (conc.opens=[]);
conc.opens.push(m);
timestamp(conc);
save_grammar(g);
open_concrete(g,ci);
}
function delete_open(g,ci,ix) {
with(g.concretes[ci]) opens=delete_ix(opens,ix);
timestamp(g.concretes[ci]);
reload_grammar(g);
}
function draw_opens(g,ci) {
var conc=g.concretes[ci];
var os=conc.opens || [] ;
var es=[];
function del(i) { return function() { delete_open(g,ci,i); }}
function show_opers(m) {
return function(event) {
var link=event.target,dst=compiler_output
function cont2(opers) {
clear(dst);
var sheet=div_class("sheet",[wrap("h3",text(m)),
wrap("pre",text(opers))])
var dy=dst.offsetTop-link.offsetTop-link.offsetHeight
sheet.style.top="-"+dy+"px";
insertFirst(dst,sheet)
setTimeout(function(){sheet.style.top="0px";},1000)
}
function cont1() { gfshell("so",cont2) }
gfshell("i -retain present/"+m+".gfo",cont1)
}
}
var first=true;
for(var i in os) {
if(!first) es.push(sep(", "))
var m=os[i];
var b=m.substr(0,m.length-conc.langcode.length);
var info=rgl_info[b];
var id=ident(m);
if(info) id.title=info;
id.onclick=show_opers(m)
id.className="ident onclick"
es.push(deletable(del(i),id,"Don't open this module"));
first=false;
}
es.push(more(g,add_open(ci),"Open more modules"));
return indent(es);
}
function draw_param(p,dp) {
function check(el) {
return ifError(dp[p.name],"Same parameter type defined twice",el);
}
return node("span",{},[check(ident(p.name)),sep(" = "),text(p.rhs)]);
}
function add_param(g,ci,el) {
function add(s) {
var p=parse_param(s);
if(p.ok) {
g.concretes[ci].params.push(p.ok);
timestamp(g.concretes[ci]);
reload_grammar(g);
return null;
}
else
return p.error
}
string_editor(el,"",add);
}
function edit_param(ci,i) {
return function (g,el) {
function replace(s) {
var p=parse_param(s);
if(p.ok) {
g.concretes[ci].params[i]=p.ok;
timestamp(g.concretes[ci]);
reload_grammar(g);
return null;
}
else
return p.error;
}
string_editor(el,show_param(g.concretes[ci].params[i]),replace);
}
}
function delete_param(g,ci,ix) {
with(g.concretes[ci]) params=delete_ix(params,ix);
timestamp(g.concretes[ci]);
reload_grammar(g);
}
function draw_params(g,ci) {
var conc=g.concretes[ci];
conc.params || (conc.params=[]);
var params=conc.params;
var es=[];
var dp={};
function del(i) { return function() { delete_param(g,ci,i); }}
function draw_eparam(i,dp) {
return editable("span",draw_param(params[i],dp),g,edit_param(ci,i),"Edit this parameter type");
}
for(var i in params) {
es.push(div_class("param",[deletable(del(i),draw_eparam(i,dp),"Delete this parameter type")]));
dp[params[i].name]=true;
}
es.push(more(g,function(g,el) { return add_param(g,ci,el)},
"Add a new parameter type"));
return indent(es);
}
function delete_lincat(g,ci,cat) {
var i;
var c=g.concretes[ci];
for(i=0;i<c.lincats.length && c.lincats[i].cat!=cat;i++);
if(i<c.lincats.length) c.lincats=delete_ix(c.lincats,i);
timestamp(c);
reload_grammar(g);
}
function draw_lincats(g,i) {
var conc=g.concretes[i];
function edit(c) {
return function(g,el) {
function check(s,cont) {
function check2(msg) {
if(!msg) {
if(c.template) conc.lincats.push({cat:c.cat,type:s});
else c.type=s;
reload_grammar(g);
}
cont(msg);
}
check_exp(s,check2);
}
string_editor(el,c.type,check,true)
}
}
function del(c) { return function() { delete_lincat(g,i,c); } }
function dlc(c,cls) {
var t=editable("span",text_ne(c.type),g,edit(c),"Edit lincat for "+c.cat);
return node("span",{"class":cls},
[ident(c.cat),sep(" = "),t]);
}
var dc=locally_defined_cats(g,{});
function draw_lincat(c) {
var cat=c.cat;
var err=!dc[cat];
var l1=dlc(c,"lincat");
var l2= deletable(del(cat),l1,"Delete this lincat");
var l=ifError(err,"lincat for undefined category",l2);
delete dc[cat];
return node_sortable("lincat",cat,[l]);
}
function dtmpl(c) {
return wrap("div",dlc({cat:c,type:"",template:true},"template")); }
var lcs=map(draw_lincat,conc.lincats);
for(var c in dc)
lcs.push(dtmpl(c));
function sort_lincats() {
conc.lincats=sort_list(this,conc.lincats,"cat");
timestamp(conc);
save_grammar(g);
}
return indent_sortable(lcs,sort_lincats);
}
/* -------------------------------------------------------------------------- */
function draw_oper(p,dp) {
function check(el) {
return ifError(dp[p.name],"Same operator definition defined twice",el);
}
return node("span",{},[check(ident(p.name)),text(" "),text(p.rhs)]);
}
function check_oper(s,ok,err) {
var p=parse_oper(s);
function check2(msg) {
if(msg) err(msg);
else ok(p.ok)
}
if(p.ok) {
// Checking oper syntax by checking an expression with a local
// definition. Some valid opers will be rejected!!
var e=p.ok.name+" where { "+p.ok.name+" "+p.ok.rhs+" }";
check_exp(e,check2);
}
else
err(p.error);
}
function add_oper(g,ci,el) {
function check(s,cont) {
function ok(oper) {
g.concretes[ci].opers.push(oper);
timestamp(g.concretes[ci]);
reload_grammar(g);
cont(null);
}
check_oper(s,ok,cont)
}
string_editor(el,"",check,true);
}
function edit_oper(ci,i) {
return function (g,el) {
function check(s,cont) {
function ok(oper) {
g.concretes[ci].opers[i]=oper;
timestamp(g.concretes[ci]);
reload_grammar(g);
cont(null);
}
check_oper(s,ok,cont)
}
string_editor(el,show_oper(g.concretes[ci].opers[i]),check,true);
}
}
function delete_oper(g,ci,ix) {
with(g.concretes[ci]) opers=delete_ix(opers,ix);
timestamp(g.concretes[ci]);
reload_grammar(g);
}
function draw_opers(g,ci) {
var conc=g.concretes[ci];
conc.opers || (conc.opers=[]);
var opers=conc.opers;
var es=[];
var dp={};
function del(i) { return function() { delete_oper(g,ci,i); }}
function draw_eoper(i,dp) {
return editable("span",draw_oper(opers[i],dp),g,edit_oper(ci,i),"Edit this operator definition");
}
for(var i in opers) {
es.push(node_sortable("oper",opers[i].name,
[deletable(del(i),draw_eoper(i,dp),
"Delete this operator definition")]));
dp[opers[i].name]=true;
}
es.push(more(g,function(g,el) { return add_oper(g,ci,el)},
"Add a new operator definition"));
function sort_opers() {
conc.opers=sort_list(this,conc.opers,"name");
timestamp(conc);
save_grammar(g);
}
return indent_sortable(es,sort_opers);
}
function delete_lin(g,ci,fun) {
var i;
var c=g.concretes[ci];
for(i=0;i<c.lins.length && c.lins[i].fun!=fun;i++);
if(i<c.lins.length) c.lins=delete_ix(c.lins,i);
timestamp(c);
reload_grammar(g);
}
/* -------------------------------------------------------------------------- */
function arg_names(type) {
function lower(s) { return s.toLowerCase(); }
var names=map(lower,type);
names.pop(); // remove result type
var n,count={},use={};
for(var i in names) n=names[i],count[n]=0,use[n]=0;
for(var i in names) count[names[i]]++;
function unique(n) {
return count[n]>1 ? n+(++use[n]) : n;
}
return map(unique,names);
}
function draw_lins(g,ci) {
var conc=g.concretes[ci];
var igs=inherited_grammars(g)
var dc=defined_cats(g);
function edit(f) {
return function(g,el) {
function check(s,cont) {
function check2(msg) {
if(!msg) {
if(f.template)
conc.lins.push({fun:f.fun,args:f.args,lin:s});
else { f.lin=s; f.eb_lin=null; }
reload_grammar(g);
}
cont(msg);
}
check_exp(s,check2);
}
string_editor(el,f.lin,check,true)
}
}
function del(fun) { return function () { delete_lin(g,ci,fun); } }
function dl(f,cls) {
var fn=ident(f.fun)
var fty=function_type(g,f.fun)
var linty=lintype(g,conc,igs,dc,fty)
if(fty)
fn.title="fun "+f.fun+": "+show_type(fty)
+"\nlin "+f.fun+": "+show_lintype(linty)
var l=[fn]
for(var i=0; i<f.args.length;i++) {
l.push(text(" "));
var an=ident(f.args[i])
an.title=f.args[i]+": "+linty[i]
l.push(an);
}
l.push(sep(" = "));
var t=editable("span",text_ne(f.lin),g,edit(f),"Edit lin for "+f.fun);
t.appendChild(exb_linbuttons(g,ci,f));
l.push(t);
return node("span",{"class":cls},l);
}
var df=locally_defined_funs(g,{});
function draw_lin(f) {
var fun=f.fun;
var err= !df[fun];
var l= deletable(del(fun),dl(f,"lin"),"Delete this linearization function")
var l=ifError(err,"Function "+fun+" is not part of the abstract syntax",l);
delete df[fun];
return node_sortable("lin",fun,[l]);
}
function largs(f) {
var funs=g.abstract.funs;
for(var i=0;i<funs.length && funs[i].name!=f;i++);
return arg_names(funs[i].type);
}
function dtmpl(f) {
return div_class("template",
[dl({fun:f,args:largs(f),lin:"",template:true},"template")]);
}
function sort_lins() {
conc.lins=sort_list(this,conc.lins,"fun");
timestamp(conc);
save_grammar(g);
}
var ls=map(draw_lin,conc.lins);
for(var f in df)
ls.push(dtmpl(f));
return indent_sortable(ls,sort_lins);
}
// Return the linearization type for the given abtract type in the given
// concrete syntax, taking into account that lincats may be defined in
// inherited grammars.
// lintype :: Grammar -> Concrete -> [Grammar] -> {Cat=>ModId} => Type -> [Term]
function lintype(g,conc,igs,dc,type) {
console.log(dc)
function ihcat_lincat(cat) {
if(dc[cat]=="Predef") return "{s:Str}" // !!! Is this right?
var ig=find_grammar_byname(igs,dc[cat])
if(!ig) return "??"
var iconc=ig.concretes[conc_index(ig,conc.langcode)]
if(!iconc) return "??"
return cat_lincat(iconc,cat) || "??"
}
function lincat(cat) {
return dc[cat]
? (dc[cat]==g.basename ? cat_lincat(conc,cat) : ihcat_lincat(cat))
|| "??"
: cat+"??"
}
return type.map(lincat)
}
function find_grammar_byname(igs,name) {
for(var i in igs) if(igs[i].basename==name) return igs[i]
return null
}
/* -------------------------------------------------------------------------- */
function draw_matrix(g) {
var row=[th(text("Abstract"))]
var t=empty_class("table","matrixview")
for(var ci in g.concretes)
row.push(th(text(concname(g.concretes[ci].langcode))))
t.appendChild(tr(row))
var dc=defined_cats(g);
var df=inherited_funs(g);
for(var i in g.abstract.cats) {
var cat=g.abstract.cats[i]
var row=[td(ident(cat))]
for(var ci in g.concretes) {
var conc=g.concretes[ci]
row.push(td(text(cat_lincat(conc,cat))))
}
t.appendChild(tr(row))
}
for(var i in g.abstract.funs) {
var fun=g.abstract.funs[i]
var row=[td(draw_fun(g,fun,dc,df))]
for(var ci in g.concretes) {
var conc=g.concretes[ci]
var lin=fun_lin(conc,fun.name)
var dl=lin ? simple_draw_lin(lin) : text(" ")
row.push(td(dl))
}
t.appendChild(tr(row))
}
return t
}
function simple_draw_lin(f) {
var l=[]
if(f.args.length>0) {
l.push(sep("\\")) /* λ */
for(var i in f.args) {
l.push(text(" "));
l.push(ident(f.args[i]));
}
l.push(sep(" → "));
}
l.push(text_ne(f.lin));
return wrap("span",l);
}
/* -------------------------------------------------------------------------- */
function defined_cats(g) { return all_defined_cats(g,inherited_grammars(g)) }
function defined_funs(g) { return all_defined_funs(g,inherited_gramamrs(g)) }
function inherited_cats(g) {
return all_inherited_cats(inherited_grammars(g),predefined_cats())
}
function inherited_funs(g) {return all_inherited_funs(inherited_grammars(g),{})}
function upload(g,cont) {
var gs=inherited_grammars(g);
gs.push(g);
upload_grammars(gs,cont);
}
// inherited_grammars :: Grammar -> [Grammar]
function inherited_grammars(g) {
// Load the available grammars once
var grammar_byname=cached_grammar_byname();
var visited={};
function exists(g) { return g; }
// Then traverse the dependencies to collect all inherited grammars
function ihgs(g) {
if(visited[g.basename]) return []; // avoid cycles and diamonds
else {
visited[g.basename]=true;
var igs=(g.extends || []).map(grammar_byname).filter(exists);
var igss=igs.map(ihgs)
for(var i in igss) igs=igs.concat(igss[i]);
return igs;
}
}
return ihgs(g)
}
// cached_grammar__byname :: () -> (ModId->Grammar)
function cached_grammar_byname() {
var gix=cached_grammar_array_byname()
function grammar_byname(name) { return gix[name]; }
return grammar_byname;
}
// cached_grammar_array_byname :: () -> {ModId=>Grammar}
function cached_grammar_array_byname() {
var gix={};
for(var i=0;i<local.count;i++) {
var g=get_grammar(i,null);
if(g) gix[g.basename]=g; // basenames are not necessarily unique!!
}
return gix
}
/* -------------------------------------------------------------------------- */
function find_langcode(concs,langcode) {
for(var ci in concs)
if(concs[ci].langcode==langcode)
return concs[ci];
return null;
}
function cleanup_deleted(files) {
var keep={}
for(var i in files) keep[files[i]]=true;
//debug("cleanup_deleted "+JSON.stringify(files))
//debug("keep "+JSON.stringify(keep))
for(var i=0;i<local.count;i++) {
var g=local.get(i,null)
if(g && g.unique_name && !keep[g.unique_name+".json"]) {
debug("cleanup "+i+" "+g.unique_name);
remove_local_grammar(i)
}
}
}
function grammar_index() {
var index={}
var count=local.count
for(var i=0;i<count;i++) {
var g=local.get(i,null)
if(g && g.unique_name) index[g.unique_name]=i
}
return index
}
function merge_grammar(i,newg) {
var oldg=get_grammar(i);
var keep="";
debug("Merging at "+i);
if(oldg) {
oldg.basename=newg.basename;
if(newg.abstract.timestamp<oldg.abstract.timestamp) {
newg.abstract=newg.abstract
keep+=" "+oldg.basename
}
for(var ci in newg.concretes) {
var conc=newg.concretes[ci];
var oldconc=find_langcode(oldg.concretes,conc.langcode);
if(oldconc && conc.timestamp<oldconc.timestamp) {
newg.concretes[ci]=oldconc;
keep+=" "+oldg.basename+conc.langcode;
}
}
}
local.put(i,newg)
return keep;
}
function timestamp(obj,prop) {
obj[prop || "timestamp"]=Date.now();
}
function draw_timestamp(obj) {
var t=obj.timestamp;
return node("small",{"class":"modtime"},
[text(t ? " -- "+new Date(t).toLocaleString() : "")]);
}
/* -------------------------------------------------------------------------- */
function delete_ix(old,ix) {
var a=[];
for(var i in old) if(i!=ix) a.push(old[i]);
return a;
}
function sort_list(list,olditems,key) {
var items=[];
function find(fun) {
for(var i=0;i<olditems.length;i++)
if(olditems[i][key]==fun) return olditems[i];
return null;
}
for(var el=list.firstChild;el;el=el.nextSibling) {
var name=el.getAttribute("ident")
if(name) {
var old=find(name);
if(old) items.push(old)
else debug("Bug: did not find "+name+" while sorting");
}
}
if(items.length==olditems.length)
return items;
else {
debug("Bug: length changed while sorting")
return olditems;
}
}
function string_editor(el,init,ok,async) {
var p=el.parentNode;
function restore() {
e.parentNode.removeChild(e);
el.style.display="";
}
function done() {
var edited=e.it.value;
restore();
function cont(msg) { if(msg) start(msg); }
if(async) ok(edited,cont)
else cont(ok(edited));
return false;
}
function start(msg) {
el.style.display="none";
m.innerHTML=msg;
insertAfter(e,el);
e.it.focus();
}
var m=empty_class("span","error_message");
var i=node("input",{"class":"string_edit",name:"it",value:init},[]);
if(init.length>10) i.size=init.length+5;
// var i=node("textarea",{name:"it",rows:"2",cols:"60"},[text(init)]);
var e=node("form",{},
[i,
node("input",{type:"submit",value:"OK"},[]),
button("Cancel",restore),
text(" "),
m])
e.onsubmit=done
start("");
}
function ifError(b,msg,el) { return b ? inError(msg,el) : el; }
function inError(msg,el) {
return node("span",{"class":"inError",title:msg},[el]);
}
function kw(txt,hint) {
var w=wrap_class("span","kw",text(txt));
if(hint) w.title=hint;
return w;
}
function sep(txt) { return wrap_class("span","sep",text(txt)); }
function ident(txt) { return wrap_class("span","ident",text(txt)); }
function unimportant(txt) { return wrap_class("small","unimportant",text(txt)); }
function indent(cs) { return div_class("indent",cs); }
function indent_sortable(cs,sort) {
var n= indent(cs);
n.onsort=sort;
return n;
}
function node_sortable(cls,name,ls) {
return node("div",{"class":cls,"ident":name},ls);
}
function extensible(cs) { return div_class("extensible",cs); }
function more(g,action,hint,label) {
var b=node("span",{"class":"more","title":hint || "Add more"},
[text(label || " + ")]);
b.onclick=function() { action(g,b); }
return b;
}
function text_ne(s) { // like text(s), but force it to be non-empty
return text(s ? s : "\xa0\xa0\xa0")
}
function editable(tag,cs,g,f,hint) {
var b=edit_button(function(){f(g,e)},hint);
var e=node(tag,{"class":"editable"},[cs,b]);
//e.onclick=b.onclick;
return e;
}
function edit_button(action,hint) {
var b=node("span",{"class":"edit","title":hint || "Edit"},[text("%")]);
b.onclick=action;
return b;
}
function deletable(del,el,hint) {
var b=delete_button(del,hint)
return node("span",{"class":"deletable"},[b,el])
}
function delete_button(action,hint) {
var b=node("span",{"class":"delete",title:hint || "Delete"},[text("×")])
b.onclick=action;
return b;
}
function touch_edit() {
var b=node("input",{type:"checkbox"},[]);
function touch() {
document.body.className=b.checked ? "nohover" : "hover";
}
b.onchange=touch;
insertAfter(b,editor);
insertAfter(wrap("small",text("Enable editing on touch devices. ")),b);
}
/* --- DOM Support ---------------------------------------------------------- */
function a(url,linked) { return node("a",{href:url},linked); }
function ul(lis) { return node("ul",{},lis); }
function li(xs) { return node("li",{},xs); }
function table(rows) { return node("table",{},rows); }
function td_right(cs) { return node("td",{"class":"right"},cs); }
function jsurl(js) { return "javascript:"+js; }
/* -------------------------------------------------------------------------- */
function download_from_cloud() {
var newdir="/tmp/"+location.hash.substr(1)
function download2(olddir) {
//debug("Starting grammar sharing in the cloud")
if(newdir!=olddir) link_directories(newdir,download3)
else download4()
}
function download3() {
//debug("Uploading local grammars to cloud");
local.put("dir",newdir);
upload_json(download4)
}
function download4() {
//debug("Downloading grammars from the cloud");
download_json()
}
with_dir(download2)
}
/* --- Initialization ------------------------------------------------------- */
//document.body.appendChild(empty_id("div","debug"));
function dir_bugfix() {
// remove trailing newline caused by bug in older version
var dir=local.get("dir");
if(dir) {
var n=dir.length;
while(dir[dir.length-1]=="\n" || dir[dir.length-1]=="\r")
dir=dir.substr(0,dir.length-1)
if(dir.length<n) {
debug("removing trailing newline")
local.put("dir",dir);
}
//debug("Server directory: "+JSON.stringify(dir))
}
else debug("No server directory yet")
}
if(editor) {
if(supports_html5_storage()) {
initial_view();
touch_edit();
dir_bugfix();
initialize_sorting(["DIV"],["fun","lin","lincat","oper"]);
}
else
editor.innerHTML="<p>This browser does not appear to support localStorage, and the grammar editor does not work without it. Sorry!"
}
//console.log("hi")