Skip to content

Action List Conditional Expressions

0xjc edited this page Jan 29, 2023 · 6 revisions

Conditional expressions

Conditional expressions allow complex composition of conditions, and are usually added to action priority list through the "if" keyword, using the following syntax:

# Uses faerie fire if:
# * There are less than three stacks
# * And neither "expose armor" nor "sunder armor" are on the target
actions+=/faerie_fire,if=debuff.faerie_fire.stack<3&!(debuff.sunder_armor.up|debuff.expose_armor.up)

The same syntax is used for the interrupt_if action modifier and the the sec modifier for wait_fixed actions.

Player Expressions are also used for raid event player_if filter.

Operators

Simple example

Consider the following APL line:

actions+=/faerie_fire,if=debuff.faerie_fire.stack<3&!(debuff.sunder_armor.up|debuff.expose_armor.up)

Given that & means "AND", ! means "NOT", and | means "OR", we can replace the symbols with their English equivalents:

debuff.faerie_fire.stack LESS THAN 3 AND NOT(debuff.sunder_armor.up OR debuff.expose_armor.up)

Therefore, this APL line says to cast Faerie Fire if there are less than three stacks of Faerie Fire and neither Sunder Armor nor Expose Armor are up.

Operator precedence

To be able to read APL expressions, it is important to know the precedence rules (i.e. the order of operations). An operator with higher precedence binds more tightly to its operands: for example, 2+3*5 is evaluated as 2+(3*5) because * has higher precedence than +. Similarly, a&b|c&d is evaluated as (a&b)|(c&d) because & has higher precedence than |.

Within the same precedence class, Simulationcraft always uses left-to-right associativity: for example, 4-5+6 is evaluated as (4-5)+6 because +, - have the same precedence but the 4-5 appears first when reading left to right.

Here is the full list of operators in Simulationcraft, in order from highest to lowest precedence (i.e. entries listed first bind more tightly):

  • Function calls: floor() ceil()
  • Unary operators: + - @ !
  • Multiplication, division, and modulus: * % %%
  • Addition and subtraction: + -
  • Max and min: <? >?
  • Comparison operators: = != < <= > >= ~ !~
  • Logical AND: &
  • Logical XOR: ^
  • Logical OR: |

Note that this is very similar to the operator precedence in other programming languages such as C++ or Javascript.

Arithmetic operators

All Simulationcraft expressions evaluate to double-precision floating-point numbers. The following operators perform floating-point arithmetic.

  • + is the addition operator: x+y evaluates to the sum of x and y. As a unary operator, it is a no-op: +x evaluates to the same as x.
  • - is the subtraction operator: x-y evaluates to x minus y. As a unary operator, it negates the operand: -x evaluates to the negation of x.
  • * is the multiplication operator: x*y evaluates to the product of x and y.
  • % is the division operator: x%y evaluates to the floating-point quotient of x by y (for example, 3%2 equals 1.5). Note that / could not be used for division since it is already being used as the separator token.
  • %% is the modulus operator: x%%y evaluates to the floating-point remainder when x is divided by y (for example, 9%%2.5 equals 1.5). The result has the same sign as x.
  • @ is the absolute value operator: @x evaluates to the absolute value of x.
  • <? is the max operator: x<?y evaluates to the greater of x and y.
  • >? is the min operator: x>?y evaluates to the lesser of x and y.
  • floor() is the floor function: floor(x) evaluates to the greatest integer value that is less than or equal to x. Note that this is different from truncation for negative operands: for example, floor(-2.5) equals -3.
  • ceil() is the ceiling function: ceil(x) evaluates to the least integer value that is greater than or equal to x.

Comparison operators

  • = is the equality operator: x=y evaluates to 1 if x is equal to y, and 0 otherwise.
  • != is the inequality operator: x!=y evaluates to y if x is not equal to y, and 0 otherwise.
  • < <= > >= are the relational operators: these should be self-explanatory given the above.

Logical operators

