2016-04-28 17:50:03 -07:00
|
|
|
import {PipeTransform, Pipe, Injectable} from '@angular/core';
|
2016-04-12 09:40:37 -07:00
|
|
|
import {
|
|
|
|
isDate,
|
|
|
|
isNumber,
|
|
|
|
DateWrapper,
|
|
|
|
isBlank,
|
2016-04-28 17:50:03 -07:00
|
|
|
} from '../../src/facade/lang';
|
|
|
|
import {DateFormatter} from '../../src/facade/intl';
|
|
|
|
import {StringMapWrapper} from '../../src/facade/collection';
|
2015-08-07 11:41:38 -07:00
|
|
|
|
|
|
|
import {InvalidPipeArgumentException} from './invalid_pipe_argument_exception';
|
|
|
|
|
2015-07-04 22:34:54 +04:30
|
|
|
|
2015-09-10 22:52:54 -07:00
|
|
|
// TODO: move to a global configurable location along with other i18n components.
|
2015-07-04 22:34:54 +04:30
|
|
|
var defaultLocale: string = 'en-US';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Formats a date value to a string based on the requested format.
|
|
|
|
*
|
2015-10-27 16:37:08 -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.
|
|
|
|
*
|
|
|
|
* ## Usage
|
2015-07-04 22:34:54 +04:30
|
|
|
*
|
|
|
|
* expression | date[:format]
|
|
|
|
*
|
|
|
|
* where `expression` is a date object or a number (milliseconds since UTC epoch) and
|
|
|
|
* `format` indicates which date/time components to include:
|
|
|
|
*
|
|
|
|
* | Component | Symbol | Short Form | Long Form | Numeric | 2-digit |
|
|
|
|
* |-----------|:------:|--------------|-------------------|-----------|-----------|
|
|
|
|
* | era | G | G (AD) | GGGG (Anno Domini)| - | - |
|
|
|
|
* | year | y | - | - | y (2015) | yy (15) |
|
|
|
|
* | month | M | MMM (Sep) | MMMM (September) | M (9) | MM (09) |
|
|
|
|
* | day | d | - | - | d (3) | dd (03) |
|
|
|
|
* | weekday | E | EEE (Sun) | EEEE (Sunday) | - | - |
|
|
|
|
* | hour | j | - | - | j (13) | jj (13) |
|
|
|
|
* | hour12 | h | - | - | h (1 PM) | hh (01 PM)|
|
|
|
|
* | 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) | - | - | - |
|
|
|
|
*
|
|
|
|
* 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
|
|
|
* On the other hand in Dart version, you can also include quoted text as well as some extra
|
|
|
|
* date/time components such as quarter. For more information see:
|
|
|
|
* https://api.dartlang.org/apidocs/channels/stable/dartdoc-viewer/intl/intl.DateFormat.
|
|
|
|
*
|
|
|
|
* `format` can also be one of the following predefined formats:
|
|
|
|
*
|
|
|
|
* - `'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)
|
|
|
|
* - `'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)
|
|
|
|
*
|
|
|
|
* Timezone of the formatted text will be the local system timezone of the end-users machine.
|
|
|
|
*
|
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
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* {@example core/pipes/ts/date_pipe/date_pipe_example.ts region='DatePipe'}
|
2015-07-04 22:34:54 +04:30
|
|
|
*/
|
2015-10-27 16:37:08 -07:00
|
|
|
@Pipe({name: 'date', pure: true})
|
2015-08-06 10:39:02 -07:00
|
|
|
@Injectable()
|
2015-08-12 12:04:54 -07:00
|
|
|
export class DatePipe implements PipeTransform {
|
2015-10-28 15:04:55 -07:00
|
|
|
/** @internal */
|
2015-10-02 17:33:21 -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-04-22 15:33:32 -07:00
|
|
|
transform(value: any, pattern: string = 'mediumDate'): string {
|
2015-08-06 10:39:02 -07:00
|
|
|
if (isBlank(value)) return null;
|
|
|
|
|
|
|
|
if (!this.supports(value)) {
|
|
|
|
throw new InvalidPipeArgumentException(DatePipe, value);
|
|
|
|
}
|
|
|
|
|
2015-07-04 22:34:54 +04:30
|
|
|
if (isNumber(value)) {
|
|
|
|
value = DateWrapper.fromMillis(value);
|
|
|
|
}
|
|
|
|
if (StringMapWrapper.contains(DatePipe._ALIASES, pattern)) {
|
|
|
|
pattern = <string>StringMapWrapper.get(DatePipe._ALIASES, pattern);
|
|
|
|
}
|
|
|
|
return DateFormatter.format(value, defaultLocale, pattern);
|
|
|
|
}
|
|
|
|
|
2015-09-04 15:09:02 -07:00
|
|
|
supports(obj: any): boolean { return isDate(obj) || isNumber(obj); }
|
2015-07-04 22:34:54 +04:30
|
|
|
}
|