2023-03-31 06:59:08 +02:00
|
|
|
#region evaluator
|
2023-04-15 14:48:29 +02:00
|
|
|
function __funcTree(symbol, l = noone, r = noone) constructor {
|
2023-03-31 06:59:08 +02:00
|
|
|
self.symbol = symbol;
|
|
|
|
self.l = l;
|
|
|
|
self.r = r;
|
|
|
|
isFunc = false;
|
|
|
|
|
2023-04-15 14:48:29 +02:00
|
|
|
static getVal = function(val, inp = 0) {
|
|
|
|
if(is_struct(val)) return val.eval();
|
|
|
|
if(is_real(val)) return val;
|
|
|
|
|
|
|
|
var splt = string_splice(val, "[");
|
|
|
|
var key = splt[0];
|
|
|
|
var _val = 0;
|
|
|
|
|
|
|
|
if(key == "value") {
|
|
|
|
_val = inp;
|
|
|
|
for( var i = 1; i < array_length(splt); i++ ) {
|
|
|
|
if(!is_array(_val)) break;
|
|
|
|
|
|
|
|
var _ind = toNumber(splt[i]);
|
|
|
|
_val = array_safe_get(_val, _ind, 0);
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
_val = nodeGetData(val);
|
|
|
|
|
|
|
|
return _val;
|
|
|
|
}
|
|
|
|
|
|
|
|
static _validate = function(val) {
|
|
|
|
if(is_real(val)) return true;
|
|
|
|
if(is_struct(val)) return val.validate();
|
|
|
|
|
|
|
|
var splt = string_splice(val, "[");
|
|
|
|
if(splt[0] == "value") return true;
|
|
|
|
|
|
|
|
var strs = string_splice(val, ".");
|
|
|
|
if(array_length(strs) == 0) return false;
|
|
|
|
if(array_length(strs) == 1) {
|
|
|
|
var splt = string_splice(val, "[");
|
|
|
|
return GLOBAL.inputExist(splt[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(strs[0] == "Project") {
|
|
|
|
switch(strs[1]) {
|
|
|
|
case "frame" :
|
|
|
|
case "frameTotal" :
|
|
|
|
case "fps" :
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
var key = strs[0];
|
|
|
|
if(!ds_map_exists(NODE_NAME_MAP, key)) return false;
|
|
|
|
|
|
|
|
var node = NODE_NAME_MAP[? key];
|
|
|
|
var splt = string_splice(strs[1], "[");
|
|
|
|
if(array_length(splt) < 2) return false;
|
|
|
|
|
|
|
|
var mmap = splt[0] == "inputs"? node.inputMap : node.outputMap;
|
|
|
|
if(array_length(splt) < 2) return false;
|
|
|
|
|
|
|
|
var mkey = string_replace_all(string_replace(splt[1], "]", ""), "\"", "");
|
|
|
|
return ds_map_exists(mmap, mkey);
|
|
|
|
}
|
|
|
|
|
|
|
|
static validate = function() {
|
|
|
|
return _validate(l) && _validate(r);
|
|
|
|
}
|
|
|
|
|
|
|
|
static eval = function(inp = 0) {
|
|
|
|
var v1 = getVal(l, inp);
|
|
|
|
var v2 = getVal(r, inp);
|
|
|
|
|
|
|
|
//print("symbol " + string(symbol));
|
|
|
|
//print("l : " + string(l));
|
|
|
|
//print("r : " + string(r));
|
|
|
|
//print("v1 : " + string(v1));
|
|
|
|
//print("v2 : " + string(v2));
|
|
|
|
//print("====================");
|
2023-03-31 06:59:08 +02:00
|
|
|
|
|
|
|
switch(symbol) {
|
|
|
|
case "+": return v1 + v2;
|
|
|
|
case "-": return v1 - v2;
|
2023-04-15 14:48:29 +02:00
|
|
|
case "*": return v1 * v2;
|
2023-03-31 06:59:08 +02:00
|
|
|
case "^": return power(v1, v2);
|
2023-04-15 14:48:29 +02:00
|
|
|
case "/": return v1 / v2;
|
2023-03-31 06:59:08 +02:00
|
|
|
|
|
|
|
case "sin" : return sin(v1);
|
|
|
|
case "cos" : return cos(v1);
|
|
|
|
case "tan" : return tan(v1);
|
|
|
|
case "abs" : return abs(v1);
|
|
|
|
case "round" : return round(v1);
|
|
|
|
case "ceil" : return ceil(v1);
|
|
|
|
case "floor" : return floor(v1);
|
|
|
|
}
|
|
|
|
|
2023-04-15 14:48:29 +02:00
|
|
|
return v1;
|
2023-03-31 06:59:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-15 14:48:29 +02:00
|
|
|
function evaluateFunctionTree(fx) {
|
2023-03-31 06:59:08 +02:00
|
|
|
var pres = global.EQUATION_PRES;
|
|
|
|
var vl = ds_stack_create();
|
|
|
|
var op = ds_stack_create();
|
|
|
|
|
|
|
|
fx = string_replace_all(fx, " ", "");
|
|
|
|
fx = string_replace_all(fx, "\n", "");
|
|
|
|
|
|
|
|
var len = string_length(fx);
|
|
|
|
var l = 1;
|
|
|
|
var ch, cch;
|
|
|
|
|
|
|
|
while(l <= len) {
|
|
|
|
ch = string_char_at(fx, l);
|
|
|
|
|
2023-04-15 14:48:29 +02:00
|
|
|
if(ds_map_exists(pres, ch)) { //symbol is operator
|
2023-03-31 06:59:08 +02:00
|
|
|
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-04-15 14:48:29 +02:00
|
|
|
while(pres[? ch] <= pres[? ds_stack_top(op)] && !ds_stack_empty(op))
|
|
|
|
ds_stack_push(vl, buildFuncTree(ds_stack_pop(op), vl));
|
2023-03-31 06:59:08 +02:00
|
|
|
ds_stack_push(op, ch);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
l++;
|
|
|
|
} else if (ch == "(") {
|
|
|
|
ds_stack_push(op, ch);
|
|
|
|
l++;
|
|
|
|
} else if (ch == ")") {
|
|
|
|
while(ds_stack_top(op) != "(" && !ds_stack_empty(op)) {
|
2023-04-15 14:48:29 +02:00
|
|
|
ds_stack_push(vl, buildFuncTree(ds_stack_pop(op), vl));
|
2023-03-31 06:59:08 +02:00
|
|
|
}
|
|
|
|
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-04-15 14:48:29 +02:00
|
|
|
case "e": ds_stack_push(vl, 2.71828); break;
|
|
|
|
case "pi": ds_stack_push(vl, pi); break;
|
|
|
|
default : ds_stack_push(vl, isNumber(vsl)? toNumber(vsl) : vsl); break;
|
2023-03-31 06:59:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-15 14:48:29 +02:00
|
|
|
while(!ds_stack_empty(op))
|
|
|
|
ds_stack_push(vl, buildFuncTree(ds_stack_pop(op), vl));
|
2023-03-31 06:59:08 +02:00
|
|
|
ds_stack_destroy(op);
|
|
|
|
|
2023-04-15 14:48:29 +02:00
|
|
|
var tree = ds_stack_empty(vl)? noone : ds_stack_pop(vl)
|
|
|
|
ds_stack_destroy(vl);
|
|
|
|
|
|
|
|
if(is_string(tree))
|
|
|
|
tree = new __funcTree("", tree);
|
|
|
|
|
|
|
|
return tree;
|
|
|
|
}
|
|
|
|
|
|
|
|
function buildFuncTree(operator, vl) {
|
|
|
|
if(ds_stack_empty(vl)) return noone;
|
|
|
|
|
|
|
|
switch(operator) {
|
|
|
|
case "+":
|
|
|
|
if(ds_stack_size(vl) >= 2) return new __funcTree("+", ds_stack_pop(vl), ds_stack_pop(vl));
|
|
|
|
case "-":
|
|
|
|
if(ds_stack_size(vl) >= 2) return new __funcTree("-", ds_stack_pop(vl), ds_stack_pop(vl));
|
|
|
|
else return new __funcTree("-", ds_stack_pop(vl), 0);
|
|
|
|
case "*":
|
|
|
|
if(ds_stack_size(vl) >= 2) return new __funcTree("*", ds_stack_pop(vl), ds_stack_pop(vl));
|
|
|
|
case "^":
|
|
|
|
if(ds_stack_size(vl) >= 2) return new __funcTree("^", ds_stack_pop(vl), ds_stack_pop(vl));
|
|
|
|
case "/":
|
|
|
|
if(ds_stack_size(vl) >= 2) return new __funcTree("/", ds_stack_pop(vl), ds_stack_pop(vl));
|
|
|
|
|
|
|
|
default: return new __funcTree(operator, ds_stack_pop(vl));
|
|
|
|
}
|
|
|
|
|
|
|
|
return noone;
|
2023-03-31 06:59:08 +02:00
|
|
|
}
|
|
|
|
#endregion
|