Этот пакет сделан, чтобы сделать работу с датами более удобной.
- Позволяет передавать в конструкторы дат и временных зон строки без необходимости создавать объект ради объекта
- В нем можно посмотреть, как сделать грамотное наследование имеющищегося в коробке PHP ужаса, чтобы этим управлять или добавить своё.
composer require gzhegow/calendar;
<?php
require_once __DIR__ . '/vendor/autoload.php';
// > настраиваем PHP
ini_set('memory_limit', '32M');
// > настраиваем обработку ошибок
(new \Gzhegow\Lib\Exception\ErrorHandler())
->useErrorReporting()
->useErrorHandler()
->useExceptionHandler()
;
// > добавляем несколько функция для тестирования
function _debug(...$values) : string
{
$lines = [];
foreach ( $values as $value ) {
$lines[] = \Gzhegow\Lib\Lib::debug()->type($value);
}
$ret = implode(' | ', $lines) . PHP_EOL;
echo $ret;
return $ret;
}
function _dump(...$values) : string
{
$lines = [];
foreach ( $values as $value ) {
$lines[] = \Gzhegow\Lib\Lib::debug()->value($value);
}
$ret = implode(' | ', $lines) . PHP_EOL;
echo $ret;
return $ret;
}
function _dump_array($value, int $maxLevel = null, bool $multiline = false) : string
{
$content = $multiline
? \Gzhegow\Lib\Lib::debug()->array_multiline($value, $maxLevel)
: \Gzhegow\Lib\Lib::debug()->array($value, $maxLevel);
$ret = $content . PHP_EOL;
echo $ret;
return $ret;
}
function _assert_output(
\Closure $fn, string $expect = null
) : void
{
$trace = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS, 1);
\Gzhegow\Lib\Lib::assert()->output($trace, $fn, $expect);
}
function _assert_microtime(
\Closure $fn, float $expectMax = null, float $expectMin = null
) : void
{
$trace = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS, 1);
\Gzhegow\Lib\Lib::assert()->microtime($trace, $fn, $expectMax, $expectMin);
}
// >>> ЗАПУСКАЕМ!
// > сначала всегда фабрика
$factory = new \Gzhegow\Calendar\CalendarFactory();
// > создаем конфигурацию
$config = new \Gzhegow\Calendar\CalendarConfig();
$config->configure(function (\Gzhegow\Calendar\CalendarConfig $config) {
// > можно указать форматы для разбора даты, если таковые не переданы прямо в функцию
$config->parser->parseDateTimeFormatsDefault = [
\Gzhegow\Calendar\Calendar::FORMAT_SQL,
\Gzhegow\Calendar\Calendar::FORMAT_JAVASCRIPT,
];
// > можно указать форматы для разбора интервалов, если таковые не переданы прямо в функцию
$config->parser->parseDateIntervalFormatsDefault = [
\Gzhegow\Calendar\Calendar::FORMAT_SQL_TIME,
];
// > можно указать "дату по-умолчанию", которая будет создана, если не передать аргументов вовсе
$config->manager->dateTimeDefault = '1970-01-01 midnight';
// > можно указать "временную зону по-умолчанию", которая будет установлена, если при разборе даты не удалось определить временную зону
$config->manager->dateTimeZoneDefault = 'Europe/Minsk';
// > можно указать "интервал по-умолчанию", который будет создаваться, если не передать аргументов
$config->manager->dateIntervalDefault = 'P0D';
// > использовать ли INTL, если такое расширение подключено (в целях тестирования установлено в FALSE)
$config->formatter->useIntl = false;
});
// > можно изменить классы дат на свои собственные реализации
$type = new \Gzhegow\Calendar\Type\CalendarType();
// > создаем парсер
$parser = new \Gzhegow\Calendar\Parser\CalendarParser($factory, $config->parser);
// > создаем менеджер
$manager = new \Gzhegow\Calendar\Manager\CalendarManager($factory, $parser, $config->manager);
// > создаем форматтер
$formatter = new \Gzhegow\Calendar\Formatter\CalendarFormatter($factory, $config->formatter);
// > создаем фасад
$calendar = new \Gzhegow\Calendar\CalendarFacade(
$factory,
$type,
$parser,
$manager,
$formatter
);
// > сохраняем фасад статически (чтобы вызывать без привязки к контейнеру)
\Gzhegow\Calendar\Calendar::setFacade($calendar);
// > TEST
// > создаем дату, временную зону и интервал
$fn = function () use ($calendar) {
_dump('TEST 1');
echo PHP_EOL;
$result = $calendar->dateTime($datetime = '', $timezone = null);
_dump($result, json_encode($result));
$result = $calendar->dateTime($datetime = '1970-01-01 00:00:00', $timezone = null);
_dump($result, json_encode($result));
$result = $calendar->dateTimeImmutable($datetime = '', $timezone = null);
_dump($result, json_encode($result));
$result = $calendar->dateTimeImmutable($datetime = '1970-01-01 00:00:00', $timezone = null);
_dump($result, json_encode($result));
$result = $calendar->dateTimeZone($timezone = '');
_dump($result, json_encode($result));
$result = $calendar->dateTimeZone($timezone = 'UTC');
_dump($result, json_encode($result));
$result = $calendar->dateInterval($duration = '');
_dump($result, json_encode($result));
$result = $calendar->dateInterval($duration = 'P1D');
_dump($result, json_encode($result));
};
_assert_output($fn, PHP_VERSION_ID >= 80000
? '
"TEST 1"
{ object # Gzhegow\Calendar\Struct\PHP8\DateTime } | "\"1970-01-01T00:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateTime } | "\"1970-01-01T00:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateTimeImmutable } | "\"1970-01-01T00:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateTimeImmutable } | "\"1970-01-01T00:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateTimeZone } | "\"Europe\/Minsk\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateTimeZone } | "\"UTC\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateInterval } | "\"P0D\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateInterval } | "\"P1D\""
'
: '
"TEST 1"
{ object # Gzhegow\Calendar\Struct\PHP7\DateTime } | "\"1970-01-01T00:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateTime } | "\"1970-01-01T00:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateTimeImmutable } | "\"1970-01-01T00:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateTimeImmutable } | "\"1970-01-01T00:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateTimeZone } | "\"Europe\/Minsk\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateTimeZone } | "\"UTC\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateInterval } | "\"P0D\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateInterval } | "\"P1D\""
');
// > TEST
// > распознаем дату, временную зону и интервал
$fn = function () use ($calendar) {
_dump('TEST 2');
echo PHP_EOL;
$result = $calendar->parseDateTime($datetime = '1970-01-01 00:00:00', $formats = [ 'Y-m-d H:i:s' ], $timezoneIfParsed = 'UTC');
_dump($result, json_encode($result));
$result = $calendar->parseDateTimeImmutable($datetime = '1970-01-01 00:00:00', $formats = [ 'Y-m-d H:i:s' ], $timezoneIfParsed = 'UTC');
_dump($result, json_encode($result));
$result = $calendar->parseDateTimeZone($timezone = 'UTC');
_dump($result, json_encode($result));
$result = $calendar->parseDateInterval($interval = 'P0D', $formats = null);
_dump($result, json_encode($result));
};
_assert_output($fn, PHP_VERSION_ID >= 80000
? '
"TEST 2"
{ object # Gzhegow\Calendar\Struct\PHP8\DateTime } | "\"1970-01-01T00:00:00.000+00:00\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateTimeImmutable } | "\"1970-01-01T00:00:00.000+00:00\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateTimeZone } | "\"UTC\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateInterval } | "\"P0D\""
'
: '
"TEST 2"
{ object # Gzhegow\Calendar\Struct\PHP7\DateTime } | "\"1970-01-01T00:00:00.000+00:00\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateTimeImmutable } | "\"1970-01-01T00:00:00.000+00:00\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateTimeZone } | "\"UTC\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateInterval } | "\"P0D\""
');
// > TEST
// > проводим действия над датой
$fn = function () use ($calendar) {
_dump('TEST 3');
echo PHP_EOL;
$dateTimeImmutable11 = $calendar->parseDateTimeImmutable($datetime = '2000-01-01 midnight', $formats = null, $timezoneIfParsed = null);
$dateTimeImmutable12 = $dateTimeImmutable11->modify('+ 10 hours');
$dateTimeImmutableDiff13 = $dateTimeImmutable11->diff($dateTimeImmutable12);
_dump($dateTimeImmutable11, json_encode($dateTimeImmutable11));
_dump($dateTimeImmutable12, json_encode($dateTimeImmutable12));
_dump($dateTimeImmutableDiff13, $calendar->formatIntervalAgo($dateTimeImmutableDiff13));
$dateTimeImmutable21 = $calendar->parseDateTimeImmutable($datetime = '2000-01-01 midnight', $formats = null, $timezoneIfParsed = null);
$dateTimeImmutable22 = $dateTimeImmutable21->add(new \DateInterval('PT10H'));
$dateTimeImmutableDiff23 = $dateTimeImmutable21->diff($dateTimeImmutable22);
_dump($dateTimeImmutable21, json_encode($dateTimeImmutable21));
_dump($dateTimeImmutable22, json_encode($dateTimeImmutable22));
_dump($dateTimeImmutableDiff23, $calendar->formatIntervalAgo($dateTimeImmutableDiff23));
$dateTimeImmutable31 = $calendar->parseDateTimeImmutable($datetime = '2000-01-01 midnight', $formats = null, $timezoneIfParsed = null);
$dateTimeImmutable32 = $dateTimeImmutable31->sub(new \DateInterval('PT10H'));
$dateTimeImmutableDiff33 = $dateTimeImmutable31->diff($dateTimeImmutable32);
_dump($dateTimeImmutable31, json_encode($dateTimeImmutable31));
_dump($dateTimeImmutable32, json_encode($dateTimeImmutable32));
_dump($dateTimeImmutableDiff33, $calendar->formatIntervalAgo($dateTimeImmutableDiff33));
};
_assert_output($fn, PHP_VERSION_ID >= 80000
? '
"TEST 3"
{ object # Gzhegow\Calendar\Struct\PHP8\DateTimeImmutable } | "\"2000-01-01T00:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateTimeImmutable } | "\"2000-01-01T10:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateInterval } | "через 10 час."
{ object # Gzhegow\Calendar\Struct\PHP8\DateTimeImmutable } | "\"2000-01-01T00:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateTimeImmutable } | "\"2000-01-01T10:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateInterval } | "через 10 час."
{ object # Gzhegow\Calendar\Struct\PHP8\DateTimeImmutable } | "\"2000-01-01T00:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateTimeImmutable } | "\"1999-12-31T14:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP8\DateInterval } | "10 час. назад"
'
: '
"TEST 3"
{ object # Gzhegow\Calendar\Struct\PHP7\DateTimeImmutable } | "\"2000-01-01T00:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateTimeImmutable } | "\"2000-01-01T10:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateInterval } | "через 10 час."
{ object # Gzhegow\Calendar\Struct\PHP7\DateTimeImmutable } | "\"2000-01-01T00:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateTimeImmutable } | "\"2000-01-01T10:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateInterval } | "через 10 час."
{ object # Gzhegow\Calendar\Struct\PHP7\DateTimeImmutable } | "\"2000-01-01T00:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateTimeImmutable } | "\"1999-12-31T14:00:00.000+03:00\""
{ object # Gzhegow\Calendar\Struct\PHP7\DateInterval } | "10 час. назад"
');
// > TEST
// > проводим действия над датой
$fn = function () use ($calendar) {
_dump('TEST 4');
echo PHP_EOL;
$dateTime = $calendar->dateTime();
$formatted = $calendar->formatHumanDate($dateTime);
_dump($dateTime, json_encode($dateTime), $formatted);
$dateTime = $calendar->dateTime();
$formatted = $calendar->formatHumanDay($dateTime);
_dump($dateTime, json_encode($dateTime), $formatted);
};
_assert_output($fn, PHP_VERSION_ID >= 80000
? '
"TEST 4"
{ object # Gzhegow\Calendar\Struct\PHP8\DateTime } | "\"1970-01-01T00:00:00.000+03:00\"" | "Thu, 01 Jan 1970 00:00:00 +0300"
{ object # Gzhegow\Calendar\Struct\PHP8\DateTime } | "\"1970-01-01T00:00:00.000+03:00\"" | "Thu, 01 Jan 1970 +0300"
'
: '
"TEST 4"
{ object # Gzhegow\Calendar\Struct\PHP7\DateTime } | "\"1970-01-01T00:00:00.000+03:00\"" | "Thu, 01 Jan 1970 00:00:00 +0300"
{ object # Gzhegow\Calendar\Struct\PHP7\DateTime } | "\"1970-01-01T00:00:00.000+03:00\"" | "Thu, 01 Jan 1970 +0300"
');