-
Notifications
You must be signed in to change notification settings - Fork 8
Mod Structure
- Getting started for mapping
- Getting started for code modding
- Adding content to your mod
- Using Helper Mods
- Packaging your mod for publishing
- Optional Dependencies for everest.yaml (advanced)
- Complete Reference
- Examples
- Create a folder named after your mod in the Mods folder, for example
Mods/yourmod
- Drop your map inside it, in
Mods/yourmod/Maps/yournickname/campaignname/1-yourmap.bin
- a "campaign" is a group of maps in chapter select. For example, vanilla Celeste is a campaign with 11 chapters (counting Prologue and Epilogue).
- the
1-
prefix in the bin name will be useful if you want to make a campaign with multiple chapters.
- Create the
Mods/yourmod/Dialog
folder (:warning: not Dialogue) and create a file named English.txt in it. Add those lines to it:
yournickname_campaignname= Your Campaign Name
yournickname_campaignname_1_yourmap= Your Map Name
Now, your map should show up in-game in its own category, and have the name you defined in your English.txt. If this doesn't work, double-check the file isn't actually called English.txt.txt. For this, check this box to see the actual file name:
- create an everest.yaml file at
Mods/yourmodname/everest.yaml
. Here is how it should look like:
- Name: YourModName
Version: 1.0.0
Dependencies:
- Name: Everest
Version: 1.3000.0
Replace 3000 with the version of Everest you're using, and YourModName with your mod name (it should be unique, and you should not change it once your map is published).
You are now ready to go and you will be able to add more folders / files to Mods/yourmod
when you will need to!
To get started and playtest your map quickly, you can also just save your map as a bin file in your Mods folder next to Celeste (Mods/mymap.bin
). It will appear as "uncategorized" in-game. This is not recommended if you have any plans to share or publish your map, but can be occasionally useful for one-off tests.
You will need to get a folder structure for your mod as soon as:
- you want to publish your map, and want to package it in a zip
- or, you need to have custom dialogue, graphics, audio, etc.
- you want to take advantage of automatic reloading when saving the map
This page covers mod structure for code mods. To get started with code modding itself, check Your First Code Mod.
Once you compiled your mod, you should get a DLL (often in a bin/Debug/net452
folder or similar). In terms of mod structure, all you need to do is:
- Create a
Mods/yourmodname
folder - Inside that folder, create a
Code
folder - Place the .dll somewhere inside the
Code
folder - create an everest.yaml file (
Mods/yourmodname/everest.yaml
) looking like this:
- Name: YourModName
Version: 1.0.0
DLL: Code/mymod.dll
Dependencies:
- Name: Everest
Version: 1.3000.0
Replace:
- 3000 with the version of Everest you're using
- YourModName with your mod name (it should be unique, and you should not change it once your map is published)
-
Code/mymod.dll
with the path to your mod's DLL (for example, here, Everest would expect the DLL to be inMods/yourmod/Code/mymod.dll
).
If you are making a helper, pay particular attention to the version number, since your mod is going to be used as a dependency: Everest uses semantic versioning 🔗. It means that version numbers are on the format MAJOR.MINOR (or MAJOR.MINOR.PATCH), and that changing MAJOR means you made a change that would break mods depending on your helper, which you usually don't want.
So, if a mod depends on YourMod 1.3:
- installing YourMod 1.0 won't work
- installing YourMod 1.3 will work (of course)
- installing YourMod 1.6 will work
- installing YourMod 2.0 won't work. The user will have to get YourMod 1.x instead.
If you want to make another chapter for your campaign, just save the map bin in Mods/yourmodname/Maps/yournickname/yourcampaignname
next to your existing chapter. They will appear next to each other in chapter select.
If you want to make a B-Side or C-Side for 1-mymap.bin
, save the map bin next to it, and name it 1-mymap-B.bin
or 1-mymap-C.bin
.
To add dialogue, just edit Mods/yourmodname/Dialog/English.txt
and add more lines to it:
yourname_campaignname_dialogid=
[MADELINE left normal]
Yay it works!
Keep in mind that all your dialog IDs should start with yourname_campaignname_
to be sure they are unique.
You can also add more languages (for example, Mods/yourmod/Dialog/French.txt
). If a language does not exist, the game will use English instead.
For more details about how dialogue works, refer to Adding Custom Dialogue.
For a reference for adding custom portraits, see here
All custom graphics go to Mods/yourmodname/Graphics
. For example,
- if you want to add decals, drop them in
Mods/yourmod/Graphics/Atlases/Gameplay/decals/yournickname/yourcampaignname/decal.png
. The decal's dimensions should be multiples of 16. - if you want to add stylegrounds, drop them in
Mods/yourmod/Graphics/Atlases/Gameplay/bgs/yournickname/yourcampaignname/bg.png
- to add Everest custom NPC textures, drop the frames in
Mods/yourmod/Graphics/Atlases/Gameplay/characters/yournickname/yourcampaignname/animXX.png
then useyournickname/yourcampaignname/anim
in the NPC's properties. - to add a custom icon to your map, drop it in
Mods/yourmod/Graphics/Atlases/Gui/areas/yournickname/yourcampaignname/mymapicon.png
and set the "title banner icon" toareas/yournickname/yourcampaignname/mymapicon
in your map's metadata in Lönn.
You should always have your nickname and campaign name in the path to prevent conflicts.
If you are code modding, you can access those textures in code as well: check Adding Sprites for more details.
Check the Audio: How Tos to learn how to make a custom bank file.
Once you built a bank with your custom audio, drop the .bank and .guids.txt file to the Mods/yourmod/Audio
folder: Mods/yourmod/Audio/yournickname_yourcampaignname.bank
and .guids.txt
. Everest will load the bank, and you will be able to use events in it as background music in Lönn, or in code mods.
You can record a tutorial with, for example, Kayden's Commands 🔗. Open the debug console, type start_rec
, do the action to record, then type stop_rec
. You'll find the result in Content/Tutorials/CustomPlaybacks/latestCustomRecording.bin
.
Move this file to Mods/yourmod/Tutorials/yournickname/yourcampaignname/tutorial.bin
. Then, place a "player playback" and set its tutorial to yournickname/yourcampaignname/tutorial
. The ghost will now replay the action you recorded!
When making maps, you can use helper mods: you can find the list on GameBanana, under the Helpers category 🔗. Just install one of them as you would install any other mod and restart Lönn to access the entities/triggers it provides.
When you use something from a helper, you must add a dependency to this helper, to be sure that people that play your map also have it installed. To do that:
- open the helper's zip, and open its everest.yaml. You should see something like this:
- Name: FrostHelper
Version: 1.17.6
DLL: Code/FrostTempleHelper/bin/Debug/FrostTempleHelper.dll
Dependencies:
- Name: Everest
Version: 1.1375.0
- pick the two first
Name
andVersion
lines. - open your own everest.yaml, and add those two lines under
Dependencies
. It should now look like this:
- Name: YourModName
Version: 1.0.0
Dependencies:
- Name: Everest
Version: 1.1375.0
- Name: FrostHelper
Version: 1.17.6
Be sure that all lines in Dependencies are aligned correctly and add/remove spaces if this is not the case. YAML is very picky about that.
If you want to check that you added the dependency correctly, simply remove the helper zip from your Mods folder. When you start up the game, Everest should tell you it is missing, and offer you to install it in Mod Options.
To get the zip to be published on GameBanana, just go to Mods/yourmodname
, select everything and compress it (for example Send to > Compressed folder
on Windows, or 7-Zip > Add to yourmodname.zip
if you have 7-Zip installed). Be sure to make a zip archive (not a rar or a 7z one).
When you open your zip, you should see Dialog
, Maps
and everest.yaml
right away, without needing to open another folder.
If you did that correctly, publish the zip on GameBanana, and an "Olympus 1-click install" button should appear automatically! If it's not there right away, don't panic - it usually takes a few minutes for the button to show up.
Optional dependencies are helpful in two cases:
- You want your mod to load after another mod if it is installed, but you don't want to force the user to have that mod (like a regular dependency would). This is for example useful if you have a skin mod that reskins vanilla and the Spring Collab: you want your skin to load after the Spring Collab to override its textures, but you don't want to force the player to download the Spring Collab if they just want to play vanilla.
- You want to prevent your mod from loading if another mod is present, but outdated (useful in case you know both mods are incompatible).
For example, if you have a mod called TestMod123
and know it will crash if a Randomizer version lower than 1.4.0 is installed, you can write:
- Name: TestMod123
Version: 1.0.0
Dependencies:
- Name: Everest
Version: 1.2002.0
OptionalDependencies:
- Name: Randomizer
Version: 1.4.0
- If
Randomizer
1.4.0 is installed,TestMod123
will load right after it. - If
Randomizer
is not installed,TestMod123
will load anyway. - If
Randomizer
1.3.0 is installed,TestMod123
will not load. The main menu will display "1 mod failed to load", and the "Install missing dependencies" button will updateRandomizer
.
Optional Dependencies were added in Everest 2002.
Everest currently supports the following formats:
- Mod
.zip
s: Best used when using other's mods or when uploading your mod somewhere. - Mod subfolders: Best used for prototyping mods on your own machine.
A mod can't contain "submods", unless a code mod loads the "submods" dynamically.
All files in a mod .zip
must be on the top level (root) and visible when just opening the .zip
file. YourMod.zip
shouldn't contain another YourMod
folder.
A code mod can dynamically load external mod content and mod assemblies.
Dynamically loaded mods aren't restricted by the above formats, but rather by how the "supporting" mod loads them.
The following file layout still applies to dynamically loaded mods.
Mods can contain custom content, both overrides / replacements (when supported) and new content. The content paths match the originals as close as possible. Paths are case sensitive.
Mods must contain a everest.yaml
metadata file.
The following content mappings are supported out of the box:
-
Graphics/Atlases/*.png
: Replace or add textures.- Supports
.meta
definition files. -
Graphics/Atlases/Gui/title.png
- Atlas:
Gui
- Texture:
title
- Atlas:
-
Graphics/Atlases/Gameplay/scenery/sign.png
- Atlas:
Gameplay
- Texture:
scenery/sign
- Atlas:
- Supports
-
Maps/*.bin
: Add new maps.- Supports
.meta
definition files. - Replacing vanilla maps not supported.
-
Maps/Author/LevelSet/1-Name.bin
- Default SID (String ID):
Author/LevelSet/1-Name
- Default LevelSet:
Author/LevelSet
- Default Name:
1-Name
- Default SID (String ID):
- Supports
-
Dialog/*.txt
: Add new texts.- Only contains the texts belonging to the mod.
- Multiple mods are allowed to occupy the same path. F.e. two mods can contain their own
Dialog/English.txt
files. - Replacing existing texts not supported.
-
Dialog/English.txt
is always used when a dialog key cannot be found in another language.
-
Audio/*.bank
+Audio/*.guids.txt
: Add new FMOD audio banks containing custom / replacement events.- Works with new banks created in the Celeste FMOD project 🔗
- Requires manually exported
GUIDs.txt
, renamed toyour bank name.guids.txt
- To replace the game's events, assign any modified events to your new mod bank. Please avoid overriding the game's original banks.
With a few exceptions, each file path can only be occupied by one file.
- If mod #1 contains
Graphics/Atlases/Gui/title.png
andGraphics/Atlases/Gui/title.meta.yaml
, it overrides the texture shipped with Celeste. - If an additional mod #2 contains
Graphics/Atlases/Gui/title.png
, it overrides the texture shipped with #1. The meta definitions of #1 still apply.
All "supported" mod content can be loaded with the usual Celeste methods. For example, a texture in an atlas can be accessed with the matching atlas.
All "unsupported" mod content can be loaded by codemods via Everest.Content.Get*
(usage) 🔗.
For content inside of .zip
s and subfolders, all mod content is directly contained in the .zip
/ subfolder. For example, the mod .zip
should directly contain a Maps
folder. The Maps
folder should not be in a Content
subfolder.
For content in form of embedded resources inside of .dlls
, all mod content requires a Content\
path prefix Embedded resources normally don't have logical file paths in the traditional sense - the C# compiler mangles the filepath into something C#-friendly. To fix content embedded into .dlls, set a logical name for all Content\
<EmbeddedResource>
s in your .csproj
:
<EmbeddedResource Include="Content\Dialog\English.txt">
<LogicalName>Content\Dialog\English.txt</LogicalName>
</EmbeddedResource>
Additionally, for embedded resources, the C# compiler requires filepaths with \
as the folder separator, but the content is accessed with /
as the folder separator. Everest replaces all \
symbols with /
symbols in embedded resource paths at runtime.
The everest.yaml
file in your mod contains a list of all "module" names (ID), version, (optional) DLL paths and any dependencies.
If you've got no mod DLLs, leave that field out. If you've got multiple modules in separate DLLs, list them separately.
The version should match the semver (semantic versioning) format 🔗:
- The MAJOR version must match to prevent breakages caused by API changes (f.e. API removals).
- The MINOR version must rise with each backwards-compatible API change (f.e. API additions). If the mod depends on a new version but an older version is installed, the mod won't load.
- The PATCH version is also checked: if the mod depends on a new version but an older version is installed, the mod won't load.
Adding a dependency to a mod with version 0.0.*
ignores the above checks at your own risk.
File list:
everest.yaml
GhostMod.dll
GhostNetMod.dll
Embedded resources:
-
Content\Dialog\English.txt
: Mod option texts.
Metadata:
- Name: GhostMod
Version: 1.2.1
DLL: GhostMod.dll
Dependencies:
- Name: Everest
Version: 1.0.0
- Name: GhostNetMod
Version: 1.3.4
DLL: GhostNetMod.dll
Dependencies:
- Name: Everest
Version: 1.0.0
- Name: GhostMod
Version: 1.2.1
- The mod will load with Everest
1.0+.*
, meaning1.0.*
,1.1.*
,1.2.*
, ... - The mod won't load with Everest older than
1.0.0
. - The mod won't load with Everest
2.*.*
,3.*.*
,4.*.*
or similar. The API has changed in a way that could break the mod.
File list:
everest.yaml
-
Dialog/English.txt
: LevelSet and chapter names. -
Maps/Cruor/Mario11/Cruor-Secret.bin
: A-side map binary. -
Maps/Cruor/Mario11/Cruor-Secret.meta.yaml
: Chapter metadata. Always the.meta
of the A-side binary. -
Maps/Cruor/Mario11/Cruor-Secret-B.bin
: B-side map binary. The A-side / chapter.meta
links to this. -
Graphics/Atlases/Gui/areas/Cruor/Mario11/secret.png
,secret_back.png
: The chapter selection screen icon and its backside (when flipping). -
Graphics/Atlases/Gameplay/decals/Cruor/Mario11/*.png
: Any decals used by the A-side or B-side map.bin
s. -
Graphics/Atlases/Ending-Cruor-Secret-1/*.png
, ...: Chapter completion screen textures. The chapter.meta
links to this.
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