2016-06-23 12:47:54 -04: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-22 20:37:48 -04:00
|
|
|
import {NgZone} from '@angular/core';
|
2016-08-30 21:07:40 -04:00
|
|
|
|
|
|
|
import {ListWrapper} from './facade/collection';
|
|
|
|
import {RegExp, StringWrapper, global, isPresent, isString} from './facade/lang';
|
|
|
|
import {getDOM} from './private_import_platform-browser';
|
2015-08-24 09:41:36 -04:00
|
|
|
|
|
|
|
export class BrowserDetection {
|
2016-05-02 01:09:58 -04:00
|
|
|
private _overrideUa: string;
|
2016-05-02 01:50:37 -04:00
|
|
|
private get _ua(): string {
|
2016-05-02 01:09:58 -04:00
|
|
|
if (isPresent(this._overrideUa)) {
|
|
|
|
return this._overrideUa;
|
|
|
|
} else {
|
|
|
|
return isPresent(getDOM()) ? getDOM().getUserAgent() : '';
|
|
|
|
}
|
|
|
|
}
|
2015-08-24 09:41:36 -04:00
|
|
|
|
2015-12-15 19:38:27 -05:00
|
|
|
static setup() { browserDetection = new BrowserDetection(null); }
|
|
|
|
|
2016-05-02 01:50:37 -04:00
|
|
|
constructor(ua: string) { this._overrideUa = ua; }
|
2015-08-24 09:41:36 -04:00
|
|
|
|
|
|
|
get isFirefox(): boolean { return this._ua.indexOf('Firefox') > -1; }
|
|
|
|
|
|
|
|
get isAndroid(): boolean {
|
|
|
|
return this._ua.indexOf('Mozilla/5.0') > -1 && this._ua.indexOf('Android') > -1 &&
|
2016-07-29 13:05:12 -04:00
|
|
|
this._ua.indexOf('AppleWebKit') > -1 && this._ua.indexOf('Chrome') == -1 &&
|
|
|
|
this._ua.indexOf('IEMobile') == -1;
|
2015-08-24 09:41:36 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
get isEdge(): boolean { return this._ua.indexOf('Edge') > -1; }
|
|
|
|
|
|
|
|
get isIE(): boolean { return this._ua.indexOf('Trident') > -1; }
|
|
|
|
|
|
|
|
get isWebkit(): boolean {
|
2016-07-29 13:05:12 -04:00
|
|
|
return this._ua.indexOf('AppleWebKit') > -1 && this._ua.indexOf('Edge') == -1 &&
|
|
|
|
this._ua.indexOf('IEMobile') == -1;
|
2015-08-24 09:41:36 -04:00
|
|
|
}
|
|
|
|
|
2015-09-09 10:41:11 -04:00
|
|
|
get isIOS7(): boolean {
|
2016-07-29 13:05:12 -04:00
|
|
|
return (this._ua.indexOf('iPhone OS 7') > -1 || this._ua.indexOf('iPad OS 7') > -1) &&
|
|
|
|
this._ua.indexOf('IEMobile') == -1;
|
2015-09-09 10:41:11 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
get isSlow(): boolean { return this.isAndroid || this.isIE || this.isIOS7; }
|
|
|
|
|
2016-08-31 18:55:13 -04:00
|
|
|
// The Intl API is only natively supported in Chrome, Firefox, IE11 and Edge.
|
|
|
|
// This detector is needed in tests to make the difference between:
|
|
|
|
// 1) IE11/Edge: they have a native Intl API, but with some discrepancies
|
|
|
|
// 2) IE9/IE10: they use the polyfill, and so no discrepancies
|
|
|
|
get supportsNativeIntlApi(): boolean {
|
|
|
|
return !!(<any>global).Intl && (<any>global).Intl !== (<any>global).IntlPolyfill;
|
|
|
|
}
|
2016-05-10 20:47:17 -04:00
|
|
|
|
|
|
|
get isChromeDesktop(): boolean {
|
2016-05-13 16:22:29 -04:00
|
|
|
return this._ua.indexOf('Chrome') > -1 && this._ua.indexOf('Mobile Safari') == -1 &&
|
2016-06-08 19:38:52 -04:00
|
|
|
this._ua.indexOf('Edge') == -1;
|
2016-05-10 20:47:17 -04:00
|
|
|
}
|
2016-08-31 18:55:13 -04:00
|
|
|
|
|
|
|
// "Old Chrome" means Chrome 3X, where there are some discrepancies in the Intl API.
|
|
|
|
// Android 4.4 and 5.X have such browsers by default (respectively 30 and 39).
|
|
|
|
get isOldChrome(): boolean {
|
|
|
|
return this._ua.indexOf('Chrome') > -1 && this._ua.indexOf('Chrome/3') > -1 &&
|
|
|
|
this._ua.indexOf('Edge') == -1;
|
|
|
|
}
|
2015-08-24 09:41:36 -04:00
|
|
|
}
|
|
|
|
|
2016-02-26 06:25:55 -05:00
|
|
|
BrowserDetection.setup();
|
|
|
|
|
2016-06-08 19:38:52 -04:00
|
|
|
export function dispatchEvent(
|
|
|
|
element: any /** TODO #9100 */, eventType: any /** TODO #9100 */): void {
|
2016-04-28 20:50:03 -04:00
|
|
|
getDOM().dispatchEvent(element, getDOM().createEvent(eventType));
|
2015-02-03 10:27:09 -05:00
|
|
|
}
|
|
|
|
|
2015-07-07 23:03:00 -04:00
|
|
|
export function el(html: string): HTMLElement {
|
2016-04-28 20:50:03 -04:00
|
|
|
return <HTMLElement>getDOM().firstChild(getDOM().content(getDOM().createTemplate(html)));
|
2015-05-26 16:57:13 -04:00
|
|
|
}
|
2015-05-27 04:22:30 -04:00
|
|
|
|
|
|
|
export function normalizeCSS(css: string): string {
|
2015-06-23 06:46:38 -04:00
|
|
|
css = StringWrapper.replaceAll(css, /\s+/g, ' ');
|
|
|
|
css = StringWrapper.replaceAll(css, /:\s/g, ':');
|
2015-07-29 12:34:02 -04:00
|
|
|
css = StringWrapper.replaceAll(css, /'/g, '"');
|
2015-08-20 12:23:57 -04:00
|
|
|
css = StringWrapper.replaceAll(css, / }/g, '}');
|
2016-06-08 19:38:52 -04:00
|
|
|
css = StringWrapper.replaceAllMapped(
|
|
|
|
css, /url\((\"|\s)(.+)(\"|\s)\)(\s*)/g,
|
|
|
|
(match: any /** TODO #9100 */) => `url("${match[2]}")`);
|
|
|
|
css = StringWrapper.replaceAllMapped(
|
|
|
|
css, /\[(.+)=([^"\]]+)\]/g, (match: any /** TODO #9100 */) => `[${match[1]}="${match[2]}"]`);
|
2015-05-27 04:22:30 -04:00
|
|
|
return css;
|
|
|
|
}
|
2015-05-22 11:47:13 -04:00
|
|
|
|
|
|
|
var _singleTagWhitelist = ['br', 'hr', 'input'];
|
2016-06-08 18:45:15 -04:00
|
|
|
export function stringifyElement(el: any /** TODO #9100 */): string {
|
2015-05-22 11:47:13 -04:00
|
|
|
var result = '';
|
2016-04-28 20:50:03 -04:00
|
|
|
if (getDOM().isElementNode(el)) {
|
|
|
|
var tagName = getDOM().tagName(el).toLowerCase();
|
2015-05-22 11:47:13 -04:00
|
|
|
|
|
|
|
// Opening tag
|
|
|
|
result += `<${tagName}`;
|
|
|
|
|
|
|
|
// Attributes in an ordered way
|
2016-04-28 20:50:03 -04:00
|
|
|
var attributeMap = getDOM().attributeMap(el);
|
2016-06-08 18:45:15 -04:00
|
|
|
var keys: any[] /** TODO #9100 */ = [];
|
2015-10-08 19:01:18 -04:00
|
|
|
attributeMap.forEach((v, k) => keys.push(k));
|
2015-05-22 11:47:13 -04:00
|
|
|
ListWrapper.sort(keys);
|
|
|
|
for (let i = 0; i < keys.length; i++) {
|
|
|
|
var key = keys[i];
|
2015-06-17 19:21:40 -04:00
|
|
|
var attValue = attributeMap.get(key);
|
2015-05-22 11:47:13 -04:00
|
|
|
if (!isString(attValue)) {
|
|
|
|
result += ` ${key}`;
|
|
|
|
} else {
|
|
|
|
result += ` ${key}="${attValue}"`;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
result += '>';
|
|
|
|
|
|
|
|
// Children
|
2016-04-28 20:50:03 -04:00
|
|
|
var childrenRoot = getDOM().templateAwareRoot(el);
|
|
|
|
var children = isPresent(childrenRoot) ? getDOM().childNodes(childrenRoot) : [];
|
2015-05-22 11:47:13 -04:00
|
|
|
for (let j = 0; j < children.length; j++) {
|
|
|
|
result += stringifyElement(children[j]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Closing tag
|
|
|
|
if (!ListWrapper.contains(_singleTagWhitelist, tagName)) {
|
|
|
|
result += `</${tagName}>`;
|
|
|
|
}
|
2016-04-28 20:50:03 -04:00
|
|
|
} else if (getDOM().isCommentNode(el)) {
|
|
|
|
result += `<!--${getDOM().nodeValue(el)}-->`;
|
2015-05-22 11:47:13 -04:00
|
|
|
} else {
|
2016-04-28 20:50:03 -04:00
|
|
|
result += getDOM().getText(el);
|
2015-05-22 11:47:13 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2016-04-28 20:50:03 -04:00
|
|
|
|
|
|
|
export var browserDetection: BrowserDetection = new BrowserDetection(null);
|
2016-08-22 20:37:48 -04:00
|
|
|
|
|
|
|
export function createNgZone(): NgZone {
|
|
|
|
return new NgZone({enableLongStackTrace: true});
|
|
|
|
}
|