blob: 7a0552b0b7d767f2e18f23d7199c8eb9bbf452ab [file] [log] [blame]
Andrey Andreevc5536aa2012-11-01 17:33:58 +02001<?php
Derek Allard2067d1a2008-11-13 22:59:24 +00002/**
3 * CodeIgniter
4 *
Phil Sturgeon07c1ac82012-03-09 17:03:37 +00005 * An open source application development framework for PHP 5.2.4 or newer
Derek Allard2067d1a2008-11-13 22:59:24 +00006 *
Derek Jonesf4a4bd82011-10-20 12:18:42 -05007 * NOTICE OF LICENSE
Eric Barnesdc3e4be2011-12-19 13:48:29 -05008 *
Derek Jonesf4a4bd82011-10-20 12:18:42 -05009 * Licensed under the Open Software License version 3.0
Eric Barnesdc3e4be2011-12-19 13:48:29 -050010 *
Derek Jonesf4a4bd82011-10-20 12:18:42 -050011 * This source file is subject to the Open Software License (OSL 3.0) that is
12 * bundled with this package in the files license.txt / license.rst. It is
13 * also available through the world wide web at this URL:
14 * http://opensource.org/licenses/OSL-3.0
15 * If you did not receive a copy of the license and are unable to obtain it
16 * through the world wide web, please send an email to
17 * licensing@ellislab.com so we can send you a copy immediately.
18 *
Derek Allard2067d1a2008-11-13 22:59:24 +000019 * @package CodeIgniter
Derek Jonesf4a4bd82011-10-20 12:18:42 -050020 * @author EllisLab Dev Team
Greg Aker0defe5d2012-01-01 18:46:41 -060021 * @copyright Copyright (c) 2008 - 2012, EllisLab, Inc. (http://ellislab.com/)
Derek Jonesf4a4bd82011-10-20 12:18:42 -050022 * @license http://opensource.org/licenses/OSL-3.0 Open Software License (OSL 3.0)
Derek Allard2067d1a2008-11-13 22:59:24 +000023 * @link http://codeigniter.com
24 * @since Version 1.0
25 * @filesource
26 */
Andrey Andreevc5536aa2012-11-01 17:33:58 +020027defined('BASEPATH') OR exit('No direct script access allowed');
Derek Allard2067d1a2008-11-13 22:59:24 +000028
Derek Allard2067d1a2008-11-13 22:59:24 +000029/**
30 * CodeIgniter Date Helpers
31 *
32 * @package CodeIgniter
33 * @subpackage Helpers
34 * @category Helpers
Derek Jonesf4a4bd82011-10-20 12:18:42 -050035 * @author EllisLab Dev Team
Derek Allard2067d1a2008-11-13 22:59:24 +000036 * @link http://codeigniter.com/user_guide/helpers/date_helper.html
37 */
38
39// ------------------------------------------------------------------------
40
Derek Allard2067d1a2008-11-13 22:59:24 +000041if ( ! function_exists('now'))
42{
Timothy Warren01b129a2012-04-27 11:36:50 -040043 /**
44 * Get "now" time
45 *
Iban Eguia74009652012-06-13 22:57:50 +020046 * Returns time() based on the timezone parameter or on the
47 * "time_reference" setting
Timothy Warren01b129a2012-04-27 11:36:50 -040048 *
Iban Eguia895e98c2012-06-08 23:01:31 +020049 * @param string
Timothy Warren01b129a2012-04-27 11:36:50 -040050 * @return int
51 */
Iban Eguia83105952012-03-27 18:18:15 +020052 function now($timezone = NULL)
Derek Allard2067d1a2008-11-13 22:59:24 +000053 {
Iban Eguiac88daba2012-06-11 13:58:30 +020054 if (empty($timezone))
Derek Allard2067d1a2008-11-13 22:59:24 +000055 {
Iban Eguiafeb14da2012-06-12 16:09:36 +020056 $timezone = config_item('time_reference');
Derek Allard2067d1a2008-11-13 22:59:24 +000057 }
Greg Akerc964e722011-08-29 19:31:29 -050058
Iban Eguiac88daba2012-06-11 13:58:30 +020059 if ($timezone === 'local' OR $timezone === date_default_timezone_get())
Iban Eguiae15e3dd2012-06-09 23:52:27 +020060 {
Iban Eguiac88daba2012-06-11 13:58:30 +020061 return time();
Iban Eguiae15e3dd2012-06-09 23:52:27 +020062 }
Derek Allard2067d1a2008-11-13 22:59:24 +000063
Iban Eguiac88daba2012-06-11 13:58:30 +020064 $datetime = new DateTime('now', new DateTimeZone($timezone));
65 sscanf($datetime->format('j-n-Y G:i:s'), '%d-%d-%d %d:%d:%d', $day, $month, $year, $hour, $minute, $second);
66
67 return mktime($hour, $minute, $second, $month, $day, $year);
Derek Allard2067d1a2008-11-13 22:59:24 +000068 }
69}
Barry Mienydd671972010-10-04 16:33:58 +020070
Derek Allard2067d1a2008-11-13 22:59:24 +000071// ------------------------------------------------------------------------
72
Derek Allard2067d1a2008-11-13 22:59:24 +000073if ( ! function_exists('mdate'))
74{
Timothy Warren01b129a2012-04-27 11:36:50 -040075 /**
76 * Convert MySQL Style Datecodes
77 *
78 * This function is identical to PHPs date() function,
79 * except that it allows date codes to be formatted using
80 * the MySQL style, where each code letter is preceded
81 * with a percent sign: %Y %m %d etc...
82 *
83 * The benefit of doing dates this way is that you don't
84 * have to worry about escaping your text letters that
85 * match the date codes.
86 *
87 * @param string
88 * @param int
89 * @return int
90 */
Derek Allard2067d1a2008-11-13 22:59:24 +000091 function mdate($datestr = '', $time = '')
92 {
Alex Bilbie773ccc32012-06-02 11:11:08 +010093 if ($datestr === '')
Greg Akerf9168392011-08-29 19:29:05 -050094 {
Eric Barnesdc3e4be2011-12-19 13:48:29 -050095 return '';
Greg Akerf9168392011-08-29 19:29:05 -050096 }
Andrey Andreeveef24062012-06-14 16:17:48 +030097 elseif (empty($time))
98 {
99 $time = now();
100 }
Barry Mienydd671972010-10-04 16:33:58 +0200101
Greg Akerf9168392011-08-29 19:29:05 -0500102 $datestr = str_replace(
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500103 '%\\',
104 '',
Andrey Andreevae31eb52012-05-17 14:54:15 +0300105 preg_replace('/([a-z]+?){1}/i', '\\\\\\1', $datestr)
Greg Akerf9168392011-08-29 19:29:05 -0500106 );
Greg Akerc964e722011-08-29 19:31:29 -0500107
Derek Allard2067d1a2008-11-13 22:59:24 +0000108 return date($datestr, $time);
109 }
110}
Barry Mienydd671972010-10-04 16:33:58 +0200111
Derek Allard2067d1a2008-11-13 22:59:24 +0000112// ------------------------------------------------------------------------
113
Derek Allard2067d1a2008-11-13 22:59:24 +0000114if ( ! function_exists('standard_date'))
115{
Timothy Warren01b129a2012-04-27 11:36:50 -0400116 /**
117 * Standard Date
118 *
119 * Returns a date formatted according to the submitted standard.
120 *
Andrey Andreevac570332012-07-04 13:04:10 +0300121 * As of PHP 5.2, the DateTime extension provides constants that
122 * serve for the exact same purpose and are used with date().
Andrey Andreevac570332012-07-04 13:04:10 +0300123 *
Andrey Andreev29d909d2012-10-27 01:05:09 +0300124 * @todo Remove in version 3.1+.
125 * @deprecated 3.0.0 Use PHP's native date() instead.
126 * @link http://www.php.net/manual/en/class.datetime.php#datetime.constants.types
Andrey Andreevac570332012-07-04 13:04:10 +0300127 *
Andrey Andreev29d909d2012-10-27 01:05:09 +0300128 * @example date(DATE_RFC822, now()); // default
129 * @example date(DATE_W3C, $time); // a different format and time
Andrey Andreevac570332012-07-04 13:04:10 +0300130 *
Andrey Andreev29d909d2012-10-27 01:05:09 +0300131 * @param string $fmt = 'DATE_RFC822' the chosen format
132 * @param int $time = NULL Unix timestamp
Timothy Warren01b129a2012-04-27 11:36:50 -0400133 * @return string
134 */
Andrey Andreevdd6f32b2012-07-04 10:08:54 +0300135 function standard_date($fmt = 'DATE_RFC822', $time = NULL)
Derek Allard2067d1a2008-11-13 22:59:24 +0000136 {
Andrey Andreevdd6f32b2012-07-04 10:08:54 +0300137 if (empty($time))
138 {
139 $time = now();
140 }
Derek Allard2067d1a2008-11-13 22:59:24 +0000141
Andrey Andreevdd6f32b2012-07-04 10:08:54 +0300142 // Procedural style pre-defined constants from the DateTime extension
143 if (strpos($fmt, 'DATE_') !== 0 OR defined($fmt) === FALSE)
Derek Allard2067d1a2008-11-13 22:59:24 +0000144 {
145 return FALSE;
146 }
Barry Mienydd671972010-10-04 16:33:58 +0200147
Andrey Andreevdd6f32b2012-07-04 10:08:54 +0300148 return date(constant($fmt), $time);
Derek Allard2067d1a2008-11-13 22:59:24 +0000149 }
150}
Barry Mienydd671972010-10-04 16:33:58 +0200151
Derek Allard2067d1a2008-11-13 22:59:24 +0000152// ------------------------------------------------------------------------
153
Derek Allard2067d1a2008-11-13 22:59:24 +0000154if ( ! function_exists('timespan'))
155{
Timothy Warren01b129a2012-04-27 11:36:50 -0400156 /**
157 * Timespan
158 *
159 * Returns a span of seconds in this format:
160 * 10 days 14 hours 36 minutes 47 seconds
161 *
162 * @param int a number of seconds
163 * @param int Unix timestamp
164 * @param int a number of display units
165 * @return string
166 */
Roger Herbert8d69aa12012-03-14 08:44:55 +0000167 function timespan($seconds = 1, $time = '', $units = 7)
Derek Allard2067d1a2008-11-13 22:59:24 +0000168 {
169 $CI =& get_instance();
170 $CI->lang->load('date');
171
Andrey Andreeveef24062012-06-14 16:17:48 +0300172 is_numeric($seconds) OR $seconds = 1;
173 is_numeric($time) OR $time = time();
174 is_numeric($units) OR $units = 7;
Roger Herbert04c146d2012-03-11 15:31:01 +0000175
Greg Akerf9168392011-08-29 19:29:05 -0500176 $seconds = ($time <= $seconds) ? 1 : $time - $seconds;
Barry Mienydd671972010-10-04 16:33:58 +0200177
Roger Herbert04c146d2012-03-11 15:31:01 +0000178 $str = array();
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500179 $years = floor($seconds / 31557600);
Barry Mienydd671972010-10-04 16:33:58 +0200180
Derek Allard2067d1a2008-11-13 22:59:24 +0000181 if ($years > 0)
Barry Mienydd671972010-10-04 16:33:58 +0200182 {
Andrey Andreeveef24062012-06-14 16:17:48 +0300183 $str[] = $years.' '.$CI->lang->line($years > 1 ? 'date_years' : 'date_year');
Barry Mienydd671972010-10-04 16:33:58 +0200184 }
185
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500186 $seconds -= $years * 31557600;
187 $months = floor($seconds / 2629743);
Barry Mienydd671972010-10-04 16:33:58 +0200188
Roger Herbertb8fb66b2012-03-11 16:15:15 +0000189 if (count($str) < $units && ($years > 0 OR $months > 0))
Derek Allard2067d1a2008-11-13 22:59:24 +0000190 {
191 if ($months > 0)
Barry Mienydd671972010-10-04 16:33:58 +0200192 {
Andrey Andreeveef24062012-06-14 16:17:48 +0300193 $str[] = $months.' '.$CI->lang->line($months > 1 ? 'date_months' : 'date_month');
Barry Mienydd671972010-10-04 16:33:58 +0200194 }
195
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500196 $seconds -= $months * 2629743;
Derek Allard2067d1a2008-11-13 22:59:24 +0000197 }
198
199 $weeks = floor($seconds / 604800);
Barry Mienydd671972010-10-04 16:33:58 +0200200
Roger Herbertb8fb66b2012-03-11 16:15:15 +0000201 if (count($str) < $units && ($years > 0 OR $months > 0 OR $weeks > 0))
Derek Allard2067d1a2008-11-13 22:59:24 +0000202 {
203 if ($weeks > 0)
Barry Mienydd671972010-10-04 16:33:58 +0200204 {
Andrey Andreeveef24062012-06-14 16:17:48 +0300205 $str[] = $weeks.' '.$CI->lang->line($weeks > 1 ? 'date_weeks' : 'date_week');
Derek Allard2067d1a2008-11-13 22:59:24 +0000206 }
Barry Mienydd671972010-10-04 16:33:58 +0200207
Derek Allard2067d1a2008-11-13 22:59:24 +0000208 $seconds -= $weeks * 604800;
Barry Mienydd671972010-10-04 16:33:58 +0200209 }
Derek Allard2067d1a2008-11-13 22:59:24 +0000210
211 $days = floor($seconds / 86400);
Barry Mienydd671972010-10-04 16:33:58 +0200212
Roger Herbertb8fb66b2012-03-11 16:15:15 +0000213 if (count($str) < $units && ($months > 0 OR $weeks > 0 OR $days > 0))
Derek Allard2067d1a2008-11-13 22:59:24 +0000214 {
215 if ($days > 0)
Barry Mienydd671972010-10-04 16:33:58 +0200216 {
Andrey Andreeveef24062012-06-14 16:17:48 +0300217 $str[] = $days.' '.$CI->lang->line($days > 1 ? 'date_days' : 'date_day');
Derek Allard2067d1a2008-11-13 22:59:24 +0000218 }
Barry Mienydd671972010-10-04 16:33:58 +0200219
Derek Allard2067d1a2008-11-13 22:59:24 +0000220 $seconds -= $days * 86400;
221 }
Barry Mienydd671972010-10-04 16:33:58 +0200222
Derek Allard2067d1a2008-11-13 22:59:24 +0000223 $hours = floor($seconds / 3600);
Barry Mienydd671972010-10-04 16:33:58 +0200224
Roger Herbertb8fb66b2012-03-11 16:15:15 +0000225 if (count($str) < $units && ($days > 0 OR $hours > 0))
Derek Allard2067d1a2008-11-13 22:59:24 +0000226 {
227 if ($hours > 0)
228 {
Andrey Andreeveef24062012-06-14 16:17:48 +0300229 $str[] = $hours.' '.$CI->lang->line($hours > 1 ? 'date_hours' : 'date_hour');
Derek Allard2067d1a2008-11-13 22:59:24 +0000230 }
Barry Mienydd671972010-10-04 16:33:58 +0200231
Derek Allard2067d1a2008-11-13 22:59:24 +0000232 $seconds -= $hours * 3600;
233 }
Barry Mienydd671972010-10-04 16:33:58 +0200234
Derek Allard2067d1a2008-11-13 22:59:24 +0000235 $minutes = floor($seconds / 60);
Barry Mienydd671972010-10-04 16:33:58 +0200236
Roger Herbertb8fb66b2012-03-11 16:15:15 +0000237 if (count($str) < $units && ($days > 0 OR $hours > 0 OR $minutes > 0))
Derek Allard2067d1a2008-11-13 22:59:24 +0000238 {
239 if ($minutes > 0)
Barry Mienydd671972010-10-04 16:33:58 +0200240 {
Andrey Andreeveef24062012-06-14 16:17:48 +0300241 $str[] = $minutes.' '.$CI->lang->line($minutes > 1 ? 'date_minutes' : 'date_minute');
Derek Allard2067d1a2008-11-13 22:59:24 +0000242 }
Barry Mienydd671972010-10-04 16:33:58 +0200243
Derek Allard2067d1a2008-11-13 22:59:24 +0000244 $seconds -= $minutes * 60;
245 }
Barry Mienydd671972010-10-04 16:33:58 +0200246
Roger Herbert597eb212012-03-14 09:06:17 +0000247 if (count($str) === 0)
Derek Allard2067d1a2008-11-13 22:59:24 +0000248 {
Andrey Andreeveef24062012-06-14 16:17:48 +0300249 $str[] = $seconds.' '.$CI->lang->line($seconds > 1 ? 'date_seconds' : 'date_second');
Derek Allard2067d1a2008-11-13 22:59:24 +0000250 }
Barry Mienydd671972010-10-04 16:33:58 +0200251
Roger Herbert04c146d2012-03-11 15:31:01 +0000252 return implode(', ', $str);
Derek Allard2067d1a2008-11-13 22:59:24 +0000253 }
254}
Barry Mienydd671972010-10-04 16:33:58 +0200255
Derek Allard2067d1a2008-11-13 22:59:24 +0000256// ------------------------------------------------------------------------
257
Derek Allard2067d1a2008-11-13 22:59:24 +0000258if ( ! function_exists('days_in_month'))
259{
Timothy Warren01b129a2012-04-27 11:36:50 -0400260 /**
261 * Number of days in a month
262 *
263 * Takes a month/year as input and returns the number of days
264 * for the given month/year. Takes leap years into consideration.
265 *
266 * @param int a numeric month
267 * @param int a numeric year
268 * @return int
269 */
Derek Allard2067d1a2008-11-13 22:59:24 +0000270 function days_in_month($month = 0, $year = '')
271 {
272 if ($month < 1 OR $month > 12)
273 {
274 return 0;
275 }
Andrey Andreeveef24062012-06-14 16:17:48 +0300276 elseif ( ! is_numeric($year) OR strlen($year) !== 4)
Derek Allard2067d1a2008-11-13 22:59:24 +0000277 {
278 $year = date('Y');
279 }
Barry Mienydd671972010-10-04 16:33:58 +0200280
Andrey Andreeveef24062012-06-14 16:17:48 +0300281 if ($year >= 1970)
282 {
283 return (int) date('t', mktime(12, 0, 0, $month, 1, $year));
284 }
285
Derek Allard2067d1a2008-11-13 22:59:24 +0000286 if ($month == 2)
287 {
Alex Bilbie773ccc32012-06-02 11:11:08 +0100288 if ($year % 400 === 0 OR ($year % 4 === 0 && $year % 100 !== 0))
Derek Allard2067d1a2008-11-13 22:59:24 +0000289 {
290 return 29;
291 }
292 }
293
294 $days_in_month = array(31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31);
295 return $days_in_month[$month - 1];
296 }
297}
Derek Jones36591092010-03-05 10:05:51 -0600298
Derek Allard2067d1a2008-11-13 22:59:24 +0000299// ------------------------------------------------------------------------
300
Derek Allard2067d1a2008-11-13 22:59:24 +0000301if ( ! function_exists('local_to_gmt'))
302{
Timothy Warren01b129a2012-04-27 11:36:50 -0400303 /**
304 * Converts a local Unix timestamp to GMT
305 *
306 * @param int Unix timestamp
307 * @return int
308 */
Derek Allard2067d1a2008-11-13 22:59:24 +0000309 function local_to_gmt($time = '')
310 {
Alex Bilbie773ccc32012-06-02 11:11:08 +0100311 if ($time === '')
Greg Akerf9168392011-08-29 19:29:05 -0500312 {
Derek Allard2067d1a2008-11-13 22:59:24 +0000313 $time = time();
Greg Akerf9168392011-08-29 19:29:05 -0500314 }
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500315
Greg Akerf9168392011-08-29 19:29:05 -0500316 return mktime(
Andrey Andreevb089e152012-06-16 19:11:40 +0300317 gmdate('G', $time),
Andrey Andreevae31eb52012-05-17 14:54:15 +0300318 gmdate('i', $time),
319 gmdate('s', $time),
Andrey Andreevb089e152012-06-16 19:11:40 +0300320 gmdate('n', $time),
321 gmdate('j', $time),
Andrey Andreevae31eb52012-05-17 14:54:15 +0300322 gmdate('Y', $time)
Greg Akerf9168392011-08-29 19:29:05 -0500323 );
Derek Allard2067d1a2008-11-13 22:59:24 +0000324 }
325}
Barry Mienydd671972010-10-04 16:33:58 +0200326
Derek Allard2067d1a2008-11-13 22:59:24 +0000327// ------------------------------------------------------------------------
328
Derek Allard2067d1a2008-11-13 22:59:24 +0000329if ( ! function_exists('gmt_to_local'))
330{
Timothy Warren01b129a2012-04-27 11:36:50 -0400331 /**
332 * Converts GMT time to a localized value
333 *
334 * Takes a Unix timestamp (in GMT) as input, and returns
335 * at the local value based on the timezone and DST setting
336 * submitted
337 *
338 * @param int Unix timestamp
339 * @param string timezone
340 * @param bool whether DST is active
341 * @return int
342 */
Derek Allard2067d1a2008-11-13 22:59:24 +0000343 function gmt_to_local($time = '', $timezone = 'UTC', $dst = FALSE)
Barry Mienydd671972010-10-04 16:33:58 +0200344 {
Alex Bilbie773ccc32012-06-02 11:11:08 +0100345 if ($time === '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000346 {
347 return now();
348 }
Barry Mienydd671972010-10-04 16:33:58 +0200349
Derek Allard2067d1a2008-11-13 22:59:24 +0000350 $time += timezones($timezone) * 3600;
351
Andrey Andreeveef24062012-06-14 16:17:48 +0300352 return ($dst === TRUE) ? $time + 3600 : $time;
Derek Allard2067d1a2008-11-13 22:59:24 +0000353 }
354}
Derek Jones36591092010-03-05 10:05:51 -0600355
Derek Allard2067d1a2008-11-13 22:59:24 +0000356// ------------------------------------------------------------------------
357
Derek Allard2067d1a2008-11-13 22:59:24 +0000358if ( ! function_exists('mysql_to_unix'))
359{
Timothy Warren01b129a2012-04-27 11:36:50 -0400360 /**
361 * Converts a MySQL Timestamp to Unix
362 *
363 * @param int Unix timestamp
364 * @return int
365 */
Derek Allard2067d1a2008-11-13 22:59:24 +0000366 function mysql_to_unix($time = '')
367 {
368 // We'll remove certain characters for backward compatibility
369 // since the formatting changed with MySQL 4.1
370 // YYYY-MM-DD HH:MM:SS
Barry Mienydd671972010-10-04 16:33:58 +0200371
Andrey Andreev3bbbd262012-06-14 13:35:32 +0300372 $time = str_replace(array('-', ':', ' '), '', $time);
Barry Mienydd671972010-10-04 16:33:58 +0200373
Derek Allard2067d1a2008-11-13 22:59:24 +0000374 // YYYYMMDDHHMMSS
Greg Akerc964e722011-08-29 19:31:29 -0500375 return mktime(
376 substr($time, 8, 2),
377 substr($time, 10, 2),
378 substr($time, 12, 2),
379 substr($time, 4, 2),
380 substr($time, 6, 2),
381 substr($time, 0, 4)
382 );
Derek Allard2067d1a2008-11-13 22:59:24 +0000383 }
384}
Barry Mienydd671972010-10-04 16:33:58 +0200385
Derek Allard2067d1a2008-11-13 22:59:24 +0000386// ------------------------------------------------------------------------
387
Derek Allard2067d1a2008-11-13 22:59:24 +0000388if ( ! function_exists('unix_to_human'))
389{
Timothy Warren01b129a2012-04-27 11:36:50 -0400390 /**
391 * Unix to "Human"
392 *
393 * Formats Unix timestamp to the following prototype: 2006-08-21 11:35 PM
394 *
395 * @param int Unix timestamp
396 * @param bool whether to show seconds
397 * @param string format: us or euro
398 * @return string
399 */
Derek Allard2067d1a2008-11-13 22:59:24 +0000400 function unix_to_human($time = '', $seconds = FALSE, $fmt = 'us')
401 {
Andrey Andreeveef24062012-06-14 16:17:48 +0300402 $r = date('Y', $time).'-'.date('m', $time).'-'.date('d', $time).' ';
Barry Mienydd671972010-10-04 16:33:58 +0200403
Alex Bilbie773ccc32012-06-02 11:11:08 +0100404 if ($fmt === 'us')
Derek Allard2067d1a2008-11-13 22:59:24 +0000405 {
406 $r .= date('h', $time).':'.date('i', $time);
407 }
408 else
409 {
410 $r .= date('H', $time).':'.date('i', $time);
411 }
Barry Mienydd671972010-10-04 16:33:58 +0200412
Derek Allard2067d1a2008-11-13 22:59:24 +0000413 if ($seconds)
414 {
415 $r .= ':'.date('s', $time);
416 }
Barry Mienydd671972010-10-04 16:33:58 +0200417
Alex Bilbie773ccc32012-06-02 11:11:08 +0100418 if ($fmt === 'us')
Derek Allard2067d1a2008-11-13 22:59:24 +0000419 {
Andrey Andreeveef24062012-06-14 16:17:48 +0300420 return $r.' '.date('A', $time);
Derek Allard2067d1a2008-11-13 22:59:24 +0000421 }
Barry Mienydd671972010-10-04 16:33:58 +0200422
Derek Allard2067d1a2008-11-13 22:59:24 +0000423 return $r;
424 }
425}
Barry Mienydd671972010-10-04 16:33:58 +0200426
Derek Allard2067d1a2008-11-13 22:59:24 +0000427// ------------------------------------------------------------------------
428
Derek Allard2067d1a2008-11-13 22:59:24 +0000429if ( ! function_exists('human_to_unix'))
430{
Timothy Warren01b129a2012-04-27 11:36:50 -0400431 /**
432 * Convert "human" date to GMT
433 *
434 * Reverses the above process
435 *
436 * @param string format: us or euro
437 * @return int
438 */
Derek Allard2067d1a2008-11-13 22:59:24 +0000439 function human_to_unix($datestr = '')
440 {
Alex Bilbie773ccc32012-06-02 11:11:08 +0100441 if ($datestr === '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000442 {
443 return FALSE;
444 }
Barry Mienydd671972010-10-04 16:33:58 +0200445
Andrey Andreevae31eb52012-05-17 14:54:15 +0300446 $datestr = preg_replace('/\040+/', ' ', trim($datestr));
Barry Mienydd671972010-10-04 16:33:58 +0200447
Andrey Andreevf11a1932012-06-14 16:35:09 +0300448 if ( ! preg_match('/^(\d{2}|\d{4})\-[0-9]{1,2}\-[0-9]{1,2}\s[0-9]{1,2}:[0-9]{1,2}(?::[0-9]{1,2})?(?:\s[AP]M)?$/i', $datestr))
Derek Allard2067d1a2008-11-13 22:59:24 +0000449 {
450 return FALSE;
451 }
Barry Mienydd671972010-10-04 16:33:58 +0200452
Andrey Andreeve24eed72012-11-02 23:33:45 +0200453 sscanf($datestr, '%d-%d-%d %s %s', $year, $month, $day, $time, $ampm);
454 sscanf($time, '%d:%d:%d', $hour, $min, $sec);
455 isset($sec) OR $sec = 0;
Derek Allard2067d1a2008-11-13 22:59:24 +0000456
Andrey Andreeve24eed72012-11-02 23:33:45 +0200457 if (isset($ampm))
Derek Allard2067d1a2008-11-13 22:59:24 +0000458 {
Andrey Andreeve24eed72012-11-02 23:33:45 +0200459 $ampm = strtolower($ampm);
Barry Mienydd671972010-10-04 16:33:58 +0200460
Andrey Andreevf11a1932012-06-14 16:35:09 +0300461 if ($ampm[0] === 'p' && $hour < 12)
Greg Akerf9168392011-08-29 19:29:05 -0500462 {
Andrey Andreevae31eb52012-05-17 14:54:15 +0300463 $hour += 12;
Greg Akerf9168392011-08-29 19:29:05 -0500464 }
Andrey Andreevf11a1932012-06-14 16:35:09 +0300465 elseif ($ampm[0] === 'a' && $hour === 12)
Greg Akerf9168392011-08-29 19:29:05 -0500466 {
Andrey Andreevf11a1932012-06-14 16:35:09 +0300467 $hour = 0;
Greg Akerf9168392011-08-29 19:29:05 -0500468 }
Derek Allard2067d1a2008-11-13 22:59:24 +0000469 }
Barry Mienydd671972010-10-04 16:33:58 +0200470
Derek Allard2067d1a2008-11-13 22:59:24 +0000471 return mktime($hour, $min, $sec, $month, $day, $year);
472 }
473}
Barry Mienydd671972010-10-04 16:33:58 +0200474
Derek Allard2067d1a2008-11-13 22:59:24 +0000475// ------------------------------------------------------------------------
476
Kyle Farris896d95a2011-08-21 23:03:54 -0300477if ( ! function_exists('nice_date'))
478{
Timothy Warren01b129a2012-04-27 11:36:50 -0400479 /**
480 * Turns many "reasonably-date-like" strings into something
481 * that is actually useful. This only works for dates after unix epoch.
482 *
483 * @param string The terribly formatted date-like string
484 * @param string Date format to return (same as php date function)
485 * @return string
486 */
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500487 function nice_date($bad_date = '', $format = FALSE)
Kyle Farris896d95a2011-08-21 23:03:54 -0300488 {
489 if (empty($bad_date))
490 {
491 return 'Unknown';
492 }
Andrey Andreevd9b44be2012-06-15 16:07:08 +0300493 elseif (empty($format))
494 {
495 $format = 'U';
496 }
Greg Akerf9168392011-08-29 19:29:05 -0500497
Kyle Farris896d95a2011-08-21 23:03:54 -0300498 // Date like: YYYYMM
Andrey Andreevf11a1932012-06-14 16:35:09 +0300499 if (preg_match('/^\d{6}$/i', $bad_date))
Kyle Farris896d95a2011-08-21 23:03:54 -0300500 {
Andrey Andreevae31eb52012-05-17 14:54:15 +0300501 if (in_array(substr($bad_date, 0, 2), array('19', '20')))
Kyle Farris896d95a2011-08-21 23:03:54 -0300502 {
503 $year = substr($bad_date, 0, 4);
504 $month = substr($bad_date, 4, 2);
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500505 }
506 else
Kyle Farris896d95a2011-08-21 23:03:54 -0300507 {
508 $month = substr($bad_date, 0, 2);
509 $year = substr($bad_date, 2, 4);
510 }
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500511
Andrey Andreevae31eb52012-05-17 14:54:15 +0300512 return date($format, strtotime($year.'-'.$month.'-01'));
Kyle Farris896d95a2011-08-21 23:03:54 -0300513 }
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500514
Kyle Farris896d95a2011-08-21 23:03:54 -0300515 // Date Like: YYYYMMDD
Andrey Andreevf11a1932012-06-14 16:35:09 +0300516 if (preg_match('/^(\d{2})\d{2}(\d{4})$/i', $bad_date, $matches))
Kyle Farris896d95a2011-08-21 23:03:54 -0300517 {
Andrey Andreevf11a1932012-06-14 16:35:09 +0300518 return date($format, strtotime($matches[1].'/01/'.$matches[2]));
Kyle Farris896d95a2011-08-21 23:03:54 -0300519 }
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500520
Kyle Farris896d95a2011-08-21 23:03:54 -0300521 // Date Like: MM-DD-YYYY __or__ M-D-YYYY (or anything in between)
Andrey Andreevf11a1932012-06-14 16:35:09 +0300522 if (preg_match('/^(\d{1,2})-(\d{1,2})-(\d{4})$/i', $bad_date, $matches))
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500523 {
Andrey Andreevf11a1932012-06-14 16:35:09 +0300524 return date($format, strtotime($matches[3].'-'.$matches[1].'-'.$matches[2]));
Kyle Farris896d95a2011-08-21 23:03:54 -0300525 }
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500526
Kyle Farris896d95a2011-08-21 23:03:54 -0300527 // Any other kind of string, when converted into UNIX time,
528 // produces "0 seconds after epoc..." is probably bad...
529 // return "Invalid Date".
Alex Bilbie773ccc32012-06-02 11:11:08 +0100530 if (date('U', strtotime($bad_date)) === '0')
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500531 {
Andrey Andreevae31eb52012-05-17 14:54:15 +0300532 return 'Invalid Date';
Kyle Farris896d95a2011-08-21 23:03:54 -0300533 }
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500534
Kyle Farris896d95a2011-08-21 23:03:54 -0300535 // It's probably a valid-ish date format already
536 return date($format, strtotime($bad_date));
537 }
538}
539
540// ------------------------------------------------------------------------
541
Derek Allard2067d1a2008-11-13 22:59:24 +0000542if ( ! function_exists('timezone_menu'))
543{
Timothy Warren01b129a2012-04-27 11:36:50 -0400544 /**
545 * Timezone Menu
546 *
547 * Generates a drop-down menu of timezones.
548 *
549 * @param string timezone
550 * @param string classname
551 * @param string menu name
Mat Whitney7540ded2012-06-22 12:02:10 -0700552 * @param mixed attributes
Timothy Warren01b129a2012-04-27 11:36:50 -0400553 * @return string
554 */
Mat Whitney7540ded2012-06-22 12:02:10 -0700555 function timezone_menu($default = 'UTC', $class = '', $name = 'timezones', $attributes = '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000556 {
557 $CI =& get_instance();
558 $CI->lang->load('date');
Barry Mienydd671972010-10-04 16:33:58 +0200559
Alex Bilbie773ccc32012-06-02 11:11:08 +0100560 $default = ($default === 'GMT') ? 'UTC' : $default;
Derek Allard2067d1a2008-11-13 22:59:24 +0000561
562 $menu = '<select name="'.$name.'"';
Barry Mienydd671972010-10-04 16:33:58 +0200563
Alex Bilbie773ccc32012-06-02 11:11:08 +0100564 if ($class !== '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000565 {
566 $menu .= ' class="'.$class.'"';
567 }
Barry Mienydd671972010-10-04 16:33:58 +0200568
Eric Barnesacedd2b2012-07-29 00:15:40 -0400569 $menu .= _stringify_attributes($attributes).">\n";
Barry Mienydd671972010-10-04 16:33:58 +0200570
Derek Allard2067d1a2008-11-13 22:59:24 +0000571 foreach (timezones() as $key => $val)
572 {
Alex Bilbie773ccc32012-06-02 11:11:08 +0100573 $selected = ($default === $key) ? ' selected="selected"' : '';
Andrey Andreevae31eb52012-05-17 14:54:15 +0300574 $menu .= '<option value="'.$key.'"'.$selected.'>'.$CI->lang->line($key)."</option>\n";
Derek Allard2067d1a2008-11-13 22:59:24 +0000575 }
576
Andrey Andreevae31eb52012-05-17 14:54:15 +0300577 return $menu.'</select>';
Derek Allard2067d1a2008-11-13 22:59:24 +0000578 }
579}
Barry Mienydd671972010-10-04 16:33:58 +0200580
Derek Allard2067d1a2008-11-13 22:59:24 +0000581// ------------------------------------------------------------------------
582
Derek Allard2067d1a2008-11-13 22:59:24 +0000583if ( ! function_exists('timezones'))
584{
Timothy Warren01b129a2012-04-27 11:36:50 -0400585 /**
586 * Timezones
587 *
588 * Returns an array of timezones. This is a helper function
589 * for various other ones in this library
590 *
591 * @param string timezone
592 * @return string
593 */
Derek Allard2067d1a2008-11-13 22:59:24 +0000594 function timezones($tz = '')
595 {
596 // Note: Don't change the order of these even though
597 // some items appear to be in the wrong order
Barry Mienydd671972010-10-04 16:33:58 +0200598
599 $zones = array(
Greg Akerc964e722011-08-29 19:31:29 -0500600 'UM12' => -12,
601 'UM11' => -11,
602 'UM10' => -10,
603 'UM95' => -9.5,
604 'UM9' => -9,
605 'UM8' => -8,
606 'UM7' => -7,
607 'UM6' => -6,
608 'UM5' => -5,
609 'UM45' => -4.5,
610 'UM4' => -4,
611 'UM35' => -3.5,
612 'UM3' => -3,
613 'UM2' => -2,
614 'UM1' => -1,
615 'UTC' => 0,
616 'UP1' => +1,
617 'UP2' => +2,
618 'UP3' => +3,
619 'UP35' => +3.5,
620 'UP4' => +4,
621 'UP45' => +4.5,
622 'UP5' => +5,
623 'UP55' => +5.5,
624 'UP575' => +5.75,
625 'UP6' => +6,
626 'UP65' => +6.5,
627 'UP7' => +7,
628 'UP8' => +8,
629 'UP875' => +8.75,
630 'UP9' => +9,
631 'UP95' => +9.5,
632 'UP10' => +10,
633 'UP105' => +10.5,
634 'UP11' => +11,
635 'UP115' => +11.5,
636 'UP12' => +12,
637 'UP1275' => +12.75,
638 'UP13' => +13,
639 'UP14' => +14
640 );
Barry Mienydd671972010-10-04 16:33:58 +0200641
Alex Bilbie773ccc32012-06-02 11:11:08 +0100642 if ($tz === '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000643 {
644 return $zones;
645 }
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500646
Andrey Andreeve92df332012-03-26 22:44:20 +0300647 return isset($zones[$tz]) ? $zones[$tz] : 0;
Derek Allard2067d1a2008-11-13 22:59:24 +0000648 }
649}
650
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200651// ------------------------------------------------------------------------
652
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200653if ( ! function_exists('date_range'))
654{
Andrey Andreev14aa3172012-05-02 13:27:30 +0300655 /**
656 * Date range
657 *
658 * Returns a list of dates within a specified period.
659 *
660 * @param int unix_start UNIX timestamp of period start date
661 * @param int unix_end|days UNIX timestamp of period end date
662 * or interval in days.
Andrey Andreev2f8bf9b2012-10-12 20:37:52 +0300663 * @param mixed is_unix Specifies whether the second parameter
Andrey Andreev14aa3172012-05-02 13:27:30 +0300664 * is a UNIX timestamp or a day interval
665 * - TRUE or 'unix' for a timestamp
666 * - FALSE or 'days' for an interval
667 * @param string date_format Output date format, same as in date()
668 * @return array
669 */
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200670 function date_range($unix_start = '', $mixed = '', $is_unix = TRUE, $format = 'Y-m-d')
671 {
672 if ($unix_start == '' OR $mixed == '' OR $format == '')
673 {
674 return FALSE;
675 }
676
677 $is_unix = ! ( ! $is_unix OR $is_unix === 'days');
678
679 // Validate input and try strtotime() on invalid timestamps/intervals, just in case
Andrey Andreev7a7ad782012-11-12 17:21:01 +0200680 if ( ( ! ctype_digit((string) $unix_start) && ($unix_start = @strtotime($unix_time)) === FALSE)
681 OR ( ! ctype_digit((string) $mixed) && ($is_unix === FALSE OR ($mixed = @strtotime($mixed)) === FALSE))
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200682 OR ($is_unix === TRUE && $mixed < $unix_start))
683 {
684 return FALSE;
685 }
686
687 if ($is_unix && ($unix_start == $mixed OR date($format, $unix_start) === date($format, $mixed)))
688 {
689 return array($start_date);
690 }
691
692 $range = array();
693
Andrey Andreev30da39b2012-03-10 15:49:17 +0200694 /* NOTE: Even though the DateTime object has many useful features, it appears that
695 * it doesn't always handle properly timezones, when timestamps are passed
696 * directly to its constructor. Neither of the following gave proper results:
697 *
698 * new DateTime('<timestamp>')
699 * new DateTime('<timestamp>', '<timezone>')
700 *
701 * --- available in PHP 5.3:
702 *
703 * DateTime::createFromFormat('<format>', '<timestamp>')
704 * DateTime::createFromFormat('<format>', '<timestamp>', '<timezone')
705 *
706 * ... so we'll have to set the timestamp after the object is instantiated.
707 * Furthermore, in PHP 5.3 we can use DateTime::setTimestamp() to do that and
708 * given that we have UNIX timestamps - we should use it.
709 */
710 $from = new DateTime();
711
712 if (is_php('5.3'))
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200713 {
Andrey Andreev30da39b2012-03-10 15:49:17 +0200714 $from->setTimestamp($unix_start);
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200715 if ($is_unix)
716 {
717 $arg = new DateTime();
Andrey Andreev30da39b2012-03-10 15:49:17 +0200718 $arg->setTimestamp($mixed);
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200719 }
720 else
721 {
722 $arg = (int) $mixed;
723 }
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200724
Andrey Andreev30da39b2012-03-10 15:49:17 +0200725 $period = new DatePeriod($from, new DateInterval('P1D'), $arg);
726 foreach ($period as $date)
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200727 {
Andrey Andreev30da39b2012-03-10 15:49:17 +0200728 $range[] = $date->format($format);
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200729 }
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200730
Andrey Andreev30da39b2012-03-10 15:49:17 +0200731 /* If a period end date was passed to the DatePeriod constructor, it might not
732 * be in our results. Not sure if this is a bug or it's just possible because
733 * the end date might actually be less than 24 hours away from the previously
734 * generated DateTime object, but either way - we have to append it manually.
735 */
736 if ( ! is_int($arg) && $range[count($range) - 1] !== $arg->format($format))
737 {
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200738 $range[] = $arg->format($format);
739 }
740
741 return $range;
742 }
743
Andrey Andreev30da39b2012-03-10 15:49:17 +0200744 $from->setDate(date('Y', $unix_start), date('n', $unix_start), date('j', $unix_start));
745 $from->setTime(date('G', $unix_start), date('i', $unix_start), date('s', $unix_start));
746 if ($is_unix)
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200747 {
Andrey Andreev30da39b2012-03-10 15:49:17 +0200748 $arg = new DateTime();
749 $arg->setDate(date('Y', $mixed), date('n', $mixed), date('j', $mixed));
750 $arg->setTime(date('G', $mixed), date('i', $mixed), date('s', $mixed));
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200751 }
Andrey Andreev30da39b2012-03-10 15:49:17 +0200752 else
753 {
754 $arg = (int) $mixed;
755 }
756 $range[] = $from->format($format);
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200757
Andrey Andreev30da39b2012-03-10 15:49:17 +0200758 if (is_int($arg)) // Day intervals
759 {
760 do
761 {
762 $from->modify('+1 day');
763 $range[] = $from->format($format);
764 }
765 while (--$arg > 0);
766 }
767 else // end date UNIX timestamp
768 {
769 for ($from->modify('+1 day'), $end_check = $arg->format('Ymd'); $from->format('Ymd') < $end_check; $from->modify('+1 day'))
770 {
771 $range[] = $from->format($format);
772 }
773
774 // Our loop only appended dates prior to our end date
775 $range[] = $arg->format($format);
776 }
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200777
778 return $range;
779 }
780}
781
Derek Allard2067d1a2008-11-13 22:59:24 +0000782/* End of file date_helper.php */
Derek Jonesa3ffbbb2008-05-11 18:18:29 +0000783/* Location: ./system/helpers/date_helper.php */