getLocale()); if ($this->getHTML5()) { $attributes['type'] = 'datetime-local'; $attributes['min'] = $this->getMinDatetime(); $attributes['max'] = $this->getMaxDatetime(); } return $attributes; } public function getSchemaDataDefaults() { $defaults = parent::getSchemaDataDefaults(); return array_merge($defaults, [ 'html5' => $this->getHTML5() ]); } public function Type() { return 'text datetime'; } public function getHTML5() { return $this->html5; } public function setHTML5($bool) { $this->html5 = $bool; return $this; } /** * Assign value posted from form submission * * @param mixed $value * @param mixed $data * @return $this */ public function setSubmittedValue($value, $data = null) { // Save raw value for later validation $this->rawValue = $value; // Null case if (!$value) { $this->value = null; return $this; } // Parse from submitted value $this->value = $this->localisedToISO8601($value); return $this; } /** * Convert date localised in the current locale to ISO 8601 date * * @param string $date * @return string The formatted date, or null if not a valid date */ public function localisedToISO8601($datetime) { if (!$datetime) { return null; } $fromFormatter = $this->getFormatter(); $toFormatter = $this->getISO8601Formatter(); $timestamp = $fromFormatter->parse($datetime); // Try to parse time without seconds, since that's a valid HTML5 submission format // See https://html.spec.whatwg.org/multipage/infrastructure.html#times if ($timestamp === false && $this->getHTML5()) { $fromFormatter->setPattern(str_replace(':ss', '', $fromFormatter->getPattern())); $timestamp = $fromFormatter->parse($datetime); } if ($timestamp === false) { return null; } return $toFormatter->format($timestamp) ?: null; } /** * Get date formatter with the standard locale / date format * * @throws \LogicException * @return IntlDateFormatter */ protected function getFormatter() { if ($this->getHTML5() && $this->datetimeFormat && $this->datetimeFormat !== DBDatetime::ISO_DATETIME) { throw new \LogicException( 'Please opt-out of HTML5 processing of ISO 8601 dates via setHTML5(false) if using setDatetimeFormat()' ); } if ($this->getHTML5() && $this->dateLength) { throw new \LogicException( 'Please opt-out of HTML5 processing of ISO 8601 dates via setHTML5(false) if using setDateLength()' ); } if ($this->getHTML5() && $this->locale) { throw new \LogicException( 'Please opt-out of HTML5 processing of ISO 8601 dates via setHTML5(false) if using setLocale()' ); } $formatter = IntlDateFormatter::create( $this->getLocale(), $this->getDateLength(), $this->getTimeLength(), $this->getTimezone() ); if ($this->getHTML5()) { // Browsers expect ISO 8601 dates, localisation is handled on the client $formatter->setPattern(DBDatetime::ISO_DATETIME); } elseif ($this->datetimeFormat) { // Don't invoke getDatetimeFormat() directly to avoid infinite loop $ok = $formatter->setPattern($this->datetimeFormat); if (!$ok) { throw new InvalidArgumentException("Invalid date format {$this->datetimeFormat}"); } } return $formatter; } /** * Get date format in CLDR standard format * * This can be set explicitly. If not, this will be generated from the current locale * with the current date length. * * @see http://userguide.icu-project.org/formatparse/datetime#TOC-Date-Field-Symbol-Table */ public function getDatetimeFormat() { if ($this->getHTML5()) { // Browsers expect ISO 8601 dates, localisation is handled on the client $this->setDatetimeFormat(DBDatetime::ISO_DATETIME); } if ($this->datetimeFormat) { return $this->datetimeFormat; } // Get from locale return $this->getFormatter()->getPattern(); } /** * Set date format in CLDR standard format. * Only applicable with {@link setHTML5(false)}. * * @see http://userguide.icu-project.org/formatparse/datetime#TOC-Date-Field-Symbol-Table * @param string $format * @return $this */ public function setDatetimeFormat($format) { $this->datetimeFormat = $format; return $this; } /** * Get formatter for converting to the target timezone, if timezone is set * Can return null if no timezone set * * @return IntlDateFormatter|null */ protected function getTimezoneFormatter() { $timezone = $this->getTimezone(); if (!$timezone) { return null; } // Build new formatter with the altered timezone $formatter = clone $this->getISO8601Formatter(); $formatter->setTimeZone($timezone); return $formatter; } /** * Get a date formatter for the ISO 8601 format * * @return IntlDateFormatter */ protected function getISO8601Formatter() { $formatter = IntlDateFormatter::create( i18n::config()->uninherited('default_locale'), IntlDateFormatter::MEDIUM, IntlDateFormatter::MEDIUM, date_default_timezone_get() // Default to server timezone ); $formatter->setLenient(false); // CLDR iso8601 date. // Note we omit timezone from this format, and we assume server TZ always. $formatter->setPattern('y-MM-dd HH:mm:ss'); return $formatter; } /** * Assign value from iso8601 string * * @param mixed $value * @param mixed $data * @return $this */ public function setValue($value, $data = null) { // Save raw value for later validation $this->rawValue = $value; // Empty value if (empty($value)) { $this->value = null; return $this; } if (is_array($value)) { throw new InvalidArgumentException("Use setSubmittedValue to assign by array"); }; // Validate iso 8601 date // If invalid, assign for later validation failure $isoFormatter = $this->getISO8601Formatter(); $timestamp = $isoFormatter->parse($value); if ($timestamp === false) { return $this; } // Cleanup date $value = $isoFormatter->format($timestamp); // Save value $this->value = $value; // Shift iso date into timezone before assignment to subfields $timezoneFormatter = $this->getTimezoneFormatter(); if ($timezoneFormatter) { $value = $timezoneFormatter->format($timestamp); } return $this; } public function Value() { return $this->iso8601ToLocalised($this->value); } /** * Convert an ISO 8601 localised date into the format specified by the * current date format. * * @param string $date * @return string The formatted date, or null if not a valid date */ public function iso8601ToLocalised($datetime) { $datetime = $this->tidyISO8601($datetime); if (!$datetime) { return null; } $fromFormatter = $this->getISO8601Formatter(); $toFormatter = $this->getFormatter(); $timestamp = $fromFormatter->parse($datetime); if ($timestamp === false) { return null; } return $toFormatter->format($timestamp) ?: null; } /** * Tidy up iso8601-ish date, or approximation * * @param string $date Date in iso8601 or approximate form * @return string iso8601 date, or null if not valid */ public function tidyISO8601($datetime) { if (!$datetime) { return null; } // Re-run through formatter to tidy up (e.g. remove time component) $formatter = $this->getISO8601Formatter(); $timestamp = $formatter->parse($datetime); if ($timestamp === false) { // Fallback to strtotime $timestamp = strtotime($datetime, DBDatetime::now()->getTimestamp()); if ($timestamp === false) { return null; } } return $formatter->format($timestamp); } /** * Get length of the date format to use. One of: * * - IntlDateFormatter::SHORT * - IntlDateFormatter::MEDIUM * - IntlDateFormatter::LONG * - IntlDateFormatter::FULL * * @see http://php.net/manual/en/class.intldateformatter.php#intl.intldateformatter-constants * @return int */ public function getDateLength() { if ($this->dateLength) { return $this->dateLength; } return IntlDateFormatter::MEDIUM; } /** * Get length of the date format to use. * Only applicable with {@link setHTML5(false)}. * * @see http://php.net/manual/en/class.intldateformatter.php#intl.intldateformatter-constants * * @param int $length * @return $this */ public function setDateLength($length) { $this->dateLength = $length; return $this; } /** * Get length of the date format to use. One of: * * - IntlDateFormatter::SHORT * - IntlDateFormatter::MEDIUM * - IntlDateFormatter::LONG * - IntlDateFormatter::FULL * * @see http://php.net/manual/en/class.intldateformatter.php#intl.intldateformatter-constants * @return int */ public function getTimeLength() { if ($this->timeLength) { return $this->timeLength; } return IntlDateFormatter::MEDIUM; } /** * Get length of the date format to use. * Only applicable with {@link setHTML5(false)}. * * @see http://php.net/manual/en/class.intldateformatter.php#intl.intldateformatter-constants * * @param int $length * @return $this */ public function setTimeLength($length) { $this->timeLength = $length; return $this; } public function setDisabled($bool) { parent::setDisabled($bool); return $this; } public function setReadonly($bool) { parent::setReadonly($bool); return $this; } /** * Set default locale for this field. If omitted will default to the current locale. * * @param string $locale * @return $this */ public function setLocale($locale) { $this->locale = $locale; return $this; } /** * Get locale for this field * * @return string */ public function getLocale() { return $this->locale ?: i18n::get_locale(); } /** * @return string */ public function getMinDatetime() { return $this->minDatetime; } /** * @param string $minDatetime * @return $this */ public function setMinDatetime($minDatetime) { $this->minDatetime = $this->tidyISO8601($minDatetime); return $this; } /** * @return string */ public function getMaxDatetime() { return $this->maxDatetime; } /** * @param string $maxDatetime * @return $this */ public function setMaxDatetime($maxDatetime) { $this->maxDatetime = $this->tidyISO8601($maxDatetime); return $this; } /** * @param Validator $validator * @return bool */ public function validate($validator) { // Don't validate empty fields if (empty($this->rawValue)) { return true; } // We submitted a value, but it couldn't be parsed if (empty($this->value)) { $validator->validationError( $this->name, _t( 'DatetimeField.VALIDDATETIMEFORMAT', "Please enter a valid date and time format ({format})", ['format' => $this->getDatetimeFormat()] ) ); return false; } // Check min date $min = $this->getMinDatetime(); if ($min) { $oops = strtotime($this->value) < strtotime($min); if ($oops) { $validator->validationError( $this->name, _t( 'DatetimeField.VALIDDATETIMEMINDATE', "Your date has to be newer or matching the minimum allowed date and time ({datetime})", ['datetime' => $this->iso8601ToLocalised($min)] ) ); return false; } } // Check max date $max = $this->getMaxDatetime(); if ($max) { $oops = strtotime($this->value) > strtotime($max); if ($oops) { $validator->validationError( $this->name, _t( 'DatetimeField.VALIDDATEMAXDATETIME', "Your date has to be older or matching the maximum allowed date and time ({datetime})", ['datetime' => $this->iso8601ToLocalised($max)] ) ); return false; } } return true; } public function performReadonlyTransformation() { $field = clone $this; $field->setReadonly(true); return $field; } /** * @return string */ public function getTimezone() { return $this->timezone; } /** * Custom timezone * * @var string */ protected $timezone = null; /** * @param string $timezone * @return $this */ public function setTimezone($timezone) { if ($this->value && $timezone !== $this->timezone) { throw new \BadMethodCallException("Can't change timezone after setting a value"); } // Note: DateField has no timezone option, and TimeField::setTimezone // should be ignored $this->timezone = $timezone; return $this; } }