roll-a-d6
is a dice rolling engine library that supports sophisticated macros and formulas that no other dice roller supports.
At its core, we have the dice roller: rollFormula(formula, macrosObject)
import * as d6 from 'roll-a-d6';
const roll = d6.rollFormula('1d20+Strength,1d8+Strength', { Strength: '5' });
console.log(roll.result.join(', '));
The components of the dice roller include the validator, tokenizer, parser, RPN converter, and evaluator. For more information, check out the API docs here.
An example of a more sophisticated dice roll that this package supports:
import * as d6 from 'roll-a-d6';
const macros = {
roll: 'r=1d20,(^adv&&^disadv)||(adv==0&&disadv==0)->r[0];(adv->r[0]>>r[1];r[0]<<r[1])',
atk: 'roll[0]==20->dmgDice[0]+dmgDice[1]+dmgMod;(roll[0]==1->0;roll[0]+bonus>ac@Defender->dmgDice[0]+dmgMod;0)',
bonus: 'prof+dmgMod',
shortSword: '1d6,dmgMod=Dex>>Str',
sneakAttack: '3d6',
sneakAttackIfPossible: 'canYouSneakAttack->sneakAttack;0',
Str: '10',
Dex: '18',
prof: '3',
};
const roll = d6.rollFormula('ac@Defender=?,canYouSneakAttack=1,dmgDice=shortSword,(atk[0]+atk[1])>0->(atk[0]+atk[1]+sneakAttackIfPossible);0', macros);
// now I know the attacker's AC is 15
const foundResult = roll.result.find(({ minValue, maxValue }) => minValue <= 15 && !(maxValue < 15));
console.log(foundResult[0]);
Rolls should be in the format of xdy, e.g., '1d20' or
Operators should be used infix, e.g., 5+7, though they are evaluated postfix.
+, -, *, /
Greater than (>
), Less than (<
), and other similar operators (<=
, >=
, ==
, and <>
(not equal)) return true or false.
Greater of (>>
) and Lesser of (<<
) return the appropriate value, e.g., 5<<7
returns 5
.
||
and &&
perform the equivalent operations.
->
and ;
are the equivalents of then and else. For example: 1d20>10->1d6;0
will first roll 1d20,
then compare that value to 10. If the value is greater than 10, it will roll 1d6 and return the result
of that roll. If the value is less than or equal to 10, it will return 0.
These operators allow you to roll the same thing many times with ease. For example, 4#+1d6
is the
equivalent of 1d6+1d6+1d6+1d6
. This is distinct from 4*1d6 because the former will not reroll those
dice; the latter will.
The following operators are supported:
#+
- Add the results together#*
- Multiply the results together#&
- AND (&&
) the results together, returning the last value if all are truthy, or the first falsy value.#|
- OR (||
) the results together, returning the last value if all are falsy, or the first truthy value.Prefixing a variable reference with ^
will cause it to be treated as 0 if it was not passed in.
By using the name of a macro, you can evaluate that macro in your formula. Macros can also reference other macros.
Here is a simple example:
import * as d6 from 'roll-a-d6';
const macros = {
macro: '1d20>10->helperMacro;0',
helperMacro: '1d10+5',
};
const formula = 'macro+macro';
const roll = d6.rollFormula(formula, macros);
You may want to check the result of a macro and then use it. There are a couple ways to do this.
One way involves using the >>
and <<
operators.
For example, to choose the higher value between macro1
and macro2
, you would write
macro1>>macro2
.
The more advanced way involves global and local macro instances.
[]
is used for local macro instances. {}
is used for global macro instances.
Local macro instance evaluations are used only for the currently evaluated macro / formula.
For example, you might want to know if the result of 1d20 was 20, 1, or if it was above a certain value.
You can do this like so: r=1d20,r[0]==20->crit;(r[0]==0->miss;r[0]>=minimum->hit)
Global macro instance evaluations are used for every macro that references them. They are more useful in very focused applications rather than in more general, reusable rolls.
As the result of a roll, you may want to change some value, like decreasing your uses of an ability. Alternatively, you may want to track a value during a roll and use it to determine what to do next. You can do this with side effects!
Currently, only three side effects are supported:
:=
(assignment)+=
(increment)-=
(decrement)If you wish to apply a side effect, you must prefix the name of the variable with $
.
You can use the ...
pseudo-operator between a side effect and the result you want to return.
One example of a side effect, using the macros from above:
const myMacros = {
...macros,
sneakAttack: '3d6',
sneakAttackIfPossible: 'canSneakAttack>0->$canSneakAttack:=0...sneakAttack;0',
attackWithSneakAttack: 'atk[0]>0->atk[0]+sneakAttackIfPossible;0',
};
const formula = 'canSneakAttack=1,2#+attackWithSneakAttack';
d6.rollFormula(formula, myMacros);