diff --git a/build.gradle b/build.gradle index 621951a279..f504291dd4 100644 --- a/build.gradle +++ b/build.gradle @@ -498,7 +498,7 @@ dependencies { implementation 'com.github.jknack:handlebars-helpers:4.3.1' // For advanced dice roller - implementation 'com.github.RPTools:advanced-dice-roller:0.1.4' + implementation 'com.github.RPTools:advanced-dice-roller:1.0.3' diff --git a/src/main/java/net/rptools/dicelib/expression/ExpressionParser.java b/src/main/java/net/rptools/dicelib/expression/ExpressionParser.java index 84f45dc425..33ad23b422 100755 --- a/src/main/java/net/rptools/dicelib/expression/ExpressionParser.java +++ b/src/main/java/net/rptools/dicelib/expression/ExpressionParser.java @@ -14,6 +14,8 @@ */ package net.rptools.dicelib.expression; +import java.util.List; +import java.util.regex.Pattern; import net.rptools.dicelib.expression.function.ArsMagicaStress; import net.rptools.dicelib.expression.function.CountSuccessDice; import net.rptools.dicelib.expression.function.DropHighestRoll; @@ -37,9 +39,11 @@ import net.rptools.dicelib.expression.function.ShadowRun5Dice; import net.rptools.dicelib.expression.function.ShadowRun5ExplodeDice; import net.rptools.dicelib.expression.function.UbiquityRoll; +import net.rptools.dicelib.expression.function.advanced.AdvancedDiceRolls; import net.rptools.parser.*; import net.rptools.parser.transform.RegexpStringTransformer; import net.rptools.parser.transform.StringLiteralTransformer; +import org.javatuples.Pair; public class ExpressionParser { private static String[][] DICE_PATTERNS = @@ -206,22 +210,15 @@ public class ExpressionParser { new String[] {"\\b[aA][sS](\\d+)[bB]#([+-]?\\d+)\\b", "arsMagicaStress($1, $2)"}, new String[] {"\\b[aA][nN][sS](\\d+)\\b", "arsMagicaStressNum($1, 0)"}, new String[] {"\\b[aA][nN][sS](\\d+)[bB]#([+-]?\\d+)\\b", "arsMagicaStressNum($1, $2)"}, - - // SW Genesys - new String[] { - "\\bsw#(([bBsSaAdDpPcCfF]\\d+)+)(\\.(([eEgGdDjJ])+))?\\b", "swgenesys('$1', '$4')" - }, - new String[] {"\\bsw#last(\\.(([eEgGdDjJ])+))*\\b", "swgenesysLast('last', '$2')"}, - - // Genesys - new String[] { - "\\bgs#(([bBsSaAdDpPcCjJ]\\d+)+)(\\.(([eEgGdD])+))?\\b", "genesys('$1', " + "'$4')" - }, - new String[] {"\\bgs#last(\\.(([eEgGdDj])+))*\\b", "genesysLast('last', '$2')"} }; private final Parser parser; + private final List> preprocessPatterns = + List.of( + new Pair<>(Pattern.compile("([A-z]+)!\"([^\"]*)\""), "advancedRoll('$1', " + "'$2')"), + new Pair<>(Pattern.compile("([A-z]+)!'([^']*)'"), "advancedRoll('$1', " + "'$2')")); + public ExpressionParser() { this(DICE_PATTERNS); } @@ -252,6 +249,7 @@ public ExpressionParser(String[][] regexpTransforms) { parser.addFunction(new KeepLowestRoll()); parser.addFunction(new ArsMagicaStress()); parser.addFunction(new GeneSysDice()); + parser.addFunction(new AdvancedDiceRolls()); parser.addFunction(new If()); @@ -291,6 +289,9 @@ public Result evaluate(String expression, VariableResolver resolver, boolean mak } RunData.setCurrent(newRunData); + // Some patterns need pre-processing before the parser is called otherwise the parser + // creation will fail + expression = preProcess(expression); synchronized (parser) { final Expression xp = makeDeterministic @@ -306,4 +307,20 @@ public Result evaluate(String expression, VariableResolver resolver, boolean mak return ret; } + + /** + * Pre-process the expression before it is parsed. This is used to convert some patterns into + * function calls that the parser can handle. + * + * @param expression The expression to pre-process + * @return The pre-processed expression + */ + private String preProcess(String expression) { + for (Pair p : preprocessPatterns) { + if (p.getValue0().matcher(expression).find()) { + return p.getValue0().matcher(expression).replaceAll(p.getValue1()); + } + } + return expression; + } } diff --git a/src/main/java/net/rptools/dicelib/expression/function/advanced/AdvancedDiceRolls.java b/src/main/java/net/rptools/dicelib/expression/function/advanced/AdvancedDiceRolls.java new file mode 100644 index 0000000000..b1acc165dd --- /dev/null +++ b/src/main/java/net/rptools/dicelib/expression/function/advanced/AdvancedDiceRolls.java @@ -0,0 +1,51 @@ +/* + * This software Copyright by the RPTools.net development team, and + * licensed under the Affero GPL Version 3 or, at your option, any later + * version. + * + * MapTool Source Code is distributed in the hope that it will be + * useful, but WITHOUT ANY WARRANTY; without even the implied warranty + * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * + * You should have received a copy of the GNU Affero General Public + * License * along with this source Code. If not, please visit + * and specifically the Affero license + * text at . + */ +package net.rptools.dicelib.expression.function.advanced; + +import java.util.List; +import net.rptools.dicelib.expression.function.advanced.GenesysDiceRolls.DiceType; +import net.rptools.maptool.language.I18N; +import net.rptools.parser.Parser; +import net.rptools.parser.ParserException; +import net.rptools.parser.VariableResolver; +import net.rptools.parser.function.AbstractFunction; + +/** Function to roll dice using the advanced dice roller. */ +public class AdvancedDiceRolls extends AbstractFunction { + + /** Constructor. */ + public AdvancedDiceRolls() { + super(2, 2, false, "advancedRoll"); + } + + @Override + public Object childEvaluate( + Parser parser, VariableResolver resolver, String functionName, List parameters) + throws ParserException { + String diceName = parameters.get(0).toString().toLowerCase(); + String diceExpression = parameters.get(1).toString(); + + try { + return switch (diceName) { + case "sw" -> new GenesysDiceRolls().roll(DiceType.StarWars, diceExpression, resolver); + case "gs" -> new GenesysDiceRolls().roll(DiceType.Genesys, diceExpression, resolver); + default -> throw new ParserException( + I18N.getText("advanced.roll.unknownDiceType", diceName)); + }; + } catch (IllegalArgumentException e) { + throw new ParserException(e.getMessage()); + } + } +} diff --git a/src/main/java/net/rptools/dicelib/expression/function/advanced/GenesysDiceRolls.java b/src/main/java/net/rptools/dicelib/expression/function/advanced/GenesysDiceRolls.java new file mode 100644 index 0000000000..f267147026 --- /dev/null +++ b/src/main/java/net/rptools/dicelib/expression/function/advanced/GenesysDiceRolls.java @@ -0,0 +1,261 @@ +/* + * This software Copyright by the RPTools.net development team, and + * licensed under the Affero GPL Version 3 or, at your option, any later + * version. + * + * MapTool Source Code is distributed in the hope that it will be + * useful, but WITHOUT ANY WARRANTY; without even the implied warranty + * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * + * You should have received a copy of the GNU Affero General Public + * License * along with this source Code. If not, please visit + * and specifically the Affero license + * text at . + */ +package net.rptools.dicelib.expression.function.advanced; + +import java.util.Map; +import javax.swing.JOptionPane; +import net.rptools.maptool.advanceddice.genesys.GenesysDiceResult; +import net.rptools.maptool.advanceddice.genesys.GenesysDiceRoller; +import net.rptools.maptool.client.MapTool; +import net.rptools.maptool.client.MapToolVariableResolver; +import net.rptools.maptool.client.ui.theme.ThemeSupport; +import net.rptools.maptool.client.ui.theme.ThemeSupport.ThemeColor; +import net.rptools.maptool.language.I18N; +import net.rptools.parser.ParserException; +import net.rptools.parser.VariableResolver; + +/** Function to roll dice using the advanced dice roller. */ +public class GenesysDiceRolls { + + /** Enum to represent the different dice types. */ + public enum DiceType { + StarWars("sw"), + Genesys("gs"); + + /** The variable prefix for the dice type. */ + public final String variablePrefix; + + /** + * Constructor. + * + * @param variablePrefix the prefix to use for variables. + */ + DiceType(String variablePrefix) { + this.variablePrefix = variablePrefix; + } + + /** + * Get the variable prefix for the dice type. + * + * @return the variable prefix. + */ + public String getVariablePrefix() { + return variablePrefix; + } + } + + /** Map of font names for the different systems. */ + private static final Map GS_FONT_NAME_MAP = + Map.of(DiceType.StarWars, "EotE Symbol", DiceType.Genesys, "Genesys Glyphs and Dice"); + + /** + * Roll the given dice string using genesys/starwars dice roll parser. + * + * @param diceType the type of dice to roll. + * @param diceExpression the expression to roll. + * @param resolver the variable resolver. + * @return the result of the roll. + * @throws ParserException if there is an error parsing the expression. + */ + Object roll(DiceType diceType, String diceExpression, VariableResolver resolver) + throws ParserException { + var roller = new GenesysDiceRoller(); + try { + var result = + roller.roll( + diceExpression, + n -> getVariable(resolver, n), + n -> getProperty(resolver, n), + n -> getPromptedValue(resolver, n)); + + if (result.hasErrors()) { + var errorSb = new StringBuilder(); + for (var error : result.getErrors()) { + String msg; + int ind = error.charPositionInLine(); + if (result.getRollString().length() > ind + 3) { + msg = result.getRollString().substring(ind, ind + 3) + "..."; + } else { + msg = result.getRollString().substring(ind); + } + var errorText = I18N.getText("advanced.roll.parserError", error.line(), ind + 1, msg); + errorSb.append(errorText).append("
"); + } + throw new ParserException(errorSb.toString()); + } + var varPrefix = diceType.getVariablePrefix() + ".lastRoll"; + setVars(resolver, varPrefix, result); + + return formatResults(diceType, result); + + } catch (IllegalArgumentException e) { + throw new ParserException(e.getMessage()); + } + } + + /** + * Format the results of the roll. + * + * @param diceType the type of dice. + * @param result the result of the roll. + * @return the formatted results. + */ + private String formatResults(DiceType diceType, GenesysDiceResult result) { + var gray = ThemeSupport.getThemeColorHexString(ThemeColor.GREY); + + var sb = new StringBuilder(); + sb.append(""); + sb.append(""); + for (var dice : result.getRolls()) { + sb.append(dice.resultType().getFontCharacters()); + } + sb.append(""); + sb.append(""); + return sb.toString(); + } + + /** + * Set the variables for the result. + * + * @param resolver the variable resolver. + * @param varPrefix the variable prefix. + * @param result the result. + * @throws ParserException if there is an error setting the variables. + */ + private void setVars(VariableResolver resolver, String varPrefix, GenesysDiceResult result) + throws ParserException { + resolver.setVariable(varPrefix + ".expression", result.getRollString()); + resolver.setVariable(varPrefix + ".success", result.getSuccessCount()); + resolver.setVariable(varPrefix + ".failure", result.getFailureCount()); + resolver.setVariable(varPrefix + ".advantage", result.getAdvantageCount()); + resolver.setVariable(varPrefix + ".threat", result.getThreatCount()); + resolver.setVariable(varPrefix + ".triumph", result.getTriumphCount()); + resolver.setVariable(varPrefix + ".despair", result.getDespairCount()); + resolver.setVariable(varPrefix + ".light", result.getLightCount()); + resolver.setVariable(varPrefix + ".dark", result.getDarkCount()); + + for (var group : result.getGroupNames()) { + var groupResult = result.getGroup(group); + setVars(resolver, varPrefix + ".group." + group, groupResult); + } + } + + /** + * Get the variable value. + * + * @param resolver the variable resolver. + * @param name the name of the variable. + * @return the value of the variable. + */ + private int getVariable(VariableResolver resolver, String name) { + if (!resolver.getVariables().contains(name.toLowerCase())) { + throw new IllegalArgumentException(I18N.getText("advanced.roll.unknownVariable", name)); + } + + try { + var value = resolver.getVariable(name); + var result = integerResult(value, false); + + if (result == null) { + throw new IllegalArgumentException(I18N.getText("advanced.roll.variableNotNumber", name)); + } + + return result; + } catch (ParserException e) { + throw new IllegalArgumentException(e); + } + } + + /** + * Get the property value for the token in context. + * + * @param resolver the variable resolver. + * @param name the name of the property. + * @return the value of the property. + */ + private int getProperty(VariableResolver resolver, String name) { + var mtResolver = (MapToolVariableResolver) resolver; + var token = mtResolver.getTokenInContext(); + if (token == null) { + throw new IllegalArgumentException(I18N.getText("advanced.roll.noTokenInContext")); + } + var value = token.getProperty(name); + + if (value == null) { + throw new IllegalArgumentException(I18N.getText("advanced.roll.unknownProperty", name)); + } + + var result = integerResult(value, true); + if (result == null) { + throw new IllegalArgumentException(I18N.getText("advanced.roll.propertyNotNumber", name)); + } + + return result; + } + + /** + * Prompt the user for a value. + * + * @param resolver the variable resolver. + * @param name the name of the value. + * @return the value. + */ + private int getPromptedValue(VariableResolver resolver, String name) { + var option = + JOptionPane.showInputDialog( + MapTool.getFrame(), + I18N.getText("lineParser.dialogValueFor", name), + I18N.getText("lineParser.dialogTitleNoToken"), + JOptionPane.QUESTION_MESSAGE, + null, + null, + 1); + try { + return Integer.parseInt(option.toString()); + } catch (NumberFormatException e) { + throw new IllegalArgumentException(I18N.getText("advanced.roll.inputNotNumber", name)); + } + } + + /** + * Get the integer result. + * + * @param value the value. + * @param parseString attempt to parse string value of object if it is not a number. + * @return the integer result. + */ + private Integer integerResult(Object value, boolean parseString) { + if (value instanceof Integer i) { + return i; + } + + if (value instanceof Number n) { + return n.intValue(); + } + + if (parseString) { + try { + return Integer.parseInt(value.toString()); + } catch (NumberFormatException e) { + return null; + } + } + + return null; + } +} diff --git a/src/main/java/net/rptools/maptool/client/MapToolVariableResolver.java b/src/main/java/net/rptools/maptool/client/MapToolVariableResolver.java index f1489caf77..ed31fbcccb 100644 --- a/src/main/java/net/rptools/maptool/client/MapToolVariableResolver.java +++ b/src/main/java/net/rptools/maptool/client/MapToolVariableResolver.java @@ -276,7 +276,7 @@ public Object getVariable(String name, VariableModifiers mods) throws ParserExce result = JOptionPane.showInputDialog( MapTool.getFrame(), - I18N.getText("lineParser.dialogValueFor") + " " + name, + I18N.getText("lineParser.dialogValueFor", name), DialogTitle, JOptionPane.QUESTION_MESSAGE, null, diff --git a/src/main/resources/net/rptools/maptool/language/i18n.properties b/src/main/resources/net/rptools/maptool/language/i18n.properties index 2d7975a06f..9da807e8fb 100644 --- a/src/main/resources/net/rptools/maptool/language/i18n.properties +++ b/src/main/resources/net/rptools/maptool/language/i18n.properties @@ -1654,7 +1654,7 @@ lineParser.countNonNeg = Count option requires a non negative numbe # Notice there are no double quotes around {0}. lineParser.dialogTitle = Input Value for {0}. lineParser.dialogTitleNoToken = Input Value. -lineParser.dialogValueFor = Value For +lineParser.dialogValueFor = Value For "{0}" lineParser.duplicateLibTokens = Duplicate "{0}" tokens found. lineParser.emptyTokenName = Cannot assign a blank or empty string to the variable token.name lineParser.errorBodyRoll = Error in body of roll. @@ -2808,3 +2808,13 @@ Label.label=Label: # StatSheet token.statSheet.legacyStatSheetDescription = Legacy (pre 1.14) Stat Sheet token.statSheet.useDefault = Default Stat Sheet for Property Type + +# Advanced Dice Rolls +advanced.roll.parserError = Dice Roll String Error line {0} column {1} "{2}". +advanced.roll.unknownDiceType = Unknown Dice Roll Type {0}. +advanced.roll.unknownVariable = Unknown Variable {0}. +advanced.roll.variableNotNumber = Variable {0} is not a number. +advanced.roll.unknownProperty = Unknown Property {0}. +advanced.roll.propertyNotNumber = Property {0} is not a number. +advanced.roll.noTokenInContext = No token in context. +advanced.roll.inputNotNumber = Input {0} is not a number. \ No newline at end of file