mathjs 10.1.1 → 10.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/HISTORY.md +17 -0
- package/docs/expressions/algebra.md +27 -0
- package/docs/reference/functions/abs.md +6 -0
- package/docs/reference/functions/acos.md +6 -0
- package/docs/reference/functions/acosh.md +6 -0
- package/docs/reference/functions/acot.md +6 -0
- package/docs/reference/functions/acoth.md +6 -0
- package/docs/reference/functions/acsc.md +6 -0
- package/docs/reference/functions/acsch.md +6 -0
- package/docs/reference/functions/add.md +6 -0
- package/docs/reference/functions/and.md +6 -0
- package/docs/reference/functions/apply.md +6 -0
- package/docs/reference/functions/arg.md +6 -0
- package/docs/reference/functions/asec.md +6 -0
- package/docs/reference/functions/asech.md +6 -0
- package/docs/reference/functions/asin.md +6 -0
- package/docs/reference/functions/asinh.md +6 -0
- package/docs/reference/functions/atan.md +6 -0
- package/docs/reference/functions/atan2.md +6 -0
- package/docs/reference/functions/atanh.md +6 -0
- package/docs/reference/functions/bellNumbers.md +6 -0
- package/docs/reference/functions/bin.md +6 -0
- package/docs/reference/functions/bitAnd.md +6 -0
- package/docs/reference/functions/bitNot.md +6 -0
- package/docs/reference/functions/bitOr.md +6 -0
- package/docs/reference/functions/bitXor.md +6 -0
- package/docs/reference/functions/catalan.md +6 -0
- package/docs/reference/functions/cbrt.md +6 -0
- package/docs/reference/functions/ceil.md +6 -0
- package/docs/reference/functions/clone.md +6 -0
- package/docs/reference/functions/column.md +6 -0
- package/docs/reference/functions/combinations.md +6 -0
- package/docs/reference/functions/combinationsWithRep.md +6 -0
- package/docs/reference/functions/compare.md +6 -0
- package/docs/reference/functions/compareNatural.md +6 -0
- package/docs/reference/functions/compareText.md +6 -0
- package/docs/reference/functions/compile.md +6 -0
- package/docs/reference/functions/composition.md +6 -0
- package/docs/reference/functions/concat.md +6 -0
- package/docs/reference/functions/conj.md +6 -0
- package/docs/reference/functions/cos.md +6 -0
- package/docs/reference/functions/cosh.md +6 -0
- package/docs/reference/functions/cot.md +6 -0
- package/docs/reference/functions/coth.md +6 -0
- package/docs/reference/functions/count.md +6 -0
- package/docs/reference/functions/cross.md +6 -0
- package/docs/reference/functions/csc.md +6 -0
- package/docs/reference/functions/csch.md +6 -0
- package/docs/reference/functions/ctranspose.md +6 -0
- package/docs/reference/functions/cube.md +6 -0
- package/docs/reference/functions/deepEqual.md +6 -0
- package/docs/reference/functions/derivative.md +6 -0
- package/docs/reference/functions/det.md +6 -0
- package/docs/reference/functions/diag.md +6 -0
- package/docs/reference/functions/diff.md +6 -0
- package/docs/reference/functions/distance.md +6 -0
- package/docs/reference/functions/divide.md +6 -0
- package/docs/reference/functions/dot.md +6 -0
- package/docs/reference/functions/dotDivide.md +6 -0
- package/docs/reference/functions/dotMultiply.md +6 -0
- package/docs/reference/functions/dotPow.md +6 -0
- package/docs/reference/functions/eigs.md +6 -0
- package/docs/reference/functions/equal.md +6 -0
- package/docs/reference/functions/equalText.md +6 -0
- package/docs/reference/functions/erf.md +6 -0
- package/docs/reference/functions/evaluate.md +6 -0
- package/docs/reference/functions/exp.md +6 -0
- package/docs/reference/functions/expm.md +6 -0
- package/docs/reference/functions/expm1.md +6 -0
- package/docs/reference/functions/factorial.md +6 -0
- package/docs/reference/functions/filter.md +6 -0
- package/docs/reference/functions/fix.md +6 -0
- package/docs/reference/functions/flatten.md +6 -0
- package/docs/reference/functions/floor.md +6 -0
- package/docs/reference/functions/forEach.md +6 -0
- package/docs/reference/functions/format.md +6 -0
- package/docs/reference/functions/gamma.md +6 -0
- package/docs/reference/functions/gcd.md +6 -0
- package/docs/reference/functions/getMatrixDataType.md +6 -0
- package/docs/reference/functions/hasNumericValue.md +6 -0
- package/docs/reference/functions/help.md +6 -0
- package/docs/reference/functions/hex.md +6 -0
- package/docs/reference/functions/hypot.md +6 -0
- package/docs/reference/functions/identity.md +6 -0
- package/docs/reference/functions/im.md +6 -0
- package/docs/reference/functions/intersect.md +6 -0
- package/docs/reference/functions/inv.md +6 -0
- package/docs/reference/functions/invmod.md +6 -0
- package/docs/reference/functions/isInteger.md +6 -0
- package/docs/reference/functions/isNaN.md +6 -0
- package/docs/reference/functions/isNegative.md +6 -0
- package/docs/reference/functions/isNumeric.md +6 -0
- package/docs/reference/functions/isPositive.md +6 -0
- package/docs/reference/functions/isPrime.md +6 -0
- package/docs/reference/functions/isZero.md +6 -0
- package/docs/reference/functions/kldivergence.md +6 -0
- package/docs/reference/functions/kron.md +6 -0
- package/docs/reference/functions/larger.md +6 -0
- package/docs/reference/functions/largerEq.md +6 -0
- package/docs/reference/functions/lcm.md +6 -0
- package/docs/reference/functions/leafCount.md +52 -0
- package/docs/reference/functions/leftShift.md +6 -0
- package/docs/reference/functions/log.md +6 -0
- package/docs/reference/functions/log10.md +6 -0
- package/docs/reference/functions/log1p.md +6 -0
- package/docs/reference/functions/log2.md +6 -0
- package/docs/reference/functions/lsolve.md +6 -0
- package/docs/reference/functions/lsolveAll.md +6 -0
- package/docs/reference/functions/lup.md +6 -0
- package/docs/reference/functions/lusolve.md +6 -0
- package/docs/reference/functions/mad.md +6 -0
- package/docs/reference/functions/map.md +6 -0
- package/docs/reference/functions/matrixFromColumns.md +6 -0
- package/docs/reference/functions/matrixFromFunction.md +6 -0
- package/docs/reference/functions/matrixFromRows.md +6 -0
- package/docs/reference/functions/max.md +6 -0
- package/docs/reference/functions/mean.md +6 -0
- package/docs/reference/functions/median.md +6 -0
- package/docs/reference/functions/min.md +6 -0
- package/docs/reference/functions/mod.md +6 -0
- package/docs/reference/functions/mode.md +6 -0
- package/docs/reference/functions/multinomial.md +6 -0
- package/docs/reference/functions/multiply.md +6 -0
- package/docs/reference/functions/norm.md +6 -0
- package/docs/reference/functions/not.md +6 -0
- package/docs/reference/functions/nthRoot.md +6 -0
- package/docs/reference/functions/nthRoots.md +6 -0
- package/docs/reference/functions/numeric.md +6 -0
- package/docs/reference/functions/oct.md +6 -0
- package/docs/reference/functions/ones.md +6 -0
- package/docs/reference/functions/or.md +6 -0
- package/docs/reference/functions/parser.md +6 -0
- package/docs/reference/functions/partitionSelect.md +6 -0
- package/docs/reference/functions/permutations.md +6 -0
- package/docs/reference/functions/pickRandom.md +6 -0
- package/docs/reference/functions/pow.md +6 -0
- package/docs/reference/functions/print.md +6 -0
- package/docs/reference/functions/prod.md +6 -0
- package/docs/reference/functions/qr.md +6 -0
- package/docs/reference/functions/quantileSeq.md +6 -0
- package/docs/reference/functions/random.md +6 -0
- package/docs/reference/functions/randomInt.md +6 -0
- package/docs/reference/functions/range.md +6 -0
- package/docs/reference/functions/rationalize.md +7 -1
- package/docs/reference/functions/re.md +6 -0
- package/docs/reference/functions/reshape.md +7 -0
- package/docs/reference/functions/resize.md +6 -0
- package/docs/reference/functions/resolve.md +46 -0
- package/docs/reference/functions/rightArithShift.md +6 -0
- package/docs/reference/functions/rightLogShift.md +6 -0
- package/docs/reference/functions/rotate.md +6 -0
- package/docs/reference/functions/rotationMatrix.md +6 -0
- package/docs/reference/functions/round.md +6 -0
- package/docs/reference/functions/row.md +6 -0
- package/docs/reference/functions/sec.md +6 -0
- package/docs/reference/functions/sech.md +6 -0
- package/docs/reference/functions/setCartesian.md +6 -0
- package/docs/reference/functions/setDifference.md +6 -0
- package/docs/reference/functions/setDistinct.md +6 -0
- package/docs/reference/functions/setIntersect.md +6 -0
- package/docs/reference/functions/setIsSubset.md +6 -0
- package/docs/reference/functions/setMultiplicity.md +6 -0
- package/docs/reference/functions/setPowerset.md +6 -0
- package/docs/reference/functions/setSize.md +6 -0
- package/docs/reference/functions/setSymDifference.md +6 -0
- package/docs/reference/functions/setUnion.md +6 -0
- package/docs/reference/functions/sign.md +6 -0
- package/docs/reference/functions/simplify.md +43 -6
- package/docs/reference/functions/simplifyCore.md +50 -0
- package/docs/reference/functions/sin.md +6 -0
- package/docs/reference/functions/sinh.md +6 -0
- package/docs/reference/functions/size.md +6 -0
- package/docs/reference/functions/slu.md +6 -0
- package/docs/reference/functions/smaller.md +6 -0
- package/docs/reference/functions/smallerEq.md +6 -0
- package/docs/reference/functions/sort.md +6 -0
- package/docs/reference/functions/sqrt.md +6 -0
- package/docs/reference/functions/sqrtm.md +6 -0
- package/docs/reference/functions/square.md +6 -0
- package/docs/reference/functions/squeeze.md +6 -0
- package/docs/reference/functions/std.md +6 -0
- package/docs/reference/functions/stirlingS2.md +6 -0
- package/docs/reference/functions/subset.md +6 -0
- package/docs/reference/functions/subtract.md +6 -0
- package/docs/reference/functions/sum.md +6 -0
- package/docs/reference/functions/tan.md +6 -0
- package/docs/reference/functions/tanh.md +6 -0
- package/docs/reference/functions/to.md +6 -0
- package/docs/reference/functions/trace.md +6 -0
- package/docs/reference/functions/transpose.md +6 -0
- package/docs/reference/functions/typeOf.md +6 -0
- package/docs/reference/functions/unaryMinus.md +6 -0
- package/docs/reference/functions/unaryPlus.md +6 -0
- package/docs/reference/functions/unequal.md +6 -0
- package/docs/reference/functions/usolve.md +6 -0
- package/docs/reference/functions/usolveAll.md +6 -0
- package/docs/reference/functions/variance.md +6 -0
- package/docs/reference/functions/xgcd.md +6 -0
- package/docs/reference/functions/xor.md +6 -0
- package/docs/reference/functions/zeros.md +6 -0
- package/docs/reference/functions.md +3 -0
- package/lib/browser/math.js +6 -6
- package/lib/browser/math.js.map +1 -1
- package/lib/cjs/core/create.js +4 -4
- package/lib/cjs/core/function/import.js +3 -3
- package/lib/cjs/core/function/typed.js +2 -2
- package/lib/cjs/defaultInstance.js +3 -3
- package/lib/cjs/entry/allFactoriesAny.js +1 -1
- package/lib/cjs/entry/allFactoriesNumber.js +1 -1
- package/lib/cjs/entry/configReadonly.js +1 -1
- package/lib/cjs/entry/dependenciesAny/dependenciesLeafCount.generated.js +23 -0
- package/lib/cjs/entry/dependenciesAny/dependenciesRationalize.generated.js +3 -0
- package/lib/cjs/entry/dependenciesAny/dependenciesResolve.generated.js +32 -0
- package/lib/cjs/entry/dependenciesAny/dependenciesSimplify.generated.js +6 -0
- package/lib/cjs/entry/dependenciesAny/dependenciesSimplifyCore.generated.js +65 -0
- package/lib/cjs/entry/dependenciesAny.generated.js +24 -0
- package/lib/cjs/entry/dependenciesNumber/dependenciesRationalize.generated.js +3 -0
- package/lib/cjs/entry/dependenciesNumber/dependenciesResolve.generated.js +32 -0
- package/lib/cjs/entry/dependenciesNumber/dependenciesSimplify.generated.js +6 -0
- package/lib/cjs/entry/dependenciesNumber/dependenciesSimplifyCore.generated.js +65 -0
- package/lib/cjs/entry/dependenciesNumber.generated.js +16 -0
- package/lib/cjs/entry/impureFunctionsAny.generated.js +67 -29
- package/lib/cjs/entry/impureFunctionsNumber.generated.js +96 -64
- package/lib/cjs/entry/pureFunctionsAny.generated.js +53 -55
- package/lib/cjs/entry/pureFunctionsNumber.generated.js +46 -48
- package/lib/cjs/expression/Help.js +4 -0
- package/lib/cjs/expression/Parser.js +1 -1
- package/lib/cjs/expression/embeddedDocs/core/typed.js +1 -1
- package/lib/cjs/expression/embeddedDocs/embeddedDocs.js +14 -5
- package/lib/cjs/expression/embeddedDocs/function/algebra/leafCount.js +15 -0
- package/lib/cjs/expression/embeddedDocs/function/algebra/resolve.js +16 -0
- package/lib/cjs/expression/embeddedDocs/function/algebra/simplify.js +1 -1
- package/lib/cjs/expression/embeddedDocs/function/algebra/simplifyCore.js +15 -0
- package/lib/cjs/expression/node/FunctionNode.js +7 -7
- package/lib/cjs/expression/node/IndexNode.js +1 -1
- package/lib/cjs/expression/node/Node.js +3 -3
- package/lib/cjs/expression/node/ObjectNode.js +1 -1
- package/lib/cjs/expression/node/utils/access.js +1 -1
- package/lib/cjs/expression/node/utils/assign.js +1 -1
- package/lib/cjs/expression/parse.js +13 -13
- package/lib/cjs/factoriesAny.js +24 -0
- package/lib/cjs/factoriesNumber.js +16 -0
- package/lib/cjs/function/algebra/decomposition/qr.js +1 -1
- package/lib/cjs/function/algebra/leafCount.js +66 -0
- package/lib/cjs/function/algebra/rationalize.js +24 -41
- package/lib/cjs/function/algebra/resolve.js +106 -0
- package/lib/cjs/function/algebra/simplify/simplifyConstant.js +5 -5
- package/lib/cjs/function/algebra/simplify/util.js +171 -33
- package/lib/cjs/function/algebra/simplify.js +554 -207
- package/lib/cjs/function/algebra/{simplify/simplifyCore.js → simplifyCore.js} +67 -43
- package/lib/cjs/function/algebra/solver/lsolveAll.js +2 -2
- package/lib/cjs/function/algebra/solver/usolveAll.js +2 -2
- package/lib/cjs/function/arithmetic/ceil.js +3 -3
- package/lib/cjs/function/arithmetic/floor.js +3 -3
- package/lib/cjs/function/arithmetic/invmod.js +1 -1
- package/lib/cjs/function/arithmetic/norm.js +1 -1
- package/lib/cjs/function/arithmetic/round.js +1 -1
- package/lib/cjs/function/matrix/eigs/complexEigs.js +5 -5
- package/lib/cjs/function/matrix/matrixFromColumns.js +1 -1
- package/lib/cjs/function/matrix/matrixFromRows.js +1 -1
- package/lib/cjs/function/probability/util/seededRNG.js +2 -2
- package/lib/cjs/function/relational/compareNatural.js +6 -6
- package/lib/cjs/header.js +2 -2
- package/lib/cjs/plain/bignumber/index.js +1 -1
- package/lib/cjs/plain/number/combinations.js +18 -6
- package/lib/cjs/type/bignumber/BigNumber.js +2 -2
- package/lib/cjs/type/bignumber/function/bignumber.js +1 -1
- package/lib/cjs/type/boolean.js +2 -2
- package/lib/cjs/type/complex/Complex.js +14 -14
- package/lib/cjs/type/complex/function/complex.js +1 -1
- package/lib/cjs/type/fraction/Fraction.js +6 -6
- package/lib/cjs/type/fraction/function/fraction.js +1 -1
- package/lib/cjs/type/matrix/DenseMatrix.js +5 -5
- package/lib/cjs/type/matrix/SparseMatrix.js +2 -2
- package/lib/cjs/type/number.js +1 -1
- package/lib/cjs/type/string.js +2 -2
- package/lib/cjs/type/unit/Unit.js +8 -8
- package/lib/cjs/utils/customs.js +2 -2
- package/lib/cjs/utils/emitter.js +1 -1
- package/lib/cjs/utils/function.js +2 -2
- package/lib/cjs/utils/is.js +6 -6
- package/lib/cjs/utils/latex.js +3 -3
- package/lib/cjs/utils/lruQueue.js +1 -1
- package/lib/cjs/utils/map.js +3 -3
- package/lib/cjs/utils/object.js +2 -2
- package/lib/cjs/utils/snapshot.js +7 -7
- package/lib/cjs/utils/string.js +2 -2
- package/lib/cjs/version.js +1 -1
- package/lib/esm/entry/dependenciesAny/dependenciesLeafCount.generated.js +12 -0
- package/lib/esm/entry/dependenciesAny/dependenciesRationalize.generated.js +2 -0
- package/lib/esm/entry/dependenciesAny/dependenciesResolve.generated.js +18 -0
- package/lib/esm/entry/dependenciesAny/dependenciesSimplify.generated.js +4 -0
- package/lib/esm/entry/dependenciesAny/dependenciesSimplifyCore.generated.js +40 -0
- package/lib/esm/entry/dependenciesAny.generated.js +3 -0
- package/lib/esm/entry/dependenciesNumber/dependenciesRationalize.generated.js +2 -0
- package/lib/esm/entry/dependenciesNumber/dependenciesResolve.generated.js +18 -0
- package/lib/esm/entry/dependenciesNumber/dependenciesSimplify.generated.js +4 -0
- package/lib/esm/entry/dependenciesNumber/dependenciesSimplifyCore.generated.js +40 -0
- package/lib/esm/entry/dependenciesNumber.generated.js +2 -0
- package/lib/esm/entry/impureFunctionsAny.generated.js +57 -22
- package/lib/esm/entry/impureFunctionsNumber.generated.js +83 -53
- package/lib/esm/entry/pureFunctionsAny.generated.js +42 -42
- package/lib/esm/entry/pureFunctionsNumber.generated.js +32 -32
- package/lib/esm/expression/Help.js +4 -0
- package/lib/esm/expression/embeddedDocs/core/typed.js +1 -1
- package/lib/esm/expression/embeddedDocs/embeddedDocs.js +6 -0
- package/lib/esm/expression/embeddedDocs/function/algebra/leafCount.js +8 -0
- package/lib/esm/expression/embeddedDocs/function/algebra/resolve.js +9 -0
- package/lib/esm/expression/embeddedDocs/function/algebra/simplify.js +1 -1
- package/lib/esm/expression/embeddedDocs/function/algebra/simplifyCore.js +8 -0
- package/lib/esm/factoriesAny.js +3 -0
- package/lib/esm/factoriesNumber.js +2 -0
- package/lib/esm/function/algebra/leafCount.js +59 -0
- package/lib/esm/function/algebra/rationalize.js +24 -40
- package/lib/esm/function/algebra/resolve.js +95 -0
- package/lib/esm/function/algebra/simplify/simplifyConstant.js +3 -3
- package/lib/esm/function/algebra/simplify/util.js +170 -34
- package/lib/esm/function/algebra/simplify.js +549 -202
- package/lib/esm/function/algebra/{simplify/simplifyCore.js → simplifyCore.js} +60 -44
- package/lib/esm/plain/number/combinations.js +18 -6
- package/lib/esm/version.js +1 -1
- package/package.json +15 -10
- package/types/index.d.ts +44 -2
- package/lib/cjs/function/algebra/simplify/resolve.js +0 -76
- package/lib/esm/function/algebra/simplify/resolve.js +0 -67
@@ -15,18 +15,14 @@ var _factory = require("../../utils/factory.js");
|
|
15
15
|
|
16
16
|
var _util = require("./simplify/util.js");
|
17
17
|
|
18
|
-
var _simplifyCore = require("./simplify/simplifyCore.js");
|
19
|
-
|
20
18
|
var _simplifyConstant = require("./simplify/simplifyConstant.js");
|
21
19
|
|
22
|
-
var _resolve = require("./simplify/resolve.js");
|
23
|
-
|
24
20
|
var _object = require("../../utils/object.js");
|
25
21
|
|
26
22
|
var _map = require("../../utils/map.js");
|
27
23
|
|
28
24
|
var name = 'simplify';
|
29
|
-
var dependencies = ['config', 'typed', 'parse', 'add', 'subtract', 'multiply', 'divide', 'pow', 'isZero', 'equal', '?fraction', '?bignumber', 'mathWithTransform', 'matrix', 'AccessorNode', 'ArrayNode', 'ConstantNode', 'FunctionNode', 'IndexNode', 'ObjectNode', 'OperatorNode', 'ParenthesisNode', 'SymbolNode'];
|
25
|
+
var dependencies = ['config', 'typed', 'parse', 'add', 'subtract', 'multiply', 'divide', 'pow', 'isZero', 'equal', 'resolve', 'simplifyCore', '?fraction', '?bignumber', 'mathWithTransform', 'matrix', 'AccessorNode', 'ArrayNode', 'ConstantNode', 'FunctionNode', 'IndexNode', 'ObjectNode', 'OperatorNode', 'ParenthesisNode', 'SymbolNode'];
|
30
26
|
var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, function (_ref) {
|
31
27
|
var config = _ref.config,
|
32
28
|
typed = _ref.typed,
|
@@ -38,6 +34,8 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
38
34
|
pow = _ref.pow,
|
39
35
|
isZero = _ref.isZero,
|
40
36
|
equal = _ref.equal,
|
37
|
+
resolve = _ref.resolve,
|
38
|
+
simplifyCore = _ref.simplifyCore,
|
41
39
|
fraction = _ref.fraction,
|
42
40
|
bignumber = _ref.bignumber,
|
43
41
|
mathWithTransform = _ref.mathWithTransform,
|
@@ -67,41 +65,23 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
67
65
|
OperatorNode: OperatorNode,
|
68
66
|
SymbolNode: SymbolNode
|
69
67
|
});
|
70
|
-
var simplifyCore = (0, _simplifyCore.createSimplifyCore)({
|
71
|
-
equal: equal,
|
72
|
-
isZero: isZero,
|
73
|
-
add: add,
|
74
|
-
subtract: subtract,
|
75
|
-
multiply: multiply,
|
76
|
-
divide: divide,
|
77
|
-
pow: pow,
|
78
|
-
AccessorNode: AccessorNode,
|
79
|
-
ArrayNode: ArrayNode,
|
80
|
-
ConstantNode: ConstantNode,
|
81
|
-
FunctionNode: FunctionNode,
|
82
|
-
IndexNode: IndexNode,
|
83
|
-
ObjectNode: ObjectNode,
|
84
|
-
OperatorNode: OperatorNode,
|
85
|
-
ParenthesisNode: ParenthesisNode
|
86
|
-
});
|
87
|
-
var resolve = (0, _resolve.createResolve)({
|
88
|
-
parse: parse,
|
89
|
-
FunctionNode: FunctionNode,
|
90
|
-
OperatorNode: OperatorNode,
|
91
|
-
ParenthesisNode: ParenthesisNode
|
92
|
-
});
|
93
68
|
|
94
69
|
var _createUtil = (0, _util.createUtil)({
|
95
70
|
FunctionNode: FunctionNode,
|
96
71
|
OperatorNode: OperatorNode,
|
97
72
|
SymbolNode: SymbolNode
|
98
73
|
}),
|
74
|
+
hasProperty = _createUtil.hasProperty,
|
99
75
|
isCommutative = _createUtil.isCommutative,
|
100
76
|
isAssociative = _createUtil.isAssociative,
|
77
|
+
mergeContext = _createUtil.mergeContext,
|
101
78
|
flatten = _createUtil.flatten,
|
102
79
|
unflattenr = _createUtil.unflattenr,
|
103
80
|
unflattenl = _createUtil.unflattenl,
|
104
|
-
createMakeNodeFunction = _createUtil.createMakeNodeFunction
|
81
|
+
createMakeNodeFunction = _createUtil.createMakeNodeFunction,
|
82
|
+
defaultContext = _createUtil.defaultContext,
|
83
|
+
realContext = _createUtil.realContext,
|
84
|
+
positiveContext = _createUtil.positiveContext;
|
105
85
|
/**
|
106
86
|
* Simplify an expression tree.
|
107
87
|
*
|
@@ -132,6 +112,19 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
132
112
|
* The default list of rules is exposed on the function as `simplify.rules`
|
133
113
|
* and can be used as a basis to built a set of custom rules.
|
134
114
|
*
|
115
|
+
* To specify a rule as a string, separate the left and right pattern by '->'
|
116
|
+
* When specifying a rule as an object, the following keys are meaningful:
|
117
|
+
* - l - the left pattern
|
118
|
+
* - r - the right pattern
|
119
|
+
* - s - in lieu of l and r, the string form that is broken at -> to give them
|
120
|
+
* - repeat - whether to repeat this rule until the expression stabilizes
|
121
|
+
* - assuming - gives a context object, as in the 'context' option to
|
122
|
+
* simplify. Every property in the context object must match the current
|
123
|
+
* context in order, or else the rule will not be applied.
|
124
|
+
* - imposeContext - gives a context object, as in the 'context' option to
|
125
|
+
* simplify. Any settings specified will override the incoming context
|
126
|
+
* for all matches of this rule.
|
127
|
+
*
|
135
128
|
* For more details on the theory, see:
|
136
129
|
*
|
137
130
|
* - [Strategies for simplifying math expressions (Stackoverflow)](https://stackoverflow.com/questions/7540227/strategies-for-simplifying-math-expressions)
|
@@ -140,12 +133,28 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
140
133
|
* An optional `options` argument can be passed as last argument of `simplify`.
|
141
134
|
* Currently available options (defaults in parentheses):
|
142
135
|
* - `consoleDebug` (false): whether to write the expression being simplified
|
143
|
-
|
136
|
+
* and any changes to it, along with the rule responsible, to console
|
137
|
+
* - `context` (simplify.defaultContext): an object giving properties of
|
138
|
+
* each operator, which determine what simplifications are allowed. The
|
139
|
+
* currently meaningful properties are commutative, associative,
|
140
|
+
* total (whether the operation is defined for all arguments), and
|
141
|
+
* trivial (whether the operation applied to a single argument leaves
|
142
|
+
* that argument unchanged). The default context is very permissive and
|
143
|
+
* allows almost all simplifications. Only properties differing from
|
144
|
+
* the default need to be specified; the default context is used as a
|
145
|
+
* fallback. Additional contexts `simplify.realContext` and
|
146
|
+
* `simplify.positiveContext` are supplied to cause simplify to perform
|
147
|
+
* just simplifications guaranteed to preserve all values of the expression
|
148
|
+
* assuming all variables and subexpressions are real numbers or
|
149
|
+
* positive real numbers, respectively. (Note that these are in some cases
|
150
|
+
* more restrictive than the default context; for example, the default
|
151
|
+
* context will allow `x/x` to simplify to 1, whereas
|
152
|
+
* `simplify.realContext` will not, as `0/0` is not equal to 1.)
|
144
153
|
* - `exactFractions` (true): whether to try to convert all constants to
|
145
|
-
|
154
|
+
* exact rational numbers.
|
146
155
|
* - `fractionsLimit` (10000): when `exactFractions` is true, constants will
|
147
|
-
|
148
|
-
|
156
|
+
* be expressed as fractions only when both numerator and denominator
|
157
|
+
* are smaller than `fractionsLimit`.
|
149
158
|
*
|
150
159
|
* Syntax:
|
151
160
|
*
|
@@ -168,7 +177,7 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
168
177
|
*
|
169
178
|
* See also:
|
170
179
|
*
|
171
|
-
* derivative, parse,
|
180
|
+
* simplifyCore, derivative, evaluate, parse, rationalize, resolve
|
172
181
|
*
|
173
182
|
* @param {Node | string} expr
|
174
183
|
* The expression to be simplified
|
@@ -217,7 +226,7 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
217
226
|
},
|
218
227
|
'Node, Array, Map, Object': function NodeArrayMapObject(expr, rules, scope, options) {
|
219
228
|
var debug = options.consoleDebug;
|
220
|
-
rules = _buildRules(rules);
|
229
|
+
rules = _buildRules(rules, options.context);
|
221
230
|
var res = resolve(expr, scope);
|
222
231
|
res = removeParens(res);
|
223
232
|
var visited = {};
|
@@ -239,8 +248,8 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
239
248
|
res = rules[i](res, options);
|
240
249
|
if (debug) rulestr = rules[i].name;
|
241
250
|
} else {
|
242
|
-
flatten(res);
|
243
|
-
res = applyRule(res, rules[i]);
|
251
|
+
flatten(res, options.context);
|
252
|
+
res = applyRule(res, rules[i], options.context);
|
244
253
|
|
245
254
|
if (debug) {
|
246
255
|
rulestr = "".concat(rules[i].l.toString(), " -> ").concat(rules[i].r.toString());
|
@@ -257,8 +266,12 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
257
266
|
laststr = newstr;
|
258
267
|
}
|
259
268
|
}
|
269
|
+
/* Use left-heavy binary tree internally,
|
270
|
+
* since custom rule functions may expect it
|
271
|
+
*/
|
260
272
|
|
261
|
-
|
273
|
+
|
274
|
+
unflattenl(res, options.context);
|
262
275
|
}
|
263
276
|
|
264
277
|
str = res.toString({
|
@@ -269,8 +282,9 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
269
282
|
return res;
|
270
283
|
}
|
271
284
|
});
|
272
|
-
simplify.
|
273
|
-
simplify.
|
285
|
+
simplify.defaultContext = defaultContext;
|
286
|
+
simplify.realContext = realContext;
|
287
|
+
simplify.positiveContext = positiveContext;
|
274
288
|
|
275
289
|
function removeParens(node) {
|
276
290
|
return node.transform(function (node, path, parent) {
|
@@ -280,8 +294,8 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
280
294
|
|
281
295
|
|
282
296
|
var SUPPORTED_CONSTANTS = {
|
283
|
-
|
284
|
-
|
297
|
+
true: true,
|
298
|
+
false: true,
|
285
299
|
e: true,
|
286
300
|
i: true,
|
287
301
|
Infinity: true,
|
@@ -322,41 +336,147 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
322
336
|
// Note initially we tend constants to the right because like-term
|
323
337
|
// collection prefers the left, and we would rather collect nonconstants
|
324
338
|
{
|
325
|
-
|
326
|
-
|
327
|
-
|
328
|
-
|
329
|
-
|
330
|
-
|
331
|
-
|
332
|
-
{
|
339
|
+
s: 'n-n1 -> n+-n1',
|
340
|
+
// temporarily replace 'subtract' so we can further flatten the 'add' operator
|
341
|
+
assuming: {
|
342
|
+
subtract: {
|
343
|
+
total: true
|
344
|
+
}
|
345
|
+
}
|
346
|
+
}, {
|
347
|
+
s: '-(c*v) -> v * (-c)',
|
348
|
+
// make non-constant terms positive
|
349
|
+
assuming: {
|
350
|
+
multiply: {
|
351
|
+
commutative: true
|
352
|
+
},
|
353
|
+
subtract: {
|
354
|
+
total: true
|
355
|
+
}
|
356
|
+
}
|
357
|
+
}, {
|
358
|
+
s: '-(c*v) -> (-c) * v',
|
359
|
+
// non-commutative version, part 1
|
360
|
+
assuming: {
|
361
|
+
multiply: {
|
362
|
+
commutative: false
|
363
|
+
},
|
364
|
+
subtract: {
|
365
|
+
total: true
|
366
|
+
}
|
367
|
+
}
|
368
|
+
}, {
|
369
|
+
s: '-(v*c) -> v * (-c)',
|
370
|
+
// non-commutative version, part 2
|
371
|
+
assuming: {
|
372
|
+
multiply: {
|
373
|
+
commutative: false
|
374
|
+
},
|
375
|
+
subtract: {
|
376
|
+
total: true
|
377
|
+
}
|
378
|
+
}
|
379
|
+
}, {
|
380
|
+
l: '-(n1/n2)',
|
381
|
+
r: '-n1/n2'
|
382
|
+
}, {
|
333
383
|
l: '-v',
|
334
384
|
r: 'v * (-1)'
|
335
|
-
},
|
385
|
+
}, // finish making non-constant terms positive
|
386
|
+
{
|
387
|
+
l: '(n1 + n2)*(-1)',
|
388
|
+
r: 'n1*(-1) + n2*(-1)',
|
389
|
+
repeat: true
|
390
|
+
}, // expand negations to achieve as much sign cancellation as possible
|
391
|
+
{
|
336
392
|
l: 'n/n1^n2',
|
337
393
|
r: 'n*n1^-n2'
|
338
394
|
}, // temporarily replace 'divide' so we can further flatten the 'multiply' operator
|
339
395
|
{
|
340
396
|
l: 'n/n1',
|
341
397
|
r: 'n*n1^-1'
|
342
|
-
},
|
398
|
+
}, {
|
399
|
+
s: '(n1*n2)^n3 -> n1^n3 * n2^n3',
|
400
|
+
assuming: {
|
401
|
+
multiply: {
|
402
|
+
commutative: true
|
403
|
+
}
|
404
|
+
}
|
405
|
+
}, {
|
406
|
+
s: '(n1*n2)^(-1) -> n2^(-1) * n1^(-1)',
|
407
|
+
assuming: {
|
408
|
+
multiply: {
|
409
|
+
commutative: false
|
410
|
+
}
|
411
|
+
}
|
412
|
+
}, // expand nested exponentiation
|
343
413
|
{
|
344
|
-
|
345
|
-
|
346
|
-
|
414
|
+
s: '(n ^ n1) ^ n2 -> n ^ (n1 * n2)',
|
415
|
+
assuming: {
|
416
|
+
divide: {
|
417
|
+
total: true
|
418
|
+
}
|
419
|
+
} // 1/(1/n) = n needs 1/n to exist
|
420
|
+
|
421
|
+
}, // collect like factors; into a sum, only do this for nonconstants
|
347
422
|
{
|
423
|
+
l: ' v * ( v * n1 + n2)',
|
424
|
+
r: 'v^2 * n1 + v * n2'
|
425
|
+
}, {
|
426
|
+
s: ' v * (v^n4 * n1 + n2) -> v^(1+n4) * n1 + v * n2',
|
427
|
+
assuming: {
|
428
|
+
divide: {
|
429
|
+
total: true
|
430
|
+
}
|
431
|
+
} // v*1/v = v^(1+-1) needs 1/v
|
432
|
+
|
433
|
+
}, {
|
434
|
+
s: 'v^n3 * ( v * n1 + n2) -> v^(n3+1) * n1 + v^n3 * n2',
|
435
|
+
assuming: {
|
436
|
+
divide: {
|
437
|
+
total: true
|
438
|
+
}
|
439
|
+
}
|
440
|
+
}, {
|
441
|
+
s: 'v^n3 * (v^n4 * n1 + n2) -> v^(n3+n4) * n1 + v^n3 * n2',
|
442
|
+
assuming: {
|
443
|
+
divide: {
|
444
|
+
total: true
|
445
|
+
}
|
446
|
+
}
|
447
|
+
}, {
|
348
448
|
l: 'n*n',
|
349
449
|
r: 'n^2'
|
350
450
|
}, {
|
351
|
-
|
352
|
-
|
451
|
+
s: 'n * n^n1 -> n^(n1+1)',
|
452
|
+
assuming: {
|
453
|
+
divide: {
|
454
|
+
total: true
|
455
|
+
}
|
456
|
+
} // n*1/n = n^(-1+1) needs 1/n
|
457
|
+
|
353
458
|
}, {
|
354
|
-
|
355
|
-
|
356
|
-
|
459
|
+
s: 'n^n1 * n^n2 -> n^(n1+n2)',
|
460
|
+
assuming: {
|
461
|
+
divide: {
|
462
|
+
total: true
|
463
|
+
}
|
464
|
+
} // ditto for n^2*1/n^2
|
465
|
+
|
466
|
+
}, // Unfortunately, to deal with more complicated cancellations, it
|
467
|
+
// becomes necessary to simplify constants twice per pass. It's not
|
468
|
+
// terribly expensive compared to matching rules, so this should not
|
469
|
+
// pose a performance problem.
|
470
|
+
simplifyConstant, // First: before collecting like terms
|
471
|
+
// collect like terms
|
357
472
|
{
|
358
|
-
|
359
|
-
|
473
|
+
s: 'n+n -> 2*n',
|
474
|
+
assuming: {
|
475
|
+
add: {
|
476
|
+
total: true
|
477
|
+
}
|
478
|
+
} // 2 = 1 + 1 needs to exist
|
479
|
+
|
360
480
|
}, {
|
361
481
|
l: 'n+-n',
|
362
482
|
r: '0'
|
@@ -369,33 +489,91 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
369
489
|
r: 'n3*(n1+n2)'
|
370
490
|
}, // All sub-monomials tried there.
|
371
491
|
{
|
492
|
+
l: 'n3^(-n4)*n1 + n3 * n2',
|
493
|
+
r: 'n3^(-n4)*(n1 + n3^(n4+1) *n2)'
|
494
|
+
}, {
|
495
|
+
l: 'n3^(-n4)*n1 + n3^n5 * n2',
|
496
|
+
r: 'n3^(-n4)*(n1 + n3^(n4+n5)*n2)'
|
497
|
+
}, {
|
498
|
+
s: 'n*v + v -> (n+1)*v',
|
499
|
+
// noncommutative additional cases
|
500
|
+
assuming: {
|
501
|
+
multiply: {
|
502
|
+
commutative: false
|
503
|
+
}
|
504
|
+
}
|
505
|
+
}, {
|
506
|
+
s: 'n1*n3 + n2*n3 -> (n1+n2)*n3',
|
507
|
+
assuming: {
|
508
|
+
multiply: {
|
509
|
+
commutative: false
|
510
|
+
}
|
511
|
+
}
|
512
|
+
}, {
|
513
|
+
s: 'n1*n3^(-n4) + n2 * n3 -> (n1 + n2*n3^(n4 + 1))*n3^(-n4)',
|
514
|
+
assuming: {
|
515
|
+
multiply: {
|
516
|
+
commutative: false
|
517
|
+
}
|
518
|
+
}
|
519
|
+
}, {
|
520
|
+
s: 'n1*n3^(-n4) + n2 * n3^n5 -> (n1 + n2*n3^(n4 + n5))*n3^(-n4)',
|
521
|
+
assuming: {
|
522
|
+
multiply: {
|
523
|
+
commutative: false
|
524
|
+
}
|
525
|
+
}
|
526
|
+
}, {
|
372
527
|
l: 'n*c + c',
|
373
528
|
r: '(n+1)*c'
|
374
|
-
},
|
375
|
-
|
376
|
-
|
377
|
-
|
378
|
-
|
379
|
-
|
380
|
-
|
381
|
-
|
382
|
-
|
529
|
+
}, {
|
530
|
+
s: 'c*n + c -> c*(n+1)',
|
531
|
+
assuming: {
|
532
|
+
multiply: {
|
533
|
+
commutative: false
|
534
|
+
}
|
535
|
+
}
|
536
|
+
}, simplifyConstant, // Second: before returning expressions to "standard form"
|
537
|
+
// make factors positive (and undo 'make non-constant terms positive')
|
383
538
|
{
|
384
|
-
|
385
|
-
|
539
|
+
s: '(-n)*n1 -> -(n*n1)',
|
540
|
+
assuming: {
|
541
|
+
subtract: {
|
542
|
+
total: true
|
543
|
+
}
|
544
|
+
}
|
545
|
+
}, {
|
546
|
+
s: 'n1*(-n) -> -(n1*n)',
|
547
|
+
// in case * non-commutative
|
548
|
+
assuming: {
|
549
|
+
subtract: {
|
550
|
+
total: true
|
551
|
+
},
|
552
|
+
multiply: {
|
553
|
+
commutative: false
|
554
|
+
}
|
555
|
+
}
|
386
556
|
}, // final ordering of constants
|
387
557
|
{
|
388
|
-
|
389
|
-
|
390
|
-
|
558
|
+
s: 'c+v -> v+c',
|
559
|
+
assuming: {
|
560
|
+
add: {
|
561
|
+
commutative: true
|
562
|
+
}
|
563
|
+
},
|
564
|
+
imposeContext: {
|
391
565
|
add: {
|
392
566
|
commutative: false
|
393
567
|
}
|
394
568
|
}
|
395
569
|
}, {
|
396
|
-
|
397
|
-
|
398
|
-
|
570
|
+
s: 'v*c -> c*v',
|
571
|
+
assuming: {
|
572
|
+
multiply: {
|
573
|
+
commutative: true
|
574
|
+
}
|
575
|
+
},
|
576
|
+
imposeContext: {
|
399
577
|
multiply: {
|
400
578
|
commutative: false
|
401
579
|
}
|
@@ -407,36 +585,127 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
407
585
|
r: 'n-n1'
|
408
586
|
}, // undo replace 'subtract'
|
409
587
|
{
|
410
|
-
|
411
|
-
|
412
|
-
|
413
|
-
|
414
|
-
|
415
|
-
|
588
|
+
s: 'n*(n1^-1) -> n/n1',
|
589
|
+
// undo replace 'divide'; for * commutative
|
590
|
+
assuming: {
|
591
|
+
multiply: {
|
592
|
+
commutative: true
|
593
|
+
}
|
594
|
+
} // o.w. / not conventional
|
595
|
+
|
416
596
|
}, {
|
417
|
-
|
418
|
-
|
597
|
+
s: 'n*n1^-n2 -> n/n1^n2',
|
598
|
+
assuming: {
|
599
|
+
multiply: {
|
600
|
+
commutative: true
|
601
|
+
}
|
602
|
+
} // o.w. / not conventional
|
603
|
+
|
419
604
|
}, {
|
420
|
-
|
421
|
-
|
422
|
-
|
605
|
+
s: 'n^-1 -> 1/n',
|
606
|
+
assuming: {
|
607
|
+
multiply: {
|
608
|
+
commutative: true
|
609
|
+
}
|
610
|
+
} // o.w. / not conventional
|
611
|
+
|
612
|
+
}, {
|
613
|
+
l: 'n^1',
|
614
|
+
r: 'n'
|
615
|
+
}, // can be produced by power cancellation
|
423
616
|
{
|
424
|
-
|
425
|
-
|
426
|
-
|
427
|
-
|
617
|
+
s: 'n*(n1/n2) -> (n*n1)/n2',
|
618
|
+
// '*' before '/'
|
619
|
+
assuming: {
|
620
|
+
multiply: {
|
621
|
+
associative: true
|
622
|
+
}
|
623
|
+
}
|
624
|
+
}, {
|
625
|
+
s: 'n-(n1+n2) -> n-n1-n2',
|
626
|
+
// '-' before '+'
|
627
|
+
assuming: {
|
628
|
+
addition: {
|
629
|
+
associative: true,
|
630
|
+
commutative: true
|
631
|
+
}
|
632
|
+
}
|
633
|
+
}, // { l: '(n1/n2)/n3', r: 'n1/(n2*n3)' },
|
428
634
|
// { l: '(n*n1)/(n*n2)', r: 'n1/n2' },
|
635
|
+
// simplifyConstant can leave an extra factor of 1, which can always
|
636
|
+
// be eliminated, since the identity always commutes
|
429
637
|
{
|
430
638
|
l: '1*n',
|
431
|
-
r: 'n'
|
432
|
-
|
433
|
-
|
434
|
-
|
435
|
-
|
639
|
+
r: 'n',
|
640
|
+
imposeContext: {
|
641
|
+
multiply: {
|
642
|
+
commutative: true
|
643
|
+
}
|
644
|
+
}
|
645
|
+
}, {
|
646
|
+
s: 'n1/(n2/n3) -> (n1*n3)/n2',
|
647
|
+
assuming: {
|
648
|
+
multiply: {
|
649
|
+
associative: true
|
650
|
+
}
|
651
|
+
}
|
436
652
|
}, {
|
437
653
|
l: 'n1/(-n2)',
|
438
654
|
r: '-n1/n2'
|
439
655
|
}];
|
656
|
+
/**
|
657
|
+
* Takes any rule object as allowed by the specification in simplify
|
658
|
+
* and puts it in a standard form used by applyRule
|
659
|
+
*/
|
660
|
+
|
661
|
+
function _canonicalizeRule(ruleObject, context) {
|
662
|
+
var newRule = {};
|
663
|
+
|
664
|
+
if (ruleObject.s) {
|
665
|
+
var lr = ruleObject.s.split('->');
|
666
|
+
|
667
|
+
if (lr.length === 2) {
|
668
|
+
newRule.l = lr[0];
|
669
|
+
newRule.r = lr[1];
|
670
|
+
} else {
|
671
|
+
throw SyntaxError('Could not parse rule: ' + ruleObject.s);
|
672
|
+
}
|
673
|
+
} else {
|
674
|
+
newRule.l = ruleObject.l;
|
675
|
+
newRule.r = ruleObject.r;
|
676
|
+
}
|
677
|
+
|
678
|
+
newRule.l = removeParens(parse(newRule.l));
|
679
|
+
newRule.r = removeParens(parse(newRule.r));
|
680
|
+
|
681
|
+
for (var _i = 0, _arr = ['imposeContext', 'repeat', 'assuming']; _i < _arr.length; _i++) {
|
682
|
+
var prop = _arr[_i];
|
683
|
+
|
684
|
+
if (prop in ruleObject) {
|
685
|
+
newRule[prop] = ruleObject[prop];
|
686
|
+
}
|
687
|
+
}
|
688
|
+
|
689
|
+
if (ruleObject.evaluate) {
|
690
|
+
newRule.evaluate = parse(ruleObject.evaluate);
|
691
|
+
}
|
692
|
+
|
693
|
+
if (isAssociative(newRule.l, context)) {
|
694
|
+
var makeNode = createMakeNodeFunction(newRule.l);
|
695
|
+
|
696
|
+
var expandsym = _getExpandPlaceholderSymbol();
|
697
|
+
|
698
|
+
newRule.expanded = {};
|
699
|
+
newRule.expanded.l = makeNode([newRule.l.clone(), expandsym]); // Push the expandsym into the deepest possible branch.
|
700
|
+
// This helps to match the newRule against nodes returned from getSplits() later on.
|
701
|
+
|
702
|
+
flatten(newRule.expanded.l, context);
|
703
|
+
unflattenr(newRule.expanded.l, context);
|
704
|
+
newRule.expanded.r = makeNode([newRule.r, expandsym]);
|
705
|
+
}
|
706
|
+
|
707
|
+
return newRule;
|
708
|
+
}
|
440
709
|
/**
|
441
710
|
* Parse the string array of rules into nodes
|
442
711
|
*
|
@@ -454,60 +723,26 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
454
723
|
* 'n1 * c1 -> c1 * n1'
|
455
724
|
*/
|
456
725
|
|
457
|
-
|
726
|
+
|
727
|
+
function _buildRules(rules, context) {
|
458
728
|
// Array of rules to be used to simplify expressions
|
459
729
|
var ruleSet = [];
|
460
730
|
|
461
731
|
for (var i = 0; i < rules.length; i++) {
|
462
732
|
var rule = rules[i];
|
463
733
|
var newRule = void 0;
|
464
|
-
var ruleType = (0, _typeof2
|
734
|
+
var ruleType = (0, _typeof2.default)(rule);
|
465
735
|
|
466
736
|
switch (ruleType) {
|
467
737
|
case 'string':
|
468
|
-
{
|
469
|
-
|
470
|
-
|
471
|
-
if (lr.length === 2) {
|
472
|
-
rule = {
|
473
|
-
l: lr[0],
|
474
|
-
r: lr[1]
|
475
|
-
};
|
476
|
-
} else {
|
477
|
-
throw SyntaxError('Could not parse rule: ' + rule);
|
478
|
-
}
|
479
|
-
}
|
738
|
+
rule = {
|
739
|
+
s: rule
|
740
|
+
};
|
480
741
|
|
481
742
|
/* falls through */
|
482
743
|
|
483
744
|
case 'object':
|
484
|
-
newRule =
|
485
|
-
l: removeParens(parse(rule.l)),
|
486
|
-
r: removeParens(parse(rule.r))
|
487
|
-
};
|
488
|
-
|
489
|
-
if (rule.context) {
|
490
|
-
newRule.context = rule.context;
|
491
|
-
}
|
492
|
-
|
493
|
-
if (rule.evaluate) {
|
494
|
-
newRule.evaluate = parse(rule.evaluate);
|
495
|
-
}
|
496
|
-
|
497
|
-
if (isAssociative(newRule.l)) {
|
498
|
-
var makeNode = createMakeNodeFunction(newRule.l);
|
499
|
-
|
500
|
-
var expandsym = _getExpandPlaceholderSymbol();
|
501
|
-
|
502
|
-
newRule.expanded = {};
|
503
|
-
newRule.expanded.l = makeNode([newRule.l.clone(), expandsym]); // Push the expandsym into the deepest possible branch.
|
504
|
-
// This helps to match the newRule against nodes returned from getSplits() later on.
|
505
|
-
|
506
|
-
flatten(newRule.expanded.l);
|
507
|
-
unflattenr(newRule.expanded.l);
|
508
|
-
newRule.expanded.r = makeNode([newRule.r, expandsym]);
|
509
|
-
}
|
510
|
-
|
745
|
+
newRule = _canonicalizeRule(rule, context);
|
511
746
|
break;
|
512
747
|
|
513
748
|
case 'function':
|
@@ -532,90 +767,155 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
532
767
|
return new SymbolNode('_p' + _lastsym++);
|
533
768
|
}
|
534
769
|
|
535
|
-
function mapRule(nodes, rule) {
|
770
|
+
function mapRule(nodes, rule, context) {
|
771
|
+
var resNodes = nodes;
|
772
|
+
|
536
773
|
if (nodes) {
|
537
774
|
for (var i = 0; i < nodes.length; ++i) {
|
538
|
-
|
775
|
+
var newNode = applyRule(nodes[i], rule, context);
|
776
|
+
|
777
|
+
if (newNode !== nodes[i]) {
|
778
|
+
if (resNodes === nodes) {
|
779
|
+
resNodes = nodes.slice();
|
780
|
+
}
|
781
|
+
|
782
|
+
resNodes[i] = newNode;
|
783
|
+
}
|
539
784
|
}
|
540
785
|
}
|
786
|
+
|
787
|
+
return resNodes;
|
541
788
|
}
|
542
789
|
/**
|
543
790
|
* Returns a simplfied form of node, or the original node if no simplification was possible.
|
544
791
|
*
|
545
792
|
* @param {ConstantNode | SymbolNode | ParenthesisNode | FunctionNode | OperatorNode} node
|
793
|
+
* @param {Object | Function} rule
|
794
|
+
* @param {Object} context -- information about assumed properties of operators
|
546
795
|
* @return {ConstantNode | SymbolNode | ParenthesisNode | FunctionNode | OperatorNode} The simplified form of `expr`, or the original node if no simplification was possible.
|
547
796
|
*/
|
548
797
|
|
549
798
|
|
550
|
-
|
551
|
-
'
|
552
|
-
|
553
|
-
|
554
|
-
|
555
|
-
|
556
|
-
|
557
|
-
|
558
|
-
|
559
|
-
|
560
|
-
} else if (res instanceof ParenthesisNode) {
|
561
|
-
if (res.content) {
|
562
|
-
res.content = applyRule(res.content, rule);
|
563
|
-
}
|
564
|
-
} else if (res instanceof ArrayNode) {
|
565
|
-
mapRule(res.items, rule);
|
566
|
-
} else if (res instanceof AccessorNode) {
|
567
|
-
if (res.object) {
|
568
|
-
res.object = applyRule(res.object, rule);
|
799
|
+
function applyRule(node, rule, context) {
|
800
|
+
// console.log('Entering applyRule("', rule.l.toString({parenthesis:'all'}), '->', rule.r.toString({parenthesis:'all'}), '",', node.toString({parenthesis:'all'}),')')
|
801
|
+
// check that the assumptions for this rule are satisfied by the current
|
802
|
+
// context:
|
803
|
+
if (rule.assuming) {
|
804
|
+
for (var symbol in rule.assuming) {
|
805
|
+
for (var property in rule.assuming[symbol]) {
|
806
|
+
if (hasProperty(symbol, property, context) !== rule.assuming[symbol][property]) {
|
807
|
+
return node;
|
808
|
+
}
|
569
809
|
}
|
810
|
+
}
|
811
|
+
}
|
570
812
|
|
571
|
-
|
572
|
-
|
573
|
-
|
574
|
-
|
575
|
-
|
576
|
-
|
577
|
-
|
578
|
-
|
813
|
+
var mergedContext = mergeContext(rule.imposeContext, context); // Do not clone node unless we find a match
|
814
|
+
|
815
|
+
var res = node; // First replace our child nodes with their simplified versions
|
816
|
+
// If a child could not be simplified, applying the rule to it
|
817
|
+
// will have no effect since the node is returned unchanged
|
818
|
+
|
819
|
+
if (res instanceof OperatorNode || res instanceof FunctionNode) {
|
820
|
+
var newArgs = mapRule(res.args, rule, context);
|
821
|
+
|
822
|
+
if (newArgs !== res.args) {
|
823
|
+
res = res.clone();
|
824
|
+
res.args = newArgs;
|
825
|
+
}
|
826
|
+
} else if (res instanceof ParenthesisNode) {
|
827
|
+
if (res.content) {
|
828
|
+
var newContent = applyRule(res.content, rule, context);
|
829
|
+
|
830
|
+
if (newContent !== res.content) {
|
831
|
+
res = new ParenthesisNode(newContent);
|
579
832
|
}
|
580
|
-
}
|
833
|
+
}
|
834
|
+
} else if (res instanceof ArrayNode) {
|
835
|
+
var newItems = mapRule(res.items, rule, context);
|
581
836
|
|
837
|
+
if (newItems !== res.items) {
|
838
|
+
res = new ArrayNode(newItems);
|
839
|
+
}
|
840
|
+
} else if (res instanceof AccessorNode) {
|
841
|
+
var newObj = res.object;
|
582
842
|
|
583
|
-
|
843
|
+
if (res.object) {
|
844
|
+
newObj = applyRule(res.object, rule, context);
|
845
|
+
}
|
584
846
|
|
585
|
-
var
|
586
|
-
// This allows us to match rules like 'n+n' to the expression '(1+x)+x' or even 'x+1+x' if the operator is commutative.
|
847
|
+
var newIndex = res.index;
|
587
848
|
|
849
|
+
if (res.index) {
|
850
|
+
newIndex = applyRule(res.index, rule, context);
|
851
|
+
}
|
588
852
|
|
589
|
-
if (
|
590
|
-
|
591
|
-
matches = _ruleMatch(rule.expanded.l, res)[0];
|
853
|
+
if (newObj !== res.object || newIndex !== res.index) {
|
854
|
+
res = new AccessorNode(newObj, newIndex);
|
592
855
|
}
|
856
|
+
} else if (res instanceof IndexNode) {
|
857
|
+
var newDims = mapRule(res.dimensions, rule, context);
|
593
858
|
|
594
|
-
if (
|
595
|
-
|
596
|
-
|
597
|
-
|
598
|
-
|
599
|
-
|
859
|
+
if (newDims !== res.dimensions) {
|
860
|
+
res = new IndexNode(newDims);
|
861
|
+
}
|
862
|
+
} else if (res instanceof ObjectNode) {
|
863
|
+
var changed = false;
|
864
|
+
var newProps = {};
|
600
865
|
|
601
|
-
|
602
|
-
|
603
|
-
} // Replace placeholders with their respective nodes without traversing deeper into the replaced nodes
|
866
|
+
for (var prop in res.properties) {
|
867
|
+
newProps[prop] = applyRule(res.properties[prop], rule, context);
|
604
868
|
|
869
|
+
if (newProps[prop] !== res.properties[prop]) {
|
870
|
+
changed = true;
|
871
|
+
}
|
872
|
+
}
|
605
873
|
|
606
|
-
|
607
|
-
|
608
|
-
return matches.placeholders[node.name].clone();
|
609
|
-
} else {
|
610
|
-
return node;
|
611
|
-
}
|
612
|
-
}); // const after = res.toString({parenthesis: 'all'})
|
613
|
-
// console.log('Simplified ' + before + ' to ' + after)
|
874
|
+
if (changed) {
|
875
|
+
res = new ObjectNode(newProps);
|
614
876
|
}
|
877
|
+
} // Try to match a rule against this node
|
615
878
|
|
616
|
-
|
879
|
+
|
880
|
+
var repl = rule.r;
|
881
|
+
|
882
|
+
var matches = _ruleMatch(rule.l, res, mergedContext)[0]; // If the rule is associative operator, we can try matching it while allowing additional terms.
|
883
|
+
// This allows us to match rules like 'n+n' to the expression '(1+x)+x' or even 'x+1+x' if the operator is commutative.
|
884
|
+
|
885
|
+
|
886
|
+
if (!matches && rule.expanded) {
|
887
|
+
repl = rule.expanded.r;
|
888
|
+
matches = _ruleMatch(rule.expanded.l, res, mergedContext)[0];
|
617
889
|
}
|
618
|
-
|
890
|
+
|
891
|
+
if (matches) {
|
892
|
+
// const before = res.toString({parenthesis: 'all'})
|
893
|
+
// Create a new node by cloning the rhs of the matched rule
|
894
|
+
// we keep any implicit multiplication state if relevant
|
895
|
+
var implicit = res.implicit;
|
896
|
+
res = repl.clone();
|
897
|
+
|
898
|
+
if (implicit && 'implicit' in repl) {
|
899
|
+
res.implicit = true;
|
900
|
+
} // Replace placeholders with their respective nodes without traversing deeper into the replaced nodes
|
901
|
+
|
902
|
+
|
903
|
+
res = res.transform(function (node) {
|
904
|
+
if (node.isSymbolNode && (0, _object.hasOwnProperty)(matches.placeholders, node.name)) {
|
905
|
+
return matches.placeholders[node.name].clone();
|
906
|
+
} else {
|
907
|
+
return node;
|
908
|
+
}
|
909
|
+
}); // const after = res.toString({parenthesis: 'all'})
|
910
|
+
// console.log('Simplified ' + before + ' to ' + after)
|
911
|
+
}
|
912
|
+
|
913
|
+
if (rule.repeat && res !== node) {
|
914
|
+
res = applyRule(res, rule, context);
|
915
|
+
}
|
916
|
+
|
917
|
+
return res;
|
918
|
+
}
|
619
919
|
/**
|
620
920
|
* Get (binary) combinations of a flattened binary node
|
621
921
|
* e.g. +(node1, node2, node3) -> [
|
@@ -625,6 +925,7 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
625
925
|
*
|
626
926
|
*/
|
627
927
|
|
928
|
+
|
628
929
|
function getSplits(node, context) {
|
629
930
|
var res = [];
|
630
931
|
var right, rightArgs;
|
@@ -638,9 +939,18 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
638
939
|
res.push(makeNode([node.args[i], right]));
|
639
940
|
}
|
640
941
|
} else {
|
641
|
-
|
642
|
-
|
643
|
-
|
942
|
+
// Keep order, but try all parenthesizations
|
943
|
+
for (var _i2 = 1; _i2 < node.args.length; _i2++) {
|
944
|
+
var left = node.args[0];
|
945
|
+
|
946
|
+
if (_i2 > 1) {
|
947
|
+
left = makeNode(node.args.slice(0, _i2));
|
948
|
+
}
|
949
|
+
|
950
|
+
rightArgs = node.args.slice(_i2);
|
951
|
+
right = rightArgs.length === 1 ? rightArgs[0] : makeNode(rightArgs);
|
952
|
+
res.push(makeNode([left, right]));
|
953
|
+
}
|
644
954
|
}
|
645
955
|
|
646
956
|
return res;
|
@@ -743,15 +1053,19 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
743
1053
|
*
|
744
1054
|
* @param {ConstantNode | SymbolNode | ParenthesisNode | FunctionNode | OperatorNode} rule
|
745
1055
|
* @param {ConstantNode | SymbolNode | ParenthesisNode | FunctionNode | OperatorNode} node
|
1056
|
+
* @param {Object} context -- provides assumed properties of operators
|
1057
|
+
* @param {Boolean} isSplit -- whether we are in process of splitting an
|
1058
|
+
* n-ary operator node into possible binary combinations.
|
1059
|
+
* Defaults to false.
|
746
1060
|
* @return {Object} Information about the match, if it exists.
|
747
1061
|
*/
|
748
1062
|
|
749
1063
|
|
750
|
-
function _ruleMatch(rule, node, isSplit) {
|
1064
|
+
function _ruleMatch(rule, node, context, isSplit) {
|
751
1065
|
// console.log('Entering _ruleMatch(' + JSON.stringify(rule) + ', ' + JSON.stringify(node) + ')')
|
752
1066
|
// console.log('rule = ' + rule)
|
753
1067
|
// console.log('node = ' + node)
|
754
|
-
// console.log('Entering _ruleMatch('
|
1068
|
+
// console.log('Entering _ruleMatch(', rule.toString({parenthesis:'all'}), ', ', node.toString({parenthesis:'all'}), ', ', context, ')')
|
755
1069
|
var res = [{
|
756
1070
|
placeholders: {}
|
757
1071
|
}];
|
@@ -769,32 +1083,65 @@ var createSimplify = /* #__PURE__ */(0, _factory.factory)(name, dependencies, fu
|
|
769
1083
|
} // rule and node match. Search the children of rule and node.
|
770
1084
|
|
771
1085
|
|
772
|
-
if (node.args.length === 1 && rule.args.length === 1 || !isAssociative(node) && node.args.length === rule.args.length || isSplit) {
|
773
|
-
// Expect non-associative operators to match exactly
|
1086
|
+
if (node.args.length === 1 && rule.args.length === 1 || !isAssociative(node, context) && node.args.length === rule.args.length || isSplit) {
|
1087
|
+
// Expect non-associative operators to match exactly,
|
1088
|
+
// except in any order if operator is commutative
|
774
1089
|
var childMatches = [];
|
775
1090
|
|
776
1091
|
for (var i = 0; i < rule.args.length; i++) {
|
777
|
-
var childMatch = _ruleMatch(rule.args[i], node.args[i]);
|
1092
|
+
var childMatch = _ruleMatch(rule.args[i], node.args[i], context);
|
778
1093
|
|
779
1094
|
if (childMatch.length === 0) {
|
780
1095
|
// Child did not match, so stop searching immediately
|
781
|
-
|
1096
|
+
break;
|
782
1097
|
} // The child matched, so add the information returned from the child to our result
|
783
1098
|
|
784
1099
|
|
785
1100
|
childMatches.push(childMatch);
|
786
1101
|
}
|
787
1102
|
|
1103
|
+
if (childMatches.length !== rule.args.length) {
|
1104
|
+
if (!isCommutative(node, context) || // exact match in order needed
|
1105
|
+
rule.args.length === 1) {
|
1106
|
+
// nothing to commute
|
1107
|
+
return [];
|
1108
|
+
}
|
1109
|
+
|
1110
|
+
if (rule.args.length > 2) {
|
1111
|
+
/* Need to generate all permutations and try them.
|
1112
|
+
* It's a bit complicated, and unlikely to come up since there
|
1113
|
+
* are very few ternary or higher operators. So punt for now.
|
1114
|
+
*/
|
1115
|
+
throw new Error('permuting >2 commutative non-associative rule arguments not yet implemented');
|
1116
|
+
}
|
1117
|
+
/* Exactly two arguments, try them reversed */
|
1118
|
+
|
1119
|
+
|
1120
|
+
var leftMatch = _ruleMatch(rule.args[0], node.args[1], context);
|
1121
|
+
|
1122
|
+
if (leftMatch.length === 0) {
|
1123
|
+
return [];
|
1124
|
+
}
|
1125
|
+
|
1126
|
+
var rightMatch = _ruleMatch(rule.args[1], node.args[0], context);
|
1127
|
+
|
1128
|
+
if (rightMatch.length === 0) {
|
1129
|
+
return [];
|
1130
|
+
}
|
1131
|
+
|
1132
|
+
childMatches = [leftMatch, rightMatch];
|
1133
|
+
}
|
1134
|
+
|
788
1135
|
res = mergeChildMatches(childMatches);
|
789
1136
|
} else if (node.args.length >= 2 && rule.args.length === 2) {
|
790
1137
|
// node is flattened, rule is not
|
791
1138
|
// Associative operators/functions can be split in different ways so we check if the rule matches each
|
792
1139
|
// them and return their union.
|
793
|
-
var splits = getSplits(node,
|
1140
|
+
var splits = getSplits(node, context);
|
794
1141
|
var splitMatches = [];
|
795
1142
|
|
796
|
-
for (var
|
797
|
-
var matchSet = _ruleMatch(rule, splits[
|
1143
|
+
for (var _i3 = 0; _i3 < splits.length; _i3++) {
|
1144
|
+
var matchSet = _ruleMatch(rule, splits[_i3], context, true); // recursing at the same tree depth here
|
798
1145
|
|
799
1146
|
|
800
1147
|
splitMatches = splitMatches.concat(matchSet);
|