400 lines
15 KiB
PHP
400 lines
15 KiB
PHP
|
<?php
|
||
|
/* vim: set expandtab tabstop=4 shiftwidth=4 softtabstop=4: */
|
||
|
|
||
|
/**
|
||
|
* Class for a group of elements used to input dates (and times).
|
||
|
*
|
||
|
* PHP versions 4 and 5
|
||
|
*
|
||
|
* LICENSE: This source file is subject to version 3.01 of the PHP license
|
||
|
* that is available through the world-wide-web at the following URI:
|
||
|
* http://www.php.net/license/3_01.txt If you did not receive a copy of
|
||
|
* the PHP License and are unable to obtain it through the web, please
|
||
|
* send a note to license@php.net so we can mail you a copy immediately.
|
||
|
*
|
||
|
* @category HTML
|
||
|
* @package HTML_QuickForm
|
||
|
* @author Alexey Borzov <avb@php.net>
|
||
|
* @copyright 2001-2009 The PHP Group
|
||
|
* @license http://www.php.net/license/3_01.txt PHP License 3.01
|
||
|
* @version CVS: $Id: date.php,v 1.62 2009/04/04 21:34:02 avb Exp $
|
||
|
* @link http://pear.php.net/package/HTML_QuickForm
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* Class for a group of form elements
|
||
|
*/
|
||
|
require_once 'HTML/QuickForm/group.php';
|
||
|
/**
|
||
|
* Class for <select></select> elements
|
||
|
*/
|
||
|
require_once 'HTML/QuickForm/select.php';
|
||
|
|
||
|
/**
|
||
|
* Class for a group of elements used to input dates (and times).
|
||
|
*
|
||
|
* Inspired by original 'date' element but reimplemented as a subclass
|
||
|
* of HTML_QuickForm_group
|
||
|
*
|
||
|
* @category HTML
|
||
|
* @package HTML_QuickForm
|
||
|
* @author Alexey Borzov <avb@php.net>
|
||
|
* @version Release: 3.2.11
|
||
|
* @since 3.1
|
||
|
*/
|
||
|
class HTML_QuickForm_date extends HTML_QuickForm_group
|
||
|
{
|
||
|
// {{{ properties
|
||
|
|
||
|
/**
|
||
|
* Various options to control the element's display.
|
||
|
*
|
||
|
* @access private
|
||
|
* @var array
|
||
|
*/
|
||
|
var $_options = array(
|
||
|
'format' => 'dMY',
|
||
|
'minYear' => 2001,
|
||
|
'maxYear' => 2012,
|
||
|
'addEmptyOption' => false,
|
||
|
'emptyOptionValue' => '',
|
||
|
'emptyOptionText' => ' ',
|
||
|
'optionIncrement' => array('i' => 1, 's' => 1)
|
||
|
);
|
||
|
|
||
|
/**
|
||
|
* These complement separators, they are appended to the resultant HTML
|
||
|
* @access private
|
||
|
* @var array
|
||
|
*/
|
||
|
var $_wrap = array('', '');
|
||
|
|
||
|
/**
|
||
|
* Locale array build from CRM_Utils_Date-provided names
|
||
|
*
|
||
|
* @access private
|
||
|
* @var array
|
||
|
*/
|
||
|
var $_locale = array();
|
||
|
|
||
|
// }}}
|
||
|
// {{{ constructor
|
||
|
|
||
|
/**
|
||
|
* Class constructor
|
||
|
*
|
||
|
* The following keys may appear in $options array:
|
||
|
* - 'language': date language
|
||
|
* - 'format': Format of the date, based on PHP's date() function.
|
||
|
* The following characters are currently recognised in format string:
|
||
|
* <pre>
|
||
|
* D => Short names of days
|
||
|
* l => Long names of days
|
||
|
* d => Day numbers
|
||
|
* M => Short names of months
|
||
|
* F => Long names of months
|
||
|
* m => Month numbers
|
||
|
* Y => Four digit year
|
||
|
* y => Two digit year
|
||
|
* h => 12 hour format
|
||
|
* H => 23 hour format
|
||
|
* i => Minutes
|
||
|
* s => Seconds
|
||
|
* a => am/pm
|
||
|
* A => AM/PM
|
||
|
* </pre>
|
||
|
* - 'minYear': Minimum year in year select
|
||
|
* - 'maxYear': Maximum year in year select
|
||
|
* - 'addEmptyOption': Should an empty option be added to the top of
|
||
|
* each select box?
|
||
|
* - 'emptyOptionValue': The value passed by the empty option.
|
||
|
* - 'emptyOptionText': The text displayed for the empty option.
|
||
|
* - 'optionIncrement': Step to increase the option values by (works for 'i' and 's')
|
||
|
*
|
||
|
* @access public
|
||
|
* @param string Element's name
|
||
|
* @param mixed Label(s) for an element
|
||
|
* @param array Options to control the element's display
|
||
|
* @param mixed Either a typical HTML attribute string or an associative array
|
||
|
*/
|
||
|
function __construct($elementName = null, $elementLabel = null, $options = array(), $attributes = null)
|
||
|
{
|
||
|
$this->_locale = array(
|
||
|
'weekdays_short'=> CRM_Utils_Date::getAbbrWeekdayNames(),
|
||
|
'weekdays_long' => CRM_Utils_Date::getFullWeekdayNames(),
|
||
|
'months_short' => CRM_Utils_Date::getAbbrMonthNames(),
|
||
|
'months_long' => CRM_Utils_Date::getFullMonthNames()
|
||
|
);
|
||
|
parent::__construct($elementName, $elementLabel, null, null, null, $attributes);
|
||
|
$this->_persistantFreeze = true;
|
||
|
$this->_appendName = true;
|
||
|
$this->_type = 'date';
|
||
|
// set the options, do not bother setting bogus ones
|
||
|
if (is_array($options)) {
|
||
|
foreach ($options as $name => $value) {
|
||
|
if (isset($this->_options[$name])) {
|
||
|
if (is_array($value) && is_array($this->_options[$name])) {
|
||
|
$this->_options[$name] = @array_merge($this->_options[$name], $value);
|
||
|
} else {
|
||
|
$this->_options[$name] = $value;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// }}}
|
||
|
// {{{ _createElements()
|
||
|
|
||
|
function _createElements()
|
||
|
{
|
||
|
$this->_separator = $this->_elements = array();
|
||
|
$separator = '';
|
||
|
$locale =& $this->_locale;
|
||
|
$backslash = false;
|
||
|
for ($i = 0, $length = strlen($this->_options['format']); $i < $length; $i++) {
|
||
|
$sign = $this->_options['format']{$i};
|
||
|
if ($backslash) {
|
||
|
$backslash = false;
|
||
|
$separator .= $sign;
|
||
|
} else {
|
||
|
$loadSelect = true;
|
||
|
switch ($sign) {
|
||
|
case 'D':
|
||
|
// Sunday is 0 like with 'w' in date()
|
||
|
$options = $locale['weekdays_short'];
|
||
|
$emptyText = ts('-day of week-');
|
||
|
break;
|
||
|
case 'l':
|
||
|
$options = $locale['weekdays_long'];
|
||
|
$emptyText = ts('-day of week-');
|
||
|
break;
|
||
|
case 'd':
|
||
|
$options = $this->_createOptionList(1, 31);
|
||
|
$emptyText = ts('-day-');
|
||
|
break;
|
||
|
case 'j':
|
||
|
// the no-zero-padding option (CRM-2793)
|
||
|
$options = $this->_createOptionList(1, 31, 1, false);
|
||
|
$emptyText = ts('-day-');
|
||
|
break;
|
||
|
case 'M':
|
||
|
$options = $locale['months_short'];
|
||
|
array_unshift($options , '');
|
||
|
unset($options[0]);
|
||
|
$emptyText = ts('-month-');
|
||
|
break;
|
||
|
case 'm':
|
||
|
$options = $this->_createOptionList(1, 12);
|
||
|
$emptyText = ts('-month-');
|
||
|
break;
|
||
|
case 'F':
|
||
|
$options = $locale['months_long'];
|
||
|
array_unshift($options , '');
|
||
|
unset($options[0]);
|
||
|
$emptyText = ts('-month-');
|
||
|
break;
|
||
|
case 'Y':
|
||
|
$options = $this->_createOptionList(
|
||
|
$this->_options['minYear'],
|
||
|
$this->_options['maxYear'],
|
||
|
$this->_options['minYear'] > $this->_options['maxYear']? -1: 1
|
||
|
);
|
||
|
$emptyText = ts('-year-');
|
||
|
break;
|
||
|
case 'y':
|
||
|
$options = $this->_createOptionList(
|
||
|
$this->_options['minYear'],
|
||
|
$this->_options['maxYear'],
|
||
|
$this->_options['minYear'] > $this->_options['maxYear']? -1: 1
|
||
|
);
|
||
|
array_walk($options, create_function('&$v,$k','$v = substr($v,-2);'));
|
||
|
$emptyText = ts('-year-');
|
||
|
break;
|
||
|
case 'h':
|
||
|
$options = $this->_createOptionList(1, 12);
|
||
|
$emptyText = ts('-hour-');
|
||
|
break;
|
||
|
case 'g':
|
||
|
$options = $this->_createOptionList(1, 12);
|
||
|
array_walk($options, create_function('&$v,$k', '$v = intval($v);'));
|
||
|
break;
|
||
|
case 'H':
|
||
|
$options = $this->_createOptionList(0, 23);
|
||
|
$emptyText = ts('-hour-');
|
||
|
break;
|
||
|
case 'i':
|
||
|
$options = $this->_createOptionList(0, 59, $this->_options['optionIncrement']['i']);
|
||
|
$emptyText = ts('-min-');
|
||
|
break;
|
||
|
case 's':
|
||
|
$options = $this->_createOptionList(0, 59, $this->_options['optionIncrement']['s']);
|
||
|
$emptyText = ts('-sec-');
|
||
|
break;
|
||
|
case 'a':
|
||
|
$options = array('am' => 'am', 'pm' => 'pm');
|
||
|
$emptyText = '-am/pm-';
|
||
|
break;
|
||
|
case 'A':
|
||
|
$options = array('AM' => 'AM', 'PM' => 'PM');
|
||
|
$emptyText = '-AM/PM-';
|
||
|
break;
|
||
|
case 'W':
|
||
|
$options = $this->_createOptionList(1, 53);
|
||
|
break;
|
||
|
case '\\':
|
||
|
$backslash = true;
|
||
|
$loadSelect = false;
|
||
|
break;
|
||
|
default:
|
||
|
$separator .= (' ' == $sign? ' ': $sign);
|
||
|
$loadSelect = false;
|
||
|
}
|
||
|
|
||
|
if ($loadSelect) {
|
||
|
if (0 < count($this->_elements)) {
|
||
|
$this->_separator[] = $separator;
|
||
|
} else {
|
||
|
$this->_wrap[0] = $separator;
|
||
|
}
|
||
|
$separator = '';
|
||
|
// Should we add an empty option to the top of the select?
|
||
|
if (!is_array($this->_options['addEmptyOption']) && $this->_options['addEmptyOption'] ||
|
||
|
is_array($this->_options['addEmptyOption']) && !empty($this->_options['addEmptyOption'][$sign])) {
|
||
|
|
||
|
// Using '+' array operator to preserve the keys
|
||
|
if (is_array($this->_options['emptyOptionText']) && !empty($this->_options['emptyOptionText'][$sign])) {
|
||
|
$text = $emptyText ? $emptyText : $this->_options['emptyOptionText'][$sign];
|
||
|
$options = array($this->_options['emptyOptionValue'] => $text) + $options;
|
||
|
} else {
|
||
|
$text = $emptyText ? $emptyText : $this->_options['emptyOptionText'];
|
||
|
$options = array($this->_options['emptyOptionValue'] => $text) + $options;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//modified autogenerated id for date select boxes.
|
||
|
$attribs = $this->getAttributes();
|
||
|
$elementName = $this->getName();
|
||
|
$attribs['id'] = $elementName.'['.$sign.']';
|
||
|
|
||
|
$this->_elements[] = new HTML_QuickForm_select($sign, null, $options, $attribs);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
$this->_wrap[1] = $separator . ($backslash? '\\': '');
|
||
|
}
|
||
|
|
||
|
// }}}
|
||
|
// {{{ _createOptionList()
|
||
|
|
||
|
/**
|
||
|
* Creates an option list containing the numbers from the start number to the end, inclusive
|
||
|
*
|
||
|
* @param int The start number
|
||
|
* @param int The end number
|
||
|
* @param int Increment by this value
|
||
|
* @param bool Whether to pad the result with leading zero (CRM-2793)
|
||
|
* @access private
|
||
|
* @return array An array of numeric options.
|
||
|
*/
|
||
|
function _createOptionList($start, $end, $step = 1, $pad = true)
|
||
|
{
|
||
|
for ($i = $start, $options = array(); $start > $end? $i >= $end: $i <= $end; $i += $step) {
|
||
|
$options[$i] = $pad ? sprintf('%02d', $i) : sprintf('%d', $i);
|
||
|
}
|
||
|
return $options;
|
||
|
}
|
||
|
|
||
|
// }}}
|
||
|
// {{{ _trimLeadingZeros()
|
||
|
|
||
|
/**
|
||
|
* Trims leading zeros from the (numeric) string
|
||
|
*
|
||
|
* @param string A numeric string, possibly with leading zeros
|
||
|
* @return string String with leading zeros removed
|
||
|
*/
|
||
|
function _trimLeadingZeros($str)
|
||
|
{
|
||
|
if (0 == strcmp($str, $this->_options['emptyOptionValue'])) {
|
||
|
return $str;
|
||
|
}
|
||
|
$trimmed = ltrim($str, '0');
|
||
|
return strlen($trimmed)? $trimmed: '0';
|
||
|
}
|
||
|
|
||
|
// }}}
|
||
|
// {{{ setValue()
|
||
|
|
||
|
function setValue($value)
|
||
|
{
|
||
|
if (empty($value)) {
|
||
|
$value = array();
|
||
|
} elseif (is_scalar($value)) {
|
||
|
if (!is_numeric($value)) {
|
||
|
$value = strtotime($value);
|
||
|
}
|
||
|
// might be a unix epoch, then we fill all possible values
|
||
|
$arr = explode('-', date('w-j-n-Y-g-G-i-s-a-A-W', (int)$value));
|
||
|
$value = array(
|
||
|
'D' => $arr[0],
|
||
|
'l' => $arr[0],
|
||
|
'd' => $arr[1],
|
||
|
'M' => $arr[2],
|
||
|
'm' => $arr[2],
|
||
|
'F' => $arr[2],
|
||
|
'Y' => $arr[3],
|
||
|
'y' => $arr[3],
|
||
|
'h' => $arr[4],
|
||
|
'g' => $arr[4],
|
||
|
'H' => $arr[5],
|
||
|
'i' => $this->_trimLeadingZeros($arr[6]),
|
||
|
's' => $this->_trimLeadingZeros($arr[7]),
|
||
|
'a' => $arr[8],
|
||
|
'A' => $arr[9],
|
||
|
'W' => $this->_trimLeadingZeros($arr[10])
|
||
|
);
|
||
|
} else {
|
||
|
$value = array_map(array($this, '_trimLeadingZeros'), $value);
|
||
|
}
|
||
|
parent::setValue($value);
|
||
|
}
|
||
|
|
||
|
// }}}
|
||
|
// {{{ toHtml()
|
||
|
|
||
|
function toHtml()
|
||
|
{
|
||
|
include_once('HTML/QuickForm/Renderer/Default.php');
|
||
|
$renderer = new HTML_QuickForm_Renderer_Default();
|
||
|
$renderer->setElementTemplate('{element}');
|
||
|
parent::accept($renderer);
|
||
|
return $this->_wrap[0] . $renderer->toHtml() . $this->_wrap[1];
|
||
|
}
|
||
|
|
||
|
// }}}
|
||
|
// {{{ accept()
|
||
|
|
||
|
function accept(&$renderer, $required = false, $error = null)
|
||
|
{
|
||
|
$renderer->renderElement($this, $required, $error);
|
||
|
}
|
||
|
|
||
|
// }}}
|
||
|
// {{{ onQuickFormEvent()
|
||
|
|
||
|
function onQuickFormEvent($event, $arg, &$caller)
|
||
|
{
|
||
|
if ('updateValue' == $event) {
|
||
|
// we need to call setValue(), 'cause the default/constant value
|
||
|
// may be in fact a timestamp, not an array
|
||
|
return HTML_QuickForm_element::onQuickFormEvent($event, $arg, $caller);
|
||
|
} else {
|
||
|
return parent::onQuickFormEvent($event, $arg, $caller);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// }}}
|
||
|
}
|
||
|
?>
|