summaryrefslogtreecommitdiff
Unidiff
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--library/datebookdb.cpp3
1 files changed, 3 insertions, 0 deletions
diff --git a/library/datebookdb.cpp b/library/datebookdb.cpp
index e4ec2bf..165a637 100644
--- a/library/datebookdb.cpp
+++ b/library/datebookdb.cpp
@@ -1,904 +1,907 @@
1/********************************************************************** 1/**********************************************************************
2** Copyright (C) 2000 Trolltech AS. All rights reserved. 2** Copyright (C) 2000 Trolltech AS. All rights reserved.
3** 3**
4** This file is part of Qtopia Environment. 4** This file is part of Qtopia Environment.
5** 5**
6** This file may be distributed and/or modified under the terms of the 6** This file may be distributed and/or modified under the terms of the
7** GNU General Public License version 2 as published by the Free Software 7** GNU General Public License version 2 as published by the Free Software
8** Foundation and appearing in the file LICENSE.GPL included in the 8** Foundation and appearing in the file LICENSE.GPL included in the
9** packaging of this file. 9** packaging of this file.
10** 10**
11** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE 11** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE
12** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. 12** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
13** 13**
14** See http://www.trolltech.com/gpl/ for GPL licensing information. 14** See http://www.trolltech.com/gpl/ for GPL licensing information.
15** 15**
16** Contact info@trolltech.com if any conditions of this licensing are 16** Contact info@trolltech.com if any conditions of this licensing are
17** not clear to you. 17** not clear to you.
18** 18**
19**********************************************************************/ 19**********************************************************************/
20 20
21#include <qasciidict.h> 21#include <qasciidict.h>
22#include <qmessagebox.h> 22#include <qmessagebox.h>
23#include <qtl.h> 23#include <qtl.h>
24 24
25#include <qpe/alarmserver.h> 25#include <qpe/alarmserver.h>
26#include <qpe/global.h> 26#include <qpe/global.h>
27#include "datebookdb.h" 27#include "datebookdb.h"
28#include <qpe/stringutil.h> 28#include <qpe/stringutil.h>
29 29
30#include <errno.h> 30#include <errno.h>
31#include <stdlib.h> 31#include <stdlib.h>
32 32
33 33
34class DateBookDBPrivate 34class DateBookDBPrivate
35{ 35{
36public: 36public:
37 bool clean; // indcate whether we need to write to disk... 37 bool clean; // indcate whether we need to write to disk...
38}; 38};
39 39
40 40
41// Helper functions 41// Helper functions
42 42
43static QString dateBookJournalFile() 43static QString dateBookJournalFile()
44{ 44{
45 QString str = getenv("HOME"); 45 QString str = getenv("HOME");
46 return QString( str +"/.caljournal" ); 46 return QString( str +"/.caljournal" );
47} 47}
48 48
49static QString dateBookFilename() 49static QString dateBookFilename()
50{ 50{
51 return Global::applicationFileName("datebook","datebook.xml"); 51 return Global::applicationFileName("datebook","datebook.xml");
52} 52}
53 53
54/* Calculating the next event of a recuring event is actually 54/* Calculating the next event of a recuring event is actually
55 computationally inexpensive, esp. compared to checking each day 55 computationally inexpensive, esp. compared to checking each day
56 individually. There are bad worse cases for say the 29th of 56 individually. There are bad worse cases for say the 29th of
57 february or the 31st of some other months. However 57 february or the 31st of some other months. However
58 these are still bounded */ 58 these are still bounded */
59bool nextOccurance(const Event &e, const QDate &from, QDateTime &next) 59bool nextOccurance(const Event &e, const QDate &from, QDateTime &next)
60{ 60{
61 // easy checks, first are we too far in the future or too far in the past? 61 // easy checks, first are we too far in the future or too far in the past?
62 QDate tmpDate; 62 QDate tmpDate;
63 int freq = e.repeatPattern().frequency; 63 int freq = e.repeatPattern().frequency;
64 int diff, diff2, a; 64 int diff, diff2, a;
65 int iday, imonth, iyear; 65 int iday, imonth, iyear;
66 int dayOfWeek = 0; 66 int dayOfWeek = 0;
67 int firstOfWeek = 0; 67 int firstOfWeek = 0;
68 int weekOfMonth; 68 int weekOfMonth;
69 69
70 70
71 if (e.repeatPattern().hasEndDate && e.repeatPattern().endDate() < from) 71 if (e.repeatPattern().hasEndDate && e.repeatPattern().endDate() < from)
72 return FALSE; 72 return FALSE;
73 73
74 if (e.start() >= from) { 74 if (e.start() >= from) {
75 next = e.start(); 75 next = e.start();
76 return TRUE; 76 return TRUE;
77 } 77 }
78 78
79 switch ( e.repeatPattern().type ) { 79 switch ( e.repeatPattern().type ) {
80 case Event::Weekly: 80 case Event::Weekly:
81 /* weekly is just daily by 7 */ 81 /* weekly is just daily by 7 */
82 /* first convert the repeatPattern.Days() mask to the next 82 /* first convert the repeatPattern.Days() mask to the next
83 day of week valid after from */ 83 day of week valid after from */
84 dayOfWeek = from.dayOfWeek(); 84 dayOfWeek = from.dayOfWeek();
85 dayOfWeek--; /* we want 0-6, doco for above specs 1-7 */ 85 dayOfWeek--; /* we want 0-6, doco for above specs 1-7 */
86 86
87 /* this is done in case freq > 1 and from in week not 87 /* this is done in case freq > 1 and from in week not
88 for this round */ 88 for this round */
89 // firstOfWeek = 0; this is already done at decl. 89 // firstOfWeek = 0; this is already done at decl.
90 while(!((1 << firstOfWeek) & e.repeatPattern().days)) 90 while(!((1 << firstOfWeek) & e.repeatPattern().days))
91 firstOfWeek++; 91 firstOfWeek++;
92 92
93 93
94 94
95 /* there is at least one 'day', or there would be no event */ 95 /* there is at least one 'day', or there would be no event */
96 while(!((1 << (dayOfWeek % 7)) & e.repeatPattern().days)) 96 while(!((1 << (dayOfWeek % 7)) & e.repeatPattern().days))
97 dayOfWeek++; 97 dayOfWeek++;
98 98
99 99
100 dayOfWeek = dayOfWeek % 7; /* the actual day of week */ 100 dayOfWeek = dayOfWeek % 7; /* the actual day of week */
101 dayOfWeek -= e.start().date().dayOfWeek() -1; 101 dayOfWeek -= e.start().date().dayOfWeek() -1;
102 102
103 firstOfWeek = firstOfWeek % 7; /* the actual first of week */ 103 firstOfWeek = firstOfWeek % 7; /* the actual first of week */
104 firstOfWeek -= e.start().date().dayOfWeek() -1; 104 firstOfWeek -= e.start().date().dayOfWeek() -1;
105 105
106 // dayOfWeek may be negitive now 106 // dayOfWeek may be negitive now
107 // day of week is number of days to add to start day 107 // day of week is number of days to add to start day
108 108
109 freq *= 7; 109 freq *= 7;
110 // FALL-THROUGH !!!!! 110 // FALL-THROUGH !!!!!
111 case Event::Daily: 111 case Event::Daily:
112 // the add is for the possible fall through from weekly */ 112 // the add is for the possible fall through from weekly */
113 if(e.start().date().addDays(dayOfWeek) > from) { 113 if(e.start().date().addDays(dayOfWeek) > from) {
114 /* first week exception */ 114 /* first week exception */
115 next = QDateTime(e.start().date().addDays(dayOfWeek), 115 next = QDateTime(e.start().date().addDays(dayOfWeek),
116 e.start().time()); 116 e.start().time());
117 if ((next.date() > e.repeatPattern().endDate()) 117 if ((next.date() > e.repeatPattern().endDate())
118 && e.repeatPattern().hasEndDate) 118 && e.repeatPattern().hasEndDate)
119 return FALSE; 119 return FALSE;
120 return TRUE; 120 return TRUE;
121 } 121 }
122 /* if from is middle of a non-week */ 122 /* if from is middle of a non-week */
123 123
124 diff = e.start().date().addDays(dayOfWeek).daysTo(from) % freq; 124 diff = e.start().date().addDays(dayOfWeek).daysTo(from) % freq;
125 diff2 = e.start().date().addDays(firstOfWeek).daysTo(from) % freq; 125 diff2 = e.start().date().addDays(firstOfWeek).daysTo(from) % freq;
126 126
127 if(diff != 0) 127 if(diff != 0)
128 diff = freq - diff; 128 diff = freq - diff;
129 if(diff2 != 0) 129 if(diff2 != 0)
130 diff2 = freq - diff2; 130 diff2 = freq - diff2;
131 diff = QMIN(diff, diff2); 131 diff = QMIN(diff, diff2);
132 132
133 next = QDateTime(from.addDays(diff), e.start().time()); 133 next = QDateTime(from.addDays(diff), e.start().time());
134 if ( (next.date() > e.repeatPattern().endDate()) 134 if ( (next.date() > e.repeatPattern().endDate())
135 && e.repeatPattern().hasEndDate ) 135 && e.repeatPattern().hasEndDate )
136 return FALSE; 136 return FALSE;
137 return TRUE; 137 return TRUE;
138 case Event::MonthlyDay: 138 case Event::MonthlyDay:
139 iday = from.day(); 139 iday = from.day();
140 iyear = from.year(); 140 iyear = from.year();
141 imonth = from.month(); 141 imonth = from.month();
142 /* find equivelent day of month for this month */ 142 /* find equivelent day of month for this month */
143 dayOfWeek = e.start().date().dayOfWeek(); 143 dayOfWeek = e.start().date().dayOfWeek();
144 weekOfMonth = (e.start().date().day() - 1) / 7; 144 weekOfMonth = (e.start().date().day() - 1) / 7;
145 145
146 /* work out when the next valid month is */ 146 /* work out when the next valid month is */
147 a = from.year() - e.start().date().year(); 147 a = from.year() - e.start().date().year();
148 a *= 12; 148 a *= 12;
149 a = a + (imonth - e.start().date().month()); 149 a = a + (imonth - e.start().date().month());
150 /* a is e.start()monthsFrom(from); */ 150 /* a is e.start()monthsFrom(from); */
151 if(a % freq) { 151 if(a % freq) {
152 a = freq - (a % freq); 152 a = freq - (a % freq);
153 imonth = from.month() + a; 153 imonth = from.month() + a;
154 if (imonth > 12) { 154 if (imonth > 12) {
155 imonth--; 155 imonth--;
156 iyear += imonth / 12; 156 iyear += imonth / 12;
157 imonth = imonth % 12; 157 imonth = imonth % 12;
158 imonth++; 158 imonth++;
159 } 159 }
160 } 160 }
161 /* imonth is now the first month after or on 161 /* imonth is now the first month after or on
162 from that matches the frequency given */ 162 from that matches the frequency given */
163 163
164 /* find for this month */ 164 /* find for this month */
165 tmpDate = QDate( iyear, imonth, 1 ); 165 tmpDate = QDate( iyear, imonth, 1 );
166 166
167 iday = 1; 167 iday = 1;
168 iday += (7 + dayOfWeek - tmpDate.dayOfWeek()) % 7; 168 iday += (7 + dayOfWeek - tmpDate.dayOfWeek()) % 7;
169 iday += 7 * weekOfMonth; 169 iday += 7 * weekOfMonth;
170 while (iday > tmpDate.daysInMonth()) { 170 while (iday > tmpDate.daysInMonth()) {
171 imonth += freq; 171 imonth += freq;
172 if (imonth > 12) { 172 if (imonth > 12) {
173 imonth--; 173 imonth--;
174 iyear += imonth / 12; 174 iyear += imonth / 12;
175 imonth = imonth % 12; 175 imonth = imonth % 12;
176 imonth++; 176 imonth++;
177 } 177 }
178 tmpDate = QDate( iyear, imonth, 1 ); 178 tmpDate = QDate( iyear, imonth, 1 );
179 /* these loops could go for a while, check end case now */ 179 /* these loops could go for a while, check end case now */
180 if ((tmpDate > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate) 180 if ((tmpDate > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate)
181 return FALSE; 181 return FALSE;
182 iday = 1; 182 iday = 1;
183 iday += (7 + dayOfWeek - tmpDate.dayOfWeek()) % 7; 183 iday += (7 + dayOfWeek - tmpDate.dayOfWeek()) % 7;
184 iday += 7 * weekOfMonth; 184 iday += 7 * weekOfMonth;
185 } 185 }
186 tmpDate = QDate(iyear, imonth, iday); 186 tmpDate = QDate(iyear, imonth, iday);
187 187
188 if (tmpDate >= from) { 188 if (tmpDate >= from) {
189 next = QDateTime(tmpDate, e.start().time()); 189 next = QDateTime(tmpDate, e.start().time());
190 if ((next.date() > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate) 190 if ((next.date() > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate)
191 return FALSE; 191 return FALSE;
192 return TRUE; 192 return TRUE;
193 } 193 }
194 194
195 /* need to find the next iteration */ 195 /* need to find the next iteration */
196 do { 196 do {
197 imonth += freq; 197 imonth += freq;
198 if (imonth > 12) { 198 if (imonth > 12) {
199 imonth--; 199 imonth--;
200 iyear += imonth / 12; 200 iyear += imonth / 12;
201 imonth = imonth % 12; 201 imonth = imonth % 12;
202 imonth++; 202 imonth++;
203 } 203 }
204 tmpDate = QDate( iyear, imonth, 1 ); 204 tmpDate = QDate( iyear, imonth, 1 );
205 /* these loops could go for a while, check end case now */ 205 /* these loops could go for a while, check end case now */
206 if ((tmpDate > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate) 206 if ((tmpDate > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate)
207 return FALSE; 207 return FALSE;
208 iday = 1; 208 iday = 1;
209 iday += (7 + dayOfWeek - tmpDate.dayOfWeek()) % 7; 209 iday += (7 + dayOfWeek - tmpDate.dayOfWeek()) % 7;
210 iday += 7 * weekOfMonth; 210 iday += 7 * weekOfMonth;
211 } while (iday > tmpDate.daysInMonth()); 211 } while (iday > tmpDate.daysInMonth());
212 tmpDate = QDate(iyear, imonth, iday); 212 tmpDate = QDate(iyear, imonth, iday);
213 213
214 next = QDateTime(tmpDate, e.start().time()); 214 next = QDateTime(tmpDate, e.start().time());
215 if ((next.date() > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate) 215 if ((next.date() > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate)
216 return FALSE; 216 return FALSE;
217 return TRUE; 217 return TRUE;
218 case Event::MonthlyDate: 218 case Event::MonthlyDate:
219 iday = e.start().date().day(); 219 iday = e.start().date().day();
220 iyear = from.year(); 220 iyear = from.year();
221 imonth = from.month(); 221 imonth = from.month();
222 222
223 a = from.year() - e.start().date().year(); 223 a = from.year() - e.start().date().year();
224 a *= 12; 224 a *= 12;
225 a = a + (imonth - e.start().date().month()); 225 a = a + (imonth - e.start().date().month());
226 /* a is e.start()monthsFrom(from); */ 226 /* a is e.start()monthsFrom(from); */
227 if(a % freq) { 227 if(a % freq) {
228 a = freq - (a % freq); 228 a = freq - (a % freq);
229 imonth = from.month() + a; 229 imonth = from.month() + a;
230 if (imonth > 12) { 230 if (imonth > 12) {
231 imonth--; 231 imonth--;
232 iyear += imonth / 12; 232 iyear += imonth / 12;
233 imonth = imonth % 12; 233 imonth = imonth % 12;
234 imonth++; 234 imonth++;
235 } 235 }
236 } 236 }
237 /* imonth is now the first month after or on 237 /* imonth is now the first month after or on
238 from that matches the frequencey given */ 238 from that matches the frequencey given */
239 239
240 /* this could go for a while, worse case, 4*12 iterations, probably */ 240 /* this could go for a while, worse case, 4*12 iterations, probably */
241 while(!QDate::isValid(iyear, imonth, iday) ) { 241 while(!QDate::isValid(iyear, imonth, iday) ) {
242 imonth += freq; 242 imonth += freq;
243 if (imonth > 12) { 243 if (imonth > 12) {
244 imonth--; 244 imonth--;
245 iyear += imonth / 12; 245 iyear += imonth / 12;
246 imonth = imonth % 12; 246 imonth = imonth % 12;
247 imonth++; 247 imonth++;
248 } 248 }
249 /* these loops could go for a while, check end case now */ 249 /* these loops could go for a while, check end case now */
250 if ((QDate(iyear, imonth, 1) > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate) 250 if ((QDate(iyear, imonth, 1) > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate)
251 return FALSE; 251 return FALSE;
252 } 252 }
253 253
254 if(QDate(iyear, imonth, iday) >= from) { 254 if(QDate(iyear, imonth, iday) >= from) {
255 /* done */ 255 /* done */
256 next = QDateTime(QDate(iyear, imonth, iday), 256 next = QDateTime(QDate(iyear, imonth, iday),
257 e.start().time()); 257 e.start().time());
258 if ((next.date() > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate) 258 if ((next.date() > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate)
259 return FALSE; 259 return FALSE;
260 return TRUE; 260 return TRUE;
261 } 261 }
262 262
263 /* ok, need to cycle */ 263 /* ok, need to cycle */
264 imonth += freq; 264 imonth += freq;
265 imonth--; 265 imonth--;
266 iyear += imonth / 12; 266 iyear += imonth / 12;
267 imonth = imonth % 12; 267 imonth = imonth % 12;
268 imonth++; 268 imonth++;
269 269
270 while(!QDate::isValid(iyear, imonth, iday) ) { 270 while(!QDate::isValid(iyear, imonth, iday) ) {
271 imonth += freq; 271 imonth += freq;
272 imonth--; 272 imonth--;
273 iyear += imonth / 12; 273 iyear += imonth / 12;
274 imonth = imonth % 12; 274 imonth = imonth % 12;
275 imonth++; 275 imonth++;
276 if ((QDate(iyear, imonth, 1) > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate) 276 if ((QDate(iyear, imonth, 1) > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate)
277 return FALSE; 277 return FALSE;
278 } 278 }
279 279
280 next = QDateTime(QDate(iyear, imonth, iday), e.start().time()); 280 next = QDateTime(QDate(iyear, imonth, iday), e.start().time());
281 if ((next.date() > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate) 281 if ((next.date() > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate)
282 return FALSE; 282 return FALSE;
283 return TRUE; 283 return TRUE;
284 case Event::Yearly: 284 case Event::Yearly:
285 iday = e.start().date().day(); 285 iday = e.start().date().day();
286 imonth = e.start().date().month(); 286 imonth = e.start().date().month();
287 iyear = from.year(); // after all, we want to start in this year 287 iyear = from.year(); // after all, we want to start in this year
288 288
289 diff = 1; 289 diff = 1;
290 if(imonth == 2 && iday > 28) { 290 if(imonth == 2 && iday > 28) {
291 /* leap year, and it counts, calculate actual frequency */ 291 /* leap year, and it counts, calculate actual frequency */
292 if(freq % 4) 292 if(freq % 4)
293 if (freq % 2) 293 if (freq % 2)
294 freq = freq * 4; 294 freq = freq * 4;
295 else 295 else
296 freq = freq * 2; 296 freq = freq * 2;
297 /* else divides by 4 already, leave freq alone */ 297 /* else divides by 4 already, leave freq alone */
298 diff = 4; 298 diff = 4;
299 } 299 }
300 300
301 a = from.year() - e.start().date().year(); 301 a = from.year() - e.start().date().year();
302 if(a % freq) { 302 if(a % freq) {
303 a = freq - (a % freq); 303 a = freq - (a % freq);
304 iyear = iyear + a; 304 iyear = iyear + a;
305 } 305 }
306 306
307 /* under the assumption we won't hit one of the special not-leap years twice */ 307 /* under the assumption we won't hit one of the special not-leap years twice */
308 if(!QDate::isValid(iyear, imonth, iday)) { 308 if(!QDate::isValid(iyear, imonth, iday)) {
309 /* must have been skipping by leap years and hit one that wasn't, (e.g. 2100) */ 309 /* must have been skipping by leap years and hit one that wasn't, (e.g. 2100) */
310 iyear += freq; 310 iyear += freq;
311 } 311 }
312 312
313 if(QDate(iyear, imonth, iday) >= from) { 313 if(QDate(iyear, imonth, iday) >= from) {
314 next = QDateTime(QDate(iyear, imonth, iday), 314 next = QDateTime(QDate(iyear, imonth, iday),
315 e.start().time()); 315 e.start().time());
316 if ((next.date() > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate) 316 if ((next.date() > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate)
317 return FALSE; 317 return FALSE;
318 return TRUE; 318 return TRUE;
319 } 319 }
320 /* iyear == from.year(), need to advance again */ 320 /* iyear == from.year(), need to advance again */
321 iyear += freq; 321 iyear += freq;
322 /* under the assumption we won't hit one of the special not-leap years twice */ 322 /* under the assumption we won't hit one of the special not-leap years twice */
323 if(!QDate::isValid(iyear, imonth, iday)) { 323 if(!QDate::isValid(iyear, imonth, iday)) {
324 /* must have been skipping by leap years and hit one that wasn't, (e.g. 2100) */ 324 /* must have been skipping by leap years and hit one that wasn't, (e.g. 2100) */
325 iyear += freq; 325 iyear += freq;
326 } 326 }
327 327
328 next = QDateTime(QDate(iyear, imonth, iday), e.start().time()); 328 next = QDateTime(QDate(iyear, imonth, iday), e.start().time());
329 if ((next.date() > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate) 329 if ((next.date() > e.repeatPattern().endDate()) && e.repeatPattern().hasEndDate)
330 return FALSE; 330 return FALSE;
331 return TRUE; 331 return TRUE;
332 default: 332 default:
333 return FALSE; 333 return FALSE;
334 } 334 }
335} 335}
336 336
337static bool nextAlarm( const Event &ev, QDateTime& when, int& warn) 337static bool nextAlarm( const Event &ev, QDateTime& when, int& warn)
338{ 338{
339 QDateTime now = QDateTime::currentDateTime(); 339 QDateTime now = QDateTime::currentDateTime();
340 if ( ev.hasRepeat() ) { 340 if ( ev.hasRepeat() ) {
341 QDateTime ralarm; 341 QDateTime ralarm;
342 if (nextOccurance(ev, now.date(), ralarm)) { 342 if (nextOccurance(ev, now.date(), ralarm)) {
343 ralarm = ralarm.addSecs(-ev.alarmTime()*60); 343 ralarm = ralarm.addSecs(-ev.alarmTime()*60);
344 if ( ralarm > now ) { 344 if ( ralarm > now ) {
345 when = ralarm; 345 when = ralarm;
346 warn = ev.alarmTime(); 346 warn = ev.alarmTime();
347 } else if ( nextOccurance(ev, now.date().addDays(1), ralarm) ) { 347 } else if ( nextOccurance(ev, now.date().addDays(1), ralarm) ) {
348 ralarm = ralarm.addSecs( -ev.alarmTime()*60 ); 348 ralarm = ralarm.addSecs( -ev.alarmTime()*60 );
349 if ( ralarm > now ) { 349 if ( ralarm > now ) {
350 when = ralarm; 350 when = ralarm;
351 warn = ev.alarmTime(); 351 warn = ev.alarmTime();
352 } 352 }
353 } 353 }
354 } 354 }
355 } else { 355 } else {
356 warn = ev.alarmTime(); 356 warn = ev.alarmTime();
357 when = ev.start().addSecs( -ev.alarmTime()*60 ); 357 when = ev.start().addSecs( -ev.alarmTime()*60 );
358 } 358 }
359 return when > now; 359 return when > now;
360} 360}
361 361
362static void addEventAlarm( const Event &ev ) 362static void addEventAlarm( const Event &ev )
363{ 363{
364 QDateTime when; 364 QDateTime when;
365 int warn; 365 int warn;
366 if ( nextAlarm(ev,when,warn) ) 366 if ( nextAlarm(ev,when,warn) )
367 AlarmServer::addAlarm( when, 367 AlarmServer::addAlarm( when,
368 "QPE/Application/datebook", 368 "QPE/Application/datebook",
369 "alarm(QDateTime,int)", warn ); 369 "alarm(QDateTime,int)", warn );
370} 370}
371 371
372static void delEventAlarm( const Event &ev ) 372static void delEventAlarm( const Event &ev )
373{ 373{
374 QDateTime when; 374 QDateTime when;
375 int warn; 375 int warn;
376 if ( nextAlarm(ev,when,warn) ) 376 if ( nextAlarm(ev,when,warn) )
377 AlarmServer::deleteAlarm( when, 377 AlarmServer::deleteAlarm( when,
378 "QPE/Application/datebook", 378 "QPE/Application/datebook",
379 "alarm(QDateTime,int)", warn ); 379 "alarm(QDateTime,int)", warn );
380} 380}
381 381
382 382
383DateBookDB::DateBookDB() 383DateBookDB::DateBookDB()
384{ 384{
385 init(); 385 init();
386} 386}
387 387
388DateBookDB::~DateBookDB() 388DateBookDB::~DateBookDB()
389{ 389{
390 save(); 390 save();
391 eventList.clear(); 391 eventList.clear();
392 repeatEvents.clear(); 392 repeatEvents.clear();
393
394 delete d;
395 d=0;
393} 396}
394 397
395 398
396//#### Why is this code duplicated in getEffectiveEvents ????? 399//#### Why is this code duplicated in getEffectiveEvents ?????
397//#### Addendum. Don't use this function, lets faze it out if we can. 400//#### Addendum. Don't use this function, lets faze it out if we can.
398QValueList<Event> DateBookDB::getEvents( const QDate &from, const QDate &to ) 401QValueList<Event> DateBookDB::getEvents( const QDate &from, const QDate &to )
399{ 402{
400 QValueList<Event> tmpList; 403 QValueList<Event> tmpList;
401 tmpList = getNonRepeatingEvents( from, to ); 404 tmpList = getNonRepeatingEvents( from, to );
402 405
403 // check for repeating events... 406 // check for repeating events...
404 for (QValueList<Event>::ConstIterator it = repeatEvents.begin(); 407 for (QValueList<Event>::ConstIterator it = repeatEvents.begin();
405 it != repeatEvents.end(); ++it) { 408 it != repeatEvents.end(); ++it) {
406 QDate itDate = from; 409 QDate itDate = from;
407 QDateTime due; 410 QDateTime due;
408 411
409 /* create a false end date, to short circuit on hard 412 /* create a false end date, to short circuit on hard
410 MonthlyDay recurences */ 413 MonthlyDay recurences */
411 Event dummy_event = *it; 414 Event dummy_event = *it;
412 Event::RepeatPattern r = dummy_event.repeatPattern(); 415 Event::RepeatPattern r = dummy_event.repeatPattern();
413 if ( !r.hasEndDate || r.endDate() > to ) { 416 if ( !r.hasEndDate || r.endDate() > to ) {
414 r.setEndDate( to ); 417 r.setEndDate( to );
415 r.hasEndDate = TRUE; 418 r.hasEndDate = TRUE;
416 } 419 }
417 dummy_event.setRepeat(TRUE, r); 420 dummy_event.setRepeat(TRUE, r);
418 421
419 while (nextOccurance(dummy_event, itDate, due)) { 422 while (nextOccurance(dummy_event, itDate, due)) {
420 if (due.date() > to) 423 if (due.date() > to)
421 break; 424 break;
422 Event newEvent = *it; 425 Event newEvent = *it;
423 newEvent.setStart(due); 426 newEvent.setStart(due);
424 newEvent.setEnd(due.addSecs((*it).start().secsTo((*it).end()))); 427 newEvent.setEnd(due.addSecs((*it).start().secsTo((*it).end())));
425 428
426 tmpList.append(newEvent); 429 tmpList.append(newEvent);
427 itDate = due.date().addDays(1); /* the next event */ 430 itDate = due.date().addDays(1); /* the next event */
428 } 431 }
429 } 432 }
430 qHeapSort(tmpList); 433 qHeapSort(tmpList);
431 return tmpList; 434 return tmpList;
432} 435}
433 436
434QValueList<Event> DateBookDB::getEvents( const QDateTime &start ) 437QValueList<Event> DateBookDB::getEvents( const QDateTime &start )
435{ 438{
436 QValueList<Event> day = getEvents(start.date(),start.date()); 439 QValueList<Event> day = getEvents(start.date(),start.date());
437 440
438 QValueListConstIterator<Event> it; 441 QValueListConstIterator<Event> it;
439 QDateTime dtTmp; 442 QDateTime dtTmp;
440 QValueList<Event> tmpList; 443 QValueList<Event> tmpList;
441 for (it = day.begin(); it != day.end(); ++it ) { 444 for (it = day.begin(); it != day.end(); ++it ) {
442 dtTmp = (*it).start(TRUE); 445 dtTmp = (*it).start(TRUE);
443 if ( dtTmp == start ) 446 if ( dtTmp == start )
444 tmpList.append( *it ); 447 tmpList.append( *it );
445 } 448 }
446 return tmpList; 449 return tmpList;
447} 450}
448 451
449//#### Why is this code duplicated in getEvents ????? 452//#### Why is this code duplicated in getEvents ?????
450 453
451QValueList<EffectiveEvent> DateBookDB::getEffectiveEvents( const QDate &from, 454QValueList<EffectiveEvent> DateBookDB::getEffectiveEvents( const QDate &from,
452 const QDate &to ) 455 const QDate &to )
453{ 456{
454 QValueList<EffectiveEvent> tmpList; 457 QValueList<EffectiveEvent> tmpList;
455 QValueListIterator<Event> it; 458 QValueListIterator<Event> it;
456 459
457 EffectiveEvent effEv; 460 EffectiveEvent effEv;
458 QDateTime dtTmp, 461 QDateTime dtTmp,
459 dtEnd; 462 dtEnd;
460 463
461 for (it = eventList.begin(); it != eventList.end(); ++it ) { 464 for (it = eventList.begin(); it != eventList.end(); ++it ) {
462 if (!(*it).isValidUid()) 465 if (!(*it).isValidUid())
463 (*it).assignUid(); // FIXME: Hack to restore cleared uids 466 (*it).assignUid(); // FIXME: Hack to restore cleared uids
464 467
465 dtTmp = (*it).start(TRUE); 468 dtTmp = (*it).start(TRUE);
466 dtEnd = (*it).end(TRUE); 469 dtEnd = (*it).end(TRUE);
467 470
468 if ( dtTmp.date() >= from && dtTmp.date() <= to ) { 471 if ( dtTmp.date() >= from && dtTmp.date() <= to ) {
469 Event tmpEv = *it; 472 Event tmpEv = *it;
470 effEv.setEvent(tmpEv); 473 effEv.setEvent(tmpEv);
471 effEv.setDate( dtTmp.date() ); 474 effEv.setDate( dtTmp.date() );
472 effEv.setStart( dtTmp.time() ); 475 effEv.setStart( dtTmp.time() );
473 if ( dtTmp.date() != dtEnd.date() ) 476 if ( dtTmp.date() != dtEnd.date() )
474 effEv.setEnd( QTime(23, 59, 0) ); 477 effEv.setEnd( QTime(23, 59, 0) );
475 else 478 else
476 effEv.setEnd( dtEnd.time() ); 479 effEv.setEnd( dtEnd.time() );
477 tmpList.append( effEv ); 480 tmpList.append( effEv );
478 } 481 }
479 // we must also check for end date information... 482 // we must also check for end date information...
480 if ( dtEnd.date() != dtTmp.date() && dtEnd.date() >= from ) { 483 if ( dtEnd.date() != dtTmp.date() && dtEnd.date() >= from ) {
481 QDateTime dt = dtTmp.addDays( 1 ); 484 QDateTime dt = dtTmp.addDays( 1 );
482 dt.setTime( QTime(0, 0, 0) ); 485 dt.setTime( QTime(0, 0, 0) );
483 QDateTime dtStop; 486 QDateTime dtStop;
484 if ( dtEnd > to ) { 487 if ( dtEnd > to ) {
485 dtStop = to; 488 dtStop = to;
486 } else 489 } else
487 dtStop = dtEnd; 490 dtStop = dtEnd;
488 while ( dt <= dtStop ) { 491 while ( dt <= dtStop ) {
489 Event tmpEv = *it; 492 Event tmpEv = *it;
490 effEv.setEvent( tmpEv ); 493 effEv.setEvent( tmpEv );
491 effEv.setDate( dt.date() ); 494 effEv.setDate( dt.date() );
492 if ( dt >= from ) { 495 if ( dt >= from ) {
493 effEv.setStart( QTime(0, 0, 0) ); 496 effEv.setStart( QTime(0, 0, 0) );
494 if ( dt.date() == dtEnd.date() ) 497 if ( dt.date() == dtEnd.date() )
495 effEv.setEnd( dtEnd.time() ); 498 effEv.setEnd( dtEnd.time() );
496 else 499 else
497 effEv.setEnd( QTime(23, 59, 59) ); 500 effEv.setEnd( QTime(23, 59, 59) );
498 tmpList.append( effEv ); 501 tmpList.append( effEv );
499 } 502 }
500 dt = dt.addDays( 1 ); 503 dt = dt.addDays( 1 );
501 } 504 }
502 } 505 }
503 } 506 }
504 // check for repeating events... 507 // check for repeating events...
505 QDateTime repeat; 508 QDateTime repeat;
506 for ( it = repeatEvents.begin(); it != repeatEvents.end(); ++it ) { 509 for ( it = repeatEvents.begin(); it != repeatEvents.end(); ++it ) {
507 if (!(*it).isValidUid()) 510 if (!(*it).isValidUid())
508 (*it).assignUid(); // FIXME: Hack to restore cleared uids 511 (*it).assignUid(); // FIXME: Hack to restore cleared uids
509 512
510 /* create a false end date, to short circuit on hard 513 /* create a false end date, to short circuit on hard
511 MonthlyDay recurences */ 514 MonthlyDay recurences */
512 Event dummy_event = *it; 515 Event dummy_event = *it;
513 int duration = (*it).start().date().daysTo( (*it).end().date() ); 516 int duration = (*it).start().date().daysTo( (*it).end().date() );
514 QDate itDate = from.addDays(-duration); 517 QDate itDate = from.addDays(-duration);
515 518
516 Event::RepeatPattern r = dummy_event.repeatPattern(); 519 Event::RepeatPattern r = dummy_event.repeatPattern();
517 if ( !r.hasEndDate || r.endDate() > to ) { 520 if ( !r.hasEndDate || r.endDate() > to ) {
518 r.setEndDate( to ); 521 r.setEndDate( to );
519 r.hasEndDate = TRUE; 522 r.hasEndDate = TRUE;
520 } 523 }
521 dummy_event.setRepeat(TRUE, r); 524 dummy_event.setRepeat(TRUE, r);
522 525
523 while (nextOccurance(dummy_event, itDate, repeat)) { 526 while (nextOccurance(dummy_event, itDate, repeat)) {
524 if(repeat.date() > to) 527 if(repeat.date() > to)
525 break; 528 break;
526 effEv.setDate( repeat.date() ); 529 effEv.setDate( repeat.date() );
527 if ((*it).type() == Event::AllDay) { 530 if ((*it).type() == Event::AllDay) {
528 effEv.setStart( QTime(0,0,0) ); 531 effEv.setStart( QTime(0,0,0) );
529 effEv.setEnd( QTime(23,59,59) ); 532 effEv.setEnd( QTime(23,59,59) );
530 } else { 533 } else {
531 /* we only occur by days, not hours/minutes/seconds. Hence 534 /* we only occur by days, not hours/minutes/seconds. Hence
532 the actual end and start times will be the same for 535 the actual end and start times will be the same for
533 every repeated event. For multi day events this is 536 every repeated event. For multi day events this is
534 fixed up later if on wronge day span */ 537 fixed up later if on wronge day span */
535 effEv.setStart( (*it).start().time() ); 538 effEv.setStart( (*it).start().time() );
536 effEv.setEnd( (*it).end().time() ); 539 effEv.setEnd( (*it).end().time() );
537 } 540 }
538 if ( duration != 0 ) { 541 if ( duration != 0 ) {
539 // multi-day repeating events 542 // multi-day repeating events
540 QDate sub_it = QMAX( repeat.date(), from ); 543 QDate sub_it = QMAX( repeat.date(), from );
541 QDate startDate = repeat.date(); 544 QDate startDate = repeat.date();
542 QDate endDate = startDate.addDays( duration ); 545 QDate endDate = startDate.addDays( duration );
543 546
544 while ( sub_it <= endDate && sub_it <= to ) { 547 while ( sub_it <= endDate && sub_it <= to ) {
545 EffectiveEvent tmpEffEv = effEv; 548 EffectiveEvent tmpEffEv = effEv;
546 Event tmpEv = *it; 549 Event tmpEv = *it;
547 tmpEffEv.setEvent( tmpEv ); 550 tmpEffEv.setEvent( tmpEv );
548 551
549 if ( sub_it != startDate ) 552 if ( sub_it != startDate )
550 tmpEffEv.setStart( QTime(0,0,0) ); 553 tmpEffEv.setStart( QTime(0,0,0) );
551 if ( sub_it != endDate ) 554 if ( sub_it != endDate )
552 tmpEffEv.setEnd( QTime(23,59,59) ); 555 tmpEffEv.setEnd( QTime(23,59,59) );
553 tmpEffEv.setDate( sub_it ); 556 tmpEffEv.setDate( sub_it );
554 tmpEffEv.setEffectiveDates( startDate, endDate ); 557 tmpEffEv.setEffectiveDates( startDate, endDate );
555 tmpList.append( tmpEffEv ); 558 tmpList.append( tmpEffEv );
556 sub_it = sub_it.addDays( 1 ); 559 sub_it = sub_it.addDays( 1 );
557 } 560 }
558 itDate = endDate; 561 itDate = endDate;
559 } else { 562 } else {
560 Event tmpEv = *it; 563 Event tmpEv = *it;
561 effEv.setEvent( tmpEv ); 564 effEv.setEvent( tmpEv );
562 tmpList.append( effEv ); 565 tmpList.append( effEv );
563 itDate = repeat.date().addDays( 1 ); 566 itDate = repeat.date().addDays( 1 );
564 } 567 }
565 } 568 }
566 } 569 }
567 570
568 qHeapSort( tmpList ); 571 qHeapSort( tmpList );
569 return tmpList; 572 return tmpList;
570} 573}
571 574
572QValueList<EffectiveEvent> DateBookDB::getEffectiveEvents( const QDateTime &dt) 575QValueList<EffectiveEvent> DateBookDB::getEffectiveEvents( const QDateTime &dt)
573{ 576{
574 QValueList<EffectiveEvent> day = getEffectiveEvents(dt.date(), dt.date()); 577 QValueList<EffectiveEvent> day = getEffectiveEvents(dt.date(), dt.date());
575 QValueListConstIterator<EffectiveEvent> it; 578 QValueListConstIterator<EffectiveEvent> it;
576 QValueList<EffectiveEvent> tmpList; 579 QValueList<EffectiveEvent> tmpList;
577 QDateTime dtTmp; 580 QDateTime dtTmp;
578 581
579 for (it = day.begin(); it != day.end(); ++it ) { 582 for (it = day.begin(); it != day.end(); ++it ) {
580 dtTmp = QDateTime( (*it).date(), (*it).start() ); 583 dtTmp = QDateTime( (*it).date(), (*it).start() );
581 // at the moment we don't have second granularity, be nice about that.. 584 // at the moment we don't have second granularity, be nice about that..
582 if ( QABS(dt.secsTo(dtTmp)) < 60 ) 585 if ( QABS(dt.secsTo(dtTmp)) < 60 )
583 tmpList.append( *it ); 586 tmpList.append( *it );
584 } 587 }
585 return tmpList; 588 return tmpList;
586} 589}
587 590
588void DateBookDB::addEvent( const Event &ev, bool doalarm ) 591void DateBookDB::addEvent( const Event &ev, bool doalarm )
589{ 592{
590 // write to the journal... 593 // write to the journal...
591 saveJournalEntry( ev, ACTION_ADD, -1, false ); 594 saveJournalEntry( ev, ACTION_ADD, -1, false );
592 addJFEvent( ev, doalarm ); 595 addJFEvent( ev, doalarm );
593 d->clean = false; 596 d->clean = false;
594} 597}
595 598
596void DateBookDB::addJFEvent( const Event &ev, bool doalarm ) 599void DateBookDB::addJFEvent( const Event &ev, bool doalarm )
597{ 600{
598 if ( doalarm && ev.hasAlarm() ) 601 if ( doalarm && ev.hasAlarm() )
599 addEventAlarm( ev ); 602 addEventAlarm( ev );
600 if ( ev.hasRepeat() ) 603 if ( ev.hasRepeat() )
601 repeatEvents.append( ev ); 604 repeatEvents.append( ev );
602 else 605 else
603 eventList.append( ev ); 606 eventList.append( ev );
604} 607}
605 608
606void DateBookDB::editEvent( const Event &old, Event &editedEv ) 609void DateBookDB::editEvent( const Event &old, Event &editedEv )
607{ 610{
608 int oldIndex=0; 611 int oldIndex=0;
609 bool oldHadRepeat = old.hasRepeat(); 612 bool oldHadRepeat = old.hasRepeat();
610 Event orig; 613 Event orig;
611 614
612 // write to the journal... 615 // write to the journal...
613 if ( oldHadRepeat ) { 616 if ( oldHadRepeat ) {
614 if ( origRepeat( old, orig ) ) // should work always... 617 if ( origRepeat( old, orig ) ) // should work always...
615 oldIndex = repeatEvents.findIndex( orig ); 618 oldIndex = repeatEvents.findIndex( orig );
616 } else 619 } else
617 oldIndex = eventList.findIndex( old ); 620 oldIndex = eventList.findIndex( old );
618 saveJournalEntry( editedEv, ACTION_REPLACE, oldIndex, oldHadRepeat ); 621 saveJournalEntry( editedEv, ACTION_REPLACE, oldIndex, oldHadRepeat );
619 622
620 // Delete old event 623 // Delete old event
621 if ( old.hasAlarm() ) 624 if ( old.hasAlarm() )
622 delEventAlarm( old ); 625 delEventAlarm( old );
623 if ( oldHadRepeat ) { 626 if ( oldHadRepeat ) {
624 if ( editedEv.hasRepeat() ) { // This mean that origRepeat was run above and 627 if ( editedEv.hasRepeat() ) { // This mean that origRepeat was run above and
625 // orig is initialized 628 // orig is initialized
626 // assumption, when someone edits a repeating event, they 629 // assumption, when someone edits a repeating event, they
627 // want to change them all, maybe not perfect, but it works 630 // want to change them all, maybe not perfect, but it works
628 // for the moment... 631 // for the moment...
629 repeatEvents.remove( orig ); 632 repeatEvents.remove( orig );
630 } else 633 } else
631 removeRepeat( old ); 634 removeRepeat( old );
632 } else { 635 } else {
633 QValueList<Event>::Iterator it = eventList.find( old ); 636 QValueList<Event>::Iterator it = eventList.find( old );
634 if ( it != eventList.end() ) 637 if ( it != eventList.end() )
635 eventList.remove( it ); 638 eventList.remove( it );
636 } 639 }
637 640
638 // Add new event 641 // Add new event
639 if ( editedEv.hasAlarm() ) 642 if ( editedEv.hasAlarm() )
640 addEventAlarm( editedEv ); 643 addEventAlarm( editedEv );
641 if ( editedEv.hasRepeat() ) 644 if ( editedEv.hasRepeat() )
642 repeatEvents.append( editedEv ); 645 repeatEvents.append( editedEv );
643 else 646 else
644 eventList.append( editedEv ); 647 eventList.append( editedEv );
645 648
646 d->clean = false; 649 d->clean = false;
647} 650}
648 651
649void DateBookDB::removeEvent( const Event &ev ) 652void DateBookDB::removeEvent( const Event &ev )
650{ 653{
651 // write to the journal... 654 // write to the journal...
652 saveJournalEntry( ev, ACTION_REMOVE, -1, false ); 655 saveJournalEntry( ev, ACTION_REMOVE, -1, false );
653 removeJFEvent( ev ); 656 removeJFEvent( ev );
654 d->clean = false; 657 d->clean = false;
655} 658}
656 659
657void DateBookDB::removeJFEvent( const Event&ev ) 660void DateBookDB::removeJFEvent( const Event&ev )
658{ 661{
659 if ( ev.hasAlarm() ) 662 if ( ev.hasAlarm() )
660 delEventAlarm( ev ); 663 delEventAlarm( ev );
661 if ( ev.hasRepeat() ) { 664 if ( ev.hasRepeat() ) {
662 removeRepeat( ev ); 665 removeRepeat( ev );
663 } else { 666 } else {
664 QValueList<Event>::Iterator it = eventList.find( ev ); 667 QValueList<Event>::Iterator it = eventList.find( ev );
665 if ( it != eventList.end() ) 668 if ( it != eventList.end() )
666 eventList.remove( it ); 669 eventList.remove( it );
667 } 670 }
668} 671}
669 672
670// also handles journaling... 673// also handles journaling...
671void DateBookDB::loadFile( const QString &strFile ) 674void DateBookDB::loadFile( const QString &strFile )
672{ 675{
673 676
674 QFile f( strFile ); 677 QFile f( strFile );
675 if ( !f.open( IO_ReadOnly ) ) 678 if ( !f.open( IO_ReadOnly ) )
676 return; 679 return;
677 680
678 enum Attribute { 681 enum Attribute {
679 FDescription = 0, 682 FDescription = 0,
680 FLocation, 683 FLocation,
681 FCategories, 684 FCategories,
682 FUid, 685 FUid,
683 FType, 686 FType,
684 FAlarm, 687 FAlarm,
685 FSound, 688 FSound,
686 FRType, 689 FRType,
687 FRWeekdays, 690 FRWeekdays,
688 FRPosition, 691 FRPosition,
689 FRFreq, 692 FRFreq,
690 FRHasEndDate, 693 FRHasEndDate,
691 FREndDate, 694 FREndDate,
692 FRStart, 695 FRStart,
693 FREnd, 696 FREnd,
694 FNote, 697 FNote,
695 FCreated, 698 FCreated,
696 FAction, 699 FAction,
697 FActionKey, 700 FActionKey,
698 FJournalOrigHadRepeat 701 FJournalOrigHadRepeat
699 }; 702 };
700 703
701 QAsciiDict<int> dict( 97 ); 704 QAsciiDict<int> dict( 97 );
702 dict.setAutoDelete( TRUE ); 705 dict.setAutoDelete( TRUE );
703 dict.insert( "description", new int(FDescription) ); 706 dict.insert( "description", new int(FDescription) );
704 dict.insert( "location", new int(FLocation) ); 707 dict.insert( "location", new int(FLocation) );
705 dict.insert( "categories", new int(FCategories) ); 708 dict.insert( "categories", new int(FCategories) );
706 dict.insert( "uid", new int(FUid) ); 709 dict.insert( "uid", new int(FUid) );
707 dict.insert( "type", new int(FType) ); 710 dict.insert( "type", new int(FType) );
708 dict.insert( "alarm", new int(FAlarm) ); 711 dict.insert( "alarm", new int(FAlarm) );
709 dict.insert( "sound", new int(FSound) ); 712 dict.insert( "sound", new int(FSound) );
710 dict.insert( "rtype", new int(FRType) ); 713 dict.insert( "rtype", new int(FRType) );
711 dict.insert( "rweekdays", new int(FRWeekdays) ); 714 dict.insert( "rweekdays", new int(FRWeekdays) );
712 dict.insert( "rposition", new int(FRPosition) ); 715 dict.insert( "rposition", new int(FRPosition) );
713 dict.insert( "rfreq", new int(FRFreq) ); 716 dict.insert( "rfreq", new int(FRFreq) );
714 dict.insert( "rhasenddate", new int(FRHasEndDate) ); 717 dict.insert( "rhasenddate", new int(FRHasEndDate) );
715 dict.insert( "enddt", new int(FREndDate) ); 718 dict.insert( "enddt", new int(FREndDate) );
716 dict.insert( "start", new int(FRStart) ); 719 dict.insert( "start", new int(FRStart) );
717 dict.insert( "end", new int(FREnd) ); 720 dict.insert( "end", new int(FREnd) );
718 dict.insert( "note", new int(FNote) ); 721 dict.insert( "note", new int(FNote) );
719 dict.insert( "created", new int(FCreated) ); 722 dict.insert( "created", new int(FCreated) );
720 dict.insert( "action", new int(FAction) ); 723 dict.insert( "action", new int(FAction) );
721 dict.insert( "actionkey", new int(FActionKey) ); 724 dict.insert( "actionkey", new int(FActionKey) );
722 dict.insert( "actionorig", new int (FJournalOrigHadRepeat) ); 725 dict.insert( "actionorig", new int (FJournalOrigHadRepeat) );
723 726
724 727
725 QByteArray ba = f.readAll(); 728 QByteArray ba = f.readAll();
726 char* dt = ba.data(); 729 char* dt = ba.data();
727 int len = ba.size(); 730 int len = ba.size();
728 int currentAction, 731 int currentAction,
729 journalKey, 732 journalKey,
730 origHadRepeat; // should be bool, but we need tri-state(not being used) 733 origHadRepeat; // should be bool, but we need tri-state(not being used)
731 734
732 int i = 0; 735 int i = 0;
733 char *point; 736 char *point;
734 // hack to get rid of segfaults after reading </DATEBOOK> 737 // hack to get rid of segfaults after reading </DATEBOOK>
735 while ( (dt+i != 0) && (( point = strstr( dt+i, "<event " ) ) != 0 )) { 738 while ( (dt+i != 0) && (( point = strstr( dt+i, "<event " ) ) != 0 )) {
736 i = point - dt; 739 i = point - dt;
737 // if we are reading in events in the general case, 740 // if we are reading in events in the general case,
738 // we are just adding them, so let the actions represent that... 741 // we are just adding them, so let the actions represent that...
739 currentAction = ACTION_ADD; 742 currentAction = ACTION_ADD;
740 journalKey = -1; 743 journalKey = -1;
741 origHadRepeat = -1; 744 origHadRepeat = -1;
742 // some temporary variables for dates and times ... 745 // some temporary variables for dates and times ...
743 //int startY = 0, startM = 0, startD = 0, starth = 0, startm = 0, starts = 0; 746 //int startY = 0, startM = 0, startD = 0, starth = 0, startm = 0, starts = 0;
744 //int endY = 0, endM = 0, endD = 0, endh = 0, endm = 0, ends = 0; 747 //int endY = 0, endM = 0, endD = 0, endh = 0, endm = 0, ends = 0;
745 //int enddtY = 0, enddtM = 0, enddtD = 0; 748 //int enddtY = 0, enddtM = 0, enddtD = 0;
746 749
747 // ... for the alarm settings ... 750 // ... for the alarm settings ...
748 int alarmTime = -1; Event::SoundTypeChoice alarmSound = Event::Silent; 751 int alarmTime = -1; Event::SoundTypeChoice alarmSound = Event::Silent;
749 // ... and for the recurrence 752 // ... and for the recurrence
750 Event::RepeatPattern rp; 753 Event::RepeatPattern rp;
751 Event e; 754 Event e;
752 755
753 i += 7; 756 i += 7;
754 757
755 while( 1 ) { 758 while( 1 ) {
756 while ( i < len && (dt[i] == ' ' || dt[i] == '\n' || dt[i] == '\r') ) 759 while ( i < len && (dt[i] == ' ' || dt[i] == '\n' || dt[i] == '\r') )
757 ++i; 760 ++i;
758 if ( i >= len-2 || (dt[i] == '/' && dt[i+1] == '>') ) 761 if ( i >= len-2 || (dt[i] == '/' && dt[i+1] == '>') )
759 break; 762 break;
760 // we have another attribute, read it. 763 // we have another attribute, read it.
761 int j = i; 764 int j = i;
762 while ( j < len && dt[j] != '=' ) 765 while ( j < len && dt[j] != '=' )
763 ++j; 766 ++j;
764 char *attr = dt+i; 767 char *attr = dt+i;
765 dt[j] = '\0'; 768 dt[j] = '\0';
766 i = ++j; // skip = 769 i = ++j; // skip =
767 while ( i < len && dt[i] != '"' ) 770 while ( i < len && dt[i] != '"' )
768 ++i; 771 ++i;
769 j = ++i; 772 j = ++i;
770 bool haveAmp = FALSE; 773 bool haveAmp = FALSE;
771 bool haveUtf = FALSE; 774 bool haveUtf = FALSE;
772 while ( j < len && dt[j] != '"' ) { 775 while ( j < len && dt[j] != '"' ) {
773 if ( dt[j] == '&' ) 776 if ( dt[j] == '&' )
774 haveAmp = TRUE; 777 haveAmp = TRUE;
775 if ( ((unsigned char)dt[j]) > 0x7f ) 778 if ( ((unsigned char)dt[j]) > 0x7f )
776 haveUtf = TRUE; 779 haveUtf = TRUE;
777 ++j; 780 ++j;
778 } 781 }
779 782
780 if ( i == j ) { 783 if ( i == j ) {
781 // leave out empty attributes 784 // leave out empty attributes
782 i = j + 1; 785 i = j + 1;
783 continue; 786 continue;
784 } 787 }
785 788
786 QString value = haveUtf ? QString::fromUtf8( dt+i, j-i ) 789 QString value = haveUtf ? QString::fromUtf8( dt+i, j-i )
787 : QString::fromLatin1( dt+i, j-i ); 790 : QString::fromLatin1( dt+i, j-i );
788 if ( haveAmp ) 791 if ( haveAmp )
789 value = Qtopia::plainString( value ); 792 value = Qtopia::plainString( value );
790 i = j + 1; 793 i = j + 1;
791 794
792 //qDebug("attr='%s' value='%s'", attr.data(), value.latin1() ); 795 //qDebug("attr='%s' value='%s'", attr.data(), value.latin1() );
793 int * find = dict[ attr ]; 796 int * find = dict[ attr ];
794#if 1 797#if 1
795 if ( !find ) { 798 if ( !find ) {
796 // custom field 799 // custom field
797 e.setCustomField(attr, value); 800 e.setCustomField(attr, value);
798 continue; 801 continue;
799 } 802 }
800 803
801 switch( *find ) { 804 switch( *find ) {
802 case FDescription: 805 case FDescription:
803 e.setDescription( value ); 806 e.setDescription( value );
804 break; 807 break;
805 case FLocation: 808 case FLocation:
806 e.setLocation( value ); 809 e.setLocation( value );
807 break; 810 break;
808 case FCategories: 811 case FCategories:
809 e.setCategories( Qtopia::Record::idsFromString( value ) ); 812 e.setCategories( Qtopia::Record::idsFromString( value ) );
810 break; 813 break;
811 case FUid: 814 case FUid:
812 e.setUid( value.toInt() ); 815 e.setUid( value.toInt() );
813 break; 816 break;
814 case FType: 817 case FType:
815 if ( value == "AllDay" ) 818 if ( value == "AllDay" )
816 e.setType( Event::AllDay ); 819 e.setType( Event::AllDay );
817 else 820 else
818 e.setType( Event::Normal ); 821 e.setType( Event::Normal );
819 break; 822 break;
820 case FAlarm: 823 case FAlarm:
821 alarmTime = value.toInt(); 824 alarmTime = value.toInt();
822 break; 825 break;
823 case FSound: 826 case FSound:
824 alarmSound = value == "loud" ? Event::Loud : Event::Silent; 827 alarmSound = value == "loud" ? Event::Loud : Event::Silent;
825 break; 828 break;
826 // recurrence stuff 829 // recurrence stuff
827 case FRType: 830 case FRType:
828 if ( value == "Daily" ) 831 if ( value == "Daily" )
829 rp.type = Event::Daily; 832 rp.type = Event::Daily;
830 else if ( value == "Weekly" ) 833 else if ( value == "Weekly" )
831 rp.type = Event::Weekly; 834 rp.type = Event::Weekly;
832 else if ( value == "MonthlyDay" ) 835 else if ( value == "MonthlyDay" )
833 rp.type = Event::MonthlyDay; 836 rp.type = Event::MonthlyDay;
834 else if ( value == "MonthlyDate" ) 837 else if ( value == "MonthlyDate" )
835 rp.type = Event::MonthlyDate; 838 rp.type = Event::MonthlyDate;
836 else if ( value == "Yearly" ) 839 else if ( value == "Yearly" )
837 rp.type = Event::Yearly; 840 rp.type = Event::Yearly;
838 else 841 else
839 rp.type = Event::NoRepeat; 842 rp.type = Event::NoRepeat;
840 break; 843 break;
841 case FRWeekdays: 844 case FRWeekdays:
842 // QtopiaDesktop 1.6 sometimes creates 'rweekdays="0"' 845 // QtopiaDesktop 1.6 sometimes creates 'rweekdays="0"'
843 // when it goes mad. This causes datebook to crash.. (se) 846 // when it goes mad. This causes datebook to crash.. (se)
844 if ( value.toInt() != 0 ) 847 if ( value.toInt() != 0 )
845 rp.days = value.toInt(); 848 rp.days = value.toInt();
846 else 849 else
847 rp.days = 1; 850 rp.days = 1;
848 break; 851 break;
849 case FRPosition: 852 case FRPosition:
850 rp.position = value.toInt(); 853 rp.position = value.toInt();
851 break; 854 break;
852 case FRFreq: 855 case FRFreq:
853 rp.frequency = value.toInt(); 856 rp.frequency = value.toInt();
854 break; 857 break;
855 case FRHasEndDate: 858 case FRHasEndDate:
856 rp.hasEndDate = value.toInt(); 859 rp.hasEndDate = value.toInt();
857 break; 860 break;
858 case FREndDate: { 861 case FREndDate: {
859 rp.endDateUTC = (time_t) value.toLong(); 862 rp.endDateUTC = (time_t) value.toLong();
860 break; 863 break;
861 } 864 }
862 case FRStart: { 865 case FRStart: {
863 e.setStart( (time_t) value.toLong() ); 866 e.setStart( (time_t) value.toLong() );
864 break; 867 break;
865 } 868 }
866 case FREnd: { 869 case FREnd: {
867 e.setEnd( (time_t) value.toLong() ); 870 e.setEnd( (time_t) value.toLong() );
868 break; 871 break;
869 } 872 }
870 case FNote: 873 case FNote:
871 e.setNotes( value ); 874 e.setNotes( value );
872 break; 875 break;
873 case FCreated: 876 case FCreated:
874 rp.createTime = value.toInt(); 877 rp.createTime = value.toInt();
875 break; 878 break;
876 case FAction: 879 case FAction:
877 currentAction = value.toInt(); 880 currentAction = value.toInt();
878 break; 881 break;
879 case FActionKey: 882 case FActionKey:
880 journalKey = value.toInt(); 883 journalKey = value.toInt();
881 break; 884 break;
882 case FJournalOrigHadRepeat: 885 case FJournalOrigHadRepeat:
883 origHadRepeat = value.toInt(); 886 origHadRepeat = value.toInt();
884 break; 887 break;
885 default: 888 default:
886 qDebug( "huh??? missing enum? -- attr.: %s", attr ); 889 qDebug( "huh??? missing enum? -- attr.: %s", attr );
887 break; 890 break;
888 } 891 }
889#endif 892#endif
890 } 893 }
891 // "post processing" (dates, times, alarm, recurrence) 894 // "post processing" (dates, times, alarm, recurrence)
892 895
893 // other half of 1169 fixlet without getting into regression 896 // other half of 1169 fixlet without getting into regression
894 // if rp.days == 0 and rp.type == Event::Weekly 897 // if rp.days == 0 and rp.type == Event::Weekly
895 if ( rp.type == Event::Weekly && rp.days == 0 ) 898 if ( rp.type == Event::Weekly && rp.days == 0 )
896 rp.days = Event::day( e.start().date().dayOfWeek() ); 899 rp.days = Event::day( e.start().date().dayOfWeek() );
897 900
898 901
899 // start date/time 902 // start date/time
900 e.setRepeat( rp.type != Event::NoRepeat, rp ); 903 e.setRepeat( rp.type != Event::NoRepeat, rp );
901 904
902 if ( alarmTime != -1 ) 905 if ( alarmTime != -1 )
903 e.setAlarm( TRUE, alarmTime, alarmSound ); 906 e.setAlarm( TRUE, alarmTime, alarmSound );
904 907