148 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			148 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| /**
 | |
|  * @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
 | |
|  */
 | |
| 
 | |
| import {NgZone} from '@angular/core';
 | |
| 
 | |
| import {ListWrapper} from './facade/collection';
 | |
| import {RegExp, StringWrapper, global, isPresent, isString} from './facade/lang';
 | |
| import {getDOM} from './private_import_platform-browser';
 | |
| 
 | |
| export class BrowserDetection {
 | |
|   private _overrideUa: string;
 | |
|   private get _ua(): string {
 | |
|     if (isPresent(this._overrideUa)) {
 | |
|       return this._overrideUa;
 | |
|     } else {
 | |
|       return isPresent(getDOM()) ? getDOM().getUserAgent() : '';
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   static setup() { browserDetection = new BrowserDetection(null); }
 | |
| 
 | |
|   constructor(ua: string) { this._overrideUa = ua; }
 | |
| 
 | |
|   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 &&
 | |
|         this._ua.indexOf('AppleWebKit') > -1 && this._ua.indexOf('Chrome') == -1 &&
 | |
|         this._ua.indexOf('IEMobile') == -1;
 | |
|   }
 | |
| 
 | |
|   get isEdge(): boolean { return this._ua.indexOf('Edge') > -1; }
 | |
| 
 | |
|   get isIE(): boolean { return this._ua.indexOf('Trident') > -1; }
 | |
| 
 | |
|   get isWebkit(): boolean {
 | |
|     return this._ua.indexOf('AppleWebKit') > -1 && this._ua.indexOf('Edge') == -1 &&
 | |
|         this._ua.indexOf('IEMobile') == -1;
 | |
|   }
 | |
| 
 | |
|   get isIOS7(): boolean {
 | |
|     return (this._ua.indexOf('iPhone OS 7') > -1 || this._ua.indexOf('iPad OS 7') > -1) &&
 | |
|         this._ua.indexOf('IEMobile') == -1;
 | |
|   }
 | |
| 
 | |
|   get isSlow(): boolean { return this.isAndroid || this.isIE || this.isIOS7; }
 | |
| 
 | |
|   // 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;
 | |
|   }
 | |
| 
 | |
|   get isChromeDesktop(): boolean {
 | |
|     return this._ua.indexOf('Chrome') > -1 && this._ua.indexOf('Mobile Safari') == -1 &&
 | |
|         this._ua.indexOf('Edge') == -1;
 | |
|   }
 | |
| 
 | |
|   // "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;
 | |
|   }
 | |
| }
 | |
| 
 | |
| BrowserDetection.setup();
 | |
| 
 | |
| export function dispatchEvent(
 | |
|     element: any /** TODO #9100 */, eventType: any /** TODO #9100 */): void {
 | |
|   getDOM().dispatchEvent(element, getDOM().createEvent(eventType));
 | |
| }
 | |
| 
 | |
| export function el(html: string): HTMLElement {
 | |
|   return <HTMLElement>getDOM().firstChild(getDOM().content(getDOM().createTemplate(html)));
 | |
| }
 | |
| 
 | |
| export function normalizeCSS(css: string): string {
 | |
|   css = StringWrapper.replaceAll(css, /\s+/g, ' ');
 | |
|   css = StringWrapper.replaceAll(css, /:\s/g, ':');
 | |
|   css = StringWrapper.replaceAll(css, /'/g, '"');
 | |
|   css = StringWrapper.replaceAll(css, / }/g, '}');
 | |
|   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]}"]`);
 | |
|   return css;
 | |
| }
 | |
| 
 | |
| var _singleTagWhitelist = ['br', 'hr', 'input'];
 | |
| export function stringifyElement(el: any /** TODO #9100 */): string {
 | |
|   var result = '';
 | |
|   if (getDOM().isElementNode(el)) {
 | |
|     var tagName = getDOM().tagName(el).toLowerCase();
 | |
| 
 | |
|     // Opening tag
 | |
|     result += `<${tagName}`;
 | |
| 
 | |
|     // Attributes in an ordered way
 | |
|     var attributeMap = getDOM().attributeMap(el);
 | |
|     var keys: any[] /** TODO #9100 */ = [];
 | |
|     attributeMap.forEach((v, k) => keys.push(k));
 | |
|     ListWrapper.sort(keys);
 | |
|     for (let i = 0; i < keys.length; i++) {
 | |
|       var key = keys[i];
 | |
|       var attValue = attributeMap.get(key);
 | |
|       if (!isString(attValue)) {
 | |
|         result += ` ${key}`;
 | |
|       } else {
 | |
|         result += ` ${key}="${attValue}"`;
 | |
|       }
 | |
|     }
 | |
|     result += '>';
 | |
| 
 | |
|     // Children
 | |
|     var childrenRoot = getDOM().templateAwareRoot(el);
 | |
|     var children = isPresent(childrenRoot) ? getDOM().childNodes(childrenRoot) : [];
 | |
|     for (let j = 0; j < children.length; j++) {
 | |
|       result += stringifyElement(children[j]);
 | |
|     }
 | |
| 
 | |
|     // Closing tag
 | |
|     if (!ListWrapper.contains(_singleTagWhitelist, tagName)) {
 | |
|       result += `</${tagName}>`;
 | |
|     }
 | |
|   } else if (getDOM().isCommentNode(el)) {
 | |
|     result += `<!--${getDOM().nodeValue(el)}-->`;
 | |
|   } else {
 | |
|     result += getDOM().getText(el);
 | |
|   }
 | |
| 
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| export var browserDetection: BrowserDetection = new BrowserDetection(null);
 | |
| 
 | |
| export function createNgZone(): NgZone {
 | |
|   return new NgZone({enableLongStackTrace: true});
 | |
| }
 |