Made constants work in calculations performed after recomputation
This commit is contained in:
@@ -1,32 +1,67 @@
|
||||
import { parse, CompilationContext } from '/imports/parser/parser.js';
|
||||
import ConstantNode from '/imports/parser/parseTree/ConstantNode.js';
|
||||
import SymbolNode from '/imports/parser/parseTree/SymbolNode.js';
|
||||
import ErrorNode from '/imports/parser/parseTree/ErrorNode.js';
|
||||
|
||||
//TODO replace constants with their parsed node
|
||||
|
||||
export default function evaluateString(string, scope, fn = 'compile', context){
|
||||
let errors = [];
|
||||
export default function evaluateString({string, scope, fn = 'compile', context}){
|
||||
if (!context){
|
||||
context = new CompilationContext({});
|
||||
}
|
||||
if (!string){
|
||||
errors.push('No string provided');
|
||||
return {result: string, errors};
|
||||
context.storeError('No string provided');
|
||||
return {result: {value: string}, context};
|
||||
}
|
||||
|
||||
if (!scope) errors.push('No scope provided');
|
||||
if (!scope) context.storeError('No scope provided');
|
||||
|
||||
// Parse the string using mathjs
|
||||
let node;
|
||||
try {
|
||||
node = parse(string);
|
||||
} catch (e) {
|
||||
errors.push(e);
|
||||
return {result: string, errors};
|
||||
}
|
||||
if (!context){
|
||||
context = new CompilationContext({});
|
||||
context.storeError(e);
|
||||
return {result: {value: string}, context};
|
||||
}
|
||||
node = replaceConstants({calc: node, context, scope});
|
||||
let result = node[fn](scope, context);
|
||||
if (result instanceof ConstantNode){
|
||||
return {result: result.value, errors: context.errors}
|
||||
} else {
|
||||
return {result: result.toString(), errors: context.errors};
|
||||
}
|
||||
return {result, context};
|
||||
}
|
||||
|
||||
// Replace constants in the calc with the right ParseNodes
|
||||
function replaceConstants({calc, context, scope}){
|
||||
let constFailed = [];
|
||||
calc = calc.replaceNodes(node => {
|
||||
if (!(node instanceof SymbolNode)) return;
|
||||
let constant = scope[node.name];
|
||||
// replace constants that aren't overridden by stats or disabled by a toggle
|
||||
if (constant && constant.type === 'constant'){
|
||||
// Fail if the constant has errors
|
||||
if (constant.errors && constant.errors.length){
|
||||
constFailed.push(node.name);
|
||||
return;
|
||||
}
|
||||
let parsedConstantNode;
|
||||
try {
|
||||
parsedConstantNode = parse(constant.calculation);
|
||||
} catch(e){
|
||||
constFailed.push(node.name);
|
||||
return;
|
||||
}
|
||||
if (!parsedConstantNode) constFailed.push(node.name);
|
||||
return parsedConstantNode;
|
||||
}
|
||||
});
|
||||
constFailed.forEach(name => {
|
||||
context.storeError({
|
||||
type: 'error',
|
||||
message: `${name} is a constant property with parsing errors`
|
||||
});
|
||||
});
|
||||
let failed = !!constFailed.length;
|
||||
if (failed){
|
||||
calc = new ErrorNode({error: 'Failed to replace constants'});
|
||||
}
|
||||
return calc;
|
||||
}
|
||||
|
||||
@@ -4,28 +4,30 @@ import VERSION from '/imports/constants/VERSION.js';
|
||||
|
||||
export default function writeCreatureVariables(memo, creatureId, fullRecompute = true) {
|
||||
const fields = [
|
||||
'name',
|
||||
'attributeType',
|
||||
'baseValue',
|
||||
'spellSlotLevelValue',
|
||||
'damage',
|
||||
'decimal',
|
||||
'reset',
|
||||
'resetMultiplier',
|
||||
'value',
|
||||
'currentValue',
|
||||
'modifier',
|
||||
'ability',
|
||||
'skillType',
|
||||
'baseProficiency',
|
||||
'abilityMod',
|
||||
'advantage',
|
||||
'passiveBonus',
|
||||
'proficiency',
|
||||
'attributeType',
|
||||
'baseProficiency',
|
||||
'baseValue',
|
||||
'calculation',
|
||||
'conditionalBenefits',
|
||||
'rollBonuses',
|
||||
'currentValue',
|
||||
'damage',
|
||||
'decimal',
|
||||
'fail',
|
||||
'level',
|
||||
'modifier',
|
||||
'name',
|
||||
'passiveBonus',
|
||||
'proficiency',
|
||||
'reset',
|
||||
'resetMultiplier',
|
||||
'rollBonuses',
|
||||
'skillType',
|
||||
'spellSlotLevelValue',
|
||||
'type',
|
||||
'value',
|
||||
];
|
||||
|
||||
if (fullRecompute){
|
||||
@@ -34,6 +36,12 @@ export default function writeCreatureVariables(memo, creatureId, fullRecompute =
|
||||
let condensedStat = pick(stat, fields);
|
||||
memo.creatureVariables[variableName] = condensedStat;
|
||||
});
|
||||
forOwn(memo.constantsByVariableName, (stat, variableName) => {
|
||||
let condensedStat = pick(stat, fields);
|
||||
if (!memo.creatureVariables[variableName]){
|
||||
memo.creatureVariables[variableName] = condensedStat;
|
||||
}
|
||||
});
|
||||
Creatures.update(creatureId, {$set: {
|
||||
variables: memo.creatureVariables,
|
||||
computeVersion: VERSION,
|
||||
|
||||
Reference in New Issue
Block a user