Skip to content

Commit

Permalink
Add dynamic change of Y axis
Browse files Browse the repository at this point in the history
(cherry picked from commit 5bc7efab2573a508c58f7e32ee5275a5caf646e1)
  • Loading branch information
PanSpagetka committed Jan 17, 2017
1 parent d922a73 commit 2a07611
Show file tree
Hide file tree
Showing 4 changed files with 120 additions and 14 deletions.
49 changes: 38 additions & 11 deletions app/assets/javascripts/miq_application.js
Original file line number Diff line number Diff line change
Expand Up @@ -1668,7 +1668,7 @@ function chartData(type, data, data2) {

// small C&U charts have very limited height
if (data.miq.flat_chart) {
var max = _.max(_.flatten(_.tail(data.data.columns).map(_.tail)));
var max = _.max(getChartColumnDataValues(data.data.columns));
data.axis.y.tick.values = [0, max];
}

Expand All @@ -1681,7 +1681,7 @@ function chartData(type, data, data2) {
return data.miq.category_table[x];
};
}
if(data.miq.zoomed){
if (data.miq.zoomed){
data.size = { height: $('#lightbox-panel').height() - 200 };
data.data.names = data.miq.name_table;
data.legend = { position : 'bottom'};
Expand All @@ -1695,18 +1695,45 @@ function chartData(type, data, data2) {
_.isObject(data.axis.y.tick) &&
_.isObject(data.axis.y.tick.format) &&
data.axis.y.tick.format.function) {

var format = data.axis.y.tick.format;
var max = _.max(_.flatten(_.tail(data.data.columns).map(_.tail)));
var min = _.min(_.flatten(_.tail(data.data.columns).map(_.tail)));
max = ManageIQ.charts.formatters[format.function].c3(format.options)(max).split(/[^0-9\,\.]/)[0];
min = ManageIQ.charts.formatters[format.function].c3(format.options)(min).split(/[^0-9\,\.]/)[0];
console.log(min);
console.log(max);
if(max - min < Math.pow(10, 1 - format.options.precision)){
format.options.precision += 1;
console.log(format);
max = _.max(getChartColumnDataValues(data.data.columns));
var min = _.min(getChartColumnDataValues(data.data.columns));
var max_showed = getChartFormatedValue(format, max);
var min_showed = getChartFormatedValue(format, min);
var change_format = true;
// if there are no valid values or there is only single values big enough, then not change formating function
// TODO MOVE IT TO FUNCTION
if (max <= min || max_showed <= min_showed) {
if (max < min || max > 10) {
change_format = false;
}
else if (max > 0){
min = 0;
}
}

if (change_format) {
// if min and max are close, labels should be more precise
if (max_showed - min_showed <= Math.pow(10, 1 - format.options.precision)) {
while (((max_showed - min_showed ) * Math.pow(10, format.options.precision)) < 9.9) {
format.options.precision += 1;
}
}
// if min and max are not close, labels should be less precise
else if ((max_showed - min_showed) >= Math.pow(10, 2 - format.options.precision)) {
while (((max_showed - min_showed ) * Math.pow(10, format.options.precision)) > 99) {
if (format.options.precision < 1) {
break;
}
format.options.precision -= 1;
}
}
}
data.axis.y.tick.format = ManageIQ.charts.formatters[format.function].c3(format.options);
data.legend.item = {
onclick: recalculateChartYLabels
}

var title_format = _.cloneDeep(format);
title_format.options.precision += 1;
Expand Down
77 changes: 77 additions & 0 deletions app/assets/javascripts/miq_c3.js
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,83 @@ function load_c3_chart(data, chart_id, height) {
ManageIQ.charts.c3[chart_id] = chart;
};


function recalculateChartYLabels (id) {
this.api.toggle(id);
var format = ManageIQ.charts.chartData.candu[this.config.bindto.split('_').pop()].xml.miq.format;
var minMax = getMinMaxFromChart(this);
// if there are no valid values or there is only single values big enough, then return
if (!minMax || minMax[0] === minMax[1]) {
if (!minMax || minMax[1] >= 10) {
return;
}
else if (minMax[1] > 0){
minMax[0] = 0;
}
}

var tmp = getChartFormatedValueWithFormat(format, minMax[0]);
var tmp2 = getChartFormatedValueWithFormat(format, minMax[1]);
var min_showed = tmp[0];
var max_showed = tmp2[0];
var min_units = tmp[1];
var max_units = tmp2[1];
if (min_units !== max_units){
return;
}
// if min and max are close, labels should be more precise
if (max_showed - min_showed <= Math.pow(10, 1 - format.options.precision)) {
while (((max_showed - min_showed ) * Math.pow(10, format.options.precision)) < 9.9) {
format.options.precision += 1;
min_showed = getChartFormatedValue(format, minMax[0]);
max_showed = getChartFormatedValue(format, minMax[1]);
}
this.config.axis_y_tick_format = ManageIQ.charts.formatters[format.function].c3(format.options);
this.api.flush();
}
// if min and max are not, labels should be less precise
else if ((max_showed - min_showed) >= Math.pow(10, 2 - format.options.precision)) {
while (((max_showed - min_showed ) * Math.pow(10, format.options.precision)) > 99) {
if(format.options.precision < 1){
break;
}
format.options.precision -= 1;
}
this.config.axis_y_tick_format = ManageIQ.charts.formatters[format.function].c3(format.options);
this.api.flush();
}
}

function getMinMaxFromChart(chart) {
var data = [];
_.forEach(chart.api.data.shown(), function(o) {
_.forEach(o.values, function(elem) {
data.push(elem.value);
});
});

var max = _.max(_.filter(data, function(o) { return o !== null; }));
var min = _.min(_.filter(data, function(o) { return o !== null; }));
if (max === -Infinity || min === Infinity) {
return false;
}
return [min, max];
}

function getChartColumnDataValues(columns) {
return _.filter(_.flatten(_.tail(columns).map(_.tail)), function(o) { return o !== null; })
}

function getChartFormatedValue(format, value) {
return numeral(ManageIQ.charts.formatters[format.function].c3(format.options)(value).split(/[^0-9\,\.]/)[0]).value();
}

function getChartFormatedValueWithFormat(format, value) {
var tmp = /^([0-9\,\.]+)(.*)/.exec(ManageIQ.charts.formatters[format.function].c3(format.options)(value));
return [numeral(tmp[1]).value(), tmp[2]];
}


c3.chart.internal.fn.categoryName = function (i) {
var config = this.config, categoryIndex = Math.ceil(i);
return i < config.axis_x_categories.length ? config.axis_x_categories[categoryIndex] : i;
Expand Down
2 changes: 1 addition & 1 deletion app/controllers/application_controller/performance.rb
Original file line number Diff line number Diff line change
Expand Up @@ -859,7 +859,7 @@ def prepare_perf_tag_chart(chart, rpt, cat_desc)
# Grab the first (and should be only) chart column
col = chart[:columns].first
# Create the new chart columns for each tag
chart[:columns] = rpt.extras[:group_by_tags].collect { |t| col + "_" + t }
chart[:columns] ||= rpt.extras[:group_by_tags].collect { |t| col + "_" + t }
end

def gen_perf_chart(chart, rpt, idx, zoom_action)
Expand Down
6 changes: 4 additions & 2 deletions lib/report_formatter/c3.rb
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,8 @@ def build_document_header
:data => {:columns => [], :names => {}},
:axis => {:x => {:tick => {}}, :y => {:tick => {}}},
:tooltip => {:format => {}},
:miq => {:name_table => {}, :category_table => {}}
:miq => {:name_table => {}, :category_table => {}},
:legend => {}
}

if chart_is_2d?
Expand Down Expand Up @@ -95,6 +96,7 @@ def build_document_header

axis_formatter = {:function => format, :options => options}
mri.chart[:axis][:y] = {:tick => {:format => axis_formatter}}
mri.chart[:miq][:format] = axis_formatter
end
end

Expand Down Expand Up @@ -142,7 +144,7 @@ def change_structure_to_timeseries
# set x axis type to timeseries and remove categories
mri.chart[:axis][:x] = {:type => 'timeseries', :tick => {}}
# set flag for performance chart
mri.chart[:miq] = {:performance_chart => true}
mri.chart[:miq][:performance_chart] = true
# this conditions are taken from build_performance_chart_area method from chart_commons.rb
if mri.db.include?("Daily") || (mri.where_clause && mri.where_clause.include?("daily"))
# set format for parsing
Expand Down

0 comments on commit 2a07611

Please sign in to comment.