aboutsummaryrefslogtreecommitdiffstats
path: root/phpBB/includes/datetime.php
blob: 577081e40b06eac0fe9e17bd49a40f962fec476c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
<?php
/**
*
* @package phpBB3
* @version $Id$
* @copyright (c) 2010 phpBB Group
* @license http://opensource.org/licenses/gpl-license.php GNU Public License
*/

/**
* phpBB custom extensions to the PHP DateTime class
* This handles the relative formats phpBB employs
*/
class phpbb_datetime extends DateTime
{
	/**
	* String used to wrap the date segment which should be replaced by today/tomorrow/yesterday
	*/
	const RELATIVE_WRAPPER = '|';

	/**
	* @var user User who is the context for this DateTime instance
	*/
	protected $_user;

	/**
	* @var array Date formats are preprocessed by phpBB, to save constant recalculation they are cached.
	*/
	static protected $format_cache = array();

	/**
	* Constructs a new instance of phpbb_datetime, expanded to include an argument to inject
	* the user context and modify the timezone to the users selected timezone if one is not set.
	*
	* @param string $time String in a format accepted by strtotime().
	* @param DateTimeZone $timezone Time zone of the time.
	* @param user User object for context.
	*/
	public function __construct($time = 'now', DateTimeZone $timezone = null, $user = null)
	{
		$this->_user	= $user ? $user : $GLOBALS['user'];

		$timezone		= (!$timezone && $this->_user->tz instanceof DateTimeZone) ? $this->_user->tz : $timezone;

		parent::__construct($time, $timezone);
	}

	/**
	* Returns a UNIX timestamp representation of the date time.
	*
	* @return int UNIX timestamp
	*/
	public function getTimestamp()
	{
		static $compat;

		if (!isset($compat))
		{
			$compat = !method_exists('DateTime', 'getTimestamp');
		}

		return !$compat ? parent::getTimestamp() : (int) parent::format('U');
	}

	/**
	* Formats the current date time into the specified format
	*
	* @param string $format Optional format to use for output, defaults to users chosen format
	* @param boolean $force_absolute Force output of a non relative date
	* @return string Formatted date time
	*/
	public function format($format = '', $force_absolute = false)
	{
		$format		= $format ? $format : $this->_user->date_format;
		$relative	= (strpos($format, self::RELATIVE_WRAPPER) !== false && !$force_absolute);
		$now		= new self('now', $this->_user->tz, $this->_user);

		$timestamp	= $this->getTimestamp();
		$now_ts		= $now->getTimeStamp();

		$delta		= $now_ts - $timestamp;

		if ($relative)
		{
			// Check the delta is less than or equal to 1 hour
			// and the delta is either greater than -5 seconds or timestamp and current time are of the same minute (they must be in the same hour already)
			// finally check that relative dates are supported by the language pack
			if ($delta <= 3600 && ($delta >= -5 || (($now_ts / 60) % 60) == (($timestamp / 60) % 60)) && isset($this->_user->lang['datetime']['AGO']))
			{
				return $this->_user->lang(array('datetime', 'AGO'), max(0, (int) floor($delta / 60)));
			}
			else
			{
				$midnight = clone $now;
				$midnight->setTime(0, 0, 0);

				$midnight	= $midnight->getTimestamp();

				if ($timestamp > $midnight + 86400)
				{
					$day = 'TOMORROW';
				}
				else if ($timestamp > $midnight)
				{
					$day = 'TODAY';
				}
				else if ($timestamp > $midnight - 86400)
				{
					$day = 'YESTERDAY';
				}

				if ($day !== false)
				{
					$format = self::_format_cache($format, $this->_user);

					// Format using the short formatting and finally swap out the relative token placeholder with the correct value
					return str_replace(self::RELATIVE_WRAPPER . self::RELATIVE_WRAPPER, $this->_user->lang['datetime'][$day], strtr(parent::format($format['format_short']), $format['lang']));
				}
			}
		}

		$format = self::_format_cache($format, $this->_user);

		return strtr(parent::format($format['format_long']), $format['lang']);
	}

	/**
	* Pre-processes the specified date format
	*
	* @param string $format Output format
	* @param user $user User object to use for localisation
	* @return array Processed date format
	*/
	static protected function _format_cache($format, $user)
	{
		$lang = $user->lang_name;

		if (!isset(self::$format_cache[$lang]))
		{
			self::$format_cache[$lang] = array();

			if (!isset(self::$format_cache[$lang][$format]))
			{
				// Is the user requesting a friendly date format (i.e. 'Today 12:42')?
				self::$format_cache[$lang][$format] = array(
					'is_short'		=> strpos($format, self::RELATIVE_WRAPPER) !== false,
					'format_short'	=> substr($format, 0, strpos($format, self::RELATIVE_WRAPPER)) . self::RELATIVE_WRAPPER . self::RELATIVE_WRAPPER . substr(strrchr($format, self::RELATIVE_WRAPPER), 1),
					'format_long'	=> str_replace(self::RELATIVE_WRAPPER, '', $format),
					'lang'			=> $user->lang['datetime'],
				);

				// Short representation of month in format? Some languages use different terms for the long and short format of May
				if ((strpos($format, '\M') === false && strpos($format, 'M') !== false) || (strpos($format, '\r') === false && strpos($format, 'r') !== false))
				{
					self::$format_cache[$lang][$format]['lang']['May'] = $user->lang['datetime']['May_short'];
				}
			}
		}

		return self::$format_cache[$lang][$format];
	}
}