require 'date' require 'active_support/version' %w{ active_support/core_ext/array/extract_options active_support/core_ext/string/conversions active_support/core_ext/date/conversions active_support/core_ext/integer/time active_support/core_ext/numeric/time }.each do |active_support_3_requirement| require active_support_3_requirement end if ActiveSupport::VERSION::MAJOR == 3 # Encapsulates a timeframe between two dates. The dates provided to the class are always until the last date. That means # that the last date is excluded. # # # from 2007-10-01 00:00:00.000 to 2007-10-31 23:59:59.999 # Timeframe.new(Date(2007,10,1), Date(2007,11,1)) # # and holds 31 days # Timeframe.new(Date(2007,10,1), Date(2007,11,1)).days #=> 31 class Timeframe attr_accessor :from, :to # Creates a new instance of Timeframe. You can either pass a start and end Date or a Hash with named arguments, # with the following options: # # :month: Start date becomes the first day of this month, and the end date becomes the first day of # the next month. If no :year is specified, the current year is used. # :year: Start date becomes the first day of this year, and the end date becomes the first day of the # next year. # # By default, Timeframe.new will die if the resulting Timeframe would cross year boundaries. This can be overridden # by setting the :skip_year_boundary_crossing_check option. # # Examples: # # Timeframe.new Date.new(2007, 2, 1), Date.new(2007, 4, 1) # February and March # Timeframe.new :year => 2004 # The year 2004 # Timeframe.new :month => 4 # April # Timeframe.new :year => 2004, :month => 2 # Feburary 2004 def initialize(*args) options = args.extract_options! if month = options[:month] month = Date.parse(month).month if month.is_a? String year = options[:year] || Date.today.year from = Date.new(year, month, 1) to = from.next_month elsif year = options[:year] from = Date.new(year, 1, 1) to = Date.new(year+1, 1, 1) end from = args.shift.to_date if from.nil? and args.any? to = args.shift.to_date if to.nil? and args.any? raise ArgumentError, "Please supply a start and end date, `#{args.map(&:inspect).to_sentence}' is not enough" if from.nil? or to.nil? raise ArgumentError, "Start date #{from} should be earlier than end date #{to}" if from > to raise ArgumentError, 'Timeframes that cross year boundaries are dangerous' unless options[:skip_year_boundary_crossing_check] or from.year == to.yesterday.year or from == to @from, @to = from, to end def inspect # :nodoc: "" end # The number of days in the timeframe # # Timeframe.new(Date.new(2007, 11, 1), Date.new(2007, 12, 1)).days #=> 30 # Timeframe.new(:month => 1).days #=> 31 # Timeframe.new(:year => 2004).days #=> 366 def days (to - from).to_i end # Returns a string representation of the timeframe def to_s if (from.year == to.year - 1) and [from.day, from.month, to.day, to.month].uniq == [1] from.year.to_s elsif from.year == to.year and from.day == 1 and to.day == 1 and to.month - from.month == 1 "#{Date::MONTHNAMES[from.month]} #{from.year}" else "the period from #{from.strftime('%d %B')} to #{to.yesterday.strftime('%d %B %Y')}" end end # Returns true when a Date or other Timeframe is included in this Timeframe def include?(obj) # puts "checking to see if #{date} is between #{from} and #{to}" if Emitter::DEBUG case obj when Date (from...to).include?(obj) when Time # (from...to).include?(obj.to_date) raise "this wasn't previously supported, but it could be" when Timeframe from <= obj.from and to >= obj.to end end # Returns true when the parameter Timeframe is properly included in the Timeframe def proper_include?(other_timeframe) raise ArgumentError, 'Proper inclusion only makes sense when testing other Timeframes' unless other_timeframe.is_a? Timeframe (from < other_timeframe.from) and (to > other_timeframe.to) end # Returns true when this timeframe is equal to the other timeframe def ==(other) # puts "checking to see if #{self} is equal to #{other}" if Emitter::DEBUG return false unless other.is_a?(Timeframe) from == other.from and to == other.to end alias :eql? :== # Calculates a hash value for the Timeframe, used for equality checking and Hash lookups. #-- # This needs to be an integer or else it won't use #eql? def hash from.hash + to.hash end # Returns an array of month-long subtimeframes #-- # TODO: rename to month_subtimeframes def months raise ArgumentError, "Please only provide whole-month timeframes to Timeframe#months" unless from.day == 1 and to.day == 1 raise ArgumentError, 'Timeframes that cross year boundaries are dangerous during Timeframe#months' unless from.year == to.yesterday.year year = from.year # therefore this only works in the from year (from.month..to.yesterday.month).map { |m| Timeframe.new :month => m, :year => year } end # Returns the relevant year as a Timeframe def year raise ArgumentError, 'Timeframes that cross year boundaries are dangerous during Timeframe#year' unless from.year == to.yesterday.year Timeframe.new :year => from.year end # Divides a Timeframe into component parts, each no more than a month long. #-- # multiyear safe def month_subtimeframes (from.year..to.yesterday.year).map do |year| (1..12).map do |month| Timeframe.new(:year => year, :month => month) & self end end.flatten.compact end # Like #month_subtimeframes, but will discard partial months # multiyear safe def full_month_subtimeframes month_subtimeframes.map { |st| Timeframe.new(:year => st.from.year, :month => st.from.month) } end # Divides a Timeframe into component parts, each no more than a year long. #-- # multiyear safe def year_subtimeframes (from.year..to.yesterday.year).map do |year| Timeframe.new(:year => year) & self end end # Like #year_subtimeframes, but will discard partial years #-- # multiyear safe def full_year_subtimeframes (from.year..to.yesterday.year).map do |year| Timeframe.new :year => year end end # Crop a Timeframe to end no later than the provided date. #-- # multiyear safe def ending_no_later_than(date) if to < date self elsif from >= date nil else Timeframe.multiyear from, date end end # Returns a timeframe representing the intersection of the given timeframes def &(other_timeframe) this_timeframe = self if other_timeframe == this_timeframe this_timeframe elsif this_timeframe.from > other_timeframe.from and this_timeframe.to < other_timeframe.to this_timeframe elsif other_timeframe.from > this_timeframe.from and other_timeframe.to < this_timeframe.to other_timeframe elsif this_timeframe.from >= other_timeframe.to or this_timeframe.to <= other_timeframe.from nil else Timeframe.new [this_timeframe.from, other_timeframe.from].max, [this_timeframe.to, other_timeframe.to].min, :skip_year_boundary_crossing_check => true end end # Returns the fraction (as a Float) of another Timeframe that this Timeframe represents def /(other_timeframe) raise ArgumentError, 'You can only divide a Timeframe by another Timeframe' unless other_timeframe.is_a? Timeframe self.days.to_f / other_timeframe.days.to_f end # Crop a Timeframe by another Timeframe def crop(container) raise ArgumentError, 'You can only crop a timeframe by another timeframe' unless container.is_a? Timeframe self.class.new [from, container.from].max, [to, container.to].min end # Returns an array of Timeframes representing the gaps left in the Timeframe after removing all given Timeframes def gaps_left_by(*timeframes) # remove extraneous timeframes timeframes.reject! { |t| t.to <= from } timeframes.reject! { |t| t.from >= to } # crop timeframes timeframes.map! { |t| t.crop self } # remove proper subtimeframes timeframes.reject! { |t| timeframes.detect { |u| u.proper_include? t } } # escape return [self] if timeframes.empty? timeframes.sort! { |x, y| x.from <=> y.from } a = [ from ] + timeframes.collect(&:to) b = timeframes.collect(&:from) + [ to ] a.zip(b).map do |gap| Timeframe.new(*gap) if gap[1] > gap[0] end.compact end # Returns true if the union of the given Timeframes includes the Timeframe def covered_by?(*timeframes) gaps_left_by(*timeframes).empty? end # Returns the same Timeframe, only a year earlier def last_year self.class.new((from - 1.year), (to - 1.year)) end # Just a string that can be processed by Timeframe.interval... identical to #to_param def to_json(*) to_param end # http://jonathanjulian.com/2010/04/rails-to_json-or-as_json/ # Not really interested in making a Hash representation. def as_json(*) to_param end # URL-friendly like "2008-10-25/2009-11-12" def to_param "#{from}/#{to}" end class << self def make_dates(from, to) # :nodoc: return from.to_date, to.to_date end # Shortcut method to return the Timeframe representing the current year (as defined by Time.now) def this_year new :year => Time.now.year end # Construct a new Timeframe, but constrain it by another def constrained_new(from, to, constraint) from, to = make_dates from, to raise ArgumentError, 'Constraint must be a Timeframe' unless constraint.is_a? Timeframe raise ArgumentError, "Start date #{from} should be earlier than end date #{to}" if from > to if to <= constraint.from or from >= constraint.to new constraint.from, constraint.from elsif from.year == to.yesterday.year new(from, to) & constraint elsif from.year < constraint.from.year and constraint.from.year < to.yesterday.year constraint else new [constraint.from, from].max, [constraint.to, to].min end end # Shortcut for #new that automatically skips year boundary crossing checks def multiyear(from, to) from, to = make_dates from, to new from, to, :skip_year_boundary_crossing_check => true end # Create a multiyear timeframe +/- number of years around today def mid(number) from = Time.zone.today - number.years to = Time.zone.today + number.years multiyear from, to end # Construct a new Timeframe by parsing an ISO 8601 time interval string # http://en.wikipedia.org/wiki/ISO_8601#Time_intervals def interval(str) raise ArgumentError, 'Intervals should be specified as a string' unless str.is_a? String raise ArgumentError, 'Intervals should be specified according to ISO 8601, method 1, eliding times' unless str =~ /^\d\d\d\d-\d\d-\d\d\/\d\d\d\d-\d\d-\d\d$/ multiyear *str.split('/') end alias :from_json :interval end end