Skip to content

Creating Custom Events

PixelPiano edited this page Dec 1, 2024 · 10 revisions

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.

Contents

EventTrigger

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.

OnCustomEvent

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.

[CustomEvent] Attribute

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.

CutsceneEntity

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 new Coroutine 🔗 to execute the cutscene within.
  • OnEnd(Level level) should be used to clean up after the coroutine has finished. If necessary, the WasSkipped 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;
    }
}

CutsceneNode

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).

Clone this wiki locally