UNPKG

angular2

Version:

Angular 2 - a web framework for modern web apps

173 lines 25.6 kB
import { DateWrapper, StringWrapper, RegExpWrapper, NumberWrapper, isPresent } from 'angular2/src/facade/lang'; import { Math } from 'angular2/src/facade/math'; import { camelCaseToDashCase } from 'angular2/src/platform/dom/util'; import { StringMapWrapper } from 'angular2/src/facade/collection'; import { DOM } from 'angular2/src/platform/dom/dom_adapter'; export class Animation { /** * Stores the start time and starts the animation * @param element * @param data * @param browserDetails */ constructor(element, data, browserDetails) { this.element = element; this.data = data; this.browserDetails = browserDetails; /** functions to be called upon completion */ this.callbacks = []; /** functions for removing event listeners */ this.eventClearFunctions = []; /** flag used to track whether or not the animation has finished */ this.completed = false; this._stringPrefix = ''; this.startTime = DateWrapper.toMillis(DateWrapper.now()); this._stringPrefix = DOM.getAnimationPrefix(); this.setup(); this.wait((timestamp) => this.start()); } /** total amount of time that the animation should take including delay */ get totalTime() { let delay = this.computedDelay != null ? this.computedDelay : 0; let duration = this.computedDuration != null ? this.computedDuration : 0; return delay + duration; } wait(callback) { // Firefox requires 2 frames for some reason this.browserDetails.raf(callback, 2); } /** * Sets up the initial styles before the animation is started */ setup() { if (this.data.fromStyles != null) this.applyStyles(this.data.fromStyles); if (this.data.duration != null) this.applyStyles({ 'transitionDuration': this.data.duration.toString() + 'ms' }); if (this.data.delay != null) this.applyStyles({ 'transitionDelay': this.data.delay.toString() + 'ms' }); } /** * After the initial setup has occurred, this method adds the animation styles */ start() { this.addClasses(this.data.classesToAdd); this.addClasses(this.data.animationClasses); this.removeClasses(this.data.classesToRemove); if (this.data.toStyles != null) this.applyStyles(this.data.toStyles); var computedStyles = DOM.getComputedStyle(this.element); this.computedDelay = Math.max(this.parseDurationString(computedStyles.getPropertyValue(this._stringPrefix + 'transition-delay')), this.parseDurationString(this.element.style.getPropertyValue(this._stringPrefix + 'transition-delay'))); this.computedDuration = Math.max(this.parseDurationString(computedStyles.getPropertyValue(this._stringPrefix + 'transition-duration')), this.parseDurationString(this.element.style.getPropertyValue(this._stringPrefix + 'transition-duration'))); this.addEvents(); } /** * Applies the provided styles to the element * @param styles */ applyStyles(styles) { StringMapWrapper.forEach(styles, (value, key) => { var dashCaseKey = camelCaseToDashCase(key); if (isPresent(DOM.getStyle(this.element, dashCaseKey))) { DOM.setStyle(this.element, dashCaseKey, value.toString()); } else { DOM.setStyle(this.element, this._stringPrefix + dashCaseKey, value.toString()); } }); } /** * Adds the provided classes to the element * @param classes */ addClasses(classes) { for (let i = 0, len = classes.length; i < len; i++) DOM.addClass(this.element, classes[i]); } /** * Removes the provided classes from the element * @param classes */ removeClasses(classes) { for (let i = 0, len = classes.length; i < len; i++) DOM.removeClass(this.element, classes[i]); } /** * Adds events to track when animations have finished */ addEvents() { if (this.totalTime > 0) { this.eventClearFunctions.push(DOM.onAndCancel(this.element, DOM.getTransitionEnd(), (event) => this.handleAnimationEvent(event))); } else { this.handleAnimationCompleted(); } } handleAnimationEvent(event) { let elapsedTime = Math.round(event.elapsedTime * 1000); if (!this.browserDetails.elapsedTimeIncludesDelay) elapsedTime += this.computedDelay; event.stopPropagation(); if (elapsedTime >= this.totalTime) this.handleAnimationCompleted(); } /** * Runs all animation callbacks and removes temporary classes */ handleAnimationCompleted() { this.removeClasses(this.data.animationClasses); this.callbacks.forEach(callback => callback()); this.callbacks = []; this.eventClearFunctions.forEach(fn => fn()); this.eventClearFunctions = []; this.completed = true; } /** * Adds animation callbacks to be called upon completion * @param callback * @returns {Animation} */ onComplete(callback) { if (this.completed) { callback(); } else { this.callbacks.push(callback); } return this; } /** * Converts the duration string to the number of milliseconds * @param duration * @returns {number} */ parseDurationString(duration) { var maxValue = 0; // duration must have at least 2 characters to be valid. (number + type) if (duration == null || duration.length < 2) { return maxValue; } else if (duration.substring(duration.length - 2) == 'ms') { let value = NumberWrapper.parseInt(this.stripLetters(duration), 10); if (value > maxValue) maxValue = value; } else if (duration.substring(duration.length - 1) == 's') { let ms = NumberWrapper.parseFloat(this.stripLetters(duration)) * 1000; let value = Math.floor(ms); if (value > maxValue) maxValue = value; } return maxValue; } /** * Strips the letters from the duration string * @param str * @returns {string} */ stripLetters(str) { return StringWrapper.replaceAll(str, RegExpWrapper.create('[^0-9]+$', ''), ''); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"animation.js","sourceRoot":"","sources":["angular2/src/animate/animation.ts"],"names":["Animation","Animation.constructor","Animation.totalTime","Animation.wait","Animation.setup","Animation.start","Animation.applyStyles","Animation.addClasses","Animation.removeClasses","Animation.addEvents","Animation.handleAnimationEvent","Animation.handleAnimationCompleted","Animation.onComplete","Animation.parseDurationString","Animation.stripLetters"],"mappings":"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","sourcesContent":["import {\n  DateWrapper,\n  StringWrapper,\n  RegExpWrapper,\n  NumberWrapper,\n  isPresent\n} from 'angular2/src/facade/lang';\nimport {Math} from 'angular2/src/facade/math';\nimport {camelCaseToDashCase} from 'angular2/src/platform/dom/util';\nimport {StringMapWrapper} from 'angular2/src/facade/collection';\nimport {DOM} from 'angular2/src/platform/dom/dom_adapter';\n\nimport {BrowserDetails} from './browser_details';\nimport {CssAnimationOptions} from './css_animation_options';\n\nexport class Animation {\n  /** functions to be called upon completion */\n  callbacks: Function[] = [];\n\n  /** the duration (ms) of the animation (whether from CSS or manually set) */\n  computedDuration: number;\n\n  /** the animation delay (ms) (whether from CSS or manually set) */\n  computedDelay: number;\n\n  /** timestamp of when the animation started */\n  startTime: number;\n\n  /** functions for removing event listeners */\n  eventClearFunctions: Function[] = [];\n\n  /** flag used to track whether or not the animation has finished */\n  completed: boolean = false;\n\n  private _stringPrefix: string = '';\n\n  /** total amount of time that the animation should take including delay */\n  get totalTime(): number {\n    let delay = this.computedDelay != null ? this.computedDelay : 0;\n    let duration = this.computedDuration != null ? this.computedDuration : 0;\n    return delay + duration;\n  }\n\n  /**\n   * Stores the start time and starts the animation\n   * @param element\n   * @param data\n   * @param browserDetails\n   */\n  constructor(public element: HTMLElement, public data: CssAnimationOptions,\n              public browserDetails: BrowserDetails) {\n    this.startTime = DateWrapper.toMillis(DateWrapper.now());\n    this._stringPrefix = DOM.getAnimationPrefix();\n    this.setup();\n    this.wait((timestamp: any) => this.start());\n  }\n\n  wait(callback: Function) {\n    // Firefox requires 2 frames for some reason\n    this.browserDetails.raf(callback, 2);\n  }\n\n  /**\n   * Sets up the initial styles before the animation is started\n   */\n  setup(): void {\n    if (this.data.fromStyles != null) this.applyStyles(this.data.fromStyles);\n    if (this.data.duration != null)\n      this.applyStyles({'transitionDuration': this.data.duration.toString() + 'ms'});\n    if (this.data.delay != null)\n      this.applyStyles({'transitionDelay': this.data.delay.toString() + 'ms'});\n  }\n\n  /**\n   * After the initial setup has occurred, this method adds the animation styles\n   */\n  start(): void {\n    this.addClasses(this.data.classesToAdd);\n    this.addClasses(this.data.animationClasses);\n    this.removeClasses(this.data.classesToRemove);\n    if (this.data.toStyles != null) this.applyStyles(this.data.toStyles);\n    var computedStyles = DOM.getComputedStyle(this.element);\n    this.computedDelay =\n        Math.max(this.parseDurationString(\n                     computedStyles.getPropertyValue(this._stringPrefix + 'transition-delay')),\n                 this.parseDurationString(\n                     this.element.style.getPropertyValue(this._stringPrefix + 'transition-delay')));\n    this.computedDuration = Math.max(this.parseDurationString(computedStyles.getPropertyValue(\n                                         this._stringPrefix + 'transition-duration')),\n                                     this.parseDurationString(this.element.style.getPropertyValue(\n                                         this._stringPrefix + 'transition-duration')));\n    this.addEvents();\n  }\n\n  /**\n   * Applies the provided styles to the element\n   * @param styles\n   */\n  applyStyles(styles: {[key: string]: any}): void {\n    StringMapWrapper.forEach(styles, (value: any, key: string) => {\n      var dashCaseKey = camelCaseToDashCase(key);\n      if (isPresent(DOM.getStyle(this.element, dashCaseKey))) {\n        DOM.setStyle(this.element, dashCaseKey, value.toString());\n      } else {\n        DOM.setStyle(this.element, this._stringPrefix + dashCaseKey, value.toString());\n      }\n    });\n  }\n\n  /**\n   * Adds the provided classes to the element\n   * @param classes\n   */\n  addClasses(classes: string[]): void {\n    for (let i = 0, len = classes.length; i < len; i++) DOM.addClass(this.element, classes[i]);\n  }\n\n  /**\n   * Removes the provided classes from the element\n   * @param classes\n   */\n  removeClasses(classes: string[]): void {\n    for (let i = 0, len = classes.length; i < len; i++) DOM.removeClass(this.element, classes[i]);\n  }\n\n  /**\n   * Adds events to track when animations have finished\n   */\n  addEvents(): void {\n    if (this.totalTime > 0) {\n      this.eventClearFunctions.push(DOM.onAndCancel(\n          this.element, DOM.getTransitionEnd(), (event: any) => this.handleAnimationEvent(event)));\n    } else {\n      this.handleAnimationCompleted();\n    }\n  }\n\n  handleAnimationEvent(event: any): void {\n    let elapsedTime = Math.round(event.elapsedTime * 1000);\n    if (!this.browserDetails.elapsedTimeIncludesDelay) elapsedTime += this.computedDelay;\n    event.stopPropagation();\n    if (elapsedTime >= this.totalTime) this.handleAnimationCompleted();\n  }\n\n  /**\n   * Runs all animation callbacks and removes temporary classes\n   */\n  handleAnimationCompleted(): void {\n    this.removeClasses(this.data.animationClasses);\n    this.callbacks.forEach(callback => callback());\n    this.callbacks = [];\n    this.eventClearFunctions.forEach(fn => fn());\n    this.eventClearFunctions = [];\n    this.completed = true;\n  }\n\n  /**\n   * Adds animation callbacks to be called upon completion\n   * @param callback\n   * @returns {Animation}\n   */\n  onComplete(callback: Function): Animation {\n    if (this.completed) {\n      callback();\n    } else {\n      this.callbacks.push(callback);\n    }\n    return this;\n  }\n\n  /**\n   * Converts the duration string to the number of milliseconds\n   * @param duration\n   * @returns {number}\n   */\n  parseDurationString(duration: string): number {\n    var maxValue = 0;\n    // duration must have at least 2 characters to be valid. (number + type)\n    if (duration == null || duration.length < 2) {\n      return maxValue;\n    } else if (duration.substring(duration.length - 2) == 'ms') {\n      let value = NumberWrapper.parseInt(this.stripLetters(duration), 10);\n      if (value > maxValue) maxValue = value;\n    } else if (duration.substring(duration.length - 1) == 's') {\n      let ms = NumberWrapper.parseFloat(this.stripLetters(duration)) * 1000;\n      let value = Math.floor(ms);\n      if (value > maxValue) maxValue = value;\n    }\n    return maxValue;\n  }\n\n  /**\n   * Strips the letters from the duration string\n   * @param str\n   * @returns {string}\n   */\n  stripLetters(str: string): string {\n    return StringWrapper.replaceAll(str, RegExpWrapper.create('[^0-9]+$', ''), '');\n  }\n}\n"]}