2017-04-26 10:44:28 -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
* /
import { AUTO_STYLE , AnimateTimings , AnimationAnimateChildMetadata , AnimationAnimateMetadata , AnimationAnimateRefMetadata , AnimationGroupMetadata , AnimationKeyframesSequenceMetadata , AnimationMetadata , AnimationMetadataType , AnimationOptions , AnimationQueryMetadata , AnimationQueryOptions , AnimationReferenceMetadata , AnimationSequenceMetadata , AnimationStaggerMetadata , AnimationStateMetadata , AnimationStyleMetadata , AnimationTransitionMetadata , AnimationTriggerMetadata , style , ɵ StyleData } from '@angular/animations' ;
2017-08-15 16:11:11 -07:00
import { AnimationDriver } from '../render/animation_driver' ;
2017-04-26 10:44:28 -07:00
import { getOrSetAsInMap } from '../render/shared' ;
2017-10-03 13:41:52 -07:00
import { ENTER_SELECTOR , LEAVE_SELECTOR , NG_ANIMATING_SELECTOR , NG_TRIGGER_SELECTOR , SUBSTITUTION_EXPR_START , copyObj , extractStyleParams , iteratorToArray , normalizeAnimationEntry , resolveTiming , validateStyleParams , visitDslNode } from '../util' ;
2017-04-26 10:44:28 -07:00
import { AnimateAst , AnimateChildAst , AnimateRefAst , Ast , DynamicTimingAst , GroupAst , KeyframesAst , QueryAst , ReferenceAst , SequenceAst , StaggerAst , StateAst , StyleAst , TimingAst , TransitionAst , TriggerAst } from './animation_ast' ;
2017-10-03 13:41:52 -07:00
import { AnimationDslVisitor } from './animation_dsl_visitor' ;
2017-04-26 10:44:28 -07:00
import { parseTransitionExpr } from './animation_transition_expr' ;
const SELF_TOKEN = ':self' ;
const SELF_TOKEN_REGEX = new RegExp ( ` \ s* ${ SELF_TOKEN } \ s*,? ` , 'g' ) ;
/ *
* [ Validation ]
* The visitor code below will traverse the animation AST generated by the animation verb functions
* ( the output is a tree of objects ) and attempt to perform a series of validations on the data . The
* following corner - cases will be validated :
*
* 1 . Overlap of animations
* Given that a CSS property cannot be animated in more than one place at the same time , it ' s
* important that this behaviour is detected and validated . The way in which this occurs is that
* each time a style property is examined , a string - map containing the property will be updated with
* the start and end times for when the property is used within an animation step .
*
* If there are two or more parallel animations that are currently running ( these are invoked by the
* group ( ) ) on the same element then the validator will throw an error . Since the start / end timing
* values are collected for each property then if the current animation step is animating the same
* property and its timing values fall anywhere into the window of time that the property is
* currently being animated within then this is what causes an error .
*
* 2 . Timing values
* The validator will validate to see if a timing value of ` duration delay easing ` or
* ` durationNumber ` is valid or not .
*
* ( note that upon validation the code below will replace the timing data with an object containing
* { duration , delay , easing } .
*
* 3 . Offset Validation
* Each of the style ( ) calls are allowed to have an offset value when placed inside of keyframes ( ) .
* Offsets within keyframes ( ) are considered valid when :
*
* - No offsets are used at all
* - Each style ( ) entry contains an offset value
* - Each offset is between 0 and 1
* - Each offset is greater to or equal than the previous one
*
* Otherwise an error will be thrown .
* /
export function buildAnimationAst (
2017-08-15 16:11:11 -07:00
driver : AnimationDriver , metadata : AnimationMetadata | AnimationMetadata [ ] ,
2017-10-03 13:41:52 -07:00
errors : any [ ] ) : Ast < AnimationMetadataType > {
2017-08-15 16:11:11 -07:00
return new AnimationAstBuilderVisitor ( driver ) . build ( metadata , errors ) ;
2017-04-26 10:44:28 -07:00
}
const ROOT_SELECTOR = '' ;
export class AnimationAstBuilderVisitor implements AnimationDslVisitor {
2017-08-15 16:11:11 -07:00
constructor ( private _driver : AnimationDriver ) { }
2017-10-03 13:41:52 -07:00
build ( metadata : AnimationMetadata | AnimationMetadata [ ] , errors : any [ ] ) :
Ast < AnimationMetadataType > {
2017-04-26 10:44:28 -07:00
const context = new AnimationAstBuilderContext ( errors ) ;
this . _resetContextStyleTimingState ( context ) ;
2017-10-03 13:41:52 -07:00
return < Ast < AnimationMetadataType > > visitDslNode (
this , normalizeAnimationEntry ( metadata ) , context ) ;
2017-04-26 10:44:28 -07:00
}
private _resetContextStyleTimingState ( context : AnimationAstBuilderContext ) {
context . currentQuerySelector = ROOT_SELECTOR ;
2017-06-22 09:42:31 -07:00
context . collectedStyles = { } ;
2017-04-26 10:44:28 -07:00
context . collectedStyles [ ROOT_SELECTOR ] = { } ;
context . currentTime = 0 ;
}
visitTrigger ( metadata : AnimationTriggerMetadata , context : AnimationAstBuilderContext ) :
TriggerAst {
let queryCount = context . queryCount = 0 ;
let depCount = context . depCount = 0 ;
const states : StateAst [ ] = [ ] ;
const transitions : TransitionAst [ ] = [ ] ;
2017-11-10 09:46:18 +02:00
if ( metadata . name . charAt ( 0 ) == '@' ) {
context . errors . push (
'animation triggers cannot be prefixed with an `@` sign (e.g. trigger(\'@foo\', [...]))' ) ;
}
2017-04-26 10:44:28 -07:00
metadata . definitions . forEach ( def = > {
this . _resetContextStyleTimingState ( context ) ;
if ( def . type == AnimationMetadataType . State ) {
const stateDef = def as AnimationStateMetadata ;
const name = stateDef . name ;
2018-03-21 19:58:37 -07:00
name . toString ( ) . split ( /\s*,\s*/ ) . forEach ( n = > {
2017-04-26 10:44:28 -07:00
stateDef . name = n ;
states . push ( this . visitState ( stateDef , context ) ) ;
} ) ;
stateDef . name = name ;
} else if ( def . type == AnimationMetadataType . Transition ) {
const transition = this . visitTransition ( def as AnimationTransitionMetadata , context ) ;
queryCount += transition . queryCount ;
depCount += transition . depCount ;
transitions . push ( transition ) ;
} else {
context . errors . push (
'only state() and transition() definitions can sit inside of a trigger()' ) ;
}
} ) ;
2017-10-03 13:41:52 -07:00
return {
type : AnimationMetadataType . Trigger ,
name : metadata.name , states , transitions , queryCount , depCount ,
options : null
} ;
2017-04-26 10:44:28 -07:00
}
visitState ( metadata : AnimationStateMetadata , context : AnimationAstBuilderContext ) : StateAst {
2017-08-07 11:40:04 -07:00
const styleAst = this . visitStyle ( metadata . styles , context ) ;
const astParams = ( metadata . options && metadata . options . params ) || null ;
if ( styleAst . containsDynamicStyles ) {
const missingSubs = new Set < string > ( ) ;
const params = astParams || { } ;
styleAst . styles . forEach ( value = > {
if ( isObject ( value ) ) {
const stylesObj = value as any ;
Object . keys ( stylesObj ) . forEach ( prop = > {
extractStyleParams ( stylesObj [ prop ] ) . forEach ( sub = > {
if ( ! params . hasOwnProperty ( sub ) ) {
missingSubs . add ( sub ) ;
}
} ) ;
} ) ;
}
} ) ;
if ( missingSubs . size ) {
const missingSubsArr = iteratorToArray ( missingSubs . values ( ) ) ;
context . errors . push (
` state(" ${ metadata . name } ", ...) must define default values for all the following style substitutions: ${ missingSubsArr . join ( ', ' ) } ` ) ;
}
}
2017-10-03 13:41:52 -07:00
return {
type : AnimationMetadataType . State ,
name : metadata.name ,
style : styleAst ,
options : astParams ? { params : astParams } : null
} ;
2017-04-26 10:44:28 -07:00
}
visitTransition ( metadata : AnimationTransitionMetadata , context : AnimationAstBuilderContext ) :
TransitionAst {
context . queryCount = 0 ;
context . depCount = 0 ;
2017-10-03 13:41:52 -07:00
const animation = visitDslNode ( this , normalizeAnimationEntry ( metadata . animation ) , context ) ;
2017-04-26 10:44:28 -07:00
const matchers = parseTransitionExpr ( metadata . expr , context . errors ) ;
2017-10-03 13:41:52 -07:00
return {
type : AnimationMetadataType . Transition ,
matchers ,
animation ,
queryCount : context.queryCount ,
depCount : context.depCount ,
options : normalizeAnimationOptions ( metadata . options )
} ;
2017-04-26 10:44:28 -07:00
}
visitSequence ( metadata : AnimationSequenceMetadata , context : AnimationAstBuilderContext ) :
SequenceAst {
2017-10-03 13:41:52 -07:00
return {
type : AnimationMetadataType . Sequence ,
steps : metadata.steps.map ( s = > visitDslNode ( this , s , context ) ) ,
options : normalizeAnimationOptions ( metadata . options )
} ;
2017-04-26 10:44:28 -07:00
}
visitGroup ( metadata : AnimationGroupMetadata , context : AnimationAstBuilderContext ) : GroupAst {
const currentTime = context . currentTime ;
let furthestTime = 0 ;
const steps = metadata . steps . map ( step = > {
context . currentTime = currentTime ;
2017-10-03 13:41:52 -07:00
const innerAst = visitDslNode ( this , step , context ) ;
2017-04-26 10:44:28 -07:00
furthestTime = Math . max ( furthestTime , context . currentTime ) ;
return innerAst ;
} ) ;
context . currentTime = furthestTime ;
2017-10-03 13:41:52 -07:00
return {
type : AnimationMetadataType . Group ,
steps ,
options : normalizeAnimationOptions ( metadata . options )
} ;
2017-04-26 10:44:28 -07:00
}
visitAnimate ( metadata : AnimationAnimateMetadata , context : AnimationAstBuilderContext ) :
AnimateAst {
const timingAst = constructTimingAst ( metadata . timings , context . errors ) ;
context . currentAnimateTimings = timingAst ;
2017-10-03 13:41:52 -07:00
let styleAst : StyleAst | KeyframesAst ;
2017-04-26 10:44:28 -07:00
let styleMetadata : AnimationMetadata = metadata . styles ? metadata.styles : style ( { } ) ;
if ( styleMetadata . type == AnimationMetadataType . Keyframes ) {
2017-10-03 13:41:52 -07:00
styleAst = this . visitKeyframes ( styleMetadata as AnimationKeyframesSequenceMetadata , context ) ;
2017-04-26 10:44:28 -07:00
} else {
let styleMetadata = metadata . styles as AnimationStyleMetadata ;
let isEmpty = false ;
if ( ! styleMetadata ) {
isEmpty = true ;
const newStyleData : { [ prop : string ] : string | number } = { } ;
if ( timingAst . easing ) {
newStyleData [ 'easing' ] = timingAst . easing ;
}
styleMetadata = style ( newStyleData ) ;
}
context . currentTime += timingAst . duration + timingAst . delay ;
2017-10-03 13:41:52 -07:00
const _styleAst = this . visitStyle ( styleMetadata , context ) ;
_styleAst . isEmptyStep = isEmpty ;
styleAst = _styleAst ;
2017-04-26 10:44:28 -07:00
}
context . currentAnimateTimings = null ;
2017-10-03 13:41:52 -07:00
return {
type : AnimationMetadataType . Animate ,
timings : timingAst ,
style : styleAst ,
options : null
} ;
2017-04-26 10:44:28 -07:00
}
visitStyle ( metadata : AnimationStyleMetadata , context : AnimationAstBuilderContext ) : StyleAst {
const ast = this . _makeStyleAst ( metadata , context ) ;
this . _validateStyleAst ( ast , context ) ;
return ast ;
}
private _makeStyleAst ( metadata : AnimationStyleMetadata , context : AnimationAstBuilderContext ) :
StyleAst {
const styles : ( ɵ StyleData | string ) [ ] = [ ] ;
if ( Array . isArray ( metadata . styles ) ) {
( metadata . styles as ( ɵ StyleData | string ) [ ] ) . forEach ( styleTuple = > {
if ( typeof styleTuple == 'string' ) {
if ( styleTuple == AUTO_STYLE ) {
styles . push ( styleTuple as string ) ;
} else {
context . errors . push ( ` The provided style string value ${ styleTuple } is not allowed. ` ) ;
}
} else {
styles . push ( styleTuple as ɵ StyleData ) ;
}
2017-07-27 16:13:16 -07:00
} ) ;
2017-04-26 10:44:28 -07:00
} else {
styles . push ( metadata . styles ) ;
}
2017-08-07 11:40:04 -07:00
let containsDynamicStyles = false ;
2017-04-26 10:44:28 -07:00
let collectedEasing : string | null = null ;
styles . forEach ( styleData = > {
if ( isObject ( styleData ) ) {
const styleMap = styleData as ɵ StyleData ;
const easing = styleMap [ 'easing' ] ;
if ( easing ) {
collectedEasing = easing as string ;
delete styleMap [ 'easing' ] ;
}
2017-08-07 11:40:04 -07:00
if ( ! containsDynamicStyles ) {
for ( let prop in styleMap ) {
const value = styleMap [ prop ] ;
if ( value . toString ( ) . indexOf ( SUBSTITUTION_EXPR_START ) >= 0 ) {
containsDynamicStyles = true ;
break ;
}
}
}
2017-04-26 10:44:28 -07:00
}
} ) ;
2017-08-07 11:40:04 -07:00
2017-10-03 13:41:52 -07:00
return {
type : AnimationMetadataType . Style ,
styles ,
easing : collectedEasing ,
offset : metadata.offset , containsDynamicStyles ,
options : null
} ;
2017-04-26 10:44:28 -07:00
}
private _validateStyleAst ( ast : StyleAst , context : AnimationAstBuilderContext ) : void {
const timings = context . currentAnimateTimings ;
let endTime = context . currentTime ;
let startTime = context . currentTime ;
if ( timings && startTime > 0 ) {
startTime -= timings . duration + timings . delay ;
}
ast . styles . forEach ( tuple = > {
if ( typeof tuple == 'string' ) return ;
Object . keys ( tuple ) . forEach ( prop = > {
2017-08-15 16:11:11 -07:00
if ( ! this . _driver . validateStyleProperty ( prop ) ) {
context . errors . push (
` The provided animation property " ${ prop } " is not a supported CSS property for animations ` ) ;
return ;
}
2017-04-26 10:44:28 -07:00
const collectedStyles = context . collectedStyles [ context . currentQuerySelector ! ] ;
const collectedEntry = collectedStyles [ prop ] ;
let updateCollectedStyle = true ;
if ( collectedEntry ) {
if ( startTime != endTime && startTime >= collectedEntry . startTime &&
endTime <= collectedEntry . endTime ) {
context . errors . push (
` The CSS property " ${ prop } " that exists between the times of " ${ collectedEntry . startTime } ms" and " ${ collectedEntry . endTime } ms" is also being animated in a parallel animation between the times of " ${ startTime } ms" and " ${ endTime } ms" ` ) ;
updateCollectedStyle = false ;
}
// we always choose the smaller start time value since we
// want to have a record of the entire animation window where
// the style property is being animated in between
startTime = collectedEntry . startTime ;
}
if ( updateCollectedStyle ) {
collectedStyles [ prop ] = { startTime , endTime } ;
}
if ( context . options ) {
validateStyleParams ( tuple [ prop ] , context . options , context . errors ) ;
}
} ) ;
} ) ;
}
visitKeyframes ( metadata : AnimationKeyframesSequenceMetadata , context : AnimationAstBuilderContext ) :
KeyframesAst {
2017-10-03 13:41:52 -07:00
const ast : KeyframesAst = { type : AnimationMetadataType . Keyframes , styles : [ ] , options : null } ;
2017-04-26 10:44:28 -07:00
if ( ! context . currentAnimateTimings ) {
context . errors . push ( ` keyframes() must be placed inside of a call to animate() ` ) ;
2017-10-03 13:41:52 -07:00
return ast ;
2017-04-26 10:44:28 -07:00
}
const MAX_KEYFRAME_OFFSET = 1 ;
let totalKeyframesWithOffsets = 0 ;
const offsets : number [ ] = [ ] ;
let offsetsOutOfOrder = false ;
let keyframesOutOfRange = false ;
let previousOffset : number = 0 ;
const keyframes : StyleAst [ ] = metadata . steps . map ( styles = > {
const style = this . _makeStyleAst ( styles , context ) ;
let offsetVal : number | null =
style . offset != null ? style.offset : consumeOffset ( style . styles ) ;
let offset : number = 0 ;
if ( offsetVal != null ) {
totalKeyframesWithOffsets ++ ;
offset = style . offset = offsetVal ;
}
keyframesOutOfRange = keyframesOutOfRange || offset < 0 || offset > 1 ;
offsetsOutOfOrder = offsetsOutOfOrder || offset < previousOffset ;
previousOffset = offset ;
offsets . push ( offset ) ;
return style ;
} ) ;
if ( keyframesOutOfRange ) {
context . errors . push ( ` Please ensure that all keyframe offsets are between 0 and 1 ` ) ;
}
if ( offsetsOutOfOrder ) {
context . errors . push ( ` Please ensure that all keyframe offsets are in order ` ) ;
}
const length = metadata . steps . length ;
let generatedOffset = 0 ;
if ( totalKeyframesWithOffsets > 0 && totalKeyframesWithOffsets < length ) {
context . errors . push ( ` Not all style() steps within the declared keyframes() contain offsets ` ) ;
} else if ( totalKeyframesWithOffsets == 0 ) {
generatedOffset = MAX_KEYFRAME_OFFSET / ( length - 1 ) ;
}
const limit = length - 1 ;
const currentTime = context . currentTime ;
const currentAnimateTimings = context . currentAnimateTimings ! ;
const animateDuration = currentAnimateTimings . duration ;
keyframes . forEach ( ( kf , i ) = > {
const offset = generatedOffset > 0 ? ( i == limit ? 1 : ( generatedOffset * i ) ) : offsets [ i ] ;
const durationUpToThisFrame = offset * animateDuration ;
context . currentTime = currentTime + currentAnimateTimings . delay + durationUpToThisFrame ;
currentAnimateTimings . duration = durationUpToThisFrame ;
this . _validateStyleAst ( kf , context ) ;
kf . offset = offset ;
2017-10-03 13:41:52 -07:00
ast . styles . push ( kf ) ;
2017-04-26 10:44:28 -07:00
} ) ;
2017-10-03 13:41:52 -07:00
return ast ;
2017-04-26 10:44:28 -07:00
}
visitReference ( metadata : AnimationReferenceMetadata , context : AnimationAstBuilderContext ) :
ReferenceAst {
2017-10-03 13:41:52 -07:00
return {
type : AnimationMetadataType . Reference ,
animation : visitDslNode ( this , normalizeAnimationEntry ( metadata . animation ) , context ) ,
options : normalizeAnimationOptions ( metadata . options )
} ;
2017-04-26 10:44:28 -07:00
}
visitAnimateChild ( metadata : AnimationAnimateChildMetadata , context : AnimationAstBuilderContext ) :
AnimateChildAst {
context . depCount ++ ;
2017-10-03 13:41:52 -07:00
return {
type : AnimationMetadataType . AnimateChild ,
options : normalizeAnimationOptions ( metadata . options )
} ;
2017-04-26 10:44:28 -07:00
}
visitAnimateRef ( metadata : AnimationAnimateRefMetadata , context : AnimationAstBuilderContext ) :
AnimateRefAst {
2017-10-03 13:41:52 -07:00
return {
type : AnimationMetadataType . AnimateRef ,
animation : this.visitReference ( metadata . animation , context ) ,
options : normalizeAnimationOptions ( metadata . options )
} ;
2017-04-26 10:44:28 -07:00
}
visitQuery ( metadata : AnimationQueryMetadata , context : AnimationAstBuilderContext ) : QueryAst {
const parentSelector = context . currentQuerySelector ! ;
const options = ( metadata . options || { } ) as AnimationQueryOptions ;
context . queryCount ++ ;
context . currentQuery = metadata ;
const [ selector , includeSelf ] = normalizeSelector ( metadata . selector ) ;
context . currentQuerySelector =
parentSelector . length ? ( parentSelector + ' ' + selector ) : selector ;
getOrSetAsInMap ( context . collectedStyles , context . currentQuerySelector , { } ) ;
2017-10-03 13:41:52 -07:00
const animation = visitDslNode ( this , normalizeAnimationEntry ( metadata . animation ) , context ) ;
2017-04-26 10:44:28 -07:00
context . currentQuery = null ;
context . currentQuerySelector = parentSelector ;
2017-10-03 13:41:52 -07:00
return {
type : AnimationMetadataType . Query ,
selector ,
limit : options.limit || 0 ,
optional : ! ! options . optional , includeSelf , animation ,
originalSelector : metadata.selector ,
options : normalizeAnimationOptions ( metadata . options )
} ;
2017-04-26 10:44:28 -07:00
}
visitStagger ( metadata : AnimationStaggerMetadata , context : AnimationAstBuilderContext ) :
StaggerAst {
if ( ! context . currentQuery ) {
context . errors . push ( ` stagger() can only be used inside of query() ` ) ;
}
const timings = metadata . timings === 'full' ?
{ duration : 0 , delay : 0 , easing : 'full' } :
resolveTiming ( metadata . timings , context . errors , true ) ;
2017-10-03 13:41:52 -07:00
return {
type : AnimationMetadataType . Stagger ,
animation : visitDslNode ( this , normalizeAnimationEntry ( metadata . animation ) , context ) , timings ,
options : null
} ;
2017-04-26 10:44:28 -07:00
}
}
function normalizeSelector ( selector : string ) : [ string , boolean ] {
const hasAmpersand = selector . split ( /\s*,\s*/ ) . find ( token = > token == SELF_TOKEN ) ? true : false ;
if ( hasAmpersand ) {
selector = selector . replace ( SELF_TOKEN_REGEX , '' ) ;
}
2017-10-05 14:03:11 -07:00
// the :enter and :leave selectors are filled in at runtime during timeline building
selector = selector . replace ( /@\*/g , NG_TRIGGER_SELECTOR )
2017-04-26 10:44:28 -07:00
. replace ( /@\w+/g , match = > NG_TRIGGER_SELECTOR + '-' + match . substr ( 1 ) )
. replace ( /:animating/g , NG_ANIMATING_SELECTOR ) ;
return [ selector , hasAmpersand ] ;
}
function normalizeParams ( obj : { [ key : string ] : any } | any ) : { [ key : string ] : any } | null {
return obj ? copyObj ( obj ) : null ;
}
export type StyleTimeTuple = {
startTime : number ; endTime : number ;
} ;
export class AnimationAstBuilderContext {
public queryCount : number = 0 ;
public depCount : number = 0 ;
public currentTransition : AnimationTransitionMetadata | null = null ;
public currentQuery : AnimationQueryMetadata | null = null ;
public currentQuerySelector : string | null = null ;
public currentAnimateTimings : TimingAst | null = null ;
public currentTime : number = 0 ;
public collectedStyles : { [ selectorName : string ] : { [ propName : string ] : StyleTimeTuple } } = { } ;
public options : AnimationOptions | null = null ;
constructor ( public errors : any [ ] ) { }
}
function consumeOffset ( styles : ɵStyleData | string | ( ɵ StyleData | string ) [ ] ) : number | null {
if ( typeof styles == 'string' ) return null ;
let offset : number | null = null ;
if ( Array . isArray ( styles ) ) {
styles . forEach ( styleTuple = > {
if ( isObject ( styleTuple ) && styleTuple . hasOwnProperty ( 'offset' ) ) {
const obj = styleTuple as ɵ StyleData ;
offset = parseFloat ( obj [ 'offset' ] as string ) ;
delete obj [ 'offset' ] ;
}
} ) ;
} else if ( isObject ( styles ) && styles . hasOwnProperty ( 'offset' ) ) {
const obj = styles as ɵ StyleData ;
offset = parseFloat ( obj [ 'offset' ] as string ) ;
delete obj [ 'offset' ] ;
}
return offset ;
}
function isObject ( value : any ) : boolean {
return ! Array . isArray ( value ) && typeof value == 'object' ;
}
function constructTimingAst ( value : string | number | AnimateTimings , errors : any [ ] ) {
let timings : AnimateTimings | null = null ;
if ( value . hasOwnProperty ( 'duration' ) ) {
timings = value as AnimateTimings ;
} else if ( typeof value == 'number' ) {
const duration = resolveTiming ( value as number , errors ) . duration ;
2017-10-03 13:41:52 -07:00
return makeTimingAst ( duration as number , 0 , '' ) ;
2017-04-26 10:44:28 -07:00
}
const strValue = value as string ;
const isDynamic = strValue . split ( /\s+/ ) . some ( v = > v . charAt ( 0 ) == '{' && v . charAt ( 1 ) == '{' ) ;
if ( isDynamic ) {
2017-10-03 13:41:52 -07:00
const ast = makeTimingAst ( 0 , 0 , '' ) as any ;
ast . dynamic = true ;
ast . strValue = strValue ;
return ast as DynamicTimingAst ;
2017-04-26 10:44:28 -07:00
}
timings = timings || resolveTiming ( strValue , errors ) ;
2017-10-03 13:41:52 -07:00
return makeTimingAst ( timings . duration , timings . delay , timings . easing ) ;
2017-04-26 10:44:28 -07:00
}
function normalizeAnimationOptions ( options : AnimationOptions | null ) : AnimationOptions {
if ( options ) {
options = copyObj ( options ) ;
if ( options [ 'params' ] ) {
options [ 'params' ] = normalizeParams ( options [ 'params' ] ) ! ;
}
} else {
options = { } ;
}
return options ;
}
2017-10-03 13:41:52 -07:00
function makeTimingAst ( duration : number , delay : number , easing : string | null ) : TimingAst {
return { duration , delay , easing } ;
}