19 declare(strict_types=1);
47 $math_tokens = $this->tokenizer->tokenizeMath($calculated_token->getValue());
49 $substituted = $this->substitution->substituteFieldValues($math_tokens);
50 [$result, $last_operator] = $this->
parseMath($substituted);
52 $from_function = $calculated_token->getFromFunction();
70 foreach ($math_tokens as $math_token) {
72 $field = $this->substitution->getFieldFromPlaceholder($math_token->getValue());
80 $tokens = $this->tokenizer->tokenize($original_token->
getValue());
81 foreach ($tokens as
$token) {
83 if (preg_match_all(
'/\[\[(.*?)\]\]/', $token->getValue(), $matches)) {
84 foreach ($matches[1] as $match) {
85 $field = $this->substitution->getFieldFromPlaceholder($match);
99 static fn(
Operators $operator):
string => $operator->value,
103 $stack =
new Stack();
104 $precedences =
new Stack();
106 foreach ($tokens as
$token) {
107 if (is_string($token)) {
112 $token = $token->getValue() ===
'' ?
'0' : $token->getValue();
114 if (is_numeric($token) || $token ===
'(') {
115 $stack->push($token);
116 if ($token ===
'(') {
119 } elseif (in_array($token, $operators)) {
121 $new_precedence = $last_operator->getPrecedence();
122 if ($new_precedence > $precedence || $in_bracket) {
124 $stack->push($token);
125 $precedences->push($new_precedence);
126 $precedence = $new_precedence;
129 while ($new_precedence <= $precedence && $stack->count() > 1) {
130 $right = (float) $stack->pop();
131 $operator = $stack->pop();
132 $left = (float) $stack->pop();
133 $result = $this->
calculate($operator, $left, $right);
134 $stack->push($result);
135 $precedence = $precedences->pop();
137 $stack->push($token);
138 $precedence = $new_precedence;
139 $precedences->push($new_precedence);
141 } elseif ($token ===
')') {
144 $elem = $stack->pop();
145 while ($elem !==
'(' && !$stack->isEmpty()) {
147 $elem = $stack->pop();
150 $stack->push($this->
parseMath(array_reverse($_tokens)));
153 throw new \ilException(
"Unrecognized token '$token'");
157 if ($stack->count() === 1) {
158 $result = $stack->pop();
160 $value = (ctype_digit((
string) $result)) ? $result : number_format($result, 2,
'.',
"'");
161 return [$value, $last_operator ??
null];
164 while ($stack->count() >= 2) {
165 $right = $stack->pop();
166 $operator = $stack->pop();
167 $left = $stack->count() ? $stack->pop() : 0;
168 $stack->push($this->
calculate($operator, $left, $right));
170 return [$stack->pop(), $last_operator];
177 static fn(
Functions $function):
string => $function->value,
181 foreach ($functions as $function) {
182 $pattern .=
"($function)\\(([^)]*)\\)|";
184 if (!preg_match_all(rtrim($pattern,
'|') .
'#', $token->
getValue(), $result)) {
190 foreach ($result[0] as $k => $to_replace) {
192 $function = $function_args[
'function'];
193 $args = $this->substitution->substituteFieldValues($function_args[
'args']);
194 $token_value = str_replace($to_replace, (
string) $this->
calculateFunction($function, $args), $token_value);
199 Functions::tryFrom($function ??
null)
212 for ($i = 1; $i < count($data); $i++) {
214 if ($_data[$index]) {
215 $function = $_data[$index];
216 $args = explode(
';', $data[$i + 1][$index]);
217 $return[
'function'] = $function;
218 $return[
'args'] = $args;
234 $count = count($args);
235 $array_sum = array_sum($args);
237 return ($count > 0) ? $array_sum / $count : 0;
239 return array_sum($args);
245 throw new ilException(
"Unrecognized function '$function'");
256 protected function calculate(
string $operator, $left, $right)
260 $result = $left + $right;
263 $result = $left - $right;
266 $result = $left * $right;
269 $result = ($right == 0) ? 0 : $left / $right;
272 $result = pow($left, $right);
275 throw new \ilException(
"Unrecognized operator '$operator'");
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
const INPUTFORMAT_DATETIME