You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
311 lines
9.8 KiB
311 lines
9.8 KiB
9 months ago
|
<?php
|
||
|
defined( 'ABSPATH' ) || die;
|
||
|
|
||
|
use MetaBox\Support\Arr;
|
||
|
|
||
|
/**
|
||
|
* The date and time picker field which allows users to select both date and time via jQueryUI datetime picker.
|
||
|
*/
|
||
|
class RWMB_Datetime_Field extends RWMB_Input_Field {
|
||
|
/**
|
||
|
* Translate date format from jQuery UI date picker to PHP date().
|
||
|
* It's used to store timestamp value of the field.
|
||
|
* Missing: '!' => '', 'oo' => '', '@' => '', "''" => "'".
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
protected static $date_formats = [
|
||
|
'd' => 'j',
|
||
|
'dd' => 'd',
|
||
|
'oo' => 'z',
|
||
|
'D' => 'D',
|
||
|
'DD' => 'l',
|
||
|
'm' => 'n',
|
||
|
'mm' => 'm',
|
||
|
'M' => 'M',
|
||
|
'MM' => 'F',
|
||
|
'y' => 'y',
|
||
|
'yy' => 'Y',
|
||
|
'o' => 'z',
|
||
|
];
|
||
|
|
||
|
/**
|
||
|
* Translate time format from jQuery UI time picker to PHP date().
|
||
|
* It's used to store timestamp value of the field.
|
||
|
* Missing: 't' => '', T' => '', 'm' => '', 's' => ''.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
protected static $time_formats = [
|
||
|
'H' => 'G',
|
||
|
'HH' => 'H',
|
||
|
'h' => 'g',
|
||
|
'hh' => 'h',
|
||
|
'mm' => 'i',
|
||
|
'ss' => 's',
|
||
|
'l' => 'u',
|
||
|
'tt' => 'a',
|
||
|
'TT' => 'A',
|
||
|
];
|
||
|
|
||
|
public static function register_assets() {
|
||
|
// jQueryUI base theme: https://github.com/jquery/jquery-ui/tree/1.13.2/themes/base
|
||
|
$url = RWMB_CSS_URL . 'jqueryui';
|
||
|
wp_register_style( 'jquery-ui-core', "$url/core.css", [], '1.13.2' );
|
||
|
wp_style_add_data( 'jquery-ui-core', 'path', RWMB_CSS_DIR . 'jqueryui/core.css' );
|
||
|
|
||
|
wp_register_style( 'jquery-ui-theme', "$url/theme.css", [], '1.13.2' );
|
||
|
wp_style_add_data( 'jquery-ui-theme', 'path', RWMB_CSS_DIR . 'jqueryui/theme.css' );
|
||
|
|
||
|
wp_register_style( 'jquery-ui-datepicker', "$url/datepicker.css", [ 'jquery-ui-core', 'jquery-ui-theme' ], '1.13.2' );
|
||
|
wp_style_add_data( 'jquery-ui-datepicker', 'path', RWMB_CSS_DIR . 'jqueryui/datepicker.css' );
|
||
|
|
||
|
wp_register_style( 'jquery-ui-slider', "$url/slider.css", [ 'jquery-ui-core', 'jquery-ui-theme' ], '1.13.2' );
|
||
|
wp_style_add_data( 'jquery-ui-slider', 'path', RWMB_CSS_DIR . 'jqueryui/slider.css' );
|
||
|
|
||
|
// jQueryUI timepicker addon: https://github.com/trentrichardson/jQuery-Timepicker-Addon
|
||
|
wp_register_style( 'jquery-ui-timepicker', "$url/jquery-ui-timepicker-addon.min.css", [ 'rwmb-date', 'jquery-ui-slider' ], '1.6.3' );
|
||
|
wp_style_add_data( 'jquery-ui-timepicker', 'path', RWMB_CSS_DIR . 'jqueryui/jquery-ui-timepicker-addon.min.css' );
|
||
|
|
||
|
wp_register_style( 'rwmb-date', RWMB_CSS_URL . 'date.css', [ 'jquery-ui-datepicker' ], RWMB_VER );
|
||
|
wp_style_add_data( 'rwmb-date', 'path', RWMB_CSS_DIR . 'date.css' );
|
||
|
|
||
|
// Scripts.
|
||
|
$url = RWMB_JS_URL . 'jqueryui';
|
||
|
wp_register_script( 'jquery-ui-timepicker', "$url/jquery-ui-timepicker-addon.min.js", [ 'jquery-ui-datepicker', 'jquery-ui-slider' ], '1.6.3', true );
|
||
|
wp_register_script( 'jquery-ui-timepicker-slider', "$url/jquery-ui-sliderAccess.js", [ 'jquery-ui-datepicker', 'jquery-ui-slider' ], '0.3', true );
|
||
|
wp_register_script( 'jquery-ui-timepicker-i18n', "$url/jquery-ui-timepicker-addon-i18n.min.js", [ 'jquery-ui-timepicker' ], '1.6.3', true );
|
||
|
|
||
|
wp_register_script( 'rwmb-datetime', RWMB_JS_URL . 'datetime.js', [ 'jquery-ui-datepicker', 'jquery-ui-timepicker-i18n', 'underscore', 'jquery-ui-button', 'jquery-ui-timepicker-slider', 'rwmb' ], RWMB_VER, true );
|
||
|
wp_register_script( 'rwmb-date', RWMB_JS_URL . 'date.js', [ 'jquery-ui-datepicker', 'underscore', 'rwmb' ], RWMB_VER, true );
|
||
|
wp_register_script( 'rwmb-time', RWMB_JS_URL . 'time.js', [ 'jquery-ui-timepicker-i18n', 'jquery-ui-button', 'jquery-ui-timepicker-slider', 'rwmb' ], RWMB_VER, true );
|
||
|
|
||
|
$handles = [ 'datetime', 'time' ];
|
||
|
$locale = str_replace( '_', '-', get_locale() );
|
||
|
$locale_short = substr( $locale, 0, 2 );
|
||
|
$data = [
|
||
|
'locale' => $locale,
|
||
|
'localeShort' => $locale_short,
|
||
|
];
|
||
|
foreach ( $handles as $handle ) {
|
||
|
RWMB_Helpers_Field::localize_script_once( "rwmb-$handle", 'RWMB_' . ucfirst( $handle ), $data );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Enqueue scripts and styles.
|
||
|
*/
|
||
|
public static function admin_enqueue_scripts() {
|
||
|
self::register_assets();
|
||
|
wp_enqueue_style( 'jquery-ui-timepicker' );
|
||
|
wp_enqueue_script( 'rwmb-datetime' );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get field HTML.
|
||
|
*
|
||
|
* @param mixed $meta The field meta value.
|
||
|
* @param array $field The field parameters.
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public static function html( $meta, $field ) {
|
||
|
$output = '';
|
||
|
|
||
|
if ( $field['timestamp'] ) {
|
||
|
$name = $field['field_name'];
|
||
|
$field = wp_parse_args( [ 'field_name' => $name . '[formatted]' ], $field );
|
||
|
$timestamp = $meta['timestamp'] ?? 0;
|
||
|
$output .= sprintf(
|
||
|
'<input type="hidden" name="%s" class="rwmb-datetime-timestamp" value="%s">',
|
||
|
esc_attr( $name . '[timestamp]' ),
|
||
|
(int) $timestamp
|
||
|
);
|
||
|
|
||
|
$meta = $meta['formatted'] ?? '';
|
||
|
}
|
||
|
|
||
|
$output .= parent::html( $meta, $field );
|
||
|
|
||
|
if ( $field['inline'] ) {
|
||
|
$output .= '<div class="rwmb-datetime-inline"></div>';
|
||
|
}
|
||
|
|
||
|
return $output;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Calculates the timestamp from the datetime string and returns it if $field['timestamp'] is set or the datetime string if not.
|
||
|
*
|
||
|
* @param mixed $new The submitted meta value.
|
||
|
* @param mixed $old The existing meta value.
|
||
|
* @param int $post_id The post ID.
|
||
|
* @param array $field The field parameters.
|
||
|
*
|
||
|
* @return string|int
|
||
|
*/
|
||
|
public static function value( $new, $old, $post_id, $field ) {
|
||
|
if ( $field['timestamp'] ) {
|
||
|
if ( is_array( $new ) ) {
|
||
|
return $new['timestamp'];
|
||
|
}
|
||
|
if ( ! is_numeric( $new ) ) {
|
||
|
return strtotime( $new );
|
||
|
}
|
||
|
return $new;
|
||
|
}
|
||
|
|
||
|
if ( $field['save_format'] ) {
|
||
|
// Fix 'c' and 'r' formats not containing WordPress timezone.
|
||
|
$timezone = in_array( $field['save_format'], [ 'c', 'r' ], true ) ? wp_timezone() : null;
|
||
|
$date = DateTimeImmutable::createFromFormat( $field['php_format'], $new, $timezone );
|
||
|
return $date === false ? $new : $date->format( $field['save_format'] );
|
||
|
}
|
||
|
|
||
|
return $new;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get meta value.
|
||
|
*
|
||
|
* @param int $post_id The post ID.
|
||
|
* @param bool $saved Whether the meta box is saved at least once.
|
||
|
* @param array $field The field parameters.
|
||
|
*
|
||
|
* @return mixed
|
||
|
*/
|
||
|
public static function meta( $post_id, $saved, $field ) {
|
||
|
$meta = parent::meta( $post_id, $saved, $field );
|
||
|
|
||
|
if ( $field['timestamp'] ) {
|
||
|
return Arr::map( $meta, __CLASS__ . '::from_timestamp', $field );
|
||
|
}
|
||
|
|
||
|
if ( $field['save_format'] && $meta ) {
|
||
|
return Arr::map( $meta, __CLASS__ . '::from_save_format', $field );
|
||
|
}
|
||
|
|
||
|
return $meta;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Format meta value if set 'timestamp'.
|
||
|
*/
|
||
|
public static function from_timestamp( $meta, array $field ): array {
|
||
|
return [
|
||
|
'timestamp' => $meta ?: null,
|
||
|
'formatted' => $meta ? gmdate( $field['php_format'], intval( $meta ) ) : '',
|
||
|
];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Transform meta value from save format to the JS format.
|
||
|
*/
|
||
|
public static function from_save_format( $meta, array $field ): string {
|
||
|
$formats = array_merge(
|
||
|
[
|
||
|
$field['save_format'] => $field['save_format'],
|
||
|
],
|
||
|
[
|
||
|
'c' => DateTimeInterface::ATOM,
|
||
|
'r' => DateTimeInterface::RFC2822,
|
||
|
]
|
||
|
);
|
||
|
$format = $formats[ $field['save_format'] ];
|
||
|
$date = DateTimeImmutable::createFromFormat( $format, $meta );
|
||
|
return false === $date ? $meta : $date->format( $field['php_format'] );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Normalize parameters for field.
|
||
|
*
|
||
|
* @param array $field The field parameters.
|
||
|
* @return array
|
||
|
*/
|
||
|
public static function normalize( $field ) {
|
||
|
$field = wp_parse_args( $field, [
|
||
|
'timestamp' => false,
|
||
|
'inline' => false,
|
||
|
'js_options' => [],
|
||
|
'save_format' => '',
|
||
|
'autocomplete' => 'off',
|
||
|
] );
|
||
|
|
||
|
// Deprecate 'format', but keep it for backward compatible.
|
||
|
// Use 'js_options' instead.
|
||
|
$field['js_options'] = wp_parse_args( $field['js_options'], [
|
||
|
'timeFormat' => 'HH:mm',
|
||
|
'separator' => ' ',
|
||
|
'dateFormat' => $field['format'] ?? 'yy-mm-dd',
|
||
|
'showButtonPanel' => true,
|
||
|
'changeYear' => true,
|
||
|
'yearRange' => '-100:+100',
|
||
|
'changeMonth' => true,
|
||
|
'oneLine' => true,
|
||
|
'controlType' => 'select', // select or slider
|
||
|
'addSliderAccess' => true,
|
||
|
'sliderAccessArgs' => [
|
||
|
'touchonly' => true, // To show sliderAccess only on touch devices
|
||
|
],
|
||
|
] );
|
||
|
|
||
|
if ( $field['inline'] ) {
|
||
|
$field['js_options'] = wp_parse_args( $field['js_options'], [ 'altFieldTimeOnly' => false ] );
|
||
|
}
|
||
|
|
||
|
$field['php_format'] = static::get_php_format( $field['js_options'] );
|
||
|
|
||
|
$field = parent::normalize( $field );
|
||
|
|
||
|
return $field;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the attributes for a field.
|
||
|
*
|
||
|
* @param array $field The field parameters.
|
||
|
* @param mixed $value The meta value.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
public static function get_attributes( $field, $value = null ) {
|
||
|
$attributes = parent::get_attributes( $field, $value );
|
||
|
$attributes = wp_parse_args( $attributes, [ 'data-options' => wp_json_encode( $field['js_options'] ) ] );
|
||
|
$attributes['type'] = 'text';
|
||
|
|
||
|
return $attributes;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns a date() compatible format string from the JavaScript format.
|
||
|
* @link http://www.php.net/manual/en/function.date.php
|
||
|
*/
|
||
|
protected static function get_php_format( array $js_options ): string {
|
||
|
return strtr( $js_options['dateFormat'], self::$date_formats )
|
||
|
. $js_options['separator']
|
||
|
. strtr( $js_options['timeFormat'], self::$time_formats );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Format a single value for the helper functions. Sub-fields should overwrite this method if necessary.
|
||
|
*
|
||
|
* @param array $field Field parameters.
|
||
|
* @param string $value The value.
|
||
|
* @param array $args Additional arguments. Rarely used. See specific fields for details.
|
||
|
* @param int|null $post_id Post ID. null for current post. Optional.
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public static function format_single_value( $field, $value, $args, $post_id ) {
|
||
|
if ( $field['timestamp'] ) {
|
||
|
$value = self::from_timestamp( $value, $field );
|
||
|
} else {
|
||
|
$value = [
|
||
|
'timestamp' => strtotime( $value ),
|
||
|
'formatted' => $value,
|
||
|
];
|
||
|
}
|
||
|
return empty( $args['format'] ) ? $value['formatted'] : gmdate( $args['format'], $value['timestamp'] );
|
||
|
}
|
||
|
}
|