Calculating Business Hours Left

P

Phrogz

current_time = Time.new
due_time = # ... some Time instance in the future
work_hours_left_before_I_am_screwed = # ... calculate

I'm writing a little task-tracking app to help me put out the most
urgent fires. As above, given any two Time instances, I'd like to know
how many hours are left to perform the task. This should skip weekends
and hours outside of specified start- and end-hours for a work day.

Anyone got such code handy? Or feel like writing it for me? :)

Right now, my best idea involves looping over the range one day at a
time and manually accumulating the available hours. It feels
inelegant, but it's all I have right now.
 
K

Keith Fahlgren

current_time = Time.new
due_time = # ... some Time instance in the future
work_hours_left_before_I_am_screwed = # ... calculate

I'm writing a little task-tracking app to help me put out the most
urgent fires. As above, given any two Time instances, I'd like to know
how many hours are left to perform the task. This should skip weekends
and hours outside of specified start- and end-hours for a work day.

Anyone got such code handy? Or feel like writing it for me? :)

Hmm... no idea if this works, but it might be a start:

#!/usr/bin/env ruby
class WorkHours
SECONDS_IN_HOUR = 3600
HOURS_I_WORK = (7..16)
DAYS_I_WORK = (1..5)
def self.diff_times(a, b)
hours = []
(a.to_i..b.to_i).step(SECONDS_IN_HOUR) {|t|
time = Time.at(t)
hours << t if DAYS_I_WORK.include?(time.wday) &&
HOURS_I_WORK.include?(time.hour)
}
return hours.length
end
end
nil

now = Time.now
=> Fri Apr 06 07:12:08 PDT 2007
twenty_days_from_now = now + 1728000
=> Thu Apr 26 07:12:08 PDT 2007
one_day_from_now = now + 86400
=> Sat Apr 07 07:12:08 PDT 2007
WorkHours.diff_times(now, twenty_days_from_now)
=> 141
WorkHours.diff_times(now, one_day_from_now)
=> 10


HTH,
Keith
 
P

Phrogz

Hmm... no idea if this works, but it might be a start:

That feels like it might be good, thanks! I'll have to investigate
more. It certainly looks cleaner than what I came up with:

class Time
def at( hrs, mins=0 )
self.class.parse strftime("%Y-%b-%d #{hrs}:#{mins}" )
end
end

# Fails for tasks one year or farther in the future
def work_hours_until( due_time, work_starts_at=9, work_ends_at=17 )
start_time = [
Time.now,
Time.parse( "#{work_starts_at}:00" )
].max
end_time = [
due_time,
due_time.at( work_ends_at )
].min

weekdays = 1..5
one_hour = 3600.0
one_day = one_hour * 24

if start_time.yday == end_time.yday
if weekdays.include?( start_time.wday )
( end_time - start_time ) / one_hour
else
0
end
else
business_hours = work_ends_at - work_starts_at

# Handle partial hours on the first day
hours = ( start_time.at( work_ends_at ) - start_time ) / one_hour

t = start_time + one_day
until t.yday == due_time.yday
hours += business_hours if weekdays.include?( t.wday )
t += one_day
end

# Handle partial hours on the last day
if weekdays.include?( t.wday )
hours += ( end_time - end_time.at( work_starts_at ) ) / one_hour
end

hours
end

end
 

Ask a Question

Want to reply to this thread or ask your own question?

You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.

Ask a Question

Members online

No members online now.

Forum statistics

Threads
473,755
Messages
2,569,537
Members
45,022
Latest member
MaybelleMa

Latest Threads

Top