* @author Joerg Lehrke * @copyright (c) 2009-16 by RalfBecker-At-outdoor-training.de * @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License * @version $Id$ */ use EGroupware\Api; /** * Recurrence rule iterator * * The constructor accepts times only as DateTime (or decendents like egw_date) to work timezone-correct. * The timezone of the event is determined by timezone of the startime, other times get converted to that timezone. * * There's a static factory method calendar_rrule::event2rrule(array $event,$usertime=true), which converts an * event read by calendar_bo::read() or calendar_bo::search() to a rrule iterator. * * The rrule iterator object can be casted to string, to get a human readable description of the rrule. * * There's an interactive test-form, if the class get's called directly: http://localhost/egroupware/calendar/inc/class.calendar_rrule.inc.php * * @todo Integrate iCal import and export, so all recurrence code resides just in this class * @todo Implement COUNT, can be stored in enddate assuming counts are far smaller then timestamps (eg. < 1000 is a count) * @todo Implement WKST (week start day), currently WKST=SU is used (this is not stored in current DB schema, it's a user preference) */ class calendar_rrule implements Iterator { /** * No recurrence */ const NONE = 0; /** * Daily recurrence */ const DAILY = 1; /** * Weekly recurrance on day(s) specified by bitfield in $data */ const WEEKLY = 2; /** * Monthly recurrence iCal: monthly_bymonthday */ const MONTHLY_MDAY = 3; /** * Monthly recurrence iCal: BYDAY (by weekday, eg. 1st Friday of month) */ const MONTHLY_WDAY = 4; /** * Yearly recurrence */ const YEARLY = 5; /** * Hourly recurrence */ const HOURLY = 8; /** * Minutely recurrence */ const MINUTELY = 7; /** * RDATE: date or period (a list of dates, instead of a RRULE) */ const RDATE = 9; /** * Translate recurrence types to labels * * @var array */ static public $types = Array( self::NONE => 'None', self::DAILY => 'Daily', self::WEEKLY => 'Weekly', self::MONTHLY_WDAY => 'Monthly (by day)', self::MONTHLY_MDAY => 'Monthly (by date)', self::YEARLY => 'Yearly', self::RDATE => 'Explicit recurrences', ); /** * @var array $recur_egw2ical_2_0 conversation of egw recur-type => ical RULE;FREQ */ static private $recur_egw2ical_2_0 = array( self::DAILY => 'DAILY', self::WEEKLY => 'WEEKLY', self::MONTHLY_WDAY => 'MONTHLY', // BYDAY={1..7, -1}{MO..SO, last workday} self::MONTHLY_MDAY => 'MONTHLY', // BYMONHTDAY={1..31, -1 for last day of month} self::YEARLY => 'YEARLY', self::HOURLY => 'HOURLY', self::MINUTELY => 'MINUTELY', self::RDATE => 'RDATE', ); /** * @var array $recur_egw2ical_1_0 conversation of egw recur-type => ical FREQ */ static private $recur_egw2ical_1_0 = array( self::DAILY => 'D', self::WEEKLY => 'W', self::MONTHLY_WDAY => 'MP', // BYDAY={1..7,-1}{MO..SO, last workday} self::MONTHLY_MDAY => 'MD', // BYMONHTDAY={1..31,-1} self::YEARLY => 'YM', ); /** * RRule type: NONE, DAILY, WEEKLY, MONTHLY_MDAY, MONTHLY_WDAY, YEARLY * * @var int */ public $type = self::NONE; /** * Interval * * @var int */ public $interval = 1; /** * Number for monthly byday: 1, ..., 5, -1=last weekday of month * * EGroupware Calendar does NOT explicitly store it, it's only implicitly defined by series start date * * @var int */ public $monthly_byday_num; /** * Number for monthly bymonthday: 1, ..., 31, -1=last day of month * * EGroupware Calendar does NOT explicitly store it, it's only implicitly defined by series start date * * @var int */ public $monthly_bymonthday; /** * Recurrence dates * @var \DateTime[] */ public $rdates = []; /** * Enddate of recurring event or null, if not ending * * @var DateTime */ public $enddate; /** * Enddate of recurring event, as Ymd integer (eg. 20091111) * * Or 5 years in future, if no enddate. So iterator is always limited. * * @var int */ public $enddate_ymd; /** * Enddate of recurring event, as timestamp * * Or 5 years in future, if no enddate. So iterator is always limited. * * @var int */ public $enddate_ts; const SUNDAY = 1; const MONDAY = 2; const TUESDAY = 4; const WEDNESDAY = 8; const THURSDAY = 16; const FRIDAY = 32; const SATURDAY = 64; const WORKDAYS = 62; // Mo, ..., Fr const ALLDAYS = 127; /** * Translate weekday bitmasks to labels * * @var array */ static public $days = array( self::MONDAY => 'Monday', self::TUESDAY => 'Tuesday', self::WEDNESDAY => 'Wednesday', self::THURSDAY => 'Thursday', self::FRIDAY => 'Friday', self::SATURDAY => 'Saturday', self::SUNDAY => 'Sunday', ); /** * Bitmask of valid weekdays for weekly repeating events: self::SUNDAY|...|self::SATURDAY * * @var integer */ public $weekdays; /** * Array of exception dates (Ymd strings) * * @var array */ public $exceptions=array(); /** * Array of exceptions as DateTime/egw_time objects * * @var array */ public $exceptions_objs=array(); /** * Starttime of series * * @var Api\DateTime */ public $time; /** * Current "position" / time or null, if invalid (out of explicit RDATEs) * * @var Api\DateTime|null */ public $current; /** * Last day of the week according to user preferences * * @var int */ protected $lastdayofweek; /** * Cached timezone data * * @var array id => data */ protected static $tz_cache = array(); /** * Constructor * * The constructor accepts on DateTime (or decendents like egw_date) for all times, to work timezone-correct. * The timezone of the event is determined by timezone of $time, other times get converted to that timezone. * * @param DateTime $time start of event in it's own timezone * @param int $type self::NONE, self::DAILY, ..., self::YEARLY * @param int $interval =1 1, 2, ... * @param DateTime $enddate =null enddate or null for no enddate (in which case we user '+5 year' on $time) * @param int $weekdays =0 self::SUNDAY=1|self::MONDAY=2|...|self::SATURDAY=64 * @param DateTime[] $exceptions =null DateTime objects with exceptions * @param DateTime[] $rdates =null DateTime objects with rdates (for type self::PERIOD) */ public function __construct(DateTime $time,$type,$interval=1,DateTime $enddate=null,$weekdays=0,array $exceptions=null,array $rdates=null) { switch($GLOBALS['egw_info']['user']['preferences']['calendar']['weekdaystarts']) { case 'Sunday': $this->lastdayofweek = self::SATURDAY; break; case 'Saturday': $this->lastdayofweek = self::FRIDAY; break; default: // Monday $this->lastdayofweek = self::SUNDAY; } $this->time = $time instanceof Api\DateTime ? $time : new Api\DateTime($time); if(!in_array($type, array(self::NONE, self::DAILY, self::WEEKLY, self::MONTHLY_MDAY, self::MONTHLY_WDAY, self::YEARLY, self::HOURLY, self::MINUTELY, self::RDATE))) { throw new Api\Exception\WrongParameter(__METHOD__."($time,$type,$interval,$enddate,$weekdays,...) type $type is NOT valid!"); } $this->type = $type; // determine only implicit defined rules for RRULE=MONTHLY,BYDAY={-1, 1, ..., 5}{MO,..,SU} if ($type == self::MONTHLY_WDAY) { // check for last week of month if (($day = $this->time->format('d')) >= 21 && $day > self::daysInMonth($this->time)-7) { $this->monthly_byday_num = -1; } else { $this->monthly_byday_num = 1 + floor(($this->time->format('d')-1) / 7); } } elseif($type == self::MONTHLY_MDAY) { $this->monthly_bymonthday = (int)$this->time->format('d'); // check for last day of month if ($this->monthly_bymonthday >= 28) { $test = clone $this->time; $test->modify('1 day'); if ($test->format('m') != $this->time->format('m')) { $this->monthly_bymonthday = -1; } } } if ((int)$interval < 1) { $interval = 1; // calendar stores no (extra) interval as null, so using default 1 here } $this->interval = (int)$interval; if ($exceptions) { foreach($exceptions as $exception) { $exception->setTimezone($this->time->getTimezone()); $this->exceptions[] = $exception->format('Ymd'); } $this->exceptions_objs = $exceptions; } $this->enddate = $enddate; if($type == self::RDATE) { foreach($rdates as $rdate) { $rdate->setTimezone($this->time->getTimezone()); $this->rdates[] = $rdate; } // if startdate is neither in the rdates, nor the exceptions --> prepend it to rdates if (!in_array($this->time, $this->rdates) && !in_array($this->time, $this->exceptions_objs)) { array_unshift($this->rdates, clone($this->time)); } $enddate = clone(count($this->rdates) ? end($this->rdates) : $this->time); // Make sure to include the last date as valid $enddate->modify('+1 second'); reset($this->rdates); } // no recurrence --> current date is enddate if ($type == self::NONE) { $enddate = clone $this->time; } // set a maximum of 5 years if no enddate given elseif (is_null($enddate)) { $enddate = clone $this->time; $enddate->modify('5 year'); } // convert enddate to timezone of time, if necessary else { $enddate->setTimezone($this->time->getTimezone()); } $this->enddate_ymd = (int)$enddate->format('Ymd'); $this->enddate_ts = $enddate->format('ts'); // if no valid weekdays are given for weekly repeating, we use just the current weekday if (!($this->weekdays = (int)$weekdays) && ($type == self::WEEKLY || $type == self::MONTHLY_WDAY)) { $this->weekdays = self::getWeekday($this->time); } } /** * Get recurrence interval duration in seconds * * @param int $type self::(DAILY|WEEKLY|MONTHLY_(M|W)DAY|YEARLY) * @param int $interval =1 * @return int */ public static function recurrence_interval($type, $interval=1) { switch($type) { default: case self::DAILY: $duration = 24*3600; break; case self::WEEKLY: $duration = 7*24*3600; break; case self::MONTHLY_MDAY: case self::MONTHLY_WDAY: $duration = 31*24*3600; break; case self::YEARLY: $duration = 366*24*3600; break; } if ($interval > 1) $duration *= $interval; return $duration; } /** * Get number of days in month of given date * * @param DateTime $time * @return int */ private static function daysInMonth(DateTime $time) { list($year,$month) = explode('-',$time->format('Y-m')); $last_day = new Api\DateTime(); $last_day->setDate($year,$month+1,0); return (int)$last_day->format('d'); } /** * Return the current element * * @return ?Api\DateTime */ public function current(): ?Api\DateTime { return $this->current ? clone $this->current : null; } /** * Return the key of the current element, we use a Ymd integer as key * * @return int */ public function key(): int { return $this->current ? (int)$this->current->format('Ymd') : 0; } /** * Move forward to next recurence, not caring for exceptions */ public function next_no_exception() { switch($this->type) { case self::NONE: // need to add at least one day, to end "series", as enddate == current date case self::DAILY: $this->current->modify($this->interval.' day'); break; case self::WEEKLY: // advance to next valid weekday do { // interval in weekly means event runs on valid days eg. each 2. week // --> on the last day of the week we have to additionally advance interval-1 weeks if ($this->interval > 1 && self::getWeekday($this->current) == $this->lastdayofweek) { $this->current->modify(($this->interval-1).' week'); } $this->current->modify('1 day'); //echo __METHOD__.'() '.$this->current->format('l').', '.$this->current.": $this->weekdays & ".self::getWeekday($this->current)."
\n"; } while(!($this->weekdays & self::getWeekday($this->current))); break; case self::MONTHLY_WDAY: // iCal: BYDAY={1, ..., 5, -1}{MO..SO} // advance to start of next month list($year,$month) = explode('-',$this->current->format('Y-m')); $month += $this->interval+($this->monthly_byday_num < 0 ? 1 : 0); $this->current->setDate($year,$month,$this->monthly_byday_num < 0 ? 0 : 1); //echo __METHOD__."() $this->monthly_byday_num".substr(self::$days[$this->monthly_byday_wday],0,2).": setDate($year,$month,1): ".$this->current->format('l').', '.$this->current."
\n"; // now advance to n-th week if ($this->monthly_byday_num > 1) { $this->current->modify(($this->monthly_byday_num-1).' week'); //echo __METHOD__."() $this->monthly_byday_num".substr(self::$days[$this->monthly_byday_wday],0,2).': modify('.($this->monthly_byday_num-1).' week): '.$this->current->format('l').', '.$this->current."
\n"; } // advance to given weekday while(!($this->weekdays & self::getWeekday($this->current))) { $this->current->modify(($this->monthly_byday_num < 0 ? -1 : 1).' day'); //echo __METHOD__."() $this->monthly_byday_num".substr(self::$days[$this->monthly_byday_wday],0,2).': modify(1 day): '.$this->current->format('l').', '.$this->current."
\n"; } break; case self::MONTHLY_MDAY: // iCal: monthly_bymonthday={1, ..., 31, -1} list($year,$month) = explode('-',$this->current->format('Y-m')); $day = $this->monthly_bymonthday+($this->monthly_bymonthday < 0 ? 1 : 0); $month += $this->interval+($this->monthly_bymonthday < 0 ? 1 : 0); $this->current->setDate($year,$month,$day); //echo __METHOD__."() setDate($year,$month,$day): ".$this->current->format('l').', '.$this->current."
\n"; break; case self::YEARLY: $this->current->modify($this->interval.' year'); break; case self::HOURLY: $this->current->modify($this->interval.' hour'); break; case self::MINUTELY: $this->current->modify($this->interval.' minute'); break; case self::RDATE: if (($next = next($this->rdates))) { $this->current->setDate($next->format('Y'), $next->format('m'), $next->format('d')); $this->current->setTime($next->format('H'), $next->format('i'), $next->format('s'), 0); } else { $this->current = null; } break; default: throw new Api\Exception\AssertionFailed(__METHOD__."() invalid type #$this->type !"); } } /** * Move forward to next recurrence, taking into account exceptions */ public function next(): void { do { $this->next_no_exception(); } while($this->current && $this->exceptions && in_array($this->current->format('Ymd'),$this->exceptions)); } /** * Get weekday of $time as self::SUNDAY=1, ..., self::SATURDAY=64 integer mask * * @param DateTime $time * @return int self::SUNDAY=1, ..., self::SATURDAY=64 */ static protected function getWeekday(DateTime $time) { //echo __METHOD__.'('.$time->format('l').' '.$time.') 1 << '.$time->format('w').' = '.(1 << (int)$time->format('w'))."
\n"; return 1 << (int)$time->format('w'); } /** * Get datetime of n-th event, 1. is original event-time * * This is identical on COUNT parameter of RRULE is evaluated, exceptions are NOT taken into account! * * @param int $count * @return DateTime */ public function count2date($count) { if ($count <= 1) { return clone $this->time; } if (isset($this->current)) $backup = $this->current; $this->rewind(); while(--$count > 0) { $this->next_no_exception(); } $ret = clone $this->current; if ($backup) $this->current = $backup; return $ret; } /** * Fix enddates which are not on a recurrence, eg. for a on Monday recurring weekly event a Tuesday * * @return DateTime */ public function normalize_enddate() { $this->rewind(); while ($this->current < $this->enddate) { $previous = clone $this->current; $this->next_no_exception(); } // if enddate is now before next occurrence, but not on same day, we use previous recurrence // this can happen if client gives an enddate which is NOT a recurrence date // e.g. for an on Monday recurring weekly event a Tuesday as enddate if ($this->enddate < $this->current && $this->current->format('Ymd') != $this->enddate->format('Ymd')) { $last = $previous; } else { $last = clone $this->current; } return $last; } /** * Rewind the Iterator to the first element (called at beginning of foreach loop) */ public function rewind(): void { if ($this->type == self::RDATE) { $this->current = $this->rdates ? clone reset($this->rdates) : null; } else { $this->current = clone $this->time; } while ($this->valid() && $this->exceptions && in_array($this->current->format('Ymd'),$this->exceptions)) { $this->next_no_exception(); } } /** * Checks if current position is valid * * @param boolean? $use_just_date true: use just date, false|null: use also time * @return boolean */ public function validDate(bool $use_just_date=null): bool { if (!$this->current) { return false; } if ($use_just_date) { return $this->current->format('Ymd') <= $this->enddate_ymd; } return $this->valid(); } /** * Checks if current position is valid * * @return boolean */ public function valid(): bool { return $this->current && $this->current->format('ts') < $this->enddate_ts; } /** * Return string representation of RRule * * @return string */ function __toString( ) { $str = ''; // Repeated Events if($this->type != self::NONE) { $str = (string)lang(self::$types[$this->type]); $str_extra = array(); switch ($this->type) { case self::MONTHLY_MDAY: $str_extra[] = ($this->monthly_bymonthday == -1 ? lang('last') : $this->monthly_bymonthday.'.').' '.lang('day'); break; case self::WEEKLY: case self::MONTHLY_WDAY: $repeat_days = array(); if ($this->weekdays == self::ALLDAYS) { $repeat_days[] = $this->type == self::WEEKLY ? lang('all') : lang('day'); } elseif($this->weekdays == self::WORKDAYS) { $repeat_days[] = $this->type == self::WEEKLY ? lang('workdays') : lang('workday'); } else { foreach (self::$days as $mask => $label) { if ($this->weekdays & $mask) { $repeat_days[] = lang($label); } } } if($this->type == self::WEEKLY && count($repeat_days)) { $str_extra[] = lang('days repeated').': '.implode(', ',$repeat_days); } elseif($this->type == self::MONTHLY_WDAY) { $str_extra[] = ($this->monthly_byday_num == -1 ? lang('last') : $this->monthly_byday_num.'.').' '.implode(', ',$repeat_days); } break; case self::RDATE: $str_extra = array_map(static function (DateTime $rdate) { return str_replace(',', '', Api\DateTime::server2user($rdate, '')); }, $this->rdates); break; } if($this->interval > 1 && $this->type != self::RDATE) { $str_extra[] = lang('Interval').': '.$this->interval; } if ($this->enddate && $this->type != self::RDATE) { if ($this->enddate->getTimezone()->getName() != Api\DateTime::$user_timezone->getName()) { $this->enddate->setTimezone(Api\DateTime::$user_timezone); } $str_extra[] = lang('ends').': '.lang($this->enddate->format('l')).', '.$this->enddate->format(Api\DateTime::$user_dateformat); } if ($this->time->getTimezone()->getName() != Api\DateTime::$user_timezone->getName()) { $str_extra[] = $this->time->getTimezone()->getName(); } if ($this->exceptions_objs) { $str_extra[] = lang('Exceptions').': '.implode(', ', array_map(static function (DateTime $exdate) { return str_replace(',', '', Api\DateTime::server2user($exdate, '')); }, $this->exceptions_objs)); } if(count($str_extra)) { $str .= ' ('.implode(', ',$str_extra).')'; } } return $str; } /** * Generate a VEVENT RRULE * @param string $version ='2.0' could be '1.0' too * * $return array vCalendar RRULE */ public function generate_rrule($version='2.0') { $repeat_days = array(); $rrule = array(); if ($this->type == self::NONE || $this->type == self::RDATE) return false; // no recurring event (with RRULE) if ($version == '1.0') { $rrule['FREQ'] = self::$recur_egw2ical_1_0[$this->type] . $this->interval; switch ($this->type) { case self::WEEKLY: foreach (self::$days as $mask => $label) { if ($this->weekdays & $mask) { $repeat_days[] = strtoupper(substr($label,0,2)); } } $rrule['BYDAY'] = implode(' ', $repeat_days); $rrule['FREQ'] = $rrule['FREQ'].' '.$rrule['BYDAY']; break; case self::MONTHLY_MDAY: // date of the month: BYMONTDAY={1..31} break; case self::MONTHLY_WDAY: // weekday of the month: BDAY={1..5}+ {MO..SO} $rrule['BYDAY'] = abs($this->monthly_byday_num); $rrule['BYDAY'] .= ($this->monthly_byday_num < 0) ? '- ' : '+ '; $rrule['BYDAY'] .= strtoupper(substr($this->time->format('l'),0,2)); $rrule['FREQ'] = $rrule['FREQ'].' '.$rrule['BYDAY']; break; } if (!$this->enddate) { $rrule['UNTIL'] = '#0'; } } else // $version == '2.0' { $rrule['FREQ'] = self::$recur_egw2ical_2_0[$this->type]; switch ($this->type) { case self::WEEKLY: foreach (self::$days as $mask => $label) { if ($this->weekdays & $mask) { $repeat_days[] = strtoupper(substr($label,0,2)); } } $rrule['BYDAY'] = implode(',', $repeat_days); break; case self::MONTHLY_MDAY: // date of the month: BYMONTDAY={1..31} $rrule['BYMONTHDAY'] = $this->monthly_bymonthday; break; case self::MONTHLY_WDAY: // weekday of the month: BDAY={1..5}{MO..SO} $rrule['BYDAY'] = $this->monthly_byday_num . strtoupper(substr($this->time->format('l'),0,2)); break; case self::PERIOD: $period = []; foreach($this->period as $date) { $period[] = $date->format("Ymd\THms\Z"); } $rrule['PERIOD'] = implode(',', $period); } if ($this->interval > 1) { $rrule['INTERVAL'] = $this->interval; } } if ($this->enddate) { // our enddate is the end-time, not start-time of last event! $this->rewind(); $enddate = $this->current(); do { $this->next_no_exception(); $occurrence = $this->current(); } while ($this->valid() && ($enddate = $occurrence)); $rrule['UNTIL'] = $enddate; } return $rrule; } /** * Get instance for a given event array * * @param array $event * @param boolean $usertime =true true: event timestamps are usertime (default for calendar_bo::(read|search), false: servertime * @param string $to_tz timezone for exports (null for event's timezone) * * @return calendar_rrule false on error */ public static function event2rrule(array $event,$usertime=true,$to_tz=null) { if (!is_array($event) || !isset($event['tzid'])) return false; if (!$to_tz) $to_tz = $event['tzid']; $timestamp_tz = $usertime ? Api\DateTime::$user_timezone : Api\DateTime::$server_timezone; $time = is_a($event['start'],'DateTime') ? $event['start'] : new Api\DateTime($event['start'],$timestamp_tz); if (!isset(self::$tz_cache[$to_tz])) { self::$tz_cache[$to_tz] = calendar_timezones::DateTimeZone($to_tz); } self::rrule2tz($event, $time, $to_tz); $time->setTimezone(self::$tz_cache[$to_tz]); if ($event['recur_enddate']) { $enddate = is_a($event['recur_enddate'],'DateTime') ? clone $event['recur_enddate'] : new Api\DateTime($event['recur_enddate'],$timestamp_tz); // Check to see if switching timezones changes the date, we'll need to adjust for that $enddate_event_timezone = clone $enddate; $enddate->setTimezone($timestamp_tz); $delta = (int)$enddate_event_timezone->format('z') - (int)$enddate->format('z'); $enddate->add("$delta days"); $end = is_a($event['end'],'DateTime') ? clone $event['end'] : new Api\DateTime($event['end'],$timestamp_tz); $end->setTimezone($enddate->getTimezone()); $enddate->setTime($end->format('H'),$end->format('i'),0); if(!empty($event['whole_day'])) { $enddate->setTime(23,59,59); } } if (is_array($event['recur_exception'])) { foreach($event['recur_exception'] as $exception) { $exceptions[] = is_a($exception,'DateTime') ? $exception : new Api\DateTime($exception, $timestamp_tz); } } if (is_array($event['recur_rdates'])) { foreach($event['recur_rdates'] as $rdate) { $rdates[] = is_a($rdate,'DateTime') ? $rdate : new Api\DateTime($rdate, $timestamp_tz); } } return new calendar_rrule($time,$event['recur_type'],$event['recur_interval'],$enddate??null,$event['recur_data'],$exceptions??null,$rdates??null); } /** * Generate a rrule from a string generated by __toString(). * * @param String $rrule Recurrence rule in string format, as generated by __toString() * @param DateTime date Optional date to work from, defaults to today */ public static function from_string(String $rrule, DateTime $date) { $time = $date ? $date : new Api\DateTime(); $type_id = self::NONE; $interval = 1; $enddate = null; $weekdays = 0; $exceptions = array(); list($type, $sub, $conds) = explode(' (', $rrule); if(!$conds) { $conds = $sub; } else { $type .= " ($sub"; } $conditions = explode(', ', substr($conds, 0, -1)); foreach(static::$types as $id => $type_name) { $str = lang($type_name); if($str == $type) { $type_id = $id; break; } } // Rejoin some extra splits for conditions with multiple values foreach($conditions as $condition_index => $condition) { if(((int)$condition || strpos($condition, lang('last')) === 0) && substr_compare( $condition, lang('day'), -strlen( lang('day') ) ) === 0) { $time->setDate($time->format('Y'), $time->format('m'), (int)($condition) ? (int)$condition : $time->format('t')); unset($conditions[$condition_index]); continue; } if(!strpos($condition, ':') && strpos($conditions[$condition_index-1], ':')) { $conditions[$condition_index-1] .= ', ' . $condition; unset($conditions[$condition_index]); } } foreach($conditions as $condition_index => $condition) { list($condition_name, $value) = explode(': ', $condition); if($condition_name == lang('days repeated')) { foreach (self::$days as $mask => $label) { if (strpos($value, $label) !== FALSE || strpos($value, lang($label)) !== FALSE) { $weekdays += $mask; } } if(stripos($condition, lang('all')) !== false) { $weekdays = self::ALLDAYS; } } else if ($condition_name == lang('interval')) { $interval = (int)$value; } else if ($condition_name == lang('ends')) { list(, $date) = explode(', ', $value); $enddate = new DateTime($date); } } return new calendar_rrule($time,$type_id,$interval,$enddate,$weekdays,$exceptions); } /** * Get recurrence data (keys 'recur_*') to merge into an event * * @return array */ public function rrule2event() { return array( 'recur_type' => $this->type, 'recur_interval' => $this->interval, 'recur_enddate' => $this->enddate ? $this->enddate->format('ts') : null, 'recur_data' => $this->weekdays, 'recur_exception' => $this->exceptions, 'recur_rdates' => $this->rdates, ); } /** * Shift a recurrence rule to a new timezone * * @param array $event recurring event * @param DateTime|string $starttime of the event (in servertime) * @param string $to_tz new timezone */ public static function rrule2tz(array &$event,$starttime,$to_tz) { // We assume that the difference between timezones can result // in a maximum of one day if (!is_array($event) || !isset($event['recur_type']) || $event['recur_type'] == self::NONE || empty($event['recur_data']) || $event['recur_data'] == self::ALLDAYS || empty($event['tzid']) || empty($to_tz) || $event['tzid'] == $to_tz) return; if (!isset(self::$tz_cache[$event['tzid']])) { self::$tz_cache[$event['tzid']] = calendar_timezones::DateTimeZone($event['tzid']); } if (!isset(self::$tz_cache[$to_tz])) { self::$tz_cache[$to_tz] = calendar_timezones::DateTimeZone($to_tz); } $time = is_a($starttime,'DateTime') ? $starttime : new Api\DateTime($starttime, Api\DateTime::$server_timezone); $time->setTimezone(self::$tz_cache[$event['tzid']]); $remote = clone $time; $remote->setTimezone(self::$tz_cache[$to_tz]); $delta = (int)$remote->format('w') - (int)$time->format('w'); if ($delta) { // We have to generate a shifted rrule switch ($event['recur_type']) { case self::MONTHLY_WDAY: case self::WEEKLY: $mask = (int)$event['recur_data']; if ($delta == 1 || $delta == -6) { $mask = $mask << 1; if ($mask & 128) $mask = $mask - 127; // overflow } else { if ($mask & 1) $mask = $mask + 128; // underflow $mask = $mask >> 1; } $event['recur_data'] = $mask; } } } /** * Parses a DateTime field and returns a unix timestamp. If the * field cannot be parsed then the original text is returned * unmodified. * * @param string $text The Icalendar datetime field value. * @param string $tzid =null A timezone identifier. * * @return integer A unix timestamp. */ private static function parseIcalDateTime($text, $tzid=null) { static $vcal = null; if (!isset($vcal)) $vcal = new Horde_Icalendar; return $vcal->_parseDateTime($text, $tzid); } /** * Parse an iCal recurrence-rule string * * @param string $recurence * @param bool $support_below_daily =false true: support FREQ=HOURLY|MINUTELY * @param array $vcardData =[] already parsed values, specially value for "start" is used if no explicit BYDAY attribute * @return array */ public static function parseRrule($recurence, $support_below_daily=false, array $vcardData=[]) { $vcardData['recur_interval'] = 1; $matches = null; $type = preg_match('/FREQ=([^;: ]+)/i',$recurence,$matches) ? $matches[1] : $recurence[0]; // vCard 2.0 values for all types if (preg_match('/UNTIL=([0-9TZ]+)/',$recurence,$matches)) { $vcardData['recur_enddate'] = self::parseIcalDateTime($matches[1]); // If it couldn't be parsed, treat it as not set if(is_string($vcardData['recur_enddate'])) { unset($vcardData['recur_enddate']); } } elseif (preg_match('/COUNT=([0-9]+)/',$recurence,$matches)) { $vcardData['recur_count'] = (int)$matches[1]; } if (preg_match('/INTERVAL=([0-9]+)/',$recurence,$matches)) { $vcardData['recur_interval'] = (int) $matches[1] ? (int) $matches[1] : 1; } $vcardData['recur_data'] = 0; switch($type) { case 'D': // 1.0 $recurenceMatches = null; if (preg_match('/D(\d+) #(\d+)/', $recurence, $recurenceMatches)) { $vcardData['recur_interval'] = $recurenceMatches[1]; $vcardData['recur_count'] = $recurenceMatches[2]; } elseif (preg_match('/D(\d+) (.*)/', $recurence, $recurenceMatches)) { $vcardData['recur_interval'] = $recurenceMatches[1]; $vcardData['recur_enddate'] = self::parseIcalDateTime(trim($recurenceMatches[2])); } else break; // fall-through case 'DAILY': // 2.0 $vcardData['recur_type'] = self::DAILY; if (stripos($recurence, 'BYDAY') === false) break; // hack to handle TYPE=DAILY;BYDAY= as WEEKLY, which is true as long as there's no interval // fall-through case 'W': case 'WEEKLY': $days = array(); if (preg_match('/W(\d+) *((?i: [AEFHMORSTUW]{2})+)?( +([^ ]*))$/',$recurence, $recurenceMatches)) // 1.0 { $vcardData['recur_interval'] = $recurenceMatches[1]; if (empty($recurenceMatches[2])) { $days[0] = strtoupper(substr(date('D', $vcardData['start']),0,2)); } else { $days = explode(' ',trim($recurenceMatches[2])); } $repeatMatches = null; if (preg_match('/#(\d+)/',$recurenceMatches[4],$repeatMatches)) { if ($repeatMatches[1]) $vcardData['recur_count'] = $repeatMatches[1]; } else { $vcardData['recur_enddate'] = self::parseIcalDateTime($recurenceMatches[4]); } } elseif (preg_match('/BYDAY=([^;: ]+)/',$recurence,$recurenceMatches)) // 2.0 { $days = explode(',',$recurenceMatches[1]); } else // no day given, use the day of dtstart { $vcardData['recur_data'] |= 1 << (int)date('w',$vcardData['start']); $vcardData['recur_type'] = self::WEEKLY; } if ($days) { foreach (self::$days as $id => $day) { if (in_array(strtoupper(substr($day,0,2)),$days)) { $vcardData['recur_data'] |= $id; } } $vcardData['recur_type'] = self::WEEKLY; } break; case 'M': if (preg_match('/MD(\d+)(?: [^ ]+)? #(\d+)/', $recurence, $recurenceMatches)) { $vcardData['recur_type'] = self::MONTHLY_MDAY; $vcardData['recur_interval'] = $recurenceMatches[1]; $vcardData['recur_count'] = $recurenceMatches[2]; } elseif (preg_match('/MD(\d+)(?: [^ ]+)? ([0-9TZ]+)/',$recurence, $recurenceMatches)) { $vcardData['recur_type'] = self::MONTHLY_MDAY; $vcardData['recur_interval'] = $recurenceMatches[1]; $vcardData['recur_enddate'] = self::parseIcalDateTime($recurenceMatches[2]); } elseif (preg_match('/MP(\d+) (.*) (.*) (.*)/',$recurence, $recurenceMatches)) { $vcardData['recur_type'] = self::MONTHLY_WDAY; $vcardData['recur_interval'] = $recurenceMatches[1]; if (preg_match('/#(\d+)/',$recurenceMatches[4],$recurenceMatches)) { $vcardData['recur_count'] = $recurenceMatches[1]; } else { $vcardData['recur_enddate'] = self::parseIcalDateTime(trim($recurenceMatches[4])); } } break; case 'Y': // 1.0 if (preg_match('/YM(\d+)(?: [^ ]+)? #(\d+)/', $recurence, $recurenceMatches)) { $vcardData['recur_interval'] = $recurenceMatches[1]; $vcardData['recur_count'] = $recurenceMatches[2]; } elseif (preg_match('/YM(\d+)(?: [^ ]+)? ([0-9TZ]+)/',$recurence, $recurenceMatches)) { $vcardData['recur_interval'] = $recurenceMatches[1]; $vcardData['recur_enddate'] = self::parseIcalDateTime($recurenceMatches[2]); } else break; // fall-through case 'YEARLY': // 2.0 if (strpos($recurence, 'BYDAY') === false) { $vcardData['recur_type'] = self::YEARLY; break; } // handle FREQ=YEARLY;BYDAY= as FREQ=MONTHLY;BYDAY= with 12*INTERVAL $vcardData['recur_interval'] = $vcardData['recur_interval'] ? 12*$vcardData['recur_interval'] : 12; // fall-through case 'MONTHLY': // does currently NOT parse BYDAY or BYMONTH, it has to be specified/identical to DTSTART $vcardData['recur_type'] = strpos($recurence,'BYDAY') !== false ? self::MONTHLY_WDAY : self::MONTHLY_MDAY; break; case 'HOURLY': if ($support_below_daily) $vcardData['recur_type'] = self::HOURLY; break; case 'MINUTELY': if ($support_below_daily) $vcardData['recur_type'] = self::MINUTELY; break; } return $vcardData; } } if (isset($_SERVER['SCRIPT_FILENAME']) && $_SERVER['SCRIPT_FILENAME'] == __FILE__) // some tests { ini_set('display_errors',1); error_reporting(E_ALL & ~E_NOTICE); function lang($str) { return $str; } $GLOBALS['egw_info']['user']['preferences']['common']['tz'] = $_REQUEST['user-tz'] ? $_REQUEST['user-tz'] : 'Europe/Berlin'; require_once('../../api/src/autoload.php'); if (!isset($_REQUEST['time'])) { $now = new Api\DateTime('now',new DateTimeZone($_REQUEST['tz'] = 'UTC')); $_REQUEST['time'] = $now->format(); $_REQUEST['type'] = calendar_rrule::WEEKLY; $_REQUEST['interval'] = 2; $now->modify('2 month'); $_REQUEST['enddate'] = $now->format('Y-m-d'); $_REQUEST['user-tz'] = 'Europe/Berlin'; } echo "\n\n\tTest calendar_rrule class\n\n\n
\n"; echo "

Date+Time: ".Api\Html::input('time',$_REQUEST['time']). Api\Html::select('tz',$_REQUEST['tz'],Api\DateTime::getTimezones())."

\n"; echo "

Type: ".Api\Html::select('type',$_REQUEST['type'],calendar_rrule::$types)."\n". "Interval: ".Api\Html::input('interval',$_REQUEST['interval'])."

\n"; echo "
\n"; echo "Weekdays:
".Api\Html::checkbox_multiselect('weekdays',$_REQUEST['weekdays'],calendar_rrule::$days,false,'','7',false,'height: 150px;')."\n"; echo "
\n"; echo "

Exceptions:
".Api\Html::textarea('exceptions',$_REQUEST['exceptions'],'style="height: 150px;"')."\n"; echo "

\n"; echo "

Enddate: ".Api\Html::input('enddate',$_REQUEST['enddate'])."

\n"; echo "

Display recurances in ".Api\Html::select('user-tz',$_REQUEST['user-tz'],Api\DateTime::getTimezones())."

\n"; echo "

".Api\Html::submit_button('calc','Calculate')."

\n"; echo "
\n"; $tz = new DateTimeZone($_REQUEST['tz']); $time = new Api\DateTime($_REQUEST['time'],$tz); if ($_REQUEST['enddate']) $enddate = new Api\DateTime($_REQUEST['enddate'],$tz); $weekdays = 0; foreach((array)$_REQUEST['weekdays'] as $mask) { $weekdays |= $mask; } if ($_REQUEST['exceptions']) foreach(preg_split("/[,\r\n]+ ?/",$_REQUEST['exceptions']) as $exception) { $exceptions[] = new Api\DateTime($exception); } $rrule = new calendar_rrule($time,$_REQUEST['type'],$_REQUEST['interval'],$enddate,$weekdays,$exceptions); echo "

".$time->format('l').', '.$time.' ('.$tz->getName().') '.$rrule."

\n"; $n = 0; foreach($rrule as $rtime) { $rtime->setTimezone(Api\DateTime::$user_timezone); echo ++$n.': '.$rtime->format('l').', '.$rtime."
\n"; } echo "\n\n"; }