2016-06-23 09:47:54 -07:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright Google Inc. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2016-08-26 09:16:01 -07:00
|
|
|
import {Inject, LOCALE_ID, Pipe, PipeTransform} from '@angular/core';
|
2016-12-13 21:34:50 +01:00
|
|
|
import {DateFormatter} from './intl';
|
2017-01-27 13:19:00 -08:00
|
|
|
import {invalidPipeArgumentError} from './invalid_pipe_argument_error';
|
2017-03-02 09:37:01 -08:00
|
|
|
import {isNumeric} from './number_pipe';
|
2015-08-07 11:41:38 -07:00
|
|
|
|
2017-01-30 21:44:19 +03:00
|
|
|
const ISO8601_DATE_REGEX =
|
|
|
|
/^(\d{4})-?(\d\d)-?(\d\d)(?:T(\d\d)(?::?(\d\d)(?::?(\d\d)(?:\.(\d+))?)?)?(Z|([+-])(\d\d):?(\d\d))?)?$/;
|
|
|
|
// 1 2 3 4 5 6 7 8 9 10 11
|
2016-12-13 21:34:50 +01:00
|
|
|
|
2015-07-04 22:34:54 +04:30
|
|
|
/**
|
2016-09-08 21:41:09 -07:00
|
|
|
* @ngModule CommonModule
|
|
|
|
* @whatItDoes Formats a date according to locale rules.
|
|
|
|
* @howToUse `date_expression | date[:format]`
|
|
|
|
* @description
|
2015-07-04 22:34:54 +04:30
|
|
|
*
|
2016-09-08 21:41:09 -07:00
|
|
|
* Where:
|
|
|
|
* - `expression` is a date object or a number (milliseconds since UTC epoch) or an ISO string
|
|
|
|
* (https://www.w3.org/TR/NOTE-datetime).
|
2017-01-24 18:21:59 +00:00
|
|
|
* - `format` indicates which date/time components to include. The format can be predefined as
|
2016-09-08 21:41:09 -07:00
|
|
|
* shown below or custom as shown in the table.
|
|
|
|
* - `'medium'`: equivalent to `'yMMMdjms'` (e.g. `Sep 3, 2010, 12:05:08 PM` for `en-US`)
|
|
|
|
* - `'short'`: equivalent to `'yMdjm'` (e.g. `9/3/2010, 12:05 PM` for `en-US`)
|
|
|
|
* - `'fullDate'`: equivalent to `'yMMMMEEEEd'` (e.g. `Friday, September 3, 2010` for `en-US`)
|
|
|
|
* - `'longDate'`: equivalent to `'yMMMMd'` (e.g. `September 3, 2010` for `en-US`)
|
|
|
|
* - `'mediumDate'`: equivalent to `'yMMMd'` (e.g. `Sep 3, 2010` for `en-US`)
|
|
|
|
* - `'shortDate'`: equivalent to `'yMd'` (e.g. `9/3/2010` for `en-US`)
|
|
|
|
* - `'mediumTime'`: equivalent to `'jms'` (e.g. `12:05:08 PM` for `en-US`)
|
|
|
|
* - `'shortTime'`: equivalent to `'jm'` (e.g. `12:05 PM` for `en-US`)
|
2015-07-04 22:34:54 +04:30
|
|
|
*
|
|
|
|
*
|
2016-10-19 20:05:13 +03:00
|
|
|
* | Component | Symbol | Narrow | Short Form | Long Form | Numeric | 2-digit |
|
|
|
|
* |-----------|:------:|--------|--------------|-------------------|-----------|-----------|
|
|
|
|
* | era | G | G (A) | GGG (AD) | GGGG (Anno Domini)| - | - |
|
|
|
|
* | year | y | - | - | - | y (2015) | yy (15) |
|
|
|
|
* | month | M | L (S) | MMM (Sep) | MMMM (September) | M (9) | MM (09) |
|
|
|
|
* | day | d | - | - | - | d (3) | dd (03) |
|
|
|
|
* | weekday | E | E (S) | EEE (Sun) | EEEE (Sunday) | - | - |
|
2017-08-08 02:47:08 +08:00
|
|
|
* | hour | j | - | - | - | j (1 PM) | jj (1 PM) |
|
|
|
|
* | hour12 | h | - | - | - | h (1) | hh (01) |
|
2016-10-19 20:05:13 +03:00
|
|
|
* | hour24 | H | - | - | - | H (13) | HH (13) |
|
|
|
|
* | minute | m | - | - | - | m (5) | mm (05) |
|
|
|
|
* | second | s | - | - | - | s (9) | ss (09) |
|
|
|
|
* | timezone | z | - | - | z (Pacific Standard Time)| - | - |
|
|
|
|
* | timezone | Z | - | Z (GMT-8:00) | - | - | - |
|
|
|
|
* | timezone | a | - | a (PM) | - | - | - |
|
2015-07-04 22:34:54 +04:30
|
|
|
*
|
|
|
|
* In javascript, only the components specified will be respected (not the ordering,
|
2015-09-22 12:11:25 +03:00
|
|
|
* punctuations, ...) and details of the formatting will be dependent on the locale.
|
2015-07-04 22:34:54 +04:30
|
|
|
*
|
2016-09-05 21:37:21 +08:00
|
|
|
* Timezone of the formatted text will be the local system timezone of the end-user's machine.
|
2015-07-04 22:34:54 +04:30
|
|
|
*
|
2016-11-10 20:57:04 -02:00
|
|
|
* When the expression is a ISO string without time (e.g. 2016-09-19) the time zone offset is not
|
|
|
|
* applied and the formatted text will have the same day, month and year of the expression.
|
|
|
|
*
|
2016-09-08 21:41:09 -07:00
|
|
|
* WARNINGS:
|
|
|
|
* - this pipe is marked as pure hence it will not be re-evaluated when the input is mutated.
|
|
|
|
* Instead users should treat the date as an immutable object and change the reference when the
|
|
|
|
* pipe needs to re-run (this is to avoid reformatting the date on every change detection run
|
|
|
|
* which would be an expensive operation).
|
|
|
|
* - this pipe uses the Internationalization API. Therefore it is only reliable in Chrome and Opera
|
|
|
|
* browsers.
|
|
|
|
*
|
2015-10-19 15:37:32 +01:00
|
|
|
* ### Examples
|
2015-07-04 22:34:54 +04:30
|
|
|
*
|
|
|
|
* Assuming `dateObj` is (year: 2015, month: 6, day: 15, hour: 21, minute: 43, second: 11)
|
|
|
|
* in the _local_ time and locale is 'en-US':
|
|
|
|
*
|
2015-11-02 15:46:59 -08:00
|
|
|
* ```
|
2015-07-04 22:34:54 +04:30
|
|
|
* {{ dateObj | date }} // output is 'Jun 15, 2015'
|
|
|
|
* {{ dateObj | date:'medium' }} // output is 'Jun 15, 2015, 9:43:11 PM'
|
|
|
|
* {{ dateObj | date:'shortTime' }} // output is '9:43 PM'
|
|
|
|
* {{ dateObj | date:'mmss' }} // output is '43:11'
|
2015-11-02 15:46:59 -08:00
|
|
|
* ```
|
|
|
|
*
|
2016-09-08 21:41:09 -07:00
|
|
|
* {@example common/pipes/ts/date_pipe.ts region='DatePipe'}
|
2016-05-27 11:24:05 -07:00
|
|
|
*
|
2016-08-23 13:58:41 -07:00
|
|
|
* @stable
|
2015-07-04 22:34:54 +04:30
|
|
|
*/
|
2015-10-27 16:37:08 -07:00
|
|
|
@Pipe({name: 'date', pure: true})
|
2015-08-12 12:04:54 -07:00
|
|
|
export class DatePipe implements PipeTransform {
|
2015-10-28 15:04:55 -07:00
|
|
|
/** @internal */
|
2016-09-11 00:27:56 -07:00
|
|
|
static _ALIASES: {[key: string]: string} = {
|
2015-07-04 22:34:54 +04:30
|
|
|
'medium': 'yMMMdjms',
|
|
|
|
'short': 'yMdjm',
|
|
|
|
'fullDate': 'yMMMMEEEEd',
|
|
|
|
'longDate': 'yMMMMd',
|
|
|
|
'mediumDate': 'yMMMd',
|
|
|
|
'shortDate': 'yMd',
|
|
|
|
'mediumTime': 'jms',
|
|
|
|
'shortTime': 'jm'
|
|
|
|
};
|
|
|
|
|
2016-08-26 09:16:01 -07:00
|
|
|
constructor(@Inject(LOCALE_ID) private _locale: string) {}
|
2015-07-04 22:34:54 +04:30
|
|
|
|
2017-03-24 09:54:02 -07:00
|
|
|
transform(value: any, pattern: string = 'mediumDate'): string|null {
|
2016-11-10 20:57:04 -02:00
|
|
|
let date: Date;
|
|
|
|
|
2017-01-26 18:23:53 +03:00
|
|
|
if (isBlank(value) || value !== value) return null;
|
2015-08-06 10:39:02 -07:00
|
|
|
|
2016-11-09 02:45:12 +03:00
|
|
|
if (typeof value === 'string') {
|
|
|
|
value = value.trim();
|
2015-08-06 10:39:02 -07:00
|
|
|
}
|
|
|
|
|
2016-11-09 02:45:12 +03:00
|
|
|
if (isDate(value)) {
|
|
|
|
date = value;
|
2017-03-02 09:37:01 -08:00
|
|
|
} else if (isNumeric(value)) {
|
2016-11-09 02:45:12 +03:00
|
|
|
date = new Date(parseFloat(value));
|
2016-11-10 20:57:04 -02:00
|
|
|
} else if (typeof value === 'string' && /^(\d{4}-\d{1,2}-\d{1,2})$/.test(value)) {
|
|
|
|
/**
|
|
|
|
* For ISO Strings without time the day, month and year must be extracted from the ISO String
|
|
|
|
* before Date creation to avoid time offset and errors in the new Date.
|
|
|
|
* If we only replace '-' with ',' in the ISO String ("2015,01,01"), and try to create a new
|
|
|
|
* date, some browsers (e.g. IE 9) will throw an invalid Date error
|
|
|
|
* If we leave the '-' ("2015-01-01") and try to create a new Date("2015-01-01") the timeoffset
|
|
|
|
* is applied
|
|
|
|
* Note: ISO months are 0 for January, 1 for February, ...
|
|
|
|
*/
|
|
|
|
const [y, m, d] = value.split('-').map((val: string) => parseInt(val, 10));
|
|
|
|
date = new Date(y, m - 1, d);
|
2016-11-09 02:45:12 +03:00
|
|
|
} else {
|
|
|
|
date = new Date(value);
|
2015-07-04 22:34:54 +04:30
|
|
|
}
|
2016-09-11 00:27:56 -07:00
|
|
|
|
2016-11-09 02:45:12 +03:00
|
|
|
if (!isDate(date)) {
|
2017-03-24 09:54:02 -07:00
|
|
|
let match: RegExpMatchArray|null;
|
2017-01-30 21:44:19 +03:00
|
|
|
if ((typeof value === 'string') && (match = value.match(ISO8601_DATE_REGEX))) {
|
|
|
|
date = isoStringToDate(match);
|
|
|
|
} else {
|
2017-01-27 13:19:00 -08:00
|
|
|
throw invalidPipeArgumentError(DatePipe, value);
|
2017-01-30 21:44:19 +03:00
|
|
|
}
|
2016-11-09 02:45:12 +03:00
|
|
|
}
|
2015-07-04 22:34:54 +04:30
|
|
|
|
2016-11-09 02:45:12 +03:00
|
|
|
return DateFormatter.format(date, this._locale, DatePipe._ALIASES[pattern] || pattern);
|
2016-03-28 23:14:32 +09:00
|
|
|
}
|
2015-07-04 22:34:54 +04:30
|
|
|
}
|
2016-11-09 02:45:12 +03:00
|
|
|
|
|
|
|
function isBlank(obj: any): boolean {
|
|
|
|
return obj == null || obj === '';
|
|
|
|
}
|
2017-01-30 21:26:21 +03:00
|
|
|
|
|
|
|
function isDate(obj: any): obj is Date {
|
|
|
|
return obj instanceof Date && !isNaN(obj.valueOf());
|
|
|
|
}
|
2017-01-30 21:44:19 +03:00
|
|
|
|
|
|
|
function isoStringToDate(match: RegExpMatchArray): Date {
|
|
|
|
const date = new Date(0);
|
|
|
|
let tzHour = 0;
|
|
|
|
let tzMin = 0;
|
|
|
|
const dateSetter = match[8] ? date.setUTCFullYear : date.setFullYear;
|
|
|
|
const timeSetter = match[8] ? date.setUTCHours : date.setHours;
|
|
|
|
|
|
|
|
if (match[9]) {
|
|
|
|
tzHour = toInt(match[9] + match[10]);
|
|
|
|
tzMin = toInt(match[9] + match[11]);
|
|
|
|
}
|
|
|
|
dateSetter.call(date, toInt(match[1]), toInt(match[2]) - 1, toInt(match[3]));
|
|
|
|
const h = toInt(match[4] || '0') - tzHour;
|
|
|
|
const m = toInt(match[5] || '0') - tzMin;
|
|
|
|
const s = toInt(match[6] || '0');
|
|
|
|
const ms = Math.round(parseFloat('0.' + (match[7] || 0)) * 1000);
|
|
|
|
timeSetter.call(date, h, m, s, ms);
|
|
|
|
return date;
|
|
|
|
}
|
|
|
|
|
|
|
|
function toInt(str: string): number {
|
|
|
|
return parseInt(str, 10);
|
|
|
|
}
|