module.js 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310
  1. /*
  2. ## Terms
  3. ### Parameters
  4. * style :: A hash of css styles
  5. * size :: top N
  6. * arrangement :: How should I arrange the query results? 'horizontal' or 'vertical'
  7. * chart :: Show a chart? 'none', 'bar', 'pie'
  8. * donut :: Only applies to 'pie' charts. Punches a hole in the chart for some reason
  9. * tilt :: Only 'pie' charts. Janky 3D effect. Looks terrible 90% of the time.
  10. * lables :: Only 'pie' charts. Labels on the pie?
  11. */
  12. define([
  13. 'angular',
  14. 'app',
  15. 'underscore',
  16. 'jquery',
  17. 'kbn'
  18. ],
  19. function (angular, app, _, $, kbn) {
  20. 'use strict';
  21. var module = angular.module('kibana.panels.terms', []);
  22. app.useModule(module);
  23. module.controller('terms', function($scope, querySrv, dashboard, filterSrv) {
  24. $scope.panelMeta = {
  25. modals : [
  26. {
  27. description: "Inspect",
  28. icon: "icon-info-sign",
  29. partial: "app/partials/inspector.html",
  30. show: $scope.panel.spyable
  31. }
  32. ],
  33. editorTabs : [
  34. {title:'Queries', src:'app/partials/querySelect.html'}
  35. ],
  36. status : "Beta",
  37. description : "Displays the results of an elasticsearch facet as a pie chart, bar chart, or a "+
  38. "table"
  39. };
  40. // Set and populate defaults
  41. var _d = {
  42. queries : {
  43. mode : 'all',
  44. ids : []
  45. },
  46. field : '_type',
  47. exclude : [],
  48. missing : true,
  49. other : true,
  50. size : 10,
  51. order : 'count',
  52. style : { "font-size": '10pt'},
  53. donut : false,
  54. tilt : false,
  55. labels : true,
  56. arrangement : 'horizontal',
  57. chart : 'bar',
  58. counter_pos : 'above',
  59. spyable : true
  60. };
  61. _.defaults($scope.panel,_d);
  62. $scope.init = function () {
  63. $scope.hits = 0;
  64. $scope.$on('refresh',function(){
  65. $scope.get_data();
  66. });
  67. $scope.get_data();
  68. };
  69. $scope.get_data = function() {
  70. // Make sure we have everything for the request to complete
  71. if(dashboard.indices.length === 0) {
  72. return;
  73. }
  74. $scope.panelMeta.loading = true;
  75. var request,
  76. results,
  77. boolQuery,
  78. queries;
  79. request = $scope.ejs.Request().indices(dashboard.indices);
  80. $scope.panel.queries.ids = querySrv.idsByMode($scope.panel.queries);
  81. queries = querySrv.getQueryObjs($scope.panel.queries.ids);
  82. // This could probably be changed to a BoolFilter
  83. boolQuery = $scope.ejs.BoolQuery();
  84. _.each(queries,function(q) {
  85. boolQuery = boolQuery.should(querySrv.toEjsObj(q));
  86. });
  87. // Terms mode
  88. request = request
  89. .facet($scope.ejs.TermsFacet('terms')
  90. .field($scope.panel.field)
  91. .size($scope.panel.size)
  92. .order($scope.panel.order)
  93. .exclude($scope.panel.exclude)
  94. .facetFilter($scope.ejs.QueryFilter(
  95. $scope.ejs.FilteredQuery(
  96. boolQuery,
  97. filterSrv.getBoolFilter(filterSrv.ids)
  98. )))).size(0);
  99. // Populate the inspector panel
  100. $scope.inspector = angular.toJson(JSON.parse(request.toString()),true);
  101. results = request.doSearch();
  102. // Populate scope when we have results
  103. results.then(function(results) {
  104. var k = 0;
  105. $scope.panelMeta.loading = false;
  106. $scope.hits = results.hits.total;
  107. $scope.data = [];
  108. _.each(results.facets.terms.terms, function(v) {
  109. var slice = { label : v.term, data : [[k,v.count]], actions: true};
  110. $scope.data.push(slice);
  111. k = k + 1;
  112. });
  113. $scope.data.push({label:'Missing field',
  114. data:[[k,results.facets.terms.missing]],meta:"missing",color:'#aaa',opacity:0});
  115. $scope.data.push({label:'Other values',
  116. data:[[k+1,results.facets.terms.other]],meta:"other",color:'#444'});
  117. $scope.$emit('render');
  118. });
  119. };
  120. $scope.build_search = function(term,negate) {
  121. if(_.isUndefined(term.meta)) {
  122. filterSrv.set({type:'terms',field:$scope.panel.field,value:term.label,
  123. mandate:(negate ? 'mustNot':'must')});
  124. } else if(term.meta === 'missing') {
  125. filterSrv.set({type:'exists',field:$scope.panel.field,
  126. mandate:(negate ? 'must':'mustNot')});
  127. } else {
  128. return;
  129. }
  130. };
  131. $scope.set_refresh = function (state) {
  132. $scope.refresh = state;
  133. };
  134. $scope.close_edit = function() {
  135. if($scope.refresh) {
  136. $scope.get_data();
  137. }
  138. $scope.refresh = false;
  139. $scope.$emit('render');
  140. };
  141. $scope.showMeta = function(term) {
  142. if(_.isUndefined(term.meta)) {
  143. return true;
  144. }
  145. if(term.meta === 'other' && !$scope.panel.other) {
  146. return false;
  147. }
  148. if(term.meta === 'missing' && !$scope.panel.missing) {
  149. return false;
  150. }
  151. return true;
  152. };
  153. });
  154. module.directive('termsChart', function(querySrv) {
  155. return {
  156. restrict: 'A',
  157. link: function(scope, elem) {
  158. // Receive render events
  159. scope.$on('render',function(){
  160. render_panel();
  161. });
  162. // Re-render if the window is resized
  163. angular.element(window).bind('resize', function(){
  164. render_panel();
  165. });
  166. // Function for rendering panel
  167. function render_panel() {
  168. var plot, chartData;
  169. // IE doesn't work without this
  170. elem.css({height:scope.panel.height||scope.row.height});
  171. // Make a clone we can operate on.
  172. chartData = _.clone(scope.data);
  173. chartData = scope.panel.missing ? chartData :
  174. _.without(chartData,_.findWhere(chartData,{meta:'missing'}));
  175. chartData = scope.panel.other ? chartData :
  176. _.without(chartData,_.findWhere(chartData,{meta:'other'}));
  177. // Populate element.
  178. require(['jquery.flot.pie'], function(){
  179. // Populate element
  180. try {
  181. // Add plot to scope so we can build out own legend
  182. if(scope.panel.chart === 'bar') {
  183. plot = $.plot(elem, chartData, {
  184. legend: { show: false },
  185. series: {
  186. lines: { show: false, },
  187. bars: { show: true, fill: 1, barWidth: 0.8, horizontal: false },
  188. shadowSize: 1
  189. },
  190. yaxis: { show: true, min: 0, color: "#c8c8c8" },
  191. xaxis: { show: false },
  192. grid: {
  193. borderWidth: 0,
  194. borderColor: '#eee',
  195. color: "#eee",
  196. hoverable: true,
  197. clickable: true
  198. },
  199. colors: querySrv.colors
  200. });
  201. }
  202. if(scope.panel.chart === 'pie') {
  203. var labelFormat = function(label, series){
  204. return '<div ng-click="build_search(panel.field,\''+label+'\')'+
  205. ' "style="font-size:8pt;text-align:center;padding:2px;color:white;">'+
  206. label+'<br/>'+Math.round(series.percent)+'%</div>';
  207. };
  208. plot = $.plot(elem, chartData, {
  209. legend: { show: false },
  210. series: {
  211. pie: {
  212. innerRadius: scope.panel.donut ? 0.4 : 0,
  213. tilt: scope.panel.tilt ? 0.45 : 1,
  214. radius: 1,
  215. show: true,
  216. combine: {
  217. color: '#999',
  218. label: 'The Rest'
  219. },
  220. stroke: {
  221. width: 0
  222. },
  223. label: {
  224. show: scope.panel.labels,
  225. radius: 2/3,
  226. formatter: labelFormat,
  227. threshold: 0.1
  228. }
  229. }
  230. },
  231. //grid: { hoverable: true, clickable: true },
  232. grid: { hoverable: true, clickable: true },
  233. colors: querySrv.colors
  234. });
  235. }
  236. // Populate legend
  237. if(elem.is(":visible")){
  238. setTimeout(function(){
  239. scope.legend = plot.getData();
  240. if(!scope.$$phase) {
  241. scope.$apply();
  242. }
  243. });
  244. }
  245. } catch(e) {
  246. elem.text(e);
  247. }
  248. });
  249. }
  250. elem.bind("plotclick", function (event, pos, object) {
  251. if(object) {
  252. scope.build_search(scope.data[object.seriesIndex]);
  253. }
  254. });
  255. var $tooltip = $('<div>');
  256. elem.bind("plothover", function (event, pos, item) {
  257. if (item) {
  258. var value = scope.panel.chart === 'bar' ? item.datapoint[1] : item.datapoint[1][0][1];
  259. $tooltip
  260. .html(
  261. kbn.query_color_dot(item.series.color, 20) + ' ' +
  262. item.series.label + " (" + value.toFixed(0)+")"
  263. )
  264. .place_tt(pos.pageX, pos.pageY);
  265. } else {
  266. $tooltip.remove();
  267. }
  268. });
  269. }
  270. };
  271. });
  272. });