AnonSec Team
Server IP : 10.2.73.233  /  Your IP : 216.73.216.223
Web Server : Apache/2.4.59 (Debian)
System : Linux polon 4.19.0-27-amd64 #1 SMP Debian 4.19.316-1 (2024-06-25) x86_64
User : www-data ( 33)
PHP Version : 5.6.40-64+0~20230107.71+debian10~1.gbp673146
Disable Function : pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,
MySQL : ON  |  cURL : ON  |  WGET : ON  |  Perl : ON  |  Python : ON
Directory (0755) :  /home/ifk/web/prado4.3.2/I18N/core/

[  Home  ][  C0mmand  ][  Upload File  ]

Current File : /home/ifk/web/prado4.3.2/I18N/core/ChoiceFormat.php
<?php

/**
 * ChoiceFormat class file.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the BSD License.
 *
 * Copyright(c) 2004 by Qiang Xue. All rights reserved.
 *
 * To contact the author write to {@link mailto:qiang.xue@gmail.com Qiang Xue}
 * The latest version of PRADO can be obtained from:
 * {@link http://prado.sourceforge.net/}
 *
 * @author Wei Zhuo <weizhuo[at]gmail[dot]com>
 */

namespace Prado\I18N\core;

use Exception;

/**
 * ChoiceFormat class.
 *
 * ChoiceFormat converts between ranges of numeric values and string
 * names for those ranges.
 *
 * A ChoiceFormat splits the real number line -Inf to +Inf into two or
 * more contiguous ranges. Each range is mapped to a string.
 * ChoiceFormat is generally used in a MessageFormat for displaying
 * grammatically correct plurals such as "There are 2 files."
 *
 * <code>
 *	$string = '[0] are no files |[1] is one file |(1,Inf] are {number} files';
 *
 *  $formatter = new MessageFormat(...); //init for a source
 *	$translated = $formatter->format($string);
 *
 *  $choice = new ChoiceFormat();
 * 	echo $choice->format($translated, 0); //shows "are no files"
 * </code>
 *
 * The message/string choices are separated by the pipe "|" followed
 * by a set notation of the form
 *  # <tt>[1,2]</tt> -- accepts values between 1 and 2, inclusive.
 *  # <tt>(1,2)</tt> -- accepts values between 1 and 2, excluding 1 and 2.
 *  # <tt>{1,2,3,4}</tt> -- only values defined in the set are accepted.
 *  # <tt>[-Inf,0)</tt> -- accepts value greater or equal to negative infinity
 *                       and strictly less than 0
 * Any non-empty combinations of the delimiters of square and round brackets
 * are acceptable.
 *
 * Since version 3.1.2 the following set notation is also possible.
 *
 *  # <tt>{n: n % 10 > 1 && n % 10 < 5}</tt> --  matches numbers like 2, 3, 4, 22, 23, 24
 *
 * Where set is defined by the expression after <tt>n:</tt>. In particular, the expression
 * accepts the following mathematical/logical operators to form a set of logical conditions
 * on the value given by <tt>n</tt>:
 *   # <tt>&lt;</tt> -- less than.
 *   # <tt>&lt;=</tt> -- less than equals.
 *   # <tt>&gt;</tt> -- greater than.
 *   # <tt>&gt=</tt> -- greater than equals.
 *   # <tt>==</tt> -- of equal value.
 *   # <tt>%</tt> -- modulo, e.g., 1 % 10 equals 1, 11 % 10 equals 1.
 *   # <tt>-</tt> -- minus, negative.
 *   # <tt>+</tt> -- addition.
 *   # <tt>&amp;</tt> -- conditional AND.
 *   # <tt>&amp;&amp;</tt> -- condition AND with short circuit.
 *   # <tt>|</tt> -- conditional OR.
 *   # <tt>||</tt> -- conditional OR with short circuit.
 *   # <tt>!</tt> -- negation.
 *
 * Additional round brackets can also be used to perform grouping.
 *
 * @author Xiang Wei Zhuo <weizhuo[at]gmail[dot]com>
 */
