Skip to content

Adding a Character

Alchyr edited this page Mar 19, 2024 · 66 revisions

A character in Slay the Spire has a lot of separate parts which are all required for it to work.

To make things easier, you should download this set of example assets. A character needs a large number of images, and it's easier to add them all at once so you can adjust them to your liking later rather than to have you make them step-by-step during the tutorial.

The main goal of this tutorial is just for you to have a rough understanding of everything that's necessary, so code and resources are provided and explained, rather than stating what you'll need to make.

After downloading them, extract the files to resources/yourmod/images/character. Be sure you're putting the files into resources and not java.

image

It should look like this once you've added the files.

image

The Character Class

The first step for making your character is to make the file for your character's class. I recommend creating a new character package in your mod's package for it first. Make sure you're doing this in the java folder, not still in the resources folder.

image

The character class requires many methods to be filled out, most of which are the same every time and so like the images, this template file is provided for you to use. Download it and copy it into the package you created. If it doesn't show up immediately, you may need to right-click->Reload from Disk to make IntelliJ update the package's contents.

image

It should have a few errors; these are intentional. First, is the package. This may or may not be a problem, depending on how you added the files.

image

If you have this error, press Alt+Enter and select Set package name to fix it. The package should end up as the first line of the file, matching the location of it in the folder structure.

Next are some unresolved methods. These are defined in the main mod file, and so they aren't imported as the specifics would depend on the mod's package.

image

Again, use Alt+Enter and make sure you choose Import static method or Qualify static call. Either one will work fine. This should include the characterPath and makeID methods.

With this done, the character class itself should be error-free. We'll rename the class later, to make this setup easier (so the class name matches what's shown).

Contents

This is a quick review of the contents of the character class file.

    //Stats
    public static final int ENERGY_PER_TURN = 3;
    public static final int MAX_HP = 70;
    public static final int STARTING_GOLD = 99;
    public static final int CARD_DRAW = 5;
    public static final int ORB_SLOTS = 0;

For the majority of characters, max HP is the only value that is adjusted. These values are defined here for convenience and used later in the constructor of the character class.

    //Strings
    private static final String ID = makeID("CharacterID"); //This should match whatever you have in the CharacterStrings.json file
    private static final CharacterStrings characterStrings = CardCrawlGame.languagePack.getCharacterString(ID);
    private static final String[] NAMES = characterStrings.NAMES;
    private static final String[] TEXT = characterStrings.TEXT;

This is the text information required for the character, which is loaded from localization files. The first two lines perform that loading process.

If you look at the resources/localization/eng/CharacterStrings.json file, you should see

