legend.ts 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305
  1. import angular from 'angular';
  2. import _ from 'lodash';
  3. import $ from 'jquery';
  4. import baron from 'baron';
  5. var module = angular.module('grafana.directives');
  6. module.directive('graphLegend', function(popoverSrv, $timeout) {
  7. return {
  8. link: function(scope, elem) {
  9. var firstRender = true;
  10. var ctrl = scope.ctrl;
  11. var panel = ctrl.panel;
  12. var data;
  13. var seriesList;
  14. var i;
  15. var legendScrollbar;
  16. const legendRightDefaultWidth = 10;
  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 getSeriesIndexForElement(el) {
  30. return el.parents('[data-series-index]').data('series-index');
  31. }
  32. function openColorSelector(e) {
  33. // if we clicked inside poup container ignore click
  34. if ($(e.target).parents('.popover').length) {
  35. return;
  36. }
  37. var el = $(e.currentTarget).find('.fa-minus');
  38. var index = getSeriesIndexForElement(el);
  39. var series = seriesList[index];
  40. $timeout(function() {
  41. popoverSrv.show({
  42. element: el[0],
  43. position: 'bottom left',
  44. targetAttachment: 'top left',
  45. template:
  46. '<series-color-picker series="series" onToggleAxis="toggleAxis" onColorChange="colorSelected">' +
  47. '</series-color-picker>',
  48. openOn: 'hover',
  49. model: {
  50. series: series,
  51. toggleAxis: function() {
  52. ctrl.toggleAxis(series);
  53. },
  54. colorSelected: function(color) {
  55. ctrl.changeSeriesColor(series, color);
  56. },
  57. },
  58. });
  59. });
  60. }
  61. function toggleSeries(e) {
  62. var el = $(e.currentTarget);
  63. var index = getSeriesIndexForElement(el);
  64. var seriesInfo = seriesList[index];
  65. var scrollPosition = $(elem.children('tbody')).scrollTop();
  66. ctrl.toggleSeries(seriesInfo, e);
  67. $(elem.children('tbody')).scrollTop(scrollPosition);
  68. }
  69. function sortLegend(e) {
  70. var el = $(e.currentTarget);
  71. var stat = el.data('stat');
  72. if (stat !== panel.legend.sort) {
  73. panel.legend.sortDesc = null;
  74. }
  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]) {
  88. return '';
  89. }
  90. var html = '<th class="pointer" data-stat="' + statName + '">' + statName;
  91. if (panel.legend.sort === statName) {
  92. var cssClass = panel.legend.sortDesc ? 'fa fa-caret-down' : 'fa fa-caret-up';
  93. html += ' <span class="' + cssClass + '"></span>';
  94. }
  95. return html + '</th>';
  96. }
  97. function render() {
  98. let legendWidth = elem.width();
  99. if (!ctrl.panel.legend.show) {
  100. elem.empty();
  101. firstRender = true;
  102. return;
  103. }
  104. if (firstRender) {
  105. elem.on('click', '.graph-legend-icon', openColorSelector);
  106. elem.on('click', '.graph-legend-alias', toggleSeries);
  107. elem.on('click', 'th', sortLegend);
  108. firstRender = false;
  109. }
  110. seriesList = data;
  111. elem.empty();
  112. // Set min-width if side style and there is a value, otherwise remove the CSS propery
  113. var width = panel.legend.rightSide && panel.legend.sideWidth ? panel.legend.sideWidth + 'px' : '';
  114. elem.css('min-width', width);
  115. elem.toggleClass('graph-legend-table', panel.legend.alignAsTable === true);
  116. var tableHeaderElem;
  117. if (panel.legend.alignAsTable) {
  118. var header = '<tr>';
  119. header += '<th colspan="2" style="text-align:left"></th>';
  120. if (panel.legend.values) {
  121. header += getTableHeaderHtml('min');
  122. header += getTableHeaderHtml('max');
  123. header += getTableHeaderHtml('avg');
  124. header += getTableHeaderHtml('current');
  125. header += getTableHeaderHtml('total');
  126. }
  127. header += '</tr>';
  128. tableHeaderElem = $(header);
  129. }
  130. if (panel.legend.sort) {
  131. seriesList = _.sortBy(seriesList, function(series) {
  132. let sort = series.stats[panel.legend.sort];
  133. if (sort === null) {
  134. sort = -Infinity;
  135. }
  136. return sort;
  137. });
  138. if (panel.legend.sortDesc) {
  139. seriesList = seriesList.reverse();
  140. }
  141. }
  142. // render first time for getting proper legend height
  143. if (!panel.legend.rightSide || (panel.legend.rightSide && legendWidth !== legendRightDefaultWidth)) {
  144. renderLegendElement(tableHeaderElem);
  145. elem.empty();
  146. }
  147. renderLegendElement(tableHeaderElem);
  148. }
  149. function renderSeriesLegendElements() {
  150. let seriesElements = [];
  151. for (i = 0; i < seriesList.length; i++) {
  152. var series = seriesList[i];
  153. if (series.hideFromLegend(panel.legend)) {
  154. continue;
  155. }
  156. var html = '<div class="graph-legend-series';
  157. if (series.yaxis === 2) {
  158. html += ' graph-legend-series--right-y';
  159. }
  160. if (ctrl.hiddenSeries[series.alias]) {
  161. html += ' graph-legend-series-hidden';
  162. }
  163. html += '" data-series-index="' + i + '">';
  164. html += '<div class="graph-legend-icon">';
  165. html += '<i class="fa fa-minus pointer" style="color:' + series.color + '"></i>';
  166. html += '</div>';
  167. html +=
  168. '<a class="graph-legend-alias pointer" title="' + series.aliasEscaped + '">' + series.aliasEscaped + '</a>';
  169. if (panel.legend.values) {
  170. var avg = series.formatValue(series.stats.avg);
  171. var current = series.formatValue(series.stats.current);
  172. var min = series.formatValue(series.stats.min);
  173. var max = series.formatValue(series.stats.max);
  174. var total = series.formatValue(series.stats.total);
  175. if (panel.legend.min) {
  176. html += '<div class="graph-legend-value min">' + min + '</div>';
  177. }
  178. if (panel.legend.max) {
  179. html += '<div class="graph-legend-value max">' + max + '</div>';
  180. }
  181. if (panel.legend.avg) {
  182. html += '<div class="graph-legend-value avg">' + avg + '</div>';
  183. }
  184. if (panel.legend.current) {
  185. html += '<div class="graph-legend-value current">' + current + '</div>';
  186. }
  187. if (panel.legend.total) {
  188. html += '<div class="graph-legend-value total">' + total + '</div>';
  189. }
  190. }
  191. html += '</div>';
  192. seriesElements.push($(html));
  193. }
  194. return seriesElements;
  195. }
  196. function renderLegendElement(tableHeaderElem) {
  197. let legendWidth = elem.width();
  198. var seriesElements = renderSeriesLegendElements();
  199. if (panel.legend.alignAsTable) {
  200. var tbodyElem = $('<tbody></tbody>');
  201. tbodyElem.append(tableHeaderElem);
  202. tbodyElem.append(seriesElements);
  203. elem.append(tbodyElem);
  204. } else {
  205. elem.append(seriesElements);
  206. }
  207. if (!panel.legend.rightSide || (panel.legend.rightSide && legendWidth !== legendRightDefaultWidth)) {
  208. addScrollbar();
  209. } else {
  210. destroyScrollbar();
  211. }
  212. }
  213. function addScrollbar() {
  214. const scrollRootClass = 'baron baron__root';
  215. const scrollerClass = 'baron__scroller';
  216. const scrollBarHTML = `
  217. <div class="baron__track">
  218. <div class="baron__bar"></div>
  219. </div>
  220. `;
  221. let scrollRoot = elem.parent();
  222. // let scroller = elem.find(':first-child').first();
  223. let scroller = elem;
  224. // clear existing scroll bar track to prevent duplication
  225. elem
  226. .parent()
  227. .find('.baron__track')
  228. .remove();
  229. scrollRoot.addClass(scrollRootClass);
  230. $(scrollBarHTML).appendTo(scrollRoot);
  231. scroller.addClass(scrollerClass);
  232. // Fix .graph-legend-content max-height
  233. // Couldn't find how to do it via CSS
  234. const legendHeight = scrollRoot.height();
  235. elem.css('max-height', legendHeight);
  236. let scrollbarParams = {
  237. root: scrollRoot[0],
  238. scroller: scroller[0],
  239. bar: '.baron__bar',
  240. track: '.baron__track',
  241. barOnCls: '_scrollbar',
  242. scrollingCls: '_scrolling',
  243. };
  244. if (!legendScrollbar) {
  245. legendScrollbar = baron(scrollbarParams);
  246. } else {
  247. destroyScrollbar();
  248. legendScrollbar = baron(scrollbarParams);
  249. }
  250. }
  251. function destroyScrollbar() {
  252. if (legendScrollbar) {
  253. legendScrollbar.dispose();
  254. legendScrollbar = undefined;
  255. }
  256. }
  257. },
  258. };
  259. });