jiangchengfeiyi-xiaochengxu/node_modules/mathjs/lib/cjs/expression/operators.js
2025-01-02 11:13:50 +08:00

388 lines
9.9 KiB
JavaScript

"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.getAssociativity = getAssociativity;
exports.getOperator = getOperator;
exports.getPrecedence = getPrecedence;
exports.isAssociativeWith = isAssociativeWith;
exports.properties = void 0;
var _object = require("../utils/object.js");
var _is = require("../utils/is.js");
// list of identifiers of nodes in order of their precedence
// also contains information about left/right associativity
// and which other operator the operator is associative with
// Example:
// addition is associative with addition and subtraction, because:
// (a+b)+c=a+(b+c)
// (a+b)-c=a+(b-c)
//
// postfix operators are left associative, prefix operators
// are right associative
//
// It's also possible to set the following properties:
// latexParens: if set to false, this node doesn't need to be enclosed
// in parentheses when using LaTeX
// latexLeftParens: if set to false, this !OperatorNode's!
// left argument doesn't need to be enclosed
// in parentheses
// latexRightParens: the same for the right argument
const properties = exports.properties = [{
// assignment
AssignmentNode: {},
FunctionAssignmentNode: {}
}, {
// conditional expression
ConditionalNode: {
latexLeftParens: false,
latexRightParens: false,
latexParens: false
// conditionals don't need parentheses in LaTeX because
// they are 2 dimensional
}
}, {
// logical or
'OperatorNode:or': {
op: 'or',
associativity: 'left',
associativeWith: []
}
}, {
// logical xor
'OperatorNode:xor': {
op: 'xor',
associativity: 'left',
associativeWith: []
}
}, {
// logical and
'OperatorNode:and': {
op: 'and',
associativity: 'left',
associativeWith: []
}
}, {
// bitwise or
'OperatorNode:bitOr': {
op: '|',
associativity: 'left',
associativeWith: []
}
}, {
// bitwise xor
'OperatorNode:bitXor': {
op: '^|',
associativity: 'left',
associativeWith: []
}
}, {
// bitwise and
'OperatorNode:bitAnd': {
op: '&',
associativity: 'left',
associativeWith: []
}
}, {
// relational operators
'OperatorNode:equal': {
op: '==',
associativity: 'left',
associativeWith: []
},
'OperatorNode:unequal': {
op: '!=',
associativity: 'left',
associativeWith: []
},
'OperatorNode:smaller': {
op: '<',
associativity: 'left',
associativeWith: []
},
'OperatorNode:larger': {
op: '>',
associativity: 'left',
associativeWith: []
},
'OperatorNode:smallerEq': {
op: '<=',
associativity: 'left',
associativeWith: []
},
'OperatorNode:largerEq': {
op: '>=',
associativity: 'left',
associativeWith: []
},
RelationalNode: {
associativity: 'left',
associativeWith: []
}
}, {
// bitshift operators
'OperatorNode:leftShift': {
op: '<<',
associativity: 'left',
associativeWith: []
},
'OperatorNode:rightArithShift': {
op: '>>',
associativity: 'left',
associativeWith: []
},
'OperatorNode:rightLogShift': {
op: '>>>',
associativity: 'left',
associativeWith: []
}
}, {
// unit conversion
'OperatorNode:to': {
op: 'to',
associativity: 'left',
associativeWith: []
}
}, {
// range
RangeNode: {}
}, {
// addition, subtraction
'OperatorNode:add': {
op: '+',
associativity: 'left',
associativeWith: ['OperatorNode:add', 'OperatorNode:subtract']
},
'OperatorNode:subtract': {
op: '-',
associativity: 'left',
associativeWith: []
}
}, {
// multiply, divide, modulus
'OperatorNode:multiply': {
op: '*',
associativity: 'left',
associativeWith: ['OperatorNode:multiply', 'OperatorNode:divide', 'Operator:dotMultiply', 'Operator:dotDivide']
},
'OperatorNode:divide': {
op: '/',
associativity: 'left',
associativeWith: [],
latexLeftParens: false,
latexRightParens: false,
latexParens: false
// fractions don't require parentheses because
// they're 2 dimensional, so parens aren't needed
// in LaTeX
},
'OperatorNode:dotMultiply': {
op: '.*',
associativity: 'left',
associativeWith: ['OperatorNode:multiply', 'OperatorNode:divide', 'OperatorNode:dotMultiply', 'OperatorNode:doDivide']
},
'OperatorNode:dotDivide': {
op: './',
associativity: 'left',
associativeWith: []
},
'OperatorNode:mod': {
op: 'mod',
associativity: 'left',
associativeWith: []
}
}, {
// Repeat multiplication for implicit multiplication
'OperatorNode:multiply': {
associativity: 'left',
associativeWith: ['OperatorNode:multiply', 'OperatorNode:divide', 'Operator:dotMultiply', 'Operator:dotDivide']
}
}, {
// unary prefix operators
'OperatorNode:unaryPlus': {
op: '+',
associativity: 'right'
},
'OperatorNode:unaryMinus': {
op: '-',
associativity: 'right'
},
'OperatorNode:bitNot': {
op: '~',
associativity: 'right'
},
'OperatorNode:not': {
op: 'not',
associativity: 'right'
}
}, {
// exponentiation
'OperatorNode:pow': {
op: '^',
associativity: 'right',
associativeWith: [],
latexRightParens: false
// the exponent doesn't need parentheses in
// LaTeX because it's 2 dimensional
// (it's on top)
},
'OperatorNode:dotPow': {
op: '.^',
associativity: 'right',
associativeWith: []
}
}, {
// factorial
'OperatorNode:factorial': {
op: '!',
associativity: 'left'
}
}, {
// matrix transpose
'OperatorNode:ctranspose': {
op: "'",
associativity: 'left'
}
}];
/**
* Returns the first non-parenthesis internal node, but only
* when the 'parenthesis' option is unset or auto.
* @param {Node} _node
* @param {string} parenthesis
* @return {Node}
*/
function unwrapParen(_node, parenthesis) {
if (!parenthesis || parenthesis !== 'auto') return _node;
let node = _node;
while ((0, _is.isParenthesisNode)(node)) node = node.content;
return node;
}
/**
* Get the precedence of a Node.
* Higher number for higher precedence, starting with 0.
* Returns null if the precedence is undefined.
*
* @param {Node} _node
* @param {string} parenthesis
* @param {string} implicit
* @param {Node} parent (for determining context for implicit multiplication)
* @return {number | null}
*/
function getPrecedence(_node, parenthesis, implicit, parent) {
let node = _node;
if (parenthesis !== 'keep') {
// ParenthesisNodes are only ignored when not in 'keep' mode
node = _node.getContent();
}
const identifier = node.getIdentifier();
let precedence = null;
for (let i = 0; i < properties.length; i++) {
if (identifier in properties[i]) {
precedence = i;
break;
}
}
// Bump up precedence of implicit multiplication, except when preceded
// by a "Rule 2" fraction ( [unaryOp]constant / constant )
if (identifier === 'OperatorNode:multiply' && node.implicit && implicit !== 'show') {
const leftArg = unwrapParen(node.args[0], parenthesis);
if (!((0, _is.isConstantNode)(leftArg) && parent && parent.getIdentifier() === 'OperatorNode:divide' && (0, _is.rule2Node)(unwrapParen(parent.args[0], parenthesis))) && !(leftArg.getIdentifier() === 'OperatorNode:divide' && (0, _is.rule2Node)(unwrapParen(leftArg.args[0], parenthesis)) && (0, _is.isConstantNode)(unwrapParen(leftArg.args[1])))) {
precedence += 1;
}
}
return precedence;
}
/**
* Get the associativity of an operator (left or right).
* Returns a string containing 'left' or 'right' or null if
* the associativity is not defined.
*
* @param {Node} _node
* @param {string} parenthesis
* @return {string|null}
* @throws {Error}
*/
function getAssociativity(_node, parenthesis) {
let node = _node;
if (parenthesis !== 'keep') {
// ParenthesisNodes are only ignored when not in 'keep' mode
node = _node.getContent();
}
const identifier = node.getIdentifier();
const index = getPrecedence(node, parenthesis);
if (index === null) {
// node isn't in the list
return null;
}
const property = properties[index][identifier];
if ((0, _object.hasOwnProperty)(property, 'associativity')) {
if (property.associativity === 'left') {
return 'left';
}
if (property.associativity === 'right') {
return 'right';
}
// associativity is invalid
throw Error('\'' + identifier + '\' has the invalid associativity \'' + property.associativity + '\'.');
}
// associativity is undefined
return null;
}
/**
* Check if an operator is associative with another operator.
* Returns either true or false or null if not defined.
*
* @param {Node} nodeA
* @param {Node} nodeB
* @param {string} parenthesis
* @return {boolean | null}
*/
function isAssociativeWith(nodeA, nodeB, parenthesis) {
// ParenthesisNodes are only ignored when not in 'keep' mode
const a = parenthesis !== 'keep' ? nodeA.getContent() : nodeA;
const b = parenthesis !== 'keep' ? nodeA.getContent() : nodeB;
const identifierA = a.getIdentifier();
const identifierB = b.getIdentifier();
const index = getPrecedence(a, parenthesis);
if (index === null) {
// node isn't in the list
return null;
}
const property = properties[index][identifierA];
if ((0, _object.hasOwnProperty)(property, 'associativeWith') && property.associativeWith instanceof Array) {
for (let i = 0; i < property.associativeWith.length; i++) {
if (property.associativeWith[i] === identifierB) {
return true;
}
}
return false;
}
// associativeWith is not defined
return null;
}
/**
* Get the operator associated with a function name.
* Returns a string with the operator symbol, or null if the
* input is not the name of a function associated with an
* operator.
*
* @param {string} Function name
* @return {string | null} Associated operator symbol, if any
*/
function getOperator(fn) {
const identifier = 'OperatorNode:' + fn;
for (const group of properties) {
if (identifier in group) {
return group[identifier].op;
}
}
return null;
}