Both sides previous revisionPrevious revisionNext revision | Previous revision |
public:t-vien-09-1:lab_5_materials [2009/02/12 01:04] – hannes | public:t-vien-09-1:lab_5_materials [2024/04/29 13:33] (current) – external edit 127.0.0.1 |
---|
====== Lab 5 - Actors and State Machines ====== | ====== Lab 5 - Actors and State Machines ====== |
| |
<note warning>Still under construction! Please wait!</note> | |
| |
==== Before You Start ==== | ==== Before You Start ==== |
- **Adding a Moving Character:**\\ Create a **Character** class derived from **FSM.FSM** (Finite State Machine). In the constructor, you first need to call the constructor of the parent like this:<code python> | - **Adding a Moving Character:**\\ Create a **Character** class derived from **FSM.FSM** (Finite State Machine). In the constructor, you first need to call the constructor of the parent like this:<code python> |
FSM.FSM.__init__(self,"Character") | FSM.FSM.__init__(self,"Character") |
</code> Then, create a new node called "mover" that will become the parent of the actor. Place the mover at **(3,5,0)**. This extra node is not always needed, but it makes it possible to redefine the "front" of the model by rotating it with respect to this new model base (think of the mover as a platform that the actor is standing on). Now you should load the **[[http://www.ru.is/kennarar/hannes/useful/PandaManual/Loading_Actors_and_Animations.1.html|Actor]]** from the model **"models/FelOrcWarriorAxe.egg"** with the animation **'anims' : "models/FelOrcWarriorAxe.egg"** and attach it to the mover. Define the front facing of the actor by rotating it **90** degrees (with respect to the mover). Notice that the animations are stored in exactly the same file as the model in this case. Furthermore, all the different motions that the Orc can perform are bundled together into a single animation here (that's just the way that this model/animation was constructed). In order to play a particular motion sequence from this animation, you always specify the //fromFrame// and //toFrame//. So, to start the Orcs walking animation, call **loop('anims',fromFrame=166, toFrame=193)** on your actor. The //fromFrame// and //toFrame// values of other motions can be found inside the text file //"FelOrcWarriorAxe.animsez". Finally create a new **Sequence** of [[http://www.ru.is/kennarar/hannes/useful/PandaManual/Lerp_Intervals.1.html|Lerp Intervals]] that first changes **mover** heading to **90** (in about 0.8 seconds) and then moves it to position **(-3,5,0)** in about 3 seconds. Rotate the mover back and bring it back to the original position. Now loop this interval and watch your actor pace around! | </code> Then, create a new node called "mover" that will become the parent of the actor. Place the mover at **(3,5,0)**. This extra node is not always needed, but it makes it possible to redefine the "front" of the model by rotating it with respect to this new model base (think of the mover as a platform that the actor is standing on). Now you should load the **[[http://www.ru.is/kennarar/hannes/useful/PandaManual/Loading_Actors_and_Animations.1.html|Actor]]** from the model **"models/FelOrcWarriorAxe.egg"** with the animation **'anims' : "models/FelOrcWarriorAxe.egg"** and attach it to the mover. Define the front facing of the actor by rotating it **90** degrees (with respect to the mover). Notice that the animations are stored in exactly the same file as the model in this case. Furthermore, all the different motions that the Orc can perform are bundled together into a single animation here (that's just the way that this model/animation was constructed). In order to play a particular motion sequence from this animation, you always specify the //fromFrame// and //toFrame//. So, to start the Orcs walking animation, call **loop('anims',fromFrame=166, toFrame=193)** on your actor. The //fromFrame// and //toFrame// values of other motions can be found inside the text file "FelOrcWarriorAxe.animsez". Finally create a new **Sequence** of [[http://www.ru.is/kennarar/hannes/useful/PandaManual/Lerp_Intervals.1.html|Lerp Intervals]] that first changes **mover** heading to **90** (in about 0.8 seconds) and then moves it to position **(-3,5,0)** in about 3 seconds. Rotate the mover back and bring it back to the original position. Now loop this interval and watch your actor pace around! |
- **Starting a State Machine:**\\ Create a state for pacing such that when he enters the state he paces around, but stops when he leaves the state. You do that by adding the methods **enterPacing(self)** and **exitPacing(self)**. In the former, you loop the **Walk** animation like before, but instead of calling **loop** on your movement interval, call **resume**. This is so that he can continue moving from where he left off when he was interrupted. In the latter, you'll have to stop the walking animation, but instead of calling **stop**, which leaves him frozen at the first frame of the animation, you should start a new idle motion loop (fromFrame=3083, toFrame=3166). Create another state (with both enter and exit methods) called **Noticing** and for now, just print out a message on the screen. Test your state machine by for example accepting two different keystrokes, one which enters the **Pacing** state by calling **self.request('Pacing')** from its handler and another that enters the **Noticing** state with **self.request('Noticing')**. | - **Starting a State Machine:**\\ Create a state for pacing such that when he enters the state he paces around, but stops when he leaves the state. You do that by adding the methods **enterPacing(self)** and **exitPacing(self)**. In the former, you loop the **Walk** animation like before, but instead of calling **loop** on your movement interval, call **resume**. This is so that he can continue moving from where he left off when he was interrupted. In the latter, you'll have to stop the walking animation, but instead of calling **stop**, which leaves him frozen at the first frame of the animation, you should start a new idle motion loop (fromFrame=3083, toFrame=3166). Create another state (with both enter and exit methods) called **Noticing** and for now, just print out a message on the screen. Test your state machine by for example accepting two different keystrokes, one which enters the **Pacing** state by calling **self.request('Pacing')** from its handler and another that enters the **Noticing** state with **self.request('Noticing')**. |
- **Making the Character Look:**\\ Now add to the actor the ability to face you when his **self.is_looking** member variable is set to **True**. Simply do this by creating a task called **look** and inside it call **lookAt(base.camera)** on the nodepath of the mover, but only if **self.is_looking** is **True**. Now set the **self.is_looking** variable to False when you enter the Pacing state and True when you enter the Noticing state. Test that this works (try approaching him from different directions). One problem that you may notice is that your character faces the wrong way once he resumes his pacing after having looked at you. You can fix this with a member variable that stores the last facing angle of the pacing state when leaving it and sets it again when re-entering it. See if you can get it fixed that way. | - **Making the Character Look:**\\ Now add to the actor the ability to face you when his **self.is_looking** member variable is set to **True**. Simply do this by creating a task called **look** and inside it call **lookAt(base.camera)** on the nodepath of the mover, but only if **self.is_looking** is **True**. Now set the **self.is_looking** variable to False when you enter the Pacing state and True when you enter the Noticing state. Test that this works (try approaching him from different directions). One problem that you may notice is that your character faces the wrong way once he resumes his pacing after having looked at you. You can fix this with a member variable that stores the last facing angle of the pacing state when leaving it and sets it again when re-entering it. See if you can get it fixed that way. |