legend.ts 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. import angular from 'angular';
  2. import _ from 'lodash';
  3. import $ from 'jquery';
  4. import PerfectScrollbar from 'perfect-scrollbar';
  5. import {updateLegendValues} from 'app/core/core';
  6. var module = angular.module('grafana.directives');
  7. module.directive('graphLegend', function(popoverSrv, $timeout) {
  8. return {
  9. link: function(scope, elem) {
  10. var firstRender = true;
  11. var ctrl = scope.ctrl;
  12. var panel = ctrl.panel;
  13. var data;
  14. var seriesList;
  15. var i;
  16. var legendScrollbar;
  17. scope.$on("$destroy", function() {
  18. if (legendScrollbar) {
  19. legendScrollbar.destroy();
  20. }
  21. });
  22. ctrl.events.on('render-legend', () => {
  23. data = ctrl.seriesList;
  24. if (data) {
  25. render();
  26. }
  27. ctrl.events.emit('legend-rendering-complete');
  28. });
  29. function updateLegendDecimals() {
  30. updateLegendValues(data, panel);
  31. }
  32. function getSeriesIndexForElement(el) {
  33. return el.parents('[data-series-index]').data('series-index');
  34. }
  35. function openColorSelector(e) {
  36. // if we clicked inside poup container ignore click
  37. if ($(e.target).parents('.popover').length) {
  38. return;
  39. }
  40. var el = $(e.currentTarget).find('.fa-minus');
  41. var index = getSeriesIndexForElement(el);
  42. var series = seriesList[index];
  43. $timeout(function() {
  44. popoverSrv.show({
  45. element: el[0],
  46. position: 'bottom left',
  47. targetAttachment: 'top left',
  48. template: '<series-color-picker series="series" onToggleAxis="toggleAxis" onColorChange="colorSelected">' +
  49. '</series-color-picker>',
  50. openOn: 'hover',
  51. model: {
  52. series: series,
  53. toggleAxis: function() {
  54. ctrl.toggleAxis(series);
  55. },
  56. colorSelected: function(color) {
  57. ctrl.changeSeriesColor(series, color);
  58. }
  59. },
  60. });
  61. });
  62. }
  63. function toggleSeries(e) {
  64. var el = $(e.currentTarget);
  65. var index = getSeriesIndexForElement(el);
  66. var seriesInfo = seriesList[index];
  67. var scrollPosition = $(elem.children('tbody')).scrollTop();
  68. ctrl.toggleSeries(seriesInfo, e);
  69. $(elem.children('tbody')).scrollTop(scrollPosition);
  70. }
  71. function sortLegend(e) {
  72. var el = $(e.currentTarget);
  73. var stat = el.data('stat');
  74. if (stat !== panel.legend.sort) { panel.legend.sortDesc = null; }
  75. // if already sort ascending, disable sorting
  76. if (panel.legend.sortDesc === false) {
  77. panel.legend.sort = null;
  78. panel.legend.sortDesc = null;
  79. ctrl.render();
  80. return;
  81. }
  82. panel.legend.sortDesc = !panel.legend.sortDesc;
  83. panel.legend.sort = stat;
  84. ctrl.render();
  85. }
  86. function getTableHeaderHtml(statName) {
  87. if (!panel.legend[statName]) { return ""; }
  88. var html = '<th class="pointer" data-stat="' + statName + '">' + statName;
  89. if (panel.legend.sort === statName) {
  90. var cssClass = panel.legend.sortDesc ? 'fa fa-caret-down' : 'fa fa-caret-up' ;
  91. html += ' <span class="' + cssClass + '"></span>';
  92. }
  93. return html + '</th>';
  94. }
  95. function render() {
  96. if (!ctrl.panel.legend.show) {
  97. elem.empty();
  98. firstRender = true;
  99. return;
  100. }
  101. if (firstRender) {
  102. elem.on('click', '.graph-legend-icon', openColorSelector);
  103. elem.on('click', '.graph-legend-alias', toggleSeries);
  104. elem.on('click', 'th', sortLegend);
  105. firstRender = false;
  106. }
  107. seriesList = data;
  108. elem.empty();
  109. // Set min-width if side style and there is a value, otherwise remove the CSS propery
  110. var width = panel.legend.rightSide && panel.legend.sideWidth ? panel.legend.sideWidth + "px" : "";
  111. elem.css("min-width", width);
  112. elem.toggleClass('graph-legend-table', panel.legend.alignAsTable === true);
  113. var tableHeaderElem;
  114. if (panel.legend.alignAsTable) {
  115. var header = '<tr>';
  116. header += '<th colspan="2" style="text-align:left"></th>';
  117. if (panel.legend.values) {
  118. header += getTableHeaderHtml('min');
  119. header += getTableHeaderHtml('max');
  120. header += getTableHeaderHtml('avg');
  121. header += getTableHeaderHtml('current');
  122. header += getTableHeaderHtml('total');
  123. }
  124. header += '</tr>';
  125. tableHeaderElem = $(header);
  126. }
  127. if (panel.legend.sort) {
  128. seriesList = _.sortBy(seriesList, function(series) {
  129. return series.stats[panel.legend.sort];
  130. });
  131. if (panel.legend.sortDesc) {
  132. seriesList = seriesList.reverse();
  133. }
  134. }
  135. // render first time for getting proper legend height
  136. if (!panel.legend.rightSide) {
  137. renderLegendElement(tableHeaderElem);
  138. updateLegendDecimals();
  139. elem.empty();
  140. } else {
  141. updateLegendDecimals();
  142. }
  143. renderLegendElement(tableHeaderElem);
  144. }
  145. function renderSeriesLegendElements() {
  146. let seriesElements = [];
  147. for (i = 0; i < seriesList.length; i++) {
  148. var series = seriesList[i];
  149. if (series.hideFromLegend(panel.legend)) {
  150. continue;
  151. }
  152. var html = '<div class="graph-legend-series';
  153. if (series.yaxis === 2) { html += ' graph-legend-series--right-y'; }
  154. if (ctrl.hiddenSeries[series.alias]) { html += ' graph-legend-series-hidden'; }
  155. html += '" data-series-index="' + i + '">';
  156. html += '<div class="graph-legend-icon">';
  157. html += '<i class="fa fa-minus pointer" style="color:' + series.color + '"></i>';
  158. html += '</div>';
  159. html += '<a class="graph-legend-alias pointer" title="' + series.aliasEscaped + '">' + series.aliasEscaped + '</a>';
  160. if (panel.legend.values) {
  161. var avg = series.formatValue(series.stats.avg);
  162. var current = series.formatValue(series.stats.current);
  163. var min = series.formatValue(series.stats.min);
  164. var max = series.formatValue(series.stats.max);
  165. var total = series.formatValue(series.stats.total);
  166. if (panel.legend.min) { html += '<div class="graph-legend-value min">' + min + '</div>'; }
  167. if (panel.legend.max) { html += '<div class="graph-legend-value max">' + max + '</div>'; }
  168. if (panel.legend.avg) { html += '<div class="graph-legend-value avg">' + avg + '</div>'; }
  169. if (panel.legend.current) { html += '<div class="graph-legend-value current">' + current + '</div>'; }
  170. if (panel.legend.total) { html += '<div class="graph-legend-value total">' + total + '</div>'; }
  171. }
  172. html += '</div>';
  173. seriesElements.push($(html));
  174. }
  175. return seriesElements;
  176. }
  177. function renderLegendElement(tableHeaderElem) {
  178. var seriesElements = renderSeriesLegendElements();
  179. if (panel.legend.alignAsTable) {
  180. var tbodyElem = $('<tbody></tbody>');
  181. tbodyElem.append(tableHeaderElem);
  182. tbodyElem.append(seriesElements);
  183. elem.append(tbodyElem);
  184. } else {
  185. elem.append(seriesElements);
  186. }
  187. if (!panel.legend.rightSide) {
  188. addScrollbar();
  189. } else {
  190. destroyScrollbar();
  191. }
  192. }
  193. function addScrollbar() {
  194. const scrollbarOptions = {
  195. // Number of pixels the content height can surpass the container height without enabling the scroll bar.
  196. scrollYMarginOffset: 2,
  197. suppressScrollX: true
  198. };
  199. if (!legendScrollbar) {
  200. legendScrollbar = new PerfectScrollbar(elem[0], scrollbarOptions);
  201. } else {
  202. legendScrollbar.update();
  203. }
  204. }
  205. function destroyScrollbar() {
  206. if (legendScrollbar) {
  207. legendScrollbar.destroy();
  208. }
  209. }
  210. }
  211. };
  212. });