class ChoiceFormat
{
	/**
	 * The pattern to validate a set notation
	 * @var string
	 */
	protected $validate = '/[\(\[\{]|[-Inf\d:\s]+|,|[\+Inf\d\s:\?\-=!><%\|&\(\)]+|[\)\]\}]/ms';

	/**
	 * The pattern to parse the formatting string.
	 * @var string
	 */
	protected $parse = '/\s*\|?([\(\[\{]([-Inf\d:\s]+,?[\+Inf\d\s:\?\-=!><%\|&\(\)]*)+[\)\]\}])\s*/';

	/**
	 * The value for positive infinity.
	 * @var float
	 */
	protected $inf;


	/**
	 * Constructor.
	 */
	public function __construct()
	{
		$this->inf = -log(0);
	}


	/**
	 * Determine if the given number belongs to a given set
	 * @param float $number the number to test.
	 * @param string $set the set, in set notation.
	 * @return bool true if number is in the set, false otherwise.
	 */
	public function isValid($number, $set)
	{
		$n = preg_match_all($this->validate, $set, $matches, PREG_SET_ORDER);

		if ($n < 3) {
			throw new Exception("Invalid set \"{$set}\"");
		}

		if (preg_match('/\{\s*n:([^\}]+)\}/', $set, $def)) {
			return $this->isValidSetNotation($number, $def[1]);
		}

		$leftBracket = $matches[0][0];
		$rightBracket = $matches[$n - 1][0];

		$i = 0;
		$elements = [];
		foreach ($matches as $match) {
			$string = $match[0];
			if ($i != 0 && $i != $n - 1 && $string !== ',') {
				if ($string == '-Inf') {
					$elements[] = -1 * $this->inf;
				} elseif ($string == '+Inf' || $string == 'Inf') {
					$elements[] = $this->inf;
				} else {
					$elements[] = (float) $string;
				}
			}
			$i++;
		}
		$total = count($elements);
		$number = (float) $number;

		if ($leftBracket == '{' && $rightBracket == '}') {
			return in_array($number, $elements);
		}

		$left = false;
		if ($leftBracket == '[') {
			$left = $number >= $elements[0];
		} elseif ($leftBracket == '(') {
			$left = $number > $elements[0];
		}

		$right = false;
		if ($rightBracket == ']') {
			$right = $number <= $elements[$total - 1];
		} elseif ($rightBracket == ')') {
			$right = $number < $elements[$total - 1];
		}

		if ($left && $right) {
			return true;
		}

		return false;
	}

	protected function isValidSetNotation($number, $set)
	{
		try {
			return eval('return ' . str_replace('n', '$number', $set) . ';');
		} catch (Exception $e) {
			return false;
		}
	}

	/**
	 * Parse a choice string and get a list of sets and a list of strings
	 * corresponding to the sets.
	 * @param string $string the string containing the choices
	 * @return array array($sets, $strings)
	 */
	public function parse($string)
	{
		$n = preg_match_all($this->parse, $string, $matches, PREG_OFFSET_CAPTURE);
		$sets = [];
		foreach ($matches[1] as $match) {
			$sets[] = $match[0];
		}
		$offset = $matches[0];
		$strings = [];
		for ($i = 0; $i < $n; $i++) {
			$len = strlen($offset[$i][0]);
			$begin = $i == 0 ? $len : $offset[$i][1] + $len;
			$end = $i == $n - 1 ? strlen($string) : $offset[$i + 1][1];
			$strings[] = substr($string, $begin, $end - $begin);
		}
		return [$sets, $strings];
	}

	/**
	 * For the choice string, and a number, find and return the
	 * string that satisfied the set within the choices.
	 * @param string $string the choices string.
	 * @param float $number the number to test.
	 * @return string the choosen string.
	 */
	public function format($string, $number)
	{
		[$sets, $strings] = $this->parse($string);
		$total = count($sets);
		for ($i = 0; $i < $total; $i++) {
			if ($this->isValid($number, $sets[$i])) {
				return $strings[$i];
			}
		}
		return false;
	}
}

AnonSec - 2021