reports_kit 0.1.0 → 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (66) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +6 -5
  3. data/app/assets/javascripts/reports_kit/lib/chart.js +33 -8
  4. data/app/assets/javascripts/reports_kit/lib/report.js +27 -26
  5. data/app/assets/javascripts/reports_kit/lib/table.js +58 -0
  6. data/app/assets/javascripts/reports_kit/vendor/jquery.tablesorter.min.js +4 -0
  7. data/app/assets/stylesheets/reports_kit/reports.css.sass +20 -0
  8. data/config/initializers/mime_types.rb +1 -0
  9. data/docs/dimensions.md +34 -26
  10. data/docs/display_options.md +15 -12
  11. data/docs/filters.md +2 -2
  12. data/docs/measures.md +4 -3
  13. data/gemfiles/mysql.gemfile.lock +14 -1
  14. data/gemfiles/postgresql.gemfile.lock +14 -1
  15. data/lib/reports_kit.rb +15 -0
  16. data/lib/reports_kit/cache.rb +37 -0
  17. data/lib/reports_kit/configuration.rb +13 -1
  18. data/lib/reports_kit/helper.rb +54 -3
  19. data/lib/reports_kit/model_configuration.rb +6 -1
  20. data/lib/reports_kit/order.rb +33 -0
  21. data/lib/reports_kit/relative_time.rb +42 -0
  22. data/lib/reports_kit/report_builder.rb +34 -15
  23. data/lib/reports_kit/reports/abstract_measure.rb +9 -0
  24. data/lib/reports_kit/reports/adapters/mysql.rb +8 -1
  25. data/lib/reports_kit/reports/adapters/postgresql.rb +8 -1
  26. data/lib/reports_kit/reports/composite_measure.rb +43 -0
  27. data/lib/reports_kit/reports/data/chart_options.rb +5 -0
  28. data/lib/reports_kit/reports/data/composite_aggregation.rb +96 -0
  29. data/lib/reports_kit/reports/data/entity.rb +7 -0
  30. data/lib/reports_kit/reports/data/format_one_dimension.rb +120 -0
  31. data/lib/reports_kit/reports/data/format_two_dimensions.rb +141 -0
  32. data/lib/reports_kit/reports/data/generate.rb +72 -25
  33. data/lib/reports_kit/reports/data/generate_for_properties.rb +75 -0
  34. data/lib/reports_kit/reports/data/one_dimension.rb +15 -49
  35. data/lib/reports_kit/reports/data/populate_one_dimension.rb +36 -0
  36. data/lib/reports_kit/reports/data/populate_two_dimensions.rb +104 -0
  37. data/lib/reports_kit/reports/data/two_dimensions.rb +15 -110
  38. data/lib/reports_kit/reports/data/utils.rb +77 -12
  39. data/lib/reports_kit/reports/data/value.rb +7 -0
  40. data/lib/reports_kit/reports/dimension.rb +4 -110
  41. data/lib/reports_kit/reports/dimension_with_measure.rb +137 -0
  42. data/lib/reports_kit/reports/filter.rb +5 -64
  43. data/lib/reports_kit/reports/filter_types/base.rb +1 -1
  44. data/lib/reports_kit/reports/filter_types/boolean.rb +9 -7
  45. data/lib/reports_kit/reports/filter_types/datetime.rb +7 -5
  46. data/lib/reports_kit/reports/filter_types/number.rb +2 -0
  47. data/lib/reports_kit/reports/filter_with_measure.rb +84 -0
  48. data/lib/reports_kit/reports/generate_autocomplete_results.rb +1 -1
  49. data/lib/reports_kit/reports/inferrable_configuration.rb +32 -13
  50. data/lib/reports_kit/reports/measure.rb +48 -12
  51. data/lib/reports_kit/reports_controller.rb +42 -3
  52. data/lib/reports_kit/version.rb +1 -1
  53. data/reports_kit.gemspec +2 -0
  54. data/spec/fixtures/generate_inputs.yml +146 -21
  55. data/spec/fixtures/generate_outputs.yml +768 -17
  56. data/spec/reports_kit/relative_time_spec.rb +29 -0
  57. data/spec/reports_kit/report_builder_spec.rb +28 -0
  58. data/spec/reports_kit/reports/data/generate_spec.rb +614 -27
  59. data/spec/reports_kit/reports/dimension_with_measure_spec.rb +69 -0
  60. data/spec/reports_kit/reports/{filter_spec.rb → filter_with_measure_spec.rb} +4 -3
  61. data/spec/spec_helper.rb +7 -2
  62. data/spec/support/config.rb +11 -0
  63. data/spec/support/helpers.rb +3 -3
  64. data/spec/support/models/issue.rb +7 -0
  65. metadata +53 -4
  66. data/spec/reports_kit/reports/dimension_spec.rb +0 -54
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA1:
3
- metadata.gz: bfc4163f7d65491eef12e9aed2a4d9a9f2eb4360
4
- data.tar.gz: 5a6b2eda37ee37daf47b5f7a7bb570c33094b58a
3
+ metadata.gz: be78c8c9d993eed5a9a37b88549beae7c7aa07af
4
+ data.tar.gz: c786cb1e077f885ab17ed25756030f9dc7f05569
5
5
  SHA512:
