Magic, Spells, and Enchantments
- Spells
- The template spell
- Mandatory fields
- Common fields
- Adding spells to professions and NPCs
- Examples
- Enchantments
Spells
Spells in Cataclysm: Dark Days Ahead consist in actions performed by a character or item, that result on a target or targets receiving an event.
This can be anything from the humble fireball spell or the simple heal, to granting states, mutations, summoning items and vehicles, spawning monsters, exploding monsters, applying auras, crowd-controlling, blinking, transforming terrain, adding or subtracting stats, granting or removing effect_type
s, and more.
Remarkably, some things that don’t seem quite “magical” at first glance may also be handled by spells, like casting Fist, casting Gun, casting Vomit, granting (or removing) drunk
enness, handling fields (field_id
s), transforming items, among other things.
By making clever use of JSON fields, interactions, combinations and descriptions, anyone can go wild in spell crafting. Different spells from the official mods can be used as reference.
The template spell
In data/mods/Magiclysm
there is a template spell, copied here for your perusal:
{
// This spell exists in json as a template for contributors to see the possible values of the spell
"id": "example_template", // id of the spell, used internally. not translated
"type": "SPELL",
"name": "Template Spell", // name of the spell that shows in game
"description": "This is a template to show off all the available values",
"valid_targets": [ "hostile", "ground", "self", "ally" ], // if a valid target is not included, you cannot cast the spell on that target.
"effect": "shallow_pit", // effects are coded in C++. A list will be provided below of possible effects that have been coded.
"effect_str": "template", // special. see below
"shape": "blast", // the "shape" of the spell's area of effect. uses the aoe stat
"extra_effects": [ { "id": "fireball", "hit_self": false, "max_level": 3 } ], // this allows you to cast multiple spells with only one spell
"affected_body_parts": [ "head", "torso", "mouth", "eyes", "arm_l", "arm_r", "hand_r", "hand_l", "leg_l", "foot_l", "foot_r" ], // body parts affected by effects
"flags": [ "SILENT", "LOUD", "SOMATIC", "VERBAL", "NO_HANDS", "NO_LEGS", "SPAWN_GROUP" ], // see "Spell Flags" below
"spell_class": "NONE", //
"base_casting_time": 1000, // this is the casting time (in moves)
"final_casting_time": 100,
"casting_time_increment": -50,
"base_energy_cost": 30, // the amount of energy (of the requisite type) to cast the spell
"final_energy_cost": 100,
"energy_increment": -6,
"energy_source": "MANA", // the type of energy used to cast the spell. types are: MANA, BIONIC, HP, STAMINA, NONE (none will not use mana)
"components": [requirement_id] // an id from a requirement, like the ones you use for crafting. spell components require to cast.
"difficulty": 12, // the difficulty to learn/cast the spell
"max_level": 10, // maximum level you can achieve in the spell
"get_level_formula_id": "jmath_get_level_formula", // The id of a jmath formula that calculates what level the spell is for a given exp value. Must be the inverse of exp_for_level_formula_id.
"exp_for_level_formula_id": "jmath_exp_for_level_formula",// The id of a jmath formula that calculates how much exp is required for a given level. Must be the inverse of get_level_formula_id.
"min_accuracy" -20, // the accuracy bonus of the spell. around -15 and it gets blocked all the time
"max_accuracy": 20, // around 20 accuracy and it's basically impossible to block
"accuracy_increment": 1.5
"min_damage": 0, // minimum damage (or "starting" damage)
"max_damage": 100, // maximum damage the spell can achieve
"damage_increment": 2.5, // to get damage (and any of the other below stats) multiply this by spell's level and add to minimum damage
"min_aoe": 0, // area of effect, or range of variance
"max_aoe": 5,
"aoe_increment": 0.1,
"min_range": 1, // range of the spell
"max_range": 10,
"range_increment": 2,
"min_dot": 0, // damage over time
"max_dot": 2,
"dot_increment": 0.1,
"min_duration": 0, // duration of spell effect in moves (if the spell has a special effect)
"max_duration": 1000,
"duration_increment": 4,
"min_pierce": 0, // how much of the spell pierces armor
"max_pierce": 1,
"pierce_increment": 0.1,
"field_id": "fd_blood", // the string id of the field (currently hardcoded)
"field_chance": 100, // one_in( field_chance ) chance of spawning a field per tile in aoe
"min_field_intensity": 10, // field intensity of fields generated
"max_field_intensity": 10,
"field_intensity_increment": 1,
"field_intensity_variance": 0.1 // the field can range in intensity from -variance as a percent to +variance as a percent i.e. this spell would be 9-11
"sound_type": "combat", // the type of sound. possible types are: background, weather, music, movement, speech, activity, destructive_activity, alarm, combat, alert, order
"sound_description": "a whoosh", // the sound description. in the form of "You hear %s" by default it is "an explosion"
"sound_ambient": true, // whether or not this is treated as an ambient sound or not
"sound_id": "misc", // the sound id
"sound_variant": "shockwave", // the sound variant
"learn_spells": { "create_atomic_light": 5, "megablast": 10 } // the caster will learn these spells when the current spell reaches the specified level. should be a map of spell_type_id and the level at which the new spell is learned.
}
The template spell above shows every JSON field that spells can have. Most of these values can be set at 0 or “NONE”, so you may leave out most of these fields if they do not pertain to your spell.
When deciding values for some of these, it is important to note that some of the formulae are not linear. For example, this is the formula for spell failure chance:
( ( ( ( spell_level - spell_difficulty ) * 2 + intelligence + spellcraft_skill ) - 30 ) / 30 ) ^ 2
Meaning a spell with difficulty 0 cast by a player with 8 intelligence, 0 spellcraft, and level 0 in the spell will have a 53% spell failure chance. On the other hand, a player with 12 intelligence, 6 spellcraft, and level 6 in the same spell will have a 0% spell failure chance.
However, experience gain is a little more complicated to calculate. The formula for how much experience you need to get to a level is below:
e ^ ( ( level + 62.5 ) * 0.146661 ) ) - 6200
Mandatory fields
As noted above, few JSON fields are actually required for spells to work. Some of the mandatory fields are:
Identifier | Description |
---|---|
id | Unique ID for the spell, used internally. Must be one continuous word, use underscores if necessary. |
type | Indicates the JSON object is a SPELL . |
name | Name of the spell that shows in game. |
description | Description of the spell that shows in game. |
valid_targets | Targets affected by the spell. If a valid target is not included, you cannot cast the spell on that target. Additionally, if the valid target is not specified, the spell aoe will not affect it. Can be ally , field , ground , hostile , item , none or self . |
effect | Hardcoded spell behaviors, roughly speaking spell “type”. See the list below. |
shape | The shape of the spell’s area of effect. See the list below. |
Depending on the spell effect, more or less fields will be required.
For example, a classic attack
spell needs the damage_type
, min_damage
, max_damage
, min_range
, max_range
, and max_level
fields.
In contrast, an attack
spell using effect_str
to grant a self
buff (without a damage component) needs the min_duration
, max_duration
, and max_level
fields instead.
Spell effects
Each spell effect is defined in the effect
field. For example, the Magus spell “Magic Missile” has the attack
effect, meaning it deals damage to a specific target:
{
"id": "magic_missile",
"effect": "attack",
"min_damage": 1
}
while the Druid spell “Nature’s Bow” has the spawn_item
effect, designating the ID of the item to spawn:
{
"id": "druid_naturebow1",
"effect": "spawn_item",
"effect_str": "druid_recurve"
}
Below is a table of currently implemented effects, along with special rules for how they work:
Effect | Description |
---|---|
add_trap | Adds a trap in the target tile. This always succeeds (unless there is an existing trap) and only places 1 trap. The effect_str is the id of the trap. |
area_pull | Pulls valid_targets in its aoe toward the target location. Currently, the pull distance is set to 1 (see directed_push ). |
area_push | Pushes valid_targets in its aoe away from the target location. Currently, the push distance is set to 1 (see directed_push ). |
attack | Causes damage to valid_targets in its aoe, and applies effect_str named effect to targets. To damage terrain use bash . |
banishment | Kills any MONSTER in the aoe up to damage hp. Any overflow hp is taken from the caster; if it’s more than the caster’s hp the spell fails. |
bash | Bashes the terrain at the target. Uses damage() as the strength of the bash. |
charm_monster | Charms a monster that has less hp than damage() for approximately duration(). |
dash | Dashes forward up to range and hits targets in a cone at the target. |
directed_push | Pushes valid_targets in aoe away from the target location, with a distance of damage(). Negative values pull instead. |
effect_on_condition | Runs the effect_on_condition from effect_str on all valid targets. The EOC will be centered on the player, with the NPC as caster and a context val location variable spell_location for the target primarily useful if the target isn’t a creature. |
emit | Causes an emit at the target. |
explosion | Causes an explosion centered on the target. Uses damage() for power and factor aoe()/10. |
flashbang | Causes a flashbang effect is centered on the target. Uses damage() for power and factor aoe()/10. |
fungalize | Fungalizes the target. |
guilt | Target gets the guilt morale as if it killed the caster. |
map | Maps out the overmap centered on the player, to a radius of aoe(). |
mod_moves | Adds damage() moves to the targets. Negative values “freeze” for that amount of time. |
morale | Gives a morale effect to NPCs or the avatar within the aoe. Uses damage() for the value. decay_start is duration() / 10. |
mutate | Mutates the targets. If effect_str is defined, mutates toward that category instead of picking at random. If the MUTATE_TRAIT flag is used, allows effect_str to be a specific trait. Damage() / 100 is the percent chance the mutation will be successful (10000 represents 100.00%). |
noise | Causes damage() amount of noise at the target. Note: the noise can be described further with sound_type , sound_description , sound_ambient , sound_id and sound_variant . |
pain_split | Evens out all of your limbs’ damage. |
pull_target | Attempts to pull the target towards the caster in a straight line. If the path is blocked by impassable furniture or terrain, the effect fails. |
recharge_vehicle | Increases or decreases the battery charge of a vehicle or battery-connected power grid. Damage is equal to the charge (negative decreases). |
recover_energy | Recovers an energy source equal to damage of the spell. The energy source is defined in effect_str and may be one of BIONIC , SLEEPINESS , PAIN , MANA or STAMINA . |
remove_effect | Removes effect_str effects from all creatures in the aoe. |
remove_field | Removes a effect_str field in the aoe. Causes teleglow of varying intensity and potentially teleportation depending on field density, if the field removed is fd_reality_tear . (see ter_transform for more versatility) |
revive | Revives a monster like a zombie necromancer. The monster must have the REVIVES flag. |
revive_dormant | Revives a dormant monster. The monster must have the REVIVES AND the DORMANT flag. |
short_range_teleport | Teleports the player randomly range spaces with aoe variation. See also the TARGET_TELEPORT and UNSAFE_TELEPORT flags. |
slime_split | The slime splits into two large or normal slimes, depending on mass. Note: hardcoded for mon_blob -type enemies, check the monster death_function + spell summon combination. |
spawn_item | Spawns an item that will disappear at the end of its duration. Default duration is 0. |
summon | Summons a MONSTER or monstergroup from effect_str that will disappear at the end of its duration. Default duration is 0. See also the SPAWN_WITH_DEATH_DROPS flag. |
summon_vehicle | Summons a vehicle from effect_str that will disappear at the end of its duration. Default duration is 0. |
targeted_polymorph | A targeted monster is permanently transformed into the MONSTER specified by effect_str , if it has less HP than the spell’s damage. If effect_str is left empty, the target will transform into a random monster with a similar difficulty rating. Alternatively, the POLYMORPH_GROUP flag can be used to pick a weighted ID from a monstergroup . The player and NPCs are immune to this spell effect. |
ter_transform | Transforms the field, furniture, terrain and trap in its aoe. The chance of any one of the points in the aoe changing is 1 / (damage). The effect_str is the ID of a ter_furn_transform (see TER_FURN_TRANSFORM.md for further details). |
timed_event | Adds a timed event to the player only. Valid timed events are: amigara , artifact_light , dim , help , robot_attack , roots_die , spawn_wyrms , temple_flood , temple_open , temple_spawn , wanted . NOTE: This was added only for artifact active effects. Support is limited, use at your own risk. |
translocate | Opens up a window that allows the caster to choose a translocation gate to teleport to. |
upgrade | Immediately upgrades a target MONSTER . |
vomit | Any creature within its aoe will instantly vomit, if it’s able to do so. |
Spell shape
Another mandatory field is the spell shape. This dictates how the area of effect works:
Shape | Description |
---|---|
blast | A circular blast centered on the impact position. Aoe value is the radius. |
cone | Fires a cone with an arc equal to aoe in degrees. |
line | Fires a line with a width equal to the aoe. |
Common fields
The following JSON fields are also used in spells and, while optional, greatly expand how spells can behave.
All fields that are numeric also support a “variable object” (see NPCs.md for full details). Some fields are ignored by monsters and items.
Field group | Description | Example |
---|---|---|
min_X , max_X , X_increment | Minimum value, maximum value, and the value increase per level. Note: a spell is not set to have max_X at max lvl, max_X is the ceiling for the value: a spell with min_damage: 0, max_damage: 100, damage_increment: 5 and max_level: 10 , will deal 5 damage at lvl 1, 10 at lvl 2 and 50 at lvl 10, as [ 0 + ( 5 * 10 ) ]. Similarly, if max_damage is instead 25, the damage will cap at spell lvl 5, as [ 0 + ( 5 * 5 ) ] | "min_damage": { "math": [ "u_skill('dodge') + u_val('intelligence')" ] }, ,"min_range": 24, ,"min_aoe": { "math": [ "( VAR_1 / 3 )" ] }, . |
min_damage , max_damage , damage_increment | “Damage” value of the spell by default (or healing, if value is negative). The spell effect can modify its function. When the spell doesn’t increase by level (such as a monster spell), min_damage and max_damage can be set at the same value ("min_damage": 15, "max_damage": 15 ), damage_increment can be omitted. | “min_damage”: 0, “max_damage”: 100, “damage_increment”: 5, |
min_duration , max_duration , duration_increment | Duration of the spell effects and effect_str , depending on the type of the ID. Written in moves, so 100 means 1 second , 1 minute as 6000 , and so on. | “min_duration”: 100, “max_duration”: 6000, “duration_increment”: 100, |
min_range , max_range , range_increment | Distance from the caster to the target. Can be omitted if the target is the caster (giving an buff/debuff, spawning an item). Note: the reality bubble diameter is ~60 tiles. | “min_range”: 2, “max_range”: 10, “range_increment”: 0.5, |
min_aoe , max_aoe , aoe_increment | Short for “area of effect”, area/zone of tiles that the spell will affect. | “min_aoe”: 0, “max_aoe”: 5, “aoe_increment”: 0.1, |
min_accuracy , max_accuracy , accuracy_increment | Accuracy of the spell. -20 accuracy will cause it to always miss, 20 will cause it always hit. Currently doesn’t work. | “min_accuracy” -20, “max_accuracy”: 20, “accuracy_increment”: 1.5 |
min_dot , max_dot , dot_increment | Short for “damage over time”. Similar to damage, positive values hurt while negative values heal. Note: decimal values use roll_remainder(); having 0.5 would result in dot dealing 1 damage randomly every 2 seconds on average. Affected by affected_body_parts field | “min_dot”: 0, “max_dot”: 2, “dot_increment”: 0.1, |
min_pierce , max_pierce , pierce_increment | Armor “piercing”, how much armor of the same damage_type the spell will ignore. | “min_pierce”: 0, “max_pierce”: 1, “pierce_increment”: 0.1, |
min_bash_scaling , max_bash_scaling , bash_scaling_increment | Converts a spell with the “attack” effect’s damage into the scaling amount of terrain damage. | “min_bash_scaling”: 1.0, “max_bash_scaling”: 2.0, “bash_scaling_increment”: 0.1, |
base_casting_time , final_casting_time , casting_time_increment | Time the caster spends when casting the spell. Similar to duration, it’s written in moves, which allows spells to be casted in fractions of a second. Ignored for monsters and items that cast spells. If several spells are chained, only the first one will apply the cost. Note: The casting time is not shown to the player (e.g. a cast of 300 will behave as if the player waits for 3 turns). | “base_casting_time”: 1000, “final_casting_time”: 100, “casting_time_increment”: -50, |
base_energy_cost , final_energy_cost , energy_increment | Amount of energy spent for cast. If several spells are chained, only the first one will apply the cost. Ignored for monsters and items that cast spells. | “base_energy_cost”: 30, “final_energy_cost”: 100, “energy_increment”: -6, |
field_id , field_chance , min_field_intensity , max_field_intensity , field_intensity_increment , field_intensity_variance | Allows the spell to spawn fields. field_id describes which field will be spawned, field_chance describes the chance as ( 1 / field_chance ). min_field_intensity , max_field_intensity and field_intensity_increment modify the field intensity and it’s growth (e.g. fd_electricity intensity 1 is “spark”, while intensity 10 is “electric cloud”). field_intensity_variance allows to randomly increase or decrease the intensity of the spell as a percent (e.g. intensity 10 and variance 0.1 means it can grow or shrink by 10%, or go from 9 to 11). | “field_id”: “fd_blood”, “field_chance”: 100, “min_field_intensity”: 10, “max_field_intensity”: 10, “field_intensity_increment”: 1, “field_intensity_variance”: 0.1 |
effect_str | The “effect” the spell has (see EFFECTS_JSON). Varies according to the spell effect field. | “effect_str”: “zapped”, “effect_str”: “mon_zombie”, |
max_level | How much you can train the spell. Default is 0. Ignored for monsters and items that cast spells. | “max_level”: 10, |
difficulty | How hard is to cast the spell. A high difficulty spell is easier to fail, failing grants spell XP at no resource cost. It also limits the maximum spellcasting skill that can be gained by casting it (e.g. difficulty 10 will train up to spellcasting lvl 10). | “difficulty”: 7, |
multiple_projectiles | If value is bigger than one, instead of shooting single spell with damage X, you will shoot multiple_projectiles amount of projectiles with damage damage/multiple_projectiles , which may damage different body parts (each projectile has 75% chance to hit torso still). Doesn’t interact with affected_body_parts | “multiple_projectiles”: 2, |
affected_body_parts | body_part where the attack , dot or effect_str will occur. For effects, set at torso by default. For attack and dot, picks a random bodypart, weighted by it’s hit_size | “affected_body_parts”: [ “head” ] |
extra_effects | Allows to cast a secondary spell id immediately after the primary spell. Allows for multiple id s. | “extra_effects”: [ { “id”: “fireball”, “hit_self”: false, “max_level”: 3 }, { “id”: “storm_chain_1” } ] |
learn_spells | Allow user to learn the spell id when they reach that spell level (e.g. "create_atomic_light": 5 means user will learn create_atomic_light when they reach level 5 of the primary spell). Allows for multiple id s. | “learn_spells”: { “create_atomic_light”: 5, “megablast”: 10 } |
teachable | Whether it’s possible to teach this spell between characters. Default is true. | "teachable": true |
message | The message that will be send in the log when the spell is cast. Default is “You cast %s!”. | “message”: “You feel refreshed.” |
sound_type , sound_description , sound_ambient , sound_id , sound_variant | Sound that will play when spell is cast. sound_type can be one of activity , alarm , alert , background , combat , destructive_activity , movement , music , order , speech , or weather . sound_description responds for the message shown in log, as “You hear %s”. Default is “an explosion”. sound_ambient whether or not this is treated as an ambient sound. | “sound_type”: “combat”, |
targeted_monster_ids | Limits the spell to target only the specified monster_id . | “targeted_monster_ids”: [ “mon_hologram” ], |
targeted_monster_species | Limits the spell to target only the specified monster SPECIES (full list at species.json). | “targeted_monster_species”: [ “ROBOT”, “CYBORG” ], |
ignored_monster_species | The opposite of targeted_monster_species : you can target everything except the specified monster SPECIES . | “ignored_monster_species”: [ “ZOMBIE”, “NETHER” ], |
Spell Flags
Flags allow you to provide additional customizations for spell effects, behavior, and limitations. Spells may have any number of flags, for example:
{
"id": "bless",
"//": "Encumbrance on the mouth (verbal) or arms (somatic) affect casting success, but not legs.",
"flags": [ "VERBAL", "SOMATIC", "NO_LEGS" ]
}
Flag | Description |
---|---|
CONCENTRATE | Focus affects spell fail %. |
EXTRA_EFFECTS_FIRST | The spell’s extra_effects will happen before the main spell effect. |
FRIENDLY_POLY | The target of a targeted_polymorph spell will become friendly to the caster if the spell resolves successfully. |
HOSTILE_SUMMON | Summon spell always spawns a hostile monster. |
HOSTILE_50 | Summoned monster spawns friendly 50% of the time. |
IGNITE_FLAMMABLE | If the spell area has anything flammable, a fire will be produced |
IGNORE_WALLS | Spell’s aoe goes through walls. |
LIQUID | Effects applied by spell will be resisted by waterproof armor if the spell targets a character’s body part. Does not currently affect damage. |
LOUD | Spell makes extra noise at target. |
MAGIC_FOCUS | Item does not interfere with hand encumbrance while spellcasting. |
MUST_HAVE_CLASS_TO_LEARN | The spell is autolearned when you have spell_class , and removed when you lost it. |
MUTATE_TRAIT | Overrides the mutate spell effect to use a specific trait_id instead of a category. |
NO_EXPLOSION_SFX | The spell will not generate a visual explosion effect. |
NO_FAIL | This spell cannot fail when cast. |
NO_HANDS | Hands do not affect spell energy cost. |
NO_LEGS | Legs do not affect casting time. |
NO_PROJECTILE | The “projectile” portion of the spell phases through walls, the epicenter of the spell effect is exactly where you target it, with no regards to obstacles. |
NON_MAGICAL | Ignores spell resistance when calculating damage mitigation and cannot be blocked by the NO_SPELLCASTING character flag. |
PAIN_NORESIST | Pain altering spells can’t be resisted (like with the deadened trait). |
PERCENTAGE_DAMAGE | The spell deals damage based on the target’s current hp. This means that the spell can’t directly kill the target. For characters (NPC or Avatar) damage applies to all body parts (damage 10 would deal 10% damage for each limb). If affected_body_parts is specified, deal percentage damage only to this body parts. Works with both attacks and DOTs |
PERMANENT | Items or creatures spawned with this spell do not disappear and die as normal. Items can only be permanent at maximum spell level; creatures can be permanent at any spell level. |
PERMANENT_ALL_LEVELS | Items spawned with this spell do not disappear even if the spell is not max level. |
POLYMORPH_GROUP | A targeted_polymorph spell will transform the target into a random monster from the monstergroup in effect_str . |
PSIONIC | Spells with this flag are not blocked by the NO_SPELLCASTING character flag, instead being blocked by NO_PSIONICS. |
RANDOM_AOE | Picks random number between (min + increment) * level and max instead of normal behavior. |
RANDOM_CRITTER | Same as RANDOM_TARGET but ignores ground. |
RANDOM_DAMAGE | Picks random number between (min + increment) * level and max instead of normal behavior. |
RANDOM_DURATION | Picks random number between (min + increment) * level and max instead of normal behavior. |
RANDOM_TARGET | Forces the spell to choose a random valid target within range instead of the caster choosing the target. This also affects extra_effects . |
RECHARM | charm_monster spell stacks its duration onto existing charm effect. |
SILENT | Spell makes no noise at target. |
SOMATIC | Arm encumbrance affects fail % and casting time (slightly). |
SPAWN_GROUP | Spawn or summon from an item_group or monstergroup , instead of the specific IDs. |
SPAWN_WITH_DEATH_DROPS | Allows summoned monsters to retain their usual death drops, otherwise they drop nothing. |
SPLIT_DAMAGE | If used, instead of dealing damage to a random body part, damage would be spread evenly across entire body, unless affected_body_parts is specificed. Works only on characters (NPC or Avatar), for monsters do not have limbs. Works with both attacks and DOTs |
SWAP_POS | A projectile spell swaps the positions of the caster and target. |
TARGET_TELEPORT | Teleport spell changes to maximum range target with aoe as variation around target. |
UNSAFE_TELEPORT | Teleport spell risks killing the caster or others. |
VERBAL | Spell makes noise at caster location, mouth encumbrance affects fail %. |
WONDER | This drastically alters the behavior of the parent spell: The spell itself doesn’t cast, but the damage and range information are used to cast the extra_effects . A n number of extra_effects will be chosen to be cast at random, where n is the current damage of the spell (stacks with the RANDOM_DAMAGE flag), the message of the casted spell will also be displayed. If this spell’s message is not wanted, make sure message is an empty string. |
Damage Types
The following are the available damage types, for those spells that have a damaging component:
Damage type | Description |
---|---|
acid | |
bash | |
biological | Internal damage such as poison. |
cold | |
cut | |
electric | |
heat | |
pure | This damage type goes through armor altogether. Set by default. |
stab |
Spell level
Spells can change effects as they level up. “Effect” in this context can be:
- accuracy
- aoe (area of effect)
- damage
- dot (damage over time)
- duration
- pierce, and
- range
The effect growth is indicated with the min_effect
, max_effect
and effect_increment
fields:
min_effect
is what the spell will do at level 0.max_effect
is where it stops growing, the level cap.effect_increment
is how much it changes per level.
Additionally, there are also included:
- energy cost (as
base_energy_cost
,final_energy_cost
,energy_increment
), and - field intensity (
min_field_intensity
,max_field_intensity
,field_intensity_increment
plusfield_intensity_variance
).
For example:
...
"min_range": 1,
"max_range": 25,
"range_increment": 5,
...
Min and max values must always have the same sign, but it can be negative e.g. in the case of spells that use a negative ‘recover’ effect to cause pain or stamina damage. For example:
{
"id": "stamina_damage",
"type": "SPELL",
"name": "Tired",
"description": "decreases stamina",
"valid_targets": [ "hostile" ],
"min_damage": -2000,
"max_damage": -10000,
"damage_increment": -3000,
"max_level": 10,
"effect": "recover_energy",
"effect_str": "STAMINA"
}
Learning Spells
There multiple ways to learn spells: learning a spell from an item (through a use_action
), from spells that have the learn_spells
field, and from traits/mutations. An example is shown below:
{
"id": "DEBUG_spellbook",
"type": "GENERIC",
"name": "A Technomancer's Guide to Debugging C:DDA",
"description": "static std::string description( spell sp ) const;",
"weight": 1,
"volume": "1 ml",
"symbol": "?",
"color": "magenta",
"use_action": {
"type": "learn_spell",
"//": "list of spells you can learn from the item",
"spells": [ "debug_hp", "debug_stamina", "example_template", "debug_bionic", "pain_split", "fireball" ]
}
}
You can study this spellbook for a rate of ~1 experience per turn depending on intelligence, spellcraft, and focus.
Below is an example of learn_spells
usage:
{
"id": "phase_door",
"type": "SPELL",
"name": "Phase Door",
"description": "Teleports you in a random direction a short distance.",
"effect": "short_range_teleport",
"shape": "blast",
"valid_targets": [ "none" ],
"max_level": 10,
"difficulty": 2,
"spell_class": "MAGUS",
"learn_spells": { "dimension_door": 10 }
}
Traits/mutations have the spells_learned
field, see the JSON_INFO documentation for details.
Extra spell effects
Another two interesting fields are extra_effects
and effect_str
:
-
extra_effects
allows to cast one or more spells simultaneously, thus enabling “chain” style casting. -
effect_str
works as a pointer, it links the main spell to a certain JSON object. Behavior can vary depending on the spell’seffect
, e.g. it may define whicheffect_type
will be applied the target, the ID of thespawn_item
,monster
orvehicle
to spawn, etc. Do note that some effect types are hardcoded, likebeartrap
,crushed
,dazed
,downed
,stunned
, etc. (non-exhaustive list!).
Adding spells to professions and NPCs
You can add spells to professions or NPC class definitions like this:
{
"id": "test_profession",
"type": "profession",
"name": "Test Professioner",
"description": "Tests professions",
"spells": [ { "id": "summon_zombie", "level": 0 }, { "id": "magic_missile", "level": 10 } ],
...
}
Note: This makes it possible to learn spells that conflict with a class. It also does not give the prompt to gain the class. Be judicious upon adding this to a profession!
Examples
The following are some spell examples, from simple to advanced:
Summon spell
{
"type": "SPELL",
"id": "test_summon", // id of the spell, used internally. not translated
"name": "Summon", // name of the spell that shows in game
"description": "Summons the creature specified in 'effect_str'",
"flags": [ "SILENT", "HOSTILE_SUMMON" ], // see "Spell Flags" in this document
"valid_targets": [ "ground" ], // if a valid target is not included, you cannot cast the spell on that target.
"min_damage": 1, // minimum number of creatures summoned (or "starting" number of creatures summoned)
"max_damage": 1, // maximum number of creatures summoned the spell can achieve
"min_aoe": 3, // area of effect of the spell, in this case the area the summons can appear in
"max_aoe": 3,
"effect": "summon", // effects are coded in C++. A list is provided in this document of possible effects that have been coded.
"effect_str": "mon_test_monster", // varies, see table of implemented effects in this document
"min_duration": 6250, // duration of spell effect in moves (if the spell has a special effect)
"max_duration": 6250
}
Self explanatory: when cast, test_summon
will silently summon 1 hostile mon_test_monster
on a random 3x3 location centered around the caster, with a duration of 62.5 seconds, after which it will disappear.
Typical attack
{
"id": "test_attack", // id of the spell, used internally. not translated
"type": "SPELL",
"name": "Ranged Strike", // name of the spell that shows in game
"description": "Deals damage to the target with 100% accuracy. Will always apply the status effect specified in 'effect_str'.",
"valid_targets": [ "ground", "hostile" ], // if a valid target is not included, you cannot cast the spell on that target.
"effect": "attack", // effects are coded in C++. A list is provided in this document of possible effects that have been coded.
"effect_str": "stunned", // varies, see table of implemented effects in this document
"min_damage": 10, // minimum damage (or "starting" damage)
"max_damage": 20, // maximum damage the spell can achieve
"damage_increment": 1.0, // How much damage increases per spell level increase
"min_range": 4, // range of the spell
"max_range": 4,
"base_casting_time": 500, // this is the casting time (in moves)
"min_duration": 200, // duration of spell effect in moves (if the spell has a special effect)
"max_duration": 300,
"duration_increment": 10, // How much longer the spell lasts per spell level
"damage_type": "stab" // type of damage
}
Explanation: classic damage spell with designated effect. After a 5 second cast, a lvl 1 test_attack
will deal 11 stab damage up to 4 tiles away, and stun the hostile
target for 2.1 seconds.
Note: valid_targets
has both ground
and hostile
so it can be targeted in an area with no line of sight.
Consecutive spell casting
{
"id": "test_combo", // id of the spell, used internally. not translated
"type": "SPELL",
"name": "Combo Strikes", // name of the spell that shows in game
"description": "Upon casting this spell, will also activate the spells specified on the 'extra_effects' in descending order.",
"flags": [ "SILENT", "RANDOM_DAMAGE", "RANDOM_AOE" ], // see "Spell Flags" in this document
"valid_targets": [ "hostile", "ground" ], // if a valid target is not included, you cannot cast the spell on that target.
"effect": "attack", // effects are coded in C++. A list is provided in this document of possible effects that have been coded.
"effect_str": "downed", // varies, see table of implemented effects in this document
"extra_effects": [ { "id": "test_atk1" }, { "id": "test_atk2" } ], // this allows you to cast multiple spells with only one spell
"min_damage": 7, // minimum damage (or "starting" damage)
"max_damage": 14, // maximum damage the spell can achieve
"damage_increment": 0.7 // damage increase per spell level
"min_aoe": 2, // area of effect
"max_aoe": 4,
"aoe_increment": 0.2, // how much wider the area of effect gets per spell level
"min_range": 10, // range of the spell
"max_range": 10,
"base_casting_time": 750, // this is the casting time (in moves)
"min_duration": 325, // duration of spell effect in moves (if the spell has a special effect)
"max_duration": 325,
"damage_type": "stab" // type of damage
}
Explanation: If you put two or more spells in extra_effects
, it will consecutively cast the spells: first extra_effects
, second extra_effects
, third extra_effects
, etc. If you wish to pick one at random, use the WONDER
flag (see below). Additionally, the extra spells will be cast at a level up to the level of the parent spell being cast, unless additional data is added to each test_atk#
.
Random spell casting
{
"id": "test_starter_spell", // id of the spell, used internally. not translated
"type": "SPELL",
"name": "Starter", // name of the spell that shows in game
"description": "Upon casting this spell, randomly selects one spell specified in 'extra_effects' to cast. The spell damage shows how many times it will randomly select from the list",
"flags": [ "SILENT", "WONDER", "RANDOM_DAMAGE" ], // see "Spell Flags" in this document
"valid_targets": [ "hostile" ], // if a valid target is not included, you cannot cast the spell on that target.
"effect": "attack", // effects are coded in C++. A list is provided in this document of possible effects that have been coded.
"extra_effects": [ // this allows you to cast multiple spells with only one spell
{ "id": "test_atk1" }, // id of the spell, used internally. not translated
{ "id": "test_atk2" }, // id of the spell, used internally. not translated
{ "id": "test_atk3" }, // id of the spell, used internally. not translated
{ "id": "test_atk4" }, // id of the spell, used internally. not translated
{ "id": "test_atk5" }, // id of the spell, used internally. not translated
{ "id": "test_atk6" } // id of the spell, used internally. not translated
],
"min_damage": 3, // minimum damage (or "starting" damage)
"max_damage": 5, // maximum damage the spell can achieve
"damage_increment": 0.2 // damage increase per spell level
"min_range": 10, // range of the spell
"max_range": 10
}
Explanation: The WONDER
flag does wonders by turning the spell into a dice: When cast, the main spell will pick the subspells in extra_effects
. The amount of “rolls” is specified via min_damage
and max_damage
plus the RANDOM_DAMAGE
flag. In this example, casting test_starter_spell
will cause any one of each test_atk#
subspells to be selected, this repeats 3 to 5 times.
Note: There must be a minimum of one spell in extra_effects
for the WONDER
flag to work properly.
Repeatedly cast the same spell
{
"type": "SPELL",
"id": "test_attack_repeat", // id of the spell, used internally. not translated
"name": "a spell", // name of the spell that shows in game
"description": "Upon casting this spell it will repeat the spell specified in `extra_effects` - the amount of repetitions is the interval `min_damage`-`max_damage` ",
"extra_effects": [ { "id": "test_attack" } ], // this allows you to cast multiple spells with only one spell
"flags": [ "SILENT", "WONDER", "RANDOM_DAMAGE" ], // see "Spell Flags" in this document
"valid_targets": [ "hostile" ], // if a valid target is not included, you cannot cast the spell on that target.
"effect": "attack", // effects are coded in C++. A list is provided in this document of possible effects that have been coded.
"effect_str": "target_message", // varies, see table of implemented effects in this document
"min_damage": 5, // minimum (starting damage)
"max_damage": 7, // maximum damage the spell can achieve
"damage_increment": 0.2 // damage increase per spell level
"min_range": 10, // range of the spell
"max_range": 10,
"min_duration": 1, // duration of spell effect in moves (if the spell has a special effect)
"max_duration": 1
}
Explanation: Notice a different approach for the WONDER
and RANDOM_DAMAGE
combo: min_damage
set to 5 and max_damage
set to 7 will cause the main spell to “roll” extra_effects
5 - 7 times, but this time there’s a single test_attack
. Because WONDER
has a 100% chance of picking a spell, test_attack
, will be repeated 5 to 7 times.
A spell that casts a note on the target and an effect on the caster
{
"id": "test_attack_note", // id of the spell, used internally. not translated
"type": "SPELL",
"name": "a note", // name of the spell that shows in game
"description": "This spell applies a harmless status effect to notify the player about the spell that the user has cast.",
"flags": [ "SILENT" ], // see "Spell Flags" in this document
"valid_targets": [ "hostile" ], // if a valid target is not included, you cannot cast the spell on that target.
"effect": "attack", // effects are coded in C++. A list is provided in this document of possible effects that have been coded.
"extra_effects": [ { "id": "sacrifice_spell", "hit_self": true }, { "id": "test_attack" } ], // this allows you to cast multiple spells with only one spell
"effect_str": "eff_test_note", // varies, see table of implemented effects in this document
"min_aoe": 6, // area of effect, or range of variance
"max_aoe": 6,
"min_duration": 1, // duration of spell effect in moves (if the spell has a special effect)
"max_duration": 1
}
Explanation: Here we have one main spell with two subspells: one on the caster and the other on the target. To do this, you must specify the ID of whatever spells you’re using with the extra_effects
field. In this case, sacrifice_spell
is stated with "hit_self": true
and will “hit” the caster, while the second spell will be cast as normal on the hostile
target. This is only necessary if we need an effect that is cast on a target and a secondary effect cast on the caster.
Monster spells
See Monster special attacks - Spells.
Custom Spell Experience Requirements
Spells may have custom formulas for their xp requirements to level by combining the get_level_formula_id
and exp_for_level_formula_id
fields. These fields both take the id of a jmath_function that they can use to calculate the xp requirement instead of the default formula. The given jmath functions must only have 1 argument (for the spell level or experience) and they should not make any calls to the alpha or beta talkers. Spell calculations are intended to be character agnostic, and while using functions such as u_skill(‘DEDUCTION’) will not error, it will always be calculated from the player characters data, even if for an npc or other uses of spell xp calculations. Also, keep in mind that changing the spell experience requirements of an existing spell will likely do strange things to any characters that already knew the spell before the xp requirements were changed.
Note: the exp_for_level_formula_id requires the total experience required for a spell level, not the difference in experience between the current and next level. IE, if a spell requires 1000 xp to level a level 10 spell should require 10,000 experience in its exp_for_level_formula_id, not 1,000.
Constant Spell Exp Requirement Example: { “id”: “test_spell”, “type”: “SPELL”, “name”: “test”, “description”: “constant spell experience formula.”, “valid_targets”: [ “hostile” ], “effect”: “attack”, “min_damage”: 1, “max_damage”: 1, “get_level_formula_id”: “magic_test_func_get_level”, “exp_for_level_formula_id”: “magic_test_func_exp_for_level” }, { “type”: “jmath_function”, “id”: “magic_test_func_get_level”, “num_args”: 1, “return”: “_0 / 1000” }, { “type”: “jmath_function”, “id”: “magic_test_func_exp_for_level”, “num_args”: 1, “return”: “1000 * _0” },
## Enchantments
Enchantments are another layer of enhancements, similar to `effect_type` and `mutation`. Unlike these which are carried by the avatar and similar entities, enchantments are bound to "containers" such as items, mutations, bionics and effects. This allows some flexibility in customizing effects and interactions according to the current state of the container, the spell being activated if any, what exactly is being granted, and more.
Identifier | Description
--- |---
`id` | Unique ID. Must be one continuous word, use underscores if necessary.
`has` | How an enchantment determines if it is in the right location in order to qualify for being active. `WIELD` when wielded in your hand, `WORN` when worn as armor, `HELD` when in your inventory.
`condition` | Determines the environment where the enchantment is active. `ALWAYS` is active always and forevermore, `ACTIVE` whenever the item, mutation, bionic, or whatever the enchantment is attached to is active, `INACTIVE` whenever the item, mutation, bionic, or whatever the enchantment is attached to is inactive. `DIALOG_CONDITION - ACTIVE` whenever the dialog condition in `condition` is true.
`hit_you_effect` | A spell that activates when you `melee_attack` a creature. The spell is centered on the location of the creature unless `"hit_self": true`, then it is centered on your location. Follows the template for defining `fake_spell`.
`hit_me_effect` | A spell that activates when you are hit by a creature. The spell is centered on your location. Follows the template for defining `fake_spell`
`intermittent_activation` | Spells that activate centered on you depending on the duration. The spells follow the `fake_spell` template.
`values` | Numbers that can be modified (see [list](#id-values)). `add` is added to the base value, `multiply` is **also added** and treated as percentage: 2.5 is +250% and -1 is -100%. `add` is always applied before `multiply`. Either `add` or `multiply` can be a variable_object/math expression (see [below](#variables) for syntax and application, and [NPCs](/NPCs.html) for the in depth explanation).
`skills` | A bonus or penalty to skills. Syntax is the same as for values, using the id of the skill name.
`emitter` | Grants the emit_id.
`modified_bodyparts` | Modifies the body plan (standard is human). `gain` adds body_part_id, `lose` removes body_part_id. Note: changes done this way stay even after the item/effect/mutation carrying the enchantment is removed.
`mutations` | Grants the mutation/trait ID. Note: enchantments effects added this way won't stack, due how mutations work.
`ench_effects` | Grants the effect_id. Requires the `intensity` for the effect.
All fields except for `type` and `id` are optional. This includes the otherwise obligatory `name` and `description`. If a name and description are set, they will be displayed in the EFFECTS tab.
There are two possible syntaxes. The first is by defining an enchantment object and then referencing the ID, the second is by directly defining the inline enchantment inside something (in this case, an item):
```json
{
"type": "enchantment",
"id": "ENCH_INVISIBILITY",
"name": { "str": "Invisibility" },
"description": "You are invisible. Just that.",
"condition": "ALWAYS",
"has": "WIELD",
"hit_you_effect": [ { "id": "AEA_FIREBALL", "hit_self": true, "once_in": 12 } ],
"hit_me_effect": [ { "id": "AEA_HEAL" } ],
"values": [ { "value": "STRENGTH", "multiply": 1.1, "add": -5 } ],
"skills": [ { "value": "computer", "add": 3 } ],
"emitter": "emit_AEP_SMOKE",
"modified_bodyparts": [ { "gain": "test_corvid_beak" }, { "lose": "torso" } ],
"mutations": [ "GILLS", "MEMBRANE", "AMPHIBIAN", "WAYFARER", "WILDSHAPE:FISH" ],
"ench_effects": [ { "effect": "invisibility", "intensity": 1 } ],
"melee_damage_bonus": [ // adds this amount of damage to attack; adding damage adds flat number to attacks, multiplier multiplies existing damage after adding
{ "type": "bash", "add": 10 }, // add 10 would straight add 10 damage of this type to each attack
{ "type": "cut", "add": -3 }, // add -3 would decrease any cut damage up to zero
{ "type": "heat", "multiply": 0.5 }, // 0.5 would mean damage be increased by 50%
{ "type": "acid", "multiply": -0.5 }, // -0.5 would mean 50% decrease of damage
{ "type": "necrotic", "add": 10, "multiply": 0.1 }, // any damage_type is supported
{ "type": "biological", "add": { "math": [ "Nemesis_iteration * -1" ] } } // supports math and stuff, works for both character/npcs and monsters. multiple `melee_damage_bonus`es of the same type do stack
],
"incoming_damage_mod": [ // any incoming damage would be modified by this value
{ "type": "bash", "add": 10 }, // adding would increase the taken damage (hurt more!)
{ "type": "stab", "add": -8 }, // negative adding would subtract damage taken by this amount
{ "type": "cut", "multiply": -0.5 }, // multiplication would multiply entire damage; -0.5 would result in 50% of damage being removed
{ "type": "bullet", "add": 1 }, // `"multiply": 1` would double all incoming bullet damage
{ "type": "electric", "add": { "math": [ "rand(3) * -1" ] } }, // supports math and stuff, works for both character/npcs and monsters. multiple `incoming_damage_mod`es of the same type do stack
{ "type": "acid", "add": 1, "multiply": 1 } // damage is subtracted before your physical armor (chain mail or similar), best works for decreasing incoming damage, as if character has additional layer of armor
],
"incoming_damage_mod_post_absorbed": [ // works exactly same as incoming_damage_mod, but is applied after your physical armor (chain mail or similar), so best works for adding damage, as if character is specifically suseptible to this type of damage
{ "type": "bash", "add": 10 },
{ "type": "stab", "add": -8 },
{ "type": "cut", "multiply": -0.5 },
{ "type": "bullet", "add": 1 },
{ "type": "electric", "add": { "math": [ "rand(3) * -1" ] } },
{ "type": "acid", "add": 1, "multiply": 1 }
],
"intermittent_activation": {
"effects": [
{
"frequency": "1 hour",
"spell_effects": [
{ "id": "AEA_ADRENALINE" }
]
}
]
}
"special_vision": [ // defines creatures (monsters or NPC) you can see in some irregular ways, thermal, supernatural, you name it
{
"condition": { "and": [ { "npc_has_flag": "ELECTRIC" }, "u_see_npc" ] }, // this need to return true to see the critter; u is character, npc is critter being watched
"distance": 20, // how far special_vision is applied. for technical reasons special_vision do not stack, having multiple special_visions of the same nature will not result in special_visions of bigger distance
"precise": true, // if true, you can aim the critter as if it's visible, if false, your ability to aim it would be limited. Default false
"ignores_aiming_cone": true, // if true, aiming the gun do not obscure sensing the creature. Default false
"descriptions": [ // if condition is true, this will assign a dedicated id (used for tiles) and text, depending on text_condition
{
"id": "infrared_creature_tiny",
"text_condition": { "math": [ "n_val('size') == 1" ] }, // optional condition for this specific id to be used. Default true. Be sure to not use condition that can change in between moves (like any math with random result, `rand(1)` etc);
// while text with tags is stored and re-evaluated only when cursor changes its position, tiles are re-evaluated every frame, even on pause
"text": "Message 1.",
"symbol": "@", // symbol, that would be used in ascii. Optional, default is `?`
"color": "c_magenta" // color of symbol, that would be used in ascii. Optional, default is `c_red`
},
{
"id": "infrared_creature_small",
"text_condition": { "math": [ "n_val('size') == 2" ] },
"text": "Message 2."
},
{
"id": "infrared_creature_medium",
"text_condition": { "math": [ "n_val('size') == 3" ] },
"text": "Message 3. <fuck_you>" // tags are also supported
},
{
"id": "infrared_creature_large",
"text_condition": { "math": [ "n_val('size') == 4" ] },
"text": "Message 4."
},
{
"id": "infrared_creature_huge",
"text_condition": { "math": [ "n_val('size') == 5" ] },
"text": "Message 5."
},
{
"id": "infrared_creature_medium",
"text": "Last message." // descriptions are read in order they are in json, adding dummy condition that always return true can be used as a fallback if none previous condition matches;
// otherwise default description is `You sense a creature here.`, and default id is `infrared_creature`
}
]
}
]
}
{
"copy-from": "mring_silver",
"type": "TOOL_ARMOR",
"id": "mring_wizardry_lesser",
"price_postapoc": "2000 USD",
"name": { "str": "lesser ring of wizardry", "str_pl": "lesser rings of wizardry" },
"description": "A thin silver band ring, engraved with two sealed scrolls. Increases mana capacity somewhat.",
"relic_data": { "passive_effects": [ { "has": "WORN", "condition": "ALWAYS", "values": [ { "value": "MAX_MANA", "add": 400 } ] } ] }
}
The relic_data
field
As seen in the last example, enchantments are added to the item as passive_effects
inside the relic_data
field. Items with this data are turned into a relic or artifact, being displayed as magenta inside the inventory view.
Also supported is charge_info
, which allows automatic charge regeneration. This in turn enables active magical items that cast spells on use:
...
"use_action": { "type": "cast_spell", "spell_id": "conj_throwing_blade3", "no_fail": true, "level": 1, "need_worn": true },
"extend": { "flags": [ "NO_UNLOAD", "NO_RELOAD" ] },
"charges_per_use": 1,
"relic_data": { "charge_info": { "recharge_type": "periodic", "time": "1 h", "regenerate_ammo": true } },
"pocket_data": [ { "pocket_type": "MAGAZINE", "holster": true, "ammo_restriction": { "crystallized_mana": 5 } } ]
...
The item consumes 1 charge per spell cast. It can’t be recharged or unloaded, relying on ammo regeneration over time for use.
Another example is a GUN
type item (e.g. a firearm). As this is a weapon that consumes ammo per use, use_action
can be omitted:
...
"clip_size": 5,
"flags": [ "NO_UNLOAD", "NO_RELOAD" ],
"relic_data": {
"charge_info": { "regenerate_ammo": true, "recharge_type": "periodic", "time": "20 s" },
"passive_effects": [ { "id": "ENCH_INVISIBILITY" } ]
},
"pocket_data": [ { "pocket_type": "MAGAZINE", "rigid": true, "ammo_restriction": { "ammo_magic_bullet": 5 } } ],
...
This weapon consumes “magic bullet” ammo every time it’s fired. Note how charge_info
and passive_effects
can be used together.
The field charge_info
supports the following:
Identifier | Description |
---|---|
regenerate_ammo | true . |
recharge_type | Can be one of: lunar , periodic , solar_cloudy , solar_sunny , or none . |
time | Time required per charge. |
recharge_condition | (optional) Similar to has from enchantments: can be one of held , worn , wield . If omitted, the item recharges regardless, even if dropped. |
Variables
From now, EOC variables can be used inside enchantments, including predefined (see NPCs.md for examples), custom variables or math equasions. Here are some examples:
{
"type": "enchantment",
"id": "MON_NEARBY_STR",
"has": "WIELD",
"condition": "ALWAYS",
"values": [ { "value": "STRENGTH", "add": { "math": [ "u_val('dexterity') + 1" ] } } ]
}
This enchantment adds the dexterity value to strength plus one: a character with str 8 and dex 10 will result with str 19 and dex 10.
{
"type": "enchantment",
"id": "MON_NEARBY_LUMINATION",
"has": "WIELD",
"condition": "ALWAYS",
"values": [
{
"value": "LUMINATION",
"add": { "math": [ "u_monsters_nearby('radius': 25) * 20" ] }
}
]
}
This enchantment checks the amount of monsters near the character (in a 25 tile range), then multiplies that number by 20, and adds the value as lumination: more monsters nearby = more light produced.
{
"type": "enchantment",
"id": "MOON_STR",
"has": "WORN",
"condition": "ALWAYS",
"values": [ { "value": "STRENGTH", "add": { "math": [ "IS_UNDER_THE_MOON * 4" ] } } ]
}
Here’s an enchantment that relies on a custom variable check, the full power of EOCs in your hand.
First, the custom variable IS_UNDER_THE_MOON is set behind the scenes, it checks if the character is under the moon’s rays (by a combination of { "not": "is_day" }
and "u_is_outside"
): if true value is 1, otherwise is 0. Then, the custom variable is used inside a math operation that multiplies the truth value by 4: character is granted [ 1 * 4 ] = 4 additional strength if outside and during the night, or [ 0 * 4 ] = 0 additional strength otherwise.
condition
field support any EoC condition
{
"type": "enchantment",
"id": "BITE_STR",
"has": "WIELD",
"condition": { "math": [ "u_effect_intensity('bite', 'bodypart': 'torso') > 1"] },
"values": [ { "value": "STRENGTH", "add": 5 } ]
}
ID values
The following is a list of possible enchantment values
:
Character status value | Description |
---|---|
ARMOR_ALL | Gives this amount of protection against any damage type except one with “no_resist”: true. For more precise changes use incoming_damage_mod or item_armor_bonus |
ATTACK_NOISE | Affects the amount of noise you make while melee attacking. |
ATTACK_SPEED | Affects attack speed of item, even if it’s not the one you’re wielding, and throwing cost (capped at 25 moves). "add": 10 adds 10 moves to each attack (makes it longer), "add": -10 makes each attack faster for 10 moves; "multiply": 1 doubles the speed of each attack |
AVOID_FRIENDRY_FIRE | Flat chance for your character to avoid friendry fire if there is a friend in the line of fire. From 0.0 (no chance) to 1.0 (never frindly fire). |
BANDAGE_BONUS | Affects the bandages_power you have when applying medicine. |
BIONIC_MANA_PENALTY | changes how big the mana penalty for having bionic energy is (default ratio is 1 kj removes 1 mana point). better to use with multiply , using add just adds or removes flat amount of mana no matter of energy level. "multiply": 1 double the ratio (1 kj removes 2 mana points), "multiply": -0.5 halves it |
BIONIC_POWER | Adds bionic power storage in millijoules (“add”: 1000000 adds 1 kJ) |
BLEED_STOP_BONUS | Affects the bleed level when applying medicine. |
BODYTEMP_SLEEP | Amount of warmth (in celcius) added to you when you sleep. Default is 0, so better to use add |
BONUS_BLOCK | Affects the number of blocks you can perform. |
BONUS_DODGE | Affects the number of dodges you can perform. Do not confuse with DODGE_CHANCE |
CARDIO_MULTIPLIER | Affects total cardio fitness by this amount. Since it’s a percent, using multiply is recommended. |
CARRY_WEIGHT | Affect the summary weight player can carry. "add": 1000 adds 1 kg of weight to carry. |
CASTING_TIME_MULTIPLIER | Same as mutation casting_time_multiplier field, changes your casting speed. Since it’s a percent, using multiply is recommended. "multiply": 2" triples the casting speed |
COMBAT_CATCHUP | Affects the rate at which you relearn combat skills (multiplier). |
CONSUME_TIME_MOD | Changes how long you consume FOOD and DRINK or something from chems itemgroup. "add": 10 adds 10 seconds to food consumption, multiply: 1 doubles it |
CLIMATE_CONTROL_HEAT | Moves body temperature up towards comfortable by number of warmth units up to value. |
CLIMATE_CONTROL_CHILL | Moves body temperature down towards comfortable by number of warmth units up to value. |
CRAFTING_SPEED_MULTIPLIER | Changes your crafting speed. Since it’s a percent, using multiply is recommended. Positive values decrease crafting time, negative values increase it. |
DEXTERITY | Affects the dexterity stat. |
DISINFECTANT_BONUS | Affects the disinfectant_power you have when applying medicine. |
DODGE_CHANCE | Modifies the probability to dodge an attack. Default is 0, so better to use add |
EFFECTIVE_HEALTH_MOD | If this is anything other than zero (which it defaults to) you will use it instead of your actual health mod. |
EQUIPMENT_DAMAGE_CHANCE | Modifies the likelihood that weapons and armor take durability damage. Since it’s a percent, using ‘multiply’ is recommended. Positive values increase likelihood of damage while negative values decrease likelihood. multiply : -1 and below result in indestructible equipment. |
EXTRA_ELEC_PAIN | Multiplier on electric damage received, the result is applied as extra pain. |
EVASION | Flat chance for your character to dodge incoming attacks regardless of other modifiers. From 0.0 (no evasion chance) to 1.0 (100% evasion chance). |
FALL_DAMAGE | Affects the amount of fall damage you take. |
SLEEPINESS | Affects how fast your sleepiness grows over time - bigger value makes you tired faster. Since it’s a percent, using multiply is recommended. |
SLEEPINESS_REGEN | Affects how much of your sleepiness and sleep deprivation drops when resting. Since it’s a percent, using multiply is recommended. |
FAT_TO_MAX_HP | Changes the amount of HP, that is given to you by your fat. Formula is ((your_calories/7716.17)/((your_height_in_cm/100)^2))*hitsize_of_all_non_bionic_bodyparts . Using add works just as adding HP, so use multiply instead |
FOOTSTEP_NOISE | |
FORCEFIELD | Chance your character reduces incoming damage to 0. From 0.0 (no chance), to 1.0 (100% chance to avoid attacks). |
HEALTHY_RATE | How much of your health is changed daily. "multiply": -1 stops any changes in health |
HEARING_MULT | How well you can hear. Remember that increased hearing means you would have a bigger “noise” written in UI; default step noise of 6, multiplied 10 times, would show it as 60 |
HUNGER | Affects how fast your hunger level changes. Do not affect actual calorie burn, the METABOLISM field is responsible for this |
INTELLIGENCE | Affects the intelligence stat. |
KCAL | Same as bio_digestion effect, increases the amount of calories obtained the food. |
KNOCKBACK_RESIST | The amount knockback effects you, 0 is the regular amount, -100 would be double effect, 100 would be no effect. |
KNOCKDOWN_RESIST | The amount knockdown effects you, currently only having 100 or greater knockdown_resist makes you immune to knockdown. |
LEARNING_FOCUS | Amount of bonus focus you have for learning purposes. |
LUMINATION | Character produces light. |
MAX_HP | |
MAX_MANA | |
MAX_STAMINA | |
MELEE_DAMAGE | Adds damage to melee attacks |
MELEE_TO_HIT | Modifies melee attacks’ to_hit . add is recommended since to_hit can be below 0 and has a small typical range. |
MELEE_STAMINA_CONSUMPTION | Changes amount of stamina used when swing in melee; stamina consumption is a negative value, so "add": 100 decreases amount of stamina consumed, when "add": -100 increases it; "multiply": 1 increases, "multiply": -0.5 decreases it. Can’t be bigger than -50. |
MENDING_MODIFIER | Changes the speed of your limb mend. Since it’s a percent, using multiply is recommended. |
METABOLISM | Multiplier for metabolic_rate_base , which respond for default bmi rate; Formula for basic bmi is metabolic_rate_base * ( (weight_in_kg / 10 ) + (6.25 * height) - (5 * age) + 5 ) ; Since it’s a percent, using multiply is recommended; Since metabolism is directly connected to weariness, at this moment decreasing it makes you more weary the less metabolism you have; zero metabolism (multiply: -1 ) is handled separately, and makes you never weary. |
MOD_HEALTH | If this is anything other than zero (which it defaults to) you will to mod your health to a max/min of MOD_HEALTH_CAP every half hour. |
MOD_HEALTH_CAP | If this is anything other than zero (which it defaults to) you will cap your MOD_HEALTH gain/loss at this every half hour. |
MOVE_COST | |
MUT_INSTABILITY_MOD | Modifies your instability score, which affects the chance to get bad mutation (scales with amount of good mutations you have, capping at 67%, check Character::roll_bad_mutation for more information). add: 1 would be equal to having 1 good mutation more, increasing the chance to get bad mutation, add: -1 would be like you have one good mutation less, decreasing the chance to get bad mutation. |
MOVECOST_FLATGROUND_MOD | How many moves you spend to move 1 tile on flat ground; shown in UI |
MOVECOST_OBSTACLE_MOD | How many moves you spend to move 1 tile, if this tile has a movecost more than 105 moves; not shown in UI |
MOVECOST_SWIM_MOD | How many moves you spend to move 1 tile in water; not shown in UI |
MOVEMENT_EXERTION_MODIFIER | Affects how much physical exertion (activity_level) is required for the player to move a single tile. The value this affects is technically a float, so addition and multiplication operations can be performed on it, but player-facing effects only happen at specific whole-integer breakpoints. This can most easily be seen in \data\json\ui\activity.json . |
NIGHT_VIS | How well you can see in darkness. ADD adds tiles, so "ADD": 3 increases night vision distance by 3 tiles. |
OBTAIN_COST_MULTIPLIER | Modifier for pulling an item from a container, as a handling penalty or bonus. "add": 100 add 100 additional moves to item wield (1 second) |
OVERKILL_DAMAGE | multiplies or contributes to the damage to an enemy corpse after death. The lower the number, the more damage caused. |
OVERMAP_SIGHT | Increases the amount of overmap tiles you can see around. |
PAIN | When gaining pain the amount gained will be modified by this much. You will still always gain at least 1 pain. |
PAIN_PENALTY_MOD_STR | Amount of this stat you lose from pain. Default value is (pain*0.005)*max_str . Can’t be lower than 1 |
PAIN_PENALTY_MOD_DEX | Amount of this stat you lose from pain. Default value is (pain*0.0075)*max_dex . Can’t be lower than 1 |
PAIN_PENALTY_MOD_INT | Amount of this stat you lose from pain. Default value is (pain*0.01)*max_int . Can’t be lower than 1 |
PAIN_PENALTY_MOD_PER | Amount of this stat you lose from pain. Default value is (pain*0.01)*max_per . Can’t be lower than 1 |
PAIN_PENALTY_MOD_SPEED | Amount of speed you lose from pain. Default value is pain^0.7 . Can’t be bigger than 50 speed. |
PAIN_REMOVE | When pain naturally decreases every five minutes the chance of pain removal will be modified by this much. You will still always have at least a chance to reduce pain. |
PERCEPTION | Affects the perception stat. |
PHASE_DISTANCE | Distance the character is able to teleport through impassible terrain. Values less than 1 do nothing and the max distance is 48 tiles. |
POWER_TRICKLE | Generates this amount of millijoules each second. Default value is zero, so better to use add |
RANGE | Modifies your characters range with firearms |
RANGED_ARMOR_PENETRATION | Adds armor penetration to ranged attacks. |
RANGED_DAMAGE | Adds damage to ranged attacks. |
RANGE_DODGE | Chance to dodge projectile attack, no matter of it’s speed; Consumes dodges similarly to melee dodges, and fails, if character has no dodges left. add and multiply behave equally. add: 0.5 would result in 50% chance to avoid projectile |
READING_EXP | Changes the minimum you learn from each reading increment. |
READING_SPEED_MULTIPLIER | Changes how fast you can read books; Lesser value means faster book reading, with cap of 1 second. |
RECOIL_MODIFIER | Affects recoil when shooting a gun. Positive value increase the dispersion, negative decrease one. |
REGEN_HP | Affects the rate you recover hp, at all time, both natural and using medicine. Default 1; Negative value (so multiply: -1.1 or smaller) would cause character lose it’s HP all the time |
REGEN_HP_AWAKE | Affects the rate you recover hp when you do not sleep. Default value is 0, meaning if this enchantment is not specified, character can not regenerate hp when awake. Negative value causes character to lost HP over time when awake. Since it’s a percent, using multiply is recommended; multiply: 0.2 would make character able to regenerate at 20% of it’s default regen |
REGEN_MANA | |
REGEN_STAMINA | |
SCENT_MASK | Amount added to your scent target scent value (default 500, assigned by scent_intensity mutation field); "add": 100 makes character a bit more smelly |
SHOUT_NOISE | Changes how loud your shouts are (default 10) |
SHOUT_NOISE_STR_MULT | Modifies the shout_multiplier , that affect how much your strength affects noise level (default 2, meaning one point of strength adds 2 units of noise ) |
SKILL_RUST_RESIST | when add , chance / 100 to resist skill rust; when multiply , multiplier for skill rust amount - the smaller, the less experience you will rust |
SLEEPY | The higher this the easier you fall asleep. |
SOCIAL_INTIMIDATE | Affects your ability to intimidate. |
SOCIAL_LIE | Affects your ability to lie. |
SOCIAL_PERSUADE | Affects your ability to persuade. |
SPEED | Affects your base speed. |
STAMINA_REGEN_MOD | Modifier to how fast you regen your stamina. It is a percent with default value of 0, so multiply is useful only in combination with add . add: 0.1 makes regen 10% bigger, add: 1 doubles it |
STEALTH_MODIFIER | Amount to be subtracted from player’s visibility range, capped to 60. Negative values work, but are not very effective due to the way vision ranges are capped. |
STOMACH_SIZE_MULTIPLIER | Changes how much food you can consume at once. "add": 1000 adds 1 L to stomach size |
STRENGTH | Affects the strength stat. Formula for all stat affecting enchantments are (base_stat + enchantment_addition) * (enchantment_multiplier + 1) . Str 8 with enchantment add 2, multiply 1 result in (8+2) * (1+1) = 10 * 2 = 20 str |
SWEAT_MULTIPLIER | Affects how much your body can sweat. Affects all bodyparts at once. Since it’s a percent, using multiply is recommended. |
THIRST | |
THROW_STR | Increases your strength for throwing purposes. Not limited by your throwing skill (you still throw it as precise as your skill allows you, just further). Only additive. Rule of thumb: one additional point of strength allow you to throw 113 g object 10 tiles further, or 1130 g object 1 tile further, limited by [ str * 3 + skill ]. Full calculations are in Character::throw_range |
THROW_DAMAGE | Increases the damage of any thrown projectile. add adds this amount of damage to the projectile as bash damage, multiply would increase all projectile damage, not only bash type. |
UGLINESS | Affects your ugliness stat, which affects NPCs’ initial opinion of you. |
VITAMIN_ABSORB_MOD | Increases amount of vitamins obtained from the food |
VOMIT_MUL | Affects your chances to vomit. |
WEAKNESS_TO_WATER | Amount of damage character gets when wet, once per second; scales with wetness, being 50% wet deal only half of damage; negative values restore hp; flat number with default value of 0, so multiply is useful only in combination with add ; Works with float numbers, so "add": -0.3 would result in restoring 1 hp with 30% change, and 70% chance to do nothing |
WEAKPOINT_ACCURACY | Increases the coverage of every weakpoint you hit, therefore, increasing chances to hit said weakpoint. Works only if weakpoint has "is_good": true (all weakpoints have it true by default) |
WEAPON_DISPERSION | Positive value increase the dispersion, negative decrease one. |
Enchanted item value | Description |
---|---|
ITEM_ATTACK_SPEED |
Enchantments on monsters
A small subset of enchantments can be applied to monsters via effects. These are listed below:
Character status value | Description |
---|---|
REGEN_HP | Affects the rate the monster recovers hp. |
VISION_RANGE | Affects monster vision range, both day and night one. |
SPEED | Affects the base speed of the monster. |
LUMINATION | Affects monster luminance |
Enchantment value examples
{ "incoming_damage_mod": [ { "type": "electric", "add": -20 } ] }, // subtracts 20 points of incoming electrical damage
{ "value": "ATTACK_SPEED", "add": -60 } // subtracts 60 attack moves, making the attacker faster
{ "incoming_damage_mod": [ { "type": "cold", "multiply": -0.4 } ] } // subtracts 40% of incoming cold damage
{ "incoming_damage_mod": [ { "type": "heat", "multiply": 0.4 } ] } // increases damage taken from fire by 40%
{ "incoming_damage_mod": [ { "type": "cut", "add": 2 } ] } // increases incoming cut damage by 2
{ "incoming_damage_mod": [ { "type": "biological", "multiply": -1.4 } ] } // subtracts 100 percent of incoming biological damage
{ "incoming_damage_mod": [ { "type": "acid", "multiply": 1.4 } ] } // increases incoming acid damage by 140%