FlowManager
Kind of class: | public class |
---|---|
Package: | |
Inherits from: |
|
Classpath: | org.asaplibrary.management.flow.FlowManager |
File last modified: | Sunday, 15 May 2011, 21:42:34 |
Enables to navigate a multi-part Flash movie using the principle of deep links.
Introduction
Flow states are represented by site 'sections', using the IFlowSection type. FlowSection inherits from LocalController, but can be assigned to any MovieClip in the site that needs to be navigated to. FlowManager registers FlowSections and uses their names to build a tree-like structure. The syntax to go to a new section is simply:FlowManager.defaultFlowManager.goto("Intro");
FlowManager finds the FlowSection with that name, and finds what sections need to be shown or hidden or even loaded.
Site structure
The structure of the site is defined by hierarchical naming of FlowSections, using dots for each level. An example of a naming structure can be found in the example demo:public static const SECTION_INTRO:String = "Intro"; public static const SECTION1:String = "Sections.Section1"; public static const SECTION1_1:String = "Sections.Section1.Section1_1"; public static const SECTION2:String = "Sections.Section2";When going from 'Sections.Section1' to 'Sections.Section2', FlowManager will detect that this is a sibling relationship. The possible types of relationships are defined in FlowOptions. You may also use a dot to indicate a level. In the next list the Section items have a higher depth than Intro:
public static const SECTION_INTRO:String = "Intro"; public static const SECTION1:String = ".Section1"; public static const SECTION1_1:String = ".Section1.Section1_1"; public static const SECTION2:String = ".Section2";This is useful when you want to let FlowManager automatically load movies based on the section names.
Starting and stopping, or showing and hiding
The default behavior forstart
is setting the visibility to true; for stop
to set it to false.
Start and stop are called from the 2 base functions FlowSection.startAction and FlowSection.stopAction. You can override the default behavior of these methods in a subclass.
When traversing from one section to the other, it depends if startSection
or stopSection
is called. When going to a child section, stop
will not called on the current section. When going to a sibling section, the current stop will be called, and right after start
.
Each startAction and stopAction function may return a IAction or a subclass thereof, including an ActionQueue. Any animation is processed sequentially - an ActionQueue is first finished before the next Action is called.
In the example demo one of the sections overrides the default startAction - it lets the clip scale from small to large in a elastic manner:
public override function get startAction () : IAction { var queue:ActionQueue = new ActionQueue("Section1_1 show"); queue.addAction(new AQSet().setVisible(this, true)); queue.addAction(new AQSet().setScale(this, .5, .5)); const CURRENT:Number = Number.NaN; var effect:Function = Elastic.easeOut; queue.addAction(new AQScale().scale(this, .8, CURRENT, CURRENT, 1, 1, effect)); return queue; }
Rules
Sometimes it is desired to manage these start and stop actions from a higher level. For example after the intro animation we want to go to section 1. To do this Rules can be defined. Rules are created with the FlowRule class:var rule:FlowRule = new FlowRule ( "Intro", OPTIONS.START_END, OPTIONS.ANY, proceedToSection1 ); FlowManager.defaultFlowManager.addRule(rule);This rule says that for a section with name
Intro
, when encountering mode START_END
(end of the start action), and ANY
relationship type, function proceedToSection1
needs to be called. And that function simply has:
protected function proceedToSection1 (inSection:IFlowSection) : void { FlowManager.defaultFlowManager.goto("Sections.Section1", this, false); }You will notice that the current section is always passed to the callback function.
Apply many
It is also possible to set a Rule for multiple sections at once. For example:var rule:FlowRule = new FlowRule ( null, OPTIONS.STOP, OPTIONS.DISTANT|OPTIONS.SIBLING, doNotHide ); FM.addRuleForSections ( rule, [AppSettings.SECTION1, AppSettings.SECTION2, AppSettings.SECTION3, AppSettings.SECTION4] );You can define combinations of options using bitwise operators. In the example,
OPTIONS.DISTANT|OPTIONS.SIBLING
means either a distant relative OR a sibling.
Function doNotHide simply voids the default behavior:
protected function doNotHide (inSection:IFlowSection) : void { // do nothing }
Enhancing default behavior
As 'man in the middle' you can control what happens before and after a startAction. In the example demo we move the stage right after starting the section. FunctionmoveSection
is called because of a Rule:
protected function moveSection (inSection:IFlowSection) : void { var x:Number, y:Number; switch (inSection.getName()) { case AppSettings.SECTION1: x = 0; y = 40; break; // etcetera } var queue:ActionQueue = moveQueue(x, y); // creates a moving animation as ActionQueue FlowManager.defaultFlowManager.addAction(inSection.startAction); FlowManager.defaultFlowManager.addAction(queue); }
Automatic loading of missing Sections
When a section is not found, FlowManager will try to load it. Right before loading it will dispatch an event with subtype FlowNavigationEvent.WILL_LOAD. After loading successfully an event with subtype FlowNavigationEvent.LOADED is sent. You may even navigate to a section within a to-be-loaded movie. After loading you can proceed to the nested section by using thedestination
property of the incoming FlowNavigationEvent.
protected function attachMovie (e:FlowNavigationEvent) : void { if (section != null) { // add child clip... FlowManager.defaultFlowManager.goto(e.destination); } }Note: calls to sections within a to be loaded movie are not supported yet.
Responding to state changes
Before a new state change, an event with subtype FlowNavigationEvent.WILL_UPDATE is sent. After the transition has been complete, an event with subtype FlowNavigationEvent.UPDATE is sent. In the example demo a MenuController listens for state changes. It is subscribed to changes using:FlowManager.defaultFlowManager.addEventListener(FlowNavigationEvent._EVENT, handleNavigationEvent);
The receiving method is:
private function handleNavigationEvent (e:FlowNavigationEvent) : void { switch (e.subtype) { case FlowNavigationEvent.WILL_UPDATE: case FlowNavigationEvent.UPDATE: e.stopImmediatePropagation(); // handle button state break; } }Because FlowNavigationEvent events bubble through and other classes may deal with update changes as well, chance is that we get stuck in a recursive loop. We end this by writing
e.stopImmediatePropagation();
.
How to start
- Create a name list of navigatable elements in your Flash site/project.
- I prefer to have a list of consts in
data/AppSettings
, so when referring to "Intro" I writeAppSettings.SECTION_INTRO
. - Section names are hierarchical. Use dots to indicate levels. For example: "Sections.Gallery" and "Sections.Gallery.Latest"
- Normally the main controller for each SWF is a LocalController. Now make it inherit from FlowSection, and pass the name of the section in the super constructor call.
- Other navigatable sections (like nested MovieClips) also need a FlowSection class.
- If your main controller is a FlowSection as well, do not forget to make it visible.
- Now write
FlowManager.defaultFlowManager.goto( "My.starting.section" );
(insert your section)
Events broadcasted to listeners
-
FlowNavigationEvent with type:
UPDATE
-
FlowNavigationEvent with type:
WILL_UPDATE
-
FlowNavigationEvent with type:
SECTIONS_STOPPING
-
FlowNavigationEvent with type:
SECTIONS_STARTING
-
FlowNavigationEvent with type:
WILL_LOAD
-
FlowNavigationEvent with type:
LOADED
-
FlowNavigationEvent with type:
FINISHED
Summary
Class properties
Instance methods
- toString () : String
-
addRule
(inRule:FlowRule) : void
- Registers a FlowRule.
-
addRuleForSections
(inRule:FlowRule, inSectionNames:Array) : void
- Applies a FlowRule to a list of sections.
-
addAction
(inAction:IAction) : void
- Adds an IAction to the existing list of actions.
-
registerFlowSection
(inFlowSection:IFlowSection, inSectionName:String = null) : void
- Stores FlowSections in a look-up hash.
-
goto
(inSectionName:String, inTrigger:Object = null, inStopEverythingFirst:Boolean = true, inUpdateState:Boolean = true) : void
- Goes to a new section.
-
removeSection
(inSectionName:String) : Boolean
- Removes a FlowSection from the list.
-
setDownloadDirectory
(inUrl:String) : void
- Sets the download directory of to be loaded movies.
-
getSectionByName
(inSectionName:String) : IFlowSection
- Gets the IFlowSection with name inSectionName, if it has been registered.
-
getFlowNavigationDataByName
(inSectionName:String) : FlowNavigationData
- Gets the FlowNavigationData with name inSectionName, if it has been registered (once goto has been called).
-
getCurrentSection
() : IFlowSection
- Gets the currently visited IFlowSection.
-
getCurrentSectionName
() : String
- Gets the name of the currently visited IFlowSection.
Class properties
defaultFlowManager
static defaultFlowManager:FlowManager(read)
Returns
- The default global instance of the FlowManager.
Instance methods
addAction
function addAction(inAction:IAction) : void
Adds an IAction to the existing list of actions.
Parameters
inAction:action to add
addRule
function addRule(inRule:FlowRule) : void
Registers a FlowRule.
Parameters
inRule:FlowRule to register
addRuleForSections
function addRuleForSections(inRule:FlowRule,
inSectionNames:Array) : void
Applies a FlowRule to a list of sections.
Parameters
inRule :FlowRule to register
inSectionNames:list of section names to apply the Rule to
getCurrentSection
function getCurrentSection() : IFlowSection
Gets the currently visited IFlowSection.
getCurrentSectionName
function getCurrentSectionName() : String
Gets the name of the currently visited IFlowSection.
getSectionByName
function getSectionByName(inSectionName:String) : IFlowSection
Gets the IFlowSection with name inSectionName, if it has been registered.
Parameters
inSectionName:name of the FlowSection
Returns
- The found FlowSection
goto
function goto(inSectionName:String,
inTrigger:Object = null,
inStopEverythingFirst:Boolean = true,
inUpdateState:Boolean = true) : void
Goes to a new section.
Parameters
inSectionName :name of the IFlowSection to move to
inStopEverythingFirst:(optional) whether the current actions are finished first (false) or stopped halfway (true); default: true
inUpdateState :(optional) whether the state is updated when going to the new section. This is not always desirable - for instance showing a navigation bar should not update the state itself. Default: true (state is updated).
Example
-
To show section "Gallery", write:
FlowManager.defaultFlowManager.goto("Gallery");
If you want to track the object or button that causes the goto call, add parameterinTrigger
:FlowManager.defaultFlowManager.goto("Gallery", gallery_btn);
To continue all current animations, set parameterinStopEverythingFirst
to true:FlowManager.defaultFlowManager.goto("Gallery", gallery_btn, true);
To prevent sending out a state update, set parameterinUpdateState
to false:FlowManager.defaultFlowManager.goto("Gallery", gallery_btn, true, false);
registerFlowSection
function registerFlowSection(inFlowSection:IFlowSection,
inSectionName:String = null) : void
Stores FlowSections in a look-up hash. Called by FlowSections.
Parameters
inFlowSection:the FlowSection to register
removeSection
function removeSection(inSectionName:String) : Boolean
Removes a FlowSection from the list. If the section was loaded, it will be removed.
Parameters
inSectionName:name of the section to remove
Implementation note
- Calls MovieManager.removeMovie.
Returns
- True if the section was successfully removed; otherwise false.
setDownloadDirectory
function setDownloadDirectory(inUrl:String) : void
Sets the download directory of to be loaded movies.
Parameters
inUrl:URL of the directory; by default the current movie directory is used
toString
override function toString() : String
Overrides
- EventDispatcher.toString