Aggregate an array of Merchant store hours that overlap to produce a single range

74 Views Asked by At

I have a list of store hours from a merchant that could potentially overlap stored in AvailableHours model. The model has a start_time and an end_time stored as a TimeOfDay object (https://github.com/JackC/tod).

I want to iterate through the list and compare the store hours so that I can just display 1 store hour. For example, if a store could had hours of Mon: 9am-2pm and Mon: 10am-5pm, I would want to display Mon: 9am-5pm. A store could also have two times in a single day, i.e. Fri: 9-5pm, 7pm-10pm. I have to do this for each day of the week.

The issue I am running into is that my code is becoming hard to manage - I started introducing a lot of nested if/else for manual checking (what a nightmare!), but my question is this:

Is there a way to use ActiveRecord to produce a hash with merged time ranges?

def ranges_overlap?(shift, current_shift)
  shift.include?(current_shift.beginning) || current_shift.include?(shift.beginning)
end

def merge_ranges(shift, current_shift)
  new_open_time = [shift.beginning, current_shift.beginning].min
  new_close_time = [shift.ending, current_shift.ending].max
  Shift.new(new_open_time, new_close_time)
end


def get_aggregate_merchant_hours 
  merchant = self
  day = nil
  shifts_array = []
  output_shift_array = []

  merchant_shifts = merchant_shifts = merchant.available_hours.order(day: :asc, open_time: :desc).map do |ah|
    merged_shift = nil
    show_output = false

    # if it's a new day
    if day.blank? || day != ah.day 
      output_shift_array = shifts_array
      shifts_array =  []        
      show_output = true if !day.blank?
      output_day = day
      day = ah.day
    end

    next if ah.open_time.blank? || ah.close_time.blank?
    open_time = ah.open_time
    close_time = ah.close_time

    current_shift = Shift.new(open_time, close_time)

    if !shifts_array.blank?
      #compare and merge
      shifts_array.each do |shift|
        merged_shift = merge_ranges(shift, current_shift) if ranges_overlap?(shift, current_shift)
      end
    end

    #replace old shift with merged shift
    if merged_shift

      delete_shift = shifts_array.find(beginning: current_shift.beginning, ending: current_shift.ending).first
      shifts_array.delete(delete_shift) if delete_shift

      shifts_array.push(merged_shift)
    else
      shifts_array.push(current_shift)
    end

    if show_output
      store_hours_string = ""
      if output_shift_array.blank?
        store_hours_string = "Closed"
      else
        output_shift_array.each do |shift|
          shift.beginning.strftime("%I:%M%p")
          shift.ending.strftime("%I:%M%p")
          if store_hours_string.blank?
            store_hours_string << "#{shift.beginning.strftime("%I:%M%p").downcase} - #{shift.ending.strftime("%I:%M%p").downcase}"
          else
            store_hours_string << ", #{shift.beginning.strftime("%I:%M%p").downcase} - #{shift.ending.strftime("%I:%M%p").downcase}"
          end
        end
      end
    "#{Date::DAYNAMES[output_day][0..2]}: #{store_hours_string}"

    end

  end

  #output last shift
  store_hours_string = ""
  if output_shift_array.blank?
    store_hours_string = "Closed"
  else
    output_shift_array.each do |shift|
      shift.beginning.strftime("%I:%M%p")
      shift.ending.strftime("%I:%M%p")
      if store_hours_string.blank?
        store_hours_string << "#{shift.beginning.strftime("%I:%M%p").downcase} - #{shift.ending.strftime("%I:%M%p").downcase}"
      else
        store_hours_string << ", #{shift.beginning.strftime("%I:%M%p").downcase} - #{shift.ending.strftime("%I:%M%p").downcase}"
      end
    end
  end
  merchant_shifts << "#{Date::DAYNAMES[day][0..2]}: #{store_hours_string}"


  merchant_shifts.compact

end
0

There are 0 best solutions below