Logical operators work with truth values. Zero is the same as false; any nonzero value is considered truthy.

  • & is the logical AND operator: x&y evaluates to 1 if both x and y are nonzero, and 0 otherwise.
  • | is the logical OR operator: x|y evaluates to 1 if either x or y or both are nonzero, and 0 otherwise.
  • ^ is the logical XOR operator: x^y evaluates to 1 if either x or y, but not both, are nonzero, and 0 otherwise.
  • ! is the logical NOT operator: !x evaluates to 1 if x is zero, and 0 otherwise.

Important note on booleans

There is no distinct boolean type, as Simulationcraft expressions are all evaluated as floating-point numbers. It is common for APLs to coerce values between boolean and arithmetic contexts freely.

  • In a boolean context, 0 is considered false, while anything nonzero is considered true. For example, 3&4 evaluates to 1 since 3 and 4 are being used in a boolean context (operands to &) and are both considered true. Logical operators and comparisons will always output either 0 or 1 regardless of their operands.
  • Example: guillotine,if=cooldown.demonic_strength.remains. The condition is equivalent to cooldown.demonic_strength.remains!=0 and it is synonymous with Demonic Strength being on cooldown: !cooldown.demonic_strength.ready means basically the same thing.
  • On the flip side, logical 0 or 1 values can be used in an arithmetic context. For example, (a==b)*c will evaluate to c if a is equal to b, or 0 otherwise.
  • Example: variable,name=next_tyrant,op=set,value=14+talent.grimoire_felguard+talent.summon_vilefiend. Since talent checks evaluate to 0 or 1 depending on whether the talent is learned, this logic starts with an initial timer of 14 and adds 1 for each of these talents learned.

SpellQuery operators

These operators may only be used in SpellQuery.

  • ~ is the string in operator: s~t evaluates to true if s is a substring of t (case-insensitive).
  • !~ is the string not in operator: s!~t evaluates to true if s is not a substring of t (case-insensitive).

Operands

Actions

Action properties are related to the action you want to perform. They require no specific syntax. The available properties are:

# If sw:p has not been used so far or its last occurrence missed, then recast it, provided it's not active or close to expiration.
actions+=/shadow_word_pain,if=(!ticking|dot.shadow_word_pain.remains<gcd+0.5)&miss_react
  • execute_time returns whichever is greater: gcd or cast_time
  • gcd is the amount of gcd-time it takes for the current action to execute, accounting for haste.
  • gcd.remains returns the amount of time until the player's gcd is ready
  • gcd.max returns the hasted gcd-time of the PLAYER, not the action. This is useful for wait commands.
  • cast_time, is the time (in seconds) which the action will need to execute.
  • cooldown, for a spell or an usable item, is the initial cooldown duration, in seconds.
  • ticking, for a dot or a hot, will return 1 if the dot is currently active on your target, 0 otherwise.
  • ticks, for a dot or a hot, will return the number of ticks done so far since the last time the dot was refreshed.
  • ticks_remain, for a dot or a hot, will return the number of remaining ticks before the dot expires.
  • remains, for a dot or a hot, will return the remaining time, in seconds, before the dot expires. It does NOT apply to buffs and debuffs.
  • full_recharge_time, will return time until all charges of a spell are fully recharged
  • tick_time, for a dot or a hot, is the time in seconds between ticks, if cast using your current haste.
  • travel_time is the timespan, in seconds, the spell will need to reach its target (a fireball for example).
  • miss_react is 1 if the last occurrence of this action missed or has never been used, 0 otherwise, taking into account the reaction time, as specified through reaction_time. If you specified 0.5s and the miss only occurred 0.3s ago, this property will return 1.
  • cooldown_react is 1 if the cooldown has elapsed, the reaction time for the abrupt reset has elapsed, or the cooldown was reset in a determined way. The expression returns 0 if the reset of the cooldown happened early (e.g., due to a proc or another action) and the cooldown specific reaction time to the abrupt reset of the cooldown has not yet elapsed.
  • cast_delay is 1 if sufficient time has elapsed after the previous player executable action's cast time (including gcd). The time is controlled by two separate parameters, brain_lag and brain_lag_stddev.
  • multiplier is the player's highest damages/healing multiplier for the schools used by the action. A buff that would increase your damages by 20% would give you a 1.2 multiplier. All damages multipliers are multiplicative (5% and 20% give 1.2*1.05).
  • casting evaluates to 1 if the action is currently being cast, 0 otherwise
  • channeling evaluates to 1 if the action is currently being channeled, 0 otherwise
  • executing evaluates to 1 if the action is currently being cast or channeled, 0 otherwise
