blob: 5f05f070172abd79b76012238a48727ae117f31b [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
Derek Jones36591092010-03-05 10:05:51 -0600453 $split = explode(' ', $datestr);
Derek Allard2067d1a2008-11-13 22:59:24 +0000454
Andrey Andreevf11a1932012-06-14 16:35:09 +0300455 list($year, $month, $day) = explode('-', $split[0]);
Derek Allard2067d1a2008-11-13 22:59:24 +0000456
Andrey Andreevae31eb52012-05-17 14:54:15 +0300457 $ex = explode(':', $split['1']);
Barry Mienydd671972010-10-04 16:33:58 +0200458
Andrey Andreevf11a1932012-06-14 16:35:09 +0300459 $hour = (int) $ex[0];
460 $min = (int) $ex[1];
461 $sec = ( ! empty($ex[2]) && preg_match('/[0-9]{1,2}/', $ex[2]))
462 ? (int) $ex[2] : 0;
Derek Allard2067d1a2008-11-13 22:59:24 +0000463
Andrey Andreev5036c9c2012-06-04 15:34:56 +0300464 if (isset($split[2]))
Derek Allard2067d1a2008-11-13 22:59:24 +0000465 {
Andrey Andreevae31eb52012-05-17 14:54:15 +0300466 $ampm = strtolower($split[2]);
Barry Mienydd671972010-10-04 16:33:58 +0200467
Andrey Andreevf11a1932012-06-14 16:35:09 +0300468 if ($ampm[0] === 'p' && $hour < 12)
Greg Akerf9168392011-08-29 19:29:05 -0500469 {
Andrey Andreevae31eb52012-05-17 14:54:15 +0300470 $hour += 12;
Greg Akerf9168392011-08-29 19:29:05 -0500471 }
Andrey Andreevf11a1932012-06-14 16:35:09 +0300472 elseif ($ampm[0] === 'a' && $hour === 12)
Greg Akerf9168392011-08-29 19:29:05 -0500473 {
Andrey Andreevf11a1932012-06-14 16:35:09 +0300474 $hour = 0;
Greg Akerf9168392011-08-29 19:29:05 -0500475 }
Derek Allard2067d1a2008-11-13 22:59:24 +0000476 }
Barry Mienydd671972010-10-04 16:33:58 +0200477
Derek Allard2067d1a2008-11-13 22:59:24 +0000478 return mktime($hour, $min, $sec, $month, $day, $year);
479 }
480}
Barry Mienydd671972010-10-04 16:33:58 +0200481
Derek Allard2067d1a2008-11-13 22:59:24 +0000482// ------------------------------------------------------------------------
483
Kyle Farris896d95a2011-08-21 23:03:54 -0300484if ( ! function_exists('nice_date'))
485{
Timothy Warren01b129a2012-04-27 11:36:50 -0400486 /**
487 * Turns many "reasonably-date-like" strings into something
488 * that is actually useful. This only works for dates after unix epoch.
489 *
490 * @param string The terribly formatted date-like string
491 * @param string Date format to return (same as php date function)
492 * @return string
493 */
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500494 function nice_date($bad_date = '', $format = FALSE)
Kyle Farris896d95a2011-08-21 23:03:54 -0300495 {
496 if (empty($bad_date))
497 {
498 return 'Unknown';
499 }
Andrey Andreevd9b44be2012-06-15 16:07:08 +0300500 elseif (empty($format))
501 {
502 $format = 'U';
503 }
Greg Akerf9168392011-08-29 19:29:05 -0500504
Kyle Farris896d95a2011-08-21 23:03:54 -0300505 // Date like: YYYYMM
Andrey Andreevf11a1932012-06-14 16:35:09 +0300506 if (preg_match('/^\d{6}$/i', $bad_date))
Kyle Farris896d95a2011-08-21 23:03:54 -0300507 {
Andrey Andreevae31eb52012-05-17 14:54:15 +0300508 if (in_array(substr($bad_date, 0, 2), array('19', '20')))
Kyle Farris896d95a2011-08-21 23:03:54 -0300509 {
510 $year = substr($bad_date, 0, 4);
511 $month = substr($bad_date, 4, 2);
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500512 }
513 else
Kyle Farris896d95a2011-08-21 23:03:54 -0300514 {
515 $month = substr($bad_date, 0, 2);
516 $year = substr($bad_date, 2, 4);
517 }
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500518
Andrey Andreevae31eb52012-05-17 14:54:15 +0300519 return date($format, strtotime($year.'-'.$month.'-01'));
Kyle Farris896d95a2011-08-21 23:03:54 -0300520 }
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500521
Kyle Farris896d95a2011-08-21 23:03:54 -0300522 // Date Like: YYYYMMDD
Andrey Andreevf11a1932012-06-14 16:35:09 +0300523 if (preg_match('/^(\d{2})\d{2}(\d{4})$/i', $bad_date, $matches))
Kyle Farris896d95a2011-08-21 23:03:54 -0300524 {
Andrey Andreevf11a1932012-06-14 16:35:09 +0300525 return date($format, strtotime($matches[1].'/01/'.$matches[2]));
Kyle Farris896d95a2011-08-21 23:03:54 -0300526 }
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500527
Kyle Farris896d95a2011-08-21 23:03:54 -0300528 // Date Like: MM-DD-YYYY __or__ M-D-YYYY (or anything in between)
Andrey Andreevf11a1932012-06-14 16:35:09 +0300529 if (preg_match('/^(\d{1,2})-(\d{1,2})-(\d{4})$/i', $bad_date, $matches))
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500530 {
Andrey Andreevf11a1932012-06-14 16:35:09 +0300531 return date($format, strtotime($matches[3].'-'.$matches[1].'-'.$matches[2]));
Kyle Farris896d95a2011-08-21 23:03:54 -0300532 }
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500533
Kyle Farris896d95a2011-08-21 23:03:54 -0300534 // Any other kind of string, when converted into UNIX time,
535 // produces "0 seconds after epoc..." is probably bad...
536 // return "Invalid Date".
Alex Bilbie773ccc32012-06-02 11:11:08 +0100537 if (date('U', strtotime($bad_date)) === '0')
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500538 {
Andrey Andreevae31eb52012-05-17 14:54:15 +0300539 return 'Invalid Date';
Kyle Farris896d95a2011-08-21 23:03:54 -0300540 }
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500541
Kyle Farris896d95a2011-08-21 23:03:54 -0300542 // It's probably a valid-ish date format already
543 return date($format, strtotime($bad_date));
544 }
545}
546
547// ------------------------------------------------------------------------
548
Derek Allard2067d1a2008-11-13 22:59:24 +0000549if ( ! function_exists('timezone_menu'))
550{
Timothy Warren01b129a2012-04-27 11:36:50 -0400551 /**
552 * Timezone Menu
553 *
554 * Generates a drop-down menu of timezones.
555 *
556 * @param string timezone
557 * @param string classname
558 * @param string menu name
Mat Whitney7540ded2012-06-22 12:02:10 -0700559 * @param mixed attributes
Timothy Warren01b129a2012-04-27 11:36:50 -0400560 * @return string
561 */
Mat Whitney7540ded2012-06-22 12:02:10 -0700562 function timezone_menu($default = 'UTC', $class = '', $name = 'timezones', $attributes = '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000563 {
564 $CI =& get_instance();
565 $CI->lang->load('date');
Barry Mienydd671972010-10-04 16:33:58 +0200566
Alex Bilbie773ccc32012-06-02 11:11:08 +0100567 $default = ($default === 'GMT') ? 'UTC' : $default;
Derek Allard2067d1a2008-11-13 22:59:24 +0000568
569 $menu = '<select name="'.$name.'"';
Barry Mienydd671972010-10-04 16:33:58 +0200570
Alex Bilbie773ccc32012-06-02 11:11:08 +0100571 if ($class !== '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000572 {
573 $menu .= ' class="'.$class.'"';
574 }
Barry Mienydd671972010-10-04 16:33:58 +0200575
Eric Barnesacedd2b2012-07-29 00:15:40 -0400576 $menu .= _stringify_attributes($attributes).">\n";
Barry Mienydd671972010-10-04 16:33:58 +0200577
Derek Allard2067d1a2008-11-13 22:59:24 +0000578 foreach (timezones() as $key => $val)
579 {
Alex Bilbie773ccc32012-06-02 11:11:08 +0100580 $selected = ($default === $key) ? ' selected="selected"' : '';
Andrey Andreevae31eb52012-05-17 14:54:15 +0300581 $menu .= '<option value="'.$key.'"'.$selected.'>'.$CI->lang->line($key)."</option>\n";
Derek Allard2067d1a2008-11-13 22:59:24 +0000582 }
583
Andrey Andreevae31eb52012-05-17 14:54:15 +0300584 return $menu.'</select>';
Derek Allard2067d1a2008-11-13 22:59:24 +0000585 }
586}
Barry Mienydd671972010-10-04 16:33:58 +0200587
Derek Allard2067d1a2008-11-13 22:59:24 +0000588// ------------------------------------------------------------------------
589
Derek Allard2067d1a2008-11-13 22:59:24 +0000590if ( ! function_exists('timezones'))
591{
Timothy Warren01b129a2012-04-27 11:36:50 -0400592 /**
593 * Timezones
594 *
595 * Returns an array of timezones. This is a helper function
596 * for various other ones in this library
597 *
598 * @param string timezone
599 * @return string
600 */
Derek Allard2067d1a2008-11-13 22:59:24 +0000601 function timezones($tz = '')
602 {
603 // Note: Don't change the order of these even though
604 // some items appear to be in the wrong order
Barry Mienydd671972010-10-04 16:33:58 +0200605
606 $zones = array(
Greg Akerc964e722011-08-29 19:31:29 -0500607 'UM12' => -12,
608 'UM11' => -11,
609 'UM10' => -10,
610 'UM95' => -9.5,
611 'UM9' => -9,
612 'UM8' => -8,
613 'UM7' => -7,
614 'UM6' => -6,
615 'UM5' => -5,
616 'UM45' => -4.5,
617 'UM4' => -4,
618 'UM35' => -3.5,
619 'UM3' => -3,
620 'UM2' => -2,
621 'UM1' => -1,
622 'UTC' => 0,
623 'UP1' => +1,
624 'UP2' => +2,
625 'UP3' => +3,
626 'UP35' => +3.5,
627 'UP4' => +4,
628 'UP45' => +4.5,
629 'UP5' => +5,
630 'UP55' => +5.5,
631 'UP575' => +5.75,
632 'UP6' => +6,
633 'UP65' => +6.5,
634 'UP7' => +7,
635 'UP8' => +8,
636 'UP875' => +8.75,
637 'UP9' => +9,
638 'UP95' => +9.5,
639 'UP10' => +10,
640 'UP105' => +10.5,
641 'UP11' => +11,
642 'UP115' => +11.5,
643 'UP12' => +12,
644 'UP1275' => +12.75,
645 'UP13' => +13,
646 'UP14' => +14
647 );
Barry Mienydd671972010-10-04 16:33:58 +0200648
Alex Bilbie773ccc32012-06-02 11:11:08 +0100649 if ($tz === '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000650 {
651 return $zones;
652 }
Eric Barnesdc3e4be2011-12-19 13:48:29 -0500653
Andrey Andreeve92df332012-03-26 22:44:20 +0300654 return isset($zones[$tz]) ? $zones[$tz] : 0;
Derek Allard2067d1a2008-11-13 22:59:24 +0000655 }
656}
657
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200658// ------------------------------------------------------------------------
659
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200660if ( ! function_exists('date_range'))
661{
Andrey Andreev14aa3172012-05-02 13:27:30 +0300662 /**
663 * Date range
664 *
665 * Returns a list of dates within a specified period.
666 *
667 * @param int unix_start UNIX timestamp of period start date
668 * @param int unix_end|days UNIX timestamp of period end date
669 * or interval in days.
Andrey Andreev2f8bf9b2012-10-12 20:37:52 +0300670 * @param mixed is_unix Specifies whether the second parameter
Andrey Andreev14aa3172012-05-02 13:27:30 +0300671 * is a UNIX timestamp or a day interval
672 * - TRUE or 'unix' for a timestamp
673 * - FALSE or 'days' for an interval
674 * @param string date_format Output date format, same as in date()
675 * @return array
676 */
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200677 function date_range($unix_start = '', $mixed = '', $is_unix = TRUE, $format = 'Y-m-d')
678 {
679 if ($unix_start == '' OR $mixed == '' OR $format == '')
680 {
681 return FALSE;
682 }
683
684 $is_unix = ! ( ! $is_unix OR $is_unix === 'days');
685
686 // Validate input and try strtotime() on invalid timestamps/intervals, just in case
687 if ( ( ! preg_match('/^[0-9]+$/', $unix_start) && ($unix_start = @strtotime($unix_time)) === FALSE)
688 OR ( ! preg_match('/^[0-9]+$/', $mixed) && ($is_unix === FALSE OR ($mixed = @strtotime($mixed)) === FALSE))
689 OR ($is_unix === TRUE && $mixed < $unix_start))
690 {
691 return FALSE;
692 }
693
694 if ($is_unix && ($unix_start == $mixed OR date($format, $unix_start) === date($format, $mixed)))
695 {
696 return array($start_date);
697 }
698
699 $range = array();
700
Andrey Andreev30da39b2012-03-10 15:49:17 +0200701 /* NOTE: Even though the DateTime object has many useful features, it appears that
702 * it doesn't always handle properly timezones, when timestamps are passed
703 * directly to its constructor. Neither of the following gave proper results:
704 *
705 * new DateTime('<timestamp>')
706 * new DateTime('<timestamp>', '<timezone>')
707 *
708 * --- available in PHP 5.3:
709 *
710 * DateTime::createFromFormat('<format>', '<timestamp>')
711 * DateTime::createFromFormat('<format>', '<timestamp>', '<timezone')
712 *
713 * ... so we'll have to set the timestamp after the object is instantiated.
714 * Furthermore, in PHP 5.3 we can use DateTime::setTimestamp() to do that and
715 * given that we have UNIX timestamps - we should use it.
716 */
717 $from = new DateTime();
718
719 if (is_php('5.3'))
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200720 {
Andrey Andreev30da39b2012-03-10 15:49:17 +0200721 $from->setTimestamp($unix_start);
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200722 if ($is_unix)
723 {
724 $arg = new DateTime();
Andrey Andreev30da39b2012-03-10 15:49:17 +0200725 $arg->setTimestamp($mixed);
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200726 }
727 else
728 {
729 $arg = (int) $mixed;
730 }
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200731
Andrey Andreev30da39b2012-03-10 15:49:17 +0200732 $period = new DatePeriod($from, new DateInterval('P1D'), $arg);
733 foreach ($period as $date)
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200734 {
Andrey Andreev30da39b2012-03-10 15:49:17 +0200735 $range[] = $date->format($format);
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200736 }
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200737
Andrey Andreev30da39b2012-03-10 15:49:17 +0200738 /* If a period end date was passed to the DatePeriod constructor, it might not
739 * be in our results. Not sure if this is a bug or it's just possible because
740 * the end date might actually be less than 24 hours away from the previously
741 * generated DateTime object, but either way - we have to append it manually.
742 */
743 if ( ! is_int($arg) && $range[count($range) - 1] !== $arg->format($format))
744 {
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200745 $range[] = $arg->format($format);
746 }
747
748 return $range;
749 }
750
Andrey Andreev30da39b2012-03-10 15:49:17 +0200751 $from->setDate(date('Y', $unix_start), date('n', $unix_start), date('j', $unix_start));
752 $from->setTime(date('G', $unix_start), date('i', $unix_start), date('s', $unix_start));
753 if ($is_unix)
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200754 {
Andrey Andreev30da39b2012-03-10 15:49:17 +0200755 $arg = new DateTime();
756 $arg->setDate(date('Y', $mixed), date('n', $mixed), date('j', $mixed));
757 $arg->setTime(date('G', $mixed), date('i', $mixed), date('s', $mixed));
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200758 }
Andrey Andreev30da39b2012-03-10 15:49:17 +0200759 else
760 {
761 $arg = (int) $mixed;
762 }
763 $range[] = $from->format($format);
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200764
Andrey Andreev30da39b2012-03-10 15:49:17 +0200765 if (is_int($arg)) // Day intervals
766 {
767 do
768 {
769 $from->modify('+1 day');
770 $range[] = $from->format($format);
771 }
772 while (--$arg > 0);
773 }
774 else // end date UNIX timestamp
775 {
776 for ($from->modify('+1 day'), $end_check = $arg->format('Ymd'); $from->format('Ymd') < $end_check; $from->modify('+1 day'))
777 {
778 $range[] = $from->format($format);
779 }
780
781 // Our loop only appended dates prior to our end date
782 $range[] = $arg->format($format);
783 }
Andrey Andreev2139ecd2012-01-11 23:58:50 +0200784
785 return $range;
786 }
787}
788
Derek Allard2067d1a2008-11-13 22:59:24 +0000789/* End of file date_helper.php */
Derek Jonesa3ffbbb2008-05-11 18:18:29 +0000790/* Location: ./system/helpers/date_helper.php */