/src/data/Numbers.js

http://github.com/mbostock/protovis · JavaScript · 313 lines · 127 code · 18 blank · 168 comment · 41 complexity · eb7510868c02c3915493bcb3b7b21642 MD5 · raw file

  1. /**
  2. * Returns an array of numbers, starting at <tt>start</tt>, incrementing by
  3. * <tt>step</tt>, until <tt>stop</tt> is reached. The stop value is
  4. * exclusive. If only a single argument is specified, this value is interpeted
  5. * as the <i>stop</i> value, with the <i>start</i> value as zero. If only two
  6. * arguments are specified, the step value is implied to be one.
  7. *
  8. * <p>The method is modeled after the built-in <tt>range</tt> method from
  9. * Python. See the Python documentation for more details.
  10. *
  11. * @see <a href="http://docs.python.org/library/functions.html#range">Python range</a>
  12. * @param {number} [start] the start value.
  13. * @param {number} stop the stop value.
  14. * @param {number} [step] the step value.
  15. * @returns {number[]} an array of numbers.
  16. */
  17. pv.range = function(start, stop, step) {
  18. if (arguments.length == 1) {
  19. stop = start;
  20. start = 0;
  21. }
  22. if (step == undefined) step = 1;
  23. if ((stop - start) / step == Infinity) throw new Error("range must be finite");
  24. var array = [], i = 0, j;
  25. stop -= (stop - start) * 1e-10; // floating point precision!
  26. if (step < 0) {
  27. while ((j = start + step * i++) > stop) {
  28. array.push(j);
  29. }
  30. } else {
  31. while ((j = start + step * i++) < stop) {
  32. array.push(j);
  33. }
  34. }
  35. return array;
  36. };
  37. /**
  38. * Returns a random number in the range [<tt>start</tt>, <tt>stop</tt>) that is
  39. * a multiple of <tt>step</tt>. More specifically, the returned number is of the
  40. * form <tt>start</tt> + <i>n</i> * <tt>step</tt>, where <i>n</i> is a
  41. * nonnegative integer. If <tt>step</tt> is not specified, it defaults to 1,
  42. * returning a random integer if <tt>start</tt> is also an integer.
  43. *
  44. * @param {number} [start] the start value value.
  45. * @param {number} stop the stop value.
  46. * @param {number} [step] the step value.
  47. * @returns {number} a random number between <i>start</i> and <i>stop</i>.
  48. */
  49. pv.random = function(start, stop, step) {
  50. if (arguments.length == 1) {
  51. stop = start;
  52. start = 0;
  53. }
  54. if (step == undefined) step = 1;
  55. return step
  56. ? (Math.floor(Math.random() * (stop - start) / step) * step + start)
  57. : (Math.random() * (stop - start) + start);
  58. };
  59. /**
  60. * Returns the sum of the specified array. If the specified array is not an
  61. * array of numbers, an optional accessor function <tt>f</tt> can be specified
  62. * to map the elements to numbers. See {@link #normalize} for an example.
  63. * Accessor functions can refer to <tt>this.index</tt>.
  64. *
  65. * @param {array} array an array of objects, or numbers.
  66. * @param {function} [f] an optional accessor function.
  67. * @returns {number} the sum of the specified array.
  68. */
  69. pv.sum = function(array, f) {
  70. var o = {};
  71. return array.reduce(f
  72. ? function(p, d, i) { o.index = i; return p + f.call(o, d); }
  73. : function(p, d) { return p + d; }, 0);
  74. };
  75. /**
  76. * Returns the maximum value of the specified array. If the specified array is
  77. * not an array of numbers, an optional accessor function <tt>f</tt> can be
  78. * specified to map the elements to numbers. See {@link #normalize} for an
  79. * example. Accessor functions can refer to <tt>this.index</tt>.
  80. *
  81. * @param {array} array an array of objects, or numbers.
  82. * @param {function} [f] an optional accessor function.
  83. * @returns {number} the maximum value of the specified array.
  84. */
  85. pv.max = function(array, f) {
  86. if (f == pv.index) return array.length - 1;
  87. return Math.max.apply(null, f ? pv.map(array, f) : array);
  88. };
  89. /**
  90. * Returns the index of the maximum value of the specified array. If the
  91. * specified array is not an array of numbers, an optional accessor function
  92. * <tt>f</tt> can be specified to map the elements to numbers. See
  93. * {@link #normalize} for an example. Accessor functions can refer to
  94. * <tt>this.index</tt>.
  95. *
  96. * @param {array} array an array of objects, or numbers.
  97. * @param {function} [f] an optional accessor function.
  98. * @returns {number} the index of the maximum value of the specified array.
  99. */
  100. pv.max.index = function(array, f) {
  101. if (!array.length) return -1;
  102. if (f == pv.index) return array.length - 1;
  103. if (!f) f = pv.identity;
  104. var maxi = 0, maxx = -Infinity, o = {};
  105. for (var i = 0; i < array.length; i++) {
  106. o.index = i;
  107. var x = f.call(o, array[i]);
  108. if (x > maxx) {
  109. maxx = x;
  110. maxi = i;
  111. }
  112. }
  113. return maxi;
  114. }
  115. /**
  116. * Returns the minimum value of the specified array of numbers. If the specified
  117. * array is not an array of numbers, an optional accessor function <tt>f</tt>
  118. * can be specified to map the elements to numbers. See {@link #normalize} for
  119. * an example. Accessor functions can refer to <tt>this.index</tt>.
  120. *
  121. * @param {array} array an array of objects, or numbers.
  122. * @param {function} [f] an optional accessor function.
  123. * @returns {number} the minimum value of the specified array.
  124. */
  125. pv.min = function(array, f) {
  126. if (f == pv.index) return 0;
  127. return Math.min.apply(null, f ? pv.map(array, f) : array);
  128. };
  129. /**
  130. * Returns the index of the minimum value of the specified array. If the
  131. * specified array is not an array of numbers, an optional accessor function
  132. * <tt>f</tt> can be specified to map the elements to numbers. See
  133. * {@link #normalize} for an example. Accessor functions can refer to
  134. * <tt>this.index</tt>.
  135. *
  136. * @param {array} array an array of objects, or numbers.
  137. * @param {function} [f] an optional accessor function.
  138. * @returns {number} the index of the minimum value of the specified array.
  139. */
  140. pv.min.index = function(array, f) {
  141. if (!array.length) return -1;
  142. if (f == pv.index) return 0;
  143. if (!f) f = pv.identity;
  144. var mini = 0, minx = Infinity, o = {};
  145. for (var i = 0; i < array.length; i++) {
  146. o.index = i;
  147. var x = f.call(o, array[i]);
  148. if (x < minx) {
  149. minx = x;
  150. mini = i;
  151. }
  152. }
  153. return mini;
  154. }
  155. /**
  156. * Returns the arithmetic mean, or average, of the specified array. If the
  157. * specified array is not an array of numbers, an optional accessor function
  158. * <tt>f</tt> can be specified to map the elements to numbers. See
  159. * {@link #normalize} for an example. Accessor functions can refer to
  160. * <tt>this.index</tt>.
  161. *
  162. * @param {array} array an array of objects, or numbers.
  163. * @param {function} [f] an optional accessor function.
  164. * @returns {number} the mean of the specified array.
  165. */
  166. pv.mean = function(array, f) {
  167. return pv.sum(array, f) / array.length;
  168. };
  169. /**
  170. * Returns the median of the specified array. If the specified array is not an
  171. * array of numbers, an optional accessor function <tt>f</tt> can be specified
  172. * to map the elements to numbers. See {@link #normalize} for an example.
  173. * Accessor functions can refer to <tt>this.index</tt>.
  174. *
  175. * @param {array} array an array of objects, or numbers.
  176. * @param {function} [f] an optional accessor function.
  177. * @returns {number} the median of the specified array.
  178. */
  179. pv.median = function(array, f) {
  180. if (f == pv.index) return (array.length - 1) / 2;
  181. array = pv.map(array, f).sort(pv.naturalOrder);
  182. if (array.length % 2) return array[Math.floor(array.length / 2)];
  183. var i = array.length / 2;
  184. return (array[i - 1] + array[i]) / 2;
  185. };
  186. /**
  187. * Returns the unweighted variance of the specified array. If the specified
  188. * array is not an array of numbers, an optional accessor function <tt>f</tt>
  189. * can be specified to map the elements to numbers. See {@link #normalize} for
  190. * an example. Accessor functions can refer to <tt>this.index</tt>.
  191. *
  192. * @param {array} array an array of objects, or numbers.
  193. * @param {function} [f] an optional accessor function.
  194. * @returns {number} the variance of the specified array.
  195. */
  196. pv.variance = function(array, f) {
  197. if (array.length < 1) return NaN;
  198. if (array.length == 1) return 0;
  199. var mean = pv.mean(array, f), sum = 0, o = {};
  200. if (!f) f = pv.identity;
  201. for (var i = 0; i < array.length; i++) {
  202. o.index = i;
  203. var d = f.call(o, array[i]) - mean;
  204. sum += d * d;
  205. }
  206. return sum;
  207. };
  208. /**
  209. * Returns an unbiased estimation of the standard deviation of a population,
  210. * given the specified random sample. If the specified array is not an array of
  211. * numbers, an optional accessor function <tt>f</tt> can be specified to map the
  212. * elements to numbers. See {@link #normalize} for an example. Accessor
  213. * functions can refer to <tt>this.index</tt>.
  214. *
  215. * @param {array} array an array of objects, or numbers.
  216. * @param {function} [f] an optional accessor function.
  217. * @returns {number} the standard deviation of the specified array.
  218. */
  219. pv.deviation = function(array, f) {
  220. return Math.sqrt(pv.variance(array, f) / (array.length - 1));
  221. };
  222. /**
  223. * Returns the logarithm with a given base value.
  224. *
  225. * @param {number} x the number for which to compute the logarithm.
  226. * @param {number} b the base of the logarithm.
  227. * @returns {number} the logarithm value.
  228. */
  229. pv.log = function(x, b) {
  230. return Math.log(x) / Math.log(b);
  231. };
  232. /**
  233. * Computes a zero-symmetric logarithm. Computes the logarithm of the absolute
  234. * value of the input, and determines the sign of the output according to the
  235. * sign of the input value.
  236. *
  237. * @param {number} x the number for which to compute the logarithm.
  238. * @param {number} b the base of the logarithm.
  239. * @returns {number} the symmetric log value.
  240. */
  241. pv.logSymmetric = function(x, b) {
  242. return (x == 0) ? 0 : ((x < 0) ? -pv.log(-x, b) : pv.log(x, b));
  243. };
  244. /**
  245. * Computes a zero-symmetric logarithm, with adjustment to values between zero
  246. * and the logarithm base. This adjustment introduces distortion for values less
  247. * than the base number, but enables simultaneous plotting of log-transformed
  248. * data involving both positive and negative numbers.
  249. *
  250. * @param {number} x the number for which to compute the logarithm.
  251. * @param {number} b the base of the logarithm.
  252. * @returns {number} the adjusted, symmetric log value.
  253. */
  254. pv.logAdjusted = function(x, b) {
  255. if (!isFinite(x)) return x;
  256. var negative = x < 0;
  257. if (x < b) x += (b - x) / b;
  258. return negative ? -pv.log(x, b) : pv.log(x, b);
  259. };
  260. /**
  261. * Rounds an input value down according to its logarithm. The method takes the
  262. * floor of the logarithm of the value and then uses the resulting value as an
  263. * exponent for the base value.
  264. *
  265. * @param {number} x the number for which to compute the logarithm floor.
  266. * @param {number} b the base of the logarithm.
  267. * @returns {number} the rounded-by-logarithm value.
  268. */
  269. pv.logFloor = function(x, b) {
  270. return (x > 0)
  271. ? Math.pow(b, Math.floor(pv.log(x, b)))
  272. : -Math.pow(b, -Math.floor(-pv.log(-x, b)));
  273. };
  274. /**
  275. * Rounds an input value up according to its logarithm. The method takes the
  276. * ceiling of the logarithm of the value and then uses the resulting value as an
  277. * exponent for the base value.
  278. *
  279. * @param {number} x the number for which to compute the logarithm ceiling.
  280. * @param {number} b the base of the logarithm.
  281. * @returns {number} the rounded-by-logarithm value.
  282. */
  283. pv.logCeil = function(x, b) {
  284. return (x > 0)
  285. ? Math.pow(b, Math.ceil(pv.log(x, b)))
  286. : -Math.pow(b, -Math.ceil(-pv.log(-x, b)));
  287. };
  288. (function() {
  289. var radians = Math.PI / 180,
  290. degrees = 180 / Math.PI;
  291. /** Returns the number of radians corresponding to the specified degrees. */
  292. pv.radians = function(degrees) { return radians * degrees; };
  293. /** Returns the number of degrees corresponding to the specified radians. */
  294. pv.degrees = function(radians) { return degrees * radians; };
  295. })();