UNPKG

dygraphs

Version:

dygraphs is a fast, flexible open source JavaScript charting library.

154 lines (142 loc) 19.1 kB
"use strict"; /** * @license * Part of dygraphs, see top-level LICENSE.txt file * MIT-licenced: https://opensource.org/licenses/MIT */ /* loader wrapper to allow browser use and ES6 imports */ (function _extras_smoothPlotter_closure() { 'use strict'; var Dygraph; if (window.Dygraph) { Dygraph = window.Dygraph; } else if (typeof module !== 'undefined') { Dygraph = require('../dygraph'); if (typeof Dygraph.NAME === 'undefined' && typeof Dygraph["default"] !== 'undefined') Dygraph = Dygraph["default"]; } /* end of loader wrapper header */ /** * Given three sequential points, p0, p1 and p2, find the left and right * control points for p1. * * The three points are expected to have x and y properties. * * The alpha parameter controls the amount of smoothing. * If α=0, then both control points will be the same as p1 (i.e. no smoothing). * * Returns [l1x, l1y, r1x, r1y] * * It's guaranteed that the line from (l1x, l1y)-(r1x, r1y) passes through p1. * Unless allowFalseExtrema is set, then it's also guaranteed that: * l1y ∈ [p0.y, p1.y] * r1y ∈ [p1.y, p2.y] * * The basic algorithm is: * 1. Put the control points l1 and r1 α of the way down (p0, p1) and (p1, p2). * 2. Shift l1 and r2 so that the line l1–r1 passes through p1 * 3. Adjust to prevent false extrema while keeping p1 on the l1–r1 line. * * This is loosely based on the HighCharts algorithm. */ function getControlPoints(p0, p1, p2, opt_alpha, opt_allowFalseExtrema) { var alpha = opt_alpha !== undefined ? opt_alpha : 1 / 3; // 0=no smoothing, 1=crazy smoothing var allowFalseExtrema = opt_allowFalseExtrema || false; if (!p2) { return [p1.x, p1.y, null, null]; } // Step 1: Position the control points along each line segment. var l1x = (1 - alpha) * p1.x + alpha * p0.x, l1y = (1 - alpha) * p1.y + alpha * p0.y, r1x = (1 - alpha) * p1.x + alpha * p2.x, r1y = (1 - alpha) * p1.y + alpha * p2.y; // Step 2: shift the points up so that p1 is on the l1–r1 line. if (l1x != r1x) { // This can be derived w/ some basic algebra. var deltaY = p1.y - r1y - (p1.x - r1x) * (l1y - r1y) / (l1x - r1x); l1y += deltaY; r1y += deltaY; } // Step 3: correct to avoid false extrema. if (!allowFalseExtrema) { if (l1y > p0.y && l1y > p1.y) { l1y = Math.max(p0.y, p1.y); r1y = 2 * p1.y - l1y; } else if (l1y < p0.y && l1y < p1.y) { l1y = Math.min(p0.y, p1.y); r1y = 2 * p1.y - l1y; } if (r1y > p1.y && r1y > p2.y) { r1y = Math.max(p1.y, p2.y); l1y = 2 * p1.y - r1y; } else if (r1y < p1.y && r1y < p2.y) { r1y = Math.min(p1.y, p2.y); l1y = 2 * p1.y - r1y; } } return [l1x, l1y, r1x, r1y]; } // i.e. is none of (null, undefined, NaN) function isOK(x) { return !!x && !isNaN(x); } ; // A plotter which uses splines to create a smooth curve. // See tests/plotters.html for a demo. // Can be controlled via smoothPlotter.smoothing function smoothPlotter(e) { var ctx = e.drawingContext, points = e.points; ctx.beginPath(); ctx.moveTo(points[0].canvasx, points[0].canvasy); // right control point for previous point var lastRightX = points[0].canvasx, lastRightY = points[0].canvasy; for (var i = 1; i < points.length; i++) { var p0 = points[i - 1], p1 = points[i], p2 = points[i + 1]; p0 = p0 && isOK(p0.canvasy) ? p0 : null; p1 = p1 && isOK(p1.canvasy) ? p1 : null; p2 = p2 && isOK(p2.canvasy) ? p2 : null; if (p0 && p1) { var controls = getControlPoints({ x: p0.canvasx, y: p0.canvasy }, { x: p1.canvasx, y: p1.canvasy }, p2 && { x: p2.canvasx, y: p2.canvasy }, smoothPlotter.smoothing); // Uncomment to show the control points: // ctx.lineTo(lastRightX, lastRightY); // ctx.lineTo(controls[0], controls[1]); // ctx.lineTo(p1.canvasx, p1.canvasy); lastRightX = lastRightX !== null ? lastRightX : p0.canvasx; lastRightY = lastRightY !== null ? lastRightY : p0.canvasy; ctx.bezierCurveTo(lastRightX, lastRightY, controls[0], controls[1], p1.canvasx, p1.canvasy); lastRightX = controls[2]; lastRightY = controls[3]; } else if (p1) { // We're starting again after a missing point. ctx.moveTo(p1.canvasx, p1.canvasy); lastRightX = p1.canvasx; lastRightY = p1.canvasy; } else { lastRightX = lastRightY = null; } } ctx.stroke(); } smoothPlotter.smoothing = 1 / 3; smoothPlotter._getControlPoints = getControlPoints; // for testing // older versions exported a global. // This will be removed in the future. // The preferred way to access smoothPlotter is via Dygraph.smoothPlotter. window.smoothPlotter = smoothPlotter; Dygraph.smoothPlotter = smoothPlotter; /* closure and loader wrapper */ Dygraph._require.add('dygraphs/src/extras/smooth-plotter.js', /* exports */{}); })(); //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_extras_smoothPlotter_closure","Dygraph","window","module","require","NAME","getControlPoints","p0","p1","p2","opt_alpha","opt_allowFalseExtrema","alpha","undefined","allowFalseExtrema","x","y","l1x","l1y","r1x","r1y","deltaY","Math","max","min","isOK","isNaN","smoothPlotter","e","ctx","drawingContext","points","beginPath","moveTo","canvasx","canvasy","lastRightX","lastRightY","i","length","controls","smoothing","bezierCurveTo","stroke","_getControlPoints","_require","add"],"sources":["../../src/extras/smooth-plotter.js"],"sourcesContent":["/**\n * @license\n * Part of dygraphs, see top-level LICENSE.txt file\n * MIT-licenced: https://opensource.org/licenses/MIT\n */\n\n/* loader wrapper to allow browser use and ES6 imports */\n(function _extras_smoothPlotter_closure() {\n'use strict';\nvar Dygraph;\nif (window.Dygraph) {\n  Dygraph = window.Dygraph;\n} else if (typeof(module) !== 'undefined') {\n  Dygraph = require('../dygraph');\n  if (typeof(Dygraph.NAME) === 'undefined' && typeof(Dygraph.default) !== 'undefined')\n    Dygraph = Dygraph.default;\n}\n/* end of loader wrapper header */\n\n/**\n * Given three sequential points, p0, p1 and p2, find the left and right\n * control points for p1.\n *\n * The three points are expected to have x and y properties.\n *\n * The alpha parameter controls the amount of smoothing.\n * If α=0, then both control points will be the same as p1 (i.e. no smoothing).\n *\n * Returns [l1x, l1y, r1x, r1y]\n *\n * It's guaranteed that the line from (l1x, l1y)-(r1x, r1y) passes through p1.\n * Unless allowFalseExtrema is set, then it's also guaranteed that:\n *   l1y ∈ [p0.y, p1.y]\n *   r1y ∈ [p1.y, p2.y]\n *\n * The basic algorithm is:\n * 1. Put the control points l1 and r1 α of the way down (p0, p1) and (p1, p2).\n * 2. Shift l1 and r2 so that the line l1–r1 passes through p1\n * 3. Adjust to prevent false extrema while keeping p1 on the l1–r1 line.\n *\n * This is loosely based on the HighCharts algorithm.\n */\nfunction getControlPoints(p0, p1, p2, opt_alpha, opt_allowFalseExtrema) {\n  var alpha = (opt_alpha !== undefined) ? opt_alpha : 1/3;  // 0=no smoothing, 1=crazy smoothing\n  var allowFalseExtrema = opt_allowFalseExtrema || false;\n\n  if (!p2) {\n    return [p1.x, p1.y, null, null];\n  }\n\n  // Step 1: Position the control points along each line segment.\n  var l1x = (1 - alpha) * p1.x + alpha * p0.x,\n      l1y = (1 - alpha) * p1.y + alpha * p0.y,\n      r1x = (1 - alpha) * p1.x + alpha * p2.x,\n      r1y = (1 - alpha) * p1.y + alpha * p2.y;\n\n  // Step 2: shift the points up so that p1 is on the l1–r1 line.\n  if (l1x != r1x) {\n    // This can be derived w/ some basic algebra.\n    var deltaY = p1.y - r1y - (p1.x - r1x) * (l1y - r1y) / (l1x - r1x);\n    l1y += deltaY;\n    r1y += deltaY;\n  }\n\n  // Step 3: correct to avoid false extrema.\n  if (!allowFalseExtrema) {\n    if (l1y > p0.y && l1y > p1.y) {\n      l1y = Math.max(p0.y, p1.y);\n      r1y = 2 * p1.y - l1y;\n    } else if (l1y < p0.y && l1y < p1.y) {\n      l1y = Math.min(p0.y, p1.y);\n      r1y = 2 * p1.y - l1y;\n    }\n\n    if (r1y > p1.y && r1y > p2.y) {\n      r1y = Math.max(p1.y, p2.y);\n      l1y = 2 * p1.y - r1y;\n    } else if (r1y < p1.y && r1y < p2.y) {\n      r1y = Math.min(p1.y, p2.y);\n      l1y = 2 * p1.y - r1y;\n    }\n  }\n\n  return [l1x, l1y, r1x, r1y];\n}\n\n// i.e. is none of (null, undefined, NaN)\nfunction isOK(x) {\n  return !!x && !isNaN(x);\n};\n\n// A plotter which uses splines to create a smooth curve.\n// See tests/plotters.html for a demo.\n// Can be controlled via smoothPlotter.smoothing\nfunction smoothPlotter(e) {\n  var ctx = e.drawingContext,\n      points = e.points;\n\n  ctx.beginPath();\n  ctx.moveTo(points[0].canvasx, points[0].canvasy);\n\n  // right control point for previous point\n  var lastRightX = points[0].canvasx, lastRightY = points[0].canvasy;\n\n  for (var i = 1; i < points.length; i++) {\n    var p0 = points[i - 1],\n        p1 = points[i],\n        p2 = points[i + 1];\n    p0 = p0 && isOK(p0.canvasy) ? p0 : null;\n    p1 = p1 && isOK(p1.canvasy) ? p1 : null;\n    p2 = p2 && isOK(p2.canvasy) ? p2 : null;\n    if (p0 && p1) {\n      var controls = getControlPoints({x: p0.canvasx, y: p0.canvasy},\n                                      {x: p1.canvasx, y: p1.canvasy},\n                                      p2 && {x: p2.canvasx, y: p2.canvasy},\n                                      smoothPlotter.smoothing);\n      // Uncomment to show the control points:\n      // ctx.lineTo(lastRightX, lastRightY);\n      // ctx.lineTo(controls[0], controls[1]);\n      // ctx.lineTo(p1.canvasx, p1.canvasy);\n      lastRightX = (lastRightX !== null) ? lastRightX : p0.canvasx;\n      lastRightY = (lastRightY !== null) ? lastRightY : p0.canvasy;\n      ctx.bezierCurveTo(lastRightX, lastRightY,\n                        controls[0], controls[1],\n                        p1.canvasx, p1.canvasy);\n      lastRightX = controls[2];\n      lastRightY = controls[3];\n    } else if (p1) {\n      // We're starting again after a missing point.\n      ctx.moveTo(p1.canvasx, p1.canvasy);\n      lastRightX = p1.canvasx;\n      lastRightY = p1.canvasy;\n    } else {\n      lastRightX = lastRightY = null;\n    }\n  }\n\n  ctx.stroke();\n}\nsmoothPlotter.smoothing = 1/3;\nsmoothPlotter._getControlPoints = getControlPoints;  // for testing\n\n// older versions exported a global.\n// This will be removed in the future.\n// The preferred way to access smoothPlotter is via Dygraph.smoothPlotter.\nwindow.smoothPlotter = smoothPlotter;\nDygraph.smoothPlotter = smoothPlotter;\n\n/* closure and loader wrapper */\nDygraph._require.add('dygraphs/src/extras/smooth-plotter.js', /* exports */ {});\n})();\n"],"mappings":"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"}