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.
Files changed (325) hide show
  1. package/HISTORY.md +17 -0
  2. package/docs/expressions/algebra.md +27 -0
  3. package/docs/reference/functions/abs.md +6 -0
  4. package/docs/reference/functions/acos.md +6 -0
  5. package/docs/reference/functions/acosh.md +6 -0
  6. package/docs/reference/functions/acot.md +6 -0
  7. package/docs/reference/functions/acoth.md +6 -0
  8. package/docs/reference/functions/acsc.md +6 -0
  9. package/docs/reference/functions/acsch.md +6 -0
  10. package/docs/reference/functions/add.md +6 -0
  11. package/docs/reference/functions/and.md +6 -0
  12. package/docs/reference/functions/apply.md +6 -0
  13. package/docs/reference/functions/arg.md +6 -0
  14. package/docs/reference/functions/asec.md +6 -0
  15. package/docs/reference/functions/asech.md +6 -0
  16. package/docs/reference/functions/asin.md +6 -0
  17. package/docs/reference/functions/asinh.md +6 -0
  18. package/docs/reference/functions/atan.md +6 -0
  19. package/docs/reference/functions/atan2.md +6 -0
  20. package/docs/reference/functions/atanh.md +6 -0
  21. package/docs/reference/functions/bellNumbers.md +6 -0
  22. package/docs/reference/functions/bin.md +6 -0
  23. package/docs/reference/functions/bitAnd.md +6 -0
  24. package/docs/reference/functions/bitNot.md +6 -0
  25. package/docs/reference/functions/bitOr.md +6 -0
  26. package/docs/reference/functions/bitXor.md +6 -0
  27. package/docs/reference/functions/catalan.md +6 -0
  28. package/docs/reference/functions/cbrt.md +6 -0
  29. package/docs/reference/functions/ceil.md +6 -0
  30. package/docs/reference/functions/clone.md +6 -0
  31. package/docs/reference/functions/column.md +6 -0
  32. package/docs/reference/functions/combinations.md +6 -0
  33. package/docs/reference/functions/combinationsWithRep.md +6 -0
  34. package/docs/reference/functions/compare.md +6 -0
  35. package/docs/reference/functions/compareNatural.md +6 -0
  36. package/docs/reference/functions/compareText.md +6 -0
  37. package/docs/reference/functions/compile.md +6 -0
  38. package/docs/reference/functions/composition.md +6 -0
  39. package/docs/reference/functions/concat.md +6 -0
  40. package/docs/reference/functions/conj.md +6 -0
  41. package/docs/reference/functions/cos.md +6 -0
  42. package/docs/reference/functions/cosh.md +6 -0
  43. package/docs/reference/functions/cot.md +6 -0
  44. package/docs/reference/functions/coth.md +6 -0
  45. package/docs/reference/functions/count.md +6 -0
  46. package/docs/reference/functions/cross.md +6 -0
  47. package/docs/reference/functions/csc.md +6 -0
  48. package/docs/reference/functions/csch.md +6 -0
  49. package/docs/reference/functions/ctranspose.md +6 -0
  50. package/docs/reference/functions/cube.md +6 -0
  51. package/docs/reference/functions/deepEqual.md +6 -0
  52. package/docs/reference/functions/derivative.md +6 -0
  53. package/docs/reference/functions/det.md +6 -0
  54. package/docs/reference/functions/diag.md +6 -0
  55. package/docs/reference/functions/diff.md +6 -0
  56. package/docs/reference/functions/distance.md +6 -0
  57. package/docs/reference/functions/divide.md +6 -0
  58. package/docs/reference/functions/dot.md +6 -0
  59. package/docs/reference/functions/dotDivide.md +6 -0
  60. package/docs/reference/functions/dotMultiply.md +6 -0
  61. package/docs/reference/functions/dotPow.md +6 -0
  62. package/docs/reference/functions/eigs.md +6 -0
  63. package/docs/reference/functions/equal.md +6 -0
  64. package/docs/reference/functions/equalText.md +6 -0
  65. package/docs/reference/functions/erf.md +6 -0
  66. package/docs/reference/functions/evaluate.md +6 -0
  67. package/docs/reference/functions/exp.md +6 -0
  68. package/docs/reference/functions/expm.md +6 -0
  69. package/docs/reference/functions/expm1.md +6 -0
  70. package/docs/reference/functions/factorial.md +6 -0
  71. package/docs/reference/functions/filter.md +6 -0
  72. package/docs/reference/functions/fix.md +6 -0
  73. package/docs/reference/functions/flatten.md +6 -0
  74. package/docs/reference/functions/floor.md +6 -0
  75. package/docs/reference/functions/forEach.md +6 -0
  76. package/docs/reference/functions/format.md +6 -0
  77. package/docs/reference/functions/gamma.md +6 -0
  78. package/docs/reference/functions/gcd.md +6 -0
  79. package/docs/reference/functions/getMatrixDataType.md +6 -0
  80. package/docs/reference/functions/hasNumericValue.md +6 -0
  81. package/docs/reference/functions/help.md +6 -0
  82. package/docs/reference/functions/hex.md +6 -0
  83. package/docs/reference/functions/hypot.md +6 -0
  84. package/docs/reference/functions/identity.md +6 -0
  85. package/docs/reference/functions/im.md +6 -0
  86. package/docs/reference/functions/intersect.md +6 -0
  87. package/docs/reference/functions/inv.md +6 -0
  88. package/docs/reference/functions/invmod.md +6 -0
  89. package/docs/reference/functions/isInteger.md +6 -0
  90. package/docs/reference/functions/isNaN.md +6 -0
  91. package/docs/reference/functions/isNegative.md +6 -0
  92. package/docs/reference/functions/isNumeric.md +6 -0
  93. package/docs/reference/functions/isPositive.md +6 -0
  94. package/docs/reference/functions/isPrime.md +6 -0
  95. package/docs/reference/functions/isZero.md +6 -0
  96. package/docs/reference/functions/kldivergence.md +6 -0
  97. package/docs/reference/functions/kron.md +6 -0
  98. package/docs/reference/functions/larger.md +6 -0
  99. package/docs/reference/functions/largerEq.md +6 -0
  100. package/docs/reference/functions/lcm.md +6 -0
  101. package/docs/reference/functions/leafCount.md +52 -0
  102. package/docs/reference/functions/leftShift.md +6 -0
  103. package/docs/reference/functions/log.md +6 -0
  104. package/docs/reference/functions/log10.md +6 -0
  105. package/docs/reference/functions/log1p.md +6 -0
  106. package/docs/reference/functions/log2.md +6 -0
  107. package/docs/reference/functions/lsolve.md +6 -0
  108. package/docs/reference/functions/lsolveAll.md +6 -0
  109. package/docs/reference/functions/lup.md +6 -0
  110. package/docs/reference/functions/lusolve.md +6 -0
  111. package/docs/reference/functions/mad.md +6 -0
  112. package/docs/reference/functions/map.md +6 -0
  113. package/docs/reference/functions/matrixFromColumns.md +6 -0
  114. package/docs/reference/functions/matrixFromFunction.md +6 -0
  115. package/docs/reference/functions/matrixFromRows.md +6 -0
  116. package/docs/reference/functions/max.md +6 -0
  117. package/docs/reference/functions/mean.md +6 -0
  118. package/docs/reference/functions/median.md +6 -0
  119. package/docs/reference/functions/min.md +6 -0
  120. package/docs/reference/functions/mod.md +6 -0
  121. package/docs/reference/functions/mode.md +6 -0
  122. package/docs/reference/functions/multinomial.md +6 -0
  123. package/docs/reference/functions/multiply.md +6 -0
  124. package/docs/reference/functions/norm.md +6 -0
  125. package/docs/reference/functions/not.md +6 -0
  126. package/docs/reference/functions/nthRoot.md +6 -0
  127. package/docs/reference/functions/nthRoots.md +6 -0
  128. package/docs/reference/functions/numeric.md +6 -0
  129. package/docs/reference/functions/oct.md +6 -0
  130. package/docs/reference/functions/ones.md +6 -0
  131. package/docs/reference/functions/or.md +6 -0
  132. package/docs/reference/functions/parser.md +6 -0
  133. package/docs/reference/functions/partitionSelect.md +6 -0
  134. package/docs/reference/functions/permutations.md +6 -0
  135. package/docs/reference/functions/pickRandom.md +6 -0
  136. package/docs/reference/functions/pow.md +6 -0
  137. package/docs/reference/functions/print.md +6 -0
  138. package/docs/reference/functions/prod.md +6 -0
  139. package/docs/reference/functions/qr.md +6 -0
  140. package/docs/reference/functions/quantileSeq.md +6 -0
  141. package/docs/reference/functions/random.md +6 -0
  142. package/docs/reference/functions/randomInt.md +6 -0
  143. package/docs/reference/functions/range.md +6 -0
  144. package/docs/reference/functions/rationalize.md +7 -1
  145. package/docs/reference/functions/re.md +6 -0
  146. package/docs/reference/functions/reshape.md +7 -0
  147. package/docs/reference/functions/resize.md +6 -0
  148. package/docs/reference/functions/resolve.md +46 -0
  149. package/docs/reference/functions/rightArithShift.md +6 -0
  150. package/docs/reference/functions/rightLogShift.md +6 -0
  151. package/docs/reference/functions/rotate.md +6 -0
  152. package/docs/reference/functions/rotationMatrix.md +6 -0
  153. package/docs/reference/functions/round.md +6 -0
  154. package/docs/reference/functions/row.md +6 -0
  155. package/docs/reference/functions/sec.md +6 -0
  156. package/docs/reference/functions/sech.md +6 -0
  157. package/docs/reference/functions/setCartesian.md +6 -0
  158. package/docs/reference/functions/setDifference.md +6 -0
  159. package/docs/reference/functions/setDistinct.md +6 -0
  160. package/docs/reference/functions/setIntersect.md +6 -0
  161. package/docs/reference/functions/setIsSubset.md +6 -0
  162. package/docs/reference/functions/setMultiplicity.md +6 -0
  163. package/docs/reference/functions/setPowerset.md +6 -0
  164. package/docs/reference/functions/setSize.md +6 -0
  165. package/docs/reference/functions/setSymDifference.md +6 -0
  166. package/docs/reference/functions/setUnion.md +6 -0
  167. package/docs/reference/functions/sign.md +6 -0
  168. package/docs/reference/functions/simplify.md +43 -6
  169. package/docs/reference/functions/simplifyCore.md +50 -0
  170. package/docs/reference/functions/sin.md +6 -0
  171. package/docs/reference/functions/sinh.md +6 -0
  172. package/docs/reference/functions/size.md +6 -0
  173. package/docs/reference/functions/slu.md +6 -0
  174. package/docs/reference/functions/smaller.md +6 -0
  175. package/docs/reference/functions/smallerEq.md +6 -0
  176. package/docs/reference/functions/sort.md +6 -0
  177. package/docs/reference/functions/sqrt.md +6 -0
  178. package/docs/reference/functions/sqrtm.md +6 -0
  179. package/docs/reference/functions/square.md +6 -0
  180. package/docs/reference/functions/squeeze.md +6 -0
  181. package/docs/reference/functions/std.md +6 -0
  182. package/docs/reference/functions/stirlingS2.md +6 -0
  183. package/docs/reference/functions/subset.md +6 -0
  184. package/docs/reference/functions/subtract.md +6 -0
  185. package/docs/reference/functions/sum.md +6 -0
  186. package/docs/reference/functions/tan.md +6 -0
  187. package/docs/reference/functions/tanh.md +6 -0
  188. package/docs/reference/functions/to.md +6 -0
  189. package/docs/reference/functions/trace.md +6 -0
  190. package/docs/reference/functions/transpose.md +6 -0
  191. package/docs/reference/functions/typeOf.md +6 -0
  192. package/docs/reference/functions/unaryMinus.md +6 -0
  193. package/docs/reference/functions/unaryPlus.md +6 -0
  194. package/docs/reference/functions/unequal.md +6 -0
  195. package/docs/reference/functions/usolve.md +6 -0
  196. package/docs/reference/functions/usolveAll.md +6 -0
  197. package/docs/reference/functions/variance.md +6 -0
  198. package/docs/reference/functions/xgcd.md +6 -0
  199. package/docs/reference/functions/xor.md +6 -0
  200. package/docs/reference/functions/zeros.md +6 -0
  201. package/docs/reference/functions.md +3 -0
  202. package/lib/browser/math.js +6 -6
  203. package/lib/browser/math.js.map +1 -1
  204. package/lib/cjs/core/create.js +4 -4
  205. package/lib/cjs/core/function/import.js +3 -3
  206. package/lib/cjs/core/function/typed.js +2 -2
  207. package/lib/cjs/defaultInstance.js +3 -3
  208. package/lib/cjs/entry/allFactoriesAny.js +1 -1
  209. package/lib/cjs/entry/allFactoriesNumber.js +1 -1
  210. package/lib/cjs/entry/configReadonly.js +1 -1
  211. package/lib/cjs/entry/dependenciesAny/dependenciesLeafCount.generated.js +23 -0
  212. package/lib/cjs/entry/dependenciesAny/dependenciesRationalize.generated.js +3 -0
  213. package/lib/cjs/entry/dependenciesAny/dependenciesResolve.generated.js +32 -0
  214. package/lib/cjs/entry/dependenciesAny/dependenciesSimplify.generated.js +6 -0
  215. package/lib/cjs/entry/dependenciesAny/dependenciesSimplifyCore.generated.js +65 -0
  216. package/lib/cjs/entry/dependenciesAny.generated.js +24 -0
  217. package/lib/cjs/entry/dependenciesNumber/dependenciesRationalize.generated.js +3 -0
  218. package/lib/cjs/entry/dependenciesNumber/dependenciesResolve.generated.js +32 -0
  219. package/lib/cjs/entry/dependenciesNumber/dependenciesSimplify.generated.js +6 -0
  220. package/lib/cjs/entry/dependenciesNumber/dependenciesSimplifyCore.generated.js +65 -0
  221. package/lib/cjs/entry/dependenciesNumber.generated.js +16 -0
  222. package/lib/cjs/entry/impureFunctionsAny.generated.js +67 -29
  223. package/lib/cjs/entry/impureFunctionsNumber.generated.js +96 -64
  224. package/lib/cjs/entry/pureFunctionsAny.generated.js +53 -55
  225. package/lib/cjs/entry/pureFunctionsNumber.generated.js +46 -48
  226. package/lib/cjs/expression/Help.js +4 -0
  227. package/lib/cjs/expression/Parser.js +1 -1
  228. package/lib/cjs/expression/embeddedDocs/core/typed.js +1 -1
  229. package/lib/cjs/expression/embeddedDocs/embeddedDocs.js +14 -5
  230. package/lib/cjs/expression/embeddedDocs/function/algebra/leafCount.js +15 -0
  231. package/lib/cjs/expression/embeddedDocs/function/algebra/resolve.js +16 -0
  232. package/lib/cjs/expression/embeddedDocs/function/algebra/simplify.js +1 -1
  233. package/lib/cjs/expression/embeddedDocs/function/algebra/simplifyCore.js +15 -0
  234. package/lib/cjs/expression/node/FunctionNode.js +7 -7
  235. package/lib/cjs/expression/node/IndexNode.js +1 -1
  236. package/lib/cjs/expression/node/Node.js +3 -3
  237. package/lib/cjs/expression/node/ObjectNode.js +1 -1
  238. package/lib/cjs/expression/node/utils/access.js +1 -1
  239. package/lib/cjs/expression/node/utils/assign.js +1 -1
  240. package/lib/cjs/expression/parse.js +13 -13
  241. package/lib/cjs/factoriesAny.js +24 -0
  242. package/lib/cjs/factoriesNumber.js +16 -0
  243. package/lib/cjs/function/algebra/decomposition/qr.js +1 -1
  244. package/lib/cjs/function/algebra/leafCount.js +66 -0
  245. package/lib/cjs/function/algebra/rationalize.js +24 -41
  246. package/lib/cjs/function/algebra/resolve.js +106 -0
  247. package/lib/cjs/function/algebra/simplify/simplifyConstant.js +5 -5
  248. package/lib/cjs/function/algebra/simplify/util.js +171 -33
  249. package/lib/cjs/function/algebra/simplify.js +554 -207
  250. package/lib/cjs/function/algebra/{simplify/simplifyCore.js → simplifyCore.js} +67 -43
  251. package/lib/cjs/function/algebra/solver/lsolveAll.js +2 -2
  252. package/lib/cjs/function/algebra/solver/usolveAll.js +2 -2
  253. package/lib/cjs/function/arithmetic/ceil.js +3 -3
  254. package/lib/cjs/function/arithmetic/floor.js +3 -3
  255. package/lib/cjs/function/arithmetic/invmod.js +1 -1
  256. package/lib/cjs/function/arithmetic/norm.js +1 -1
  257. package/lib/cjs/function/arithmetic/round.js +1 -1
  258. package/lib/cjs/function/matrix/eigs/complexEigs.js +5 -5
  259. package/lib/cjs/function/matrix/matrixFromColumns.js +1 -1
  260. package/lib/cjs/function/matrix/matrixFromRows.js +1 -1
  261. package/lib/cjs/function/probability/util/seededRNG.js +2 -2
  262. package/lib/cjs/function/relational/compareNatural.js +6 -6
  263. package/lib/cjs/header.js +2 -2
  264. package/lib/cjs/plain/bignumber/index.js +1 -1
  265. package/lib/cjs/plain/number/combinations.js +18 -6
  266. package/lib/cjs/type/bignumber/BigNumber.js +2 -2
  267. package/lib/cjs/type/bignumber/function/bignumber.js +1 -1
  268. package/lib/cjs/type/boolean.js +2 -2
  269. package/lib/cjs/type/complex/Complex.js +14 -14
  270. package/lib/cjs/type/complex/function/complex.js +1 -1
  271. package/lib/cjs/type/fraction/Fraction.js +6 -6
  272. package/lib/cjs/type/fraction/function/fraction.js +1 -1
  273. package/lib/cjs/type/matrix/DenseMatrix.js +5 -5
  274. package/lib/cjs/type/matrix/SparseMatrix.js +2 -2
  275. package/lib/cjs/type/number.js +1 -1
  276. package/lib/cjs/type/string.js +2 -2
  277. package/lib/cjs/type/unit/Unit.js +8 -8
  278. package/lib/cjs/utils/customs.js +2 -2
  279. package/lib/cjs/utils/emitter.js +1 -1
  280. package/lib/cjs/utils/function.js +2 -2
  281. package/lib/cjs/utils/is.js +6 -6
  282. package/lib/cjs/utils/latex.js +3 -3
  283. package/lib/cjs/utils/lruQueue.js +1 -1
  284. package/lib/cjs/utils/map.js +3 -3
  285. package/lib/cjs/utils/object.js +2 -2
  286. package/lib/cjs/utils/snapshot.js +7 -7
  287. package/lib/cjs/utils/string.js +2 -2
  288. package/lib/cjs/version.js +1 -1
  289. package/lib/esm/entry/dependenciesAny/dependenciesLeafCount.generated.js +12 -0
  290. package/lib/esm/entry/dependenciesAny/dependenciesRationalize.generated.js +2 -0
  291. package/lib/esm/entry/dependenciesAny/dependenciesResolve.generated.js +18 -0
  292. package/lib/esm/entry/dependenciesAny/dependenciesSimplify.generated.js +4 -0
  293. package/lib/esm/entry/dependenciesAny/dependenciesSimplifyCore.generated.js +40 -0
  294. package/lib/esm/entry/dependenciesAny.generated.js +3 -0
  295. package/lib/esm/entry/dependenciesNumber/dependenciesRationalize.generated.js +2 -0
  296. package/lib/esm/entry/dependenciesNumber/dependenciesResolve.generated.js +18 -0
  297. package/lib/esm/entry/dependenciesNumber/dependenciesSimplify.generated.js +4 -0
  298. package/lib/esm/entry/dependenciesNumber/dependenciesSimplifyCore.generated.js +40 -0
  299. package/lib/esm/entry/dependenciesNumber.generated.js +2 -0
  300. package/lib/esm/entry/impureFunctionsAny.generated.js +57 -22
  301. package/lib/esm/entry/impureFunctionsNumber.generated.js +83 -53
  302. package/lib/esm/entry/pureFunctionsAny.generated.js +42 -42
  303. package/lib/esm/entry/pureFunctionsNumber.generated.js +32 -32
  304. package/lib/esm/expression/Help.js +4 -0
  305. package/lib/esm/expression/embeddedDocs/core/typed.js +1 -1
  306. package/lib/esm/expression/embeddedDocs/embeddedDocs.js +6 -0
  307. package/lib/esm/expression/embeddedDocs/function/algebra/leafCount.js +8 -0
  308. package/lib/esm/expression/embeddedDocs/function/algebra/resolve.js +9 -0
  309. package/lib/esm/expression/embeddedDocs/function/algebra/simplify.js +1 -1
  310. package/lib/esm/expression/embeddedDocs/function/algebra/simplifyCore.js +8 -0
  311. package/lib/esm/factoriesAny.js +3 -0
  312. package/lib/esm/factoriesNumber.js +2 -0
  313. package/lib/esm/function/algebra/leafCount.js +59 -0
  314. package/lib/esm/function/algebra/rationalize.js +24 -40
  315. package/lib/esm/function/algebra/resolve.js +95 -0
  316. package/lib/esm/function/algebra/simplify/simplifyConstant.js +3 -3
  317. package/lib/esm/function/algebra/simplify/util.js +170 -34
  318. package/lib/esm/function/algebra/simplify.js +549 -202
  319. package/lib/esm/function/algebra/{simplify/simplifyCore.js → simplifyCore.js} +60 -44
  320. package/lib/esm/plain/number/combinations.js +18 -6
  321. package/lib/esm/version.js +1 -1
  322. package/package.json +15 -10
  323. package/types/index.d.ts +44 -2
  324. package/lib/cjs/function/algebra/simplify/resolve.js +0 -76
  325. 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
