| 
									
										
										
										
											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-04-28 17:50:03 -07:00
										 |  |  | import {getDOM} from '../src/dom/dom_adapter'; | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  | import {ListWrapper} from '../src/facade/collection'; | 
					
						
							| 
									
										
										
										
											2016-06-17 15:34:39 -07:00
										 |  |  | import {RegExp, RegExpWrapper, StringWrapper, global, isPresent, isString} from '../src/facade/lang'; | 
					
						
							| 
									
										
										
										
											2015-08-24 15:41:36 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | export class BrowserDetection { | 
					
						
							| 
									
										
										
										
											2016-05-01 22:09:58 -07:00
										 |  |  |   private _overrideUa: string; | 
					
						
							| 
									
										
										
										
											2016-05-01 22:50:37 -07:00
										 |  |  |   private get _ua(): string { | 
					
						
							| 
									
										
										
										
											2016-05-01 22:09:58 -07:00
										 |  |  |     if (isPresent(this._overrideUa)) { | 
					
						
							|  |  |  |       return this._overrideUa; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       return isPresent(getDOM()) ? getDOM().getUserAgent() : ''; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-08-24 15:41:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-15 16:38:27 -08:00
										 |  |  |   static setup() { browserDetection = new BrowserDetection(null); } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-01 22:50:37 -07:00
										 |  |  |   constructor(ua: string) { 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-06-08 16:38:52 -07:00
										 |  |  |         this._ua.indexOf('AppleWebKit') > -1 && this._ua.indexOf('Chrome') == -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 { | 
					
						
							|  |  |  |     return this._ua.indexOf('AppleWebKit') > -1 && this._ua.indexOf('Edge') == -1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-09 16:41:11 +02:00
										 |  |  |   get isIOS7(): boolean { | 
					
						
							|  |  |  |     return this._ua.indexOf('iPhone OS 7') > -1 || this._ua.indexOf('iPad OS 7') > -1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   get isSlow(): boolean { return this.isAndroid || this.isIE || this.isIOS7; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-24 15:41:36 +02:00
										 |  |  |   // The Intl API is only properly supported in recent Chrome and Opera.
 | 
					
						
							|  |  |  |   // Note: Edge is disguised as Chrome 42, so checking the "Edge" part is needed,
 | 
					
						
							|  |  |  |   // see https://msdn.microsoft.com/en-us/library/hh869301(v=vs.85).aspx
 | 
					
						
							| 
									
										
										
										
											2016-06-17 15:34:39 -07:00
										 |  |  |   get supportsIntlApi(): boolean { return !!(<any>global).Intl; } | 
					
						
							| 
									
										
										
										
											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
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-08-24 15:41:36 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-26 12:25:55 +01:00
										 |  |  | BrowserDetection.setup(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  | export function dispatchEvent( | 
					
						
							|  |  |  |     element: any /** TODO #9100 */, eventType: any /** TODO #9100 */): void { | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |   getDOM().dispatchEvent(element, getDOM().createEvent(eventType)); | 
					
						
							| 
									
										
										
										
											2015-02-03 07:27:09 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-07 20:03:00 -07:00
										 |  |  | export function el(html: string): HTMLElement { | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |   return <HTMLElement>getDOM().firstChild(getDOM().content(getDOM().createTemplate(html))); | 
					
						
							| 
									
										
										
										
											2015-05-26 13:57:13 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-27 10:22:30 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | export function normalizeCSS(css: string): string { | 
					
						
							| 
									
										
										
										
											2015-06-23 12:46:38 +02:00
										 |  |  |   css = StringWrapper.replaceAll(css, /\s+/g, ' '); | 
					
						
							|  |  |  |   css = StringWrapper.replaceAll(css, /:\s/g, ':'); | 
					
						
							| 
									
										
										
										
											2015-07-29 18:34:02 +02:00
										 |  |  |   css = StringWrapper.replaceAll(css, /'/g, '"'); | 
					
						
							| 
									
										
										
										
											2015-08-20 18:23:57 +02:00
										 |  |  |   css = StringWrapper.replaceAll(css, / }/g, '}'); | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07: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 10:22:30 +02:00
										 |  |  |   return css; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | var _singleTagWhitelist = ['br', 'hr', 'input']; | 
					
						
							| 
									
										
										
										
											2016-06-08 15:45:15 -07:00
										 |  |  | export function stringifyElement(el: any /** TODO #9100 */): string { | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  |   var result = ''; | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |   if (getDOM().isElementNode(el)) { | 
					
						
							|  |  |  |     var tagName = getDOM().tagName(el).toLowerCase(); | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Opening tag
 | 
					
						
							|  |  |  |     result += `<${tagName}`; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Attributes in an ordered way
 | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |     var attributeMap = getDOM().attributeMap(el); | 
					
						
							| 
									
										
										
										
											2016-06-08 15:45:15 -07:00
										 |  |  |     var keys: any[] /** TODO #9100 */ = []; | 
					
						
							| 
									
										
										
										
											2015-10-08 16:01:18 -07:00
										 |  |  |     attributeMap.forEach((v, k) => keys.push(k)); | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  |     ListWrapper.sort(keys); | 
					
						
							|  |  |  |     for (let i = 0; i < keys.length; i++) { | 
					
						
							|  |  |  |       var key = keys[i]; | 
					
						
							| 
									
										
										
										
											2015-06-17 16:21:40 -07:00
										 |  |  |       var attValue = attributeMap.get(key); | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  |       if (!isString(attValue)) { | 
					
						
							|  |  |  |         result += ` ${key}`; | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         result += ` ${key}="${attValue}"`; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     result += '>'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Children
 | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |     var childrenRoot = getDOM().templateAwareRoot(el); | 
					
						
							|  |  |  |     var children = isPresent(childrenRoot) ? getDOM().childNodes(childrenRoot) : []; | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02: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 17:50:03 -07:00
										 |  |  |   } else if (getDOM().isCommentNode(el)) { | 
					
						
							|  |  |  |     result += `<!--${getDOM().nodeValue(el)}-->`; | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |     result += getDOM().getText(el); | 
					
						
							| 
									
										
										
										
											2015-05-22 17:47:13 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return result; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | export var browserDetection: BrowserDetection = new BrowserDetection(null); |