]> err.no Git - sope/blob - sope-ical/NGiCal/iCalRepeatableEntityObject.m
improved API, bugfix for recurrent dates
[sope] / sope-ical / NGiCal / iCalRepeatableEntityObject.m
1 /*
2   Copyright (C) 2004-2005 SKYRIX Software AG
3
4   This file is part of SOPE.
5
6   SOPE is free software; you can redistribute it and/or modify it under
7   the terms of the GNU Lesser General Public License as published by the
8   Free Software Foundation; either version 2, or (at your option) any
9   later version.
10
11   SOPE is distributed in the hope that it will be useful, but WITHOUT ANY
12   WARRANTY; without even the implied warranty of MERCHANTABILITY or
13   FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public
14   License for more details.
15
16   You should have received a copy of the GNU Lesser General Public
17   License along with SOPE; see the file COPYING.  If not, write to the
18   Free Software Foundation, 59 Temple Place - Suite 330, Boston, MA
19   02111-1307, USA.
20 */
21
22 #include "iCalRepeatableEntityObject.h"
23 #include <NGExtensions/NGCalendarDateRange.h>
24 #include "iCalRecurrenceRule.h"
25 #include "iCalRecurrenceCalculator.h"
26 #include "common.h"
27
28 @implementation iCalRepeatableEntityObject
29
30 - (void)dealloc {
31   [self->rRules  release];
32   [self->exRules release];
33   [self->exDates release];
34   [super dealloc];
35 }
36
37 /* Accessors */
38
39 - (void)removeAllRecurrenceRules {
40   [self->rRules removeAllObjects];
41 }
42 - (void)addToRecurrenceRules:(id)_rrule {
43   if (_rrule == nil) return;
44   if (self->rRules == nil)
45     self->rRules = [[NSMutableArray alloc] initWithCapacity:1];
46   [self->rRules addObject:_rrule];
47 }
48 - (void)setRecurrenceRules:(NSArray *)_rrules {
49   if (_rrules == self->rRules)
50     return;
51   [self->rRules release];
52   self->rRules = [_rrules mutableCopy];
53 }
54 - (BOOL)hasRecurrenceRules {
55   return [self->rRules count] > 0 ? YES : NO;
56 }
57 - (NSArray *)recurrenceRules {
58   return self->rRules;
59 }
60
61 - (void)removeAllExceptionRules {
62   [self->exRules removeAllObjects];
63 }
64 - (void)addToExceptionRules:(id)_rrule {
65   if (_rrule == nil) return;
66   if (self->exRules == nil)
67     self->exRules = [[NSMutableArray alloc] initWithCapacity:1];
68   [self->exRules addObject:_rrule];
69 }
70 - (void)setExceptionRules:(NSArray *)_rrules {
71   if (_rrules == self->exRules)
72     return;
73   [self->exRules release];
74   self->exRules = [_rrules mutableCopy];
75 }
76 - (BOOL)hasExceptionRules {
77   return [self->exRules count] > 0 ? YES : NO;
78 }
79 - (NSArray *)exceptionRules {
80   return self->exRules;
81 }
82
83 - (void)removeAllExceptionDates {
84   [self->exDates removeAllObjects];
85 }
86 - (void)setExceptionDates:(NSArray *)_exDates {
87   if (_exDates == self->exDates)
88     return;
89   [self->exDates release];
90   self->exDates = [_exDates mutableCopy];
91 }
92 - (void)addToExceptionDates:(id)_date {
93   if (_date == nil) return;
94   if (self->exDates == nil)
95     self->exDates = [[NSMutableArray alloc] initWithCapacity:4];
96   [self->exDates addObject:_date];
97 }
98 - (BOOL)hasExceptionDates {
99   return [self->exDates count] > 0 ? YES : NO;
100 }
101 - (NSArray *)exceptionDates {
102   return self->exDates;
103 }
104
105 /* Convenience */
106
107 - (BOOL)isRecurrent {
108   return [self hasRecurrenceRules] ? YES : NO;
109 }
110
111 /* Matching */
112
113 - (BOOL)isWithinCalendarDateRange:(NGCalendarDateRange *)_range
114   firstInstanceCalendarDateRange:(NGCalendarDateRange *)_fir
115 {
116   NSArray *ranges;
117   
118   ranges = [self recurrenceRangesWithinCalendarDateRange:_range
119                  firstInstanceCalendarDateRange:_fir];
120   return [ranges count] > 0;
121 }
122
123 - (NSArray *)recurrenceRangesWithinCalendarDateRange:(NGCalendarDateRange *)_r
124   firstInstanceCalendarDateRange:(NGCalendarDateRange *)_fir
125 {
126   return [iCalRecurrenceCalculator recurrenceRangesWithinCalendarDateRange:_r
127                                    firstInstanceCalendarDateRange:_fir
128                                    recurrenceRules:self->rRules
129                                    exceptionRules:self->exRules
130                                    exceptionDates:self->exDates];
131 }
132
133
134 /* this is the outmost bound possible, not necessarily the real last date */
135 - (NSCalendarDate *)lastPossibleRecurrenceStartDateUsingFirstInstanceCalendarDateRange:(NGCalendarDateRange *)_r
136 {
137   NSCalendarDate *date;
138   unsigned       i, count;
139   
140   count = [self->rRules count];
141   if (!count)
142     return nil;
143
144   date  = nil;
145   for (i = 0; i < count; i++) {
146     iCalRecurrenceRule       *rule;
147     iCalRecurrenceCalculator *calc;
148     NSCalendarDate           *rdate;
149
150     rule = [self->rRules objectAtIndex:i];
151     if ([rule isInfinite])
152       return nil; /* rule is not bound, hence no limit */
153     calc  = [iCalRecurrenceCalculator recurrenceCalculatorForRecurrenceRule:rule
154                                       withFirstInstanceCalendarDateRange:_r];
155     rdate = [[calc lastInstanceCalendarDateRange] startDate];
156     if (date == nil || [date compare:rdate] == NSOrderedAscending)
157       date = rdate;
158   }
159   return date;
160 }
161
162 @end