legend.ts 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291
  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:
  49. '<series-color-picker series="series" onToggleAxis="toggleAxis" onColorChange="colorSelected">' +
  50. "</series-color-picker>",
  51. openOn: "hover",
  52. model: {
  53. series: series,
  54. toggleAxis: function() {
  55. ctrl.toggleAxis(series);
  56. },
  57. colorSelected: function(color) {
  58. ctrl.changeSeriesColor(series, color);
  59. }
  60. }
  61. });
  62. });
  63. }
  64. function toggleSeries(e) {
  65. var el = $(e.currentTarget);
  66. var index = getSeriesIndexForElement(el);
  67. var seriesInfo = seriesList[index];
  68. var scrollPosition = $(elem.children("tbody")).scrollTop();
  69. ctrl.toggleSeries(seriesInfo, e);
  70. $(elem.children("tbody")).scrollTop(scrollPosition);
  71. }
  72. function sortLegend(e) {
  73. var el = $(e.currentTarget);
  74. var stat = el.data("stat");
  75. if (stat !== panel.legend.sort) {
  76. panel.legend.sortDesc = null;
  77. }
  78. // if already sort ascending, disable sorting
  79. if (panel.legend.sortDesc === false) {
  80. panel.legend.sort = null;
  81. panel.legend.sortDesc = null;
  82. ctrl.render();
  83. return;
  84. }
  85. panel.legend.sortDesc = !panel.legend.sortDesc;
  86. panel.legend.sort = stat;
  87. ctrl.render();
  88. }
  89. function getTableHeaderHtml(statName) {
  90. if (!panel.legend[statName]) {
  91. return "";
  92. }
  93. var html =
  94. '<th class="pointer" data-stat="' + statName + '">' + statName;
  95. if (panel.legend.sort === statName) {
  96. var cssClass = panel.legend.sortDesc
  97. ? "fa fa-caret-down"
  98. : "fa fa-caret-up";
  99. html += ' <span class="' + cssClass + '"></span>';
  100. }
  101. return html + "</th>";
  102. }
  103. function render() {
  104. if (!ctrl.panel.legend.show) {
  105. elem.empty();
  106. firstRender = true;
  107. return;
  108. }
  109. if (firstRender) {
  110. elem.on("click", ".graph-legend-icon", openColorSelector);
  111. elem.on("click", ".graph-legend-alias", toggleSeries);
  112. elem.on("click", "th", sortLegend);
  113. firstRender = false;
  114. }
  115. seriesList = data;
  116. elem.empty();
  117. // Set min-width if side style and there is a value, otherwise remove the CSS propery
  118. var width =
  119. panel.legend.rightSide && panel.legend.sideWidth
  120. ? panel.legend.sideWidth + "px"
  121. : "";
  122. elem.css("min-width", width);
  123. elem.toggleClass(
  124. "graph-legend-table",
  125. panel.legend.alignAsTable === true
  126. );
  127. var tableHeaderElem;
  128. if (panel.legend.alignAsTable) {
  129. var header = "<tr>";
  130. header += '<th colspan="2" style="text-align:left"></th>';
  131. if (panel.legend.values) {
  132. header += getTableHeaderHtml("min");
  133. header += getTableHeaderHtml("max");
  134. header += getTableHeaderHtml("avg");
  135. header += getTableHeaderHtml("current");
  136. header += getTableHeaderHtml("total");
  137. }
  138. header += "</tr>";
  139. tableHeaderElem = $(header);
  140. }
  141. if (panel.legend.sort) {
  142. seriesList = _.sortBy(seriesList, function(series) {
  143. return series.stats[panel.legend.sort];
  144. });
  145. if (panel.legend.sortDesc) {
  146. seriesList = seriesList.reverse();
  147. }
  148. }
  149. // render first time for getting proper legend height
  150. if (!panel.legend.rightSide) {
  151. renderLegendElement(tableHeaderElem);
  152. updateLegendDecimals();
  153. elem.empty();
  154. } else {
  155. updateLegendDecimals();
  156. }
  157. renderLegendElement(tableHeaderElem);
  158. }
  159. function renderSeriesLegendElements() {
  160. let seriesElements = [];
  161. for (i = 0; i < seriesList.length; i++) {
  162. var series = seriesList[i];
  163. if (series.hideFromLegend(panel.legend)) {
  164. continue;
  165. }
  166. var html = '<div class="graph-legend-series';
  167. if (series.yaxis === 2) {
  168. html += " graph-legend-series--right-y";
  169. }
  170. if (ctrl.hiddenSeries[series.alias]) {
  171. html += " graph-legend-series-hidden";
  172. }
  173. html += '" data-series-index="' + i + '">';
  174. html += '<div class="graph-legend-icon">';
  175. html +=
  176. '<i class="fa fa-minus pointer" style="color:' +
  177. series.color +
  178. '"></i>';
  179. html += "</div>";
  180. html +=
  181. '<a class="graph-legend-alias pointer" title="' +
  182. series.aliasEscaped +
  183. '">' +
  184. series.aliasEscaped +
  185. "</a>";
  186. if (panel.legend.values) {
  187. var avg = series.formatValue(series.stats.avg);
  188. var current = series.formatValue(series.stats.current);
  189. var min = series.formatValue(series.stats.min);
  190. var max = series.formatValue(series.stats.max);
  191. var total = series.formatValue(series.stats.total);
  192. if (panel.legend.min) {
  193. html += '<div class="graph-legend-value min">' + min + "</div>";
  194. }
  195. if (panel.legend.max) {
  196. html += '<div class="graph-legend-value max">' + max + "</div>";
  197. }
  198. if (panel.legend.avg) {
  199. html += '<div class="graph-legend-value avg">' + avg + "</div>";
  200. }
  201. if (panel.legend.current) {
  202. html +=
  203. '<div class="graph-legend-value current">' + current + "</div>";
  204. }
  205. if (panel.legend.total) {
  206. html +=
  207. '<div class="graph-legend-value total">' + total + "</div>";
  208. }
  209. }
  210. html += "</div>";
  211. seriesElements.push($(html));
  212. }
  213. return seriesElements;
  214. }
  215. function renderLegendElement(tableHeaderElem) {
  216. var seriesElements = renderSeriesLegendElements();
  217. if (panel.legend.alignAsTable) {
  218. var tbodyElem = $("<tbody></tbody>");
  219. tbodyElem.append(tableHeaderElem);
  220. tbodyElem.append(seriesElements);
  221. elem.append(tbodyElem);
  222. } else {
  223. elem.append(seriesElements);
  224. }
  225. if (!panel.legend.rightSide) {
  226. addScrollbar();
  227. } else {
  228. destroyScrollbar();
  229. }
  230. }
  231. function addScrollbar() {
  232. const scrollbarOptions = {
  233. // Number of pixels the content height can surpass the container height without enabling the scroll bar.
  234. scrollYMarginOffset: 2,
  235. suppressScrollX: true
  236. };
  237. if (!legendScrollbar) {
  238. legendScrollbar = new PerfectScrollbar(elem[0], scrollbarOptions);
  239. } else {
  240. legendScrollbar.update();
  241. }
  242. }
  243. function destroyScrollbar() {
  244. if (legendScrollbar) {
  245. legendScrollbar.destroy();
  246. }
  247. }
  248. }
  249. };
  250. });