module Repor
  module Serializers
    class BaseSerializer
      include ActionView::Helpers::TextHelper

      attr_reader :report

      def initialize(report)
        @report = report
      end

      # Consider overriding many of these methods to use I18n with keys based
      # on the aggregator or dimension name.

      def human_aggregator_label(aggregator)
        aggregator.name.to_s.humanize
      end

      def human_dimension_label(dimension)
        dimension.name.to_s.humanize
      end

      def human_null_value_label(dimension)
        "No #{human_dimension_label(dimension)}"
      end

      def human_aggregator_value_label(aggregator, value)
        value
      end

      def human_dimension_value_label(dimension, value)
        return human_null_value_label(dimension) if value.nil?

        case dimension
        when Repor::Dimensions::CategoryDimension
          human_category_value_label(dimension, value)
        when Repor::Dimensions::NumberDimension
          human_number_value_label(dimension, value)
        when Repor::Dimensions::TimeDimension
          human_time_value_label(dimension, value)
        else
          value
        end
      end

      def human_category_value_label(dimension, value)
        value
      end

      def human_number_value_label(dimension, value)
        begin
          min, max = value.values_at(:min, :max)
        rescue
          min, max = value.min, value.max
        end
        if min && max
          "[#{min.round(2)}, #{max.round(2)})"
        elsif min
          ">= #{min.round(2)}"
        elsif max
          "< #{max.round(2)}"
        else
          human_null_value_label(dimension)
        end
      end

      def time_formats
        {
          minutes: '%F %k:%M', hours: '%F %k', days: '%F',
          weeks: 'week of %F', months: '%Y-%m', years: '%Y'
        }
      end

      def human_time_value_label(dimension, value)
        min, max = value.min, value.max
        if min && max
          time_formats.each do |step, format|
            return min.strftime(format) if max == min.advance(step => 1)
          end
          "#{min} to #{max}"
        elsif min
          "after #{min}"
        elsif max
          "before #{max}"
        else
          human_null_value_label(dimension)
        end
      end

      def record_type
        report.table_name.singularize.humanize
      end

      def axis_summary
        y = human_aggregator_label(report.aggregator)
        xes = report.groupers.map(&method(:human_dimension_label))
        count = pluralize(report.records.count, record_type)
        "#{y} by #{xes.to_sentence} for #{count}"
      end

      def filter_summary
        report.filters.flat_map do |dimension|
          human_dimension_label(dimension) + " = " + dimension.filter_values.map do |value|
            human_dimension_value_label(dimension, value)
          end.to_sentence(last_word_connector: ', or ')
        end.join('; ')
      end
    end
  end
end