var editor=element("editor"); var compiler_output=element("compiler_output") /* -------------------------------------------------------------------------- */ 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/1307545089_weather_04.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;i0 && !local.get(local.count-1)) local.count--; } function delete_grammar(i) { var g=local.get(i); var ok=confirm("Do you really want to delete the grammar "+g.basename+"?") if(ok) { remove_local_grammar(i) remove_cloud_grammar(g) initial_view(); } } function clone_grammar(i) { var old=local.get(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=local.get(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) { return div_class("namebar", [table([tr([td(draw_name(g)), td_right([minibar_button(g,files), quiz_button(g), compile_button(g), draw_plainbutton(g,files), draw_closebutton(g)])])])]) } 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="
"+show_grammar(g)+"
" 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 show_compile_error(res) { var dst=compiler_output 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_button(g) { function compile() { replaceInnerHTML(compiler_output,"

Compiling...

"); upload(g,show_compile_error); } 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) { var b2; function show_editor() { edit_grammar(g); } function goto_minibar(res) { show_compile_error(res); if(res.errorcode=="OK") { //location.href=res.minibar_url; clear(files); files.appendChild(div_id("minibar")); 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, default_source_language: "Eng", try_google: true } var minibar=new Minibar(pgf_server,minibar_options); b.style.display="none"; if(b2) b2.style.display=""; else { b2=button("Show editor",show_editor); insertAfter(b2,b); } } } function compile() { replaceInnerHTML(compiler_output,"

Compiling...

"); upload(g,goto_minibar); } var b=button("Minibar",compile); b.title="Upload the grammar and test it in the minibar"; return b; } function quiz_button(g) { function goto_quiz(res) { show_compile_error(res); if(res.errorcode=="OK") location.href="../TransQuiz/translation_quiz.html?"+local.get("dir")+"/" } function compile() { replaceInnerHTML(compiler_output,"

Compiling...

"); upload(g,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 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=local.get(ix); var cs=g.concretes; var ci; for(var ci=0;ci0) { cs.push(cs[g.current-1]); // old and new are shared at this point save_grammar(g); // serialization loses sharing g=local.get(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"),sep(" = "),m]); } function draw_conc_extends(g,conc) { var kw_extends=kw("extends ") kw_extends.title="This grammar is an extension of the grammars listed here." 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 ") var exts= g.extends || []; kw_extends.title="This grammar is an extension of the grammars listed here." 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;i0) 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=local.get(gix); var ig=local.get(igix); if(!g.extends) g.extends=[]; g.extends.push(ig.basename); //timestamp(g) reload_grammar(g); } function draw_abstract(g) { var kw_cat = kw("cat"); kw_cat.title = "The categories (nonterminals) of the grammar are enumerated here. [C.3.2]"; var kw_fun = kw("fun"); kw_fun.title = "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_lincat=kw("lincat") kw_lincat.title="The linearization type for each catagory in the abstract syntax is given here. [C.3.8]" var kw_lin=kw("lin") kw_lin.title="The linearization function for each function in the abstract syntax is given here. [C.3.9]" var kw_param=kw("param") kw_param.title="Parameter type definitions can be added here. [C.3.12]" var kw_oper=kw("oper") kw_oper.title="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=["Paradigms","Syntax","Lexicon","Extra"]; 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]) list.push(li([a(jsurl("add_open2("+g.index+","+ci+",'"+m+"')"), [text(m)])])); } 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=local.get(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); }} var first=true; for(var i in os) { if(!first) es.push(sep(", ")) es.push(deletable(del(i),ident(os[i]),"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;i1 ? n+(++use[n]) : n; } return map(unique,names); } function draw_lins(g,ci) { var conc=g.concretes[ci]; 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 l=[ident(f.fun)] for(var i in f.args) { l.push(text(" ")); l.push(ident(f.args[i])); } 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 [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;i10) 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) { return wrap_class("span","kw",text(txt)); } 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 div_id(id,cs) { return node("div",{id:id},cs); } function div_class(cls,cs) { return node("div",{"class":cls},cs); } 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