-
Notifications
You must be signed in to change notification settings - Fork 8
Creating Custom Events
This is a guide on how to define a custom event in a code mod. If you are not familiar with the C# language, or do not have prior experience
in making code mods for Celeste, Lua Cutscenes 🔗 may be a more suitable option.
If you are interested in making code mods but don't know how, read Your First Code Mod to get started.
In order to use any sort of custom event, an EventTrigger
needs to be placed somewhere in the level.
Event Triggers work the same way as any other Trigger
, and are activated when the player enters their hitbox.
The event
data field is the ID of the event that will be triggered.
Event IDs should be unique, and should include the mod name or a nickname that is unlikely to be reused by another mod.
If an Event Trigger is placed on the edge of a screen, it may be necessary to set the onSpawn
field to true in order to trigger the event immediately upon entering the screen.
Hooking EventTrigger.OnEventTrigger
provides a way to execute nearly any action when an EventTrigger is entered.
Simply check that the eventID matches the desired event, and return true
if it does, to notify the game that an appropriate event has been found.
Important
While this can be a convenient way to test code, it is only recommended if there are no other predefined options.
For example, an EventTrigger
could be used to set level flags, or end a level, but if no other behaviour is desired, a FlagTrigger
or CompleteAreaTrigger
is a better alternative.
To create a custom entity that will be automatically added when the appropriate EventTrigger
has been entered, create a class that extends Monocle.Entity
and annotate it with the [CustomEvent]
attribute
so that the game can detect it when it loads a map:
[CustomEvent("mymodname/myevent")]
class MyEvent : Entity { ... }
You have to define a constructor for the game to be able to build your event. The allowed signatures for this constructor are, in order of precedence:
public MyEvent(EventTrigger trigger, Player player, string eventID)
public MyEvent()
You can also give a custom event multiple IDs (useful for backwards compatibility):
[CustomEvent("mymodname/myevent", "mynewmodname/myevent")]
or have different IDs call different static generator methods for your entity:
[CustomEvent(
"mymodname/myeventup = LoadUp",
"mymodname/myeventdown = LoadDown"
)]
public class MyEvent : Entity {
public static Entity LoadUp(EventTrigger trigger, Player player, string eventID)
=> new MyEvent(player, eventID, Directions.Up);
public static Entity LoadDown(EventTrigger trigger, Player player, string eventID)
=> new MyEvent(player, eventID, Directions.Down);
[...]
}
If no generator method is specified in the CustomEvent ID, Everest will look for a generator method named Load
.
Note
A generator method, if provided, will take precedence over any defined constructors.
One major use of Event Triggers is in triggering cutscenes, which is done by adding a CutsceneEntity
to the Level.
This can be done using either of the methods described above.
CutsceneEntity
is an abstract class that contains two required methods:
-
OnBegin(Level level)
should be used to set up the cutscene, and to add a newCoroutine
🔗 to execute the cutscene within. -
OnEnd(Level level)
should be used to clean up after the coroutine has finished. If necessary, theWasSkipped
field should be checked in case the cutscene was ended prematurely.
EndCutscene(Level level, bool removeSelf)
should be called at the end of the coroutine, to let the level know it has completed.
Example Cutscene:
[CustomEvent("MyModName/MyCustomEvent")] //"MyModName" should be replaced by the name of your mod. This is to prevent duplicate events from existing.
public class TestEvent : CutsceneEntity
{
private Player player;
public TestEvent(EventTrigger trigger, Player player, string eventID) : base()
{
this.player = player;
}
public override void OnBegin(Level level)
{
Add(new Coroutine(cutscene()));
}
private IEnumerator cutscene()
{
//simple cutscene that turns the player to the left, waits one second, turns the player to the right, waits 0.5 seconds, then ends.
player.StateMachine.State = Player.StDummy;
player.Facing = Facings.Left;
yield return 1f;
player.Facing = Facings.Right;
yield return 0.5f;
EndCutscene(Level); //Tells the level the cutscene has been completed and calls "OnEnd".
}
public override void OnEnd(Level level)
{
if (WasSkipped)
{
//make sure the player is facing right, as they would have been facing right at the end had the cutscene not been skipped.
player.Facing = Facings.Right;
}
//set the player's state back to normal
player.StateMachine.State = Player.StNormal;
}
}
Cutscene nodes are named points that can be placed in a level, and are used for reference from within a cutscene.
They can be retrieved using CutsceneNode.Find(string name)
.
Home
Contributing
FAQ
Useful Links
Your First Custom Map
Your First Texture Pack
Mod Setup
Custom Maps
Texture Packs
Uploading Mods
Generated Dialog Keys
Reference Code Mod🔗
Vanilla Audio IDs
Vanilla Decal Registry Reference
Character Portraits
Mod Structure
Debug Mode
Debug Map
Command Line Arguments
Environment Variables
Install Issues
Common Crashes
Latency Guide
everest.yaml Setup
Mapping FAQ
Map Metadata
Vanilla Metadata Reference
Adding Custom Dialogue
Overworld Customisation
Entity & Trigger Documentation
Custom Entity List🔗
Camera
Ahorn Scripts
Custom Tilesets
Tileset Format Reference
Stylegrounds
Reskinning Entities
Skinmods
Decal Registry
Chapter Complete Screen
Custom Portraits
Adding Custom Audio
Advanced Custom Audio
Code Mod Setup
Making Code Mods
Settings, SaveData and Session
Everest Events
Understanding Input
Logging
Cross-Mod Functionality
Recommended Practices
Core Migration Guide
Lönn Integration🔗
Custom Events
Adding Sprites
Adding Preexisting Audio