# Consumes a potion as soon as we have enough stacked buffs to reach a minimum 30% multiplier.
actions+=/golemblood_potion,if=multiplier>1.3

Buffs and debuffs

The character's buffs and the raid-friendly debuffs on your target (sunder armor, curses, etc) can all be used through the following syntax: buff.<aura_name>.<aura_property> (recommended) or aura.<aura_name>.<aura_property> (not recommended, see below). As usual, for the aura name, use underscores instead of white spaces and ignore non-alphanumeric characters.

SimC doesn't have separate concepts of "buff" and "debuff" - we have one generic buff object, so when you apply e.g. a mortal_strike debuff to the target, you're really giving them a "buff" named mortal_strike. However, we do have APL syntax for both buffs and debuffs based on where you want to search for a given buff.

For example, debuff.mortal_strike.up will check the target for a mortal_strike buff first, and if it doesn't find anything, will then check the player's buffs. The conditional buff.mortal_strike.up, on the other hand, will only check the player, and give up if it doesn't find anything.

# Here is how we can check the number of stacks of faerie fire on the target.
actions+=/faerie_fire,if=debuff.faerie_fire.stack<3
# This is how we would check for the presence of a beneficial buff on the player.
actions+=/avengers_shield,if=buff.grand_crusader.react

Regarding the list of buffs...

  1. The "buff "list will contains a smartly filtered list of buffs and target debuffs. For example, if many warriors are in the raid, every one of them will only have his own version of colossus smash, so that he cannot react to a CS from another warrior. However, this list will contain "sunder_armor", "expose_armor" and "faerie_fire" and those ones will be updated whenever someone in the raid applies it on the target. Finally, it will contains all buffs the character has.
  2. The "aura" list, however, is kinda strange and it is advised you not use it unless you're sure you should.
  3. The "buff" list also contains special buffs such as:
  • "bleeding" (target is bleeding)
  • "casting" (character is casting)
  • "flying" (character is flying)
  • "raid_movement" (character is moving because of a "movement" raid event)
  • "self_movement" (character is moving because of a start_move action)
  • "stunned" (character is stunned)
  • "invulnerable" (target is invulnerable)
  • "vulnerable" (target is vulnerable)
  • "mortal_wounds" (character has the mortal strike debuff)
  • "damage_taken" (Since Simulationcraft 6.0.1) (stacking debuff that causes 1% extra damage taken per stack, see enemy action options).
  1. The "buff" list also contains buffs for
  • potions (speed_potion, tolvir_potion, etc)
  • trinkets (crushing_matter, etc)
  • stances (defensive_stance, etc)
  • shapes (moonkin_form, etc)
  • enchants (landslide_mh for main hand, etc).
  1. If you want to check the complete "buff" and "aura" lists, examine the application's output (not the html report). Under every player's detail, you will see a listing of constant and dynamic buffs: those are the content of the "buff" list. Now, past the players' and targets' details, is a "Auras:" label, with also constant and dynamic buffs: those are the content of the "aura" list.
  2. (Since Simulationcraft 6.0.1) You can also refer to any potion-based buff by using the potion alias as the buff name (i.e., buff.potion.up).

For every buff/debuff, the available properties are:

  • remains is the remaining time, in seconds, for this debuff. Debuffs with a an infinite duration have a zero value.
  • cooldown_remains is the remaining time, in seconds, for spell's cooldown triggering this buff.
  • up is 1 when the debuff currently exists and is active, 0 otherwise.
  • down is 0 when the debuff currently exists and is active, 1 otherwise.
  • stack is the number of stacks of this buff.
  • max_stack is the maximum possible stack size of this buff.
  • stack_pct returns 100 * stack / max_stack.
  • react is the number of stacks of this buff, taking into account your reaction time, as specified through reaction_time. If you specified 0.5s, the returned value will be the number of stacks there were 0.5s ago.
  • value is the buff's value. For a 1200 spell power buff, for example, it will be 1200.

