Bethesda Tutorial Papyrus Events and Properties
Bethesda Tutorial Papyrus Events and Properties | |
---|---|
Scripting Series, Chapter N/A | |
Return to Tutorial Hub | |
Previous Tutorial | Next Tutorial |
Overview
This tutorial introduces Papyrus Events and Properties.
You will learn:
- Basic information about Papyrus Events and how they are triggered.
- Basic information about Papyrus Properties, and how to create, fill, and use them.
- How the game interacts with Papyrus by communicating events.
- How Papyrus interacts with the game by performing actions on properties.
This tutorial builds on the events in Lokir's Tomb, the sample dungeon created in the Level Design Tutorials. If you haven't completed that tutorial, you can download a plugin with the finished level, but be aware that this tutorial assumes you are already familiar with the elements introduced there, including things like:
- Basic editor navigation.
- Concepts like Ambushes, Activation, and Activate Parents.
The Plan
Currently, when you enter the final chamber in Lokir's Tomb, the boss Draugr rises from his sarcophagus and attacks. Let's make this battle more exciting by introducing some unique scripted elements. We'll start by adding two dead Draugr to the room that we reanimate (with full spell effects) as the boss emerges from his tomb.
Setting the Stage
First, let's set up the draugr we want to resurrect. In the editor, open the cell LokirsTomb and focus your view on the cave area; this is our boss chamber. In the Object Window, navigate to Actors>Actor - or all - and use the filter to locate "LvlDraugrMissileMale" and "LvlDraugrWarlockMale". Drag and drop one of each into the room.
Placed actors like these Druagr start alive (well, relatively speaking), and we need them to be dead. Double-click on each of them and check the box marked "Starts Dead".
Save your plugin and run it in-game to check things out so far. You'll notice that two dead Draugr now lie in ragdoll on the floor of the room. That's what we want for now - but let's head back to the Creation Kit and get them back on their feet.
Event Planning
Introduction to Events
We want these Draugr to be resurrected as the boss emerges from his tomb. That is, we want this to happen in response to an Event.
"Events" are actions or state changes that the game notifies Papyrus about. There are hundreds of different events, such as:
- Interacting with an object or character (pulling a lever, opening a door, looting a body).
- Picking up, equipping, unequipping, or dropping items.
- Entering or leaving a Location.
- Entering or leaving combat, getting hit in combat, dying.
- And much, much more.
Simply put - if we want to react to something in the game, Events are the way to do it.
Triggering the Activation Event
You may not realize it, but our example already uses an event sent to Papyrus. This is how the boss knows when to get out of his sarcophagus. If you've done the Ambushes Tutorial, you may recall that the boss has uses a trigger volume as its Activate Parent. When the player steps into the trigger volume, the trigger activates the boss, causing him to get up.
There are pre-existing scripts at play here, but here's what's basically happening: - The player collides with the trigger, causing an onTriggerEnter event
- A script on trigger sends an Activate event in response to this.
- The Draugr boss is an activate child of the trigger, and therefore also receives an activation
- A script on the Draugr boss responds to the Activation by making him climb out of his tomb
We can use the same trigger to also send Activate events to our two dead draugr.
For each of the Draugr:
- Double-click on the Draugr to open its properties window.
- Locate and choose the Activate Parents tab.
- Right-click in the empty list and select 'New'. The Activate Ref Selection dialog appears.
- Click 'Select Reference in Render Window'
- When the Crosshair cursor appears, double-click on the trigger volume.
Both Draugr will now receive an Activate Event when the player enters the trigger. We aren't actually using that event to do anything yet, however. Next you'll write a script that responds to that event.
Scripting the Resurrection
Initial Setup
It's time to start scripting. Open your preferred text editor (we provide setups for Notepad++ and Sublime Text), and create a new file named "LokirsDraugrResurrection.psc".
Start with the following two lines of script:
scriptName LokirsDraugrResurrection extends Actor
{Resurrects the two dead Draugr in Lokir's Tomb.}
We want to tell it to listen for the onActivate event. Activation occurs when another entity, such as the player or an NPC, tries to "use" or "activate" the object. In this case, the script on the ambush trigger will be sending the activation.
scriptName LokirsDraugrResurrection extends Actor
{Resurrects the two dead Draugr in Lokir's Tomb.}
Event OnActivate(ObjectReference akActionRef)
; Cast a Reanimate spell on the Draugr
EndEvent
Casting the Spell
Next, we need to cast the Reanimate spell on the Draugr when the Activation event is received. To do this, we'll need to create a "Property". In this case, it's a property of the Spell type we'll be using. This will allow us to tell the Creation Kit which specific spell to use.
For those who used the "legacy" scripting language in editors for earlier Bethesda tools, this is a significant change. In Papyrus, you can't simply write in the Editor ID of the object you want to manipulate-- you have to use a property. This allows much more flexible and reusable scripts, since each instance of the script can have a different values for its properties. This page provides an overview of properties and variables in Papyrus
Begin by "declaring" the Spell property, as below:
scriptName LokirsDraugrResurrection extends Actor
{Resurrects the two dead Draugr in Lokir's Tomb.}
Spell property reanimateSpell auto
EVENT OnActivate(ObjectReference akActionRef)
; Cast a Reanimate spell on the Draugr
EndEVENT
- Spell - Is the property type. It tells Papyrus that we'll provide it with a Spell to use.
- property - Is a reserved word this is how we tell Papyrus to expect a new property.
- reanimateSpell - This is the name we're giving to our variable. In-game, Papyrus will use our spell whenever we reference this property.
- auto - Another reserved word - it tells Papyrus to automatically generate 'get' and 'set' functions for the variable. Don't worry about the details here; this usually doesn't matter.
- ; ... - Semicolons are the comment character in Papyrus. Use these to keep notes on what your script is doing. Anything after a semicolon is ignored by the game, so you can write whatever you want.
We've defined a spell, but it isn't doing anything yet. Spell properites can use the Cast function, which we'll use next. Copy the new line of script inside the onActivate event below:
scriptName LokirsDraugrResurrection extends Actor
{Resurrects the two dead Draugr in Lokir's Tomb.}
Spell property reanimateSpell Auto
EVENT OnActivate(ObjectReference akActionRef)
; Cast a Reanimate spell on the Draugr
reanimateSpell.Cast(Self, Self)
EndEVENT
Like many functions, we have to pass one or more "arguments" to let the function know what to do. In this case, Cast requires a source and a target. For this example, we just needed the draugr to "cast" the spell at itself. The reserved word "Self" is useful in this case; by passing it into both fields, Papyrus automatically knows we want to reference the same entity the script is attached to. In this case, then, "self" is the same as creating a new property for the draugr we'll be resurrecting - only much simpler.
Save and compile the script, then return to the Creation Kit.
Attaching the Script
Back in the Editor, double-click on one of the Draugr to open its Properties window, then go to the Scripts tab (you can hit the 'End' key to jump right there; it's the last tab in the list). This is where we'll hook up our script:
- Click Add.
- In the Add script... Window, enter the name of our new script, 'LokirsDraugrResurrection'.
- Double-click on the script to add it to the Draugr.
That's attached the script to this reference, but we still need to tell the Creation Kit what spell to cast. We'll be using "dunReanimateSelf", a special non-playable spell created especially for events like this one.
- Select LokirsDraugrResurrection in the Scripts Tab
- ClickProperties
- The list that appears only has a single property - reanimateSpell.
- Select reanimateSpell and click Edit Value.
- Notice that this list only displays valid forms - spells, in this case
- Select dunReanimateSelf. Click OK.
Repeat this process for the other Draugr. Save your work and give it a try in the game!
Debugging the Resurrection
At first glance, everything probably worked out as expected. However, if you kept playing, you may have noticed that the resurrection will occur any time the draugr receives an activate event. This includes when the player loots the body! That's not the sort of nasty surprise we intended on! Unexpected bugs like this can be handled using a bit of extra logic, however.
There are a number of ways we could fix this, like making sure the resurrection only happens once by using a control variable. We could also make sure the script ignores activation from the player, but that could still result in accidental re-resurrection by an NPC.
Instead, we'll ensure that the resurrection only occurs when the triggerbox sends the activation. We'll do this by creating another property - an Object Reference this time - which we'll assign to our trigger.
We'll compare this new property to "akActionRef", the parameter supplied by the OnActivate() event. This parameter is a sort of special variable, usable only within this onActivate event, which Papyrus automatically assigns to the reference responsible for the activation.
scriptName LokirsDraugrResurrection extends Actor
{This Script lives on the dead minion draugr in Lokir's Tomb. It handles their resurrection}
Spell property reanimateSpell Auto ; this is the special self-resurrection spell to use
objectReference property myTrigger auto ; This is the reference we are waiting on to send an activate
Event OnActivate(ObjectReference akActionRef)
; I've been activated - see if was my trigger
if (akActionRef == myTrigger)
; Cast a Reanimate spell on the Draugr
reanimateSpell.Cast(Self, Self)
EndIf
EndEvent
objectReference This is the data type of our new property myTrigger As before, this is a name we've chosen for our new property if This is the beginning of an "if statement". If the statement behind the if is true, then the script below will run. Otherwise, Papyrus skips ahead and resumes processing commands after the corresponding "endif" below. (akActionRef == myTrigger) This is our test condition. The "==" is a logical operator which translates to "is equal to". In this case, it's telling the if statement to continue only if akActionRef and myTrigger are filled with he same reference. endIf This is necessary to close any if statement. It's possible to have multiple if statements within each other for more complicated operations, so it becomes important to keep track of your logic using these.