# -*- coding: utf-8 -*- class RailsDataExplorer class Chart # Responsibilities: # * Render a stacked bar chart for bivariate analysis of a temporal and a # categorical data series. # # Collaborators: # * DataSet # class StackedHistogramTemporal < Chart def initialize(_data_set, options = {}) @data_set = _data_set @options = {}.merge(options) end def compute_chart_attrs x_candidates = @data_set.data_series.find_all { |ds| (ds.chart_roles[Chart::StackedHistogramTemporal] & [:x, :any]).any? }.sort { |a,b| b.uniq_vals.length <=> a.uniq_vals.length } y_candidates = @data_set.data_series.find_all { |ds| (ds.chart_roles[Chart::StackedHistogramTemporal] & [:y, :any]).any? } x_ds = x_candidates.first y_ds = (y_candidates - [x_ds]).first return false if x_ds.nil? || y_ds.nil? # initialize data_matrix data_matrix = { _sum: { _sum: 0 } } x_ds.uniq_vals.each { |x_val| data_matrix[x_val] = {} data_matrix[x_val][:_sum] = 0 y_ds.uniq_vals.each { |y_val| data_matrix[x_val][y_val] = 0 data_matrix[:_sum][y_val] = 0 } } # populate data_matrix x_ds.values.length.times { |idx| x_val = x_ds.values[idx] y_val = y_ds.values[idx] data_matrix[x_val][y_val] += 1 data_matrix[:_sum][y_val] += 1 data_matrix[x_val][:_sum] += 1 data_matrix[:_sum][:_sum] += 1 } x_sorted_keys = x_ds.uniq_vals.sort( &x_ds.label_sorter( nil, lambda { |a,b| data_matrix[b][:_sum] <=> data_matrix[a][:_sum] } ) ) y_sorted_keys = y_ds.uniq_vals.sort( &y_ds.label_sorter( nil, lambda { |a,b| data_matrix[:_sum][b] <=> data_matrix[:_sum][a] } ) ) values = case @data_set.dimensions_count when 2 y_sorted_keys.map { |y_val| x_sorted_keys.map { |x_val| { x: x_val, y: data_matrix[x_val][y_val], c: y_val } } }.flatten else raise(ArgumentError.new("Exactly two data series required for contingency table.")) end { values: values, x_axis_label: x_ds.name, x_axis_tick_format: 'function(d) { return d }', y_axis_label: "#{ y_ds.name } distribution [%]", y_axis_tick_format: "d3.format('.1%')", } end def render return '' unless render? ca = compute_chart_attrs return '' unless ca render_vega(ca) end def render_vega(ca) %(