Trinket Procs

These commands can be combined with trinkets if you wish to create action lists that react to trinket procs.

Syntax: trinket[.1|2|name].has_stacking_stat_.stat.specific stat._buff expression

All commands that work with buffs will also work with this.

The slot (1, 2, or Since Simulationcraft 8.2.0 release 1, name) is not required, and can be skipped. If skipped, it will scan both trinket slots. If both trinkets satisfy the buff expression, the maximum value of the buff expressions will be chosen. If tokenized trinket name is specified, and the name is invalid, the item is not a trinket, or the item is not found on the actor, an expression that always returns 0 will be generated.

Examples:

# Flags as true if the trinket in slot 2 has the ability to proc an agility stacking proc.
trinket.2.has_stacking_stat.agility

# Flags as true if any trinket has a non-stacking strength proc.
trinket.has_stat.strength

# Flags as true if the trinket in slot 1 has agility proc with more than 10 stacks.
trinket.1.stacking_stat.agility.stack>10

# Flags as true if the trinket in slot 2 procs for over 3000 mastery, non-stacking.
trinket.2.stat.mastery.value>3000

# Flags as true if any trinket has a non-stacking strength proc with more than 5 seconds left on the internal cooldown.
trinket.stat.strength.cooldown_remains>5

They can be combined with other conditions:

# Save recklessness if the cooldown on the trinket proc is greater than 15, and the cooldown on skull banner is below 20.
actions+=/recklessness,if=trinket.stat.strength.cooldown_remains>15&cooldown.skull_banner.remains<20

Trinket Cooldowns

You can also express trinket cooldowns through the expression system. Generic syntax is trinket.(has_|)cooldown.<cooldown_expr>, where <cooldown_expr> refers to any cooldown expression we support. has_cooldown requires no cooldown expression. Positional trinket parameters are also supported, see above for example. If the actor has no trinkets with cooldown, both expressions will always return 0. If both trinket slots have a cooldown, the larger cooldown expression return value will be chosen. The system will scan both Equip and Use trinkets in the slot(s), and use either type, if available.

NOTE: This syntax will not be evaluated against the 20s lockout typically shared by on-use items. To check if both the trinket's cooldown and the shared lockout are up, use the expression trinket[1|2|name].ready_cooldown.

# Use Ascendance, if there are no trinket cooldowns ready in the next 10 seconds.
actions +=/ascendance,if=!trinket.has_cooldown|trinket.cooldown.remains>10

Character properties

Character properties require no specific syntax. The available properties are:

  • level is the player level.
# Shadow fiend is only acquired on lv66 so if we want to test low-level toons...
actions+=/shadowfiend,if=level>=66
  • name and self both return a unique integer (called the "actor_index") for the player. By default, if the sim encounters an unknown operand, it will attempt to match it to the names of the actors and return the actor_index if it finds a match. Thus, you can use name and self to perform tests like follows:
# Use Reckoning if the boss is targeting another player (see Simulationcraft For Tanks)
actions+=/reckoning,if=target.current_target!=self
# Cast Flash of Light as long as the target's name isn't Bob (see Target Properties section)
actions+=/flash_of_light,if=target.name!=Bob
  • in_combat is zero when not in combat, 1 otherwise.
# Set stance before we enter combat
actions=/stance,choose=berserker,if=!in_combat
  • ptr is zero when using the mechanics from the live server, 1 when using the modifications on ptr.
