mirror of
https://github.com/LCPQ/QUESTDB_website.git
synced 2025-01-12 22:18:29 +01:00
386 lines
11 KiB
JavaScript
386 lines
11 KiB
JavaScript
|
// source: https://github.com/kercl/LaTeX-to-Unicode/tree/master
|
|||
|
|
|||
|
texparser = {
|
|||
|
replace_chars: function(x, table) {
|
|||
|
var res = "";
|
|||
|
for(i in x)
|
|||
|
if(x[i] in table)
|
|||
|
res = res + table[x[i]];
|
|||
|
else
|
|||
|
res = res + x[i];
|
|||
|
return res;
|
|||
|
},
|
|||
|
|
|||
|
trim_tokens: function(tokens) {
|
|||
|
var beg = 0, end = tokens.length - 1;
|
|||
|
for(; beg < tokens.length; beg++)
|
|||
|
if(!this.whitespace(tokens[beg].object))
|
|||
|
break;
|
|||
|
for(; end >= 0; end--)
|
|||
|
if(!this.whitespace(tokens[end].object))
|
|||
|
break;
|
|||
|
return tokens.slice(beg, end);
|
|||
|
},
|
|||
|
|
|||
|
strings_disjoint: function(x, y) {
|
|||
|
if(x.length != y.length)
|
|||
|
return false;
|
|||
|
|
|||
|
for(var i = 0; i < x.length; i++)
|
|||
|
if(x[i] == y[i])
|
|||
|
return false;
|
|||
|
return true;
|
|||
|
},
|
|||
|
|
|||
|
optimize: function(tokens) {
|
|||
|
var single = "";
|
|||
|
var sliceat = -1;
|
|||
|
|
|||
|
for(var i = 0; i < tokens.length; i++) {
|
|||
|
if((tokens[i].object.length == 1 &&
|
|||
|
"_^{}$".indexOf(tokens[i].object) == -1) || this.whitespace(tokens[i].object))
|
|||
|
single = single + tokens[i].object;
|
|||
|
else {
|
|||
|
sliceat = i;
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
if(sliceat == -1)
|
|||
|
return [{object:single}];
|
|||
|
return [{object:single}].concat(tokens.slice(sliceat));
|
|||
|
},
|
|||
|
|
|||
|
tokenize: function(str) {
|
|||
|
var ret = [];
|
|||
|
|
|||
|
str = str.replace(/\u200B/g, "\\");
|
|||
|
|
|||
|
while(str != "") {
|
|||
|
var s = "";
|
|||
|
if("_^{}$".indexOf(str[0]) > -1) {
|
|||
|
s = str[0];
|
|||
|
}else if(str[0] == '\\') {
|
|||
|
s = str.match(/^\\([a-zA-Z]+|\$|\\|\{|\}| |\_|\^)/g);
|
|||
|
if(s != null)
|
|||
|
s = s[0];
|
|||
|
else
|
|||
|
s = str[0];
|
|||
|
}else {
|
|||
|
var s1 = str.match(/^[^_\\\$\^\{\}\s]/g),
|
|||
|
s2 = str.match(/^[^_\\\$\^\{\}\S]+/g);
|
|||
|
|
|||
|
if(s1 == null) {
|
|||
|
s = s2[0];
|
|||
|
}else if(s2 == null) { // not whitespace
|
|||
|
s = s1[0];
|
|||
|
}else {
|
|||
|
s = s[0];
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
ret.push({object:s});
|
|||
|
str = str.substring(Math.max(1,s.length));
|
|||
|
}
|
|||
|
var ret2 = this.optimize(ret);
|
|||
|
return ret2;
|
|||
|
},
|
|||
|
|
|||
|
whitespace: function(str) {
|
|||
|
return str.match(/^\s+/g) != null;
|
|||
|
},
|
|||
|
|
|||
|
tag: function(tok) {
|
|||
|
return (tok.object[0] == "\\" && tok.object.length > 1) || tok.object == "^" || tok.object == "_";
|
|||
|
},
|
|||
|
|
|||
|
extract_block: function(tokens, begin, start_token, end_token) {
|
|||
|
start_token = typeof start_token !== 'undefined' ? start_token : "{";
|
|||
|
end_token = typeof end_token !== 'undefined' ? end_token : "}";
|
|||
|
|
|||
|
if(tokens[begin] == undefined)
|
|||
|
return [];
|
|||
|
|
|||
|
if((this.tag(tokens[begin]) || tokens[begin].object == "\\") && tokens[begin].caret != undefined)
|
|||
|
return [];
|
|||
|
|
|||
|
if(tokens[begin].object != start_token)
|
|||
|
return [tokens[begin]];
|
|||
|
if(tokens[begin].closed != true)
|
|||
|
return [];
|
|||
|
|
|||
|
var bc = 1;
|
|||
|
var res = [tokens[begin]];
|
|||
|
|
|||
|
for(var i = begin+1; i < tokens.length; i++) {
|
|||
|
if(tokens[i].object == start_token)
|
|||
|
bc++;
|
|||
|
else if(tokens[i].object == end_token) {
|
|||
|
bc--;
|
|||
|
if(bc == 0) {
|
|||
|
res.push(tokens[i]);
|
|||
|
return res;
|
|||
|
}
|
|||
|
}
|
|||
|
res.push(tokens[i]);
|
|||
|
}
|
|||
|
|
|||
|
return [];
|
|||
|
},
|
|||
|
|
|||
|
finish: function(str) {
|
|||
|
return str;
|
|||
|
},
|
|||
|
|
|||
|
reformat_math: function(str) {
|
|||
|
var res = tag_table["\\textit"].value(str.replace(/\\ /g, "\u00A0"));
|
|||
|
res = res.replace(/ /g, "");
|
|||
|
return res.replace(/[><=≌≊≆≈⋍∽≅⋞⋟⪖⪕⩵≡≧⩾≥⟵≫⪊≩⪈≳⪆⋛⪌≷⇔↔≦⩽⪅⋚⪋≲≤⪉≨⪇≴←⟵⇐↔⇔→⟶⇒↦≹∈∋∌∉≸≮≯≠≾≼≼⪹⪵⇒≿⫅⊆⫋⊊⊂≽≽⪺⪶⋩≻⫆⊇⫌⊋⊃⋑⋐]|:./g, function(x) {
|
|||
|
if(x.match(/:./g))
|
|||
|
return ": " + x[1];
|
|||
|
return "\u2009" + x + "\u2009";
|
|||
|
}).trim();
|
|||
|
},
|
|||
|
|
|||
|
parse_str: function(str, cursorpos) {
|
|||
|
var bracketstack = [], beginstack = [];
|
|||
|
var tokens = this.tokenize(str);
|
|||
|
|
|||
|
var carettrace = 0;
|
|||
|
var mathmodebegin = -1;
|
|||
|
|
|||
|
for(var i = 0; i < tokens.length; i++) {
|
|||
|
if(tokens[i].object == "{") {
|
|||
|
bracketstack.push(i);
|
|||
|
}else if(tokens[i].object == "}" && bracketstack.length > 0) {
|
|||
|
tokens[bracketstack[bracketstack.length-1]].closed = true;
|
|||
|
tokens[i].closed = true;
|
|||
|
bracketstack.pop();
|
|||
|
}
|
|||
|
|
|||
|
if(tokens[i].object == "\\begin") {
|
|||
|
beginstack.push(i);
|
|||
|
}else if(tokens[i].object == "\\end" && beginstack.length > 0) {
|
|||
|
tokens[beginstack[beginstack.length-1]].closed = true;
|
|||
|
tokens[i].closed = true;
|
|||
|
beginstack.pop();
|
|||
|
}
|
|||
|
|
|||
|
if(tokens[i].object == "$") {
|
|||
|
if(mathmodebegin != -1) {
|
|||
|
tokens[mathmodebegin].closed = true;
|
|||
|
tokens[i].closed = true;
|
|||
|
}else {
|
|||
|
mathmodebegin = i;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if(cursorpos > carettrace && cursorpos <= carettrace + tokens[i].object.length)
|
|||
|
tokens[i].caret = cursorpos - carettrace;
|
|||
|
carettrace = carettrace + tokens[i].object.length;
|
|||
|
}
|
|||
|
|
|||
|
for(var i = 0; i < bracketstack.length; i++)
|
|||
|
tokens[bracketstack[i]].closed = false;
|
|||
|
for(var i = 0; i < beginstack.length; i++)
|
|||
|
tokens[beginstack[i]].closed = false;
|
|||
|
|
|||
|
var res = this.parse(tokens);
|
|||
|
return res;
|
|||
|
},
|
|||
|
|
|||
|
"itemize": function(tokens) {
|
|||
|
console.log("itemize");
|
|||
|
|
|||
|
var tmp_tokens = [];
|
|||
|
for(var i = 0; i < tokens.length - 1; i++) {
|
|||
|
if(tokens[i].object == "\\item" && this.whitespace(tokens[i + 1].object)) {
|
|||
|
tmp_tokens.push(tokens[i]);
|
|||
|
i = i + 1;
|
|||
|
}else if(tokens[i + 1].object == "\\item" && this.whitespace(tokens[i].object)) {
|
|||
|
}else if(tokens[i].object.indexOf("\n") > -1) {
|
|||
|
tmp_tokens.push({object:"\n"});
|
|||
|
}else {
|
|||
|
tmp_tokens.push(tokens[i]);
|
|||
|
}
|
|||
|
}
|
|||
|
console.log(tmp_tokens);
|
|||
|
tag_table["\\item"] = {type:"symbol",value:"\n • "};
|
|||
|
var res = this.parse(tmp_tokens);
|
|||
|
tag_table["\\item"] = undefined;
|
|||
|
|
|||
|
res.text = res.text.replace(/\n( • )?/g, function(x) { if(x.length == 1) return "\n "; else return x; }) + "\n\n";
|
|||
|
|
|||
|
return res;
|
|||
|
},
|
|||
|
|
|||
|
"theorem": function(tokens) {
|
|||
|
var res = this.parse(this.trim_tokens(tokens));
|
|||
|
res.text = tag_table["\\textbf"].value("Theorem: ") + res.text + "\n";
|
|||
|
return res;
|
|||
|
},
|
|||
|
|
|||
|
"proof": function(tokens) {
|
|||
|
var res = this.parse(this.trim_tokens(tokens));
|
|||
|
res.text = tag_table["\\textbf"].value("Proof: ") + res.text + "\n\u200F□\u200F\n";
|
|||
|
return res;
|
|||
|
},
|
|||
|
|
|||
|
"align*": function(tokens) {
|
|||
|
var res = this.parse( [{object:"$",closed:true}].concat(this.trim_tokens(tokens)).concat([{object:"$",closed:true}]) );
|
|||
|
res.text = "\n " + res.text.replace(/\n/g, "\n ") + "\n\n";
|
|||
|
return res;
|
|||
|
},
|
|||
|
|
|||
|
parse_depth:0,
|
|||
|
parse: function(tokens) {
|
|||
|
this.parse_depth++;
|
|||
|
|
|||
|
var res = "", mathmode = null;
|
|||
|
var cursorpos = -1;
|
|||
|
|
|||
|
var decorator_stack = [];
|
|||
|
|
|||
|
for(var i = 0; i < tokens.length; i++) {
|
|||
|
if(tokens[i].caret != undefined && tokens[i].closed == undefined) {
|
|||
|
if(tokens[i].object != "\\\\"
|
|||
|
&& tokens[i].object != "\\_"
|
|||
|
&& tokens[i].object != "\\}"
|
|||
|
&& tokens[i].object != "\\{"
|
|||
|
&& tokens[i].object != "\\$"
|
|||
|
&& tokens[i].object != "\\^"
|
|||
|
&& tokens[i].object != "\\$") {
|
|||
|
cursorpos = res.length + tokens[i].caret;
|
|||
|
res = res + tokens[i].object;
|
|||
|
continue;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if(tokens[i].object == "{" || tokens[i].object == "}") {
|
|||
|
if(tokens[i].closed != true)
|
|||
|
res = res + tokens[i].object;
|
|||
|
if(tokens[i].caret != undefined && tokens[i].object == "}") {
|
|||
|
cursorpos = cursorpos + res.length + 1;
|
|||
|
}
|
|||
|
}else if(tokens[i].object == "$" && tokens[i].closed == true) {
|
|||
|
if(mathmode == null) {
|
|||
|
mathmode = res;
|
|||
|
res = "";
|
|||
|
}else {
|
|||
|
res = mathmode + this.reformat_math(res);
|
|||
|
mathmode = null;
|
|||
|
}
|
|||
|
}else {
|
|||
|
if(this.tag(tokens[i])) {
|
|||
|
if(tag_table[tokens[i].object] != undefined) {
|
|||
|
if(tag_table[tokens[i].object].type == "symbol") {
|
|||
|
var val = tag_table[tokens[i].object].value;
|
|||
|
res = res + val;
|
|||
|
if(tokens[i].caret != undefined && cursorpos == -1)
|
|||
|
cursorpos = res.length + val.length - 1;
|
|||
|
}else if(tag_table[tokens[i].object].type == "decorator" && i < tokens.length - 1) {
|
|||
|
var subblock = this.extract_block(tokens, i+1);
|
|||
|
if(subblock.length > 0) {
|
|||
|
sret = this.parse(subblock);
|
|||
|
var subs = tag_table[tokens[i].object].value(sret.text);
|
|||
|
res = res + subs;
|
|||
|
if(sret.caret != -1 && cursorpos == -1) {
|
|||
|
cursorpos = res.length;
|
|||
|
}
|
|||
|
i = i + subblock.length;
|
|||
|
}else {
|
|||
|
res = res + tokens[i].object;
|
|||
|
}
|
|||
|
}else if(tag_table[tokens[i].object].type == "decorator2" && i < tokens.length - 1) {
|
|||
|
var subblock = this.extract_block(tokens, i+1);
|
|||
|
var init_i = i;
|
|||
|
if(subblock.length > 0) {
|
|||
|
i = i + subblock.length;
|
|||
|
var subblock2 = this.extract_block(tokens, i+1);
|
|||
|
if(subblock2.length > 0) {
|
|||
|
var sret1 = this.parse(subblock);
|
|||
|
var sret2 = this.parse(subblock2);
|
|||
|
if(cursorpos == -1)
|
|||
|
cursorpos = sret1.caret + res.length;
|
|||
|
if(cursorpos == -1)
|
|||
|
cursorpos = sret2.caret + res.length;
|
|||
|
var subs = tag_table[tokens[init_i].object].value(sret1.text, sret2.text);
|
|||
|
res = res + subs;
|
|||
|
|
|||
|
i = i + subblock2.length;
|
|||
|
|
|||
|
if((sret1.caret != -1 || sret2.caret != -1) && cursorpos == -1) {
|
|||
|
cursorpos = res.length + subs.length + 2;
|
|||
|
}else if(tokens[i].caret != undefined) {
|
|||
|
cursorpos = res.length + subs.length + 1;
|
|||
|
|
|||
|
}
|
|||
|
}else {
|
|||
|
res = res + tokens[init_i].object;
|
|||
|
i = init_i;
|
|||
|
if(tokens[i + 1].object == "{") {
|
|||
|
tokens[i + 1].closed = undefined;
|
|||
|
console.log(tokens[i + subblock.length]);
|
|||
|
tokens[i + subblock.length].closed = undefined;
|
|||
|
}
|
|||
|
console.log("frac not finished:");
|
|||
|
console.log(tokens);
|
|||
|
}
|
|||
|
}else {
|
|||
|
res = res + tokens[i].object;
|
|||
|
}
|
|||
|
}else
|
|||
|
res = res + tokens[i].object;
|
|||
|
}else if(tokens[i].object == "\\begin" && i+1 < tokens.length) {
|
|||
|
if(tokens[i+1].object == "{") {
|
|||
|
var subblock = this.extract_block(tokens, i, "\\begin", "\\end");
|
|||
|
var argument = this.extract_block(tokens, i+1);
|
|||
|
if(argument.length > 0) {
|
|||
|
tokens[i+1].closed = false;
|
|||
|
tokens[i+argument.length].closed = false;
|
|||
|
|
|||
|
if(subblock.length > 0) {
|
|||
|
var argument_parsed = this.parse(argument.slice(1,argument.length-1));
|
|||
|
if(argument_parsed.text != this.parse(this.extract_block(tokens, i + subblock.length)).text) {
|
|||
|
res = res + tokens[i].object;
|
|||
|
continue;
|
|||
|
}
|
|||
|
if(argument_parsed.text in this) {
|
|||
|
i = i + subblock.length + argument.length;
|
|||
|
var parsed_block = this[argument_parsed.text](subblock.slice(argument.length + 1, subblock.length));
|
|||
|
res = res + parsed_block.text;
|
|||
|
|
|||
|
}else {
|
|||
|
res = res + tokens[i].object;
|
|||
|
}
|
|||
|
}else
|
|||
|
res = res + tokens[i].object;
|
|||
|
}else {
|
|||
|
res = res + tokens[i].object;
|
|||
|
}
|
|||
|
}else
|
|||
|
res = res + tokens[i].object;
|
|||
|
}else if(tokens[i].object == "\\end" && i+1 < tokens.length) {
|
|||
|
if(tokens[i+1].object == "{") {
|
|||
|
var argument = this.extract_block(tokens, i+1);
|
|||
|
if(argument.length > 0) {
|
|||
|
tokens[i+1].closed = false;
|
|||
|
tokens[i+argument.length].closed = false;
|
|||
|
res = res + tokens[i].object;
|
|||
|
}else {
|
|||
|
res = res + tokens[i].object;
|
|||
|
}
|
|||
|
}else
|
|||
|
res = res + tokens[i].object;
|
|||
|
}else {
|
|||
|
res = res + tokens[i].object;
|
|||
|
}
|
|||
|
}else {
|
|||
|
res = res + tokens[i].object;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
this.parse_depth--;
|
|||
|
return {text:res, caret:cursorpos};
|
|||
|
}
|
|||
|
}
|