1 gettext.inc _locale_import_parse_arithmetic($string)

Parses and sanitizes an arithmetic formula into a PHP expression.

While parsing, we ensure, that the operators have the right precedence and associativity.

Parameters

$string: A string containing the arithmetic formula.

Return value

The PHP version of the formula.:

Related topics

File

core/includes/gettext.inc, line 656
Gettext parsing and generating API.

Code

function _locale_import_parse_arithmetic($string) {
  // Operator precedence table.
  $precedence = array("(" => -1, ")" => -1, "?" => 1, ":" => 1, "||" => 3, "&&" => 4, "==" => 5, "!=" => 5, "<" => 6, ">" => 6, "<=" => 6, ">=" => 6, "+" => 7, "-" => 7, "*" => 8, "/" => 8, "%" => 8);
  // Right associativity.
  $right_associativity = array("?" => 1, ":" => 1);

  $tokens = _locale_import_tokenize_formula($string);

  // Parse by converting into infix notation then back into postfix.
  // Operator stack - holds math operators and symbols.
  $operator_stack = array();
  // Element Stack - holds data to be operated on.
  $element_stack = array();

  foreach ($tokens as $token) {
    $current_token = $token;

    // Numbers and the $n variable are pushed into $element_stack.
    if (is_numeric($token)) {
      $element_stack[] = $current_token;
    }
    elseif ($current_token == "n") {
      $element_stack[] = '$n';
    }
    elseif ($current_token == "(") {
      $operator_stack[] = $current_token;
    }
    elseif ($current_token == ")") {
      $to_pop = array_pop($operator_stack);
      while (isset($to_pop) && ($to_pop != "(")) {
        $element_stack[] = $to_pop;
        $to_pop = array_pop($operator_stack);
      }
    }
    elseif (!empty($precedence[$current_token])) {
      // If it's an operator, then pop from $operator_stack into $element_stack
      // until the precedence in $operator_stack is less than current, then push
      // into $operator_stack.
      $to_pop = array_pop($operator_stack);
      while (isset($to_pop) && ($precedence[$to_pop] >= $precedence[$current_token]) && !(($precedence[$to_pop] == $precedence[$current_token]) && !empty($right_associativity[$to_pop]) && !empty($right_associativity[$current_token]))) {
        $element_stack[] = $to_pop;
        $to_pop = array_pop($operator_stack);
      }
      if ($to_pop) {
        // Return element to top.
        $operator_stack[] = $to_pop;
      }
      // Parentheses are not needed.
      $operator_stack[] = $current_token;
    }
    else {
      return FALSE;
    }
  }

  // Flush operator stack.
  $to_pop = array_pop($operator_stack);
  while ($to_pop != NULL) {
    $element_stack[] = $to_pop;
    $to_pop = array_pop($operator_stack);
  }

  // Now extract formula from stack.
  $previous_size = count($element_stack) + 1;
  while (count($element_stack) < $previous_size) {
    $previous_size = count($element_stack);
    for ($i = 2; $i < count($element_stack); $i++) {
      $op = $element_stack[$i];
      if (!empty($precedence[$op])) {
        if ($op == ":") {
          $f = $element_stack[$i - 2] . "):" . $element_stack[$i - 1] . ")";
        }
        elseif ($op == "?") {
          $f = "(" . $element_stack[$i - 2] . "?(" . $element_stack[$i - 1];
        }
        else {
          $f = "(" . $element_stack[$i - 2] . $op . $element_stack[$i - 1] . ")";
        }
        array_splice($element_stack, $i - 2, 3, $f);
        break;
      }
    }
  }

  // If only one element is left, the number of operators is appropriate.
  if (count($element_stack) == 1) {
    return $element_stack[0];
  }
  else {
    return FALSE;
  }
}