2023-03-31 06:59:08 +02:00
|
|
|
#region data
|
2023-07-29 10:10:48 +02:00
|
|
|
global.LOG_EXPRESSION = false;
|
|
|
|
|
2023-07-14 20:34:35 +02:00
|
|
|
global.EVALUATE_HEAD = noone;
|
|
|
|
|
2023-07-12 21:00:05 +02:00
|
|
|
global.EQUATION_PRES = ds_map_create();
|
2023-06-13 21:24:05 +02:00
|
|
|
global.EQUATION_PRES[? "+"] = 1;
|
|
|
|
global.EQUATION_PRES[? "-"] = 1;
|
2023-06-26 18:55:41 +02:00
|
|
|
global.EQUATION_PRES[? "∸"] = 9; //unary negative
|
2023-06-13 21:24:05 +02:00
|
|
|
global.EQUATION_PRES[? "*"] = 2;
|
|
|
|
global.EQUATION_PRES[? "/"] = 2;
|
2023-07-12 21:00:05 +02:00
|
|
|
global.EQUATION_PRES[? "%"] = 2;
|
2023-06-13 21:24:05 +02:00
|
|
|
global.EQUATION_PRES[? "$"] = 3;
|
|
|
|
|
|
|
|
global.EQUATION_PRES[? "&"] = 5;
|
|
|
|
global.EQUATION_PRES[? "|"] = 4;
|
|
|
|
global.EQUATION_PRES[? "^"] = 3;
|
|
|
|
global.EQUATION_PRES[? "<"] = 3;
|
|
|
|
global.EQUATION_PRES[? "»"] = 6;
|
|
|
|
global.EQUATION_PRES[? "«"] = 6;
|
|
|
|
global.EQUATION_PRES[? "~"] = 9;
|
|
|
|
|
2023-07-12 21:00:05 +02:00
|
|
|
global.EQUATION_PRES[? "="] = -99;
|
|
|
|
global.EQUATION_PRES[? "⩵"] = -1; //==
|
|
|
|
global.EQUATION_PRES[? "≠"] = -1; //!=
|
|
|
|
global.EQUATION_PRES[? "<"] = 0;
|
|
|
|
global.EQUATION_PRES[? ">"] = 0;
|
|
|
|
global.EQUATION_PRES[? "≤"] = 0;
|
|
|
|
global.EQUATION_PRES[? "≥"] = 0;
|
2023-06-13 21:24:05 +02:00
|
|
|
|
|
|
|
global.EQUATION_PRES[? "@"] = 5; //array accerssor symbol
|
|
|
|
|
2023-07-11 20:36:44 +02:00
|
|
|
global.FUNCTIONS = ds_map_create();
|
2023-07-12 16:28:32 +02:00
|
|
|
global.FUNCTIONS[? "sin"] = [ ["radian"], function(val) { return sin(val[0]); } ];
|
|
|
|
global.FUNCTIONS[? "cos"] = [ ["radian"], function(val) { return cos(val[0]); } ];
|
|
|
|
global.FUNCTIONS[? "tan"] = [ ["radian"], function(val) { return tan(val[0]); } ];
|
2023-07-14 20:34:35 +02:00
|
|
|
|
2023-07-12 16:28:32 +02:00
|
|
|
global.FUNCTIONS[? "abs"] = [ ["number"], function(val) { return abs(val[0]); } ];
|
|
|
|
global.FUNCTIONS[? "round"] = [ ["number"], function(val) { return round(val[0]); } ];
|
|
|
|
global.FUNCTIONS[? "ceil"] = [ ["number"], function(val) { return ceil(val[0]); } ];
|
|
|
|
global.FUNCTIONS[? "floor"] = [ ["number"], function(val) { return floor(val[0]); } ];
|
2023-07-11 20:36:44 +02:00
|
|
|
|
2023-07-12 21:00:05 +02:00
|
|
|
global.FUNCTIONS[? "lerp"] = [ ["number_0", "number_1", "amount"], function(val) { return lerp(array_safe_get(val, 0), array_safe_get(val, 1), array_safe_get(val, 2)); } ];
|
|
|
|
|
2023-07-14 20:34:35 +02:00
|
|
|
global.FUNCTIONS[? "wiggle"] = [ ["time", "frequency", "octave = 1", "seed = 0"], function(val) {
|
|
|
|
return wiggle(0, 1, array_safe_get(val, 1),
|
|
|
|
array_safe_get(val, 0),
|
|
|
|
array_safe_get(val, 3, 0),
|
|
|
|
array_safe_get(val, 2, 1));
|
|
|
|
} ];
|
|
|
|
global.FUNCTIONS[? "random"] = [ ["min = 0", "max = 1"], function(val) {
|
|
|
|
return random_range(array_safe_get(val, 0, 0),
|
|
|
|
array_safe_get(val, 1, 1));
|
|
|
|
} ];
|
|
|
|
global.FUNCTIONS[? "irandom"] = [ ["min = 0", "max = 1"], function(val) {
|
|
|
|
return irandom_range(array_safe_get(val, 0, 0),
|
|
|
|
array_safe_get(val, 1, 1));
|
|
|
|
} ];
|
2023-07-17 19:58:33 +02:00
|
|
|
|
2023-07-18 17:51:40 +02:00
|
|
|
global.FUNCTIONS[? "range"] = [ ["length", "start = 0", "step = 1"], function(val) {
|
|
|
|
var arr = array_create(array_safe_get(val, 0, 0));
|
2023-07-25 20:12:40 +02:00
|
|
|
for( var i = 0, n = array_length(arr); i < n; i++ )
|
2023-07-18 17:51:40 +02:00
|
|
|
arr[i] = array_safe_get(val, 1, 0) + i * array_safe_get(val, 2, 1);
|
|
|
|
return arr;
|
|
|
|
} ];
|
2023-03-31 06:59:08 +02:00
|
|
|
#endregion
|
2023-03-23 06:57:31 +01:00
|
|
|
|
2023-06-13 21:24:05 +02:00
|
|
|
function functionStringClean(fx) {
|
2023-07-12 16:28:32 +02:00
|
|
|
var ch = "", ind = 0, len = string_length(fx);
|
|
|
|
var _fx = "", str = false;
|
|
|
|
while(ind++ <= len) {
|
|
|
|
ch = string_char_at(fx, ind);
|
|
|
|
|
|
|
|
if(ch == " ") {
|
|
|
|
if(str)
|
|
|
|
_fx += ch;
|
|
|
|
} else
|
|
|
|
_fx += ch;
|
|
|
|
|
|
|
|
if(ch == "\"")
|
|
|
|
str = !str;
|
|
|
|
}
|
|
|
|
|
|
|
|
fx = _fx;
|
|
|
|
|
|
|
|
|
2023-06-13 21:24:05 +02:00
|
|
|
fx = string_replace_all(fx, "\n", "");
|
|
|
|
fx = string_replace_all(fx, "**", "$");
|
|
|
|
fx = string_replace_all(fx, "<<", "«");
|
|
|
|
fx = string_replace_all(fx, ">>", "»");
|
|
|
|
|
2023-07-12 21:00:05 +02:00
|
|
|
fx = string_replace_all(fx, "==", "⩵");
|
2023-06-13 21:24:05 +02:00
|
|
|
fx = string_replace_all(fx, "!=", "≠");
|
|
|
|
fx = string_replace_all(fx, "<>", "≠");
|
|
|
|
fx = string_replace_all(fx, ">=", "≥");
|
|
|
|
fx = string_replace_all(fx, "<=", "≤");
|
|
|
|
|
2023-07-12 21:00:05 +02:00
|
|
|
fx = string_trim(fx);
|
2023-06-13 21:24:05 +02:00
|
|
|
|
|
|
|
return fx;
|
|
|
|
}
|
|
|
|
|
2023-03-31 06:59:08 +02:00
|
|
|
#region evaluator
|
2023-06-20 19:43:19 +02:00
|
|
|
enum EXPRESS_TREE_ANIM {
|
|
|
|
none,
|
|
|
|
base_value,
|
|
|
|
animated
|
|
|
|
}
|
|
|
|
|
2023-07-12 21:00:05 +02:00
|
|
|
function __funcList() constructor {
|
|
|
|
funcTrees = [];
|
|
|
|
|
|
|
|
static addFunction = function(fn) {
|
|
|
|
array_push(funcTrees, fn);
|
|
|
|
}
|
|
|
|
|
|
|
|
static validate = function() {
|
2023-07-25 20:12:40 +02:00
|
|
|
for( var i = 0, n = array_length(funcTrees); i < n; i++ )
|
2023-07-12 21:00:05 +02:00
|
|
|
if(!funcTrees[i].validate())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static isAnimated = function() {
|
2023-07-25 20:12:40 +02:00
|
|
|
for( var i = 0, n = array_length(funcTrees); i < n; i++ )
|
2023-07-12 21:00:05 +02:00
|
|
|
if(!funcTrees[i].isAnimated())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static eval = function(params = {}) {
|
|
|
|
//var _params = variable_clone(params);
|
|
|
|
var val = 0;
|
|
|
|
|
2023-07-25 20:12:40 +02:00
|
|
|
for( var i = 0, n = array_length(funcTrees); i < n; i++ )
|
2023-07-12 21:00:05 +02:00
|
|
|
val = funcTrees[i].eval(params);
|
|
|
|
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function __funcIf() constructor {
|
|
|
|
condition = noone;
|
|
|
|
if_true = new __funcList();
|
|
|
|
if_false = new __funcList();
|
|
|
|
|
|
|
|
static validate = function() {
|
|
|
|
if(condition != noone && !condition.validate()) return false;
|
|
|
|
if(if_true != noone && !if_true.validate()) return false;
|
|
|
|
if(if_false != noone && !if_false.validate()) return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static isAnimated = function() {
|
|
|
|
if(condition != noone && !condition.isAnimated()) return false;
|
|
|
|
if(if_true != noone && !if_true.isAnimated()) return false;
|
|
|
|
if(if_false != noone && !if_false.isAnimated()) return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static eval = function(params = {}) {
|
|
|
|
if(condition == noone) return 0;
|
|
|
|
|
|
|
|
var res = condition.eval(params);
|
2023-07-29 10:10:48 +02:00
|
|
|
printIf(global.LOG_EXPRESSION, $"<<<<<< IF {res} >>>>>>");
|
2023-07-12 21:00:05 +02:00
|
|
|
|
|
|
|
if(res) return if_true == noone? 0 : if_true.eval(params);
|
|
|
|
else return if_false == noone? 0 : if_false.eval(params);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function __funcFor() constructor {
|
|
|
|
itr_array = false;
|
|
|
|
|
|
|
|
cond_init = noone;
|
|
|
|
cond_iter = noone;
|
|
|
|
cond_term = noone;
|
|
|
|
|
|
|
|
cond_arr = noone;
|
|
|
|
|
|
|
|
cond_step = 1;
|
|
|
|
action = new __funcList();
|
|
|
|
|
|
|
|
static validate = function() {
|
|
|
|
if(itr_array) {
|
|
|
|
if(cond_arr == noone || !cond_arr.validate()) return false;
|
|
|
|
} else {
|
|
|
|
if(cond_init == noone || !cond_init.validate()) return false;
|
|
|
|
if(cond_term == noone || !cond_term.validate()) return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(action != noone && !action.validate()) return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static isAnimated = function() {
|
|
|
|
if(itr_array) {
|
|
|
|
if(cond_arr == noone || !cond_arr.isAnimated()) return false;
|
|
|
|
} else {
|
|
|
|
if(cond_init == noone || !cond_init.isAnimated()) return false;
|
|
|
|
if(cond_term == noone || !cond_term.isAnimated()) return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(action != noone && !action.isAnimated()) return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static eval = function(params = {}) {
|
|
|
|
if(itr_array) {
|
|
|
|
var _arr = cond_arr.eval(params);
|
2023-07-29 10:10:48 +02:00
|
|
|
printIf(global.LOG_EXPRESSION, $"<<<<<< FOR EACH {_arr} >>>>>>");
|
2023-07-25 20:12:40 +02:00
|
|
|
for( var i = 0, n = array_length(_arr); i < n; i++ ) {
|
2023-07-12 21:00:05 +02:00
|
|
|
var val = _arr[i];
|
|
|
|
params[$ cond_iter] = val;
|
|
|
|
|
2023-07-29 10:10:48 +02:00
|
|
|
printIf(global.LOG_EXPRESSION, $"<< ITER {i}: {cond_iter} = {val} >>");
|
2023-07-12 21:00:05 +02:00
|
|
|
action.eval(params);
|
|
|
|
}
|
|
|
|
} else {
|
2023-07-29 10:10:48 +02:00
|
|
|
printIf(global.LOG_EXPRESSION, "<< FOR >>");
|
2023-07-12 21:00:05 +02:00
|
|
|
cond_init.eval(params);
|
|
|
|
|
|
|
|
while(cond_term.eval(params)) {
|
|
|
|
action.eval(params);
|
|
|
|
cond_iter.eval(params);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-20 19:43:19 +02:00
|
|
|
function __funcTree(symbol, l = noone, r = noone) constructor {
|
|
|
|
self.symbol = symbol;
|
|
|
|
self.l = l;
|
|
|
|
self.r = r;
|
2023-07-12 16:28:32 +02:00
|
|
|
dependency = [];
|
2023-06-20 19:43:19 +02:00
|
|
|
|
|
|
|
static _string = function(str) {
|
|
|
|
return string_char_at(str, 1) == "\"" &&
|
|
|
|
string_char_at(str, string_length(str)) == "\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
static _string_trim = function(str) {
|
|
|
|
return _string(str)? string_copy(str, 2, string_length(str) - 2) : string(str);
|
|
|
|
}
|
|
|
|
|
2023-07-11 20:36:44 +02:00
|
|
|
static getVal = function(val, params = {}, getRaw = false) {
|
2023-07-23 20:21:35 +02:00
|
|
|
if(is_struct(val)) return val.eval(params, getRaw);
|
|
|
|
if(is_real(val)) return val;
|
|
|
|
if(getRaw) return val;
|
2023-07-12 21:00:05 +02:00
|
|
|
|
2023-07-11 20:36:44 +02:00
|
|
|
if(is_string(val)) val = string_trim(val);
|
2023-06-20 19:43:19 +02:00
|
|
|
|
2023-07-29 10:10:48 +02:00
|
|
|
//printIf(global.LOG_EXPRESSION, $" [ get struct {params}[{val}] ]");
|
2023-07-12 21:00:05 +02:00
|
|
|
|
2023-06-20 19:43:19 +02:00
|
|
|
if(struct_has(params, val))
|
|
|
|
return struct_try_get(params, val);
|
|
|
|
|
|
|
|
if(_string(string_trim(val)))
|
|
|
|
return string_trim(val);
|
|
|
|
|
|
|
|
return nodeGetData(val);
|
|
|
|
}
|
|
|
|
|
|
|
|
static _validate = function(val) {
|
|
|
|
if(is_real(val)) return true;
|
|
|
|
if(is_string(val)) return true;
|
|
|
|
if(is_struct(val)) return val.validate();
|
|
|
|
|
|
|
|
if(val == "value") return true;
|
2023-07-06 19:49:16 +02:00
|
|
|
if(PROJECT.globalNode.inputExist(val)) return true;
|
2023-06-20 19:43:19 +02:00
|
|
|
|
|
|
|
var strs = string_splice(val, ".");
|
|
|
|
if(array_length(strs) < 2) return false;
|
|
|
|
|
2023-07-12 16:28:32 +02:00
|
|
|
if(strs[0] == "Project")
|
|
|
|
return ds_map_exists(PROJECT_VARIABLES, strs[1]);
|
|
|
|
|
|
|
|
if(!ds_map_exists(PROJECT.nodeNameMap, strs[0]))
|
2023-06-20 19:43:19 +02:00
|
|
|
return false;
|
|
|
|
|
2023-07-14 20:34:35 +02:00
|
|
|
array_push_unique(dependency, strs[0]);
|
2023-07-12 16:28:32 +02:00
|
|
|
return true;
|
2023-06-20 19:43:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static validate = function() {
|
2023-07-12 16:28:32 +02:00
|
|
|
dependency = [];
|
|
|
|
|
2023-07-11 20:36:44 +02:00
|
|
|
if(ds_map_exists(global.FUNCTIONS, symbol)) {
|
|
|
|
if(!is_array(l)) return false;
|
2023-07-25 20:12:40 +02:00
|
|
|
for( var i = 0, n = array_length(l); i < n; i++ )
|
2023-07-11 20:36:44 +02:00
|
|
|
if(!_validate(l[i])) return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-06-20 19:43:19 +02:00
|
|
|
switch(symbol) {
|
|
|
|
case "@": return _validate(l);
|
2023-07-11 20:36:44 +02:00
|
|
|
case "【": return true;
|
|
|
|
case "": return true;
|
2023-06-20 19:43:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return _validate(l) && _validate(r);
|
|
|
|
}
|
|
|
|
|
|
|
|
static _isAnimated = function(val) {
|
|
|
|
if(is_real(val)) return EXPRESS_TREE_ANIM.none;
|
|
|
|
if(is_struct(val)) return val._isAnimated();
|
|
|
|
|
|
|
|
if(val == "value") return EXPRESS_TREE_ANIM.base_value;
|
2023-07-06 19:49:16 +02:00
|
|
|
if(PROJECT.globalNode.inputExist(val)) {
|
|
|
|
var _inp = PROJECT.globalNode.getInput(val);
|
2023-06-20 19:43:19 +02:00
|
|
|
if(_inp.is_anim) return EXPRESS_TREE_ANIM.animated;
|
|
|
|
}
|
|
|
|
|
|
|
|
return EXPRESS_TREE_ANIM.none;
|
|
|
|
}
|
|
|
|
|
|
|
|
static isAnimated = function() {
|
|
|
|
var anim = EXPRESS_TREE_ANIM.none;
|
|
|
|
anim = max(anim, _isAnimated(l));
|
|
|
|
if(symbol != "@")
|
|
|
|
anim = max(anim, _isAnimated(r));
|
|
|
|
|
|
|
|
return anim;
|
|
|
|
}
|
|
|
|
|
2023-07-23 20:21:35 +02:00
|
|
|
static eval = function(params = {}, isLeft = false) {
|
2023-07-11 20:36:44 +02:00
|
|
|
if(ds_map_exists(global.FUNCTIONS, symbol)) {
|
|
|
|
if(!is_array(l)) return 0;
|
|
|
|
|
|
|
|
var _fn = global.FUNCTIONS[? symbol];
|
|
|
|
var _ev = _fn[1];
|
|
|
|
var _l = array_create(array_length(l));
|
|
|
|
|
2023-07-25 20:12:40 +02:00
|
|
|
for( var i = 0, n = array_length(l); i < n; i++ )
|
2023-07-11 20:36:44 +02:00
|
|
|
_l[i] = getVal(l[i], params);
|
|
|
|
|
|
|
|
var res = _ev(_l);
|
2023-07-29 10:10:48 +02:00
|
|
|
printIf(global.LOG_EXPRESSION, $"Function {symbol}{_l} = {res}");
|
|
|
|
printIf(global.LOG_EXPRESSION, "====================");
|
2023-07-11 20:36:44 +02:00
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2023-07-23 20:21:35 +02:00
|
|
|
var v1 = getVal(l, params, symbol == "=" || symbol == "【" || isLeft);
|
2023-07-11 20:36:44 +02:00
|
|
|
var v2 = getVal(r, params);
|
|
|
|
|
|
|
|
var res = 0;
|
2023-06-20 19:43:19 +02:00
|
|
|
|
2023-07-11 20:36:44 +02:00
|
|
|
if(symbol == "") {
|
|
|
|
res = v1;
|
|
|
|
} else if(symbol == "【") { //array builder
|
|
|
|
res = array_create(array_length(v1));
|
2023-07-25 20:12:40 +02:00
|
|
|
for( var i = 0, n = array_length(res); i < n; i++ )
|
2023-07-11 20:36:44 +02:00
|
|
|
res[i] = getVal(v1[i], params);
|
|
|
|
} else if(symbol == "@") {
|
2023-07-23 20:21:35 +02:00
|
|
|
if(isLeft) res = [ v1, v2 ];
|
|
|
|
else res = is_real(v2)? array_safe_get(v1, v2) : 0;
|
2023-07-12 21:00:05 +02:00
|
|
|
} else if(symbol == "=") {
|
2023-07-23 20:21:35 +02:00
|
|
|
if(is_array(v1)) {
|
|
|
|
var val = params[$ v1[0]];
|
|
|
|
array_safe_set(val, v1[1], v2);
|
|
|
|
params[$ v1[0]] = val;
|
|
|
|
res = val;
|
|
|
|
} else {
|
|
|
|
params[$ v1] = v2;
|
|
|
|
res = v2;
|
|
|
|
}
|
2023-07-11 20:36:44 +02:00
|
|
|
} else if(is_array(v1) && !is_array(v2)) {
|
|
|
|
res = array_create(array_length(v1));
|
2023-07-25 20:12:40 +02:00
|
|
|
for( var i = 0, n = array_length(res); i < n; i++ )
|
2023-07-11 20:36:44 +02:00
|
|
|
res[i] = eval_real(array_safe_get(v1, i), v2);
|
|
|
|
} else if(!is_array(v1) && is_array(v2)) {
|
|
|
|
res = array_create(array_length(v2));
|
2023-07-25 20:12:40 +02:00
|
|
|
for( var i = 0, n = array_length(res); i < n; i++ )
|
2023-07-11 20:36:44 +02:00
|
|
|
res[i] = eval_real(v1, array_safe_get(v2, i));
|
|
|
|
} else if(is_array(v1) && is_array(v2)) {
|
|
|
|
res = array_create(max(array_length(v1), array_length(v2)));
|
2023-07-25 20:12:40 +02:00
|
|
|
for( var i = 0, n = array_length(res); i < n; i++ )
|
2023-07-11 20:36:44 +02:00
|
|
|
res[i] = eval_real(array_safe_get(v1, i), array_safe_get(v2, i));
|
|
|
|
} else
|
|
|
|
res = eval_real(v1, v2);
|
|
|
|
|
2023-07-29 10:10:48 +02:00
|
|
|
printIf(global.LOG_EXPRESSION, $"|{v1}|{symbol}|{v2}| = {res}");
|
|
|
|
printIf(global.LOG_EXPRESSION, $"symbol : {symbol}");
|
|
|
|
printIf(global.LOG_EXPRESSION, $"l : | {typeof(l)} |{l}|");
|
|
|
|
printIf(global.LOG_EXPRESSION, $"r : | {typeof(r)} |{r}|");
|
|
|
|
printIf(global.LOG_EXPRESSION, "====================");
|
2023-06-20 19:43:19 +02:00
|
|
|
|
2023-07-11 20:36:44 +02:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static eval_real = function(v1, v2) {
|
2023-06-20 19:43:19 +02:00
|
|
|
switch(symbol) {
|
|
|
|
case "+":
|
|
|
|
if(_string(v1) || _string(v2))
|
|
|
|
return _string_trim(v1) + _string_trim(v2);
|
|
|
|
if(is_real(v1) && is_real(v2))
|
|
|
|
return v1 + v2;
|
|
|
|
return 0;
|
|
|
|
case "-": return (is_real(v1) && is_real(v2))? v1 - v2 : 0;
|
2023-06-26 18:55:41 +02:00
|
|
|
case "∸": return is_real(v1)? -v1 : 0;
|
2023-06-20 19:43:19 +02:00
|
|
|
case "*": return (is_real(v1) && is_real(v2))? v1 * v2 : 0;
|
|
|
|
case "$": return (is_real(v1) && is_real(v2))? power(v1, v2) : 0;
|
2023-07-11 20:36:44 +02:00
|
|
|
case "/": return (is_real(v1) && is_real(v2) && v2 != 0)? v1 / v2 : 0;
|
2023-07-12 21:00:05 +02:00
|
|
|
case "%": return (is_real(v1) && is_real(v2) && v2 != 0)? v1 % v2 : 0;
|
2023-06-20 19:43:19 +02:00
|
|
|
|
|
|
|
case "&": return (is_real(v1) && is_real(v2))? v1 & v2 : 0;
|
|
|
|
case "|": return (is_real(v1) && is_real(v2))? v1 | v2 : 0;
|
|
|
|
case "^": return (is_real(v1) && is_real(v2))? v1 ^ v2 : 0;
|
|
|
|
case "«": return (is_real(v1) && is_real(v2))? v1 << v2 : 0;
|
|
|
|
case "»": return (is_real(v1) && is_real(v2))? v1 >> v2 : 0;
|
|
|
|
case "~": return is_real(v1)? ~v1 : 0;
|
|
|
|
|
2023-07-12 21:00:05 +02:00
|
|
|
case "⩵": return (is_real(v1) && is_real(v2))? v1 == v2 : 0;
|
2023-06-20 19:43:19 +02:00
|
|
|
case "≠": return (is_real(v1) && is_real(v2))? v1 != v2 : 0;
|
|
|
|
case "≤": return (is_real(v1) && is_real(v2))? v1 <= v2 : 0;
|
|
|
|
case "≥": return (is_real(v1) && is_real(v2))? v1 >= v2 : 0;
|
|
|
|
case ">": return (is_real(v1) && is_real(v2))? v1 > v2 : 0;
|
|
|
|
case "<": return (is_real(v1) && is_real(v2))? v1 < v2 : 0;
|
|
|
|
|
|
|
|
case "sin" : return is_real(v1)? sin(v1) : 0;
|
|
|
|
case "cos" : return is_real(v1)? cos(v1) : 0;
|
|
|
|
case "tan" : return is_real(v1)? tan(v1) : 0;
|
|
|
|
case "abs" : return is_real(v1)? abs(v1) : 0;
|
|
|
|
case "round" : return is_real(v1)? round(v1) : 0;
|
|
|
|
case "ceil" : return is_real(v1)? ceil(v1) : 0;
|
|
|
|
case "floor" : return is_real(v1)? floor(v1) : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return v1;
|
|
|
|
}
|
|
|
|
}
|
2023-07-12 21:00:05 +02:00
|
|
|
|
|
|
|
function functionStrip(fx) {
|
|
|
|
var el_st = 1;
|
|
|
|
var el_ed = 1;
|
|
|
|
|
|
|
|
for( var i = 1; i <= string_length(fx); i++ ) {
|
|
|
|
var cch = string_char_at(fx, i);
|
|
|
|
if(cch == "(") {
|
|
|
|
el_st = i + 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for( var i = string_length(fx); i >= 1; i-- ) {
|
|
|
|
var cch = string_char_at(fx, i);
|
|
|
|
if(cch == ")") {
|
|
|
|
el_ed = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return string_copy(fx, el_st, el_ed - el_st)
|
|
|
|
}
|
|
|
|
|
|
|
|
function evaluateFunctionList(fx) {
|
|
|
|
var fxs = string_split(fx, "\n");
|
|
|
|
var flist = new __funcList();
|
|
|
|
|
|
|
|
var call_st = ds_stack_create();
|
|
|
|
ds_stack_push(call_st, flist);
|
|
|
|
|
2023-07-25 20:12:40 +02:00
|
|
|
for( var i = 0, n = array_length(fxs); i < n; i++ ) {
|
2023-07-12 21:00:05 +02:00
|
|
|
var _fx = functionStringClean(fxs[i]);
|
|
|
|
if(_fx == "") continue;
|
|
|
|
|
|
|
|
var _fx_sp = string_split(_fx, "(");
|
|
|
|
if(string_char_at(_fx, 1) == "}") {
|
|
|
|
ds_stack_pop(call_st);
|
|
|
|
_fx = string_replace(_fx, "}", "");
|
|
|
|
}
|
|
|
|
|
|
|
|
if(array_length(_fx_sp) > 1) {
|
|
|
|
var _cond = functionStrip(_fx);
|
|
|
|
|
|
|
|
switch(_fx_sp[0]) {
|
|
|
|
case "if":
|
|
|
|
var con_if = new __funcIf();
|
|
|
|
con_if.condition = evaluateFunctionTree(_cond);
|
|
|
|
ds_stack_top(call_st).addFunction(con_if);
|
|
|
|
ds_stack_push(call_st, con_if.if_true);
|
|
|
|
continue;
|
|
|
|
case "else if":
|
|
|
|
var con_if = new __funcIf();
|
|
|
|
con_if.condition = evaluateFunctionTree(_cond);
|
|
|
|
ds_stack_top(call_st).addFunction(con_if);
|
|
|
|
ds_stack_push(call_st, con_if.if_true);
|
|
|
|
continue;
|
|
|
|
case "else":
|
|
|
|
ds_stack_push(call_st, con_if.if_false);
|
|
|
|
continue;
|
|
|
|
case "for":
|
|
|
|
var con_for = new __funcFor();
|
|
|
|
var cond = string_splice(_cond, ":");
|
|
|
|
if(array_length(cond) == 2) {
|
|
|
|
con_for.itr_array = true;
|
|
|
|
con_for.cond_iter = cond[0];
|
|
|
|
con_for.cond_arr = evaluateFunctionTree(cond[1]);
|
|
|
|
} else if(array_length(cond) == 3) {
|
|
|
|
con_for.itr_array = false;
|
|
|
|
con_for.cond_init = evaluateFunctionTree(cond[0]);
|
|
|
|
con_for.cond_iter = evaluateFunctionTree(cond[1]);
|
|
|
|
con_for.cond_term = evaluateFunctionTree(cond[2]);
|
|
|
|
}
|
|
|
|
ds_stack_top(call_st).addFunction(con_for);
|
|
|
|
ds_stack_push(call_st, con_for.action);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ds_stack_top(call_st).addFunction(evaluateFunctionTree(_fx));
|
|
|
|
}
|
|
|
|
|
|
|
|
var val = ds_stack_pop(call_st);
|
|
|
|
ds_stack_destroy(call_st);
|
|
|
|
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2023-06-20 19:43:19 +02:00
|
|
|
function evaluateFunctionTree(fx) {
|
|
|
|
static __BRACKETS = [ "(", ")", "[", "]" ];
|
|
|
|
|
2023-03-23 06:57:31 +01:00
|
|
|
var pres = global.EQUATION_PRES;
|
|
|
|
var vl = ds_stack_create();
|
|
|
|
var op = ds_stack_create();
|
2023-07-11 20:36:44 +02:00
|
|
|
var last_push = "";
|
2022-01-13 05:24:03 +01:00
|
|
|
|
|
|
|
var len = string_length(fx);
|
2023-03-23 06:57:31 +01:00
|
|
|
var l = 1;
|
2023-06-20 19:43:19 +02:00
|
|
|
var ch = "";
|
|
|
|
var cch = "";
|
|
|
|
var _ch = "";
|
|
|
|
var in_str = false;
|
2022-01-13 05:24:03 +01:00
|
|
|
|
2023-07-29 10:10:48 +02:00
|
|
|
printIf(global.LOG_EXPRESSION, $"===== Evaluating function: {fx} =====");
|
2023-07-12 16:28:32 +02:00
|
|
|
|
2022-01-13 05:24:03 +01:00
|
|
|
while(l <= len) {
|
|
|
|
ch = string_char_at(fx, l);
|
|
|
|
|
2023-07-12 21:00:05 +02:00
|
|
|
//print($"Analyzing {ch}");
|
|
|
|
|
2023-06-20 19:43:19 +02:00
|
|
|
if(ds_map_exists(pres, ch)) { //symbol is operator
|
2023-07-15 20:01:29 +02:00
|
|
|
last_push = "op";
|
|
|
|
|
2022-01-13 05:24:03 +01:00
|
|
|
if(ds_stack_empty(op)) ds_stack_push(op, ch);
|
|
|
|
else {
|
2023-07-11 20:36:44 +02:00
|
|
|
var _top = ds_stack_top(op);
|
|
|
|
if(_top == "(" || ds_map_exists(global.FUNCTIONS, _top) || pres[? ch] > pres[? _top]) {
|
|
|
|
ds_stack_push(op, ch);
|
|
|
|
} else {
|
2023-06-26 18:55:41 +02:00
|
|
|
if(ch == "-" && ds_map_exists(pres, _ch)) ch = "∸"; //unary negative
|
2023-05-22 20:31:55 +02:00
|
|
|
|
|
|
|
while(pres[? ch] <= pres[? ds_stack_top(op)] && !ds_stack_empty(op))
|
2023-06-20 19:43:19 +02:00
|
|
|
ds_stack_push(vl, buildFuncTree(ds_stack_pop(op), vl));
|
2022-01-13 05:24:03 +01:00
|
|
|
ds_stack_push(op, ch);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
l++;
|
|
|
|
} else if (ch == "(") {
|
2023-07-11 20:36:44 +02:00
|
|
|
if(last_push == "fn") ds_stack_push(op, [ "〚", ds_stack_size(vl) ]);
|
|
|
|
else ds_stack_push(op, ch);
|
|
|
|
last_push = "op";
|
2022-01-13 05:24:03 +01:00
|
|
|
l++;
|
|
|
|
} else if (ch == ")") {
|
2023-07-11 20:36:44 +02:00
|
|
|
while(!ds_stack_empty(op)) {
|
|
|
|
var _top = ds_stack_pop(op);
|
|
|
|
if(_top == "(") break;
|
|
|
|
if(is_array(_top) && _top[0] == "〚") {
|
|
|
|
var arr = [];
|
|
|
|
while(ds_stack_size(vl) > _top[1])
|
|
|
|
array_insert(arr, 0, ds_stack_pop(vl));
|
|
|
|
|
|
|
|
ds_stack_push(vl, new __funcTree(ds_stack_pop(op), arr));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ds_stack_push(vl, buildFuncTree(_top, vl));
|
2023-06-20 19:43:19 +02:00
|
|
|
}
|
2023-07-11 20:36:44 +02:00
|
|
|
|
|
|
|
last_push = "vl";
|
2023-06-20 19:43:19 +02:00
|
|
|
l++;
|
|
|
|
} else if (ch == "[") {
|
2023-07-15 20:01:29 +02:00
|
|
|
if(last_push == "vl") {
|
|
|
|
ds_stack_push(op, "@");
|
|
|
|
ds_stack_push(op, ch);
|
|
|
|
} else
|
|
|
|
ds_stack_push(op, [ "{", ds_stack_size(vl) ]);
|
|
|
|
|
2023-07-11 20:36:44 +02:00
|
|
|
last_push = "op";
|
2023-06-20 19:43:19 +02:00
|
|
|
l++;
|
|
|
|
} else if (ch == "]") {
|
2023-07-11 20:36:44 +02:00
|
|
|
while(!ds_stack_empty(op)) {
|
|
|
|
var _top = ds_stack_pop(op);
|
|
|
|
if(_top == "[") break;
|
|
|
|
if(is_array(_top) && _top[0] == "{") {
|
|
|
|
var arr = [];
|
|
|
|
while(ds_stack_size(vl) > _top[1])
|
|
|
|
array_insert(arr, 0, ds_stack_pop(vl));
|
2023-07-12 21:00:05 +02:00
|
|
|
ds_stack_push(vl, new __funcTree("【", arr));
|
2023-07-11 20:36:44 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ds_stack_push(vl, buildFuncTree(_top, vl));
|
|
|
|
}
|
|
|
|
|
|
|
|
last_push = "vl";
|
|
|
|
l++;
|
|
|
|
} else if (ch == ",") {
|
|
|
|
while(!ds_stack_empty(op)) {
|
|
|
|
var _top = ds_stack_top(op);
|
|
|
|
if(_top == "[" || _top == "(" || (is_array(_top) && _top[0] == "{")) break;
|
|
|
|
|
|
|
|
ds_stack_push(vl, buildFuncTree(_top, vl));
|
|
|
|
}
|
2023-06-20 19:43:19 +02:00
|
|
|
|
2023-07-11 20:36:44 +02:00
|
|
|
last_push = "vl";
|
2022-01-13 05:24:03 +01:00
|
|
|
l++;
|
|
|
|
} else {
|
|
|
|
var vsl = "";
|
|
|
|
|
|
|
|
while(l <= len) {
|
|
|
|
cch = string_char_at(fx, l);
|
2023-07-12 16:28:32 +02:00
|
|
|
if(ds_map_exists(pres, cch) || array_exists(__BRACKETS, cch)) break;
|
|
|
|
if(cch == ",") {
|
|
|
|
l++;
|
|
|
|
break;
|
|
|
|
}
|
2022-01-13 05:24:03 +01:00
|
|
|
|
|
|
|
vsl += cch;
|
|
|
|
l++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(vsl == "") continue;
|
|
|
|
|
2023-07-11 20:36:44 +02:00
|
|
|
if(ds_map_exists(global.FUNCTIONS, vsl)) { //function
|
2022-01-13 05:24:03 +01:00
|
|
|
ds_stack_push(op, vsl);
|
2023-07-11 20:36:44 +02:00
|
|
|
last_push = "fn";
|
2022-01-13 05:24:03 +01:00
|
|
|
} else {
|
2023-07-11 20:36:44 +02:00
|
|
|
vsl = string_trim(vsl);
|
2023-07-15 20:01:29 +02:00
|
|
|
|
2022-01-13 05:24:03 +01:00
|
|
|
switch(vsl) {
|
2023-06-20 19:43:19 +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;
|
2022-01-13 05:24:03 +01:00
|
|
|
}
|
2023-07-11 20:36:44 +02:00
|
|
|
|
|
|
|
last_push = "vl";
|
2022-01-13 05:24:03 +01:00
|
|
|
}
|
|
|
|
}
|
2023-05-22 20:31:55 +02:00
|
|
|
|
2023-07-12 21:00:05 +02:00
|
|
|
//print($"op: {ds_stack_size(op)}; vl: {ds_stack_size(vl)}");
|
|
|
|
|
2023-05-22 20:31:55 +02:00
|
|
|
_ch = ch;
|
2022-01-13 05:24:03 +01:00
|
|
|
}
|
|
|
|
|
2023-06-20 19:43:19 +02:00
|
|
|
while(!ds_stack_empty(op))
|
|
|
|
ds_stack_push(vl, buildFuncTree(ds_stack_pop(op), vl));
|
2022-01-13 05:24:03 +01:00
|
|
|
|
2023-07-12 21:00:05 +02:00
|
|
|
var tree = ds_stack_empty(vl)? noone : ds_stack_pop(vl);
|
|
|
|
|
|
|
|
ds_stack_destroy(op);
|
2023-06-20 19:43:19 +02:00
|
|
|
ds_stack_destroy(vl);
|
|
|
|
|
2023-07-11 20:36:44 +02:00
|
|
|
if(!is_struct(tree))
|
2023-06-20 19:43:19 +02:00
|
|
|
tree = new __funcTree("", tree);
|
|
|
|
|
2023-07-29 10:10:48 +02:00
|
|
|
printIf(global.LOG_EXPRESSION, tree);
|
|
|
|
printIf(global.LOG_EXPRESSION, "");
|
2023-07-11 20:36:44 +02:00
|
|
|
|
2023-06-20 19:43:19 +02:00
|
|
|
return tree;
|
2022-01-13 05:24:03 +01:00
|
|
|
}
|
|
|
|
|
2023-06-20 19:43:19 +02:00
|
|
|
function buildFuncTree(operator, vl) {
|
|
|
|
if(ds_stack_empty(vl)) return noone;
|
2023-06-13 21:24:05 +02:00
|
|
|
|
2023-07-12 16:28:32 +02:00
|
|
|
if(ds_map_exists(global.FUNCTIONS, operator)) {
|
|
|
|
if(ds_stack_empty(vl))
|
|
|
|
return noone;
|
|
|
|
|
|
|
|
var _v1 = ds_stack_pop(vl);
|
|
|
|
return new __funcTree(operator, _v1);
|
|
|
|
}
|
|
|
|
|
2023-06-20 19:43:19 +02:00
|
|
|
switch(operator) {
|
2023-06-26 18:55:41 +02:00
|
|
|
case "-": //deal with preceeding negative number -5
|
2023-06-20 19:43:19 +02:00
|
|
|
if(ds_stack_size(vl) >= 2) {
|
|
|
|
var _v1 = ds_stack_pop(vl);
|
|
|
|
var _v2 = ds_stack_pop(vl);
|
|
|
|
return new __funcTree("-", _v2, _v1);
|
|
|
|
} else
|
|
|
|
return new __funcTree("-", ds_stack_pop(vl), 0);
|
|
|
|
|
2023-07-11 20:36:44 +02:00
|
|
|
case "@":
|
|
|
|
var _v1 = ds_stack_pop(vl);
|
2023-07-12 21:00:05 +02:00
|
|
|
var _v2 = ds_stack_pop(vl);
|
|
|
|
return new __funcTree(operator, _v2, _v1);
|
2023-07-11 20:36:44 +02:00
|
|
|
|
2023-06-20 19:43:19 +02:00
|
|
|
case "+": //binary operators
|
2023-03-11 01:40:17 +01:00
|
|
|
case "*":
|
2023-06-13 21:24:05 +02:00
|
|
|
case "$":
|
2022-01-13 05:24:03 +01:00
|
|
|
case "/":
|
2023-07-12 21:00:05 +02:00
|
|
|
case "%":
|
2023-06-20 19:43:19 +02:00
|
|
|
|
2023-06-13 21:24:05 +02:00
|
|
|
case "|":
|
2023-06-20 19:43:19 +02:00
|
|
|
case "&":
|
2023-06-13 21:24:05 +02:00
|
|
|
case "^":
|
|
|
|
case "»":
|
|
|
|
case "«":
|
2023-06-20 19:43:19 +02:00
|
|
|
|
2023-06-13 21:24:05 +02:00
|
|
|
case "=":
|
2023-07-12 21:00:05 +02:00
|
|
|
case "⩵":
|
2023-06-13 21:24:05 +02:00
|
|
|
case "≠":
|
|
|
|
case "≤":
|
|
|
|
case "≥":
|
2023-06-20 19:43:19 +02:00
|
|
|
case "<":
|
|
|
|
case ">":
|
|
|
|
|
|
|
|
if(ds_stack_size(vl) >= 2) {
|
|
|
|
var _v1 = ds_stack_pop(vl);
|
|
|
|
var _v2 = ds_stack_pop(vl);
|
|
|
|
return new __funcTree(operator, _v2, _v1);
|
|
|
|
}
|
|
|
|
|
|
|
|
default: return new __funcTree(operator, ds_stack_pop(vl));
|
2022-01-13 05:24:03 +01:00
|
|
|
}
|
|
|
|
|
2023-06-20 19:43:19 +02:00
|
|
|
return noone;
|
|
|
|
}
|
|
|
|
|
|
|
|
function evaluateFunction(fx, params = {}) {
|
2023-06-23 15:39:24 +02:00
|
|
|
if(isNumber(fx)) return toNumber(fx);
|
2023-07-12 21:00:05 +02:00
|
|
|
return evaluateFunctionList(fx).eval(params);
|
2022-01-13 05:24:03 +01:00
|
|
|
}
|
|
|
|
#endregion
|