-
-
Notifications
You must be signed in to change notification settings - Fork 148
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
added addDate(), addTime() & addDateTime()
- Loading branch information
Showing
11 changed files
with
722 additions
and
9 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,213 @@ | ||
<?php | ||
|
||
/** | ||
* This file is part of the Nette Framework (https://nette.org) | ||
* Copyright (c) 2004 David Grudl (https://davidgrudl.com) | ||
*/ | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Nette\Forms\Controls; | ||
|
||
use Nette; | ||
use Nette\Forms\Form; | ||
|
||
|
||
/** | ||
* Selects date or time or date & time. | ||
*/ | ||
class DateTimeControl extends BaseControl | ||
{ | ||
public const | ||
TypeDate = 1, | ||
TypeTime = 2, | ||
TypeDateTime = 3; | ||
|
||
public const | ||
FormatObject = 'object', | ||
FormatTimestamp = 'timestamp'; | ||
|
||
/** @var int */ | ||
private $type; | ||
|
||
/** @var bool */ | ||
private $withSeconds; | ||
|
||
/** @var string */ | ||
private $format = self::FormatObject; | ||
|
||
|
||
public function __construct($label = null, int $type = self::TypeDate, bool $withSeconds = false) | ||
{ | ||
$this->type = $type; | ||
$this->withSeconds = $withSeconds; | ||
parent::__construct($label); | ||
$this->control->step = $withSeconds ? 1 : null; | ||
$this->setOption('type', 'datetime'); | ||
} | ||
|
||
|
||
/** | ||
* Format of returned value. Allowed values are string (ie 'Y-m-d'), DateTimeControl::FormatObject and DateTimeControl::FormatTimestamp. | ||
* @return static | ||
*/ | ||
public function setFormat(string $format) | ||
{ | ||
$this->format = $format; | ||
return $this; | ||
} | ||
|
||
|
||
/** | ||
* @param \DateTimeInterface|string|int|null $value | ||
* @return static | ||
*/ | ||
public function setValue($value) | ||
{ | ||
$this->value = $value === null || $value === '' | ||
? null | ||
: $this->normalizeValue($value); | ||
return $this; | ||
} | ||
|
||
|
||
/** | ||
* @return \DateTimeImmutable|string|int|null | ||
*/ | ||
public function getValue() | ||
{ | ||
if ($this->format === self::FormatObject) { | ||
return $this->value; | ||
} elseif ($this->format === self::FormatTimestamp) { | ||
return $this->value ? $this->value->getTimestamp() : null; | ||
} else { | ||
return $this->value ? $this->value->format($this->format) : null; | ||
} | ||
} | ||
|
||
|
||
/** | ||
* @param \DateTimeInterface|string|int $value | ||
*/ | ||
private function normalizeValue($value): \DateTimeImmutable | ||
{ | ||
if (is_numeric($value)) { | ||
$dt = (new \DateTimeImmutable)->setTimestamp((int) $value); | ||
} elseif (is_string($value) && $value !== '') { | ||
$dt = $this->createDateTime($value); | ||
} elseif ($value instanceof \DateTime) { | ||
$dt = \DateTimeImmutable::createFromMutable($value); | ||
} elseif ($value instanceof \DateTimeImmutable) { | ||
$dt = $value; | ||
} else { | ||
throw new \TypeError('Value must be DateTimeInterface|string|int|null, ' . gettype($value) . ' given.'); | ||
} | ||
|
||
[$h, $m, $s] = [(int) $dt->format('H'), (int) $dt->format('i'), $this->withSeconds ? (int) $dt->format('s') : 0]; | ||
if ($this->type === self::TypeDate) { | ||
return $dt->setTime(0, 0); | ||
} elseif ($this->type === self::TypeTime) { | ||
return $dt->setDate(0, 1, 1)->setTime($h, $m, $s); | ||
} elseif ($this->type === self::TypeDateTime) { | ||
return $dt->setTime($h, $m, $s); | ||
} | ||
} | ||
|
||
|
||
public function loadHttpData(): void | ||
{ | ||
try { | ||
parent::loadHttpData(); | ||
} catch (\Throwable $e) { | ||
$this->value = null; | ||
} | ||
} | ||
|
||
|
||
public function getControl(): Nette\Utils\Html | ||
{ | ||
return parent::getControl()->addAttributes($this->getAttributesFromRules())->addAttributes([ | ||
'value' => $this->value ? $this->formatHtmlValue($this->value) : null, | ||
'type' => [self::TypeDate => 'date', self::TypeTime => 'time', self::TypeDateTime => 'datetime-local'][$this->type], | ||
]); | ||
} | ||
|
||
|
||
/** | ||
* Formats a date/time for HTML attributes. | ||
* @param \DateTimeInterface|string|int $value | ||
*/ | ||
public function formatHtmlValue($value): string | ||
{ | ||
$value = $this->normalizeValue($value); | ||
return $value->format([ | ||
self::TypeDate => 'Y-m-d', | ||
self::TypeTime => $this->withSeconds ? 'H:i:s' : 'H:i', | ||
self::TypeDateTime => $this->withSeconds ? 'Y-m-d\\TH:i:s' : 'Y-m-d\\TH:i', | ||
][$this->type]); | ||
} | ||
|
||
|
||
/** | ||
* Formats a date/time according to the locale and formatting options. | ||
* @param \DateTimeInterface|string|int $value | ||
*/ | ||
public function formatLocaleText($value): string | ||
{ | ||
$value = $this->normalizeValue($value); | ||
if ($this->type === self::TypeDate) { | ||
return \IntlDateFormatter::formatObject($value, [\IntlDateFormatter::MEDIUM, \IntlDateFormatter::NONE]); | ||
} elseif ($this->type === self::TypeTime) { | ||
return \IntlDateFormatter::formatObject($value, [\IntlDateFormatter::NONE, $this->withSeconds ? \IntlDateFormatter::MEDIUM : \IntlDateFormatter::SHORT]); | ||
} elseif ($this->type === self::TypeDateTime) { | ||
return \IntlDateFormatter::formatObject($value, [\IntlDateFormatter::MEDIUM, $this->withSeconds ? \IntlDateFormatter::MEDIUM : \IntlDateFormatter::SHORT]); | ||
} | ||
} | ||
|
||
|
||
private function getAttributesFromRules(): array | ||
{ | ||
$attrs = []; | ||
$format = function ($val) { | ||
return is_scalar($val) || $val instanceof \DateTimeInterface | ||
? $this->formatHtmlValue($val) | ||
: null; | ||
}; | ||
foreach ($this->getRules() as $rule) { | ||
if ($rule->branch) { | ||
} elseif (!$rule->canExport()) { | ||
break; | ||
} elseif ($rule->validator === Form::Min) { | ||
$attrs['min'] = $format($rule->arg); | ||
} elseif ($rule->validator === Form::Max) { | ||
$attrs['max'] = $format($rule->arg); | ||
} elseif ($rule->validator === Form::Range) { | ||
$attrs['min'] = $format($rule->arg[0] ?? null); | ||
$attrs['max'] = $format($rule->arg[1] ?? null); | ||
} | ||
} | ||
return $attrs; | ||
} | ||
|
||
|
||
public function validateMinMax($min, $max): bool | ||
{ | ||
$value = $this->normalizeValue($this->value); | ||
$min = $min === null ? null : $this->normalizeValue($min); | ||
$max = $max === null ? null : $this->normalizeValue($max); | ||
return $this->type === self::TypeTime && $min > $max | ||
? $value >= $min || $value <= $max | ||
: $value >= $min && ($max === null || $value <= $max); | ||
} | ||
|
||
|
||
private function createDateTime(string $value): \DateTimeImmutable | ||
{ | ||
$dt = new \DateTimeImmutable($value); | ||
$errors = \DateTimeImmutable::getLastErrors(); | ||
if ($errors && $errors['warnings']) { | ||
throw new Nette\InvalidArgumentException(Nette\Utils\Arrays::first($errors['warnings']) . " '$value'"); | ||
} | ||
return $dt; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,44 @@ | ||
<?php | ||
|
||
/** | ||
* Test: Nette\Forms\Controls\DateTimeControl. | ||
*/ | ||
|
||
declare(strict_types=1); | ||
|
||
use Nette\Forms\Controls\DateTimeControl; | ||
use Nette\Forms\Form; | ||
use Tester\Assert; | ||
|
||
|
||
require __DIR__ . '/../bootstrap.php'; | ||
|
||
|
||
test('string format', function () { | ||
$form = new Form; | ||
$input = $form->addDate('date') | ||
->setValue('2023-10-22 10:30') | ||
->setFormat('j.n.Y'); | ||
|
||
Assert::same('22.10.2023', $input->getValue()); | ||
}); | ||
|
||
|
||
test('timestamp format', function () { | ||
$form = new Form; | ||
$input = $form->addDate('date') | ||
->setValue('2023-10-22 10:30') | ||
->setFormat(DateTimeControl::FormatTimestamp); | ||
|
||
Assert::same(1697925600, $input->getValue()); | ||
}); | ||
|
||
|
||
test('object format', function () { | ||
$form = new Form; | ||
$input = $form->addDate('date') | ||
->setValue('2023-10-22 10:30') | ||
->setFormat(DateTimeControl::FormatObject); | ||
|
||
Assert::equal(new DateTimeImmutable('2023-10-22 00:00'), $input->getValue()); | ||
}); |
Oops, something went wrong.