<?php
namespace nur\b\date;

use nur\b\IllegalAccessException;
use nur\b\ValueException;
use nur\b\values\IValueState;

/**
 * Class Time: une heure allant de 0h à 24h inclus.
 *
 * la seule utilisation autorisée de "24h" est comme borne supérieure pour une
 * plage horaire.
 *
 * Cet objet est considéré comme immutable. les seules méthodes publiques qui
 * permettent de le modifier en place sont {@link Time::wrapStart()},
 * {@link Time::wrapEnd()} et {@link Time::set()}
 */
class Time implements IValueState {
  const UNIT_HOURS = 3600;
  const UNIT_MINUTES = 60;
  const UNIT_SECONDS = 1;

  /** @var string format normalisé pour l'analyse */
  const PATTERN = '/^(\d+):(\d{2}):(\d{2})$/';
  /** @var string format par défaut pour l'affichage */
  const FORMAT = "HH:MM:SS";

  static function parse_time($time, int $unit=self::UNIT_SECONDS): int {
    if ($time instanceof Time) {
      $seconds = $time->getSeconds();
    } elseif ($time === null || is_int($time)) {
      $seconds = $time !== null? $time: 0;
      if ($unit !== null) $seconds *= $unit;
    } elseif (is_array($time)) {
      [$h, $m, $s] = $time;
      $seconds = $h * 3600 + $m * 60 + $s;
    } elseif (is_string($time) && preg_match(self::PATTERN, $time, $ms)) {
      $seconds = intval($ms[1]) * 3600 + intval($ms[2]) * 60 + intval($ms[3]);
    } else {
      throw ValueException::invalid_value($time, "time");
    }
    return $seconds;
  }

  /**
   * @return int la valeur de l'unité en secondes, pour le constructeur et les
   * méthodes {@link newu()}, {@link addu()} et {@link subu()}
   */
  function UNIT(): int {
    return static::UNIT;
  } const UNIT = self::UNIT_SECONDS;

  /** @return bool s'il faut garder les heures dans la plage [0, 24h] */
  protected function WRAP(): bool {
    return static::WRAP;
  } const WRAP = true;

  function __construct($time=null) {
    if ($time !== false) {
      $this->setSeconds(self::parse_time($time, $this->UNIT()));
    }
  }

  /** cet objet est-il l'instance nulle? */
  function isNull(): bool { return false; }

  /** cet objet est-il l'heure indéfinie? */
  function isUndef(): bool { return false; }

  /**
   * wrapper l'heure pour la garder dans la plage [0h, 24h[ ce qui la rend
   * propice à l'utilisation comme borne inférieure d'une période
   */
  function wrapStart(): self {
    $seconds = $this->seconds;
    while ($seconds < 0) $seconds += 86400;
    if ($seconds >= 86400) $seconds %= 86400;
    $this->seconds = $seconds;
    return $this;
  }

  /**
   * wrapper l'heure pour la garder dans la plage [0h, 24h] ce qui la rend
   * propice à l'utilisation comme borne supérieure d'une période
   */
  function wrapEnd(): self {
    $seconds = $this->seconds;
    while ($seconds < 0) $seconds += 86400;
    if ($seconds > 86400) $seconds %= 86400;
    $this->seconds = $seconds;
    return $this;
  }

  protected function afterUpdate(): void {
    if ($this->WRAP()) $this->wrapEnd();
  }

  /** @var int */
  protected $seconds;

  /** @return int le nombre de secondes */
  function getSeconds(): int {
    return $this->seconds;
  }

  /**
   * mettre à jour cet objet avec le nombre de secondes spécifié
   *
   * @return int le nombre de seconde effectif, après correction
   */
  protected function setSeconds(?int $seconds): int {
    if ($seconds === null) {
      $hms = date('His');
      $seconds = intval(substr($hms, 0, 2)) * 3600
        + intval(substr($hms, 2, 2)) * 60
        + intval(substr($hms, 4, 2));
    }
    $adjust = $seconds % $this->UNIT();
    if ($seconds < 0) $adjust = -$adjust;
    $seconds -= $adjust;
    $this->seconds = $seconds;
    $this->afterUpdate();
    return $this->seconds;
  }

  /** formatter cette heure pour affichage */
  function format(?string $format=null): string {
    if ($format === null) $format = static::FORMAT;
    $v = $this->seconds;
    $h = intdiv($v, 3600); $v = $v % 3600;
    $m = intdiv($v, 60);
    $s = $v % 60;
    $searches = [
      "HH", "H",
      "MM", "M",
      "SS", "S",
    ];
    $replaces = [
      sprintf("%02u", $h), strval($h),
      sprintf("%02u", $m), strval($m),
      sprintf("%02u", $s), strval($s),
    ];
    return str_replace($searches, $replaces, $format);
  }

