/*jshint globalstrict:true */ /*global angular:true */ /* ## Hits ### Parameters * style :: A hash of css styles * arrangement :: How should I arrange the query results? 'horizontal' or 'vertical' * chart :: Show a chart? 'none', 'bar', 'pie' * donut :: Only applies to 'pie' charts. Punches a hole in the chart for some reason * tilt :: Only 'pie' charts. Janky 3D effect. Looks terrible 90% of the time. * lables :: Only 'pie' charts. Labels on the pie? */ 'use strict'; angular.module('kibana.hits', []) .controller('hits', function($scope, querySrv, dashboard, filterSrv) { $scope.panelMeta = { editorTabs : [ {title:'Queries', src:'partials/querySelect.html'} ], status : "Stable", description : "The total hits for a query or set of queries. Can be a pie chart, bar chart, "+ "list, or absolute total of all queries combined" }; // Set and populate defaults var _d = { queries : { mode : 'all', ids : [] }, style : { "font-size": '10pt'}, arrangement : 'horizontal', chart : 'bar', counter_pos : 'above', donut : false, tilt : false, labels : true, spyable : true }; _.defaults($scope.panel,_d); $scope.init = function () { $scope.hits = 0; $scope.$on('refresh',function(){ $scope.get_data(); }); $scope.get_data(); }; $scope.get_data = function(segment,query_id) { delete $scope.panel.error; $scope.panelMeta.loading = true; // Make sure we have everything for the request to complete if(dashboard.indices.length === 0) { return; } var _segment = _.isUndefined(segment) ? 0 : segment; var request = $scope.ejs.Request().indices(dashboard.indices[_segment]); $scope.panel.queries.ids = querySrv.idsByMode($scope.panel.queries); // Build the question part of the query _.each($scope.panel.queries.ids, function(id) { var _q = $scope.ejs.FilteredQuery( querySrv.getEjsObj(id), filterSrv.getBoolFilter(filterSrv.ids)); request = request .facet($scope.ejs.QueryFacet(id) .query(_q) ).size(0); }); // Populate the inspector panel $scope.inspector = angular.toJson(JSON.parse(request.toString()),true); // Then run it var results = request.doSearch(); // Populate scope when we have results results.then(function(results) { $scope.panelMeta.loading = false; if(_segment === 0) { $scope.hits = 0; $scope.data = []; query_id = $scope.query_id = new Date().getTime(); } // Check for error and abort if found if(!(_.isUndefined(results.error))) { $scope.panel.error = $scope.parse_error(results.error); return; } // Convert facet ids to numbers var facetIds = _.map(_.keys(results.facets),function(k){return parseInt(k, 10);}); // Make sure we're still on the same query/queries if($scope.query_id === query_id && _.intersection(facetIds,$scope.panel.queries.ids).length === $scope.panel.queries.ids.length ) { var i = 0; _.each($scope.panel.queries.ids, function(id) { var v = results.facets[id]; var hits = _.isUndefined($scope.data[i]) || _segment === 0 ? v.count : $scope.data[i].hits+v.count; $scope.hits += v.count; // Create series $scope.data[i] = { info: querySrv.list[id], id: id, hits: hits, data: [[i,hits]] }; i++; }); $scope.$emit('render'); if(_segment < dashboard.indices.length-1) { $scope.get_data(_segment+1,query_id); } } }); }; $scope.set_refresh = function (state) { $scope.refresh = state; }; $scope.close_edit = function() { if($scope.refresh) { $scope.get_data(); } $scope.refresh = false; $scope.$emit('render'); }; function set_time(time) { $scope.time = time; $scope.get_data(); } }).directive('hitsChart', function(querySrv) { return { restrict: 'A', link: function(scope, elem, attrs, ctrl) { // Receive render events scope.$on('render',function(){ render_panel(); }); // Re-render if the window is resized angular.element(window).bind('resize', function(){ render_panel(); }); // Function for rendering panel function render_panel() { // IE doesn't work without this elem.css({height:scope.panel.height||scope.row.height}); try { _.each(scope.data,function(series) { series.label = series.info.alias; series.color = series.info.color; }); } catch(e) {return;} var scripts = $LAB.script("common/lib/panels/jquery.flot.js").wait() .script("common/lib/panels/jquery.flot.pie.js"); // Populate element. scripts.wait(function(){ // Populate element try { // Add plot to scope so we can build out own legend if(scope.panel.chart === 'bar') { scope.plot = $.plot(elem, scope.data, { legend: { show: false }, series: { lines: { show: false, }, bars: { show: true, fill: 1, barWidth: 0.8, horizontal: false }, shadowSize: 1 }, yaxis: { show: true, min: 0, color: "#c8c8c8" }, xaxis: { show: false }, grid: { borderWidth: 0, borderColor: '#eee', color: "#eee", hoverable: true, }, colors: querySrv.colors }); } if(scope.panel.chart === 'pie') { scope.plot = $.plot(elem, scope.data, { legend: { show: false }, series: { pie: { innerRadius: scope.panel.donut ? 0.4 : 0, tilt: scope.panel.tilt ? 0.45 : 1, radius: 1, show: true, combine: { color: '#999', label: 'The Rest' }, stroke: { width: 0 }, label: { show: scope.panel.labels, radius: 2/3, formatter: function(label, series){ return '
'+ label+'
'+Math.round(series.percent)+'%
'; }, threshold: 0.1 } } }, //grid: { hoverable: true, clickable: true }, grid: { hoverable: true, clickable: true }, colors: querySrv.colors }); } } catch(e) { elem.text(e); } }); } var $tooltip = $('
'); elem.bind("plothover", function (event, pos, item) { if (item) { var value = scope.panel.chart === 'bar' ? item.datapoint[1] : item.datapoint[1][0][1]; $tooltip .html(kbn.query_color_dot(item.series.color, 20) + ' ' + value.toFixed(0)) .place_tt(pos.pageX, pos.pageY); } else { $tooltip.remove(); } }); } }; });