| 
									
										
										
										
											2016-06-23 09:47:54 -07:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @license | 
					
						
							| 
									
										
										
										
											2020-05-19 12:08:49 -07:00
										 |  |  |  * Copyright Google LLC All Rights Reserved. | 
					
						
							| 
									
										
										
										
											2016-06-23 09:47:54 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * 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
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-16 09:00:03 -07:00
										 |  |  | import {CUSTOM_ELEMENTS_SCHEMA, NO_ERRORS_SCHEMA, SchemaMetadata, SecurityContext} from '../core'; | 
					
						
							| 
									
										
										
										
											2016-11-08 15:45:30 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-13 10:37:45 -07:00
										 |  |  | import {isNgContainer, isNgContent} from '../ml_parser/tags'; | 
					
						
							| 
									
										
										
										
											2016-11-08 15:45:30 -08:00
										 |  |  | import {dashCaseToCamelCase} from '../util'; | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-26 14:35:27 -07:00
										 |  |  | import {SECURITY_SCHEMA} from './dom_security_schema'; | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  | import {ElementSchemaRegistry} from './element_schema_registry'; | 
					
						
							| 
									
										
										
										
											2016-04-21 15:11:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  | const BOOLEAN = 'boolean'; | 
					
						
							|  |  |  | const NUMBER = 'number'; | 
					
						
							|  |  |  | const STRING = 'string'; | 
					
						
							|  |  |  | const OBJECT = 'object'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * This array represents the DOM schema. It encodes inheritance, properties, and events. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ## Overview | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Each line represents one kind of element. The `element_inheritance` and properties are joined | 
					
						
							| 
									
										
										
										
											2016-08-23 10:52:40 -07:00
										 |  |  |  * using `element_inheritance|properties` syntax. | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * ## Element Inheritance | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The `element_inheritance` can be further subdivided as `element1,element2,...^parentElement`. | 
					
						
							|  |  |  |  * Here the individual elements are separated by `,` (commas). Every element in the list | 
					
						
							|  |  |  |  * has identical properties. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * An `element` may inherit additional properties from `parentElement` If no `^parentElement` is | 
					
						
							|  |  |  |  * specified then `""` (blank) element is assumed. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-09-01 19:37:09 -07:00
										 |  |  |  * NOTE: The blank element inherits from root `[Element]` element, the super element of all | 
					
						
							|  |  |  |  * elements. | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-05-26 16:17:35 -07:00
										 |  |  |  * NOTE an element prefix such as `:svg:` has no special meaning to the schema. | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * ## Properties | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Each element has a set of properties separated by `,` (commas). Each property can be prefixed | 
					
						
							|  |  |  |  * by a special character designating its type: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * - (no prefix): property is a string. | 
					
						
							|  |  |  |  * - `*`: property represents an event. | 
					
						
							|  |  |  |  * - `!`: property is a boolean. | 
					
						
							|  |  |  |  * - `#`: property is a number. | 
					
						
							|  |  |  |  * - `%`: property is an object. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ## Query | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-08-23 10:52:40 -07:00
										 |  |  |  * The class creates an internal squas representation which allows to easily answer the query of | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |  * if a given property exist on a given element. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * NOTE: We don't yet support querying for types or events. | 
					
						
							| 
									
										
										
										
											2016-05-26 16:17:35 -07:00
										 |  |  |  * NOTE: This schema is auto extracted from `schema_extractor.ts` located in the test folder, | 
					
						
							|  |  |  |  *       see dom_element_schema_registry_spec.ts | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-05-26 14:35:27 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | // =================================================================================================
 | 
					
						
							|  |  |  | // =================================================================================================
 | 
					
						
							|  |  |  | // =========== S T O P   -  S T O P   -  S T O P   -  S T O P   -  S T O P   -  S T O P  ===========
 | 
					
						
							|  |  |  | // =================================================================================================
 | 
					
						
							|  |  |  | // =================================================================================================
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //                       DO NOT EDIT THIS DOM SCHEMA WITHOUT A SECURITY REVIEW!
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Newly added properties must be security reviewed and assigned an appropriate SecurityContext in
 | 
					
						
							| 
									
										
										
										
											2016-06-02 10:27:33 -07:00
										 |  |  | // dom_security_schema.ts. Reach out to mprobst & rjamet for details.
 | 
					
						
							| 
									
										
										
										
											2016-05-26 14:35:27 -07:00
										 |  |  | //
 | 
					
						
							|  |  |  | // =================================================================================================
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-06 12:11:34 -07:00
										 |  |  | const SCHEMA: string[] = [ | 
					
						
							|  |  |  |   '[Element]|textContent,%classList,className,id,innerHTML,*beforecopy,*beforecut,*beforepaste,*copy,*cut,*paste,*search,*selectstart,*webkitfullscreenchange,*webkitfullscreenerror,*wheel,outerHTML,#scrollLeft,#scrollTop,slot' + | 
					
						
							|  |  |  |       /* added manually to avoid breaking changes */ | 
					
						
							|  |  |  |       ',*message,*mozfullscreenchange,*mozfullscreenerror,*mozpointerlockchange,*mozpointerlockerror,*webglcontextcreationerror,*webglcontextlost,*webglcontextrestored', | 
					
						
							|  |  |  |   '[HTMLElement]^[Element]|accessKey,contentEditable,dir,!draggable,!hidden,innerText,lang,*abort,*auxclick,*blur,*cancel,*canplay,*canplaythrough,*change,*click,*close,*contextmenu,*cuechange,*dblclick,*drag,*dragend,*dragenter,*dragleave,*dragover,*dragstart,*drop,*durationchange,*emptied,*ended,*error,*focus,*gotpointercapture,*input,*invalid,*keydown,*keypress,*keyup,*load,*loadeddata,*loadedmetadata,*loadstart,*lostpointercapture,*mousedown,*mouseenter,*mouseleave,*mousemove,*mouseout,*mouseover,*mouseup,*mousewheel,*pause,*play,*playing,*pointercancel,*pointerdown,*pointerenter,*pointerleave,*pointermove,*pointerout,*pointerover,*pointerup,*progress,*ratechange,*reset,*resize,*scroll,*seeked,*seeking,*select,*show,*stalled,*submit,*suspend,*timeupdate,*toggle,*volumechange,*waiting,outerText,!spellcheck,%style,#tabIndex,title,!translate', | 
					
						
							|  |  |  |   'abbr,address,article,aside,b,bdi,bdo,cite,code,dd,dfn,dt,em,figcaption,figure,footer,header,i,kbd,main,mark,nav,noscript,rb,rp,rt,rtc,ruby,s,samp,section,small,strong,sub,sup,u,var,wbr^[HTMLElement]|accessKey,contentEditable,dir,!draggable,!hidden,innerText,lang,*abort,*auxclick,*blur,*cancel,*canplay,*canplaythrough,*change,*click,*close,*contextmenu,*cuechange,*dblclick,*drag,*dragend,*dragenter,*dragleave,*dragover,*dragstart,*drop,*durationchange,*emptied,*ended,*error,*focus,*gotpointercapture,*input,*invalid,*keydown,*keypress,*keyup,*load,*loadeddata,*loadedmetadata,*loadstart,*lostpointercapture,*mousedown,*mouseenter,*mouseleave,*mousemove,*mouseout,*mouseover,*mouseup,*mousewheel,*pause,*play,*playing,*pointercancel,*pointerdown,*pointerenter,*pointerleave,*pointermove,*pointerout,*pointerover,*pointerup,*progress,*ratechange,*reset,*resize,*scroll,*seeked,*seeking,*select,*show,*stalled,*submit,*suspend,*timeupdate,*toggle,*volumechange,*waiting,outerText,!spellcheck,%style,#tabIndex,title,!translate', | 
					
						
							|  |  |  |   'media^[HTMLElement]|!autoplay,!controls,%controlsList,%crossOrigin,#currentTime,!defaultMuted,#defaultPlaybackRate,!disableRemotePlayback,!loop,!muted,*encrypted,*waitingforkey,#playbackRate,preload,src,%srcObject,#volume', | 
					
						
							|  |  |  |   ':svg:^[HTMLElement]|*abort,*auxclick,*blur,*cancel,*canplay,*canplaythrough,*change,*click,*close,*contextmenu,*cuechange,*dblclick,*drag,*dragend,*dragenter,*dragleave,*dragover,*dragstart,*drop,*durationchange,*emptied,*ended,*error,*focus,*gotpointercapture,*input,*invalid,*keydown,*keypress,*keyup,*load,*loadeddata,*loadedmetadata,*loadstart,*lostpointercapture,*mousedown,*mouseenter,*mouseleave,*mousemove,*mouseout,*mouseover,*mouseup,*mousewheel,*pause,*play,*playing,*pointercancel,*pointerdown,*pointerenter,*pointerleave,*pointermove,*pointerout,*pointerover,*pointerup,*progress,*ratechange,*reset,*resize,*scroll,*seeked,*seeking,*select,*show,*stalled,*submit,*suspend,*timeupdate,*toggle,*volumechange,*waiting,%style,#tabIndex', | 
					
						
							|  |  |  |   ':svg:graphics^:svg:|', | 
					
						
							|  |  |  |   ':svg:animation^:svg:|*begin,*end,*repeat', | 
					
						
							|  |  |  |   ':svg:geometry^:svg:|', | 
					
						
							|  |  |  |   ':svg:componentTransferFunction^:svg:|', | 
					
						
							|  |  |  |   ':svg:gradient^:svg:|', | 
					
						
							|  |  |  |   ':svg:textContent^:svg:graphics|', | 
					
						
							|  |  |  |   ':svg:textPositioning^:svg:textContent|', | 
					
						
							|  |  |  |   'a^[HTMLElement]|charset,coords,download,hash,host,hostname,href,hreflang,name,password,pathname,ping,port,protocol,referrerPolicy,rel,rev,search,shape,target,text,type,username', | 
					
						
							|  |  |  |   'area^[HTMLElement]|alt,coords,download,hash,host,hostname,href,!noHref,password,pathname,ping,port,protocol,referrerPolicy,rel,search,shape,target,username', | 
					
						
							|  |  |  |   'audio^media|', | 
					
						
							|  |  |  |   'br^[HTMLElement]|clear', | 
					
						
							|  |  |  |   'base^[HTMLElement]|href,target', | 
					
						
							|  |  |  |   'body^[HTMLElement]|aLink,background,bgColor,link,*beforeunload,*blur,*error,*focus,*hashchange,*languagechange,*load,*message,*offline,*online,*pagehide,*pageshow,*popstate,*rejectionhandled,*resize,*scroll,*storage,*unhandledrejection,*unload,text,vLink', | 
					
						
							|  |  |  |   'button^[HTMLElement]|!autofocus,!disabled,formAction,formEnctype,formMethod,!formNoValidate,formTarget,name,type,value', | 
					
						
							|  |  |  |   'canvas^[HTMLElement]|#height,#width', | 
					
						
							|  |  |  |   'content^[HTMLElement]|select', | 
					
						
							|  |  |  |   'dl^[HTMLElement]|!compact', | 
					
						
							|  |  |  |   'datalist^[HTMLElement]|', | 
					
						
							|  |  |  |   'details^[HTMLElement]|!open', | 
					
						
							|  |  |  |   'dialog^[HTMLElement]|!open,returnValue', | 
					
						
							|  |  |  |   'dir^[HTMLElement]|!compact', | 
					
						
							|  |  |  |   'div^[HTMLElement]|align', | 
					
						
							|  |  |  |   'embed^[HTMLElement]|align,height,name,src,type,width', | 
					
						
							|  |  |  |   'fieldset^[HTMLElement]|!disabled,name', | 
					
						
							|  |  |  |   'font^[HTMLElement]|color,face,size', | 
					
						
							|  |  |  |   'form^[HTMLElement]|acceptCharset,action,autocomplete,encoding,enctype,method,name,!noValidate,target', | 
					
						
							|  |  |  |   'frame^[HTMLElement]|frameBorder,longDesc,marginHeight,marginWidth,name,!noResize,scrolling,src', | 
					
						
							|  |  |  |   'frameset^[HTMLElement]|cols,*beforeunload,*blur,*error,*focus,*hashchange,*languagechange,*load,*message,*offline,*online,*pagehide,*pageshow,*popstate,*rejectionhandled,*resize,*scroll,*storage,*unhandledrejection,*unload,rows', | 
					
						
							|  |  |  |   'hr^[HTMLElement]|align,color,!noShade,size,width', | 
					
						
							|  |  |  |   'head^[HTMLElement]|', | 
					
						
							|  |  |  |   'h1,h2,h3,h4,h5,h6^[HTMLElement]|align', | 
					
						
							|  |  |  |   'html^[HTMLElement]|version', | 
					
						
							|  |  |  |   'iframe^[HTMLElement]|align,!allowFullscreen,frameBorder,height,longDesc,marginHeight,marginWidth,name,referrerPolicy,%sandbox,scrolling,src,srcdoc,width', | 
					
						
							|  |  |  |   'img^[HTMLElement]|align,alt,border,%crossOrigin,#height,#hspace,!isMap,longDesc,lowsrc,name,referrerPolicy,sizes,src,srcset,useMap,#vspace,#width', | 
					
						
							|  |  |  |   'input^[HTMLElement]|accept,align,alt,autocapitalize,autocomplete,!autofocus,!checked,!defaultChecked,defaultValue,dirName,!disabled,%files,formAction,formEnctype,formMethod,!formNoValidate,formTarget,#height,!incremental,!indeterminate,max,#maxLength,min,#minLength,!multiple,name,pattern,placeholder,!readOnly,!required,selectionDirection,#selectionEnd,#selectionStart,#size,src,step,type,useMap,value,%valueAsDate,#valueAsNumber,#width', | 
					
						
							|  |  |  |   'li^[HTMLElement]|type,#value', | 
					
						
							|  |  |  |   'label^[HTMLElement]|htmlFor', | 
					
						
							|  |  |  |   'legend^[HTMLElement]|align', | 
					
						
							|  |  |  |   'link^[HTMLElement]|as,charset,%crossOrigin,!disabled,href,hreflang,integrity,media,referrerPolicy,rel,%relList,rev,%sizes,target,type', | 
					
						
							|  |  |  |   'map^[HTMLElement]|name', | 
					
						
							|  |  |  |   'marquee^[HTMLElement]|behavior,bgColor,direction,height,#hspace,#loop,#scrollAmount,#scrollDelay,!trueSpeed,#vspace,width', | 
					
						
							|  |  |  |   'menu^[HTMLElement]|!compact', | 
					
						
							|  |  |  |   'meta^[HTMLElement]|content,httpEquiv,name,scheme', | 
					
						
							|  |  |  |   'meter^[HTMLElement]|#high,#low,#max,#min,#optimum,#value', | 
					
						
							|  |  |  |   'ins,del^[HTMLElement]|cite,dateTime', | 
					
						
							|  |  |  |   'ol^[HTMLElement]|!compact,!reversed,#start,type', | 
					
						
							|  |  |  |   'object^[HTMLElement]|align,archive,border,code,codeBase,codeType,data,!declare,height,#hspace,name,standby,type,useMap,#vspace,width', | 
					
						
							|  |  |  |   'optgroup^[HTMLElement]|!disabled,label', | 
					
						
							|  |  |  |   'option^[HTMLElement]|!defaultSelected,!disabled,label,!selected,text,value', | 
					
						
							|  |  |  |   'output^[HTMLElement]|defaultValue,%htmlFor,name,value', | 
					
						
							|  |  |  |   'p^[HTMLElement]|align', | 
					
						
							|  |  |  |   'param^[HTMLElement]|name,type,value,valueType', | 
					
						
							|  |  |  |   'picture^[HTMLElement]|', | 
					
						
							|  |  |  |   'pre^[HTMLElement]|#width', | 
					
						
							|  |  |  |   'progress^[HTMLElement]|#max,#value', | 
					
						
							|  |  |  |   'q,blockquote,cite^[HTMLElement]|', | 
					
						
							|  |  |  |   'script^[HTMLElement]|!async,charset,%crossOrigin,!defer,event,htmlFor,integrity,src,text,type', | 
					
						
							| 
									
										
										
										
											2021-02-20 10:29:35 +01:00
										 |  |  |   'select^[HTMLElement]|autocomplete,!autofocus,!disabled,#length,!multiple,name,!required,#selectedIndex,#size,value', | 
					
						
							| 
									
										
										
										
											2017-07-06 12:11:34 -07:00
										 |  |  |   'shadow^[HTMLElement]|', | 
					
						
							|  |  |  |   'slot^[HTMLElement]|name', | 
					
						
							|  |  |  |   'source^[HTMLElement]|media,sizes,src,srcset,type', | 
					
						
							|  |  |  |   'span^[HTMLElement]|', | 
					
						
							|  |  |  |   'style^[HTMLElement]|!disabled,media,type', | 
					
						
							|  |  |  |   'caption^[HTMLElement]|align', | 
					
						
							|  |  |  |   'th,td^[HTMLElement]|abbr,align,axis,bgColor,ch,chOff,#colSpan,headers,height,!noWrap,#rowSpan,scope,vAlign,width', | 
					
						
							|  |  |  |   'col,colgroup^[HTMLElement]|align,ch,chOff,#span,vAlign,width', | 
					
						
							|  |  |  |   'table^[HTMLElement]|align,bgColor,border,%caption,cellPadding,cellSpacing,frame,rules,summary,%tFoot,%tHead,width', | 
					
						
							|  |  |  |   'tr^[HTMLElement]|align,bgColor,ch,chOff,vAlign', | 
					
						
							|  |  |  |   'tfoot,thead,tbody^[HTMLElement]|align,ch,chOff,vAlign', | 
					
						
							|  |  |  |   'template^[HTMLElement]|', | 
					
						
							| 
									
										
										
										
											2021-02-20 10:29:35 +01:00
										 |  |  |   'textarea^[HTMLElement]|autocapitalize,autocomplete,!autofocus,#cols,defaultValue,dirName,!disabled,#maxLength,#minLength,name,placeholder,!readOnly,!required,#rows,selectionDirection,#selectionEnd,#selectionStart,value,wrap', | 
					
						
							| 
									
										
										
										
											2017-07-06 12:11:34 -07:00
										 |  |  |   'title^[HTMLElement]|text', | 
					
						
							|  |  |  |   'track^[HTMLElement]|!default,kind,label,src,srclang', | 
					
						
							|  |  |  |   'ul^[HTMLElement]|!compact,type', | 
					
						
							|  |  |  |   'unknown^[HTMLElement]|', | 
					
						
							|  |  |  |   'video^media|#height,poster,#width', | 
					
						
							|  |  |  |   ':svg:a^:svg:graphics|', | 
					
						
							|  |  |  |   ':svg:animate^:svg:animation|', | 
					
						
							|  |  |  |   ':svg:animateMotion^:svg:animation|', | 
					
						
							|  |  |  |   ':svg:animateTransform^:svg:animation|', | 
					
						
							|  |  |  |   ':svg:circle^:svg:geometry|', | 
					
						
							|  |  |  |   ':svg:clipPath^:svg:graphics|', | 
					
						
							|  |  |  |   ':svg:defs^:svg:graphics|', | 
					
						
							|  |  |  |   ':svg:desc^:svg:|', | 
					
						
							|  |  |  |   ':svg:discard^:svg:|', | 
					
						
							|  |  |  |   ':svg:ellipse^:svg:geometry|', | 
					
						
							|  |  |  |   ':svg:feBlend^:svg:|', | 
					
						
							|  |  |  |   ':svg:feColorMatrix^:svg:|', | 
					
						
							|  |  |  |   ':svg:feComponentTransfer^:svg:|', | 
					
						
							|  |  |  |   ':svg:feComposite^:svg:|', | 
					
						
							|  |  |  |   ':svg:feConvolveMatrix^:svg:|', | 
					
						
							|  |  |  |   ':svg:feDiffuseLighting^:svg:|', | 
					
						
							|  |  |  |   ':svg:feDisplacementMap^:svg:|', | 
					
						
							|  |  |  |   ':svg:feDistantLight^:svg:|', | 
					
						
							|  |  |  |   ':svg:feDropShadow^:svg:|', | 
					
						
							|  |  |  |   ':svg:feFlood^:svg:|', | 
					
						
							|  |  |  |   ':svg:feFuncA^:svg:componentTransferFunction|', | 
					
						
							|  |  |  |   ':svg:feFuncB^:svg:componentTransferFunction|', | 
					
						
							|  |  |  |   ':svg:feFuncG^:svg:componentTransferFunction|', | 
					
						
							|  |  |  |   ':svg:feFuncR^:svg:componentTransferFunction|', | 
					
						
							|  |  |  |   ':svg:feGaussianBlur^:svg:|', | 
					
						
							|  |  |  |   ':svg:feImage^:svg:|', | 
					
						
							|  |  |  |   ':svg:feMerge^:svg:|', | 
					
						
							|  |  |  |   ':svg:feMergeNode^:svg:|', | 
					
						
							|  |  |  |   ':svg:feMorphology^:svg:|', | 
					
						
							|  |  |  |   ':svg:feOffset^:svg:|', | 
					
						
							|  |  |  |   ':svg:fePointLight^:svg:|', | 
					
						
							|  |  |  |   ':svg:feSpecularLighting^:svg:|', | 
					
						
							|  |  |  |   ':svg:feSpotLight^:svg:|', | 
					
						
							|  |  |  |   ':svg:feTile^:svg:|', | 
					
						
							|  |  |  |   ':svg:feTurbulence^:svg:|', | 
					
						
							|  |  |  |   ':svg:filter^:svg:|', | 
					
						
							|  |  |  |   ':svg:foreignObject^:svg:graphics|', | 
					
						
							|  |  |  |   ':svg:g^:svg:graphics|', | 
					
						
							|  |  |  |   ':svg:image^:svg:graphics|', | 
					
						
							|  |  |  |   ':svg:line^:svg:geometry|', | 
					
						
							|  |  |  |   ':svg:linearGradient^:svg:gradient|', | 
					
						
							|  |  |  |   ':svg:mpath^:svg:|', | 
					
						
							|  |  |  |   ':svg:marker^:svg:|', | 
					
						
							|  |  |  |   ':svg:mask^:svg:|', | 
					
						
							|  |  |  |   ':svg:metadata^:svg:|', | 
					
						
							|  |  |  |   ':svg:path^:svg:geometry|', | 
					
						
							|  |  |  |   ':svg:pattern^:svg:|', | 
					
						
							|  |  |  |   ':svg:polygon^:svg:geometry|', | 
					
						
							|  |  |  |   ':svg:polyline^:svg:geometry|', | 
					
						
							|  |  |  |   ':svg:radialGradient^:svg:gradient|', | 
					
						
							|  |  |  |   ':svg:rect^:svg:geometry|', | 
					
						
							|  |  |  |   ':svg:svg^:svg:graphics|#currentScale,#zoomAndPan', | 
					
						
							|  |  |  |   ':svg:script^:svg:|type', | 
					
						
							|  |  |  |   ':svg:set^:svg:animation|', | 
					
						
							|  |  |  |   ':svg:stop^:svg:|', | 
					
						
							|  |  |  |   ':svg:style^:svg:|!disabled,media,title,type', | 
					
						
							|  |  |  |   ':svg:switch^:svg:graphics|', | 
					
						
							|  |  |  |   ':svg:symbol^:svg:|', | 
					
						
							|  |  |  |   ':svg:tspan^:svg:textPositioning|', | 
					
						
							|  |  |  |   ':svg:text^:svg:textPositioning|', | 
					
						
							|  |  |  |   ':svg:textPath^:svg:textContent|', | 
					
						
							|  |  |  |   ':svg:title^:svg:|', | 
					
						
							|  |  |  |   ':svg:use^:svg:graphics|', | 
					
						
							|  |  |  |   ':svg:view^:svg:|#zoomAndPan', | 
					
						
							|  |  |  |   'data^[HTMLElement]|value', | 
					
						
							|  |  |  |   'keygen^[HTMLElement]|!autofocus,challenge,!disabled,form,keytype,name', | 
					
						
							|  |  |  |   'menuitem^[HTMLElement]|type,label,icon,!disabled,!checked,radiogroup,!default', | 
					
						
							|  |  |  |   'summary^[HTMLElement]|', | 
					
						
							|  |  |  |   'time^[HTMLElement]|dateTime', | 
					
						
							|  |  |  |   ':svg:cursor^:svg:|', | 
					
						
							|  |  |  | ]; | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-23 10:52:40 -07:00
										 |  |  | const _ATTR_TO_PROP: {[name: string]: string} = { | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |   'class': 'className', | 
					
						
							| 
									
										
										
										
											2016-11-10 00:21:27 +01:00
										 |  |  |   'for': 'htmlFor', | 
					
						
							| 
									
										
										
										
											2016-05-26 14:35:27 -07:00
										 |  |  |   'formaction': 'formAction', | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |   'innerHtml': 'innerHTML', | 
					
						
							|  |  |  |   'readonly': 'readOnly', | 
					
						
							| 
									
										
										
										
											2016-09-01 19:37:09 -07:00
										 |  |  |   'tabindex': 'tabIndex', | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-04 14:43:26 -08:00
										 |  |  | // Invert _ATTR_TO_PROP.
 | 
					
						
							|  |  |  | const _PROP_TO_ATTR: {[name: string]: string} = | 
					
						
							|  |  |  |     Object.keys(_ATTR_TO_PROP).reduce((inverted, attr) => { | 
					
						
							|  |  |  |       inverted[_ATTR_TO_PROP[attr]] = attr; | 
					
						
							|  |  |  |       return inverted; | 
					
						
							|  |  |  |     }, {} as {[prop: string]: string}); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												feat: security implementation in Angular 2.
Summary:
This adds basic security hooks to Angular 2.
* `SecurityContext` is a private API between core, compiler, and
  platform-browser. `SecurityContext` communicates what context a value is used
  in across template parser, compiler, and sanitization at runtime.
* `SanitizationService` is the bare bones interface to sanitize values for a
  particular context.
* `SchemaElementRegistry.securityContext(tagName, attributeOrPropertyName)`
  determines the security context for an attribute or property (it turns out
  attributes and properties match for the purposes of sanitization).
Based on these hooks:
* `DomSchemaElementRegistry` decides what sanitization applies in a particular
  context.
* `DomSanitizationService` implements `SanitizationService` and adds *Safe
  Value*s, i.e. the ability to mark a value as safe and not requiring further
  sanitization.
* `url_sanitizer` and `style_sanitizer` sanitize URLs and Styles, respectively
  (surprise!).
`DomSanitizationService` is the default implementation bound for browser
applications, in the three contexts (browser rendering, web worker rendering,
server side rendering).
BREAKING CHANGES:
*** SECURITY WARNING ***
Angular 2 Release Candidates do not implement proper contextual escaping yet.
Make sure to correctly escape all values that go into the DOM.
*** SECURITY WARNING ***
Reviewers: IgorMinar
Differential Revision: https://reviews.angular.io/D103
											
										 
											2016-04-29 16:04:08 -07:00
										 |  |  | export class DomElementSchemaRegistry extends ElementSchemaRegistry { | 
					
						
							| 
									
										
										
										
											2016-08-23 10:52:40 -07:00
										 |  |  |   private _schema: {[element: string]: {[property: string]: string}} = {}; | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   constructor() { | 
					
						
							| 
									
										
											  
											
												feat: security implementation in Angular 2.
Summary:
This adds basic security hooks to Angular 2.
* `SecurityContext` is a private API between core, compiler, and
  platform-browser. `SecurityContext` communicates what context a value is used
  in across template parser, compiler, and sanitization at runtime.
* `SanitizationService` is the bare bones interface to sanitize values for a
  particular context.
* `SchemaElementRegistry.securityContext(tagName, attributeOrPropertyName)`
  determines the security context for an attribute or property (it turns out
  attributes and properties match for the purposes of sanitization).
Based on these hooks:
* `DomSchemaElementRegistry` decides what sanitization applies in a particular
  context.
* `DomSanitizationService` implements `SanitizationService` and adds *Safe
  Value*s, i.e. the ability to mark a value as safe and not requiring further
  sanitization.
* `url_sanitizer` and `style_sanitizer` sanitize URLs and Styles, respectively
  (surprise!).
`DomSanitizationService` is the default implementation bound for browser
applications, in the three contexts (browser rendering, web worker rendering,
server side rendering).
BREAKING CHANGES:
*** SECURITY WARNING ***
Angular 2 Release Candidates do not implement proper contextual escaping yet.
Make sure to correctly escape all values that go into the DOM.
*** SECURITY WARNING ***
Reviewers: IgorMinar
Differential Revision: https://reviews.angular.io/D103
											
										 
											2016-04-29 16:04:08 -07:00
										 |  |  |     super(); | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |     SCHEMA.forEach(encodedType => { | 
					
						
							| 
									
										
										
										
											2016-11-12 14:08:58 +01:00
										 |  |  |       const type: {[property: string]: string} = {}; | 
					
						
							| 
									
										
										
										
											2016-08-23 10:52:40 -07:00
										 |  |  |       const [strType, strProperties] = encodedType.split('|'); | 
					
						
							|  |  |  |       const properties = strProperties.split(','); | 
					
						
							|  |  |  |       const [typeNames, superName] = strType.split('^'); | 
					
						
							|  |  |  |       typeNames.split(',').forEach(tag => this._schema[tag.toLowerCase()] = type); | 
					
						
							| 
									
										
										
										
											2016-09-01 17:32:47 -07:00
										 |  |  |       const superType = superName && this._schema[superName.toLowerCase()]; | 
					
						
							| 
									
										
										
										
											2016-08-23 10:52:40 -07:00
										 |  |  |       if (superType) { | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |         Object.keys(superType).forEach((prop: string) => { | 
					
						
							|  |  |  |           type[prop] = superType[prop]; | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |       } | 
					
						
							|  |  |  |       properties.forEach((property: string) => { | 
					
						
							| 
									
										
										
										
											2016-08-23 10:52:40 -07:00
										 |  |  |         if (property.length > 0) { | 
					
						
							|  |  |  |           switch (property[0]) { | 
					
						
							|  |  |  |             case '*': | 
					
						
							|  |  |  |               // We don't yet support events.
 | 
					
						
							|  |  |  |               // If ever allowing to bind to events, GO THROUGH A SECURITY REVIEW, allowing events
 | 
					
						
							|  |  |  |               // will
 | 
					
						
							|  |  |  |               // almost certainly introduce bad XSS vulnerabilities.
 | 
					
						
							|  |  |  |               // type[property.substring(1)] = EVENT;
 | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |             case '!': | 
					
						
							|  |  |  |               type[property.substring(1)] = BOOLEAN; | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |             case '#': | 
					
						
							|  |  |  |               type[property.substring(1)] = NUMBER; | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |             case '%': | 
					
						
							|  |  |  |               type[property.substring(1)] = OBJECT; | 
					
						
							|  |  |  |               break; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |               type[property] = STRING; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-08-27 16:29:02 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-25 03:02:57 -07:00
										 |  |  |   hasProperty(tagName: string, propName: string, schemaMetas: SchemaMetadata[]): boolean { | 
					
						
							| 
									
										
										
										
											2016-08-19 16:05:34 -07:00
										 |  |  |     if (schemaMetas.some((schema) => schema.name === NO_ERRORS_SCHEMA.name)) { | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-23 10:52:40 -07:00
										 |  |  |     if (tagName.indexOf('-') > -1) { | 
					
						
							| 
									
										
										
										
											2017-04-13 10:37:45 -07:00
										 |  |  |       if (isNgContainer(tagName) || isNgContent(tagName)) { | 
					
						
							| 
									
										
										
										
											2016-06-15 09:57:58 -07:00
										 |  |  |         return false; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2016-08-23 10:52:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-26 11:25:38 -07:00
										 |  |  |       if (schemaMetas.some((schema) => schema.name === CUSTOM_ELEMENTS_SCHEMA.name)) { | 
					
						
							|  |  |  |         // Can't tell now as we don't know which properties a custom element will get
 | 
					
						
							|  |  |  |         // once it is instantiated
 | 
					
						
							|  |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2015-07-29 14:01:18 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-08-19 16:05:34 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-23 10:52:40 -07:00
										 |  |  |     const elementProperties = this._schema[tagName.toLowerCase()] || this._schema['unknown']; | 
					
						
							|  |  |  |     return !!elementProperties[propName]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   hasElement(tagName: string, schemaMetas: SchemaMetadata[]): boolean { | 
					
						
							|  |  |  |     if (schemaMetas.some((schema) => schema.name === NO_ERRORS_SCHEMA.name)) { | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (tagName.indexOf('-') > -1) { | 
					
						
							| 
									
										
										
										
											2017-04-13 10:37:45 -07:00
										 |  |  |       if (isNgContainer(tagName) || isNgContent(tagName)) { | 
					
						
							| 
									
										
										
										
											2016-08-23 10:52:40 -07:00
										 |  |  |         return true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (schemaMetas.some((schema) => schema.name === CUSTOM_ELEMENTS_SCHEMA.name)) { | 
					
						
							|  |  |  |         // Allow any custom elements
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2016-07-26 11:25:38 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-08-23 10:52:40 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return !!this._schema[tagName.toLowerCase()]; | 
					
						
							| 
									
										
										
										
											2015-07-29 14:01:18 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												feat: security implementation in Angular 2.
Summary:
This adds basic security hooks to Angular 2.
* `SecurityContext` is a private API between core, compiler, and
  platform-browser. `SecurityContext` communicates what context a value is used
  in across template parser, compiler, and sanitization at runtime.
* `SanitizationService` is the bare bones interface to sanitize values for a
  particular context.
* `SchemaElementRegistry.securityContext(tagName, attributeOrPropertyName)`
  determines the security context for an attribute or property (it turns out
  attributes and properties match for the purposes of sanitization).
Based on these hooks:
* `DomSchemaElementRegistry` decides what sanitization applies in a particular
  context.
* `DomSanitizationService` implements `SanitizationService` and adds *Safe
  Value*s, i.e. the ability to mark a value as safe and not requiring further
  sanitization.
* `url_sanitizer` and `style_sanitizer` sanitize URLs and Styles, respectively
  (surprise!).
`DomSanitizationService` is the default implementation bound for browser
applications, in the three contexts (browser rendering, web worker rendering,
server side rendering).
BREAKING CHANGES:
*** SECURITY WARNING ***
Angular 2 Release Candidates do not implement proper contextual escaping yet.
Make sure to correctly escape all values that go into the DOM.
*** SECURITY WARNING ***
Reviewers: IgorMinar
Differential Revision: https://reviews.angular.io/D103
											
										 
											2016-04-29 16:04:08 -07:00
										 |  |  |   /** | 
					
						
							|  |  |  |    * securityContext returns the security context for the given property on the given DOM tag. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Tag and property name are statically known and cannot change at runtime, i.e. it is not | 
					
						
							|  |  |  |    * possible to bind a value into a changing attribute or tag name. | 
					
						
							|  |  |  |    * | 
					
						
							| 
									
										
										
										
											2019-03-29 12:52:56 -07:00
										 |  |  |    * The filtering is based on a list of allowed tags|attributes. All attributes in the schema | 
					
						
							|  |  |  |    * above are assumed to have the 'NONE' security context, i.e. that they are safe inert | 
					
						
							|  |  |  |    * string values. Only specific well known attack vectors are assigned their appropriate context. | 
					
						
							| 
									
										
											  
											
												feat: security implementation in Angular 2.
Summary:
This adds basic security hooks to Angular 2.
* `SecurityContext` is a private API between core, compiler, and
  platform-browser. `SecurityContext` communicates what context a value is used
  in across template parser, compiler, and sanitization at runtime.
* `SanitizationService` is the bare bones interface to sanitize values for a
  particular context.
* `SchemaElementRegistry.securityContext(tagName, attributeOrPropertyName)`
  determines the security context for an attribute or property (it turns out
  attributes and properties match for the purposes of sanitization).
Based on these hooks:
* `DomSchemaElementRegistry` decides what sanitization applies in a particular
  context.
* `DomSanitizationService` implements `SanitizationService` and adds *Safe
  Value*s, i.e. the ability to mark a value as safe and not requiring further
  sanitization.
* `url_sanitizer` and `style_sanitizer` sanitize URLs and Styles, respectively
  (surprise!).
`DomSanitizationService` is the default implementation bound for browser
applications, in the three contexts (browser rendering, web worker rendering,
server side rendering).
BREAKING CHANGES:
*** SECURITY WARNING ***
Angular 2 Release Candidates do not implement proper contextual escaping yet.
Make sure to correctly escape all values that go into the DOM.
*** SECURITY WARNING ***
Reviewers: IgorMinar
Differential Revision: https://reviews.angular.io/D103
											
										 
											2016-04-29 16:04:08 -07:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2016-10-24 09:58:52 -07:00
										 |  |  |   securityContext(tagName: string, propName: string, isAttribute: boolean): SecurityContext { | 
					
						
							|  |  |  |     if (isAttribute) { | 
					
						
							|  |  |  |       // NB: For security purposes, use the mapped property name, not the attribute name.
 | 
					
						
							|  |  |  |       propName = this.getMappedPropName(propName); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-26 14:35:27 -07:00
										 |  |  |     // Make sure comparisons are case insensitive, so that case differences between attribute and
 | 
					
						
							|  |  |  |     // property names do not have a security impact.
 | 
					
						
							|  |  |  |     tagName = tagName.toLowerCase(); | 
					
						
							|  |  |  |     propName = propName.toLowerCase(); | 
					
						
							| 
									
										
										
										
											2018-08-16 14:37:21 -07:00
										 |  |  |     let ctx = SECURITY_SCHEMA()[tagName + '|' + propName]; | 
					
						
							| 
									
										
										
										
											2016-08-23 10:52:40 -07:00
										 |  |  |     if (ctx) { | 
					
						
							|  |  |  |       return ctx; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-16 14:37:21 -07:00
										 |  |  |     ctx = SECURITY_SCHEMA()['*|' + propName]; | 
					
						
							| 
									
										
										
										
											2016-08-23 10:52:40 -07:00
										 |  |  |     return ctx ? ctx : SecurityContext.NONE; | 
					
						
							| 
									
										
											  
											
												feat: security implementation in Angular 2.
Summary:
This adds basic security hooks to Angular 2.
* `SecurityContext` is a private API between core, compiler, and
  platform-browser. `SecurityContext` communicates what context a value is used
  in across template parser, compiler, and sanitization at runtime.
* `SanitizationService` is the bare bones interface to sanitize values for a
  particular context.
* `SchemaElementRegistry.securityContext(tagName, attributeOrPropertyName)`
  determines the security context for an attribute or property (it turns out
  attributes and properties match for the purposes of sanitization).
Based on these hooks:
* `DomSchemaElementRegistry` decides what sanitization applies in a particular
  context.
* `DomSanitizationService` implements `SanitizationService` and adds *Safe
  Value*s, i.e. the ability to mark a value as safe and not requiring further
  sanitization.
* `url_sanitizer` and `style_sanitizer` sanitize URLs and Styles, respectively
  (surprise!).
`DomSanitizationService` is the default implementation bound for browser
applications, in the three contexts (browser rendering, web worker rendering,
server side rendering).
BREAKING CHANGES:
*** SECURITY WARNING ***
Angular 2 Release Candidates do not implement proper contextual escaping yet.
Make sure to correctly escape all values that go into the DOM.
*** SECURITY WARNING ***
Reviewers: IgorMinar
Differential Revision: https://reviews.angular.io/D103
											
										 
											2016-04-29 16:04:08 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |   getMappedPropName(propName: string): string { | 
					
						
							|  |  |  |     return _ATTR_TO_PROP[propName] || propName; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-07-28 19:39:10 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |   getDefaultComponentElementName(): string { | 
					
						
							|  |  |  |     return 'ng-component'; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-09-28 02:10:02 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   validateProperty(name: string): {error: boolean, msg?: string} { | 
					
						
							|  |  |  |     if (name.toLowerCase().startsWith('on')) { | 
					
						
							|  |  |  |       const msg = `Binding to event property '${name}' is disallowed for security reasons, ` + | 
					
						
							|  |  |  |           `please use (${name.slice(2)})=...` + | 
					
						
							|  |  |  |           `\nIf '${name}' is a directive input, make sure the directive is imported by the` + | 
					
						
							|  |  |  |           ` current module.`; | 
					
						
							|  |  |  |       return {error: true, msg: msg}; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       return {error: false}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   validateAttribute(name: string): {error: boolean, msg?: string} { | 
					
						
							|  |  |  |     if (name.toLowerCase().startsWith('on')) { | 
					
						
							|  |  |  |       const msg = `Binding to event attribute '${name}' is disallowed for security reasons, ` + | 
					
						
							|  |  |  |           `please use (${name.slice(2)})=...`; | 
					
						
							|  |  |  |       return {error: true, msg: msg}; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       return {error: false}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-10-24 09:58:52 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |   allKnownElementNames(): string[] { | 
					
						
							|  |  |  |     return Object.keys(this._schema); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-11-08 15:45:30 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-04 14:43:26 -08:00
										 |  |  |   allKnownAttributesOfElement(tagName: string): string[] { | 
					
						
							|  |  |  |     const elementProperties = this._schema[tagName.toLowerCase()] || this._schema['unknown']; | 
					
						
							|  |  |  |     // Convert properties to attributes.
 | 
					
						
							|  |  |  |     return Object.keys(elementProperties).map(prop => _PROP_TO_ATTR[prop] ?? prop); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-08 15:45:30 -08:00
										 |  |  |   normalizeAnimationStyleProperty(propName: string): string { | 
					
						
							|  |  |  |     return dashCaseToCamelCase(propName); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   normalizeAnimationStyleValue(camelCaseProp: string, userProvidedProp: string, val: string|number): | 
					
						
							|  |  |  |       {error: string, value: string} { | 
					
						
							| 
									
										
										
										
											2016-11-12 14:08:58 +01:00
										 |  |  |     let unit: string = ''; | 
					
						
							|  |  |  |     const strVal = val.toString().trim(); | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     let errorMsg: string = null!; | 
					
						
							| 
									
										
										
										
											2016-11-08 15:45:30 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (_isPixelDimensionStyle(camelCaseProp) && val !== 0 && val !== '0') { | 
					
						
							|  |  |  |       if (typeof val === 'number') { | 
					
						
							|  |  |  |         unit = 'px'; | 
					
						
							|  |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2016-11-12 14:08:58 +01:00
										 |  |  |         const valAndSuffixMatch = val.match(/^[+-]?[\d\.]+([a-z]*)$/); | 
					
						
							| 
									
										
										
										
											2016-11-08 15:45:30 -08:00
										 |  |  |         if (valAndSuffixMatch && valAndSuffixMatch[1].length == 0) { | 
					
						
							|  |  |  |           errorMsg = `Please provide a CSS unit value for ${userProvidedProp}:${val}`; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return {error: errorMsg, value: strVal + unit}; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function _isPixelDimensionStyle(prop: string): boolean { | 
					
						
							|  |  |  |   switch (prop) { | 
					
						
							|  |  |  |     case 'width': | 
					
						
							|  |  |  |     case 'height': | 
					
						
							|  |  |  |     case 'minWidth': | 
					
						
							|  |  |  |     case 'minHeight': | 
					
						
							|  |  |  |     case 'maxWidth': | 
					
						
							|  |  |  |     case 'maxHeight': | 
					
						
							|  |  |  |     case 'left': | 
					
						
							|  |  |  |     case 'top': | 
					
						
							|  |  |  |     case 'bottom': | 
					
						
							|  |  |  |     case 'right': | 
					
						
							|  |  |  |     case 'fontSize': | 
					
						
							|  |  |  |     case 'outlineWidth': | 
					
						
							|  |  |  |     case 'outlineOffset': | 
					
						
							|  |  |  |     case 'paddingTop': | 
					
						
							|  |  |  |     case 'paddingLeft': | 
					
						
							|  |  |  |     case 'paddingBottom': | 
					
						
							|  |  |  |     case 'paddingRight': | 
					
						
							|  |  |  |     case 'marginTop': | 
					
						
							|  |  |  |     case 'marginLeft': | 
					
						
							|  |  |  |     case 'marginBottom': | 
					
						
							|  |  |  |     case 'marginRight': | 
					
						
							|  |  |  |     case 'borderRadius': | 
					
						
							|  |  |  |     case 'borderWidth': | 
					
						
							|  |  |  |     case 'borderTopWidth': | 
					
						
							|  |  |  |     case 'borderLeftWidth': | 
					
						
							|  |  |  |     case 'borderRightWidth': | 
					
						
							|  |  |  |     case 'borderBottomWidth': | 
					
						
							|  |  |  |     case 'textIndent': | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-07-29 14:01:18 +02:00
										 |  |  | } |