mirror of
https://github.com/josdejong/mathjs.git
synced 2025-12-08 19:46:04 +00:00
146 lines
3.7 KiB
JavaScript
146 lines
3.7 KiB
JavaScript
var number = require('../../util/number'),
|
|
Node = require('./Node'),
|
|
|
|
BigNumber = require('bignumber.js'),
|
|
Range = require('../../type/Range'),
|
|
Matrix = require('../../type/Matrix'),
|
|
|
|
toNumber = number.toNumber;
|
|
|
|
/**
|
|
* @constructor RangeNode
|
|
* @extends {Node}
|
|
* create a range
|
|
* @param {Object} math The math namespace containing all functions
|
|
* @param {Node[]} params Array [start, end] or [start, end, step]
|
|
*/
|
|
function RangeNode (math, params) {
|
|
this.math = math;
|
|
|
|
if (params.length != 2 && params.length != 3) {
|
|
// TODO: better error message
|
|
throw new SyntaxError('Wrong number of arguments');
|
|
}
|
|
|
|
this.start = params[0]; // included lower-bound
|
|
this.end = params[1]; // included upper-bound
|
|
this.step = params[2]; // optional step
|
|
}
|
|
|
|
RangeNode.prototype = new Node();
|
|
|
|
// TODO: remove redundant functions from RangeNode
|
|
|
|
/**
|
|
* Evaluate the range
|
|
* @return {*} result
|
|
*/
|
|
// TODO: cleanup
|
|
RangeNode.prototype._eval = function() {
|
|
// evaluate the parameters
|
|
var range = this._evalParams(),
|
|
includeEnd = true;
|
|
|
|
// create the range
|
|
return this.math.range(range.start, range.end, range.step, includeEnd);
|
|
};
|
|
|
|
/**
|
|
* Compile the node to javascript code
|
|
* @param {Object} defs Object which can be used to define functions
|
|
* or constants globally available for the compiled
|
|
* expression
|
|
* @return {String} js
|
|
* @private
|
|
*/
|
|
RangeNode.prototype._compile = function (defs) {
|
|
return 'math.range(' +
|
|
this.start._compile(defs) + ', ' +
|
|
this.end._compile(defs) + ', ' +
|
|
(this.step ? (this.step._compile(defs) + ', ') : '') +
|
|
'true)'; // parameter includeEnd = true
|
|
};
|
|
|
|
/**
|
|
* Create a Range from a RangeNode
|
|
* @return {Range} range
|
|
*/
|
|
RangeNode.prototype.toRange = function() {
|
|
// evaluate the parameters
|
|
var range = this._evalParams();
|
|
|
|
// convert big numbers to numbers
|
|
if (range.start instanceof BigNumber) range.start = toNumber(range.start);
|
|
if (range.end instanceof BigNumber) range.end = toNumber(range.end);
|
|
if (range.step instanceof BigNumber) range.step = toNumber(range.step);
|
|
|
|
// upper-bound must be included, so compensate for that
|
|
// NOTE: this only works for integer numbers!
|
|
range.end = this.math.add(range.end, (range.step > 0) ? 1 : -1);
|
|
|
|
// create the range
|
|
return new Range(range.start, range.end, range.step);
|
|
};
|
|
|
|
/**
|
|
* Evaluate the range parameters start, step, end
|
|
* @returns {{start: Number, end: Number, step: Number}} range
|
|
* @private
|
|
*/
|
|
RangeNode.prototype._evalParams = function _evalParams() {
|
|
var start = this.start.eval();
|
|
var end = this.end.eval();
|
|
var step = this.step ? this.step.eval() :
|
|
((start instanceof BigNumber) ? new BigNumber(1) : 1);
|
|
|
|
return {
|
|
start: start,
|
|
end: end,
|
|
step: step
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Find all nodes matching given filter
|
|
* @param {Object} filter See Node.find for a description of the filter settings
|
|
* @returns {Node[]} nodes
|
|
*/
|
|
RangeNode.prototype.find = function (filter) {
|
|
var nodes = [];
|
|
|
|
// check itself
|
|
if (this.match(filter)) {
|
|
nodes.push(this);
|
|
}
|
|
|
|
// search in parameters
|
|
if (this.start) {
|
|
nodes = nodes.concat(this.start.find(filter));
|
|
}
|
|
if (this.step) {
|
|
nodes = nodes.concat(this.step.find(filter));
|
|
}
|
|
if (this.end) {
|
|
nodes = nodes.concat(this.end.find(filter));
|
|
}
|
|
|
|
return nodes;
|
|
};
|
|
|
|
/**
|
|
* Get string representation
|
|
* @return {String} str
|
|
*/
|
|
RangeNode.prototype.toString = function() {
|
|
// format the range like "start:step:end"
|
|
var str = this.start.toString();
|
|
if (this.step) {
|
|
str += ':' + this.step.toString();
|
|
}
|
|
str += ':' + this.end.toString();
|
|
|
|
return str;
|
|
};
|
|
|
|
module.exports = RangeNode;
|