Pixel-Composer/scripts/string_eval/string_eval.gml

136 lines
3.8 KiB
Text
Raw Normal View History

2023-03-31 06:59:08 +02:00
#region data
2023-03-23 06:57:31 +01:00
global.EQUATION_PRES = ds_map_create();
global.EQUATION_PRES[? "+"] = 1;
global.EQUATION_PRES[? "-"] = 1;
2023-05-22 20:31:55 +02:00
global.EQUATION_PRES[? "_"] = 9; //unary negative
2023-03-23 06:57:31 +01:00
global.EQUATION_PRES[? "*"] = 2;
global.EQUATION_PRES[? "/"] = 2;
global.EQUATION_PRES[? "^"] = 3;
2023-05-03 21:42:17 +02:00
global.EQUATION_PRES[? "|"] = 5; //array accerssor symbol
2023-03-23 06:57:31 +01:00
global.EQUATION_PRES[? "sin"] = 5;
global.EQUATION_PRES[? "cos"] = 5;
global.EQUATION_PRES[? "tan"] = 5;
global.EQUATION_PRES[? "abs"] = 5;
global.EQUATION_PRES[? "round"] = 5;
global.EQUATION_PRES[? "ceil"] = 5;
global.EQUATION_PRES[? "floor"] = 5;
2023-03-31 06:59:08 +02:00
#endregion
2023-03-23 06:57:31 +01:00
2023-03-31 06:59:08 +02:00
#region evaluator
2023-01-09 03:14:20 +01:00
function evaluateFunction(fx, params = {}) {
2023-03-23 06:57:31 +01:00
var pres = global.EQUATION_PRES;
var vl = ds_stack_create();
var op = ds_stack_create();
2022-01-13 05:24:03 +01:00
2023-03-11 01:40:17 +01:00
fx = string_replace_all(fx, " ", "");
2022-01-13 05:24:03 +01:00
fx = string_replace_all(fx, "\n", "");
var len = string_length(fx);
2023-03-23 06:57:31 +01:00
var l = 1;
2023-05-22 20:31:55 +02:00
var ch, cch, _ch;
2022-01-13 05:24:03 +01:00
while(l <= len) {
ch = string_char_at(fx, l);
if(ds_map_exists(pres, ch)) {
if(ds_stack_empty(op)) ds_stack_push(op, ch);
else {
if(pres[? ch] > pres[? ds_stack_top(op)] || ds_stack_top(op) == "(") ds_stack_push(op, ch);
else {
2023-05-22 20:31:55 +02:00
if(ch == "-" && ds_map_exists(pres, _ch)) ch = "_"; //unary negative
while(pres[? ch] <= pres[? ds_stack_top(op)] && !ds_stack_empty(op))
2022-01-13 05:24:03 +01:00
ds_stack_push(vl, evalToken(ds_stack_pop(op), vl));
ds_stack_push(op, ch);
}
}
l++;
} else if (ch == "(") {
ds_stack_push(op, ch);
l++;
} else if (ch == ")") {
2023-05-22 20:31:55 +02:00
while(ds_stack_top(op) != "(" && !ds_stack_empty(op))
2022-01-13 05:24:03 +01:00
ds_stack_push(vl, evalToken(ds_stack_pop(op), vl));
ds_stack_pop(op);
l++;
} else {
var vsl = "";
while(l <= len) {
cch = string_char_at(fx, l);
if(ds_map_exists(pres, cch) || cch == ")" || cch == "(") break;
vsl += cch;
l++;
}
if(vsl == "") continue;
if(ds_map_exists(pres, vsl)) {
ds_stack_push(op, vsl);
} else {
switch(vsl) {
2023-01-09 03:14:20 +01:00
case "e": ds_stack_push(vl, 2.71828); break;
2022-01-13 05:24:03 +01:00
case "pi": ds_stack_push(vl, pi); break;
2022-12-18 03:20:38 +01:00
default :
2023-01-09 03:14:20 +01:00
if(variable_struct_exists(params, vsl))
ds_stack_push(vl, variable_struct_get(params, vsl));
else
ds_stack_push(vl, toNumber(vsl));
2022-01-13 05:24:03 +01:00
break;
}
}
}
2023-05-22 20:31:55 +02:00
_ch = ch;
2022-01-13 05:24:03 +01:00
}
while(!ds_stack_empty(op)) {
ds_stack_push(vl, evalToken(ds_stack_pop(op), vl));
}
ds_stack_destroy(op);
return ds_stack_empty(vl)? 0 : ds_stack_pop(vl);
}
function evalToken(operator, vl) {
if(ds_stack_empty(vl)) return 0;
switch(operator) {
2023-03-11 01:40:17 +01:00
case "+":
if(ds_stack_size(vl) >= 2)
return ds_stack_pop(vl) + ds_stack_pop(vl);
2022-01-13 05:24:03 +01:00
case "-":
2023-03-11 01:40:17 +01:00
if(ds_stack_size(vl) >= 2)
return -ds_stack_pop(vl) + ds_stack_pop(vl);
else
return -ds_stack_pop(vl);
2023-05-22 20:31:55 +02:00
case "_":
return -ds_stack_pop(vl);
2023-03-11 01:40:17 +01:00
case "*":
if(ds_stack_size(vl) >= 2)
return ds_stack_pop(vl) * ds_stack_pop(vl);
2023-01-17 08:11:55 +01:00
case "^":
2023-03-11 01:40:17 +01:00
if(ds_stack_size(vl) < 2) return 1;
var ex = ds_stack_pop(vl);
var bs = ds_stack_pop(vl);
return power(bs, ex);
2022-01-13 05:24:03 +01:00
case "/":
2023-03-11 01:40:17 +01:00
if(ds_stack_size(vl) < 2) return 0;
var _d = ds_stack_pop(vl);
if(_d == 0) return 0;
return ds_stack_pop(vl) / _d;
2022-01-13 05:24:03 +01:00
2023-03-11 01:40:17 +01:00
case "sin" : if(ds_stack_size(vl) >= 1) return sin(ds_stack_pop(vl));
case "cos" : if(ds_stack_size(vl) >= 1) return cos(ds_stack_pop(vl));
case "tan" : if(ds_stack_size(vl) >= 1) return tan(ds_stack_pop(vl));
case "abs" : if(ds_stack_size(vl) >= 1) return abs(ds_stack_pop(vl));
case "round" : if(ds_stack_size(vl) >= 1) return round(ds_stack_pop(vl));
case "ceil" : if(ds_stack_size(vl) >= 1) return ceil(ds_stack_pop(vl));
case "floor" : if(ds_stack_size(vl) >= 1) return floor(ds_stack_pop(vl));
2022-01-13 05:24:03 +01:00
}
return 0;
}
#endregion