X-Git-Url: https://git.rapsys.eu/packbundle/blobdiff_plain/8989e866f7ba949de31856943049d6f908fdc46d..569db7a24c07ddc92ca69be46ae6a90d3c0d04a8:/Util/IntlUtil.php

diff --git a/Util/IntlUtil.php b/Util/IntlUtil.php
index 7556838..80b9530 100644
--- a/Util/IntlUtil.php
+++ b/Util/IntlUtil.php
@@ -15,99 +15,153 @@ use Twig\Error\SyntaxError;
 use Twig\Environment;
 
 /**
- * Helps manage intl conversions
+ * Manages intl conversions
  */
 class IntlUtil {
 	/**
-	 * Construct intl util
+	 * Format currency
 	 */
-	public function __construct() {
+	public function currency(int|float $number, string $currency, ?string $locale = null) {
+		//Get formatter
+		$formatter = $this->getNumberFormatter($locale, 'currency');
+
+		//Return formatted currency
+		return $formatter->formatCurrency($number, $currency);
 	}
 
-	public function date(Environment $env, $date, $dateFormat = 'medium', $timeFormat = 'medium', $locale = null, $timezone = null, $format = null, $calendar = 'gregorian') {
+	/**
+	 * Format date
+	 */
+	public function date(Environment $env, \DateTime $date, string $dateFormat = 'medium', string $timeFormat = 'medium', ?string $locale = null, \IntlTimeZone|\DateTimeZone|string|null $timezone = null, ?string $calendar = null, ?string $pattern = null) {
+		//Get converted date
 		$date = twig_date_converter($env, $date, $timezone);
 
-		$formatValues = array(
-			'none' => IntlDateFormatter::NONE,
-			'short' => IntlDateFormatter::SHORT,
-			'medium' => IntlDateFormatter::MEDIUM,
-			'long' => IntlDateFormatter::LONG,
-			'full' => IntlDateFormatter::FULL,
-		);
-
-		$formatter = IntlDateFormatter::create(
+		//Set date and time formatters
+		$formatters = [
+			'none' => \IntlDateFormatter::NONE,
+			'short' => \IntlDateFormatter::SHORT,
+			'medium' => \IntlDateFormatter::MEDIUM,
+			'long' => \IntlDateFormatter::LONG,
+			'full' => \IntlDateFormatter::FULL,
+		];
+
+		//Get formatter
+		$formatter = \IntlDateFormatter::create(
 			$locale,
-			$formatValues[$dateFormat],
-			$formatValues[$timeFormat],
-			IntlTimeZone::createTimeZone($date->getTimezone()->getName()),
-			'gregorian' === $calendar ? IntlDateFormatter::GREGORIAN : IntlDateFormatter::TRADITIONAL,
-			$format
+			$formatters[$dateFormat],
+			$formatters[$timeFormat],
+			\IntlTimeZone::createTimeZone($date->getTimezone()->getName()),
+			'traditional' === $calendar ? \IntlDateFormatter::TRADITIONAL : \IntlDateFormatter::GREGORIAN,
+			$pattern
 		);
 
+		//Return formatted date
 		return $formatter->format($date->getTimestamp());
 	}
 
-	public function number($number, $style = 'decimal', $type = 'default', $locale = null) {
-		static $typeValues = array(
-			'default' => NumberFormatter::TYPE_DEFAULT,
-			'int32' => NumberFormatter::TYPE_INT32,
-			'int64' => NumberFormatter::TYPE_INT64,
-			'double' => NumberFormatter::TYPE_DOUBLE,
-			'currency' => NumberFormatter::TYPE_CURRENCY,
-		);
+	/**
+	 * Compute eastern for selected year
+	 *
+	 * @param string $year The eastern year
+	 *
+	 * @return DateTime The eastern date
+	 */
+	public function getEastern(string $year): \DateTime {
+		//Set static results
+		static $results = [];
+
+		//Check if already computed
+		if (isset($results[$year])) {
+			//Return computed eastern
+			return $results[$year];
+		}
 
-		$formatter = $this->getNumberFormatter($locale, $style);
+		$d = (19 * ($year % 19) + 24) % 30;
 
-		if (!isset($typeValues[$type])) {
-			throw new SyntaxError(sprintf('The type "%s" does not exist. Known types are: "%s"', $type, implode('", "', array_keys($typeValues))));
-		}
+		$e = (2 * ($year % 4) + 4 * ($year % 7) + 6 * $d + 5) % 7;
 
-		return $formatter->format($number, $typeValues[$type]);
-	}
+		$day = 22 + $d + $e;
 
-	public function currency($number, $currency = null, $locale = null) {
-		$formatter = $this->getNumberFormatter($locale, 'currency');
+		$month = 3;
 
-		return $formatter->formatCurrency($number, $currency);
+		if ($day > 31) {
+			$day = $d + $e - 9;
+			$month = 4;
+		} elseif ($d == 29 && $e == 6) {
+			$day = 10;
+			$month = 4;
+		} elseif ($d == 28 && $e == 6) {
+			$day = 18;
+			$month = 4;
+		}
+
+		//Store eastern in data
+		return ($results[$year] = new \DateTime(sprintf('%04d-%02d-%02d', $year, $month, $day)));
 	}
 
 	/**
-	 * Gets a number formatter instance according to given locale and formatter.
+	 * Gets number formatter instance matching locale and style.
 	 *
-	 * @param string $locale Locale in which the number would be formatted
-	 * @param int    $style  Style of the formatting
+	 * @param ?string $locale Locale in which the number would be formatted
+	 * @param string $style Style of the formatting
 	 *
 	 * @return NumberFormatter A NumberFormatter instance
 	 */
-	protected function getNumberFormatter($locale, $style): NumberFormatter {
-		static $formatter, $currentStyle;
+	protected function getNumberFormatter(?string $locale, string $style): \NumberFormatter {
+		//Set static formatters
+		static $formatters = [];
 
-		$locale = null !== $locale ? $locale : Locale::getDefault();
+		//Set locale
+		$locale = null !== $locale ? $locale : \Locale::getDefault();
 
-		if ($formatter && $formatter->getLocale() === $locale && $currentStyle === $style) {
-			// Return same instance of NumberFormatter if parameters are the same
-			// to those in previous call
-			return $formatter;
+		//With existing formatter
+		if (isset($formatters[$locale][$style])) {
+			//Return the instance from previous call
+			return $formatters[$locale][$style];
 		}
 
-		static $styleValues = array(
-			'decimal' => NumberFormatter::DECIMAL,
-			'currency' => NumberFormatter::CURRENCY,
-			'percent' => NumberFormatter::PERCENT,
-			'scientific' => NumberFormatter::SCIENTIFIC,
-			'spellout' => NumberFormatter::SPELLOUT,
-			'ordinal' => NumberFormatter::ORDINAL,
-			'duration' => NumberFormatter::DURATION,
-		);
-
-		if (!isset($styleValues[$style])) {
+		//Set styles
+		static $styles = [
+			'decimal' => \NumberFormatter::DECIMAL,
+			'currency' => \NumberFormatter::CURRENCY,
+			'percent' => \NumberFormatter::PERCENT,
+			'scientific' => \NumberFormatter::SCIENTIFIC,
+			'spellout' => \NumberFormatter::SPELLOUT,
+			'ordinal' => \NumberFormatter::ORDINAL,
+			'duration' => \NumberFormatter::DURATION
+		];
+
+		//Without styles
+		if (!isset($styles[$style])) {
 			throw new SyntaxError(sprintf('The style "%s" does not exist. Known styles are: "%s"', $style, implode('", "', array_keys($styleValues))));
 		}
 
-		$currentStyle = $style;
+		//Return number formatter
+		return ($formatters[$locale][$style] = \NumberFormatter::create($locale, $styles[$style]));
+	}
 
-		$formatter = NumberFormatter::create($locale, $styleValues[$style]);
+	/**
+	 * Format number
+	 */
+	public function number(int|float $number, $style = 'decimal', $type = 'default', ?string $locale = null) {
+		//Set types
+		static $types = [
+			'default' => \NumberFormatter::TYPE_DEFAULT,
+			'int32' => \NumberFormatter::TYPE_INT32,
+			'int64' => \NumberFormatter::TYPE_INT64,
+			'double' => \NumberFormatter::TYPE_DOUBLE,
+			'currency' => \NumberFormatter::TYPE_CURRENCY
+		];
+
+		//Get formatter
+		$formatter = $this->getNumberFormatter($locale, $style);
+
+		//Without type
+		if (!isset($types[$type])) {
+			throw new SyntaxError(sprintf('The type "%s" does not exist. Known types are: "%s"', $type, implode('", "', array_keys($types))));
+		}
 
-		return $formatter;
+		//Return formatted number
+		return $formatter->format($number, $types[$type]);
 	}
 }