"${modID}:CharacterID": {

This is the identifier for the text that follows. Adjust "CharacterID" in both places to be whatever you'd like your character to be called. If you were making the Ironclad, you would have

private static final String ID = makeID("Ironclad"); in the character class and "${modID}:Ironclad": { in the json.

You should also adjust the rest of the contents of CharacterStrings.json.

    //Image file paths
    private static final String SHOULDER_1 = characterPath("shoulder.png"); //Shoulder 1 and 2 are used at rest sites.
    private static final String SHOULDER_2 = characterPath("shoulder2.png");
    private static final String CORPSE = characterPath("corpse.png"); //Corpse is when you die.

These are the file paths to some of the images used by the character. If you put the example assets in the correct folder, these shouldn't require any adjustments. You can Ctrl+Click the characterPath method to see how it creates the file path.

Enums

    public static class Enums {
        //These are used to identify your character, as well as your character's card color.
        //Library color is basically the same as card color, but you need both because that's how the game was made.
        @SpireEnum
        public static AbstractPlayer.PlayerClass YOUR_CHARACTER;
        @SpireEnum(name = "CHARACTER_GRAY_COLOR") // These two MUST match. Change it to something unique for your character.
        public static AbstractCard.CardColor CARD_COLOR;
        @SpireEnum(name = "CHARACTER_GRAY_COLOR") @SuppressWarnings("unused")
        public static CardLibrary.LibraryType LIBRARY_COLOR;
    }

These are enum values, representing your character's class and card color. Slay the Spire uses an enum to hold these, which normally it is not possible to add on to. @SpireEnum is a feature of ModTheSpire which adds onto these enums. Enum values are generally named using all capital letters. Note that these are just names; changing card appearance requires modifying their image files.

Change YOUR_CHARACTER and "CHARACTER_GRAY_COLOR" to unique identifications fitting for your character. As the comment mentions, you must change "CHARACTER_GRAY_COLOR" in both places, and they must be identical. CARD_COLOR and LIBRARY_COLOR don't need to be changed.

Constructor

For now, we'll skip over the constructor. It'll work without changing anything, so we'll come back to it when it's time to explain how to change how your character looks.

Methods

The methods are ordered based on how likely you are to need to change them, with the things you'll have to change at the top and the things you shouldn't change at the bottom.

The two really important ones are getStartingDeck() and getStartingRelic(). As their names suggest, these are used to define your character's starting deck and starting relic. For now, they're using some existing content as a placeholder. Once you make some cards and a starting relic, you'll have to adjust these. The rest are less important, and well commented. Feel free to read them, and adjust some of them if you want.

Card "Color"

Next you'll need to define the character's cards' color. This isn't an actual color, just a categorization to define how they look. Each card color is defined by a set of images used to render the card. Every card in Slay the Spire is connected to a "color", such as Colorless, Curse, and the colors for each individual character. This will be defined in the main mod file.

Images

The method you need to call is BaseMod.addColor. The method requires the file paths to a large number of images, so we'll start by defining those. Just as the character template defined some image file paths, you'll have to do the same. I recommend placing these file paths in your main mod file, since this is where you'll be using them.

As an example of how these are defined (you don't have to do it yourself): to define a constant value in Java, you define it as static final. We only need these in this file, so you can make it private. And finally, the type is String.

image

In the left sidebar, expand the resources folder and find resources/yourmod/images/character/cardback. This folder contains the images you need. For the file bg_attack.png, the full path would be yourmod/images/character/cardback/bg_attack.png. You don't have to worry about the resources/ part, it doesn't exist once your mod is packaged.

private static final String BG_ATTACK = "yourmod/images/character/cardback/bg_attack.png";

While you could include the full file path for every image, it's not really necessary since the paths for these images are all almost the same. That's why you use methods to shorten it, like in the character template. The characterPath method takes a String and adds on yourmod/images/character/, reducing the necessary repetition. So, you can just copy and paste this chunk of constants.

    private static final String BG_ATTACK = characterPath("cardback/bg_attack.png");
    private static final String BG_ATTACK_P = characterPath("cardback/bg_attack_p.png");
    private static final String BG_SKILL = characterPath("cardback/bg_skill.png");
    private static final String BG_SKILL_P = characterPath("cardback/bg_skill_p.png");
    private static final String BG_POWER = characterPath("cardback/bg_power.png");
    private static final String BG_POWER_P = characterPath("cardback/bg_power_p.png");
    private static final String ENERGY_ORB = characterPath("cardback/energy_orb.png");
    private static final String ENERGY_ORB_P = characterPath("cardback/energy_orb_p.png");
    private static final String SMALL_ORB = characterPath("cardback/small_orb.png");

This goes around the top of the main mod class (the one with @SpireInitializer) next to the other stuff defined with static at the top. These images are what determine the appearance of cards of this color (more details about this below).

    //Add the new stuff below these lines.
    private static final String resourcesFolder = checkResourcesPath();
    public static final Logger logger = LogManager.getLogger(modID); //Used to output to the console.

Actual Color

The last thing you need is a color, as in a red-green-blue kind of color. Pick a color that seems fitting for your character, and define it as such. Make sure you import the com.badlogic.gdx.graphics kind of Color, and not the java.awt or org.lwjgl.util Color. This is used for coloring a few things that depend on the character, such as the character's compendium tab, and card trails. This doesn't affect how the cards themselves look! For that, you have to modify the images used.

    private static final Color cardColor = new Color(128f/255f, 128f/255f, 128f/255f, 1f);
    //red, green, blue, alpha. alpha is transparency, which should just be 1.

The parameters are all numbers between 0 and 1. Most color pickers will give you a hex code and rgb (red/green/blue) values based on 255. This example divides by 255 to adjust those rgb values appropriately. The f after each number simply means that it's a float value and not an int value. If it was an int, it would perform integer division, where there are no decimals allowed, meaning you would either have a 0 or a 1, instead of a value somewhere in-between.

Calling the method

Go to your main mod file and find the initialize() method. This is where you'll actually be adding the color.

image

The first parameter is the CardColor enum. This is defined in the example character class, which is why you had to do that first.

image

The next parameter is the actual color that you defined, the cardColor variable. You can call the method with multiple colors to set a bunch of specific variables, but only one of them actually does anything, so just using one color is fine.

Following cardColor will be all the images. It should look like this:

        BaseMod.addColor(MyCharacter.Enums.CARD_COLOR, cardColor, 
                BG_ATTACK, BG_SKILL, BG_POWER, ENERGY_ORB, 
                BG_ATTACK_P, BG_SKILL_P, BG_POWER_P, ENERGY_ORB_P,
                SMALL_ORB);

First are the regular images, then the _P images, which are twice as large and used when viewing a single card in the detailed view (known as SingleCardView in code). Then there's SMALL_ORB, which is used to display the energy icon in card text, such as Gain [E] [E] .

With that, you've added the color. Including where it's defined, it should look similar to this (MyMod and MyCharacter are what should be different).

    public static void initialize() {
        new MyMod();

        BaseMod.addColor(MyCharacter.Enums.CARD_COLOR, cardColor,
                BG_ATTACK, BG_SKILL, BG_POWER, ENERGY_ORB,
                BG_ATTACK_P, BG_SKILL_P, BG_POWER_P, ENERGY_ORB_P,
                SMALL_ORB);
    }

Adding the Character

The last step in making your character functional is registering the character class. For this, you'll have to add a subscriber to your main mod file and provide a few more images.

First, you'll need to define the paths for those images. These are the character select button and portrait images, under character/select. These can go right next to all the card images in the main mod file.

    private static final String CHAR_SELECT_BUTTON = characterPath("select/button.png");
    private static final String CHAR_SELECT_PORTRAIT = characterPath("select/portrait.png");

Subscriber

Right up at the top of the class, after implements, add EditCharactersSubscriber. Don't forget the comma afterwards!

image

That'll leave you with the following error:

image

If you don't see it, click where the red lines are and hit Alt+Enter. Choose Implement methods and click Ok on the following window.

This will make an empty method.

    @Override
    public void receiveEditCharacters() {

    }

If you don't like where it is, cut it and paste it somewhere else (in the same file). In this method, you need to register your character using the method BaseMod.addCharacter. The necessary parameters are an instance of your class, the button image file path, the portrait image file path, and the character enum.

    @Override
    public void receiveEditCharacters() {
        BaseMod.addCharacter(new MyCharacter(), CHAR_SELECT_BUTTON, CHAR_SELECT_PORTRAIT);
    }

Your character should now be selectable in-game. Test it and make sure everything is working. You'll crash at the end of combat, however, as your character doesn't have enough cards yet.

If it does, congratulations! You've gotten through one of the harder steps of the process. Now that it's setup properly, feel free to right click your character class and choose Refactor->Rename to call it whatever you'd like.

Note that you will crash at the end of combats for now (more details below).

What Next

From here, all that's left is adding more content like cards, relics, and maybe some potions. And of course, all the details like the visuals (see below for more details on that). If you're feeling extra creative, maybe some events? It's up to you.

Standard characters have a pool of 75 cards, in a ratio of approximately 1:2:1 Common:Uncommon:Rare.

You'll need at least 3 common, uncommon, and rare cards to avoid card rewards freezing/crashing the game, and two attacks, two skills, and one power for shops to not crash. If you pick up question card you'd need 4 of each rarity. Until you fill out the card pool, expect crashes during testing. You can use Prismatic Shard as the starting relic if you want to avoid dealing with this right away.

They have 1 starter relic, and some number of relics spread out in the common, uncommon, and rare pools. All basegame characters have 1 common relic. Watcher has 2 uncommon relics, the others have 1. Ironclad and Silent have 3 rare relics; Watcher has 2, and Defect has 1. Characters generally have 3 boss relics. 1 starter relic upgrade (please make yours better than the basegame upgrades), 1 non-energy relic, and 1 energy relic. Watcher only has 2 boss relics.

Focus more on cards than relics; character specific relics (besides the starter) are relatively uncommon to encounter and don't harm the experience too much if they're missing. But they do add a bit of nice variety to the gameplay since they can interact directly with character mechanics, unlike generic relics.

The Things Not Required For Functionality

Visuals

Now that you have things working, you can worry a bit about making things look good.

In the constructor of your character class, there's a line labeled //orb and a line labeled //animation.

These are the big energy orb in the bottom left and the animation of your character respectively.

For the animation, you can view this page for details on what kind of animations you can use.

To just use a static image:

        super(NAMES[0], Enums.YOUR_CHARACTER,
                new CustomEnergyOrb(null, null, null), //Energy Orb
                new AbstractAnimation() { //Change the animation line to this.
                    @Override
                    public Type type() {
                        return Type.NONE; //A NONE animation results in img being used instead.
                    }
                });
        
        initializeClass(characterPath("image.png"),
                SHOULDER_2,
                SHOULDER_1,
                CORPSE,

This example loads resources/yourmod/images/character/image.png as the character image.

Orb

For the energy orb, there's much less framework.

CustomOrb is the default energy orb for custom players. You can give it an array of images along with numbers representing their rotation speed, and it'll render those images on top of each other spinning at those speeds. orbVfxPath is the file for an image that will expand out and fade when the player gains energy.

These images should be 128x128, except for the vfx image which should be 256x256. You can find an example of using this in DefaultMod: files, setup

You don't have to use CustomOrb, though. You can make your own class that implements EnergyOrbInterface and do anything you want with it. No detailed instructions here, since what you do at that point is up to your own creativity. Note that if you aren't using CustomOrb, you must override this method in your character class.

https://github.com/daviscook477/BaseMod/blob/1a8a344ecfde07613d29f39ff52b380cf50196f9/mod/src/main/java/basemod/abstracts/CustomPlayer.java#L156

Cards

The character's cards are displayed using the images in the cardback folder. To change how they look, you'll have to edit these images. energy_orb.png is the orb on the top left of the card where cost is displayed. energy_orb_p.png is the same, but when viewing a single card in the larger view. small_orb.png is used in card text to represent energy. The various bg_ images are the backgrounds for the various card types, with the _p versions for portrait (single card) view. Note that all cards other than attacks and powers use the skill background (curses, statuses).

Colors

Of the colors defined in the character class, cardRenderColor, cardTrailColor, and slashAttackColor, cardTrailColor is the most visible one. But you should probably change all three regardless.

Energy Font

If you want an example of loading and using a custom energy font, you can look here.

https://github.com/Alchyr/sumireko/blob/master/src/main/java/sumireko/patches/EnergyFontGen.java

https://github.com/Alchyr/sumireko/blob/master/src/main/java/sumireko/character/Sumireko.java#L220

Heart Kill Cutscene

To change this from the default Ironclad cutscene, override the getCutscenePanels method in your player class.

An example. The glow_fade can be ignored, as that is specific to this example.

It should return a list of CutscenePanel, each of which has one image and an optional sound effect to play when that image appears. The images should be 1920x1200, placed to fit within the cutscene border. Example images

If you've read all the comments and still have questions, go to the #modding-technical channel in the Slay the Spire discord.

Clone this wiki locally