require "rails" module SimpleCalendar class Calendar PARAM_KEY_BLACKLIST = :authenticity_token, :commit, :utf8, :_method, :script_name attr_accessor :view_context, :options def initialize(view_context, options = {}, &block) @view_context = view_context @locals = options.delete(:locals) || {} @options = options @block = block # Next and previous view links should use the same params as the current view @params = @view_context.respond_to?(:params) ? @view_context.params : {} @params = @params.to_unsafe_h if @params.respond_to?(:to_unsafe_h) @params = @params.with_indifferent_access.except(*PARAM_KEY_BLACKLIST) # Add in any additional params the user passed in @params.merge!(@options.fetch(:params, {})) end def render_in(view_context, &block) @block = block view_context.render( partial: partial_name, locals: locals ) end def locals(&block) @locals.merge( passed_block: @block, calendar: self, date_range: date_range, start_date: start_date, sorted_events: sorted_events ) end def td_classes_for(day) today = Date.current td_class = ["day"] td_class << "wday-#{day.wday}" td_class << "today" if today == day td_class << "past" if today > day td_class << "future" if today < day td_class << "start-date" if day.to_date == start_date.to_date td_class << "prev-month" if start_date.month != day.month && day < start_date td_class << "next-month" if start_date.month != day.month && day > start_date td_class << "current-month" if start_date.month == day.month td_class << "has-events" if sorted_events.fetch(day, []).any? td_class end def tr_classes_for(week) today = Date.current tr_class = ["week"] tr_class << "current-week" if week.include?(today) tr_class end def url_for_next_view view_context.url_for(@params.merge(start_date_param => (date_range.last + 1.day).iso8601)) end def url_for_previous_view view_context.url_for(@params.merge(start_date_param => (date_range.first - date_range.count.days).iso8601)) end def url_for_today_view view_context.url_for(@params.merge(start_date_param => Time.current.to_date.iso8601)) end def date_range (start_date..(start_date + additional_days.days)).to_a end def partial_name @options[:partial] || self.class.name.underscore end def attribute options.fetch(:attribute, :start_time).to_sym end def end_attribute options.fetch(:end_attribute, :end_time).to_sym end def start_date_param options.fetch(:start_date_param, :start_date).to_sym end def sorted_events @sorted_events ||= begin events = Array.wrap(options[:events]).reject { |e| e.send(attribute).nil? }.sort_by(&attribute) group_events_by_date(events) end end def sorted_events_for(day) Array.wrap(sorted_events[day]) end def group_events_by_date(events) events_grouped_by_date = Hash.new { |h, k| h[k] = [] } events.each do |event| event_start_date = event.send(attribute).to_date event_end_date = (event.respond_to?(end_attribute) && !event.send(end_attribute).nil?) ? event.send(end_attribute).to_date : event_start_date (event_start_date..event_end_date.to_date).each do |enumerated_date| events_grouped_by_date[enumerated_date] << event end end events_grouped_by_date end def start_date if options.has_key?(:start_date) options.fetch(:start_date).to_date else view_context.params.fetch(start_date_param, Date.current).to_date end end def end_date date_range.last end def additional_days options.fetch(:number_of_days, 4) - 1 end end end