-
Notifications
You must be signed in to change notification settings - Fork 71
/
Copy path08-FlowOfExecution.html
135 lines (94 loc) · 8.48 KB
/
08-FlowOfExecution.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
<head>
<link rel="stylesheet" type="text/css" href="stylesheet.css"/>
<script type="text/javascript" src="js/pageToc.js"></script>
<script type="text/javascript" src="js/sh/scripts/shCore.js"></script>
<script type="text/javascript" src="js/sh/scripts/shBrushJScript.js"></script>
<script type="text/javascript" src="js/sh/scripts/shBrushPhp.js"></script>
<script type="text/javascript" src="js/sh/scripts/shBrushPlain.js"></script>
<script type="text/javascript" src="js/sh/scripts/shBrushXml.js"></script>
<link type="text/css" rel="stylesheet" href="js/sh/styles/shCore.css"/>
<link type="text/css" rel="stylesheet" href="js/sh/styles/shThemeDefault.css"/>
<script type="text/javascript">
SyntaxHighlighter.config.clipboardSwf = 'js/sh/scripts/clipboard.swf';
SyntaxHighlighter.all();
</script>
<title>Flow Of Execution</title>
</head>
<body>
<h1>Flow Of Execution</h1>
<p>An important part of understanding PushButton Engine is understanding what is run when. While most of the behavior described in this chapter is completely automated, it is helpful background information when you are developing or debugging components.</p>
<p><b>Note:</b> Because PushButton Engine is very modular, different components may execute different code. Components should document this behavior. Most components will build off "core" PushButton Engine code like the ProcessManager and there will have behavior outlined in this document.</p>
<div id="pageToc"></div>
<div id="contentArea">
<h2>Main Loop</h2>
<p>PushButton Engine does not implement its own main loop or take "control" of your game. Everything is instigated by you, the developer, as this chapter outlines. In general, code and managers are created and executed on demand - for instance the first time you use the TemplateManager, it initializes itself. When you add the first object to the ProcessManager, it registers for callbacks, and when you unregister the last object it turns itself off.</p>
<p>As a rule: you pay for what you use.</p>
<h2>Engine Startup</h2>
<p>If you are using any of the PushButton Engine MXML tags, these are processed as early as possible in application startup. Level-related tags are processed when the applicationComplete event fires. Resource-related tags are processed as soon as they are loaded by Flex - usually in the first few frames.</p>
<p>As a side effect of these tags, the ResourceManager, LevelManager, and other managers may be initialized. They are all created on demand, as mentioned above, when you first access them via their static Instance member.</p>
<p>If you are manually initializing your game, by directly calling the managers without using the tags, then it's up to you when everything starts.</p>
<h2>ProcessManager</h2>
<p>The main piece of code that will be run after initialization is done is the ProcessManager, which is called at the start of every frame. The ProcessManager is responsible for managing the processing of time-related code. It does this with two kinds of callbacks - <b>per tick</b> and <b>per frame</b>.</p>
<p>The ProcessManager allows you to set a priority for when in the course of processing time your object is called back. Higher priority objects are called first, followed by lower priority objects. Priority defaults to zero.</p>
<h3>Frame Callbacks</h3>
<p>Per-frame callbacks happen via the IAnimatedObject interface. Simply implement the IAnimatedObject interface, and call <i>ProcessManager.instance.addAnimatedObject(this);</i> to get callbacks. For example:</p>
<pre class="brush: js">
// Example of a component that receives per-frame callbacks.
public class DemoAnimatedComponent extends EntityComponent implements IAnimatedObject
{
public function onFrame(elapsed:Number):void
{
trace("Frame happened, and " + elapsed + "ms passed since last time.");
}
protected override function onAdd():void
{
ProcessManager.instance.addAnimatedObject(this);
}
protected override function onRemove():void
{
ProcessManager.instance.removeAnimatedObject(this);
}
}</pre>
<p>Because you have to implement the above logic for every animated object, there is a helper base class provided with the engine, PBLabs.Engine.Components.AnimatedComponent, which not only automatically registers/unregisters for you, but lets you specify a priority.</p>
<h3>Ticks</h3>
<p><b>Ticks</b> are extremely useful when trying to implement gameplay that doesn't break at very high or low framerates. Ticks are guaranteed to happen at a fixed interval (see ProcessManager API documentation for details on the interval). At the start of each frame, the ProcessManager checks to see how long it has been since the last tick was processed. If it has been one or more tick-intervals, then it processes a tick. If it has been a long time, then it will run many ticks - however are many needed to stay caught up with realtime.</p>
<p>This is great for gameplay because it means the game runs consistently no matter what - internally, everything is processed at 32Hz (or whatever you set it the tick rate to). The rule is to implement any gameplay code in the onTick callbacks, and any "display" code - for instance updating a particle system or interpolating position for smoother display - in onFrame.<p>
<p>A simple component that receives ticked looks like this. Notice it implements the ITickedObject interface:</p>
<pre class="brush: js">
// Example of a component which receives tick callbacks.
public class DemoTickedComponent extends EntityComponent implements ITickedObject
{
public function onTick(tickRate:Number):void
{
trace("A tick happened!");
}
protected override function onAdd():void
{
ProcessManager.instance.addTickedObject(this);
}
protected override function onRemove():void
{
ProcessManager.instance.removeTickedObject(this);
}
}</pre>
<p>As with the AnimatedComponent base class, there is a TickedComponent base class which simplifies creating a component that creates ticks.</p>
<p>To summarize the behavior of ProcessManager, at the start of every frame (on the ENTER_FRAME event), it does the following:</p>
<ol>
<li>Process IAnimatedObject callbacks.</li>
<li>Process any pending ticks by calling ITickedObject onTick zero or more times.</li>
<li>Process any pending events.</li>
</ol>
<h3>Managing Time</h3>
<p>There are several convenient values the ProcessManager calculates for you. These are virtualTime, which is milliseconds of simulated time since the game started based on elapsed ticks, and interpolationFactor, which is identical to the parameter passed to onInterpolateTick. In general, you should use virtualTime instead of flash.utils.getTimer() and interpolationFactor when you need to be aware of where you are relative to ticks.</p>
<p>You may want to experiment with altering timeScale on the ProcessManager. This will allow you to do bullet time type effects or speed up the action. Unfortunately, you cannot reverse time.</p>
<h2>Shutdown</h2>
<p>Currently, no code is run at game shutdown time.</p>
<h2>Low-level Explanations</h2>
<p>The ProcessManager exists to provide a consistent environment for gameplay code to execute. If you are concerned with how time is processed in Flash, you might benefit from reading Sean Christmann's article on <a href="http://www.craftymind.com/2008/04/18/updated-elastic-racetrack-for-flash-9-and-avm2/">Updated 'Elastic Racetrack' for Flash 9 and AVM2</a>, based on Ted Patrick's original <a href="http://www.onflex.org/ted/2005/07/flash-player-mental-model-elastic.php">Elastic Racetrack</a> article.</p>
<h2>Conclusion</h2>
<p>PushButton Engine starts its managers on demand. There is no strict main loop. If you are concerned about the specific details of exactly how a specific piece of code is run, you should review its API docs and source code - this will be the most up to date and accurate.</p>
</div>
</body>
</html>