topical media & game development 
  
 
 
 
 
  
    
    
  
lib-flex-animation-code-10-org-as3lib-kitchensync-action-AbstractAction.ax
lib-flex-animation-code-10-org-as3lib-kitchensync-action-AbstractAction.ax
(swf
) 
[ flash
]
flex
  package org.as3lib.kitchensync.action
  {
          import flash.errors.*;
          import flash.events.*;
          
          import org.as3lib.kitchensync.*;
          import org.as3lib.kitchensync.core.*;
          import org.as3lib.kitchensync.utils.*;
          import org.as3lib.utils.AbstractEnforcer;
          
          [Event(name="actionStart", type="org.as3lib.kitchensync.KitchenSyncEvent")]
          [Event(name="actionPause", type="org.as3lib.kitchensync.KitchenSyncEvent")]
          [Event(name="actionUnpause", type="org.as3lib.kitchensync.KitchenSyncEvent")]
          [Event(name="actionComplete", type="org.as3lib.kitchensync.KitchenSyncEvent")]
          
          
 This can be any action that takes place at a specifity time and uses the Synchronizer class to coordinate
 timing. 
   
           // todo - better implementation of ids
          public class @ax-lib-flex-animation-code-10-org-as3lib-kitchensync-action-AbstractAction extends EventDispatcher implements IAction
          {        
                  
                  
 The timeStringParser will determine how strings are parsed into valid 
 time values.
 
	 see:  org.as3lib.kitchensync.ITimeStringParser
	 see:  org.as3lib.kitchensync.TimeStringParser_en
  
                  public static var timeStringParser:ITimeStringParser;
                  
                  
                  
 duration is the length of time that the action will run.
 Will accept an integer or a parsable string.
 
	 see:  org.as3lib.kitchensync.ITimeStringParser
	 see:  #timeUnit
  
                  public function get duration():int { return _duration; }
                  public function set duration(duration:*):void { 
                          if (!isNaN(duration)) {
                                  _duration = duration;
                          } else {
                                  var timeString:String = duration.toString();
                                  _duration = timeStringParser.parseTimeString(timeString);
                          }
                  }
                  protected var _duration:int = 0;
                  
                  
                  
 delay is the time that will pass after the start() method is called
 before the action begins.
 Will accept an integer or a parsable string.
 
	 see:  org.as3lib.kitchensync.ITimeStringParser
  
                  public function get delay():int { return _delay; }
                  public function set delay(delay:*):void { 
                          if (!isNaN(delay)) {
                                  _delay = delay;
                          } else {
                                  var timeString:String = delay.toString();
                                  _delay = timeStringParser.parseTimeString(timeString);
                          }
                  }
                  protected var _delay:int = 0;
                  
                  
Returns true if the action will occur instantaneously when started
  
                  public function get isInstantaneous():Boolean {
                          return ( _delay <= 0 && _duration <= 0 );
                  }
                  
                  
 legacy accessors. synonomous with delay.
  
                  public function get offset():int { return offset; }
                  public function set offset(offset:*):void { this.offset = offset; }
                  
                  
                  
 autoDelete is a flag that indicates whether the action should be killed 
 when it is done executing. The default is set to false so the actions must 
 be deleted manually.
  
                  public function get autoDelete():Boolean { return _autoDelete; }
                  public function set autoDelete(autoDelete:Boolean):void { _autoDelete = autoDelete; }
                  protected var _autoDelete:Boolean;
                  
                  
                  
 Setting sync to true will cause the action to sync up with real time
 even if framerate drops. Otherwise, the action will be synced to frames.
   
                  public function get sync():Boolean { return _sync; }
                  public function set sync(sync:Boolean):void { _sync = sync; }
                  protected var _sync:Boolean;
                  
                  
                  // removed for now.
                  
 The human-readable name of this action. 
                public function get name():String { return _name; }
                public function set name(name:String):void { _name = name; }
                protected var _name:String;
  
                  
                  
                  
 Will return true when the action is running (after start() has been called).
 Will continue running until stop() is called or until the action is completed.
 Pausing does not change the value of isRunning.
  
                  public function get isRunning ():Boolean { return _running; }
                  protected var _running:Boolean = false;
                  
                  
                  
 Will return true if the action is paused (after pause() has been called).
 Calling unpause() or stop() will return the value to false.
   
                  public function get isPaused ():Boolean { return _paused; }
                  protected var _paused:Boolean = false;
                  
                  
                  
 The time at which the action was last started.
  
                  public function get startTime():Timestamp { return _startTime; }
                  protected var _startTime:Timestamp;
                  
                  
 The time at which the action was last paused.
  
                  public function get pauseTime():Timestamp { return _pauseTime; }
                  protected var _pauseTime:Timestamp;
                  
                  
                  
 Constructor.
 @abstract
  
                  public function @ax-lib-flex-animation-code-10-org-as3lib-kitchensync-action-AbstractAction()
                  {
                          super();
                          timeStringParser = KitchenSyncDefaults.timeStringParser;
                          autoDelete = KitchenSyncDefaults.autoDelete;
                          sync = KitchenSyncDefaults.sync;
                          
                          AbstractEnforcer.enforceConstructor(this, @ax-lib-flex-animation-code-10-org-as3lib-kitchensync-action-AbstractAction);
                  }
                  
                  
 Adds the action as a listener to the Synchronizer's update event.
  
                  internal function register():void {
                          Synchronizer.getInstance().registerClient(this);
                          
                          // since the first update won't occur until the next frame, force one here to make it
                          // happen right away.
                          forceUpdate();
                  }
                  
                  
 Removes the action as a listener to the Synchronizer's update event.
  
                  internal function unregister():void {
                          Synchronizer.getInstance().unregisterClient(this);
                  }
                  
                  
 Starts the timer for this action.
 Registers the action with the synchronizer.
 
 @throws flash.errors.IllegalOperationError - if the method is called while the action is already running.
  
                  public function start():IAction {
                          if (_paused) {
                                  unpause();                                
                          } else {
                                  if (!_running) {
                                          _running = true;
                                          _startTime = Synchronizer.getInstance().currentTimestamp;
                                          register();
                                          dispatchEvent(new KitchenSyncEvent(KitchenSyncEvent.START, _startTime));
                                  } else {
                                          throw new IllegalOperationError("The start() method cannot be called when the action is already running. Try stopping the action first or using the clone() method to create a copy of it.");
                                  }
                          }
                          return this;
                  }
                  
                  
 Causes the action to be paused. The action temporarily ignores update events from the Synchronizer 
 and the onUpdate() handler will not be called. When unpause() is called,
 the action will continue at the point where it was paused.
 If the pause() method affects the start time even if the delay time hasn't expired yet. 
  
                  public function pause():void {
                          if (!_running) {
                                  // Do nothing
                                  
                                  //throw new IllegalOperationError("The pause() method cannot be called when the action is not already running or after it has finished running. Use the start() method first.");
                          } else if (_paused) {
                                  //throw new IllegalOperationError("The pause() method cannot be called when the action is already paused. Use the unpause() method first.");
                          } else {
                                  _pauseTime = Synchronizer.getInstance().currentTimestamp;
                                  _paused = true;
                                  unregister();
                                  dispatchEvent(new KitchenSyncEvent(KitchenSyncEvent.PAUSE, _pauseTime));
                          }
                  }
                  
                  
 Resumes the action at the point where it was paused.
  
                  public function unpause():void {
                          if (!_running) {
                                  //throw new IllegalOperationError("The unpause() method cannot be called when the action is not already running or after it has finished running. Use the start() method first.");
                          } else if (!_paused) {
                                  //throw new IllegalOperationError("The unpause() method cannot be called when the action is not already paused. Use the pause() method first.");
                          } else {
                                  register();
                                  _paused = false;
                                  var timeSincePause:Timestamp = TimestampUtil.subtract(Synchronizer.getInstance().currentTimestamp, _pauseTime);
                                  _startTime = TimestampUtil.add(_startTime, timeSincePause); 
                                  dispatchEvent(new KitchenSyncEvent(KitchenSyncEvent.UNPAUSE, _startTime));
                                  //trace("_pauseTime:", _pauseTime);
                                  //trace("_startTime:", _startTime);
                                  //trace("timeSincePause:", timeSincePause);
                                  
                          }
                  }
                  
                  
                  
 Stops the action from running and resets the timer.
  
                  public function stop():void {
                          if (_running) { 
                                  _paused = false;
                                  _running = false;
                                  _startTime = null;
                                  unregister();
                          }
                  }
                  
                  
 Causes the action to start playing when another event completes.
 
	 parameter:  trigger Another action that will trigger the start of this action.
 @throws flash.errors.Error If the trigger action is the same as this action.
  
                  public function addTrigger(trigger:IAction):void {
                           if (trigger == this) { throw new Error("An action cannot be triggered by itself."); }
                          trigger.addEventListener(KitchenSyncEvent.COMPLETE, onTrigger);
                  }
  
                  
 Removes a trigger added with addTrigger().
 
	 parameter:  trigger Another action that triggers the start of this action.
  
                  public function removeTrigger(trigger:IAction):void {
                           trigger.removeEventListener(KitchenSyncEvent.COMPLETE, onTrigger);
                  }
                  
                  
 Causes the action to start playing when a specified event is fired.
 
	 parameter:  dispatcher The object that will dispatch the event.
	 parameter:  eventType The event type to listen for.
  
                  public function addEventTrigger(dispatcher:IEventDispatcher, eventType:String):void {
                          dispatcher.addEventListener(eventType, onTrigger);
                  }
  
                  
 Removes an event trigger added by addEventTrigger().
 
	 parameter:  dispatcher The event dispatcher to remove.
	 parameter:  eventType The event type to listen for.
  
                  public function removeEventTrigger(dispatcher:IEventDispatcher, eventType:String):void {
                          dispatcher.removeEventListener(eventType, onTrigger);
                  }
                  
                  
 Handler that starts playing the action that is called by a trigger event.
	 see:  #addTrigger()
	 see:  #addEventTrigger()
 
  
                   // todo - make sure this doesn't screw up if there are multiple triggers or if the thing isn't meant to repeat.
                  protected function onTrigger(event:Event):void {
                          if (!_running) { start(); }
                  }
                  
                  
                  
 This function will be registered by the register method to respond to update events from the synchronizer.
 Code that performs the action associated with this object should go in this function.
 This function must be implemented by the subclass.
 The internal allows certain other classes such as the AbstractSynchronizedGroup to force an update 
 of its children.
 
 @abstract
	 parameter:  currentTimestamp The current timestamp from the Synchronizer.
  
                  public function update(currentTimestamp:Timestamp):void {
                          AbstractEnforcer.enforceMethod();
                  }
                  
                  
 Foreces the update() method to fire without being triggered by Synchronizer.
 
	 see:  #update()
  
                  protected function forceUpdate():void {
                          update(Synchronizer.getInstance().currentTimestamp);
                  }
                  
                  
 Checks to see whether the start time delay has elapsed and if the _startTime is defined. In other words, 
 checks to see whether the action is ready to execute. Duration is handled seperately.
	 returns:  false if _startTime is null, true if the delay has elapsed.
  
                   public function get startTimeHasElapsed():Boolean {
                           if (!_startTime || !_running || _paused) { return false; }
                          if (_sync) {
                                  if (_startTime.currentTime + _delay <= Synchronizer.getInstance().currentTimestamp.currentTime) { return true; }
                          } else {
                                  if (_startTime.currentFrame + TimestampUtil.millisecondsToFrames(_delay) <= Synchronizer.getInstance().currentTimestamp.currentFrame) { return true; }
                          }
                           return false;
                   }
                  
                  
 Checks to see whether the duration of the action has elapsed and if the _startTime is defined. In other words, 
 checks to see whether the action is finished executing. 
	 returns:  false if _startTime is null, true if the duration has elapsed.
  
                   public function get durationHasElapsed():Boolean {
                           if (!_startTime || !_running || _paused) { return false; }
                           if (_sync) {
                                   if (_startTime.currentTime + _delay + _duration <= Synchronizer.getInstance().currentTimestamp.currentTime) { return true; }                                 
                           } else {
                                   if (_startTime.currentFrame + TimestampUtil.millisecondsToFrames(_delay) + TimestampUtil.millisecondsToFrames(_duration)-1 < Synchronizer.getInstance().currentTimestamp.currentFrame) { return true; }
                           }
                           return false;
                   }
                   
  
                  
 Creates a copy of the object with all the property values of the original and returns it.
 This method should be overrided by child classes to ensure that all properties are copied.
 
 @abstract
	 returns:  AbstractSyncrhonizedAction - A copy of the original object. Type casting may be necessary.
  
                  public function clone():IAction {
                          AbstractEnforcer.enforceMethod();
                          return this;
                  }
                                  
                  
 Call this when the action has completed.
  
                  protected function complete():void {
                          _running = false;
                          unregister();
                          dispatchEvent(new KitchenSyncEvent(KitchenSyncEvent.COMPLETE, Synchronizer.getInstance().currentTimestamp));
                          if (_autoDelete) { kill(); }
                  }                
                  
                  
 Unregisters the function and removes any refrerences to objects that it may be holding onto.
 Subclass this function to remove references to objects used by the action.
  
                   public function kill():void {
                           if (_running) { complete(); }
                   }
          }
  }
  
  
(C) Æliens 
19/08/2009
You may not copy or print any of this material without explicit permission of the author or the publisher. 
In case of other copyright issues, contact the author.