| 
									
										
										
										
											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
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-22 19:16:25 -07:00
										 |  |  | import {ɵgetDOM as getDOM} from '@angular/common'; | 
					
						
							| 
									
										
										
										
											2017-03-01 15:18:10 -08:00
										 |  |  | import {NgZone, ɵglobal as global} from '@angular/core'; | 
					
						
							| 
									
										
										
										
											2016-08-30 18:07:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-24 15:41:36 +02:00
										 |  |  | export class BrowserDetection { | 
					
						
							| 
									
										
										
										
											2017-03-24 09:59:41 -07:00
										 |  |  |   private _overrideUa: string|null; | 
					
						
							| 
									
										
										
										
											2016-05-01 22:50:37 -07:00
										 |  |  |   private get _ua(): string { | 
					
						
							| 
									
										
										
										
											2016-11-03 16:58:27 -07:00
										 |  |  |     if (typeof this._overrideUa === 'string') { | 
					
						
							| 
									
										
										
										
											2016-05-01 22:09:58 -07:00
										 |  |  |       return this._overrideUa; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-11-03 16:58:27 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return getDOM() ? getDOM().getUserAgent() : ''; | 
					
						
							| 
									
										
										
										
											2016-05-01 22:09:58 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-08-24 15:41:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-03 09:15:49 +01:00
										 |  |  |   static setup() { return new BrowserDetection(null); } | 
					
						
							| 
									
										
										
										
											2015-12-15 16:38:27 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-24 09:59:41 -07:00
										 |  |  |   constructor(ua: string|null) { this._overrideUa = ua; } | 
					
						
							| 
									
										
										
										
											2015-08-24 15:41:36 +02: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 19:05:12 +02:00
										 |  |  |         this._ua.indexOf('AppleWebKit') > -1 && this._ua.indexOf('Chrome') == -1 && | 
					
						
							|  |  |  |         this._ua.indexOf('IEMobile') == -1; | 
					
						
							| 
									
										
										
										
											2015-08-24 15:41:36 +02: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 19:05:12 +02:00
										 |  |  |     return this._ua.indexOf('AppleWebKit') > -1 && this._ua.indexOf('Edge') == -1 && | 
					
						
							|  |  |  |         this._ua.indexOf('IEMobile') == -1; | 
					
						
							| 
									
										
										
										
											2015-08-24 15:41:36 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-09 16:41:11 +02:00
										 |  |  |   get isIOS7(): boolean { | 
					
						
							| 
									
										
										
										
											2016-07-29 19:05:12 +02:00
										 |  |  |     return (this._ua.indexOf('iPhone OS 7') > -1 || this._ua.indexOf('iPad OS 7') > -1) && | 
					
						
							|  |  |  |         this._ua.indexOf('IEMobile') == -1; | 
					
						
							| 
									
										
										
										
											2015-09-09 16:41:11 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   get isSlow(): boolean { return this.isAndroid || this.isIE || this.isIOS7; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-01 00:55:13 +02: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 17:47:17 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   get isChromeDesktop(): boolean { | 
					
						
							| 
									
										
										
										
											2016-05-13 13:22:29 -07:00
										 |  |  |     return this._ua.indexOf('Chrome') > -1 && this._ua.indexOf('Mobile Safari') == -1 && | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  |         this._ua.indexOf('Edge') == -1; | 
					
						
							| 
									
										
										
										
											2016-05-10 17:47:17 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-09-01 00:55:13 +02: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; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-12 15:58:13 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   get supportsCustomElements() { return (typeof(<any>global).customElements !== 'undefined'); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   get supportsDeprecatedCustomCustomElementsV0() { | 
					
						
							|  |  |  |     return (typeof(document as any).registerElement !== 'undefined'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-06 22:29:20 +08:00
										 |  |  |   get supportsRegExUnicodeFlag(): boolean { return RegExp.prototype.hasOwnProperty('unicode'); } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-12 15:58:13 -07:00
										 |  |  |   get supportsShadowDom() { | 
					
						
							|  |  |  |     const testEl = document.createElement('div'); | 
					
						
							| 
									
										
										
										
											2018-07-15 18:53:18 -07:00
										 |  |  |     return (typeof testEl.attachShadow !== 'undefined'); | 
					
						
							| 
									
										
										
										
											2018-07-12 15:58:13 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   get supportsDeprecatedShadowDomV0() { | 
					
						
							|  |  |  |     const testEl = document.createElement('div') as any; | 
					
						
							| 
									
										
										
										
											2018-07-15 18:53:18 -07:00
										 |  |  |     return (typeof testEl.createShadowRoot !== 'undefined'); | 
					
						
							| 
									
										
										
										
											2018-07-12 15:58:13 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-08-24 15:41:36 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-03 09:15:49 +01:00
										 |  |  | export const browserDetection: BrowserDetection = BrowserDetection.setup(); | 
					
						
							| 
									
										
										
										
											2016-02-26 12:25:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 22:36:21 -08:00
										 |  |  | export function dispatchEvent(element: any, eventType: any): void { | 
					
						
							| 
									
										
										
										
											2019-08-27 16:21:39 -07:00
										 |  |  |   const evt: Event = getDOM().getDefaultDocument().createEvent('Event'); | 
					
						
							|  |  |  |   evt.initEvent(eventType, true, true); | 
					
						
							|  |  |  |   getDOM().dispatchEvent(element, evt); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function createMouseEvent(eventType: string): MouseEvent { | 
					
						
							|  |  |  |   const evt: MouseEvent = getDOM().getDefaultDocument().createEvent('MouseEvent'); | 
					
						
							|  |  |  |   evt.initEvent(eventType, true, true); | 
					
						
							|  |  |  |   return evt; | 
					
						
							| 
									
										
										
										
											2015-02-03 07:27:09 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-07 20:03:00 -07:00
										 |  |  | export function el(html: string): HTMLElement { | 
					
						
							| 
									
										
										
										
											2019-08-24 08:01:24 -07:00
										 |  |  |   return <HTMLElement>getContent(createTemplate(html)).firstChild; | 
					
						
							| 
									
										
										
										
											2015-05-26 13:57:13 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-27 10:22:30 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | export function normalizeCSS(css: string): string { | 
					
						
							| 
									
										
										
										
											2016-10-06 15:10:27 -07:00
										 |  |  |   return css.replace(/\s+/g, ' ') | 
					
						
							|  |  |  |       .replace(/:\s/g, ':') | 
					
						
							|  |  |  |       .replace(/'/g, '"') | 
					
						
							|  |  |  |       .replace(/ }/g, '}') | 
					
						
							|  |  |  |       .replace(/url\((\"|\s)(.+)(\"|\s)\)(\s*)/g, (...match: string[]) => `url("${match[2]}")`) | 
					
						
							|  |  |  |       .replace(/\[(.+)=([^"\]]+)\]/g, (...match: string[]) => `[${match[1]}="${match[2]}"]`); | 
					
						
							| 
									
										
										
										
											2015-05-27 10:22:30 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-22 16:06:15 -07:00
										 |  |  | function getAttributeMap(element: any): Map<string, string> { | 
					
						
							|  |  |  |   const res = new Map<string, string>(); | 
					
						
							|  |  |  |   const elAttrs = element.attributes; | 
					
						
							|  |  |  |   for (let i = 0; i < elAttrs.length; i++) { | 
					
						
							|  |  |  |     const attrib = elAttrs.item(i); | 
					
						
							|  |  |  |     res.set(attrib.name, attrib.value); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-29 10:56:21 -07:00
										 |  |  | const _selfClosingTags = ['br', 'hr', 'input']; | 
					
						
							| 
									
										
										
										
											2016-06-08 15:45:15 -07:00
										 |  |  | export function stringifyElement(el: any /** TODO #9100 */): string { | 
					
						
							| 
									
										
										
										
											2016-11-12 14:08:58 +01:00
										 |  |  |   let result = ''; | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |   if (getDOM().isElementNode(el)) { | 
					
						
							| 
									
										
										
										
											2019-08-22 19:48:08 -07:00
										 |  |  |     const tagName = el.tagName.toLowerCase(); | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Opening tag
 | 
					
						
							|  |  |  |     result += `<${tagName}`; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Attributes in an ordered way
 | 
					
						
							| 
									
										
										
										
											2019-08-22 16:06:15 -07:00
										 |  |  |     const attributeMap = getAttributeMap(el); | 
					
						
							| 
									
										
										
										
											2018-08-22 19:17:59 +03:00
										 |  |  |     const sortedKeys = Array.from(attributeMap.keys()).sort(); | 
					
						
							|  |  |  |     for (const key of sortedKeys) { | 
					
						
							| 
									
										
										
										
											2018-01-03 14:05:33 +01:00
										 |  |  |       const lowerCaseKey = key.toLowerCase(); | 
					
						
							| 
									
										
										
										
											2018-08-22 19:17:59 +03:00
										 |  |  |       let attValue = attributeMap.get(key); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-19 13:42:39 -07:00
										 |  |  |       if (typeof attValue !== 'string') { | 
					
						
							| 
									
										
										
										
											2018-01-03 14:05:33 +01:00
										 |  |  |         result += ` ${lowerCaseKey}`; | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2018-08-22 19:17:59 +03:00
										 |  |  |         // Browsers order style rules differently. Order them alphabetically for consistency.
 | 
					
						
							|  |  |  |         if (lowerCaseKey === 'style') { | 
					
						
							|  |  |  |           attValue = attValue.split(/; ?/).filter(s => !!s).sort().map(s => `${s};`).join(' '); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-03 14:05:33 +01:00
										 |  |  |         result += ` ${lowerCaseKey}="${attValue}"`; | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     result += '>'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Children
 | 
					
						
							| 
									
										
										
										
											2019-08-22 19:48:08 -07:00
										 |  |  |     const childrenRoot = templateAwareRoot(el); | 
					
						
							| 
									
										
										
										
											2019-08-24 08:01:24 -07:00
										 |  |  |     const children = childrenRoot ? childrenRoot.childNodes : []; | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  |     for (let j = 0; j < children.length; j++) { | 
					
						
							|  |  |  |       result += stringifyElement(children[j]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Closing tag
 | 
					
						
							| 
									
										
										
										
											2019-03-29 10:56:21 -07:00
										 |  |  |     if (_selfClosingTags.indexOf(tagName) == -1) { | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  |       result += `</${tagName}>`; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-08-22 16:14:18 -07:00
										 |  |  |   } else if (isCommentNode(el)) { | 
					
						
							|  |  |  |     result += `<!--${el.nodeValue}-->`; | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2019-08-23 13:28:33 -07:00
										 |  |  |     result += el.textContent; | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return result; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-22 17:37:48 -07:00
										 |  |  | export function createNgZone(): NgZone { | 
					
						
							| 
									
										
										
										
											2019-05-17 20:50:02 +09:00
										 |  |  |   return new NgZone({enableLongStackTrace: true, shouldCoalesceEventChangeDetection: false}); | 
					
						
							| 
									
										
										
										
											2017-02-17 12:55:55 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-08-22 16:14:18 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | export function isCommentNode(node: Node): boolean { | 
					
						
							|  |  |  |   return node.nodeType === Node.COMMENT_NODE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-22 19:48:08 -07:00
										 |  |  | export function isTextNode(node: Node): boolean { | 
					
						
							|  |  |  |   return node.nodeType === Node.TEXT_NODE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function getContent(node: Node): Node { | 
					
						
							|  |  |  |   if ('content' in node) { | 
					
						
							|  |  |  |     return (<any>node).content; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     return node; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function templateAwareRoot(el: Node): any { | 
					
						
							|  |  |  |   return getDOM().isElementNode(el) && el.nodeName === 'TEMPLATE' ? getContent(el) : el; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-22 16:14:18 -07:00
										 |  |  | export function setCookie(name: string, value: string) { | 
					
						
							|  |  |  |   // document.cookie is magical, assigning into it assigns/overrides one cookie value, but does
 | 
					
						
							|  |  |  |   // not clear other cookies.
 | 
					
						
							|  |  |  |   document.cookie = encodeURIComponent(name) + '=' + encodeURIComponent(value); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-08-22 17:49:43 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | export function supportsWebAnimation(): boolean { | 
					
						
							|  |  |  |   return typeof(<any>Element).prototype['animate'] === 'function'; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-08-23 12:32:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | export function hasStyle(element: any, styleName: string, styleValue?: string | null): boolean { | 
					
						
							|  |  |  |   const value = element.style[styleName] || ''; | 
					
						
							|  |  |  |   return styleValue ? value == styleValue : value.length > 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function hasClass(element: any, className: string): boolean { | 
					
						
							|  |  |  |   return element.classList.contains(className); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function sortedClassList(element: any): any[] { | 
					
						
							|  |  |  |   return Array.prototype.slice.call(element.classList, 0).sort(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-08-23 13:28:33 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | export function createTemplate(html: any): HTMLElement { | 
					
						
							|  |  |  |   const t = getDOM().getDefaultDocument().createElement('template'); | 
					
						
							|  |  |  |   t.innerHTML = html; | 
					
						
							|  |  |  |   return t; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function childNodesAsList(el: Node): any[] { | 
					
						
							|  |  |  |   const childNodes = el.childNodes; | 
					
						
							|  |  |  |   const res = []; | 
					
						
							|  |  |  |   for (let i = 0; i < childNodes.length; i++) { | 
					
						
							|  |  |  |     res[i] = childNodes[i]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return res; | 
					
						
							|  |  |  | } |