6
- metadata.gz: 8e9dbc3e6cbad326e9b4a29fc5b7ac15c06dd51f415c982d1f5be98fa1284e43ca40fd4d9b5a17ec971d1664d38e217e871e19d094ac6b8c597767c00a4bb2a2
7
- data.tar.gz: 797554809a04e2427d41c27d4c14ca529979c5068aeb85fd3ff03f2ebbe2680229273b72137fa7e4fe175b1c4e5d3f620ae59897ee73349f475f1a30147caf1c
6
+ metadata.gz: afccc60cd1df78ddb7c686c89ee73ca43e666a377e3b7c2c5b5c466547177969e8b9c709c51be60a1382769d95d220aa1ebcf25d68bbac8257cd38b429a07e24
7
+ data.tar.gz: ebeeeeefab073d568870b1d6b93af59e7ef0bf305ad0f2463ceb31cac144d600b1263d69b143be84ef754a03d70a531409b950a43bf03d81811aff91d01b80f1
data/README.md CHANGED
@@ -69,7 +69,7 @@ In any view, create a chart that shows the number of records of a model (e.g. `u
69
69
 
70
70
  `app/views/users/index.html.haml`
71
71
  ```haml
72
- = render_report measure: 'user', dimensions: ['created_at']
72
+ = render_report measure: { key: 'user', dimensions: ['created_at'] }
73
73
  ```
74
74
 
75
75
  You're done! `render_report` will render the following chart:
@@ -80,9 +80,10 @@ Instead of passing a hash to `render_report`, you can alternatively configure yo
80
80
 
81
81
  `config/reports_kit/reports/my_users.yml`
82
82
  ```yaml
83
- measure: user
84
- dimensions:
85
- - created_at
83
+ measure:
84
+ key: user
85
+ dimensions:
86
+ - created_at
86
87
  ```
87
88
 
88
89
  `app/views/users/index.html.haml`
@@ -108,7 +109,7 @@ Many other form controls are available; see [Filters](docs/filters.md) for more.
108
109
 
109
110
  ### How It Works
110
111
 
111
- In the Quick Start chart, `measure: 'user'` tells ReportsKit to count the number of `User` records, and `dimensions: ['created_at']` tells it to group by the week of the `created_at` column. Since `created_at` is a `datetime` column, ReportsKit knows that it should sort the results chronologically.
112
+ In the Quick Start chart, `key: 'user'` tells ReportsKit to count the number of `User` records, and `dimensions: ['created_at']` tells it to group by the week of the `created_at` column. Since `created_at` is a `datetime` column, ReportsKit knows that it should sort the results chronologically.
112
113
 
113
114
  To learn how to use more of ReportsKit's features, check out the following resources:
114
115
 
@@ -6,33 +6,58 @@ ReportsKit.Chart = (function(options) {
6
6
  self.report = options.report;
7
7
  self.el = self.report.el;
8
8
 
9
- self.canvas = $('<canvas />').appendTo(self.el);
9
+ self.noResultsEl = $('<div>No data was found</div>').appendTo(self.report.visualizationEl).hide();
10
+ self.loadingIndicatorEl = $('<div class="loading_indicator"></div>').appendTo(self.report.visualizationEl).hide();
11
+ self.canvas = $('<canvas />').appendTo(self.report.visualizationEl);
10
12
  };
11
13
 
12
14
  self.render = function() {
13
- var path = self.el.data('path') + 'reports_kit/reports';
14
- path += '?properties=' + JSON.stringify(self.report.properties());
15
+ var path = self.el.data('path');
16
+ var separator = path.indexOf('?') === -1 ? '?' : '&';
17
+ path += separator + 'properties=' + encodeURIComponent(JSON.stringify(self.report.properties()));
18
+ self.loadingIndicatorEl.fadeIn(5000);
15
19
  $.getJSON(path, function(response) {
16
20
  var data = response.data;
17
- var chart_data = data.chart_data;
18
- var options = chart_data.options;
21
+ var chartData = data.chart_data;
22
+ var options = chartData.options;
23
+ options = self.addAdditionalOptions(options, chartData.standard_options)
19
24
 
20
25
  var args = {
21
26
  type: data.type,
22
- data: chart_data,
27
+ data: chartData,
23
28
  options: options
24
29
  };
30
+ self.loadingIndicatorEl.stop(true, true).hide();
25
31
 
26
32
  if (self.chart) {
27
- self.chart.data.datasets = chart_data.datasets;
28
- self.chart.data.labels = chart_data.labels;
33
+ self.chart.data.datasets = chartData.datasets;
34
+ self.chart.data.labels = chartData.labels;
29
35
  self.chart.update();
30
36
  } else {
31
37
  self.chart = new Chart(self.canvas, args);
32
38
  }
39
+ self.noResultsEl.toggle(self.chart.data.labels.length === 0);
33
40
  });
34
41
  };
35
42
 
43
+ self.addAdditionalOptions = function(options, standardOptions) {
44
+ var additionalOptions = {};
45
+ var maxItems = standardOptions && standardOptions.legend && standardOptions.legend.max_items;
46
+ if (maxItems) {
47
+ additionalOptions = {
48
+ legend: {
49
+ labels: {
50
+ filter: function(item) {
51
+ return item.index < maxItems;
52
+ }
53
+ }
54
+ }
55
+ };
56
+ options = $.extend(true, options, additionalOptions);
57
+ }
58
+ return options;
59
+ };
60
+
36
61
  self.initialize(options);
37
62
 
38
63
  return self;
@@ -4,6 +4,7 @@ ReportsKit.Report = (function(options) {
4
4
  self.initialize = function(options) {
5
5
  self.options = options;
6
6
  self.el = options.el;
7
+ self.visualizationEl = self.el.find('.reports_kit_visualization');
7
8
 
8
9
  self.defaultProperties = self.el.data('properties');
9
10
  self.form = self.el.find('.reports_kit_report_form');
@@ -11,17 +12,20 @@ ReportsKit.Report = (function(options) {
11
12
  self.initializeElements();
12
13
  self.initializeEvents();
13
14
 
14
- self.chart = new ReportsKit.Chart({ report: self });
15
+ if (self.defaultProperties.format == 'table') {
16
+ self.visualization = new ReportsKit.Table({ report: self });
17
+ } else {
18
+ self.visualization = new ReportsKit.Chart({ report: self });
19
+ }
15
20
  self.render();
16
21
  };
17
22
 
18
23
  self.initializeElements = function() {
24
+ self.exportButtons = self.el.find('[data-role=reports_kit_export_button]');
19
25
  self.form.find('.date_range_picker').daterangepicker({
20
26
  locale: {
21
27
  format: 'MMM D, YYYY'
22
28
  },
23
- startDate: moment().subtract(3, 'months'),
24
- endDate: moment(),
25
29
  ranges: {
26
30
  'Today': [moment(), moment()],
27
31
  'Last 7 Days': [moment().subtract(7, 'days'), moment()],
@@ -30,7 +34,8 @@ ReportsKit.Report = (function(options) {
30
34
  'Last 3 Months': [moment().subtract(3, 'months'), moment()],
31
35
  'Last 4 Months': [moment().subtract(4, 'months'), moment()],
32
36
  'Last 6 Months': [moment().subtract(6, 'months'), moment()],
33
- 'Last 12 Months': [moment().subtract(12, 'months'), moment()]
37
+ 'Last 12 Months': [moment().subtract(12, 'months'), moment()],
38
+ 'Year To Date': [moment().startOf('year'), moment()]
34
39
  }
35
40
  });
36
41
  self.form.find('.select2').each(function(index, el) {
@@ -50,11 +55,6 @@ ReportsKit.Report = (function(options) {
50
55
  page: params.page
51
56
  };
52
57
  data = $.extend(data, elParams);
53
- var staticParams = $('[data-role=static_params]').val();
54
- if (staticParams) {
55
- staticParams = JSON.parse(staticParams);
56
- data = $.extend(data, staticParams);
57
- }
58
58
  return data;
59
59
  },
60
60
  processResults: function(data, params) {
@@ -75,11 +75,11 @@ ReportsKit.Report = (function(options) {
75
75
  self.render();
76
76
  return false;
77
77
  })
78
+ self.exportButtons.on('click', self.onClickExportButton);
78
79
  };
79
80
 
80
81
  self.properties = function() {
81
82
  var filterKeysValues = {};
82
- var checkboxKeysEnableds = {};
83
83
  self.form.find('select,:text').each(function(index, el) {
84
84
  var filter = $(el);
85
85
  var key = filter.attr('name');
@@ -88,29 +88,30 @@ ReportsKit.Report = (function(options) {
88
88
  self.form.find(':checkbox').each(function(index, el) {
89
89
  var filter = $(el);
90
90
  var key = filter.attr('name');
91
- checkboxKeysEnableds[key] = filter.prop('checked');
91
+ filterKeysValues[key] = filter.prop('checked');
92
92
  });
93
93
  var properties = $.extend({}, self.defaultProperties);
94
- properties.measure.filters = $.map(properties.measure.filters, function(filter) {
95
- var value = filterKeysValues[filter.key];
96
- if (value !== undefined) {
97
- filter.criteria.value = value;
98
- }
99
- var enabled = checkboxKeysEnableds[filter.key];
100
- if (enabled !== undefined) {
101
- if (enabled) {
102
- filter.criteria.operator = 'true';
103
- } else {
104
- filter.criteria = {};
105
- }
106
- }
107
- return filter;
94
+
95
+ properties.ui_filters = {};
96
+ Object.keys(filterKeysValues).forEach(function(key, index) {
97
+ var value = filterKeysValues[key];
98
+ properties.ui_filters[key] = value;
108
99
  });
100
+
109
101
  return properties;
110
102
  };
111
103
 
104
+ self.onClickExportButton = function(event) {
105
+ var el = $(event.target);
106
+ var path = el.data('path');
107
+ var separator = path.indexOf('?') === -1 ? '?' : '&';
108
+ path += separator + 'properties=' + encodeURIComponent(JSON.stringify(self.properties()));
109
+ window.open(path, '_blank');
110
+ return false;
111
+ };
112
+
112
113
  self.render = function() {
113
- self.chart.render();
114
+ self.visualization.render();
114
115
  };
115
116
 
116
117
  self.initialize(options);
@@ -0,0 +1,58 @@
1
+ ReportsKit.Table = (function(options) {
2
+ var self = this;
3
+
4
+ self.initialize = function(options) {
5
+ self.options = options;
6
+ self.report = options.report;
7
+ self.el = self.report.el;
8
+
9
+ self.loadingIndicatorEl = $('<div class="loading_indicator"></div>').appendTo(self.report.visualizationEl).hide();
10
+ self.table = $('<table />', { 'class': 'table table-striped table-hover' }).appendTo(self.report.visualizationEl);
11
+ };
12
+
13
+ self.render = function() {
14
+ var path = self.el.data('path');
15
+ var separator = path.indexOf('?') === -1 ? '?' : '&';
16
+ path += separator + 'properties=' + encodeURIComponent(JSON.stringify(self.report.properties()));
17
+ self.loadingIndicatorEl.fadeIn(100);
18
+ $.getJSON(path, function(response) {
19
+ var data = response.data;
20
+ var tableData = data.table_data;
21
+
22
+ self.loadingIndicatorEl.stop(true, true).hide();
23
+
24
+ var html = '';
25
+ for(var i = 0; i < tableData.length; i++) {
26
+ if (i == 0) {
27
+ html += '<thead><tr>';
28
+ } else if (i == 1) {
29
+ html += '<tbody><tr>';
30
+ } else {
31
+ html += '<tr>';
32
+ }
33
+
34
+ for(var j = 0; j < tableData[i].length; j++) {
35
+ if (i == 0 || j == 0) {
36
+ html += '<th>' + (tableData[i][j] || '') + '</th>';
37
+ } else {
38
+ html += '<td>' + tableData[i][j] + '</td>';
39
+ }
40
+ }
41
+
42
+ if (i == 0) {
43
+ html += '</tr></thead>';
44
+ } else if (i == tableData.length) {
45
+ html += '</tr></tbody>';
46
+ } else {
47
+ html += '</tr>';
48
+ }
49
+ }
50
+ self.table.html(html);
51
+ self.table.tablesorter();
52
+ });
53
+ };
54
+
55
+ self.initialize(options);
56
+
57
+ return self;
58
+ });
@@ -0,0 +1,4 @@
1
+
2
+ (function($){$.extend({tablesorter:new
3
+ function(){var parsers=[],widgets=[];this.defaults={cssHeader:"header",cssAsc:"headerSortUp",cssDesc:"headerSortDown",cssChildRow:"expand-child",sortInitialOrder:"asc",sortMultiSortKey:"shiftKey",sortForce:null,sortAppend:null,sortLocaleCompare:true,textExtraction:"simple",parsers:{},widgets:[],widgetZebra:{css:["even","odd"]},headers:{},widthFixed:false,cancelSelection:true,sortList:[],headerList:[],dateFormat:"us",decimal:'/\.|\,/g',onRenderHeader:null,selectorHeaders:'thead th',debug:false};function benchmark(s,d){log(s+","+(new Date().getTime()-d.getTime())+"ms");}this.benchmark=benchmark;function log(s){if(typeof console!="undefined"&&typeof console.debug!="undefined"){console.log(s);}else{alert(s);}}function buildParserCache(table,$headers){if(table.config.debug){var parsersDebug="";}if(table.tBodies.length==0)return;var rows=table.tBodies[0].rows;if(rows[0]){var list=[],cells=rows[0].cells,l=cells.length;for(var i=0;i<l;i++){var p=false;if($.metadata&&($($headers[i]).metadata()&&$($headers[i]).metadata().sorter)){p=getParserById($($headers[i]).metadata().sorter);}else if((table.config.headers[i]&&table.config.headers[i].sorter)){p=getParserById(table.config.headers[i].sorter);}if(!p){p=detectParserForColumn(table,rows,-1,i);}if(table.config.debug){parsersDebug+="column:"+i+" parser:"+p.id+"\n";}list.push(p);}}if(table.config.debug){log(parsersDebug);}return list;};function detectParserForColumn(table,rows,rowIndex,cellIndex){var l=parsers.length,node=false,nodeValue=false,keepLooking=true;while(nodeValue==''&&keepLooking){rowIndex++;if(rows[rowIndex]){node=getNodeFromRowAndCellIndex(rows,rowIndex,cellIndex);nodeValue=trimAndGetNodeText(table.config,node);if(table.config.debug){log('Checking if value was empty on row:'+rowIndex);}}else{keepLooking=false;}}for(var i=1;i<l;i++){if(parsers[i].is(nodeValue,table,node)){return parsers[i];}}return parsers[0];}function getNodeFromRowAndCellIndex(rows,rowIndex,cellIndex){return rows[rowIndex].cells[cellIndex];}function trimAndGetNodeText(config,node){return $.trim(getElementText(config,node));}function getParserById(name){var l=parsers.length;for(var i=0;i<l;i++){if(parsers[i].id.toLowerCase()==name.toLowerCase()){return parsers[i];}}return false;}function buildCache(table){if(table.config.debug){var cacheTime=new Date();}var totalRows=(table.tBodies[0]&&table.tBodies[0].rows.length)||0,totalCells=(table.tBodies[0].rows[0]&&table.tBodies[0].rows[0].cells.length)||0,parsers=table.config.parsers,cache={row:[],normalized:[]};for(var i=0;i<totalRows;++i){var c=$(table.tBodies[0].rows[i]),cols=[];if(c.hasClass(table.config.cssChildRow)){cache.row[cache.row.length-1]=cache.row[cache.row.length-1].add(c);continue;}cache.row.push(c);for(var j=0;j<totalCells;++j){cols.push(parsers[j].format(getElementText(table.config,c[0].cells[j]),table,c[0].cells[j]));}cols.push(cache.normalized.length);cache.normalized.push(cols);cols=null;};if(table.config.debug){benchmark("Building cache for "+totalRows+" rows:",cacheTime);}return cache;};function getElementText(config,node){var text="";if(!node)return"";if(!config.supportsTextContent)config.supportsTextContent=node.textContent||false;if(config.textExtraction=="simple"){if(config.supportsTextContent){text=node.textContent;}else{if(node.childNodes[0]&&node.childNodes[0].hasChildNodes()){text=node.childNodes[0].innerHTML;}else{text=node.innerHTML;}}}else{if(typeof(config.textExtraction)=="function"){text=config.textExtraction(node);}else{text=$(node).text();}}return text;}function appendToTable(table,cache){if(table.config.debug){var appendTime=new Date()}var c=cache,r=c.row,n=c.normalized,totalRows=n.length,checkCell=(n[0].length-1),tableBody=$(table.tBodies[0]),rows=[];for(var i=0;i<totalRows;i++){var pos=n[i][checkCell];rows.push(r[pos]);if(!table.config.appender){var l=r[pos].length;for(var j=0;j<l;j++){tableBody[0].appendChild(r[pos][j]);}}}if(table.config.appender){table.config.appender(table,rows);}rows=null;if(table.config.debug){benchmark("Rebuilt table:",appendTime);}applyWidget(table);setTimeout(function(){$(table).trigger("sortEnd");},0);};function buildHeaders(table){if(table.config.debug){var time=new Date();}var meta=($.metadata)?true:false;var header_index=computeTableHeaderCellIndexes(table);$tableHeaders=$(table.config.selectorHeaders,table).each(function(index){this.column=header_index[this.parentNode.rowIndex+"-"+this.cellIndex];this.order=formatSortingOrder(table.config.sortInitialOrder);this.count=this.order;if(checkHeaderMetadata(this)||checkHeaderOptions(table,index))this.sortDisabled=true;if(checkHeaderOptionsSortingLocked(table,index))this.order=this.lockedOrder=checkHeaderOptionsSortingLocked(table,index);if(!this.sortDisabled){var $th=$(this).addClass(table.config.cssHeader);if(table.config.onRenderHeader)table.config.onRenderHeader.apply($th);}table.config.headerList[index]=this;});if(table.config.debug){benchmark("Built headers:",time);log($tableHeaders);}return $tableHeaders;};function computeTableHeaderCellIndexes(t){var matrix=[];var lookup={};var thead=t.getElementsByTagName('THEAD')[0];var trs=thead.getElementsByTagName('TR');for(var i=0;i<trs.length;i++){var cells=trs[i].cells;for(var j=0;j<cells.length;j++){var c=cells[j];var rowIndex=c.parentNode.rowIndex;var cellId=rowIndex+"-"+c.cellIndex;var rowSpan=c.rowSpan||1;var colSpan=c.colSpan||1
4
+ var firstAvailCol;if(typeof(matrix[rowIndex])=="undefined"){matrix[rowIndex]=[];}for(var k=0;k<matrix[rowIndex].length+1;k++){if(typeof(matrix[rowIndex][k])=="undefined"){firstAvailCol=k;break;}}lookup[cellId]=firstAvailCol;for(var k=rowIndex;k<rowIndex+rowSpan;k++){if(typeof(matrix[k])=="undefined"){matrix[k]=[];}var matrixrow=matrix[k];for(var l=firstAvailCol;l<firstAvailCol+colSpan;l++){matrixrow[l]="x";}}}}return lookup;}function checkCellColSpan(table,rows,row){var arr=[],r=table.tHead.rows,c=r[row].cells;for(var i=0;i<c.length;i++){var cell=c[i];if(cell.colSpan>1){arr=arr.concat(checkCellColSpan(table,headerArr,row++));}else{if(table.tHead.length==1||(cell.rowSpan>1||!r[row+1])){arr.push(cell);}}}return arr;};function checkHeaderMetadata(cell){if(($.metadata)&&($(cell).metadata().sorter===false)){return true;};return false;}function checkHeaderOptions(table,i){if((table.config.headers[i])&&(table.config.headers[i].sorter===false)){return true;};return false;}function checkHeaderOptionsSortingLocked(table,i){if((table.config.headers[i])&&(table.config.headers[i].lockedOrder))return table.config.headers[i].lockedOrder;return false;}function applyWidget(table){var c=table.config.widgets;var l=c.length;for(var i=0;i<l;i++){getWidgetById(c[i]).format(table);}}function getWidgetById(name){var l=widgets.length;for(var i=0;i<l;i++){if(widgets[i].id.toLowerCase()==name.toLowerCase()){return widgets[i];}}};function formatSortingOrder(v){if(typeof(v)!="Number"){return(v.toLowerCase()=="desc")?1:0;}else{return(v==1)?1:0;}}function isValueInArray(v,a){var l=a.length;for(var i=0;i<l;i++){if(a[i][0]==v){return true;}}return false;}function setHeadersCss(table,$headers,list,css){$headers.removeClass(css[0]).removeClass(css[1]);var h=[];$headers.each(function(offset){if(!this.sortDisabled){h[this.column]=$(this);}});var l=list.length;for(var i=0;i<l;i++){h[list[i][0]].addClass(css[list[i][1]]);}}function fixColumnWidth(table,$headers){var c=table.config;if(c.widthFixed){var colgroup=$('<colgroup>');$("tr:first td",table.tBodies[0]).each(function(){colgroup.append($('<col>').css('width',$(this).width()));});$(table).prepend(colgroup);};}function updateHeaderSortCount(table,sortList){var c=table.config,l=sortList.length;for(var i=0;i<l;i++){var s=sortList[i],o=c.headerList[s[0]];o.count=s[1];o.count++;}}function multisort(table,sortList,cache){if(table.config.debug){var sortTime=new Date();}var dynamicExp="var sortWrapper = function(a,b) {",l=sortList.length;for(var i=0;i<l;i++){var c=sortList[i][0];var order=sortList[i][1];var s=(table.config.parsers[c].type=="text")?((order==0)?makeSortFunction("text","asc",c):makeSortFunction("text","desc",c)):((order==0)?makeSortFunction("numeric","asc",c):makeSortFunction("numeric","desc",c));var e="e"+i;dynamicExp+="var "+e+" = "+s;dynamicExp+="if("+e+") { return "+e+"; } ";dynamicExp+="else { ";}var orgOrderCol=cache.normalized[0].length-1;dynamicExp+="return a["+orgOrderCol+"]-b["+orgOrderCol+"];";for(var i=0;i<l;i++){dynamicExp+="}; ";}dynamicExp+="return 0; ";dynamicExp+="}; ";if(table.config.debug){benchmark("Evaling expression:"+dynamicExp,new Date());}eval(dynamicExp);cache.normalized.sort(sortWrapper);if(table.config.debug){benchmark("Sorting on "+sortList.toString()+" and dir "+order+" time:",sortTime);}return cache;};function makeSortFunction(type,direction,index){var a="a["+index+"]",b="b["+index+"]";if(type=='text'&&direction=='asc'){return"("+a+" == "+b+" ? 0 : ("+a+" === null ? Number.POSITIVE_INFINITY : ("+b+" === null ? Number.NEGATIVE_INFINITY : ("+a+" < "+b+") ? -1 : 1 )));";}else if(type=='text'&&direction=='desc'){return"("+a+" == "+b+" ? 0 : ("+a+" === null ? Number.POSITIVE_INFINITY : ("+b+" === null ? Number.NEGATIVE_INFINITY : ("+b+" < "+a+") ? -1 : 1 )));";}else if(type=='numeric'&&direction=='asc'){return"("+a+" === null && "+b+" === null) ? 0 :("+a+" === null ? Number.POSITIVE_INFINITY : ("+b+" === null ? Number.NEGATIVE_INFINITY : "+a+" - "+b+"));";}else if(type=='numeric'&&direction=='desc'){return"("+a+" === null && "+b+" === null) ? 0 :("+a+" === null ? Number.POSITIVE_INFINITY : ("+b+" === null ? Number.NEGATIVE_INFINITY : "+b+" - "+a+"));";}};function makeSortText(i){return"((a["+i+"] < b["+i+"]) ? -1 : ((a["+i+"] > b["+i+"]) ? 1 : 0));";};function makeSortTextDesc(i){return"((b["+i+"] < a["+i+"]) ? -1 : ((b["+i+"] > a["+i+"]) ? 1 : 0));";};function makeSortNumeric(i){return"a["+i+"]-b["+i+"];";};function makeSortNumericDesc(i){return"b["+i+"]-a["+i+"];";};function sortText(a,b){if(table.config.sortLocaleCompare)return a.localeCompare(b);return((a<b)?-1:((a>b)?1:0));};function sortTextDesc(a,b){if(table.config.sortLocaleCompare)return b.localeCompare(a);return((b<a)?-1:((b>a)?1:0));};function sortNumeric(a,b){return a-b;};function sortNumericDesc(a,b){return b-a;};function getCachedSortType(parsers,i){return parsers[i].type;};this.construct=function(settings){return this.each(function(){if(!this.tHead||!this.tBodies)return;var $this,$document,$headers,cache,config,shiftDown=0,sortOrder;this.config={};config=$.extend(this.config,$.tablesorter.defaults,settings);$this=$(this);$.data(this,"tablesorter",config);$headers=buildHeaders(this);this.config.parsers=buildParserCache(this,$headers);cache=buildCache(this);var sortCSS=[config.cssDesc,config.cssAsc];fixColumnWidth(this);$headers.click(function(e){var totalRows=($this[0].tBodies[0]&&$this[0].tBodies[0].rows.length)||0;if(!this.sortDisabled&&totalRows>0){$this.trigger("sortStart");var $cell=$(this);var i=this.column;this.order=this.count++%2;if(this.lockedOrder)this.order=this.lockedOrder;if(!e[config.sortMultiSortKey]){config.sortList=[];if(config.sortForce!=null){var a=config.sortForce;for(var j=0;j<a.length;j++){if(a[j][0]!=i){config.sortList.push(a[j]);}}}config.sortList.push([i,this.order]);}else{if(isValueInArray(i,config.sortList)){for(var j=0;j<config.sortList.length;j++){var s=config.sortList[j],o=config.headerList[s[0]];if(s[0]==i){o.count=s[1];o.count++;s[1]=o.count%2;}}}else{config.sortList.push([i,this.order]);}};setTimeout(function(){setHeadersCss($this[0],$headers,config.sortList,sortCSS);appendToTable($this[0],multisort($this[0],config.sortList,cache));},1);return false;}}).mousedown(function(){if(config.cancelSelection){this.onselectstart=function(){return false};return false;}});$this.bind("update",function(){var me=this;setTimeout(function(){me.config.parsers=buildParserCache(me,$headers);cache=buildCache(me);},1);}).bind("updateCell",function(e,cell){var config=this.config;var pos=[(cell.parentNode.rowIndex-1),cell.cellIndex];cache.normalized[pos[0]][pos[1]]=config.parsers[pos[1]].format(getElementText(config,cell),cell);}).bind("sorton",function(e,list){$(this).trigger("sortStart");config.sortList=list;var sortList=config.sortList;updateHeaderSortCount(this,sortList);setHeadersCss(this,$headers,sortList,sortCSS);appendToTable(this,multisort(this,sortList,cache));}).bind("appendCache",function(){appendToTable(this,cache);}).bind("applyWidgetId",function(e,id){getWidgetById(id).format(this);}).bind("applyWidgets",function(){applyWidget(this);});if($.metadata&&($(this).metadata()&&$(this).metadata().sortlist)){config.sortList=$(this).metadata().sortlist;}if(config.sortList.length>0){$this.trigger("sorton",[config.sortList]);}applyWidget(this);});};this.addParser=function(parser){var l=parsers.length,a=true;for(var i=0;i<l;i++){if(parsers[i].id.toLowerCase()==parser.id.toLowerCase()){a=false;}}if(a){parsers.push(parser);};};this.addWidget=function(widget){widgets.push(widget);};this.formatFloat=function(s){var i=parseFloat(s);return(isNaN(i))?0:i;};this.formatInt=function(s){var i=parseInt(s);return(isNaN(i))?0:i;};this.isDigit=function(s,config){return/^[-+]?\d*$/.test($.trim(s.replace(/[,.']/g,'')));};this.clearTableBody=function(table){if($.browser.msie){function empty(){while(this.firstChild)this.removeChild(this.firstChild);}empty.apply(table.tBodies[0]);}else{table.tBodies[0].innerHTML="";}};}});$.fn.extend({tablesorter:$.tablesorter.construct});var ts=$.tablesorter;ts.addParser({id:"text",is:function(s){return true;},format:function(s){return $.trim(s.toLocaleLowerCase());},type:"text"});ts.addParser({id:"digit",is:function(s,table){var c=table.config;return $.tablesorter.isDigit(s,c);},format:function(s){return $.tablesorter.formatFloat(s);},type:"numeric"});ts.addParser({id:"currency",is:function(s){return/^[£$€?.]/.test(s);},format:function(s){return $.tablesorter.formatFloat(s.replace(new RegExp(/[£$€]/g),""));},type:"numeric"});ts.addParser({id:"ipAddress",is:function(s){return/^\d{2,3}[\.]\d{2,3}[\.]\d{2,3}[\.]\d{2,3}$/.test(s);},format:function(s){var a=s.split("."),r="",l=a.length;for(var i=0;i<l;i++){var item=a[i];if(item.length==2){r+="0"+item;}else{r+=item;}}return $.tablesorter.formatFloat(r);},type:"numeric"});ts.addParser({id:"url",is:function(s){return/^(https?|ftp|file):\/\/$/.test(s);},format:function(s){return jQuery.trim(s.replace(new RegExp(/(https?|ftp|file):\/\//),''));},type:"text"});ts.addParser({id:"isoDate",is:function(s){return/^\d{4}[\/-]\d{1,2}[\/-]\d{1,2}$/.test(s);},format:function(s){return $.tablesorter.formatFloat((s!="")?new Date(s.replace(new RegExp(/-/g),"/")).getTime():"0");},type:"numeric"});ts.addParser({id:"percent",is:function(s){return/\%$/.test($.trim(s));},format:function(s){return $.tablesorter.formatFloat(s.replace(new RegExp(/%/g),""));},type:"numeric"});ts.addParser({id:"usLongDate",is:function(s){return s.match(new RegExp(/^[A-Za-z]{3,10}\.? [0-9]{1,2}, ([0-9]{4}|'?[0-9]{2}) (([0-2]?[0-9]:[0-5][0-9])|([0-1]?[0-9]:[0-5][0-9]\s(AM|PM)))$/));},format:function(s){return $.tablesorter.formatFloat(new Date(s).getTime());},type:"numeric"});ts.addParser({id:"shortDate",is:function(s){return/\d{1,2}[\/\-]\d{1,2}[\/\-]\d{2,4}/.test(s);},format:function(s,table){var c=table.config;s=s.replace(/\-/g,"/");if(c.dateFormat=="us"){s=s.replace(/(\d{1,2})[\/\-](\d{1,2})[\/\-](\d{4})/,"$3/$1/$2");}else if (c.dateFormat == "pt") {s = s.replace(/(\d{1,2})[\/\-](\d{1,2})[\/\-](\d{4})/, "$3/$2/$1");} else if(c.dateFormat=="uk"){s=s.replace(/(\d{1,2})[\/\-](\d{1,2})[\/\-](\d{4})/,"$3/$2/$1");}else if(c.dateFormat=="dd/mm/yy"||c.dateFormat=="dd-mm-yy"){s=s.replace(/(\d{1,2})[\/\-](\d{1,2})[\/\-](\d{2})/,"$1/$2/$3");}return $.tablesorter.formatFloat(new Date(s).getTime());},type:"numeric"});ts.addParser({id:"time",is:function(s){return/^(([0-2]?[0-9]:[0-5][0-9])|([0-1]?[0-9]:[0-5][0-9]\s(am|pm)))$/.test(s);},format:function(s){return $.tablesorter.formatFloat(new Date("2000/01/01 "+s).getTime());},type:"numeric"});ts.addParser({id:"metadata",is:function(s){return false;},format:function(s,table,cell){var c=table.config,p=(!c.parserMetadataName)?'sortValue':c.parserMetadataName;return $(cell).metadata()[p];},type:"numeric"});ts.addWidget({id:"zebra",format:function(table){if(table.config.debug){var time=new Date();}var $tr,row=-1,odd;$("tr:visible",table.tBodies[0]).each(function(i){$tr=$(this);if(!$tr.hasClass(table.config.cssChildRow))row++;odd=(row%2==0);$tr.removeClass(table.config.widgetZebra.css[odd?0:1]).addClass(table.config.widgetZebra.css[odd?1:0])});if(table.config.debug){$.tablesorter.benchmark("Applying Zebra widget",time);}}});})(jQuery);
@@ -5,3 +5,23 @@
5
5
  width: 100%
6
6
  .date_range_picker
7
7
  width: 180px
8
+ .loading_indicator
9
+ position: absolute
10
+ padding: 22px
11
+ background: url() no-repeat center
12
+
13
+ .reports_kit_visualization
14
+ table
15
+ thead
16
+ .header
17
+ color: #337ab7
18
+ cursor: pointer
19
+ .headerSortUp, .headerSortDown
20
+ &::after
21
+ margin-left: 6px
22
+ font-size: 0.8em
23
+ font-family: "Arial"
24
+ .headerSortUp::after
25
+ content: "\25BC"
26
+ .headerSortDown::after
27
+ content: "\25B2"
@@ -0,0 +1 @@
1
+ Mime::Type.register 'application/xls', :xls
@@ -15,64 +15,71 @@ end
15
15
  You can then use `dimensions: ['carrier']` to count the number of Flights per Carrier:
16
16
 
17
17
  ```yaml
18
- measure: flight
19
- dimensions:
20
- - carrier
18
+ measure:
19
+ key: flight
20
+ dimensions:
21
+ - carrier
21
22
  ```
22
23
  [<img src="images/flights_by_carrier.png?raw=true" width="500" />](images/flights_by_carrier.png?raw=true)
23
24
 
24
25
  You can also use two dimensions:
25
26
 
26
27
  ```yaml
27
- measure: flight
28
- dimensions:
29
- - carrier
30
- - flight_at
28
+ measure:
29
+ key: flight
30
+ dimensions:
31
+ - carrier
32
+ - flight_at
31
33
  ```
32
34
  [<img src="images/flights_by_carrier_and_flight_at.png?raw=true" width="500" />](images/flights_by_carrier_and_flight_at.png?raw=true)
33
35
 
34
36
  Dimensions can be configured using a string (`carrier`):
35
37
 
36
38
  ```yaml
37
- measure: flight
38
- dimensions:
39
- - carrier
39
+ measure:
40
+ key: flight
41
+ dimensions:
42
+ - carrier
40
43
  ```
41
44
 
42
45
  Or, if you need to use options, you can configure them using a hash:
43
46
 
44
47
  ```yaml
45
- measure: flight
46
- dimensions:
47
- - key: carrier
48
- limit: 5
48
+ measure:
49
+ key: flight
50
+ dimensions:
51
+ - key: carrier
52
+ limit: 5
49
53
  ```
50
54
  #### Types
51
55
 
52
56
  ##### Association
53
57
 
54
58
  ```yaml
55
- measure: flight
56
- dimensions:
57
- - carrier
59
+ measure:
60
+ key: flight
61
+ dimensions:
62
+ - carrier
58
63
  ```
59
64
  [<img src="images/flights_by_carrier.png?raw=true" width="500" />](images/flights_by_carrier.png?raw=true)
60
65
 
61
66
  ##### Datetime Column
62
67
 
63
68
  ```yaml
64
- measure: flight
65
- dimensions:
66
- - flight_at
69
+ measure:
70
+ key: flight
71
+ dimensions:
72
+ - flight_at
67
73
  ```
68
74
  [<img src="images/flights_by_flight_at.png?raw=true" width="500" />](images/flights_by_flight_at.png?raw=true)
69
75
 
70
76
  ##### Integer Column
71
77
 
72
78
  ```yaml
73
- measure: flight
74
- dimensions:
75
- - delay
79
+ measure:
80
+ key: flight
81
+ dimensions:
82
+ - delay
76
83
  ```
77
84
  [<img src="images/flights_by_delay.png?raw=true" width="500" />](images/flights_by_delay.png?raw=true)
78
85
 
@@ -93,9 +100,10 @@ end
93
100
  We can then use the `hours_delayed` dimension:
94
101
 
95
102
  ```yaml
96
- measure: flight
97
- dimensions:
98
- - hours_delayed
103
+ measure:
104
+ key: flight
105
+ dimensions:
106
+ - hours_delayed
99
107
  ```
100
108
  [<img src="images/flights_by_hours_delayed.png?raw=true" width="500" />](images/flights_by_hours_delayed.png?raw=true)
101
109
 
@@ -11,9 +11,10 @@ You can use any `type` value supported by Chart.js, including `bar`, `line`, `ho
11
11
  Here's an example of a horizontal bar chart:
12
12
 
13
13
  ```yaml
14
- measure: flight
15
- dimensions:
16
- - carrier
14
+ measure:
15
+ key: flight
16
+ dimensions:
17
+ - carrier
17
18
  chart:
18
19
  type: horizontalBar
19
20
  options:
@@ -36,10 +37,11 @@ You can use any `options` that are supported by Chart.js.
36
37
  Here's an example of a chart with Chart.js options:
37
38
 
38
39
  ```yaml
39
- measure: flight
40
- dimensions:
41
- - origin_market
42
- - carrier
40
+ measure:
41
+ key: flight
42
+ dimensions:
43
+ - origin_market
44
+ - carrier
43
45
  chart:
44
46
  type: horizontalBar
45
47
  options:
@@ -64,11 +66,12 @@ You can use any `datasets` options that are supported by Chart.js.
64
66
  Here's an example of a chart with `datasets` options:
65
67
 
66
68
  ```yaml
67
- measure: flight
68
- dimensions:
69
- - flight_at
70
- - key: carrier
71
- limit: 3
69
+ measure:
70
+ key: flight
71
+ dimensions:
72
+ - flight_at
73
+ - key: carrier
74
+ limit: 3
72
75
  chart:
73
76
  type: line
74
77
  datasets: