-
-
Notifications
You must be signed in to change notification settings - Fork 24
Home
If you want to have your card display a custom card cost in either Act 1 (Leshy's Cabin) or Act 2 (pixel cards), you can simply hook into one of the following events:
using InscryptionCommunityPatch.Card;
using InscryptionAPI.Helpers;
Part1CardCostRender.UpdateCardCost += delegate(CardInfo card, List<Texture2D> costs)
{
int myCustomCost = card.GetExtensionPropertyAsInt("myCustomCardCost");
if (myCustomCost > 0)
costs.Add(TextureHelper.GetImageAsTexture($"custom_cost_{myCustomCost}.png"));
}
Card costs for Act 1 cards must be exactly 64x28 pixels. Card costs for Pixel cards (GBC) must be exactly 30x8 pixels.
With the API installed, the energy management drone can be made available in Act 1 and in Kaycee's Mod. It will appear automatically if any cards with an energy or gem cost are in the Act 1 card pool, and can be forced to appear by modifying the configuration for the API.
You can also force these drones to appear in different areas of the game by overriding the following values.
using InscryptionCommunityPatch.ResourceManagers;
EnergyDrone.ZoneConfigs[CardTemple.Nature].ConfigEnergy = true; // Enables energy
EnergyDrone.ZoneConfigs[CardTemple.Nature].ConfigDrone = true; // Makes the drone appear
EnergyDrone.ZoneConfigs[CardTemple.Nature].ConfigMox = true; // Enables moxen management
EnergyDrone.ZoneConfigs[CardTemple.Nature].ConfigDroneMox = true; // Makes the mox drone appear
At the time this README was written, the only zones where these settings will have any effect are CardTemple.Nature (Leshy's cabin) and CardTemple.Undead (Grimora's cabin).
Cards in Act 1 can now display multiple costs at the same time, and can display gem and energy cost in addition to blood and bones cost.
Cards in Act 2 can now display multiple costs at the same time.
The base game uses a number of hard-coded lists, called 'Enumerations' or 'Enums,' to manage behaviors. For example, the ability "Brittle" is assigned to a card using the enumerated value Ability.Brittle. We can expand these lists, but it requires care, and it is managed by the GuidManager class. This handles the creation of new enumerations and making sure those are handled consistently across mods.
Lets say that you want to create a new story event. These are managed by the enumeration StoryEvent. To create a new story event, you should use this pattern to create a single static reference to that new value:
public static readonly StoryEvent MyEvent = GuidManager.GetEnumValue<StoryEvent>(MyPlugin.guid, "MyNewStoryEvent");
GuidManager requires you to give it the guid of your plugin as well as a friendly name for the value you want to create (the plugin guid is required to avoid any issues if multiple mods try to create a new value with the same name).
If you want to get a value that was created by another mod (for example: you want to make a card that uses an ability created by another mod), you can follow this exact same pattern. You just need to know the plugin guid for the mod that it is contained in:
public static readonly Ability OtherAbility = GuidManager.GetEnumValue<Ability>("other.mod.plugin.guid", "Ability Name");
All of these values are stored in the modded save file.
If your mod needs to save data, the ModdedSaveManager class is here to help. There are two chunks of extra save data that you can access here: 'SaveData' (which persists across runs) and 'RunState' (which is reset on every run). Note that these require you to pass in a GUID, which should be your mod's plugin GUID, and an arbitrary key, which you can select for each property to you want to save.
The easiest way to use these helpers is to map them behind static properties, like so:
public static int NumberOfItems
{
get { return ModdedSaveManager.SaveData.GetValueAsInt(Plugin.PluginGuid, "NumberOfItems"); }
set { ModdedSaveManager.SaveData.SetValue(Plugin.PluginGuid, "NumberOfItems", value); }
}
When written like this, the static property "NumberOfItems" now automatically syncs to the save file.
Card management is handled through InscryptionAPI.Card.CardManager. You can simply call CardManager.Add with a CardInfo object and that card will immediately be added to the card pool:
CardInfo myCard = ...;
CardManager.Add(myCard); // Boom: done
You can create CardInfo objects however you want. However, there are some helper methods available to simplify this process for you. The most import of these is CardManager.New(name, displayName, attack, health, optional description) which creates a new card and adds it for you automatically:
CardInfo myCard = CardManager.New("example_card", "Sample Card", 2, 2, "This is just a sample card");
From here, you can modify the card as you wish, and the changes will stay synced with the game:
CardInfo myCard = CardManager.New("example_card", "Sample Card", 2, 2, "This is just a sample card");
myCard.cost = 2;
However, there are also a number of extension methods you can chain together to perform a number of common tasks when creating a new card. Here is an example of them in action, followed by a full list:
CardInfo myCard = CardManager.New("example_card", "Sample Card", 2, 2, "This is just a sample card")
.SetDefaultPart1Card()
.SetCost(bloodCost: 1, bonesCost: 2)
.SetPortrait("/art/sample_portrait.png", "/art/sample_emissive_portrait.png")
.AddAbilities(Ability.BuffEnemies, Ability.TailOnHit)
.SetTail("Geck", "/art/sample_lost_tail.png")
.SetRare();
The following card extensions are available:
- SetPortrait: Assigns the cards primary art, and optionally its emissive portrait as well. You can supply Texture2D directly, or supply a path to the card's art.
- SetEmissivePortrait: If a card already has a portrait and you just want to modify it's emissive portrait, you can use this. Note that this will throw an exception if the card does not have a portrait already.
- SetAltPortrait: Assigns the card's alternate portrait
- SetPixelPortrait: Assigns the card's pixel portrait (for GBC mode)
- SetCost: Sets the cost for the card
- SetDefaultPart1Card: Sets all of the metadata necessary to make this card playable in Part 1 (Leshy's cabin).
- SetGBCPlayable: Sets all of the metadata necessary to make this card playable in Part 2.
- SetDefaultPart3Card: Sets all of the metadata necessary to make this card playable in Part 3 (P03's cabin).
- SetRare: Sets all of the metadata ncessary to make this card look and play as Rare.
- SetTerrain: Sets all of the metadata necessary to make this card look and play as terrain
- SetTail: Creates tail parameters. Note that you must also add the TailOnHit ability for this to do anything
- SetIceCube: Creates ice cube parameters. Note that you must also add the IceCube ability for this to do anything
- SetEvolve: Creates evolve parameters. Note that you must also add the Evolve ability for this to do anything
- AddAbilities: Add any number of abilities to the card. This will add duplicates.
- AddAppearances: Add any number of appearance behaviors to the card. No duplicates will be added.
- AddMetaCategories: Add any number of metacategories to the card. No duplicates will be added.
- AddTraits: Add any number of traits to the card. No duplicates will be added.
- AddTribes: Add any number of tribes to the card. No duplicates will be added.
- AddSpecialAbilities: Add any number of special abilities to the card. No duplicates will be added.
It's possible that at the time your card is built, the card that you want to evolve into has not been built yet. You can use the event handler to delay building the evolve/icecube/tail parameters of your card, or you can use the extension methods above which will handle that for you.
However, note that if you use these extension methods to build these parameters, and the card does not exist yet, the parameters will come back null. You will not see the evolve parameters until you add the evolution to the card list and you get a fresh copy of the card from CardLoader (as would happen in game).
CardManager.New("Example", "Base", "Base Card", 2, 2).SetEvolve("Evolve Card", 1); // "Evolve Card" hasn't been built yet
Plugin.Log.LogInfo(CardLoader.GetCardByName("Example_Base").evolveParams == null); // TRUE!
CardInfo myEvolveCard = CardManager.New("Example", "Evolve", "Evolve Card", 2, 5);
Plugin.Log.LogInfo(CardLoader.GetCardByName("Example_Base").evolveParams == null); // FALSE!
If you want to edit a card that comes with the base game, you can simply find that card in the BaseGameCards list in CardManager, then edit it directly:
CardInfo card = CardManager.BaseGameCards.CardByName("Porcupine");
card.AddTraits(Trait.KillsSurvivors);
There is also an advanced editing pattern that you can use to not only edit base game cards, but also potentially edit cards that might be added by other mods. To do this, you will add an event handler to the CardManager.ModifyCardList event. This handler must accept a list of CardInfo objects and return a list of CardInfo objects. In that handlers, look for the cards you want to modify and modify them there.
In this example, we want to make all cards that have either the Touch of Death or Sharp Quills ability to also gain the trait "Kills Survivors":
CardManager.ModifyCardList += delegate(List<CardInfo> cards)
{
foreach (CardInfo card in cards.Where(c => c.HasAbility(Ability.Sharp) || c.HasAbility(Ability.Deathtouch)))
card.AddTraits(Trait.KillsSurvivors);
return cards;
};
By doing this, you can ensure that not on all of the base game cards get modified, but also all other cards added by other mods.
The API allows you to add custom properties to a card, and then retrieve them for use inside of abilities. In the same way that you can use Evolve parameters to make the evolve ability work, or the Ice Cube parameters to make the IceCube ability work, this can allow you to set custom parameters to make your custom abilities work.
CardInfo sample = CardLoader.CardByName("MyCustomCard");
sample.SetExtendedProperty("CustomPropertyName", "CustomPropertyValue");
string propValue = sample.GetExtendedProperty("CustomPropertyName");
Abilities are unfortunately a little more difficult to manage than cards. First of all, they have an attached 'AbilityBehaviour' type which you must implement. Second, the texture for the ability is not actually stored on the AbilityInfo object itself; it is managed separately (bizarrely, the pixel ability icon is on the AbilityInfo object, but we won't get into all that).
Regardless, the API will help you manage all of this with the helpful AbilityManager class. Simply create an AbilityInfo object, and then call AbilityManager.Add with that info object, the texture for the icon, and the type that implements the ability.
Abilities inherit from DiskCardGame.AbilityBehaviour
AbilityInfo myinfo = ...;
Texture2D myTexture = ...;
AbilityManager.Add(MyPlugin.guid, myInfo, typeof(MyAbilityType), myTexture);
You can also use the AbilityManager.New method to simplify some of this process:
AbilityInfo myInfo = AbilityManager.New(MyPlugin.guid, "Ability Name", "Ability Description", typeof(MyAbilityType), "/art/my_icon.png");
And there are also extension methods to help you here as well:
AbilityManager.New(MyPlugin.guid, "Ability Name", "Ability Description", typeof(MyAbilityType), "/art/my_icon.png")
.SetDefaultPart1Ability()
.SetPixelIcon("/art/my_pixel_icon.png");
- SetCustomFlippedTexture: Use this to give the ability a custom texture when it belongs to the opponent.
- SetPixelIcon: Use this to set the texture used when rendered as a GBC card.
- AddMetaCategories: Adds any number of metacategories to the ability. Will not duplicate.
- SetDefaultPart1Ability: Makes this appear in the part 1 rulebook and randomly appear on mid-tier trader cards and totems.
- SetDefaultPart3Ability: Makes this appear in the part 3 rulebook and be valid for create-a-card (make sure your power level is accurate here!)
Abilities require an instance of AbilityInfo that contains the information about the ability, but they also require you to write your own class that inherits from AbilityBehaviour and describes how the ability functions.
AbilityBehaviour contains a lot of virtual methods. For each event that can happen during a battle, there will be a 'RespondsToXXX' and an 'OnXXX' method that you need to override. The purpose of the 'RespondsToXXX' is to indicate if your ability cares about that event - you must return True in that method for the ability to fire. Then, to actually make the ability function, you need to implement your custom behavior in the 'OnXXX' method.
See this example from the base game:
public class Sharp : AbilityBehaviour
{
public override Ability Ability => Ability.Sharp;
public override bool RespondsToTakeDamage(PlayableCard source) => source != null && source.Health > 0;
public override IEnumerator OnTakeDamage(PlayableCard source)
{
yield return base.PreSuccessfulTriggerSequence();
base.Card.Anim.StrongNegationEffect();
yield return new WaitForSeconds(0.55f);
yield return source.TakeDamage(1, base.Card);
yield return base.LearnAbility(0.4f);
yield break;
}
}
There are two specific common use cases for abilities that are not given to you by the standard AbilityBehaviour class. Fortunately, this API comes with an ExtendedAbilityBehaviour class that will allow you to do the following:
To do this, you need to override RespondsToGetOpposingSlots to return true (like all RespondsToXXX overrides, you can make this conditional), and then override GetOpposingSlots to return the list of card slots that your ability wants the card to attack. If you want to override the default slot (the one directly across from the card) instead of adding an additional card slot, you need to override RemoveDefaultAttackSlot to return true.
To do this, you need to override GetPassiveAttackBuff(PlayableCard target) or GetPassiveAttackBuff(PlayableCard target) to calculate the appropriate buffs. These return an int representing the buff to give to 'target'.
In battle, the game will iterate across all cards on the board and check whether they should receive the buffs; this is what 'target' refers to; the current card being checked. You will need to write the logic for determining what cards should get the buff, as well as what buff they should receive.
Note: you need to be very careful about how complicated the logic is in GetPassiveAttackBuffs and GetPassiveHealthBuffs. These methods will be called every frame for every instance of the ability!! If you're not careful, you could bog the game down substantially!
ExtendedAbilityBehaviour also allows you to control whether the ability's triggers will activate when the card is facedown. You will need to override TriggerWhenFaceDown to return true. There are also 2 other bools you can override for more control over what triggers should activate: ShouldTriggerWhenFaceDown, which controls whether vanilla triggers will activate; and ShouldTriggerCustomWhenFaceDown, which control whether custom triggers will activate.
It should be noted that usage of ExtendedAbilityBehaviour is not required; all the functionalities mentioned can also be accessed by inheriting from the appropriate interface: IGetOpposingSlots, IActivateWhenFacedown, IPassiveAttackBuff, and IPassiveHealthBuff.
Think of these like abilities for your stats (like the Ant power or Bell power from the vanilla game). You need to create a StatIconInfo object and build a type that inherits from VariableStatBehaviour in order to implement a special stat. By now, the pattern used by this API should be apparent.
Special stat icons inherit from DiskCardGame.VariableStatBehaviour
StatIconInfo myinfo = ...;
SpecialStatIconManager.Add(MyPlugin.guid, myInfo, typeof(MyStatBehaviour));
or
SpecialStatIconManager.Add(MyPlugin.guid, "Stat", "My special stat", typeof(MyStatBehaviour))
.SetIcon("/art/special_stat_icon.png")
.SetPixelIcon("/art/special_pixel_stat_icon.png")
.SetDefaultPart1Ability()
.SetDefaultPart3Ability();
Because StatIconInfo is so simple, there aren't very many helpers for it.
Stat icons require an instance of StatIconInfo that contains the information about the ability, but they also require you to write your own class that inherits from VariableStatBehaviour and describes how the ability functions.
When you implement a variable stat behavior, you need to implement the abstract method GetStateValues. This method returns an array of integers: the value at index 0 is the variable attack power, and the value at index 1 is the variable health.
Here is an example from the base game:
public class BellProximity : VariableStatBehaviour
{
protected override SpecialStatIcon IconType => SpecialStatIcon.Bell;
protected override int[] GetStatValues()
{
int num = BoardManager.Instance.PlayerSlotsCopy.Count - base.PlayableCard.Slot.Index;
int[] array = new int[2];
array[0] = num;
return array;
}
}
Note: you need to be very careful about how complicated the logic is in GetStatValues. This will be called every frame!! If you're not careful, you could bog the game down substantially.
Special triggered abilities are a lot like regular abilities; however, they are 'invisible' to the player (that is, they do not have icons or rulebook entries). As such, the API for these is very simple. You simply need to provide your plugin guid, the name of the ability, and the type implementing the ability, and you will be given back a wrapper containing the ID of your newly created special triggered ability.
Special triggered abilities inherit from DiskCardGame.SpecialCardBehaviour
public readonly static SpecialTriggeredAbility MyAbilityID = SpecialTriggeredAbilityManager.Add(MyPlugin.guid, "Special Ability", typeof(MySpecialTriggeredAbility)).Id;
And now MyAbilityID can be added to CardInfo objects.
Special abilities are the same as regular abilities, except they do not have a metadata object associated with them (because they are not described or documented for the player) and the inherit from SpecialCardBehaviour instead of AbilityBehaviour.
Here is an example from the base game:
public class TrapSpawner : SpecialCardBehaviour
{
public override bool RespondsToDie(bool wasSacrifice, PlayableCard killer) => base.PlayableCard.OnBoard;
public override IEnumerator OnDie(bool wasSacrifice, PlayableCard killer)
{
yield return new WaitForSeconds(0.35f);
yield return BoardManager.Instance.CreateCardInSlot(CardLoader.GetCardByName("Trap"), base.PlayableCard.Slot, 0.1f, true);
yield return new WaitForSeconds(0.35f);
yield break;
}
}
These behave the same as special triggered abilities from the perspective of the API.
Special triggered abilities inherit from DiskCardGame.CardAppearanceBehaviour
public readonly static CardAppearanceBehaviour.Appearance MyAppearanceID = CardAppearanceBehaviourManager.Add(MyPlugin.guid, "Special Appearance", typeof(MyAppearanceBehaviour)).Id;
Appearance behaviours implement CardAppearanceBehaviour. There is an abstract method called ApplyAppearance that you must implement - here you override the default appearance of the card. There are also three other virtual methods: ResetAppearance, OnCardAddedToDeck, and OnPreRenderCard that give other hooks in which you can change the card's appearance.
Here is an example from the base game:
public class RedEmission : CardAppearanceBehaviour
{
public override void ApplyAppearance()
{
base.Card.RenderInfo.forceEmissivePortrait = true;
base.Card.StatsLayer.SetEmissionColor(GameColors.Instance.glowRed);
}
public override void ResetAppearance()
{
base.Card.RenderInfo.forceEmissivePortrait = false;
base.Card.StatsLayer.SetEmissionColor(GameColors.Instance.glowSeafoam);
}
}
Unlike abilities, encounters are encoded into the game's data through a combination of enumerations and strings. For example, Opponents are enumerated by the enumeration Opponent.Type, but special sequencers and unique AI rules are represented as strings buried in each encounters data.
To create a custom encounter (for example, a custom boss fight), you will need some combination of opponents, special sequencers, and AI.
The card(s) that will be used in the encounter.
public static readonly EncounterBlueprintData.CardBlueprint bp_Bonehound = new EncounterBlueprintData.CardBlueprint
{
card = CardSpawner.GetCardByName("Bonehound")
};
public static readonly EncounterBlueprintData.CardBlueprint bp_Bonepile = new EncounterBlueprintData.CardBlueprint
{
card = CardSpawner.GetCardByName("Bonepile")
};
For setting up the EncounterBlueprintData.
- In this example, the card Bonehound will spawn twice on the first turn. The slot that they spawn in is randomized.
- Second and Third turns nothing spawns.
- Fourth and fifth turns a Bonepile spawns.
- No spawns on sixth turn.
- Seventh turn a another Bonepile spawn.
public static EncounterBlueprintData BuildBlueprintTwo()
{
var blueprint = ScriptableObject.CreateInstance<EncounterBlueprintData>();
blueprint.name = "TurnPlan_2";
blueprint.turns = new List<List<EncounterBlueprintData.CardBlueprint>>
{
new List<EncounterBlueprintData.CardBlueprint> { bp_Bonehound, bp_Bonehound },
new List<EncounterBlueprintData.CardBlueprint> { },
new List<EncounterBlueprintData.CardBlueprint>(),
new List<EncounterBlueprintData.CardBlueprint> { bp_Bonepile },
new List<EncounterBlueprintData.CardBlueprint> { bp_Bonepile },
new List<EncounterBlueprintData.CardBlueprint>(),
new List<EncounterBlueprintData.CardBlueprint> { bp_Bonepile }
};
return blueprint;
}
Now that you have your encounter blueprint setup, add it with the manager:
EncounterManager.Add(BuildBlueprintTwo());
Special sequencers are essentially 'global abilities;' they listen to the same triggers that cards do and can execute code based on these triggers (such as whenever cards are played or die, at the start of each turn, etc).
While you can inherit directly from SpecialBattleSequencer, there is a hierarchy of existing special sequencers that you may wish to inherit from depending upon your use case. You can use dnSpy to see all of them, but these are three you should be specifically aware of:
- SpecialBattleSequencer: The base class for all special sequencers. Use this by default.
- BossBattleSequencer: Used for boss battles
- Part1BossBattleSequencer: Use for boss battles in Act 1 (Angler, Prospector, Trapper/Trader, and Leshy)
Special sequencers are set directly on the NodeData instance corresponding to the map node that the player touches to start the battle. This is done using a string value corresponding to the sequencer. To set this yourself, use the SpecialSequenceManager class in the API:
public class MyCustomBattleSequencer : Part1BossBattleSequencer
{
public static readonly string ID = SpecialSequenceManager.Add(Plugin.PluginGuid, "MySequencer", typeof(MyCustomBattleSequencer));
}
In the context of this API, think of opponents as basically just bosses. There is a enumeration of Opponents called Opponent.Type that refers to the opponent. This name can be confusing in some IDEs, as they may simply show the parameter type as Type, which should not be confused with the System.Type type.
Like everything else, Opponents are classes that you have to write that are responsible for handling the weird things that happen during battle. Depending upon which type of opponent you are buiding, you will need to use one of the following base classes:
- Opponent: The base class for all Opponents. You should probably never inherit from this one directly.
- Part1Opponent: Used for battles in Leshy's cabin
- Part1BossOpponent: Used for boss battles in Leshy's cabin
- PixelOpponent: Used for battles in the GBC game
- PixelBossOpponent: Used for boss battles in the GBC game
- Part3Opponent: Used for battles in Botopia
- Part3BossOpponent: Used for boss battles in Botopia
Custom opponents will need a custom Special Sequencer as a companion (please, don't ask why we need Special Sequencers and Opponents and why they couldn't just be the same thing). So in this case, the following code snippet shows how to create an opponent that is linked to its special sequencer:
public class MyBossOpponent : Part1BossOpponent
{
public static readonly Opponent.Type ID = OpponentManager.Add(Plugin.PluginGuid, "MyBossOpponent", MyCustomBattleSequencer.ID, typeof(MyBossOpponent)).Id;
}
Note that the third parameter in OpponentManager.Add is a string, and that string is the same ID that was set in the code snippet for the special sequencer in the previous example.
If you have a specific blueprint that you want your opponent type to use, using the example for setting up the blueprint from before, then in in the Awake
, Start
, or IntroSequence
override you can set it there if you didn't already set it in the EncounterData object:
public override IEnumerator IntroSequence(EncounterData encounter)
{
encounterData.Blueprint = EncounterManager.AllEncountersCopy.Find(enc => enc.name == "TurnPlan_2");
List<List<CardInfo>> plan = EncounterBuilder.BuildOpponentTurnPlan(encounterData.Blueprint, difficulty, removeLockedCards);
base.ReplaceAndAppendTurnPlan(plan);
yield return QueueNewCards();
yield return base.IntroSequence(encounter);
}
In most cases, you will probably not need to create a custom AI. By default, the game will look at the cards that the computer is preparing to play and then use brute force to test every possible slot that the computer could queue those slots for. It will then simulate a full turn of the game for each of those positions and determine which one has the best outcome. There are very few exceptions to this rule.
As an example, one exception is the Prospector boss fight, where the computer will always play the Pack Mule in slot 0 and a coyote in slot 1. It's important that these cards always get played in this position, so the game has a custom AI that detects this part of the game and overrides the default AI.
If you want to use a custom AI, you need to build a class that inherits from DiskCardGame.AI and overrides the virtual method SelectSlotsForCards. That one method is where you implement your custom AI logic.
To register a new AI with the game, use the AIManager class in the API, and keep a reference to the string ID that you receive back:
public class MyCustomAI : AI
{
public static readonly string ID = AIManager.Add(Plugin.PluginGuid, "MyAI", typeof(MyCustomAI)).Id;
public override List<CardSlot> SelectSlotsForCards(List<CardInfo> cards, CardSlot[] slots)
{
// Do stuff here
}
}
To actually use your new AI, you need to set it inside of a special sequencer; specifically in the BuildCustomEncounter method:
public class MyCustomBattleSequencer : Part1BossBattleSequencer
{
public override EncounterData BuildCustomEncounter(CardBattleNodeData nodeData)
{
EncounterData data = base.BuildCustomEncounter(nodeData);
data.aiId = MyCustomAI.ID;
return data;
}
}
If you want to add a new node to the game map, you need to be prepared to implement potentially complex game logic inside of a "sequencer" class. Whenever the game moves to a map node, it hands control of the game over to a sequencer, which is responsible for creating game objects, animating objects, manipulating the deck, or whatever else it is that you want your custom node to do. Once done, the sequencer hands control of the game back over the main game loop. This sequencer is where you will be spending most of your time programming, testing, and debugging. It is highly recommended that you check out the example mod and/or decompile the base game's sequencers to get a feel for how to do this.
Once you have created a sequencer for your event, you also need to create an animation for your node to appear on the map. This animation is an array of precisely four textures, each of which should be 49x49 pixels in size. You need to register the sequencer and the associated texture array with the API. This is done using the NodeManager.Add generic method:
NodeManager.Add<MyCustomSequencer>(
new Texture2D[] {
TextureHelper.GetImageAsTexture("animated_node_1.png", typeof(MyCustomSequencer).Assembly),
TextureHelper.GetImageAsTexture("animated_node_2.png", typeof(MyCustomSequencer).Assembly),
TextureHelper.GetImageAsTexture("animated_node_3.png", typeof(MyCustomSequencer).Assembly),
TextureHelper.GetImageAsTexture("animated_node_4.png", typeof(MyCustomSequencer).Assembly)
},
NodeManager.NodePosition.SpecialEventRandom
);
Note the final parameter to "Add" is a flag that tells the API where the node should appear. The NodeManager.NodePosition enumeration is a flag-style enumeration, which means that multiple values can be combined using the bitwise-or operator (|). So, for example, if you want the node to appear randomly throughout the map but also to be forced to appear directly before the boss, you can use this combination:
NodeManager.NodePosition.SpecialEventRandom | NodeManager.NodePosition.PreBoss
The values of NodePosition are:
- NotGenerated: This node will not be autogenerated on the map, but can still be manually added (for example, if you are manually manipulating Act 3 maps).
- Act1Available: This node can appear in Act 1 maps, not just in Kaycee's Mod maps. Without this set, the custom node will only ever appear in a Kaycee's Mod run of Act 1.
- MapStart: This node will be forced to appear at the start of each map, unless the node fails a prerequisite condition.
- CardChoiceRandom: This node will appear in the pool of random events that happen right after a battle.
- SpecialEventRandom: This node will appear in the pool of random events that happen right before a battle.
- PreBoss: This node will be forced to appear right before the boss of each map, unless the node fails a prerequisite condition
- PostBoss: This node will be forced to appear right after the boss of each map, unless the node fails a prerequisite condition
Map nodes can implement logic to determine whether or not they should appear. To do this, you need to implement a custom 'node data' object which inherits from CustomNodeData. There is a virtual method called Initialize that you should override. Here, you can add prerequisite conditions and forced generation conditions; the former will prevent a node from generating if any condition returns false, and the latter will force the node to generate if any condition returns true.
You can add two types of conditions: simple, or map-aware. A simple condition takes no parameters and returns true or false based on whatever parameters you like (for example, if a certain ascension challenge is active or not). A map-aware condition takes two parameters: the first is the y-value of the node's position, and the second is a list of all map nodes that have previously been generated on this instnace of the map.
Here is a code sample:
public class MySpecialNodeData : CustomNodeData
{
public override void Initialize()
{
// This node can only generate if the BaseDifficulty challenge is active
this.AddGenerationPrerequisite(() => AscensionSaveData.Data.ChallengeIsActive(AscensionChallenge.BaseDifficulty));
// This node is forced to generate if more than one BaseDifficulty challenge is active
// and a deck trial node has been placed on the map already
this.AddForceGenerationCondition((y, nodes) => AscensionSaveData.Data.GetNumChallengesOfTypeActive(AscensionChallenge.BaseDifficulty) > 1
&& nodes.Exists(n => n is DeckTrialNodeData));
}
}
If you are using your own node data, you have to register it at the same time as your custom sequencer:
NodeManager.Add<MyCustomSequencer, MySpecialNodeData>(
new Texture2D[] {
TextureHelper.GetImageAsTexture("animated_node_1.png", typeof(MyCustomSequencer).Assembly),
TextureHelper.GetImageAsTexture("animated_node_2.png", typeof(MyCustomSequencer).Assembly),
TextureHelper.GetImageAsTexture("animated_node_3.png", typeof(MyCustomSequencer).Assembly),
TextureHelper.GetImageAsTexture("animated_node_4.png", typeof(MyCustomSequencer).Assembly)
},
NodeManager.NodePosition.SpecialEventRandom
);
The API supports adding new Challenges as part of Kaycee's Mod using the ChallengeManager class. You can add a new Challenge using ChallengeManager.Add, either by passing in a new AscensionChallengeInfo object or by passing the individual properties of your challenge (which will construct the information object for you). This will make your challenge automatically appear in the challenge selection screen.
If you use the overload of Add that takes an AscensionChallengeInfo object, note that the "challengeType" field of type AscensionChallenge is completely irrelevant. This is an enumerated value, and it will be set for you by the ChallengeManager to ensure there is no collision with other challenges created by other mods. As such, you need to save the ID that is returned by the Add method of ChallengeManager so that you can reference it later.
If your challenge can stack, set the stackable flag to 'true' when adding your challenge. This will cause it to appear twice in the challenge selection screen. Yes, this means stackable challenges are currently limited to a max of two.
You will be responsible to write all of the necessary patches for your challenge to function. When writing those patches, you must make sure that the challenge is active before you do anything. This is entirely up to you - there is nothing in the API that can detect this for you.
You should also make sure to alert the user whenever your challenge has triggered some change in the game. The ChallengeActivationUI class (from DiskCardGame) has a helper to alert the user.
private static AscensionChallenge ID = ChallengeManager.Add
(
Plugin.PluginGuid,
"Dummy Challenge",
"This challenge doesn't do anything",
15,
Resources.Load<Texture2D>("Path/To/Texture")
);
[HarmonyPatch(...)]
public static void DoSomething()
{
if (AscensionSaveData.Data.ChallengeIsActive(ID))
{
ChallengeActivationUI.Instance.ShowActivation(ID);
// Do some actual stuff
}
}
Starter decks are relatively simple. They simply need a title, an icon, and a set of three cards. You can also optionally add an "unlock level" which prevents your starter deck from being unlocked until the player reaches a certain challenge level. This defaults to 0, which means that the starter deck will always be unlocked.
What if you want to make a starter deck with cards from another mod? What if you aren't 100% sure that those cards are loaded at the time that your starter deck is loaded? You can solve this manually by creating a dependency, which forces BepInEx to load the other mod first. However, if that mod uses JSON Loader, you need to create a dependency on JSON Loader instead, which could (in some very bizarre situations) create a situation where you both need to load before and after JSON Loader. Rather than sort out all of these possible scenarios one-by-one, you can instead create a starter deck with a set of three card names (strings) instead of three CardInfo objects. This will create a "delayed loading" scenario where the actual starter deck info won't be built until right before the Starter Deck screen loads.
Both patterns are shown here:
StarterDeckInfo myDeck = ScriptableObject.CreateInstance<StarterDeckInfo>();
myDeck.title = "Pelts";
myDeck.iconSprite = TextureHelper.GetImageAsSprite("art/pelts_deck_icon.png", TextureHelper.SpriteType.StarterDeckIcon);
myDeck.cards = new () { CardLoader.GetCardByName("PeltWolf"), CardLoader.GetCardByName("PeltHare"), CardLoader.GetCardByName("PeltHare") };
StarterDeckManager.Add(MyPlugin.Guid, myDeck);
StarterDeckManager.New(MyPlugin.Guid, "Pelts", "art/pelts_deck_icon.png", new string[] { "PeltWolf", "PeltHare", "PeltHare"});
This API supports adding new screens to Kaycee's Mod that execute before a run starts. New screens can use the AscensionScreenSort attribute to influence their sort order. Custom screens will execute in the following order:
- Starter Decks
- Select Challenges
- Custom Screens
- Requires Start
- Prefers Start
- No Preference (default)
- Prefers End
- Requires End
- Start Run
To create a custom screen, you need to write a special screen behavior class that inherits from AscensionRunSetupScreenBase (which in turn inherits from MonoBehavior). You will be required to override the following:
- headerText: The displayed title on the screen
- showCardDisplayer: Set this to return true if you want the panel on your screen that shows card information
- showCardPanel: Set this to return true if you want the scrollable panel on your screen that shows selectable cards
There is also a virtual method called InitializeScreen which you should use to build the content of your screen.
In general, you are responsible for doing all the hard work of building your screen. However, all the boilerplate content is built for you. You will automatically be given the continue and back buttons, the header (which shows the current challenge level), the footer (which displays changes as the challenge level changes), and the title of your screen. You can also optionally be given a scrollable card selection panel and card information displayer panel if you need them (using the properties shown above).
Once you've written the custom screen class, you need to register it with AscensionScreenManager like so:
AscensionScreenManager.RegisterScreen<MyCustomScreen>();
This API supports adding a custom model for specific Tribes that appear in the Wood Carver node. Natively there is already a default model that is used when a custom Tribe is added using TribeManager. However if your Tribe does not have any cards that use it then it will not appear in the WoodCarver.
If you want to add your own model for your tribe then you can use the example below.
TotemManager.NewTopPiece<CustomIconTotemTopPiece>("NameOfTotem", Plugin.PluginGuid, Tribe, prefab);
If you are using a model that you have created then here is an example of how to use asset bundles to include it.
if (AssetBundleHelper.TryGet("pathToAssetBundle", "nameOfPrefabInAssetBundle", out GameObject prefab))
{
TotemManager.NewTopPiece<CustomIconTotemTopPiece>("NameOfTotem", Plugin.PluginGuid, Tribe, prefab);
}
You will need a new class for your totem top so it doesn't look for an icon to populate from a tribe.
public class MyCustomTotemTopPiece : CompositeTotemPiece
{
protected virtual string EmissionGameObjectName => "GameObjectName";
public override void SetData(ItemData data)
{
base.SetData(data);
// Set emissiveRenderer so the game knows what to highlight when hovering their mouse over the totem top
emissiveRenderer = this.gameObject.FindChild(EmissionGameObjectName);
if (emissiveRenderer != null)
{
emissiveRenderer = icon.GetComponent<Renderer>();
}
if (emissiveRenderer == null)
{
InscryptionAPIPlugin.Logger.LogError($"emissiveRenderer not assigned to totem top!");
}
}
}
Then add your totem with your new class
if (AssetBundleHelper.TryGet("pathToAssetBundle", "nameOfPrefabInAssetBundle", out GameObject prefab))
{
TotemManager.NewTopPiece<MyCustomTotemTopPiece>("NameOfTotem", Plugin.PluginGuid, Tribe, prefab);
}
This API supports adding a custom item into the game.
Specify the class for your item and what happens when its used.
public class CustomConsumableItem : ConsumableItem
{
public override IEnumerator ActivateSequence()
{
base.PlayExitAnimation();
yield return new WaitForSeconds(0.25f);
yield return base.StartCoroutine(Singleton<ResourcesManager>.Instance.AddBones(4, null));
yield break;
}
public override bool ExtraActivationPrerequisitesMet()
{
if (!base.ExtraActivationPrerequisitesMet())
{
return false;
}
// Optional: Stop player from using the item!
return true;
}
}
Add the new item to the ConsumableItemManager.
If you don't have a custom model you can specify one of the default types from ConsumableItemManager.ModelType.
ConsumableItemManager.ModelType modelType = ConsumableItemManager.ModelType.Basic;
ConsumableItemManager.New(Plugin.PluginGuid, "Custom Item", "Does a thing!", textureOrSprite, typeof(CustomConsumableItem), modelType)
.SetDescription(learnText)
.SetAct1();
To add a card that has a card ina bottle use this
ConsumableItemManager.NewCardInABottle(PluginGuid, cardInfo.name)
.SetAct1();
If you have a custom model for your item you can specify it in the different constructor
GameObject prefab = ...
ConsumableItemManager.New(Plugin.PluginGuid, "Custom Item", "Does a thing!", textureOrSprite, typeof(CustomConsumableItem), prefab)
.SetDescription(learnText)
.SetAct1();
This API supports adding a custom pelt to be sold by the trapper and traded by the trader.
CardInfo bonePeltInfo = CardManager.New(PluginGuid, "Bone Pelt", "Bone Pelt", 0, 2);
bonePeltInfo.portraitTex = TextureHelper.GetImageAsTexture(Path.Combine(PluginDirectory, "Art/portrait_skin_bone.png")).ConvertTexture();
bonePeltInfo.cardComplexity = CardComplexity.Simple;
bonePeltInfo.AddTraits(Trait.Pelt);
bonePeltInfo.temple = CardTemple.Nature;
bonePeltInfo.AddSpecialAbilities(SpecialTriggeredAbility.SpawnLice);
bonePeltInfo.AddAppearances(CardAppearanceBehaviour.Appearance.TerrainBackground, CardAppearanceBehaviour.Appearance.TerrainLayout);
PeltManager.New(new PeltManager.CustomPeltData()
{
AbilityCount = 0,
CostCallback = ()=>3,
AvailableAtTrader = true,
CardNameOfPelt = bonePeltInfo.name,
MaxChoices = 8,
PluginGUID = PluginGuid,
GetChoicesCallback = () =>
{
return CardManager.AllCardsCopy.FindAll((a) =>
a.BonesCost > 0 && a.temple == CardTemple.Nature);
},
});
This API supports adding new tracks to the Gramophone in Leshy's Cabin. (A user must have the Hex disc unlocked in order to be able to listen to the new tracks.)
All you need for that is a regular audio file. The API will do all of the conversion. The file should be inside the 'plugins' folder. and the supported audio formats are MP3, WAV, OGG and AIFF.
You can register your track like this:
GramophoneManager.AddTrack(PluginGuid, "MyFile.wav", 0.5f);
The first parameter should be your plugin's GUID. The second parameter should be your file's name. The third parameter is optional, and determines the volume of your track, from 0 to 1f.
This API provides a helper method for converting audio files to Unity AudioClip objects so that they can be played in-game with the AudioSource component. You can use this to replace in-game music through patches, or to play your own sound effects.
The audio file should be located inside of the 'plugins' folder. The supported audio formats are MP3, WAV, OGG and AIFF.
You can convert your audio file into an AudioClip object like this:
AudioClip audio = SoundManager.LoadAudioClip("Example.mp3");
This allows you to change a mask already added to Inscyrption. Can be any Vanilla masks or one someone else has added.
MaskManager.Override("guid", "nameOfNewMask", LeshyAnimationController.Mask.Angler, "pathToTexture");
This example shows changing the mask the Angler uses to have a custom texture we are using.
NOTE: This also changes the model so we can use a texture without the fuss of UV mapping the Anglers actual mask.
If you still want to use the Anglers model then use .SetModelType(MaskManager.ModelType.Angler)
If you want to add a new mask that will be randomly chosen when Leshy goes to put on a mask use this.
MaskManager.AddRandom("guid", "nameOfNewMask", LeshyAnimationController.Mask.Prospector, "pathToTexture");
This example shows adding a new mask that when going to the Prospector boss fight, Leshy will choose between the default Prospector mask and this new one.
You can add as many random masks as you want. There is no limit.
MaskManager.Add("guid", "nameOfNewMask", "pathToTexture");
ResourceLookup resourceLookup = new ResourceLookup();
resourceLookup.FromAssetBundle("pathToAssetBundle", "prefabNameInsideBundle");
MaskManager.ModelType modelType = MaskManager.RegisterPrefab("guid", "nameOfModel", resourceLookup);
var mask = MaskManager.Add("guid", "nameOfMask");
mask.SetModelType(modelType);
This will tell Leshy to push a mask on his face.
Useful for when you have your own boss sequence and you want to tell Leshy to put on your new mask!
LeshyAnimationController.Instance.PutOnMask(LeshyAnimationController.Mask.Woodcarver, false);
public class Plugin : BaseUnityPlugin
{
private void Awake()
{
MyCustomMask.Setup();
}
}
public class MyCustomMask : MaskBehaviour
{
public static LeshyAnimationController.Mask ID;
public static void Setup()
{
var mask = MaskManager.Add("guid", "nameOfNewMask", "pathToTexture");
mask.SetMaskBehaviour(typeof(MyCustomMask));
ID = mask.ID;
}
}
Asset bundles are how you can import your own models, texture, gameobjects and more into Inscryption.
Think of them as fancy .zip's that's supported by Unity.
- Make a Unity project. Make sure you are using 2014.4.24f1 or your models will not show in-game.
- Install the AssetBundleBrowser package. (Window->Package Manager)
- Select the assets you want to be in the bundle (They need to be in the hierarchy, not in a scene!)
- At the bottom of the Inspector window you'll see a section labedled "Asset Bundle"
- Assign a new asset bundle name (example: testbundleexample)
- Build Asset bundles Window->AssetBundle Browser
- Go to the output path using file explorer
- There should be a file called 'testbundleexample' in that folder (It will not have an extension!)
- Copy this file into your mod folder
if (AssetBundleHelper.TryGet<GameObject>("pathToBundleFile", "nameOfPrefabInsideAssetBundle", out GameObject prefab))
{
GameObject clone = GameObject.Instantiate(prefab);
// Do things with gameobject!
}
First parameter is the path to the asset bundle that we copied to your mod folder in #9
Second parameter is the name of the prefab or texture... etc that you changed to have the asset bundle name in #4
Third parameter is the result of taking the object out of the asset bundle.
NOTE: Getting a prefab from an asset bundle does not laod it into the world. You need to clone it with Instantiate!
Make sure you are using 2019.4.24f1 to build the asset bundle? If not the model won't show!
This API supports creating new talking cards from scratch, without the need to load up your own Unity prefabs or anything of the sort!
All you have to do is create a class that implements the ITalkingCard interface, which contains the following fields:
Field | Type | Description |
---|---|---|
CardName | string | The name of an existing card. |
Emotions | List<EmotionData> | Your talking card's emotions. |
FaceInfo | FaceInfo | A bit of info about your talking card: blink rate and voice. |
DialogueAbility | SpecialTriggeredAbility | The special ability that controls your talking card's dialogue. |
And after that, all you need to do is register your talking card with the TalkingCardManager.New<T>()
method as such:
TalkingCardManager.New<ExampleClass>();
Additionally, as you can see from the last item in the table above, you're gonna need to create a new SpecialTriggeredAbility that controls your talking card's dialogue.
Don't worry, I'll explain exactly what you need to do!
Fortunately for you, this API already supports creating new special abilities. Additionally, you won't have to do all the work: the base game already defines abstract classes you should inherit from when creating a special ability for your talking card:
Act | Class To Inherit From |
---|---|
Act 1 | PaperTalkingCard |
Act 3 | DiskTalkingCard |
Additionally, to make the task of creating Act 1 talking cards even easier for you, I've defined an abstract class that inherits from PaperTalkingCard, includes the ITalkingCard interface and adds a few small tweaks behind the scenes! It's called CustomPaperTalkingCard.
All you have to do is create a class that inherits from CustomPaperTalkingCard and implement its abstract fields as such:
public class ExampleTalkingCard : CustomPaperTalkingCard
{
public override string CardName => "example_ExampleCard";
public override List<EmotionData> Emotions => new List<EmotionData>()
{
new EmotionData(emotion: Emotion.Neutral,
face: "Example_Face.png",
eyes: ("Example_EyesOpen.png", "Example_EyesClosed.png"),
mouth: ("Example_MouthOpen.png", "Example_MouthClosed.png"),
emission: ("Example_EyesOpenEmission.png", "_"))
};
public override FaceInfo FaceInfo => new FaceInfo(blinkRate: 1.5f, voiceSoundPitch: 1.1f);
public override SpecialTriggeredAbility DialogueAbility => dialogueAbility;
private SpecialTriggeredAbility dialogueAbility = SpecialTriggeredAbilityManager.Add(
guid: Plugin.PluginGuid,
abilityName: "ExampleDialogueAbility",
behavior: typeof(ExampleTalkingCard)).Id;
public override string OnDrawnDialogueId => "Example_OnDrawn";
public override string OnPlayFromHandDialogueId => "Example_OnPlayFromHand";
public override string OnAttackedDialogueId => "Example_OnAttacked";
public override string OnBecomeSelectablePositiveDialogueId => "Example_OnSelectedPositive";
public override string OnBecomeSelectableNegativeDialogueId => "Example_OnSelectedNegative";
public override Dictionary<Opponent.Type, string> OnDrawnSpecialOpponentDialogueIds => new Dictionary<Opponent.Type, string>();
}
And, after that, all you need to do is register your new talking card with the TalkingCardManager.New<T>()
method as such:
TalkingCardManager.New<ExampleTalkingCard>();
Note: CustomPaperTalkingCard can only be used for Act 1 talking cards. If you want to make an Act 3 talking card, you're gonna have to inherit from DiskTalkingCard directly!
Below I'm going to explain a few important things about talking cards in-depth!
The EmotionData class is a container for the sprites for one of your character's emotions. It has a constructor that takes Unity Sprites as parameters, and another constructor that takes the path to your image as a string for each image.
The constructors take the following parameters:
Parameter | Description |
---|---|
emotion | The chosen emotion. |
face | A sprite for your character's face. |
eyes | A pair of sprites for your character's eyes: open and closed. |
mouth | A pair of sprites for your character's mouth: open and closed. |
emission | A pair of sprites for your character's eyes' emission: open and closed. |
If you ever need to use an empty portrait texture, you can use TalkingCardManager.EmptyPortrait
!
The string constructor also contains shorthand for an empty texture, if you need it: Just add an "_" as your string for that given parameter.
A guide on how to use emotions in your card's dialogue can be found in this section below.
The FaceInfo class contains a bit of info about your talking card: namely, blink rate, voice pitch, and the sound used for your character's voice in general.
The constructor takes the following parameters:
Parameter | Type | Description |
---|---|---|
blinkRate | float | How often your character blinks. |
voiceId | string | Your character's "voice". Will explain more below. |
voiceSoundPitch | float | Your character's voice's pitch. |
customVoice | string | A custom voice for your character. Will be explained below. |
"voiceId" can only be one of these three strings:
- female1_voice
- cat_voice
- kobold_voice
Most talking cards in the game use the first and simply change the pitch.
You can add a custom voice to your character instead of using one of the default voices. For that, all you need to is pass the path to your audio file as the "customVoice" parameter.
The supported audio formats are MP3, WAV, OGG and AIFF!
Please use a very short audio file for your voice. Typically, you want only a very short 'vowel' sound for this, since it's going to be played in rapid repetition.
If you pass anything as "customVoice", then the contents of the "voiceId" parameter will not matter.
After looking at the example above, you might be wondering "What's all of that DialogueId stuff about? How do I make my own dialogue events?".
I'm gonna explain everything to you in detail now!
Talking cards can respond to a variety of game events. If you want your card to respond to a given event, you can override that property and return the ID of your new dialogue event.
Some properties are abstract and must be implemented: namely, OnDrawnDialogueId
, OnPlayFromHandDialogueId
, OnAttackedDialogueId
, OnBecomeSelectablePositiveDialogueId
and OnBecomeSelectableNegativeDialogueId
.
And here's a full list of triggers you can override, and a small explanation of each.
Note: All of these names end in "DialogueId", so I've omitted that last part of the name to be concise.
Trigger | Description |
---|---|
OnDrawn | Plays when your card is drawn. |
OnPlayFromHand | Plays when your card is played. |
OnAttacked | Plays when your card is attacked. |
OnBecomeSelectablePositive | Plays when your card becomes selectable for a positive effect. |
OnBecomeSelectableNegative | Plays when your card becomes selectable for a negative effect. |
OnSacrificed | Plays when your card is sacrificed. |
OnSelectedForDeckTrial | Plays when your card is selected in the deck trial node. |
OnSelectedForCardMerge | Plays before your card receives the sigil in the sigil node. |
OnSelectedForCardRemove | Plays when your card is selected for removal. |
Additionally, the OnDrawnSpecialOpponentDialogueIds
dictionary lets you add special dialogue when your card is drawn in a specific boss battle, like this:
public override Dictionary<Opponent.Type, string> OnDrawnSpecialOpponentDialogueIds => new Dictionary<Opponent.Type, string>()
{
{ Opponent.Type.ProspectorBoss, "Example_TalkAboutProspector" }
};
You can create your own dialogue events with this API's DialogueManager.GenerateEvent()
method, like this:
DialogueManager.GenerateEvent(
pluginGUID: Plugin.PluginGuid,
name: "Example_OnPlayFromHand",
mainLines: new List<CustomLine>() { "This is a main line." },
repeatLines: new List<List<CustomLine>>()
{
new List<CustomLine>() { "This is a repeat line!" },
new List<CustomLine>() { "This is another repeat line!" }
}
);
A brief explanation of each parameter:
Field | Description |
---|---|
pluginGUID | Your mod's GUID. |
name | The name for your dialogue event; this is what you'll use to refer to it! |
mainLines | A set of lines that plays in the very first time this event runs. |
repeatLines | Multiple sets of lines that are played after the first time this event has run. |
A really neat feature of Inscryption's dialogue events are dialogue codes. They add a lot of life to dialogue!
The dialogue codes most relevant to talking cards will be explained below. All of these work with talking cards.
This is by far the dialogue code you'll wanna use the most. It's also the one the game itself uses the most in all of its dialogue.
The "[w:x]" dialogue code adds a pause of x seconds before the rest of a sentence plays.
You can use it like this:
"Hello.[w:1] How are you?"
In this example, after saying "Hello.", the character waits 1 second before saying "How are you?".
The number of seconds does not have to be an integer. Using "[w:0.2]" to wait only 0.2 seconds is valid, for example, and used often throughout the base game's dialogue.
This being said, I'd advise you not to go below [w:0.1], as I don't know how small the number can go before issues arise. (And there's no point in going below that, anyhow.)
The [c:] dialogue code changes the color of a portion of your text.
You can use it like this:
"[c:R]This text is red.[c:] This text is not."
In this example, the part after [c:R] is colored in the color that matches the code 'R', which is the color red, and the part after [c:] has the default text color. You can think of this as "switching on" the colorful text mode and then switching it off.
"But how do I know the codes for each color?" Fear not! Here's a comprehensive table of all available colors and their respective codes:
Code | Color |
---|---|
B | Blue |
bB | Bright Blue |
bG | Bright Gold |
blGr | Bright Lime Green |
bR | Bright Red |
brnO | Brown Orange |
dB | Dark Blue |
dlGr | Dark Lime Green |
dSG | Dark Seafoam |
bSG | Glow Seafoam |
G | Gold |
gray | Gray |
lGr | Lime Green |
O | Orange |
R | Red |
(For the record: These are the colors the game has available, built-in. I did not choose them. Yes, it's a very odd selection of colors.)
I have added a way to use custom colors with dialogue codes. In place of one of the color codes in the table above, you can instead use a hex color code, and this mod will parse the code into an usable Color for the text.
Here's an example:
"You must be... [w:0.4][c:#7f35e6]confused[c:][w:1].",
In this example, the word "confused" is colored in the color #7f35e6. Which, if you don't wanna look it up, is this color!
Please note that for compatibility reasons, your hex color code should include the '#'.
The [leshy:x] dialogue code makes Leshy say x. This color code is very useful for making Leshy and your card talk a bit between each other!
You can use it like this:
"We're all doomed.[leshy:Quiet now.][w:2]",
In this example, the character says "We're all doomed." and then Leshy says "Quiet now." right after. The text remains on the screen for 2 seconds.
There are a few things to note from that example:
- You don't need to put quotation marks around the line Leshy is going to say.
- The "Wait" dialogue code is still usable with Leshy's lines.
You can change your character's emotion in their dialogue lines with the dialogue code [e:x]
, where 'x' is the name of an emotion. You can look at the table above for the names of all the available emotions.
This mod adds patches to make the emotion names not case-sensitive, which means the following lines are all equally valid:
"[e:Anger]I'm angry."
"[e:anger]I'm angry."
"[e:AnGeR]I'm angry and my keyboard is acting up."
If you prefer, you can use the numeric value associated with an emotion instead of its name! This is perfectly valid, for example:
"[e:2]I'm angry."