# (c) 2003-2008 Vlado Keselj http://www.cs.dal.ca/~vlado # # $Id: Schedule.pm,v 1.26 2008/09/03 11:45:33 vlado Exp $ # package Calendar::Schedule; use strict; require Exporter; use POSIX; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK %EXPORT_TAGS); # Exporter vars our @ISA = qw(Exporter); our %EXPORT_TAGS = ( 'all' => [ qw( parse_time ) ] ); our @EXPORT_OK = ( @{ $EXPORT_TAGS{'all'} } ); our @EXPORT = qw(new); #{version}';"!> #+ our $VERSION = '1.01'; #- use vars qw($Version $Revision); $Version = $VERSION; ($Revision = substr(q$Revision: 1.26 $, 10)) =~ s/\s+$//; # non-exported package globals use vars qw( $REweekday3 $REmonth3 ); $REweekday3 = qr/Mon|Tue|Wed|Thu|Fri|Sat|Sun/; $REmonth3 = qr/Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec/; =head1 NAME Calendar::Schedule - for managing calendar schedules =head1 SYNOPSIS use Calendar::Schedule qw/:all/; my $TTable = Calendar::Schedule->new(); # manually adding an entry $TTable->add_entry('2003-09-09 Tue 18-20 Some meeting'); # reading entries from a file $TTable->add_entries_from("$ENV{'HOME'}/.calendar"); # producing entries in HTML tables $TTable->set_first_week('now'); print "

\n" . $TTable->generate_table(); print "

\n" . $TTable->generate_table(); print "

\n" . $TTable->generate_table(); # etc. See EXAMPLES section The file .calendar may look like this: # comments can start with # * lines starting with * are treated as general todo entries ... # empty lines are fine to: Mon 9:00-10:00 this is a weekly entry Mon 13-14 a biweekly entry :biweekly :start Mar 8, 2004 Mon,Wed,Fri 15:30-16:30 several-days-a-week entry Wed :biweekly garbage collection 2004-03-06 Sat 14-16 fixed entry. The week day is redundant, but may\ help to detect errors (error will be reported if a wrong\ weekday is entered). BTW, an entry can go for several lines as\ long as there is a backslash at the end of each line. May 6 birthday (yearly entry) =head1 DESCRIPTION Description ... Attempted to match the internal data representation with the iCalendar standard (RFC2445). Examples of the iCalendar fields: DTSTART, DTEND, SUMMARY, RRULE (e.g. RRULE:FREQ=WEEKLY, RRULE:FREQ=WEEKLY;INTERVAL=2 for biweekly, RRULE:FREQ=WEEKLY;UNTIL=20040408 ) etc. =head1 EXAMPLES First example: use Calendar::Schedule qw/:all/; my $TTable = Calendar::Schedule->new(); # manually adding an entry $TTable->add_entry('2003-09-09 Tue 18-20 Some meeting'); # reading entries from a file $TTable->add_entries_from("$ENV{'HOME'}/.calendar"); # producing entries in HTML tables $TTable->set_first_week('2003-12-15'); print "

\n" . $TTable->generate_table(); print "

\n" . $TTable->generate_table(); print "

\n" . $TTable->generate_table(); Example with generating a weekly schedule (example2): use Calendar::Schedule; $TTable = Calendar::Schedule->new(); $TTable->{'ColLabel'} = "%A"; $TTable->add_entries(<\n" . $TTable->generate_table(); This will produce the following HTML code (if run before Apr 8, 2005): =for html

  Monday Tuesday Wednesday Thursday Friday Saturday Sunday
