themeRiverLayout.js 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. var zrUtil = require("zrender/lib/core/util");
  2. var numberUtil = require("../../util/number");
  3. /**
  4. * @file Using layout algorithm transform the raw data to layout information.
  5. * @author Deqing Li(annong035@gmail.com)
  6. */
  7. function _default(ecModel, api) {
  8. ecModel.eachSeriesByType('themeRiver', function (seriesModel) {
  9. var data = seriesModel.getData();
  10. var single = seriesModel.coordinateSystem;
  11. var layoutInfo = {}; // use the axis boundingRect for view
  12. var rect = single.getRect();
  13. layoutInfo.rect = rect;
  14. var boundaryGap = seriesModel.get('boundaryGap');
  15. var axis = single.getAxis();
  16. layoutInfo.boundaryGap = boundaryGap;
  17. if (axis.orient === 'horizontal') {
  18. boundaryGap[0] = numberUtil.parsePercent(boundaryGap[0], rect.height);
  19. boundaryGap[1] = numberUtil.parsePercent(boundaryGap[1], rect.height);
  20. var height = rect.height - boundaryGap[0] - boundaryGap[1];
  21. themeRiverLayout(data, seriesModel, height);
  22. } else {
  23. boundaryGap[0] = numberUtil.parsePercent(boundaryGap[0], rect.width);
  24. boundaryGap[1] = numberUtil.parsePercent(boundaryGap[1], rect.width);
  25. var width = rect.width - boundaryGap[0] - boundaryGap[1];
  26. themeRiverLayout(data, seriesModel, width);
  27. }
  28. data.setLayout('layoutInfo', layoutInfo);
  29. });
  30. }
  31. /**
  32. * The layout information about themeriver
  33. *
  34. * @param {module:echarts/data/List} data data in the series
  35. * @param {module:echarts/model/Series} seriesModel the model object of themeRiver series
  36. * @param {number} height value used to compute every series height
  37. */
  38. function themeRiverLayout(data, seriesModel, height) {
  39. if (!data.count()) {
  40. return;
  41. }
  42. var coordSys = seriesModel.coordinateSystem; // the data in each layer are organized into a series.
  43. var layerSeries = seriesModel.getLayerSeries(); // the points in each layer.
  44. var layerPoints = zrUtil.map(layerSeries, function (singleLayer) {
  45. return zrUtil.map(singleLayer.indices, function (idx) {
  46. var pt = coordSys.dataToPoint(data.get('time', idx));
  47. pt[1] = data.get('value', idx);
  48. return pt;
  49. });
  50. });
  51. var base = computeBaseline(layerPoints);
  52. var baseLine = base.y0;
  53. var ky = height / base.max; // set layout information for each item.
  54. var n = layerSeries.length;
  55. var m = layerSeries[0].indices.length;
  56. var baseY0;
  57. for (var j = 0; j < m; ++j) {
  58. baseY0 = baseLine[j] * ky;
  59. data.setItemLayout(layerSeries[0].indices[j], {
  60. layerIndex: 0,
  61. x: layerPoints[0][j][0],
  62. y0: baseY0,
  63. y: layerPoints[0][j][1] * ky
  64. });
  65. for (var i = 1; i < n; ++i) {
  66. baseY0 += layerPoints[i - 1][j][1] * ky;
  67. data.setItemLayout(layerSeries[i].indices[j], {
  68. layerIndex: i,
  69. x: layerPoints[i][j][0],
  70. y0: baseY0,
  71. y: layerPoints[i][j][1] * ky
  72. });
  73. }
  74. }
  75. }
  76. /**
  77. * Compute the baseLine of the rawdata
  78. * Inspired by Lee Byron's paper Stacked Graphs - Geometry & Aesthetics
  79. *
  80. * @param {Array.<Array>} data the points in each layer
  81. * @return {Object}
  82. */
  83. function computeBaseline(data) {
  84. var layerNum = data.length;
  85. var pointNum = data[0].length;
  86. var sums = [];
  87. var y0 = [];
  88. var max = 0;
  89. var temp;
  90. var base = {};
  91. for (var i = 0; i < pointNum; ++i) {
  92. for (var j = 0, temp = 0; j < layerNum; ++j) {
  93. temp += data[j][i][1];
  94. }
  95. if (temp > max) {
  96. max = temp;
  97. }
  98. sums.push(temp);
  99. }
  100. for (var k = 0; k < pointNum; ++k) {
  101. y0[k] = (max - sums[k]) / 2;
  102. }
  103. max = 0;
  104. for (var l = 0; l < pointNum; ++l) {
  105. var sum = sums[l] + y0[l];
  106. if (sum > max) {
  107. max = sum;
  108. }
  109. }
  110. base.y0 = y0;
  111. base.max = max;
  112. return base;
  113. }
  114. module.exports = _default;