  function __toString(): string {
    return $this->format();
  }

  /** @return int le nombre d'unités */
  function getu(): int {
    return intdiv($this->seconds, $this->UNIT());
  }

  /** créer une nouvelle heure avec le nombre d'unités spécifiées */
  function newu(int $count): self {
    if ($this->isNull()) return self::null();
    if ($this->isUndef()) return self::undef();
    $clone = clone $this;
    $clone->setSeconds($count * $this->UNIT());
    return $clone;
  }

  /** créer une nouvelle heure en ajoutant à cette heure le nombre d'unités spécifiées */
  function addu(int $count): self {
    if ($this->isNull()) return self::null();
    if ($this->isUndef()) return self::undef();
    $clone = clone $this;
    $clone->setSeconds($clone->getSeconds() + $count * $this->UNIT());
    return $clone;
  }

  /** créer une nouvelle heure en soustrayant à cette heure le nombre d'unités spécifiées */
  function subu(int $count): self {
    if ($this->isNull()) return self::null();
    if ($this->isUndef()) return self::undef();
    $clone = clone $this;
    $clone->setSeconds($clone->getSeconds() - $count * $this->UNIT());
    return $clone;
  }

  /** mettre à jour cette heure avec le nombre de secondes d'une autre instance */
  function set(Time $time): self {
    $this->setSeconds($time->getSeconds());
    return $this;
  }

  /** créer une nouvelle heure copie de l'heure spécifiée */
  function new(Time $time): self {
    if ($this->isNull()) return self::null();
    if ($this->isUndef() || $time->isUndef()) return self::undef();
    $clone = clone $this;
    $clone->setSeconds($time->getSeconds());
    return $clone;
  }

  /** créer une nouvelle heure en ajoutant à cette heure l'heure spécifiée */
  function add(Time $time): self {
    if ($this->isNull()) return self::null();
    if ($this->isUndef() || $time->isUndef()) return self::undef();
    $clone = clone $this;
    $clone->setSeconds($clone->getSeconds() + $time->getSeconds());
    return $clone;
  }

  /** créer une nouvelle heure en soustrayant à cette heure l'heure spécifiée */
  function sub(Time $time): self {
    if ($this->isNull()) return self::null();
    if ($this->isUndef() || $time->isUndef()) return self::undef();
    $clone = clone $this;
    $clone->setSeconds($clone->getSeconds() - $time->getSeconds());
    return $clone;
  }

  /**
   * comparer avec l'heure spécifiée. retourner une valeur négative, égale à
   * zéro ou positive suivant le résultat de la comparaison
   */
  function cmp(Time $time): int {
    return $this->seconds - $time->getSeconds();
  }

  /** tester si cette heure est avant l'heure spécifiée */
  function before(Time $other): bool {
    return $this->seconds <= $other->getSeconds();
  }

  /** tester si cette heure est après l'heure spécifiée */
  function after(Time $other): bool {
    return $this->seconds >= $other->getSeconds();
  }

  #############################################################################

  private static $null;

  /**
   * @return Time une instance immutable représentant la valeur nulle.
   *
   * En général, on utilise directement null pour indiquer qu'il n'y a pas de
   * valeur, mais cette instance peut etre utilisée dans les contextes où null
   * n'est pas autorisé
   */
  static final function null(): self {
    if (self::$null === null) {
      self::$null = new class extends Time {
        function __construct() {
          parent::__construct(false);
          $this->seconds = 0;
        }
        function isNull(): bool { return true; }
        function wrapStart(): Time {
          throw IllegalAccessException::immutable_object();
        }
        function wrapEnd(): Time {
          throw IllegalAccessException::immutable_object();
        }
        function setSeconds(?int $seconds): int {
          throw IllegalAccessException::immutable_object();
        }
        function __toString(): string {
          return "";
        }
      };
    }
    return self::$null;
  }

  private static $undef;

  /**
   * @return Time une instance immutable représentant une heure non définie,
   * (c'est à dire qu'on ne sait pas de quelle heure il s'agit). Cette
   * instance peut etre utilisée s'il faut pouvoir faire la différence entre
   * pas de valeur (null) et une valeur non définie (false)
   */
  static final function undef(): self {
    if (self::$undef === null) {
      self::$undef = new class extends Time {
        function __construct() {
          parent::__construct(false);
          $this->seconds = 0;
        }
        function isUndef(): bool { return true; }
        function wrapStart(): Time {
          throw IllegalAccessException::immutable_object();
        }
        function wrapEnd(): Time {
          throw IllegalAccessException::immutable_object();
        }
        function setSeconds(?int $seconds): int {
          throw IllegalAccessException::immutable_object();
        }
        function __toString(): string {
          return "";
        }
      };
    }
    return self::$undef;
  }
}