08:00              
10:00   Teaching (ECMM 6014)   Teaching (ECMM 6014)      
11:30              
12:00              
13:30     DNLP        
14:30         MALNIS    
15:30 Teaching (CSCI 3136)   Teaching (CSCI 3136)   Teaching (CSCI 3136)    
16:00   WIFL    
16:30            
17:00              
=head2 Conflicts Time conflicts are handled by producing several columns in a table for the same day. For example, the following code (example3): use Calendar::Schedule; $TTable = Calendar::Schedule->new(); $TTable->{'ColLabel'} = "%A"; $TTable->add_entries(<\n" . $TTable->generate_table(); will produce the following table (if run before Apr 8, 2005): =for html

  Monday Tuesday Wednesday Thursday Friday Saturday Sunday
08:00                  
10:00   Teaching (ECMM 6014)       Teaching (ECMM 6014)      
11:30                  
12:00                  
13:30     DNLP            
14:30             MALNIS    
15:00       meeting        
15:30 Teaching (CSCI 3136)   Teaching (CSCI 3136) another meeting   Teaching (CSCI 3136)    
16:00     WIFL    
16:30              
17:00                
18:00                  
=head1 STATE VARIABLES =over 4 =item StartTime Start time for various uses. Usually it is the the beginning of the first interesting week. =back =head1 METHODS =over 4 =item new() Creates a new C object and returns it. =cut sub new { my $proto = shift; my $class = ref($proto) || $proto; my $self = { VEvents => [ ], Entries => [ ], Entries1 => [ ], DayEntries => [ ], ToDo => [ ], RowLabels => [ ], StartTime => 0, ColLabel => "%A
%Y-%m-%d", ShowDays => 'all', # 'workdays' }; bless($self, $class); $self->{'DefaultRowLabels'} = [ qw( 08:00 12:00 17:00 ) ]; $self->{'RowLabels'} = [ @{ $self->{'DefaultRowLabels'} } ]; $self->set_first_week(time); return $self; } =item set_first_week(time) sets start time at the last Monday before given date. see parse_time =cut sub set_first_week { my $self = shift; my $arg = shift; my $starttime = &parse_time($arg); $self->{'StartTime'} = $self->{'ContextTime'} = &find_week_start($starttime); } =item set_ColLabel(pattern) sets C pattern for column (day) labels. The default pattern is "C<%AEbrE%Y-%m-%d>", which produces labels like: Friday 2003-12-19 In order to have just a weekday name, use "C<%A>". =cut sub set_ColLabel { my $self = shift; my $arg = shift; $self->{'ColLabel'} = $arg; } sub find_week_start { my $starttime = shift; while ((localtime($starttime))[6] != 1) { $starttime -= 86400 } while ((localtime($starttime))[2] != 0) { $starttime -= 3600 } while ((localtime($starttime))[1] != 0) { $starttime -= 60 } while ((localtime($starttime))[0] != 0) { $starttime -- } return $starttime; } =item parse_time(time_specification[,prefix]) Parses time specification and returns the calendar time (see mktime in Perl). The functions dies if the time cannot be completely recognized. If prefix is set to true (1), then only a prefix of the string can be a time specification. If prefix is set to 1, then in an array context it will return a 2-element list: the calendar time and the remainder of the string. Format examples: 2004-03-17 now Mar 8, 2004 1-Jul-2005 =cut #mktime(sec,min,hour,mday,mon,year,wday=0,yday=0,isdst=0) #mon,wday,yday start with 0,wday starts with Sun,year starts with 1900 # usually set last 3 to -1 # ('YYYY-MM-DD') now sub parse_time { my $time = shift; my $prefix = shift; my $endrex = ( $prefix ? qr// : qr/\s*$/ ); my ($ret, $ret2); my $monrex = $REmonth3; if ($time =~ /^(\d\d\d\d)-(\d\d)-(\d\d) (\d\d?):(\d\d)$endrex/) { $ret = mktime(0,$5,$4,$3,$2-1,$1-1900,-1,-1,-1) } elsif ($time =~ /^(\d\d\d\d)-(\d\d)-(\d\d)$endrex/) { $ret = mktime(0,0,0,$3,$2-1,$1-1900,-1,-1,-1) } elsif ($time =~ /^(\d\d)-(\d\d)-(\d\d\d\d)$endrex/) { $ret = mktime(0,0,0,$1,$2-1,$3-1900,-1,-1,-1) } elsif ($time =~ /^(\d?\d)-($monrex)-(\d\d\d\d)\b$endrex/) { $ret = mktime(0,0,0,$1,&month_to_digits($2),$3-1900,-1,-1,-1) } elsif ($time =~ /^($monrex) (\d?\d), (\d\d\d\d)\b$endrex/) { $ret = mktime(0,0,0,$2,&month_to_digits($1),$3-1900,-1,-1,-1) } elsif ($time =~ /^\d+$endrex/) { $ret = $time } elsif ($time =~/^now\b$endrex/) { $ret = time } else { use Carp; confess "cannot parse time:($time)" } $ret2 = $'; return wantarray ? ($ret, $ret2) : $ret; } =item add_entries_from(file_name) Adds entries from a file. See method add_entries and add_entry for format explanation. =cut sub add_entries_from { my $self = shift; my $fname = shift; return $self->add_entries(scalar(_getfile($fname))); } =item add_entries(list_of_entries) Adds more entries. Each entry may contain several entries separated by a new-line, except if the line ends with \. Empty lines and lines that start with \s*# are ignored. See add_entry for further explanation of format. =cut sub add_entries { my $self = shift; while ($#_ > -1) { my $entries = shift; foreach my $en (split(/(?add_entry($en); } } } =item add_entry(list_of_entries) Adds more entries. It is different from add_entries because this method does not break entries on new-lines, although it does accept a list of entries as arguments. Examples: $TTable->add_entry('Mon 8-17', 'Labour Day'); $TTable->add_entry('2003-09-09 Tue 18-20 Some meeting'); More format examples: Wed 3-4:30pm meeting Mon,Wed,Fri 15:30-16:30 meeting (product team) Mon 13-14 seminar :biweekly :start Mar 8, 2004 Tue,Thu 10-11:30 Class (ECMM 6014) Location: MCCAIN ARTS&SS 2022 :until Apr 8, 2004 =cut sub add_entry { my $self = shift; if ($#_ <= 1) { # entry not structured, needs to be # parsed (string) my $timeslot = shift; my $description; if ($#_ == 0) { $description = shift } else { local $_ = $timeslot; #2003-09-09 Tue 18-20 if (/^\d\d\d\d-\d\d-\d\d $REweekday3 \d\d?(:\d\d)?-\d\d?(:\d\d)?([ap]m)? /) { $timeslot = $&; $description = $'; } elsif (/^\d\d\d\d-\d\d-\d\d \d\d?(:\d\d)?-\d\d?(:\d\d)?([ap]m)? /) { $timeslot = $&; $description = $'; } # elsif (/^$REweekday3(?:,$REweekday3)*\s+\d\d?(:\d\d)?-\d\d?(:\d\d)?([ap]m)? /) { $timeslot = $&; $description = $'; } #iso8601 thanks to Mike Vasiljevs elsif (/^(\d\d\d\d-\d\d-\d\d)T(\d\d:\d\d:\d\d)- (\d\d\d\d-\d\d-\d\d)T(\d\d:\d\d:\d\d)?/x) { $timeslot = $&; $description = $'; } elsif (/^(\d\d\d\d-\d\d-\d\d) / || /^(\d?\d-\w\w\w-\d\d\d\d) / ) { $timeslot = parse_time($1); $description = $'; push @{ $self->{'DayEntries'}}, { date => $timeslot, description => $description }; return; } elsif (/^\*\s*/) { push @{ $self->{'ToDo'}}, { desc=>$' }; return; } # elsif (/^($REweekday3)\b\s*/) { $timeslot=$1; $description=$'; } else { ($timeslot, $description) = parse_time($_, 1) } $timeslot =~ s/\s+$//; } my ($starttime, $endtime); if ($timeslot =~ /^($REweekday3(?:,$REweekday3)*)\s+(\d\d?(?::\d\d)?)-(\d\d?(?::\d\d)?)((?:[ap]m)?)$/) { my ($days,$stime,$etime,$ampm) = ($1, $2, $3, $4); $stime .= $ampm; $etime .= $ampm; my $rrule = 'FREQ=WEEKLY'; if ($description =~ /\s*:biweekly\b\s*/) { $description = "$` $'"; $rrule .= ':INTERVAL=2'; } if ($description =~ /\s*:until\s+/) { my $p1 = $`; my $p2 = $'; my ($t, $p2n) = parse_time($p2, 1); $description = "$p1 $p2n"; $rrule .= ";UNTIL=".$self->find_next_time("23:59", $t); } my $starttime = $self->{'StartTime'}; if ($description =~ /:start\s+/) { my $d1 = $`; my $d2 = $'; ($starttime, $d2) = parse_time($d2, 1); $description = "$d1$d2"; } foreach my $d (split(/,/, $days)) { my %vevent = (); $vevent{'RRULE'} = $rrule; $vevent{'DTSTART'} = $self->find_next_time("$d $stime", $starttime); $vevent{'DTEND'} = $self->find_next_time("$d $etime", $vevent{'DTSTART'}); $vevent{'SUMMARY'} = $description; push @{ $self->{'VEvents'} }, \%vevent; } return; } # thanks to Mike Vasiljevs: # 25 may 2006, adding matching for iso8601 dates # elsif ($timeslot =~ /^(\d\d\d\d-\d\d-\d\d)T(\d\d:\d\d:\d\d)- (\d\d\d\d-\d\d-\d\d)T(\d\d:\d\d:\d\d)$/x) { my ($hstart, $mstart, $sstart) = split(":", $2); my ($hend, $mend, $send) = split(":", $4); $starttime = parse_time("$1 $hstart:$mstart"); $endtime = parse_time("$1 $hend:$mend"); ##correct is to use second date in endtime, but it may lead to time leaks!? #$endtime = parse_time("$3 $hend$mend"); } elsif ($timeslot =~ /^($REweekday3(?:,$REweekday3)*)$/) { my ($days) = ($1); my $rrule = 'FREQ=WEEKLY'; if ($description =~ /\s*:biweekly\b\s*/) { $description = "$` $'"; $rrule .= ':INTERVAL=2'; } if ($description =~ /\s*:until\s+/) { my $p1 = $`; my $p2 = $'; my ($t, $p2n) = parse_time($p2, 1); $description = "$p1 $p2n"; $rrule .= ";UNTIL=".$self->find_next_time("23:59", $t); } my $starttime = $self->{'StartTime'}; if ($description =~ /:start\s+/) { my $d1 = $`; my $d2 = $'; ($starttime, $d2) = parse_time($d2, 1); $description = "$d1$d2"; } foreach my $d (split(/,/, $days)) { my %vevent = (); $vevent{'DTSTART'} = $self->find_next_time("$d 00:00", $starttime); # not DTEND signals DayEntry $vevent{'RRULE'} = $rrule; $vevent{'SUMMARY'} = $description; push @{ $self->{'VEvents'} }, \%vevent; } return; } elsif ($timeslot =~ /^(\d\d\d\d-\d\d-\d\d)((?: $REweekday3)?) (\d\d?)((?::\d\d)?)-(\d\d?)((?::\d\d)?)(?:am)?$/) { my $minstart = $4; $minstart = ":00" unless $minstart ne ''; my $minend = $6; $minend = ":00" unless $minend ne ''; $starttime = parse_time("$1 $3$minstart"); $endtime = parse_time("$1 $5$minend"); my $w3 = $2; $w3 =~ s/^\s+//; die "wrong weekday:($timeslot)" if $w3 ne '' && (strftime("%a",localtime($starttime)) ne $w3 || strftime("%a",localtime($endtime)) ne $w3); } #2003-09-08 Mon 1-2pm elsif ($timeslot =~ /^(\d\d\d\d-\d\d-\d\d)((?: $REweekday3)?) (\d\d?)((?::\d\d)?)-(\d\d?)((?::\d\d)?)pm$/) { my $minstart = $4; $minstart = ":00" unless $minstart ne ''; my $minend = $6; $minend = ":00" unless $minend ne ''; $starttime = parse_time("$1 $3$minstart"); $endtime = parse_time("$1 $5$minend"); if ($starttime < $endtime) { $starttime += 12*60*60 }; $endtime += 12*60*60; my $w3 = $2; $w3 =~ s/^\s+//; die "wrong weekday:($timeslot)" if $w3 ne '' && (strftime("%a",localtime($starttime)) ne $w3 || strftime("%a",localtime($endtime)) ne $w3); } else { die "cannot parse timeslot:($timeslot)" } die "start>end: $timeslot" if $starttime > $endtime; push @{ $self->{'Entries1'}} , { starttime => $starttime, endtime => $endtime, description => $description }; } else { my $col = shift; my $start = shift; my $end = shift; my $text = shift; push @{ $self->{'Entries'} }, { col => $col, start => $start, end => $end, text => $text }; } } # end of add_entry =item find_next_time(time_spec[,start_time]) Finds next time starting from start_time according to time_spec specification and returns it. If the start_time is not given, the variable StartTime is used. Examples: $t = $schedule->find_next_time("23:59", $t); =cut sub find_next_time { my $self = shift; my $timedesc = shift; my $starttime = ( $#_ == -1 ? $self->{'StartTime'} : shift @_ ); my $pattern_wday = '*'; my $pattern_hour = '*'; my $pattern_min = '*'; my $pattern_sec = 0; if ($timedesc =~ /^($REweekday3) (\d\d?(?::\d\d)?)((?:[ap]m)?)$/) { my $apm = $3; $pattern_wday = weekday_to_digits($1); $pattern_hour = $2; $pattern_min = 0; if ($pattern_hour =~ /:/) { $pattern_min=$'; $pattern_hour=$` } $pattern_sec = 0; if ($apm eq 'pm') { die unless $pattern_hour <= 12; if ($pattern_hour < 12) { $pattern_hour += 12 } } elsif ($apm eq 'am') { die unless $pattern_hour <= 12; if ($pattern_hour == 12) { $pattern_hour = 0 } } } elsif ($timedesc =~ /^(\d?\d):(\d?\d)$/) { $pattern_hour = $1; $pattern_min=$2; } else { die "cannot parse:($timedesc)" } # find seconds if ($pattern_sec ne '*') { while ((localtime($starttime))[0] != $pattern_sec) { $starttime ++ } } # find minutes if ($pattern_min ne '*') { while ((localtime($starttime))[1] != $pattern_min) { $starttime += 60 } } # find hour if ($pattern_hour ne '*') { while ((localtime($starttime))[2] != $pattern_hour) { $starttime += 3600 } } # find weekday if ($pattern_wday ne '*') { while ((localtime($starttime))[6] != $pattern_wday) { $starttime += 3600*24 } } return $starttime; } sub add_time_label { my $self = shift; my $t = shift; my @r = (); while (@{$self->{'RowLabels'}} and $t gt $self->{'RowLabels'}[0]) { push @r, shift(@{$self->{'RowLabels'}}) } push @r, $t unless @{$self->{'RowLabels'}} and $t eq $self->{'RowLabels'}[0]; push @r, @{$self->{'RowLabels'}}; $self->{'RowLabels'} = \@r; } sub todo_list { my $self = shift; my $r = "TO DO list: "; if (! @{ $self->{'ToDo'} } ) { $r .= "" } else { $r .= "

    \n". join('', map { "
  1. $_->{'desc'}\n" } @{ $self->{'ToDo'} }). "
\n"; } return $r; } =item generate_table() Returns a weekly table in HTML. Starts with NextTableTime (or StartTime if NextTableTime does not exist), and updates NextTableTime so that consecutive call produces a new table. The table column headers can be can be changed by setting the field $obj->{ColLabel} to a format as used by the standard function strftime. The default format is: ColLabel => "%AEE%Y-%m-%d", which looks something like: Monday 2008-09-01 The format "%A" would produce just the weekday name. Use $obj->{ShowDays} = 'workdays'; to display only work-days; i.e., Monday to Friday. =cut sub generate_table { my $self = shift; my (@prepareEntries, @dayEntries); $self->{'NextTableTime'} = $self->{'StartTime'} if ! exists($self->{'NextTableTime'}); my $mondaytime = $self->{'NextTableTime'}; my @showdays = 0..6; # ShowDays: all, workdays if ($self->{ShowDays} eq 'workdays') { @showdays = 0..4 } my @col_label; { my $p = $self->{'ColLabel'}; @col_label = map { strftime($p, localtime($mondaytime + $_*86400)) } @showdays; } foreach my $ve ( @{ $self->{'VEvents'} } ) { if (exists($ve->{'RRULE'}) && $ve->{'RRULE'} =~ /\bFREQ=WEEKLY\b/) { my $d = 0; my $interval = 1; if ($ve->{'RRULE'} =~ /\bINTERVAL=(\d+)/) { $interval = $1 } my $until = undef; if ($ve->{'RRULE'} =~ /\bUNTIL=(\d+)/) { $until = $1 } while ($d + $ve->{'DTSTART'} < $mondaytime + 86400*scalar(@showdays)) { if (defined($until) && $d+$ve->{'DTSTART'} > $until) { last } if ($d+$ve->{'DTSTART'} >= $mondaytime) { if (exists($ve->{'DTEND'})) { push @prepareEntries, { starttime => $d+$ve->{'DTSTART'}, endtime => $d+$ve->{'DTEND'}, description => $ve->{'SUMMARY'} }; } else { push @dayEntries, { date => $d+$ve->{'DTSTART'}, description => $ve->{'SUMMARY'} }; } } my @a = localtime($d+$ve->{'DTSTART'}); $d += 86400*7*$interval; my @b; if (exists($ve->{'DTEND'})) { @b = localtime($d+$ve->{'DTEND'}); } else { @b = localtime($d+$ve->{'DTSTART'} + 60) } $d += ($a[8]-$b[8])*3600; # daylight saving } } } push @prepareEntries, @{ $self->{'Entries1'} }; foreach my $entry ( @{ $self->{'Entries'} } ) { $self->add_time_label( $entry->{'start'} ); $self->add_time_label( $entry->{'end'} ); } foreach my $entry ( @prepareEntries ) { my $starttime = $entry->{'starttime'}; my $endtime = $entry->{'endtime'}; my $col = floor(($starttime - $mondaytime) / 86400); next if $col < 0 || $col >= scalar(@showdays); my $startlabel = strftime("%H:%M", localtime($starttime)); my $endlabel = strftime("%H:%M", localtime($endtime)); $self->add_time_label($startlabel); $self->add_time_label($endlabel); } my %eprep; $self->{'overlap'} = [ ]; foreach my $entry ( @{ $self->{'Entries'} } ) { my $col = $entry->{'col'}; my $start = $entry->{'start'}; my $end = $entry->{'end'}; my $text = $entry->{'text'}; $self->_table_add(\%eprep,$col, $start, $text, $end); } foreach my $entry ( @prepareEntries ) { my $starttime = $entry->{'starttime'}; my $endtime = $entry->{'endtime'}; my $description = $entry->{'description'}; my $col = floor(($starttime - $mondaytime) / 86400); next if $col < 0 || $col >= scalar(@showdays); my $startlabel = strftime("%H:%M", localtime($starttime)); my $endlabel = strftime("%H:%M", localtime($endtime)); $self->_table_add(\%eprep,$col, $startlabel, $description, $endlabel); } my $r = "\n". "\n". "\n"; my @op = @{ $self->{overlap} }; foreach my $di (0 .. $#col_label) { if ($op[$di] > 0) { $r.= "\n"; } $r .= "\n"; # check if there are any DayEntries push @dayEntries, grep { $_->{'date'} - $mondaytime >=0 && $_->{'date'} - $mondaytime <= scalar(@showdays)*86400 } @{ $self->{'DayEntries'} }; if ( @dayEntries ) { $r .= ''; foreach my $i (0 .. $#col_label) { my $r1; foreach my $de (grep { $_->{'date'} - $mondaytime == $i*86400 } @dayEntries ) { $r1 .= $de->{'description'}."
\n" } $r1 = ' ' unless $r1; $r .= ($op[$i]==0 ? ""; } $r .= "\n"; } my $num_of_timelabels = @{$self->{'RowLabels'}}; foreach my $ti (0 .. $num_of_timelabels - 1) { my $t = $self->{'RowLabels'}[$ti]; $r.= "\n"; foreach my $di (0 .. $#col_label) { foreach my $oi (0 .. $op[$di]) { next if $oi == 1; my @ind = (\%eprep, $di, $t); @ind = (\%eprep, $di, $t, $oi) if ($oi > 0); if (! $self->_table_get(@ind)) { $r .= "\n" } elsif ($self->_table_get(@ind) eq 'continue') { $r.= "\n" } else { my $counter = 1; my $j=$ti+1; my @ind1 = (\%eprep, $di, $self->{'RowLabels'}[$j]); @ind1 = (\%eprep, $di, $self->{'RowLabels'}[$j], $oi) if $oi > 0; if ($oi == 0) { while ($j <= $num_of_timelabels-1 && $self->_table_get(\%eprep, $di, $self->{'RowLabels'}[$j]) eq 'continue') { ++ $counter; ++$j } } else { while ($j <= $num_of_timelabels-1 && $self->_table_get(\%eprep, $di, $self->{'RowLabels'}[$j], $oi) eq 'continue') { ++ $counter; ++$j } } $r.= "\n"; } } } $r.= "\n"; } $r.="
" } else { $r.="" } $r .= $col_label[$di]."
 " : "") . "$r1
$t 1 ? " rowspan=$counter" : ''). ">".$self->_table_get(@ind)."
\n"; $self->{'NextTableTime'} = # fix for daylight saving &find_week_start( $self->{'NextTableTime'} + 86400 * 7 + 7200 ); $self->{'RowLabels'} = [ @{ $self->{'DefaultRowLabels'} } ]; return $r; } sub weekday_to_digits { local $_ = shift; s/\b(?:SUN(?:DAY)?|Sun(?:day)?)\b/00/g; s/\b(?:MON(?:DAY)?|Mon(?:day)?)\b/01/xg; s/\b(?:TUE(?:SDAY)?|Tue(?:sday)?)\b/02/xg; s/\b(?:WED(?:NESDAY)?|Wed(?:nesday)?)\b/03/xg; s/\b(?:THU(?:RSDAY)?|Thu(?:rsday)?)\b/04/xg; s/\b(?:FRI(?:DAY)?|Fri(?:day)?)\b/05/xg; s/\b(?:SAT(?:URDAY)?|Sat(?:urday)?)\b/06/xg; return $_; } sub month_to_digits { local $_ = shift; s/\b(?:JAN(?:UARY)?|Jan(?:uary)?)\b/00/g; s/\b(?:FEB(?:RUARY)?|Feb(?:ruary)?)\b/01/xg; s/\b(?:MAR(?:CH)?|Mar(?:ch)?)\b/02/xg; s/\b(?:APR(?:IL)?|Apr(?:il)?)\b/03/xg; s/\b(?:MAY(?:)?|May(?:)?)\b/04/xg; s/\b(?:JUN(?:E)?|Jun(?:e)?)\b/05/xg; s/\b(?:JUL(?:Y)?|Jul(?:y)?)\b/06/xg; s/\b(?:AUG(?:UST)?|Aug(?:ust)?)\b/07/xg; s/\b(?:SEP(?:TEMBER)?|Sep(?:tember)?)\b/08/xg; s/\b(?:OCT(?:OBER)?|Oct(?:ober)?)\b/09/xg; s/\b(?:NOV(?:EMBER)?|Nov(?:ember)?)\b/10/xg; s/\b(?:DEC(?:EMBER)?|Dec(?:ember)?)\b/11/xg; return $_; } sub _table_add { my $self = shift; my $epr = shift; my $col = shift; my $row = shift; my $des = shift; my $end = shift; my @rows = @{$self->{'RowLabels'}}; while (@rows && $rows[0] ne $row) { shift @rows } die unless @rows; if (!$end || $row eq $end) { splice(@rows,1) } else { my @t = (shift @rows); while ($rows[0] ne $end) { die unless @rows; push @t, ( shift @rows ); } @rows = @t; } my $overlap = 0; { my @trows = @rows; while (@trows) { my $r = shift @trows; my $oldoverlap = $overlap; if ($overlap==0 && defined $epr->{$col, $r}) { #$epr->{$col, $r} .= " -CONFLICT- " . $des; $overlap = 2; } while ($overlap > 0 && defined($epr->{$col,$r,$overlap})) { ++ $overlap } if ($overlap > $oldoverlap) { push @trows, @rows } } } $self->{overlap}[$col] = $overlap if $overlap > $self->{overlap}[$col]; $row = shift @rows; if ($overlap == 0) { $epr->{$col, $row} = $des; foreach my $r (@rows) { $epr->{$col, $r} = 'continue' } } else { $epr->{$col, $row, $overlap} = $des; foreach my $r (@rows) { $epr->{$col, $r, $overlap } = 'continue'; #$epr->{$col, $r} .= " -CONFLICT- continue"; } } } sub _table_get { my $self = shift; my $epr = shift; my $col = shift; my $row = shift; my $overlap = shift; return $overlap > 0 ? $epr->{$col, $row, $overlap} : $epr->{$col, $row}; } =pod =back =cut sub _getfile($) { my $f = shift; local *F; open(F, "<$f") or die "getfile:cannot open $f:$!"; my @r = ; close(F); return wantarray ? @r : join ('', @r); } 1; __END__ =head1 THANKS I would like to thank Mike Vasiljevs for his suggestions and patches for ISO8601 format. =head1 AUTHOR Copyright 2003-2008 Vlado Keselj http://www.cs.dal.ca/~vlado This script is provided "as is" without expressed or implied warranty. This is free software; you can redistribute it and/or modify it under the same terms as Perl itself. The latest version can be found at F. =cut