# Starfall is going to be nerfed in the next patch? Then we won't use it anymore.
actions+=/starfall,if=ptr=0
  • bugs is zero when bugs are disabled, else it returns 1. Useful when a bug is implemented but will likely be fixed soon. (It's enabled by using the token option "bugs=1" or "bugs=0")
# In 7.1.5, casting Shadow Dance before going in combat let you extends the stealth buff, so it's worth to use with Subterfuge talent. Will likely be fixed in 7.2!
actions.precombat+=/shadow_dance,if=talent.subterfuge.enabled&bugs
  • is_add is true if the player is a add.
  • is_enemy is true if the player is a enemy.
  • spell_haste and attack_haste are the player's attack and spell factors. For example, 50% haste on your character sheet corresponds to a spell_haste or attack_haste value of 1/1.5=0.667.
# Only cast some_slow_spell if its cast time will be reduced to 70% of its base time.
actions+=/some_slow_spell,if=spell_haste<0.7
  • attack_speed is the player's attack speed.
  • mastery_value is the player's mastery value.
  • position_front is true if the player is in front of its target.
  • position_back is true if the player is in the back of its target.
  • time_to_bloodlust is the time until the next scheduled bloodlust cast, as defined by the bloodlust_time and bloodlust_percent parameters.
# Use avenging wrath as long as bloodlust will not be cast within the next 110 seconds
actions+=/avenging_wrath,if=time_to_bloodlust>110
  • raw_haste_pct is the players raw haste percentage.

    Note: raw means: Base + Passive + Gear (overridden or items) + Player Enchants + Global Enchants

  • incoming_damage_X will return the damage done to the player in the past X seconds. For example, a Death Knight may want to put a conditional on Death Strike usage such as:

# If damage taken in last 5 seconds exceeds 30% of max health, Death Strike
actions+=/death_strike,if=incoming_damage_5s>health.max*0.3

Note: The time can be specified in seconds or milliseconds, but must be an integer. For fractional parts of a second, milliseconds must be used; for example, incoming_damage_1500ms will return the damage taken in the last 1.5 seconds.

  • incoming_magic_damage_X will return the magical damage done to the player in the past X seconds.

  • time_to_die is the estimated remaining time, in seconds, before the player dies.

  • time_to_pct<health_pct> is the estimated remaining time, in seconds, before the player reaches <health_pct> of health.

Resource expressions

  • rage, mana, energy, focus, runic_power, health and soul_shards are the corresponding resources.
# Use heroic strike if we have a large enough rage pool
actions+=/heroic_strike,if=rage>60
  • <resource>.max is the current resource maximum amount (should always be 100 for rage and energy).
  • <resource>.pct is the current resource percentage, between 0 and 100.
  • <resource>.deficit is the number of lacking resource points for a full resource bar, between 0 and <resource>.max.
  • <resource>.max_nonproc is the resource value a player has before factoring in procs. Basically, it's the maximum resource value a player has out of combat.
  • <resource>.pct_nonproc is the resource percentage a player currently has, without factoring in procs. Can go above 100.
# Use some_expensive_spell if we have enough mana.
# All three actions are equivalent.
actions+=/some_expensive_spell,if=mana>mana.max*0.5
actions+=/some_expensive_spell,if=mana.deficit<mana.max*0.5
actions+=/some_expensive_spell,if=mana.pct>50
  • <resource>.regen is the amount of regenerated points per second for the corresponding resources.
  • <resource>.time_to_max is the time, in seconds, you would need to fully regenerate the corresponding resources assuming you would not spend anymore. It only takes into account the natural regeneration, not possible procs and such.
  • <resource>.time_to_x is the time, in seconds, you would need to regenerate to the specified amount under similar rules as above. Example: energy.time_to_50 would check when the player would reach 50 energy (or greater.)
  • <resource>.net_regen is the net resource regen a player has had so far in combat. Basically, it's the resources gained minus resources lost, divided by the current time.
  • stat.<x> is the value of a certain stat. <x> can be any of: strength, agility, stamina, intellect, spirit, health, maximum_health, mana, maximum_mana, rage, maximum_rage, energy, maximum_energy, focus, maximum_focus, runic, maximum_runic, spell_power, mp5, attack_power, expertise_rating, inverse_expertise_rating, hit_rating, inverse_hit_rating, crit_rating, haste_rating, weapon_dps, weapon_speed, weapon_offhand_dps, weapon_offhand_speed, armor, bonus_armor, resilience_rating, dodge_rating, parry_rating, block_rating, mastery_rating.

Split expressions

2 Parts
  • variable.<variablename> evaluates to the current value of the player's variable.

  • equipped.<item_id> or equipped.<item_name> or equipped.<on_use_effect_name> or equipped.<on_equip_effect_name> evaluates to true if the player has equipped the specified item or an item with the specified effect.

  • main_hand.1h, main_hand.2h, off_hand.1h, off_hand.2h evaluates to true if the player has the specified weapon type equipped.

  • race.<racename> evaluates true if <racename> is equal to the player's race.

# Use Berserking only if race is Troll.
actions+=/berserking,if=race.troll
  • spec.<spec_name> evaluates true if <spec_name> is equal to the player's specialization.
# Use Haunt only if spec is affliction
actions+=/haunt,if=spec.affliction
  • role.<role_name> evaluates true if <role_name> is equal to the player's role. The major role types are:

    • attack (melee dps and hunters)
    • heal
    • spell (caster dps)
    • tank

    There are some deprecated roles that will also parse (dps, hybrid), but as they're deprecated they (hopefully) won't show up very often.

   # This will cast Seal of Truth if the players role is "attack"
   actions+=/seal_of_truth,if=role.attack
  • stat.<stat_name> returns the amount of stat you have.
# Use smash only if players strength is above 1000
actions+=/smash,if=stat.strenght>1000
  • using_apl.<apl_name> returns true if the specified apl is explicitly used by the player.

  • active_dot.<dot_name> returns the number of dots with the given name currently active on all targets.

# Fire nova if enough Flame Shocks are active
actions+=/fire_nova,if=active_dot.flame_shock>=5
  • movement.remains returns the amount of remaining movement time of the player.
  • movement.distance returns the amount of remaining movement distance of the player.
  • movement.speed returns the current movement speed.
3 Parts
  • spell.<spell_name>.exists returns true if a spell with the given name exists for the player.
  • talent.<talent_name>.enabled returns true if the talent with the given name is active/selected for the player.
  • talent.<talent_name>.rank returns the current rank of the talent with the given name for the player. If not talented returns 0.
  • artifact.<artifact_spell_name>.enabled returns true if the artifact-spell with the given name is active for the player.
  • artifact.<artifact_spell_name>.rank returns the active rank of the artifact-spell with the given name for the player.

Set bonuses

You can check whether the character currently has a set bonus with the following syntax: set_bonus.tier<number>_[1pc/2pc/.../8pc]. It will return 1 if the set bonus is active, 0 otherwise.

# Only use some_spell if the character currently has the T21 4pc bonus
actions+=/some_spell,if=set_bonus.tier21_4pc

Class-specific

See the relevant page for each class.

Cooldowns

The character's cooldowns can be used through the following syntax: cooldown.<spell_name>.<cooldown_property>. As usual, for the spell name, use underscores instead of white spaces and ignore non-alphanumeric characters. The available properties are:

  • duration is the initial duration, in seconds (not the remaining duration).
  • remains is the remaining duration, in seconds, before the cooldown is done.
  • up or ready indicates if the cooldown is done (i.e., the associated ability is ready).
  • charges returns the amount of charges the cooldown has. If the spell has no charges it returns 1 if ready and 0 if not.
  • charges_fractional returns the amount of charges including fractional charges the cooldown has left.
  • full_recharge_time returns the amount of time left until all charges of the cooldown are ready again.
  • max_charges returns the maximum amount of charges the cooldown has left.
  • duration_expected and remains_expected are the initial/remaining duration, adjusted for cooldown affecting mechanics. The expected duration is evaluated from the difference between the elapsed time since the cooldown started and how much the cooldown duration has actually gone down.
# Use "aimed shot" if at least 5 seconds remain on "chimera shot".
actions+=/aimed_shot,if=cooldown.chimera_shot.remains>5

Dots

The dots and hots applied by the character on the target can be used through the following syntax: dot.<dot_name>.<dot_property>. As usual, for the dot name, use underscores instead of white spaces and ignore non-alphanumeric characters.

# If frost fever is going to expire on the target (less than 2 seconds remaining), refresh it with howling blast.
actions+=/howling_blast,if=dot.frost_fever.remains<=2

The available properties are:

  • duration is the initial duration, in seconds (not the remaining duration) of the current dot. Note that this returns 0 if the dot is not currently ticking.
  • modifier is the damages or healing modifier. If your character has a 20% general modifier and a 30% modifier for this dot, it will have a total of 1.2*1.3=1.56.
  • remains is the remaining duration, in seconds, before the dot/hot expires.
  • ticking is 1 if the dot is still active on the target, 0 if it faded out.
  • ticks_added is the number of additional ticks that have been added to the dot while it is active. This does not include ticks added from haste at the dot's application.
  • tick_dmg is the non-critical damage of the last tick.
  • ticks_remain are the number of ticks remaining for the current active dot.
  • spell_power last spell_power snapshot used for dot damage calculation.
  • attack_power last attack_power snapshot used for dot damage calculation.
  • multiplier last multiplier, excluding dynamic target multipliers.
  • haste_pct last haste multiplier.
  • current_ticks the total number of ticks for the current application of the dot.
  • ticks the number of ticks that have already happened for the current application of the dot.
  • crit_pct last critical strike percentage snapshot used for dot damage calculation.
  • crit_dmg last critical damage strike bonus percentage used for dot damage calculation.
  • tick_time_remains remaining time on the ongoing tick in seconds. 0 if the dot is not ticking.

Helpful Expressions

You can use active_dot.<dot_name> to return the amount of targets that currently have that dot active.

# Fire nova if enough Flame Shocks are active
actions+=/fire_nova,if=active_dot.flame_shock>=5

General

General properties require no specific syntax. The available properties are:

  • time is the elapsed time, in seconds, since the beginning of the fight
# Use bloodlust after at least 20s minimum
actions+=/bloodlust,if=time>20
  • active_enemies is the number of active enemy targets.
# Use Consecration if there are at least 3 enemies
actions+=/consecration,if=active_enemies>=3
  • expected_combat_length is the expected duration of the combat on that iteration. Added in Simulationcraft 801-02.

Pet

You can always refer to your pet's properties through the pet.<pet_name>.<pet_property> syntax. The pet's name is formatted using underscores instead of white spaces and ignoring non-alphanumeric characters.

# Use ice lance if the water elemental's "freeze" cooldown will expired before the end of the gcd.
actions+=/ice_lance,if=pet.water_elemental.cooldown.freeze.remains<gcd

Also, note that pets have one additional properties: active is zero when not active, 1 otherwise.

# Use "some_spell" if "some_pet" is active.
actions+=/some_spell,if=pet.some_pet.active

If the pet is temporary, you can evaluate to its remaining time in Azeroth with the remains expression. If the pet is not active, or it is a permanent pet, the expression evaluates to zero.

# Cast Frostbolt if Prismatic Crystal is alive during the casting
actions+=/frostbolt,if=pet.prismatic_crystal.remains>cast_time

Finally, within a pet's actions list, you can always use its owner's properties through the owner.<owner_property> syntax.

# Use some_spell if the owner's level is higher than 60.
actions+=/some_spell,if=owner.level>60

Raid Events

You can check for upcoming Raid Events with the syntax raid_event.event_type.filter. This will return a value depending on the particular event_type and filter you choose.

  • With WoW 8.0 (BFA) you can also filter by custom Raid Events names. The new syntax is: raid_event.type_or_name.filter. You can name a raid event through its ,name= option.

The event_type should be the name of the raid event as described in the Raid Events documentation. The filter is the property of the raid event that you're interested in.

# This will cast ice_floes if there's a movement raid event happening within the next second
actions+=/ice_floes,if=raid_event.movement.in<1
# This will only use ice_floes if the duration of that event is greater than 5 seconds
actions+=/ice_floes,if=raid_event.movement.in<1&raid_event.movement.duration>5

The list of event types and filters are given below. Event types are described in more detail on the Raid Events documentation page.

Event Types

  • adds
  • casting
  • damage_taken
  • damage
  • distraction
  • flying
  • heal
  • interrupt
  • invulnerable
  • movement
  • position_switch
  • stun
  • vulnerable

Filters

  • in - returns the time until the next event of the appropriate event type.
  • amount - returns the amount of the next appropriate event type. For damage and heal event types this will return the amount of damage or healing done by the event, and for the damage_taken event type it will return the number of stacks of the damage_taken debuff being applied to the player. For all other event types this will evaluate to zero.
  • duration - returns the duration of the next event of the appropriate event type (zero for event types that have no duration).
  • cooldown - returns the cooldown of the next event of the appropriate event type. Note that this is the total cooldown of the ability, not the remaining cooldown.
  • exists - returns 1 if an event type of that kind exists in the simulation, 0 otherwise.
  • distance or max_distance - returns the maximum distance of the next event of the appropriate event type.
  • min_distance - returns the minimum distance of the next event of the appropriate event type.
  • to_pct - returns the target health percentage of a heal event, if specified, 0 otherwise.

With WoW 8.0 (BFA):

  • up - returns true if there is at least 1 raid event currently up. Prefer up over remains if it satisfies your needs.
  • remains - returns the remaining duration of the raid event currently up which has the longest remaining duration. Does not take into account not yet up raid events.

For Tier 17 and beyond:

# Only use some_spell if the character currently has the T17 4pc bonus
actions+=/some_spell,if=set_bonus.tier17_4pc

See also equipment - set bonuses.

Spell flights

You can check whether a casted spell is still flying towards its target (a fireball for example) with the following syntax: action.<spell_name>.in_flight. It will return 1 if the spell if flying, 0 otherwise. Additionally you can use action.<spell_name>.in_flight_remains to return the lowest time left until the action hits any target. action.<spell_name>.in_flight_to_target checks if this action is flying towards the current target and will return 1 if this is the case and 0 otherwise.

Swings

The remaining time, in seconds, on weapon swings can be checked through the following syntax: swing.<weapon>.remains. The weapon can be one of the following values: "mh", "oh", "mainhand", "offhand", "main_hand", "off_hand".

# Arms warriors in burning crusade (at the time slam used to reset the swing timer) could use this to only fire slam if at least 3s were remaining before the next swing:
actions+=/slam,if=swing.mh.remains>3

Target's properties

You can check the target's properties with the following syntax: target.<property>

# Use cleave if the target has at least 1 add
actions+=/cleave,if=target.adds>0

The available properties are:

  • level is the target's level.
  • health_pct is the target's health percent, between 0 and 100.
  • adds is the number of adds the target has.
  • adds_never is 0 if the target will never have adds (no initial adds and no "adds" raid event), 1 otherwise.
  • distance is the distance to the target. Handy for checking spells who have damage based on the distance from the target, such as the Priest Level 90 talents (Divine Star, Cascade, and Halo)
  • current_target (Simulationcraft 6.0.1 release 1 and later) is the target's target, usually the tank that currently has "aggro." This can be useful for making conditional tank swaps (see Simulationcraft For Tanks).
  • name is the target's unique identifier (actor_index). The sim attempts to match unknown operands to the names of each actor, and returns the actor_index if it finds a match, so this can be used to test against the target's name:
# Only cast Smite on Fluffy_Pillow
actions+=/smite,if=target.name=Fluffy_Pillow

(Simulationcraft 6.0.2 release 1 and later) In addition, you can simply use target to refer to the current target of the action. This is currently only useful for Mages.

Related settings

  • reaction_time (scope: global; default: 0.5) is the reaction time, in seconds. It is the time your players will need to notice an aura application (for action conditional expressions using the react keyword) or a spell miss (for action conditional expressions using the miss_react keyword).
# No, I am not that slow!
reaction_time=0.25
  • skip_actions (scope: current character; default: "") is a sequence of action names to ignore, separated by "/". Actions mentioned in this string will never be performed.
# With only three adds, it is wise for John the warrior to still use his poor aoe spells? Let's check!
armory=us,illidan,John
skip_actions=cleave/whirlwind
  • default_actions (scope: global; default: 0), when different from zero, will force the application to ignore your custom action lists and only use the default action lists for all characters and pets.
default_actions=1
Clone this wiki locally