- and any changes to it, along with the rule responsible, to console
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
- exact rational numbers.
154
+ * exact rational numbers.
146
155
  * - `fractionsLimit` (10000): when `exactFractions` is true, constants will
147
- be expressed as fractions only when both numerator and denominator
148
- are smaller than `fractionsLimit`.
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, evaluate, rationalize
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
- unflattenl(res); // using left-heavy binary tree here since custom rule functions may expect it
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.simplifyCore = simplifyCore;
273
- simplify.resolve = resolve;
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
- "true": true,
284
- "false": true,
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
- l: 'n-n1',
326
- r: 'n+-n1'
327
- }, // temporarily replace 'subtract' so we can further flatten the 'add' operator
328
- {
329
- l: '-(c*v)',
330
- r: 'v * (-c)'
331
- }, // make non-constant terms positive
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
- }, simplifyConstant, // expand nested exponentiation
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
- l: '(n ^ n1) ^ n2',
345
- r: 'n ^ (n1 * n2)'
346
- }, // collect like factors
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
- l: 'n * n^n1',
352
- r: 'n^(n1+1)'
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
- l: 'n^n1 * n^n2',
355
- r: 'n^(n1+n2)'
356
- }, // collect like terms
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
- l: 'n+n',
359
- r: '2*n'
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
- }, // remove parenthesis in the case of negating a quantity
375
- // (It might seem this rule should precede collecting like terms,
376
- // but putting it after gives another chance of noticing like terms,
377
- // and any new like terms produced by this will be collected
378
- // on the next pass through all the rules.)
379
- {
380
- l: 'n1 + (n2 + n3)*(-1)',
381
- r: 'n1 + n2*(-1) + n3*(-1)'
382
- }, // make factors positive (and undo 'make non-constant terms positive')
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
- l: '(-n)*n1',
385
- r: '-(n*n1)'
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
- l: 'c+v',
389
- r: 'v+c',
390
- context: {
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
- l: 'v*c',
397
- r: 'c*v',
398
- context: {
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
- l: 'n*(n1^-1)',
411
- r: 'n/n1'
412
- }, // undo replace 'divide'
413
- {
414
- l: 'n*n1^-n2',
415
- r: 'n/n1^n2'
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
- l: 'n1^-1',
418
- r: '1/n1'
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
- l: 'n*(n1/n2)',
421
- r: '(n*n1)/n2'
422
- }, // '*' before '/'
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
- l: 'n-(n1+n2)',
425
- r: 'n-n1-n2'
426
- }, // '-' before '+'
427
- // { l: '(n1/n2)/n3', r: 'n1/(n2*n3)' },
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
- }, // this pattern can be produced by simplifyConstant
433
- {
434
- l: 'n1/(n2/n3)',
435
- r: '(n1*n3)/n2'
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
- function _buildRules(rules) {
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["default"])(rule);
734
+ var ruleType = (0, _typeof2.default)(rule);
465
735
 
466
736
  switch (ruleType) {
467
737
  case 'string':
468
- {
469
- var lr = rule.split('->');
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
- nodes[i] = applyRule(nodes[i], rule);
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
- var applyRule = typed('applyRule', {
551
- 'Node, Object': function NodeObject(node, rule) {
552
- // console.log('Entering applyRule(' + node.toString() + ')')
553
- // Do not clone node unless we find a match
554
- var res = node; // First replace our child nodes with their simplified versions
555
- // If a child could not be simplified, applying the rule to it
556
- // will have no effect since the node is returned unchanged
557
-
558
- if (res instanceof OperatorNode || res instanceof FunctionNode) {
559
- mapRule(res.args, rule);
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
- if (res.index) {
572
- res.index = applyRule(res.index, rule);
573
- }
574
- } else if (res instanceof IndexNode) {
575
- mapRule(res.dimensions, rule);
576
- } else if (res instanceof ObjectNode) {
577
- for (var prop in res.properties) {
578
- res.properties[prop] = applyRule(res.properties[prop], rule);
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
- } // Try to match a rule against this node
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
- var repl = rule.r;
843
+ if (res.object) {
844
+ newObj = applyRule(res.object, rule, context);
845
+ }
584
846
 
585
- var matches = _ruleMatch(rule.l, res)[0]; // If the rule is associative operator, we can try matching it while allowing additional terms.
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 (!matches && rule.expanded) {
590
- repl = rule.expanded.r;
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 (matches) {
595
- // const before = res.toString({parenthesis: 'all'})
596
- // Create a new node by cloning the rhs of the matched rule
597
- // we keep any implicit multiplication state if relevant
598
- var implicit = res.implicit;
599
- res = repl.clone();
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
- if (implicit && 'implicit' in repl) {
602
- res.implicit = true;
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
- res = res.transform(function (node) {
607
- if (node.isSymbolNode && (0, _object.hasOwnProperty)(matches.placeholders, node.name)) {
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
- return res;
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
- rightArgs = node.args.slice(1);
642
- right = rightArgs.length === 1 ? rightArgs[0] : makeNode(rightArgs);
643
- res.push(makeNode([node.args[0], right]));
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(' + rule.toString() + ', ' + node.toString() + ')')
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
- return [];
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, rule.context);
1140
+ var splits = getSplits(node, context);
794
1141
  var splitMatches = [];
795
1142
 
796
- for (var _i = 0; _i < splits.length; _i++) {
797
- var matchSet = _ruleMatch(rule, splits[_i], true); // recursing at the same tree depth here
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);