summaryrefslogtreecommitdiff
Unidiff
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--libopie/pim/otodoaccessxml.cpp1
-rw-r--r--libopie2/opiepim/backend/otodoaccessxml.cpp1
2 files changed, 2 insertions, 0 deletions
diff --git a/libopie/pim/otodoaccessxml.cpp b/libopie/pim/otodoaccessxml.cpp
index 21f93a0..c3416cb 100644
--- a/libopie/pim/otodoaccessxml.cpp
+++ b/libopie/pim/otodoaccessxml.cpp
@@ -1,648 +1,649 @@
1#include <errno.h> 1#include <errno.h>
2#include <fcntl.h> 2#include <fcntl.h>
3 3
4#include <sys/mman.h> 4#include <sys/mman.h>
5#include <sys/stat.h> 5#include <sys/stat.h>
6#include <sys/types.h> 6#include <sys/types.h>
7 7
8#include <unistd.h> 8#include <unistd.h>
9 9
10 10
11#include <qfile.h> 11#include <qfile.h>
12#include <qvector.h> 12#include <qvector.h>
13 13
14#include <qpe/global.h> 14#include <qpe/global.h>
15#include <qpe/stringutil.h> 15#include <qpe/stringutil.h>
16#include <qpe/timeconversion.h> 16#include <qpe/timeconversion.h>
17 17
18#include <opie/xmltree.h> 18#include <opie/xmltree.h>
19 19
20#include "otodoaccessxml.h" 20#include "otodoaccessxml.h"
21 21
22OTodoAccessXML::OTodoAccessXML( const QString& appName, 22OTodoAccessXML::OTodoAccessXML( const QString& appName,
23 const QString& fileName ) 23 const QString& fileName )
24 : OTodoAccessBackend(), m_app( appName ), m_opened( false ), m_changed( false ) 24 : OTodoAccessBackend(), m_app( appName ), m_opened( false ), m_changed( false )
25{ 25{
26 if (!fileName.isEmpty() ) 26 if (!fileName.isEmpty() )
27 m_file = fileName; 27 m_file = fileName;
28 else 28 else
29 m_file = Global::applicationFileName( "todolist", "todolist.xml" ); 29 m_file = Global::applicationFileName( "todolist", "todolist.xml" );
30} 30}
31OTodoAccessXML::~OTodoAccessXML() { 31OTodoAccessXML::~OTodoAccessXML() {
32 32
33} 33}
34bool OTodoAccessXML::load() { 34bool OTodoAccessXML::load() {
35 m_opened = true; 35 m_opened = true;
36 m_changed = false; 36 m_changed = false;
37 /* initialize dict */ 37 /* initialize dict */
38 /* 38 /*
39 * UPDATE dict if you change anything!!! 39 * UPDATE dict if you change anything!!!
40 */ 40 */
41 QAsciiDict<int> dict(21); 41 QAsciiDict<int> dict(21);
42 dict.setAutoDelete( TRUE ); 42 dict.setAutoDelete( TRUE );
43 dict.insert("Categories" , new int(OTodo::Category) ); 43 dict.insert("Categories" , new int(OTodo::Category) );
44 dict.insert("Uid" , new int(OTodo::Uid) ); 44 dict.insert("Uid" , new int(OTodo::Uid) );
45 dict.insert("HasDate" , new int(OTodo::HasDate) ); 45 dict.insert("HasDate" , new int(OTodo::HasDate) );
46 dict.insert("Completed" , new int(OTodo::Completed) ); 46 dict.insert("Completed" , new int(OTodo::Completed) );
47 dict.insert("Description" , new int(OTodo::Description) ); 47 dict.insert("Description" , new int(OTodo::Description) );
48 dict.insert("Summary" , new int(OTodo::Summary) ); 48 dict.insert("Summary" , new int(OTodo::Summary) );
49 dict.insert("Priority" , new int(OTodo::Priority) ); 49 dict.insert("Priority" , new int(OTodo::Priority) );
50 dict.insert("DateDay" , new int(OTodo::DateDay) ); 50 dict.insert("DateDay" , new int(OTodo::DateDay) );
51 dict.insert("DateMonth" , new int(OTodo::DateMonth) ); 51 dict.insert("DateMonth" , new int(OTodo::DateMonth) );
52 dict.insert("DateYear" , new int(OTodo::DateYear) ); 52 dict.insert("DateYear" , new int(OTodo::DateYear) );
53 dict.insert("Progress" , new int(OTodo::Progress) ); 53 dict.insert("Progress" , new int(OTodo::Progress) );
54 dict.insert("Completed", new int(OTodo::Completed) ); 54 dict.insert("Completed", new int(OTodo::Completed) );
55 dict.insert("CrossReference", new int(OTodo::CrossReference) ); 55 dict.insert("CrossReference", new int(OTodo::CrossReference) );
56 dict.insert("State", new int(OTodo::State) ); 56 dict.insert("State", new int(OTodo::State) );
57 dict.insert("Recurrence", new int(OTodo::Recurrence) ); 57 dict.insert("Recurrence", new int(OTodo::Recurrence) );
58 dict.insert("Alarms", new int(OTodo::Alarms) ); 58 dict.insert("Alarms", new int(OTodo::Alarms) );
59 dict.insert("Reminders", new int(OTodo::Reminders) ); 59 dict.insert("Reminders", new int(OTodo::Reminders) );
60 dict.insert("Notifiers", new int(OTodo::Notifiers) ); 60 dict.insert("Notifiers", new int(OTodo::Notifiers) );
61 dict.insert("Maintainer", new int(OTodo::Maintainer) ); 61 dict.insert("Maintainer", new int(OTodo::Maintainer) );
62 62
63 // here the custom XML parser from TT it's GPL 63 // here the custom XML parser from TT it's GPL
64 // but we want to push OpiePIM... to TT..... 64 // but we want to push OpiePIM... to TT.....
65 // mmap part from zecke :) 65 // mmap part from zecke :)
66 int fd = ::open( QFile::encodeName(m_file).data(), O_RDONLY ); 66 int fd = ::open( QFile::encodeName(m_file).data(), O_RDONLY );
67 struct stat attribut; 67 struct stat attribut;
68 if ( fd < 0 ) return false; 68 if ( fd < 0 ) return false;
69 69
70 if ( fstat(fd, &attribut ) == -1 ) { 70 if ( fstat(fd, &attribut ) == -1 ) {
71 ::close( fd ); 71 ::close( fd );
72 return false; 72 return false;
73 } 73 }
74 void* map_addr = ::mmap(NULL, attribut.st_size, PROT_READ, MAP_SHARED, fd, 0 ); 74 void* map_addr = ::mmap(NULL, attribut.st_size, PROT_READ, MAP_SHARED, fd, 0 );
75 if ( map_addr == ( (caddr_t)-1) ) { 75 if ( map_addr == ( (caddr_t)-1) ) {
76 ::close(fd ); 76 ::close(fd );
77 return false; 77 return false;
78 } 78 }
79 /* advise the kernel who we want to read it */ 79 /* advise the kernel who we want to read it */
80 ::madvise( map_addr, attribut.st_size, MADV_SEQUENTIAL ); 80 ::madvise( map_addr, attribut.st_size, MADV_SEQUENTIAL );
81 /* we do not the file any more */ 81 /* we do not the file any more */
82 ::close( fd ); 82 ::close( fd );
83 83
84 char* dt = (char*)map_addr; 84 char* dt = (char*)map_addr;
85 int len = attribut.st_size; 85 int len = attribut.st_size;
86 int i = 0; 86 int i = 0;
87 char *point; 87 char *point;
88 const char* collectionString = "<Task "; 88 const char* collectionString = "<Task ";
89 int strLen = strlen(collectionString); 89 int strLen = strlen(collectionString);
90 while ( dt+i != 0 && ( point = strstr( dt+i, collectionString ) ) != 0l ) { 90 while ( dt+i != 0 && ( point = strstr( dt+i, collectionString ) ) != 0l ) {
91 i = point -dt; 91 i = point -dt;
92 i+= strLen; 92 i+= strLen;
93 qWarning("Found a start at %d %d", i, (point-dt) ); 93 qWarning("Found a start at %d %d", i, (point-dt) );
94 94
95 OTodo ev; 95 OTodo ev;
96 m_year = m_month = m_day = 0; 96 m_year = m_month = m_day = 0;
97 97
98 while ( TRUE ) { 98 while ( TRUE ) {
99 while ( i < len && (dt[i] == ' ' || dt[i] == '\n' || dt[i] == '\r') ) 99 while ( i < len && (dt[i] == ' ' || dt[i] == '\n' || dt[i] == '\r') )
100 ++i; 100 ++i;
101 if ( i >= len-2 || (dt[i] == '/' && dt[i+1] == '>') ) 101 if ( i >= len-2 || (dt[i] == '/' && dt[i+1] == '>') )
102 break; 102 break;
103 103
104 // we have another attribute, read it. 104 // we have another attribute, read it.
105 int j = i; 105 int j = i;
106 while ( j < len && dt[j] != '=' ) 106 while ( j < len && dt[j] != '=' )
107 ++j; 107 ++j;
108 QCString attr( dt+i, j-i+1); 108 QCString attr( dt+i, j-i+1);
109 109
110 i = ++j; // skip = 110 i = ++j; // skip =
111 111
112 // find the start of quotes 112 // find the start of quotes
113 while ( i < len && dt[i] != '"' ) 113 while ( i < len && dt[i] != '"' )
114 ++i; 114 ++i;
115 j = ++i; 115 j = ++i;
116 116
117 bool haveUtf = FALSE; 117 bool haveUtf = FALSE;
118 bool haveEnt = FALSE; 118 bool haveEnt = FALSE;
119 while ( j < len && dt[j] != '"' ) { 119 while ( j < len && dt[j] != '"' ) {
120 if ( ((unsigned char)dt[j]) > 0x7f ) 120 if ( ((unsigned char)dt[j]) > 0x7f )
121 haveUtf = TRUE; 121 haveUtf = TRUE;
122 if ( dt[j] == '&' ) 122 if ( dt[j] == '&' )
123 haveEnt = TRUE; 123 haveEnt = TRUE;
124 ++j; 124 ++j;
125 } 125 }
126 if ( i == j ) { 126 if ( i == j ) {
127 // empty value 127 // empty value
128 i = j + 1; 128 i = j + 1;
129 continue; 129 continue;
130 } 130 }
131 131
132 QCString value( dt+i, j-i+1 ); 132 QCString value( dt+i, j-i+1 );
133 i = j + 1; 133 i = j + 1;
134 134
135 QString str = (haveUtf ? QString::fromUtf8( value ) 135 QString str = (haveUtf ? QString::fromUtf8( value )
136 : QString::fromLatin1( value ) ); 136 : QString::fromLatin1( value ) );
137 if ( haveEnt ) 137 if ( haveEnt )
138 str = Qtopia::plainString( str ); 138 str = Qtopia::plainString( str );
139 139
140 /* 140 /*
141 * add key + value 141 * add key + value
142 */ 142 */
143 todo( &dict, ev, attr, str ); 143 todo( &dict, ev, attr, str );
144 144
145 } 145 }
146 /* 146 /*
147 * now add it 147 * now add it
148 */ 148 */
149 qWarning("End at %d", i ); 149 qWarning("End at %d", i );
150 if (m_events.contains( ev.uid() ) || ev.uid() == 0) { 150 if (m_events.contains( ev.uid() ) || ev.uid() == 0) {
151 ev.setUid( 1 ); 151 ev.setUid( 1 );
152 m_changed = true; 152 m_changed = true;
153 } 153 }
154 if ( ev.hasDueDate() ) { 154 if ( ev.hasDueDate() ) {
155 ev.setDueDate( QDate(m_year, m_month, m_day) ); 155 ev.setDueDate( QDate(m_year, m_month, m_day) );
156 } 156 }
157 m_events.insert(ev.uid(), ev ); 157 m_events.insert(ev.uid(), ev );
158 m_year = m_month = m_day = -1; 158 m_year = m_month = m_day = -1;
159 } 159 }
160 160
161 munmap(map_addr, attribut.st_size ); 161 munmap(map_addr, attribut.st_size );
162 162
163 qWarning("counts %d records loaded!", m_events.count() ); 163 qWarning("counts %d records loaded!", m_events.count() );
164 return true; 164 return true;
165} 165}
166bool OTodoAccessXML::reload() { 166bool OTodoAccessXML::reload() {
167 m_events.clear();
167 return load(); 168 return load();
168} 169}
169bool OTodoAccessXML::save() { 170bool OTodoAccessXML::save() {
170// qWarning("saving"); 171// qWarning("saving");
171 if (!m_opened || !m_changed ) { 172 if (!m_opened || !m_changed ) {
172// qWarning("not saving"); 173// qWarning("not saving");
173 return true; 174 return true;
174 } 175 }
175 QString strNewFile = m_file + ".new"; 176 QString strNewFile = m_file + ".new";
176 QFile f( strNewFile ); 177 QFile f( strNewFile );
177 if (!f.open( IO_WriteOnly|IO_Raw ) ) 178 if (!f.open( IO_WriteOnly|IO_Raw ) )
178 return false; 179 return false;
179 180
180 int written; 181 int written;
181 QString out; 182 QString out;
182 out = "<!DOCTYPE Tasks>\n<Tasks>\n"; 183 out = "<!DOCTYPE Tasks>\n<Tasks>\n";
183 184
184 // for all todos 185 // for all todos
185 QMap<int, OTodo>::Iterator it; 186 QMap<int, OTodo>::Iterator it;
186 for (it = m_events.begin(); it != m_events.end(); ++it ) { 187 for (it = m_events.begin(); it != m_events.end(); ++it ) {
187 out+= "<Task " + toString( (*it) ) + " />\n"; 188 out+= "<Task " + toString( (*it) ) + " />\n";
188 QCString cstr = out.utf8(); 189 QCString cstr = out.utf8();
189 written = f.writeBlock( cstr.data(), cstr.length() ); 190 written = f.writeBlock( cstr.data(), cstr.length() );
190 191
191 /* less written then we wanted */ 192 /* less written then we wanted */
192 if ( written != (int)cstr.length() ) { 193 if ( written != (int)cstr.length() ) {
193 f.close(); 194 f.close();
194 QFile::remove( strNewFile ); 195 QFile::remove( strNewFile );
195 return false; 196 return false;
196 } 197 }
197 out = QString::null; 198 out = QString::null;
198 } 199 }
199 200
200 out += "</Tasks>"; 201 out += "</Tasks>";
201 QCString cstr = out.utf8(); 202 QCString cstr = out.utf8();
202 written = f.writeBlock( cstr.data(), cstr.length() ); 203 written = f.writeBlock( cstr.data(), cstr.length() );
203 204
204 if ( written != (int)cstr.length() ) { 205 if ( written != (int)cstr.length() ) {
205 f.close(); 206 f.close();
206 QFile::remove( strNewFile ); 207 QFile::remove( strNewFile );
207 return false; 208 return false;
208 } 209 }
209 /* flush before renaming */ 210 /* flush before renaming */
210 f.close(); 211 f.close();
211 212
212 if( ::rename( strNewFile.latin1(), m_file.latin1() ) < 0 ) { 213 if( ::rename( strNewFile.latin1(), m_file.latin1() ) < 0 ) {
213// qWarning("error renaming"); 214// qWarning("error renaming");
214 QFile::remove( strNewFile ); 215 QFile::remove( strNewFile );
215 } 216 }
216 217
217 m_changed = false; 218 m_changed = false;
218 return true; 219 return true;
219} 220}
220QArray<int> OTodoAccessXML::allRecords()const { 221QArray<int> OTodoAccessXML::allRecords()const {
221 QArray<int> ids( m_events.count() ); 222 QArray<int> ids( m_events.count() );
222 QMap<int, OTodo>::ConstIterator it; 223 QMap<int, OTodo>::ConstIterator it;
223 int i = 0; 224 int i = 0;
224 225
225 for ( it = m_events.begin(); it != m_events.end(); ++it ) { 226 for ( it = m_events.begin(); it != m_events.end(); ++it ) {
226 ids[i] = it.key(); 227 ids[i] = it.key();
227 i++; 228 i++;
228 } 229 }
229 return ids; 230 return ids;
230} 231}
231QArray<int> OTodoAccessXML::queryByExample( const OTodo&, int ) { 232QArray<int> OTodoAccessXML::queryByExample( const OTodo&, int ) {
232 QArray<int> ids(0); 233 QArray<int> ids(0);
233 return ids; 234 return ids;
234} 235}
235OTodo OTodoAccessXML::find( int uid )const { 236OTodo OTodoAccessXML::find( int uid )const {
236 OTodo todo; 237 OTodo todo;
237 todo.setUid( 0 ); // isEmpty() 238 todo.setUid( 0 ); // isEmpty()
238 QMap<int, OTodo>::ConstIterator it = m_events.find( uid ); 239 QMap<int, OTodo>::ConstIterator it = m_events.find( uid );
239 if ( it != m_events.end() ) 240 if ( it != m_events.end() )
240 todo = it.data(); 241 todo = it.data();
241 242
242 return todo; 243 return todo;
243} 244}
244void OTodoAccessXML::clear() { 245void OTodoAccessXML::clear() {
245 if (m_opened ) 246 if (m_opened )
246 m_changed = true; 247 m_changed = true;
247 248
248 m_events.clear(); 249 m_events.clear();
249} 250}
250bool OTodoAccessXML::add( const OTodo& todo ) { 251bool OTodoAccessXML::add( const OTodo& todo ) {
251// qWarning("add"); 252// qWarning("add");
252 m_changed = true; 253 m_changed = true;
253 m_events.insert( todo.uid(), todo ); 254 m_events.insert( todo.uid(), todo );
254 255
255 return true; 256 return true;
256} 257}
257bool OTodoAccessXML::remove( int uid ) { 258bool OTodoAccessXML::remove( int uid ) {
258 m_changed = true; 259 m_changed = true;
259 m_events.remove( uid ); 260 m_events.remove( uid );
260 261
261 return true; 262 return true;
262} 263}
263bool OTodoAccessXML::replace( const OTodo& todo) { 264bool OTodoAccessXML::replace( const OTodo& todo) {
264 m_changed = true; 265 m_changed = true;
265 m_events.replace( todo.uid(), todo ); 266 m_events.replace( todo.uid(), todo );
266 267
267 return true; 268 return true;
268} 269}
269QArray<int> OTodoAccessXML::effectiveToDos( const QDate& start, 270QArray<int> OTodoAccessXML::effectiveToDos( const QDate& start,
270 const QDate& end, 271 const QDate& end,
271 bool includeNoDates ) { 272 bool includeNoDates ) {
272 QArray<int> ids( m_events.count() ); 273 QArray<int> ids( m_events.count() );
273 QMap<int, OTodo>::Iterator it; 274 QMap<int, OTodo>::Iterator it;
274 275
275 int i = 0; 276 int i = 0;
276 for ( it = m_events.begin(); it != m_events.end(); ++it ) { 277 for ( it = m_events.begin(); it != m_events.end(); ++it ) {
277 if ( !it.data().hasDueDate() ) { 278 if ( !it.data().hasDueDate() ) {
278 if ( includeNoDates ) { 279 if ( includeNoDates ) {
279 ids[i] = it.key(); 280 ids[i] = it.key();
280 i++; 281 i++;
281 } 282 }
282 }else if ( it.data().dueDate() >= start && 283 }else if ( it.data().dueDate() >= start &&
283 it.data().dueDate() <= end ) { 284 it.data().dueDate() <= end ) {
284 ids[i] = it.key(); 285 ids[i] = it.key();
285 i++; 286 i++;
286 } 287 }
287 } 288 }
288 ids.resize( i ); 289 ids.resize( i );
289 return ids; 290 return ids;
290} 291}
291QArray<int> OTodoAccessXML::overDue() { 292QArray<int> OTodoAccessXML::overDue() {
292 QArray<int> ids( m_events.count() ); 293 QArray<int> ids( m_events.count() );
293 int i = 0; 294 int i = 0;
294 295
295 QMap<int, OTodo>::Iterator it; 296 QMap<int, OTodo>::Iterator it;
296 for ( it = m_events.begin(); it != m_events.end(); ++it ) { 297 for ( it = m_events.begin(); it != m_events.end(); ++it ) {
297 if ( it.data().isOverdue() ) { 298 if ( it.data().isOverdue() ) {
298 ids[i] = it.key(); 299 ids[i] = it.key();
299 i++; 300 i++;
300 } 301 }
301 } 302 }
302 ids.resize( i ); 303 ids.resize( i );
303 return ids; 304 return ids;
304} 305}
305 306
306 307
307/* private */ 308/* private */
308void OTodoAccessXML::todo( QAsciiDict<int>* dict, OTodo& ev, 309void OTodoAccessXML::todo( QAsciiDict<int>* dict, OTodo& ev,
309 const QCString& attr, const QString& val) { 310 const QCString& attr, const QString& val) {
310// qWarning("parse to do from XMLElement" ); 311// qWarning("parse to do from XMLElement" );
311 312
312 int *find=0; 313 int *find=0;
313 314
314 find = (*dict)[ attr.data() ]; 315 find = (*dict)[ attr.data() ];
315 if (!find ) { 316 if (!find ) {
316// qWarning("Unknown option" + it.key() ); 317// qWarning("Unknown option" + it.key() );
317 ev.setCustomField( attr, val ); 318 ev.setCustomField( attr, val );
318 return; 319 return;
319 } 320 }
320 321
321 switch( *find ) { 322 switch( *find ) {
322 case OTodo::Uid: 323 case OTodo::Uid:
323 ev.setUid( val.toInt() ); 324 ev.setUid( val.toInt() );
324 break; 325 break;
325 case OTodo::Category: 326 case OTodo::Category:
326 ev.setCategories( ev.idsFromString( val ) ); 327 ev.setCategories( ev.idsFromString( val ) );
327 break; 328 break;
328 case OTodo::HasDate: 329 case OTodo::HasDate:
329 ev.setHasDueDate( val.toInt() ); 330 ev.setHasDueDate( val.toInt() );
330 break; 331 break;
331 case OTodo::Completed: 332 case OTodo::Completed:
332 ev.setCompleted( val.toInt() ); 333 ev.setCompleted( val.toInt() );
333 break; 334 break;
334 case OTodo::Description: 335 case OTodo::Description:
335 ev.setDescription( val ); 336 ev.setDescription( val );
336 break; 337 break;
337 case OTodo::Summary: 338 case OTodo::Summary:
338 ev.setSummary( val ); 339 ev.setSummary( val );
339 break; 340 break;
340 case OTodo::Priority: 341 case OTodo::Priority:
341 ev.setPriority( val.toInt() ); 342 ev.setPriority( val.toInt() );
342 break; 343 break;
343 case OTodo::DateDay: 344 case OTodo::DateDay:
344 m_day = val.toInt(); 345 m_day = val.toInt();
345 break; 346 break;
346 case OTodo::DateMonth: 347 case OTodo::DateMonth:
347 m_month = val.toInt(); 348 m_month = val.toInt();
348 break; 349 break;
349 case OTodo::DateYear: 350 case OTodo::DateYear:
350 m_year = val.toInt(); 351 m_year = val.toInt();
351 break; 352 break;
352 case OTodo::Progress: 353 case OTodo::Progress:
353 ev.setProgress( val.toInt() ); 354 ev.setProgress( val.toInt() );
354 break; 355 break;
355 case OTodo::CrossReference: 356 case OTodo::CrossReference:
356 { 357 {
357 /* 358 /*
358 * A cross refernce looks like 359 * A cross refernce looks like
359 * appname,id;appname,id 360 * appname,id;appname,id
360 * we need to split it up 361 * we need to split it up
361 */ 362 */
362 QStringList refs = QStringList::split(';', val ); 363 QStringList refs = QStringList::split(';', val );
363 QStringList::Iterator strIt; 364 QStringList::Iterator strIt;
364 for (strIt = refs.begin(); strIt != refs.end(); ++strIt ) { 365 for (strIt = refs.begin(); strIt != refs.end(); ++strIt ) {
365 int pos = (*strIt).find(','); 366 int pos = (*strIt).find(',');
366 if ( pos > -1 ) 367 if ( pos > -1 )
367 ; // ev.addRelation( (*strIt).left(pos), (*strIt).mid(pos+1).toInt() ); 368 ; // ev.addRelation( (*strIt).left(pos), (*strIt).mid(pos+1).toInt() );
368 369
369 } 370 }
370 break; 371 break;
371 } 372 }
372 default: 373 default:
373 break; 374 break;
374 } 375 }
375} 376}
376QString OTodoAccessXML::toString( const OTodo& ev )const { 377QString OTodoAccessXML::toString( const OTodo& ev )const {
377 QString str; 378 QString str;
378 379
379 str += "Completed=\"" + QString::number( ev.isCompleted() ) + "\" "; 380 str += "Completed=\"" + QString::number( ev.isCompleted() ) + "\" ";
380 str += "HasDate=\"" + QString::number( ev.hasDueDate() ) + "\" "; 381 str += "HasDate=\"" + QString::number( ev.hasDueDate() ) + "\" ";
381 str += "Priority=\"" + QString::number( ev.priority() ) + "\" "; 382 str += "Priority=\"" + QString::number( ev.priority() ) + "\" ";
382 str += "Progress=\"" + QString::number(ev.progress() ) + "\" "; 383 str += "Progress=\"" + QString::number(ev.progress() ) + "\" ";
383 384
384 str += "Categories=\"" + toString( ev.categories() ) + "\" "; 385 str += "Categories=\"" + toString( ev.categories() ) + "\" ";
385 str += "Description=\"" + Qtopia::escapeString( ev.description() ) + "\" "; 386 str += "Description=\"" + Qtopia::escapeString( ev.description() ) + "\" ";
386 str += "Summary=\"" + Qtopia::escapeString( ev.summary() ) + "\" "; 387 str += "Summary=\"" + Qtopia::escapeString( ev.summary() ) + "\" ";
387 388
388 if ( ev.hasDueDate() ) { 389 if ( ev.hasDueDate() ) {
389 str += "DateYear=\"" + QString::number( ev.dueDate().year() ) + "\" "; 390 str += "DateYear=\"" + QString::number( ev.dueDate().year() ) + "\" ";
390 str += "DateMonth=\"" + QString::number( ev.dueDate().month() ) + "\" "; 391 str += "DateMonth=\"" + QString::number( ev.dueDate().month() ) + "\" ";
391 str += "DateDay=\"" + QString::number( ev.dueDate().day() ) + "\" "; 392 str += "DateDay=\"" + QString::number( ev.dueDate().day() ) + "\" ";
392 } 393 }
393// qWarning( "Uid %d", ev.uid() ); 394// qWarning( "Uid %d", ev.uid() );
394 str += "Uid=\"" + QString::number( ev.uid() ) + "\" "; 395 str += "Uid=\"" + QString::number( ev.uid() ) + "\" ";
395 396
396// append the extra options 397// append the extra options
397 /* FIXME Qtopia::Record this is currently not 398 /* FIXME Qtopia::Record this is currently not
398 * possible you can set custom fields 399 * possible you can set custom fields
399 * but don' iterate over the list 400 * but don' iterate over the list
400 * I may do #define private protected 401 * I may do #define private protected
401 * for this case - cough --zecke 402 * for this case - cough --zecke
402 */ 403 */
403 /* 404 /*
404 QMap<QString, QString> extras = ev.extras(); 405 QMap<QString, QString> extras = ev.extras();
405 QMap<QString, QString>::Iterator extIt; 406 QMap<QString, QString>::Iterator extIt;
406 for (extIt = extras.begin(); extIt != extras.end(); ++extIt ) 407 for (extIt = extras.begin(); extIt != extras.end(); ++extIt )
407 str += extIt.key() + "=\"" + extIt.data() + "\" "; 408 str += extIt.key() + "=\"" + extIt.data() + "\" ";
408 */ 409 */
409 // cross refernce 410 // cross refernce
410 411
411 412
412 return str; 413 return str;
413} 414}
414QString OTodoAccessXML::toString( const QArray<int>& ints ) const { 415QString OTodoAccessXML::toString( const QArray<int>& ints ) const {
415 return Qtopia::Record::idsToString( ints ); 416 return Qtopia::Record::idsToString( ints );
416} 417}
417 418
418/* internal class for sorting 419/* internal class for sorting
419 * 420 *
420 * Inspired by todoxmlio.cpp from TT 421 * Inspired by todoxmlio.cpp from TT
421 */ 422 */
422 423
423struct OTodoXMLContainer { 424struct OTodoXMLContainer {
424 OTodo todo; 425 OTodo todo;
425}; 426};
426 427
427namespace { 428namespace {
428 inline QString string( const OTodo& todo) { 429 inline QString string( const OTodo& todo) {
429 return todo.summary().isEmpty() ? 430 return todo.summary().isEmpty() ?
430 todo.description().left(20 ) : 431 todo.description().left(20 ) :
431 todo.summary(); 432 todo.summary();
432 } 433 }
433 inline int completed( const OTodo& todo1, const OTodo& todo2) { 434 inline int completed( const OTodo& todo1, const OTodo& todo2) {
434 int ret = 0; 435 int ret = 0;
435 if ( todo1.isCompleted() ) ret++; 436 if ( todo1.isCompleted() ) ret++;
436 if ( todo2.isCompleted() ) ret--; 437 if ( todo2.isCompleted() ) ret--;
437 return ret; 438 return ret;
438 } 439 }
439 inline int priority( const OTodo& t1, const OTodo& t2) { 440 inline int priority( const OTodo& t1, const OTodo& t2) {
440 return ( t1.priority() - t2.priority() ); 441 return ( t1.priority() - t2.priority() );
441 } 442 }
442 inline int description( const OTodo& t1, const OTodo& t2) { 443 inline int description( const OTodo& t1, const OTodo& t2) {
443 return QString::compare( string(t1), string(t2) ); 444 return QString::compare( string(t1), string(t2) );
444 } 445 }
445 inline int deadline( const OTodo& t1, const OTodo& t2) { 446 inline int deadline( const OTodo& t1, const OTodo& t2) {
446 int ret = 0; 447 int ret = 0;
447 if ( t1.hasDueDate() && 448 if ( t1.hasDueDate() &&
448 t2.hasDueDate() ) 449 t2.hasDueDate() )
449 ret = t2.dueDate().daysTo( t1.dueDate() ); 450 ret = t2.dueDate().daysTo( t1.dueDate() );
450 else if ( t1.hasDueDate() ) 451 else if ( t1.hasDueDate() )
451 ret = -1; 452 ret = -1;
452 else if ( t2.hasDueDate() ) 453 else if ( t2.hasDueDate() )
453 ret = 1; 454 ret = 1;
454 else 455 else
455 ret = 0; 456 ret = 0;
456 457
457 return ret; 458 return ret;
458 } 459 }
459 460
460}; 461};
461 462
462/* 463/*
463 * Returns: 464 * Returns:
464 * 0 if item1 == item2 465 * 0 if item1 == item2
465 * 466 *
466 * non-zero if item1 != item2 467 * non-zero if item1 != item2
467 * 468 *
468 * This function returns int rather than bool so that reimplementations 469 * This function returns int rather than bool so that reimplementations
469 * can return one of three values and use it to sort by: 470 * can return one of three values and use it to sort by:
470 * 471 *
471 * 0 if item1 == item2 472 * 0 if item1 == item2
472 * 473 *
473 * > 0 (positive integer) if item1 > item2 474 * > 0 (positive integer) if item1 > item2
474 * 475 *
475 * < 0 (negative integer) if item1 < item2 476 * < 0 (negative integer) if item1 < item2
476 * 477 *
477 */ 478 */
478class OTodoXMLVector : public QVector<OTodoXMLContainer> { 479class OTodoXMLVector : public QVector<OTodoXMLContainer> {
479public: 480public:
480 OTodoXMLVector(int size, bool asc, int sort) 481 OTodoXMLVector(int size, bool asc, int sort)
481 : QVector<OTodoXMLContainer>( size ) 482 : QVector<OTodoXMLContainer>( size )
482 { 483 {
483 setAutoDelete( true ); 484 setAutoDelete( true );
484 m_asc = asc; 485 m_asc = asc;
485 m_sort = sort; 486 m_sort = sort;
486 } 487 }
487 /* return the summary/description */ 488 /* return the summary/description */
488 QString string( const OTodo& todo) { 489 QString string( const OTodo& todo) {
489 return todo.summary().isEmpty() ? 490 return todo.summary().isEmpty() ?
490 todo.description().left(20 ) : 491 todo.description().left(20 ) :
491 todo.summary(); 492 todo.summary();
492 } 493 }
493 /** 494 /**
494 * we take the sortorder( switch on it ) 495 * we take the sortorder( switch on it )
495 * 496 *
496 */ 497 */
497 int compareItems( Item d1, Item d2 ) { 498 int compareItems( Item d1, Item d2 ) {
498 bool seComp, sePrio, seDesc, seDeadline; 499 bool seComp, sePrio, seDesc, seDeadline;
499 seComp = sePrio = seDeadline = seDesc = false; 500 seComp = sePrio = seDeadline = seDesc = false;
500 int ret =0; 501 int ret =0;
501 OTodoXMLContainer* con1 = (OTodoXMLContainer*)d1; 502 OTodoXMLContainer* con1 = (OTodoXMLContainer*)d1;
502 OTodoXMLContainer* con2 = (OTodoXMLContainer*)d2; 503 OTodoXMLContainer* con2 = (OTodoXMLContainer*)d2;
503 504
504 /* same item */ 505 /* same item */
505 if ( con1->todo.uid() == con2->todo.uid() ) 506 if ( con1->todo.uid() == con2->todo.uid() )
506 return 0; 507 return 0;
507 508
508 switch ( m_sort ) { 509 switch ( m_sort ) {
509 /* completed */ 510 /* completed */
510 case 0: { 511 case 0: {
511 ret = completed( con1->todo, con2->todo ); 512 ret = completed( con1->todo, con2->todo );
512 seComp = TRUE; 513 seComp = TRUE;
513 break; 514 break;
514 } 515 }
515 /* priority */ 516 /* priority */
516 case 1: { 517 case 1: {
517 ret = priority( con1->todo, con2->todo ); 518 ret = priority( con1->todo, con2->todo );
518 sePrio = TRUE; 519 sePrio = TRUE;
519 break; 520 break;
520 } 521 }
521 /* description */ 522 /* description */
522 case 2: { 523 case 2: {
523 ret = description( con1->todo, con2->todo ); 524 ret = description( con1->todo, con2->todo );
524 seDesc = TRUE; 525 seDesc = TRUE;
525 break; 526 break;
526 } 527 }
527 /* deadline */ 528 /* deadline */
528 case 3: { 529 case 3: {
529 ret = deadline( con1->todo, con2->todo ); 530 ret = deadline( con1->todo, con2->todo );
530 seDeadline = TRUE; 531 seDeadline = TRUE;
531 break; 532 break;
532 } 533 }
533 default: 534 default:
534 ret = 0; 535 ret = 0;
535 break; 536 break;
536 }; 537 };
537 /* 538 /*
538 * FIXME do better sorting if the first sort criteria 539 * FIXME do better sorting if the first sort criteria
539 * ret equals 0 start with complete and so on... 540 * ret equals 0 start with complete and so on...
540 */ 541 */
541 542
542 /* twist it we're not ascending*/ 543 /* twist it we're not ascending*/
543 if (!m_asc) 544 if (!m_asc)
544 ret = ret * -1; 545 ret = ret * -1;
545 546
546 if ( ret ) 547 if ( ret )
547 return ret; 548 return ret;
548 549
549 // default did not gave difference let's try it other way around 550 // default did not gave difference let's try it other way around
550 /* 551 /*
551 * General try if already checked if not test 552 * General try if already checked if not test
552 * and return 553 * and return
553 * 1.Completed 554 * 1.Completed
554 * 2.Priority 555 * 2.Priority
555 * 3.Description 556 * 3.Description
556 * 4.DueDate 557 * 4.DueDate
557 */ 558 */
558 if (!seComp ) { 559 if (!seComp ) {
559 if ( (ret = completed( con1->todo, con2->todo ) ) ) { 560 if ( (ret = completed( con1->todo, con2->todo ) ) ) {
560 if (!m_asc ) ret *= -1; 561 if (!m_asc ) ret *= -1;
561 return ret; 562 return ret;
562 } 563 }
563 } 564 }
564 if (!sePrio ) { 565 if (!sePrio ) {
565 if ( (ret = priority( con1->todo, con2->todo ) ) ) { 566 if ( (ret = priority( con1->todo, con2->todo ) ) ) {
566 if (!m_asc ) ret *= -1; 567 if (!m_asc ) ret *= -1;
567 return ret; 568 return ret;
568 } 569 }
569 } 570 }
570 if (!seDesc ) { 571 if (!seDesc ) {
571 if ( (ret = description(con1->todo, con2->todo ) ) ) { 572 if ( (ret = description(con1->todo, con2->todo ) ) ) {
572 if (!m_asc) ret *= -1; 573 if (!m_asc) ret *= -1;
573 return ret; 574 return ret;
574 } 575 }
575 } 576 }
576 if (!seDeadline) { 577 if (!seDeadline) {
577 if ( (ret = deadline( con1->todo, con2->todo ) ) ) { 578 if ( (ret = deadline( con1->todo, con2->todo ) ) ) {
578 if (!m_asc) ret *= -1; 579 if (!m_asc) ret *= -1;
579 return ret; 580 return ret;
580 } 581 }
581 } 582 }
582 583
583 return 0; 584 return 0;
584 } 585 }
585 private: 586 private:
586 bool m_asc; 587 bool m_asc;
587 int m_sort; 588 int m_sort;
588 589
589}; 590};
590 591
591QArray<int> OTodoAccessXML::sorted( bool asc, int sortOrder, 592QArray<int> OTodoAccessXML::sorted( bool asc, int sortOrder,
592 int sortFilter, int cat ) { 593 int sortFilter, int cat ) {
593 qWarning("sorted! %d cat", cat); 594 qWarning("sorted! %d cat", cat);
594 OTodoXMLVector vector(m_events.count(), asc,sortOrder ); 595 OTodoXMLVector vector(m_events.count(), asc,sortOrder );
595 QMap<int, OTodo>::Iterator it; 596 QMap<int, OTodo>::Iterator it;
596 int item = 0; 597 int item = 0;
597 598
598 bool bCat = sortFilter & 1 ? true : false; 599 bool bCat = sortFilter & 1 ? true : false;
599 bool bOnly = sortFilter & 2 ? true : false; 600 bool bOnly = sortFilter & 2 ? true : false;
600 bool comp = sortFilter & 4 ? true : false; 601 bool comp = sortFilter & 4 ? true : false;
601 for ( it = m_events.begin(); it != m_events.end(); ++it ) { 602 for ( it = m_events.begin(); it != m_events.end(); ++it ) {
602 603
603 /* show category */ 604 /* show category */
604 if ( bCat && cat != 0) 605 if ( bCat && cat != 0)
605 if (!(*it).categories().contains( cat ) ) { 606 if (!(*it).categories().contains( cat ) ) {
606 qWarning("category mis match"); 607 qWarning("category mis match");
607 continue; 608 continue;
608 } 609 }
609 /* isOverdue but we should not show overdue - why?*/ 610 /* isOverdue but we should not show overdue - why?*/
610/* if ( (*it).isOverdue() && !bOnly ) { 611/* if ( (*it).isOverdue() && !bOnly ) {
611 qWarning("item is overdue but !bOnly"); 612 qWarning("item is overdue but !bOnly");
612 continue; 613 continue;
613 } 614 }
614*/ 615*/
615 if ( !(*it).isOverdue() && bOnly ) { 616 if ( !(*it).isOverdue() && bOnly ) {
616 qWarning("item is not overdue but bOnly checked"); 617 qWarning("item is not overdue but bOnly checked");
617 continue; 618 continue;
618 } 619 }
619 620
620 if ((*it).isCompleted() && comp ) { 621 if ((*it).isCompleted() && comp ) {
621 qWarning("completed continue!"); 622 qWarning("completed continue!");
622 continue; 623 continue;
623 } 624 }
624 625
625 626
626 OTodoXMLContainer* con = new OTodoXMLContainer(); 627 OTodoXMLContainer* con = new OTodoXMLContainer();
627 con->todo = (*it); 628 con->todo = (*it);
628 vector.insert(item, con ); 629 vector.insert(item, con );
629 item++; 630 item++;
630 } 631 }
631 qWarning("XXX %d Items added", item); 632 qWarning("XXX %d Items added", item);
632 vector.resize( item ); 633 vector.resize( item );
633 /* sort it now */ 634 /* sort it now */
634 vector.sort(); 635 vector.sort();
635 /* now get the uids */ 636 /* now get the uids */
636 QArray<int> array( vector.count() ); 637 QArray<int> array( vector.count() );
637 for (uint i= 0; i < vector.count(); i++ ) { 638 for (uint i= 0; i < vector.count(); i++ ) {
638 array[i] = ( vector.at(i) )->todo.uid(); 639 array[i] = ( vector.at(i) )->todo.uid();
639 } 640 }
640 qWarning("array count = %d %d", array.count(), vector.count() ); 641 qWarning("array count = %d %d", array.count(), vector.count() );
641 return array; 642 return array;
642}; 643};
643void OTodoAccessXML::removeAllCompleted() { 644void OTodoAccessXML::removeAllCompleted() {
644 for ( QMap<int, OTodo>::Iterator it = m_events.begin(); it != m_events.end(); ++it ) { 645 for ( QMap<int, OTodo>::Iterator it = m_events.begin(); it != m_events.end(); ++it ) {
645 if ( (*it).isCompleted() ) 646 if ( (*it).isCompleted() )
646 m_events.remove( it ); 647 m_events.remove( it );
647 } 648 }
648} 649}
diff --git a/libopie2/opiepim/backend/otodoaccessxml.cpp b/libopie2/opiepim/backend/otodoaccessxml.cpp
index 21f93a0..c3416cb 100644
--- a/libopie2/opiepim/backend/otodoaccessxml.cpp
+++ b/libopie2/opiepim/backend/otodoaccessxml.cpp
@@ -1,648 +1,649 @@
1#include <errno.h> 1#include <errno.h>
2#include <fcntl.h> 2#include <fcntl.h>
3 3
4#include <sys/mman.h> 4#include <sys/mman.h>
5#include <sys/stat.h> 5#include <sys/stat.h>
6#include <sys/types.h> 6#include <sys/types.h>
7 7
8#include <unistd.h> 8#include <unistd.h>
9 9
10 10
11#include <qfile.h> 11#include <qfile.h>
12#include <qvector.h> 12#include <qvector.h>
13 13
14#include <qpe/global.h> 14#include <qpe/global.h>
15#include <qpe/stringutil.h> 15#include <qpe/stringutil.h>
16#include <qpe/timeconversion.h> 16#include <qpe/timeconversion.h>
17 17
18#include <opie/xmltree.h> 18#include <opie/xmltree.h>
19 19
20#include "otodoaccessxml.h" 20#include "otodoaccessxml.h"
21 21
22OTodoAccessXML::OTodoAccessXML( const QString& appName, 22OTodoAccessXML::OTodoAccessXML( const QString& appName,
23 const QString& fileName ) 23 const QString& fileName )
24 : OTodoAccessBackend(), m_app( appName ), m_opened( false ), m_changed( false ) 24 : OTodoAccessBackend(), m_app( appName ), m_opened( false ), m_changed( false )
25{ 25{
26 if (!fileName.isEmpty() ) 26 if (!fileName.isEmpty() )
27 m_file = fileName; 27 m_file = fileName;
28 else 28 else
29 m_file = Global::applicationFileName( "todolist", "todolist.xml" ); 29 m_file = Global::applicationFileName( "todolist", "todolist.xml" );
30} 30}
31OTodoAccessXML::~OTodoAccessXML() { 31OTodoAccessXML::~OTodoAccessXML() {
32 32
33} 33}
34bool OTodoAccessXML::load() { 34bool OTodoAccessXML::load() {
35 m_opened = true; 35 m_opened = true;
36 m_changed = false; 36 m_changed = false;
37 /* initialize dict */ 37 /* initialize dict */
38 /* 38 /*
39 * UPDATE dict if you change anything!!! 39 * UPDATE dict if you change anything!!!
40 */ 40 */
41 QAsciiDict<int> dict(21); 41 QAsciiDict<int> dict(21);
42 dict.setAutoDelete( TRUE ); 42 dict.setAutoDelete( TRUE );
43 dict.insert("Categories" , new int(OTodo::Category) ); 43 dict.insert("Categories" , new int(OTodo::Category) );
44 dict.insert("Uid" , new int(OTodo::Uid) ); 44 dict.insert("Uid" , new int(OTodo::Uid) );
45 dict.insert("HasDate" , new int(OTodo::HasDate) ); 45 dict.insert("HasDate" , new int(OTodo::HasDate) );
46 dict.insert("Completed" , new int(OTodo::Completed) ); 46 dict.insert("Completed" , new int(OTodo::Completed) );
47 dict.insert("Description" , new int(OTodo::Description) ); 47 dict.insert("Description" , new int(OTodo::Description) );
48 dict.insert("Summary" , new int(OTodo::Summary) ); 48 dict.insert("Summary" , new int(OTodo::Summary) );
49 dict.insert("Priority" , new int(OTodo::Priority) ); 49 dict.insert("Priority" , new int(OTodo::Priority) );
50 dict.insert("DateDay" , new int(OTodo::DateDay) ); 50 dict.insert("DateDay" , new int(OTodo::DateDay) );
51 dict.insert("DateMonth" , new int(OTodo::DateMonth) ); 51 dict.insert("DateMonth" , new int(OTodo::DateMonth) );
52 dict.insert("DateYear" , new int(OTodo::DateYear) ); 52 dict.insert("DateYear" , new int(OTodo::DateYear) );
53 dict.insert("Progress" , new int(OTodo::Progress) ); 53 dict.insert("Progress" , new int(OTodo::Progress) );
54 dict.insert("Completed", new int(OTodo::Completed) ); 54 dict.insert("Completed", new int(OTodo::Completed) );
55 dict.insert("CrossReference", new int(OTodo::CrossReference) ); 55 dict.insert("CrossReference", new int(OTodo::CrossReference) );
56 dict.insert("State", new int(OTodo::State) ); 56 dict.insert("State", new int(OTodo::State) );
57 dict.insert("Recurrence", new int(OTodo::Recurrence) ); 57 dict.insert("Recurrence", new int(OTodo::Recurrence) );
58 dict.insert("Alarms", new int(OTodo::Alarms) ); 58 dict.insert("Alarms", new int(OTodo::Alarms) );
59 dict.insert("Reminders", new int(OTodo::Reminders) ); 59 dict.insert("Reminders", new int(OTodo::Reminders) );
60 dict.insert("Notifiers", new int(OTodo::Notifiers) ); 60 dict.insert("Notifiers", new int(OTodo::Notifiers) );
61 dict.insert("Maintainer", new int(OTodo::Maintainer) ); 61 dict.insert("Maintainer", new int(OTodo::Maintainer) );
62 62
63 // here the custom XML parser from TT it's GPL 63 // here the custom XML parser from TT it's GPL
64 // but we want to push OpiePIM... to TT..... 64 // but we want to push OpiePIM... to TT.....
65 // mmap part from zecke :) 65 // mmap part from zecke :)
66 int fd = ::open( QFile::encodeName(m_file).data(), O_RDONLY ); 66 int fd = ::open( QFile::encodeName(m_file).data(), O_RDONLY );
67 struct stat attribut; 67 struct stat attribut;
68 if ( fd < 0 ) return false; 68 if ( fd < 0 ) return false;
69 69
70 if ( fstat(fd, &attribut ) == -1 ) { 70 if ( fstat(fd, &attribut ) == -1 ) {
71 ::close( fd ); 71 ::close( fd );
72 return false; 72 return false;
73 } 73 }
74 void* map_addr = ::mmap(NULL, attribut.st_size, PROT_READ, MAP_SHARED, fd, 0 ); 74 void* map_addr = ::mmap(NULL, attribut.st_size, PROT_READ, MAP_SHARED, fd, 0 );
75 if ( map_addr == ( (caddr_t)-1) ) { 75 if ( map_addr == ( (caddr_t)-1) ) {
76 ::close(fd ); 76 ::close(fd );
77 return false; 77 return false;
78 } 78 }
79 /* advise the kernel who we want to read it */ 79 /* advise the kernel who we want to read it */
80 ::madvise( map_addr, attribut.st_size, MADV_SEQUENTIAL ); 80 ::madvise( map_addr, attribut.st_size, MADV_SEQUENTIAL );
81 /* we do not the file any more */ 81 /* we do not the file any more */
82 ::close( fd ); 82 ::close( fd );
83 83
84 char* dt = (char*)map_addr; 84 char* dt = (char*)map_addr;
85 int len = attribut.st_size; 85 int len = attribut.st_size;
86 int i = 0; 86 int i = 0;
87 char *point; 87 char *point;
88 const char* collectionString = "<Task "; 88 const char* collectionString = "<Task ";
89 int strLen = strlen(collectionString); 89 int strLen = strlen(collectionString);
90 while ( dt+i != 0 && ( point = strstr( dt+i, collectionString ) ) != 0l ) { 90 while ( dt+i != 0 && ( point = strstr( dt+i, collectionString ) ) != 0l ) {
91 i = point -dt; 91 i = point -dt;
92 i+= strLen; 92 i+= strLen;
93 qWarning("Found a start at %d %d", i, (point-dt) ); 93 qWarning("Found a start at %d %d", i, (point-dt) );
94 94
95 OTodo ev; 95 OTodo ev;
96 m_year = m_month = m_day = 0; 96 m_year = m_month = m_day = 0;
97 97
98 while ( TRUE ) { 98 while ( TRUE ) {
99 while ( i < len && (dt[i] == ' ' || dt[i] == '\n' || dt[i] == '\r') ) 99 while ( i < len && (dt[i] == ' ' || dt[i] == '\n' || dt[i] == '\r') )
100 ++i; 100 ++i;
101 if ( i >= len-2 || (dt[i] == '/' && dt[i+1] == '>') ) 101 if ( i >= len-2 || (dt[i] == '/' && dt[i+1] == '>') )
102 break; 102 break;
103 103
104 // we have another attribute, read it. 104 // we have another attribute, read it.
105 int j = i; 105 int j = i;
106 while ( j < len && dt[j] != '=' ) 106 while ( j < len && dt[j] != '=' )
107 ++j; 107 ++j;
108 QCString attr( dt+i, j-i+1); 108 QCString attr( dt+i, j-i+1);
109 109
110 i = ++j; // skip = 110 i = ++j; // skip =
111 111
112 // find the start of quotes 112 // find the start of quotes
113 while ( i < len && dt[i] != '"' ) 113 while ( i < len && dt[i] != '"' )
114 ++i; 114 ++i;
115 j = ++i; 115 j = ++i;
116 116
117 bool haveUtf = FALSE; 117 bool haveUtf = FALSE;
118 bool haveEnt = FALSE; 118 bool haveEnt = FALSE;
119 while ( j < len && dt[j] != '"' ) { 119 while ( j < len && dt[j] != '"' ) {
120 if ( ((unsigned char)dt[j]) > 0x7f ) 120 if ( ((unsigned char)dt[j]) > 0x7f )
121 haveUtf = TRUE; 121 haveUtf = TRUE;
122 if ( dt[j] == '&' ) 122 if ( dt[j] == '&' )
123 haveEnt = TRUE; 123 haveEnt = TRUE;
124 ++j; 124 ++j;
125 } 125 }
126 if ( i == j ) { 126 if ( i == j ) {
127 // empty value 127 // empty value
128 i = j + 1; 128 i = j + 1;
129 continue; 129 continue;
130 } 130 }
131 131
132 QCString value( dt+i, j-i+1 ); 132 QCString value( dt+i, j-i+1 );
133 i = j + 1; 133 i = j + 1;
134 134
135 QString str = (haveUtf ? QString::fromUtf8( value ) 135 QString str = (haveUtf ? QString::fromUtf8( value )
136 : QString::fromLatin1( value ) ); 136 : QString::fromLatin1( value ) );
137 if ( haveEnt ) 137 if ( haveEnt )
138 str = Qtopia::plainString( str ); 138 str = Qtopia::plainString( str );
139 139
140 /* 140 /*
141 * add key + value 141 * add key + value
142 */ 142 */
143 todo( &dict, ev, attr, str ); 143 todo( &dict, ev, attr, str );
144 144
145 } 145 }
146 /* 146 /*
147 * now add it 147 * now add it
148 */ 148 */
149 qWarning("End at %d", i ); 149 qWarning("End at %d", i );
150 if (m_events.contains( ev.uid() ) || ev.uid() == 0) { 150 if (m_events.contains( ev.uid() ) || ev.uid() == 0) {
151 ev.setUid( 1 ); 151 ev.setUid( 1 );
152 m_changed = true; 152 m_changed = true;
153 } 153 }
154 if ( ev.hasDueDate() ) { 154 if ( ev.hasDueDate() ) {
155 ev.setDueDate( QDate(m_year, m_month, m_day) ); 155 ev.setDueDate( QDate(m_year, m_month, m_day) );
156 } 156 }
157 m_events.insert(ev.uid(), ev ); 157 m_events.insert(ev.uid(), ev );
158 m_year = m_month = m_day = -1; 158 m_year = m_month = m_day = -1;
159 } 159 }
160 160
161 munmap(map_addr, attribut.st_size ); 161 munmap(map_addr, attribut.st_size );
162 162
163 qWarning("counts %d records loaded!", m_events.count() ); 163 qWarning("counts %d records loaded!", m_events.count() );
164 return true; 164 return true;
165} 165}
166bool OTodoAccessXML::reload() { 166bool OTodoAccessXML::reload() {
167 m_events.clear();
167 return load(); 168 return load();
168} 169}
169bool OTodoAccessXML::save() { 170bool OTodoAccessXML::save() {
170// qWarning("saving"); 171// qWarning("saving");
171 if (!m_opened || !m_changed ) { 172 if (!m_opened || !m_changed ) {
172// qWarning("not saving"); 173// qWarning("not saving");
173 return true; 174 return true;
174 } 175 }
175 QString strNewFile = m_file + ".new"; 176 QString strNewFile = m_file + ".new";
176 QFile f( strNewFile ); 177 QFile f( strNewFile );
177 if (!f.open( IO_WriteOnly|IO_Raw ) ) 178 if (!f.open( IO_WriteOnly|IO_Raw ) )
178 return false; 179 return false;
179 180
180 int written; 181 int written;
181 QString out; 182 QString out;
182 out = "<!DOCTYPE Tasks>\n<Tasks>\n"; 183 out = "<!DOCTYPE Tasks>\n<Tasks>\n";
183 184
184 // for all todos 185 // for all todos
185 QMap<int, OTodo>::Iterator it; 186 QMap<int, OTodo>::Iterator it;
186 for (it = m_events.begin(); it != m_events.end(); ++it ) { 187 for (it = m_events.begin(); it != m_events.end(); ++it ) {
187 out+= "<Task " + toString( (*it) ) + " />\n"; 188 out+= "<Task " + toString( (*it) ) + " />\n";
188 QCString cstr = out.utf8(); 189 QCString cstr = out.utf8();
189 written = f.writeBlock( cstr.data(), cstr.length() ); 190 written = f.writeBlock( cstr.data(), cstr.length() );
190 191
191 /* less written then we wanted */ 192 /* less written then we wanted */
192 if ( written != (int)cstr.length() ) { 193 if ( written != (int)cstr.length() ) {
193 f.close(); 194 f.close();
194 QFile::remove( strNewFile ); 195 QFile::remove( strNewFile );
195 return false; 196 return false;
196 } 197 }
197 out = QString::null; 198 out = QString::null;
198 } 199 }
199 200
200 out += "</Tasks>"; 201 out += "</Tasks>";
201 QCString cstr = out.utf8(); 202 QCString cstr = out.utf8();
202 written = f.writeBlock( cstr.data(), cstr.length() ); 203 written = f.writeBlock( cstr.data(), cstr.length() );
203 204
204 if ( written != (int)cstr.length() ) { 205 if ( written != (int)cstr.length() ) {
205 f.close(); 206 f.close();
206 QFile::remove( strNewFile ); 207 QFile::remove( strNewFile );
207 return false; 208 return false;
208 } 209 }
209 /* flush before renaming */ 210 /* flush before renaming */
210 f.close(); 211 f.close();
211 212
212 if( ::rename( strNewFile.latin1(), m_file.latin1() ) < 0 ) { 213 if( ::rename( strNewFile.latin1(), m_file.latin1() ) < 0 ) {
213// qWarning("error renaming"); 214// qWarning("error renaming");
214 QFile::remove( strNewFile ); 215 QFile::remove( strNewFile );
215 } 216 }
216 217
217 m_changed = false; 218 m_changed = false;
218 return true; 219 return true;
219} 220}
220QArray<int> OTodoAccessXML::allRecords()const { 221QArray<int> OTodoAccessXML::allRecords()const {
221 QArray<int> ids( m_events.count() ); 222 QArray<int> ids( m_events.count() );
222 QMap<int, OTodo>::ConstIterator it; 223 QMap<int, OTodo>::ConstIterator it;
223 int i = 0; 224 int i = 0;
224 225
225 for ( it = m_events.begin(); it != m_events.end(); ++it ) { 226 for ( it = m_events.begin(); it != m_events.end(); ++it ) {
226 ids[i] = it.key(); 227 ids[i] = it.key();
227 i++; 228 i++;
228 } 229 }
229 return ids; 230 return ids;
230} 231}
231QArray<int> OTodoAccessXML::queryByExample( const OTodo&, int ) { 232QArray<int> OTodoAccessXML::queryByExample( const OTodo&, int ) {
232 QArray<int> ids(0); 233 QArray<int> ids(0);
233 return ids; 234 return ids;
234} 235}
235OTodo OTodoAccessXML::find( int uid )const { 236OTodo OTodoAccessXML::find( int uid )const {
236 OTodo todo; 237 OTodo todo;
237 todo.setUid( 0 ); // isEmpty() 238 todo.setUid( 0 ); // isEmpty()
238 QMap<int, OTodo>::ConstIterator it = m_events.find( uid ); 239 QMap<int, OTodo>::ConstIterator it = m_events.find( uid );
239 if ( it != m_events.end() ) 240 if ( it != m_events.end() )
240 todo = it.data(); 241 todo = it.data();
241 242
242 return todo; 243 return todo;
243} 244}
244void OTodoAccessXML::clear() { 245void OTodoAccessXML::clear() {
245 if (m_opened ) 246 if (m_opened )
246 m_changed = true; 247 m_changed = true;
247 248
248 m_events.clear(); 249 m_events.clear();
249} 250}
250bool OTodoAccessXML::add( const OTodo& todo ) { 251bool OTodoAccessXML::add( const OTodo& todo ) {
251// qWarning("add"); 252// qWarning("add");
252 m_changed = true; 253 m_changed = true;
253 m_events.insert( todo.uid(), todo ); 254 m_events.insert( todo.uid(), todo );
254 255
255 return true; 256 return true;
256} 257}
257bool OTodoAccessXML::remove( int uid ) { 258bool OTodoAccessXML::remove( int uid ) {
258 m_changed = true; 259 m_changed = true;
259 m_events.remove( uid ); 260 m_events.remove( uid );
260 261
261 return true; 262 return true;
262} 263}
263bool OTodoAccessXML::replace( const OTodo& todo) { 264bool OTodoAccessXML::replace( const OTodo& todo) {
264 m_changed = true; 265 m_changed = true;
265 m_events.replace( todo.uid(), todo ); 266 m_events.replace( todo.uid(), todo );
266 267
267 return true; 268 return true;
268} 269}
269QArray<int> OTodoAccessXML::effectiveToDos( const QDate& start, 270QArray<int> OTodoAccessXML::effectiveToDos( const QDate& start,
270 const QDate& end, 271 const QDate& end,
271 bool includeNoDates ) { 272 bool includeNoDates ) {
272 QArray<int> ids( m_events.count() ); 273 QArray<int> ids( m_events.count() );
273 QMap<int, OTodo>::Iterator it; 274 QMap<int, OTodo>::Iterator it;
274 275
275 int i = 0; 276 int i = 0;
276 for ( it = m_events.begin(); it != m_events.end(); ++it ) { 277 for ( it = m_events.begin(); it != m_events.end(); ++it ) {
277 if ( !it.data().hasDueDate() ) { 278 if ( !it.data().hasDueDate() ) {
278 if ( includeNoDates ) { 279 if ( includeNoDates ) {
279 ids[i] = it.key(); 280 ids[i] = it.key();
280 i++; 281 i++;
281 } 282 }
282 }else if ( it.data().dueDate() >= start && 283 }else if ( it.data().dueDate() >= start &&
283 it.data().dueDate() <= end ) { 284 it.data().dueDate() <= end ) {
284 ids[i] = it.key(); 285 ids[i] = it.key();
285 i++; 286 i++;
286 } 287 }
287 } 288 }
288 ids.resize( i ); 289 ids.resize( i );
289 return ids; 290 return ids;
290} 291}
291QArray<int> OTodoAccessXML::overDue() { 292QArray<int> OTodoAccessXML::overDue() {
292 QArray<int> ids( m_events.count() ); 293 QArray<int> ids( m_events.count() );
293 int i = 0; 294 int i = 0;
294 295
295 QMap<int, OTodo>::Iterator it; 296 QMap<int, OTodo>::Iterator it;
296 for ( it = m_events.begin(); it != m_events.end(); ++it ) { 297 for ( it = m_events.begin(); it != m_events.end(); ++it ) {
297 if ( it.data().isOverdue() ) { 298 if ( it.data().isOverdue() ) {
298 ids[i] = it.key(); 299 ids[i] = it.key();
299 i++; 300 i++;
300 } 301 }
301 } 302 }
302 ids.resize( i ); 303 ids.resize( i );
303 return ids; 304 return ids;
304} 305}
305 306
306 307
307/* private */ 308/* private */
308void OTodoAccessXML::todo( QAsciiDict<int>* dict, OTodo& ev, 309void OTodoAccessXML::todo( QAsciiDict<int>* dict, OTodo& ev,
309 const QCString& attr, const QString& val) { 310 const QCString& attr, const QString& val) {
310// qWarning("parse to do from XMLElement" ); 311// qWarning("parse to do from XMLElement" );
311 312
312 int *find=0; 313 int *find=0;
313 314
314 find = (*dict)[ attr.data() ]; 315 find = (*dict)[ attr.data() ];
315 if (!find ) { 316 if (!find ) {
316// qWarning("Unknown option" + it.key() ); 317// qWarning("Unknown option" + it.key() );
317 ev.setCustomField( attr, val ); 318 ev.setCustomField( attr, val );
318 return; 319 return;
319 } 320 }
320 321
321 switch( *find ) { 322 switch( *find ) {
322 case OTodo::Uid: 323 case OTodo::Uid:
323 ev.setUid( val.toInt() ); 324 ev.setUid( val.toInt() );
324 break; 325 break;
325 case OTodo::Category: 326 case OTodo::Category:
326 ev.setCategories( ev.idsFromString( val ) ); 327 ev.setCategories( ev.idsFromString( val ) );
327 break; 328 break;
328 case OTodo::HasDate: 329 case OTodo::HasDate:
329 ev.setHasDueDate( val.toInt() ); 330 ev.setHasDueDate( val.toInt() );
330 break; 331 break;
331 case OTodo::Completed: 332 case OTodo::Completed:
332 ev.setCompleted( val.toInt() ); 333 ev.setCompleted( val.toInt() );
333 break; 334 break;
334 case OTodo::Description: 335 case OTodo::Description:
335 ev.setDescription( val ); 336 ev.setDescription( val );
336 break; 337 break;
337 case OTodo::Summary: 338 case OTodo::Summary:
338 ev.setSummary( val ); 339 ev.setSummary( val );
339 break; 340 break;
340 case OTodo::Priority: 341 case OTodo::Priority:
341 ev.setPriority( val.toInt() ); 342 ev.setPriority( val.toInt() );
342 break; 343 break;
343 case OTodo::DateDay: 344 case OTodo::DateDay:
344 m_day = val.toInt(); 345 m_day = val.toInt();
345 break; 346 break;
346 case OTodo::DateMonth: 347 case OTodo::DateMonth:
347 m_month = val.toInt(); 348 m_month = val.toInt();
348 break; 349 break;
349 case OTodo::DateYear: 350 case OTodo::DateYear:
350 m_year = val.toInt(); 351 m_year = val.toInt();
351 break; 352 break;
352 case OTodo::Progress: 353 case OTodo::Progress:
353 ev.setProgress( val.toInt() ); 354 ev.setProgress( val.toInt() );
354 break; 355 break;
355 case OTodo::CrossReference: 356 case OTodo::CrossReference:
356 { 357 {
357 /* 358 /*
358 * A cross refernce looks like 359 * A cross refernce looks like
359 * appname,id;appname,id 360 * appname,id;appname,id
360 * we need to split it up 361 * we need to split it up
361 */ 362 */
362 QStringList refs = QStringList::split(';', val ); 363 QStringList refs = QStringList::split(';', val );
363 QStringList::Iterator strIt; 364 QStringList::Iterator strIt;
364 for (strIt = refs.begin(); strIt != refs.end(); ++strIt ) { 365 for (strIt = refs.begin(); strIt != refs.end(); ++strIt ) {
365 int pos = (*strIt).find(','); 366 int pos = (*strIt).find(',');
366 if ( pos > -1 ) 367 if ( pos > -1 )
367 ; // ev.addRelation( (*strIt).left(pos), (*strIt).mid(pos+1).toInt() ); 368 ; // ev.addRelation( (*strIt).left(pos), (*strIt).mid(pos+1).toInt() );
368 369
369 } 370 }
370 break; 371 break;
371 } 372 }
372 default: 373 default:
373 break; 374 break;
374 } 375 }
375} 376}
376QString OTodoAccessXML::toString( const OTodo& ev )const { 377QString OTodoAccessXML::toString( const OTodo& ev )const {
377 QString str; 378 QString str;
378 379
379 str += "Completed=\"" + QString::number( ev.isCompleted() ) + "\" "; 380 str += "Completed=\"" + QString::number( ev.isCompleted() ) + "\" ";
380 str += "HasDate=\"" + QString::number( ev.hasDueDate() ) + "\" "; 381 str += "HasDate=\"" + QString::number( ev.hasDueDate() ) + "\" ";
381 str += "Priority=\"" + QString::number( ev.priority() ) + "\" "; 382 str += "Priority=\"" + QString::number( ev.priority() ) + "\" ";
382 str += "Progress=\"" + QString::number(ev.progress() ) + "\" "; 383 str += "Progress=\"" + QString::number(ev.progress() ) + "\" ";
383 384
384 str += "Categories=\"" + toString( ev.categories() ) + "\" "; 385 str += "Categories=\"" + toString( ev.categories() ) + "\" ";
385 str += "Description=\"" + Qtopia::escapeString( ev.description() ) + "\" "; 386 str += "Description=\"" + Qtopia::escapeString( ev.description() ) + "\" ";
386 str += "Summary=\"" + Qtopia::escapeString( ev.summary() ) + "\" "; 387 str += "Summary=\"" + Qtopia::escapeString( ev.summary() ) + "\" ";
387 388
388 if ( ev.hasDueDate() ) { 389 if ( ev.hasDueDate() ) {
389 str += "DateYear=\"" + QString::number( ev.dueDate().year() ) + "\" "; 390 str += "DateYear=\"" + QString::number( ev.dueDate().year() ) + "\" ";
390 str += "DateMonth=\"" + QString::number( ev.dueDate().month() ) + "\" "; 391 str += "DateMonth=\"" + QString::number( ev.dueDate().month() ) + "\" ";
391 str += "DateDay=\"" + QString::number( ev.dueDate().day() ) + "\" "; 392 str += "DateDay=\"" + QString::number( ev.dueDate().day() ) + "\" ";
392 } 393 }
393// qWarning( "Uid %d", ev.uid() ); 394// qWarning( "Uid %d", ev.uid() );
394 str += "Uid=\"" + QString::number( ev.uid() ) + "\" "; 395 str += "Uid=\"" + QString::number( ev.uid() ) + "\" ";
395 396
396// append the extra options 397// append the extra options
397 /* FIXME Qtopia::Record this is currently not 398 /* FIXME Qtopia::Record this is currently not
398 * possible you can set custom fields 399 * possible you can set custom fields
399 * but don' iterate over the list 400 * but don' iterate over the list
400 * I may do #define private protected 401 * I may do #define private protected
401 * for this case - cough --zecke 402 * for this case - cough --zecke
402 */ 403 */
403 /* 404 /*
404 QMap<QString, QString> extras = ev.extras(); 405 QMap<QString, QString> extras = ev.extras();
405 QMap<QString, QString>::Iterator extIt; 406 QMap<QString, QString>::Iterator extIt;
406 for (extIt = extras.begin(); extIt != extras.end(); ++extIt ) 407 for (extIt = extras.begin(); extIt != extras.end(); ++extIt )
407 str += extIt.key() + "=\"" + extIt.data() + "\" "; 408 str += extIt.key() + "=\"" + extIt.data() + "\" ";
408 */ 409 */
409 // cross refernce 410 // cross refernce
410 411
411 412
412 return str; 413 return str;
413} 414}
414QString OTodoAccessXML::toString( const QArray<int>& ints ) const { 415QString OTodoAccessXML::toString( const QArray<int>& ints ) const {
415 return Qtopia::Record::idsToString( ints ); 416 return Qtopia::Record::idsToString( ints );
416} 417}
417 418
418/* internal class for sorting 419/* internal class for sorting
419 * 420 *
420 * Inspired by todoxmlio.cpp from TT 421 * Inspired by todoxmlio.cpp from TT
421 */ 422 */
422 423
423struct OTodoXMLContainer { 424struct OTodoXMLContainer {
424 OTodo todo; 425 OTodo todo;
425}; 426};
426 427
427namespace { 428namespace {
428 inline QString string( const OTodo& todo) { 429 inline QString string( const OTodo& todo) {
429 return todo.summary().isEmpty() ? 430 return todo.summary().isEmpty() ?
430 todo.description().left(20 ) : 431 todo.description().left(20 ) :
431 todo.summary(); 432 todo.summary();
432 } 433 }
433 inline int completed( const OTodo& todo1, const OTodo& todo2) { 434 inline int completed( const OTodo& todo1, const OTodo& todo2) {
434 int ret = 0; 435 int ret = 0;
435 if ( todo1.isCompleted() ) ret++; 436 if ( todo1.isCompleted() ) ret++;
436 if ( todo2.isCompleted() ) ret--; 437 if ( todo2.isCompleted() ) ret--;
437 return ret; 438 return ret;
438 } 439 }
439 inline int priority( const OTodo& t1, const OTodo& t2) { 440 inline int priority( const OTodo& t1, const OTodo& t2) {
440 return ( t1.priority() - t2.priority() ); 441 return ( t1.priority() - t2.priority() );
441 } 442 }
442 inline int description( const OTodo& t1, const OTodo& t2) { 443 inline int description( const OTodo& t1, const OTodo& t2) {
443 return QString::compare( string(t1), string(t2) ); 444 return QString::compare( string(t1), string(t2) );
444 } 445 }
445 inline int deadline( const OTodo& t1, const OTodo& t2) { 446 inline int deadline( const OTodo& t1, const OTodo& t2) {
446 int ret = 0; 447 int ret = 0;
447 if ( t1.hasDueDate() && 448 if ( t1.hasDueDate() &&
448 t2.hasDueDate() ) 449 t2.hasDueDate() )
449 ret = t2.dueDate().daysTo( t1.dueDate() ); 450 ret = t2.dueDate().daysTo( t1.dueDate() );
450 else if ( t1.hasDueDate() ) 451 else if ( t1.hasDueDate() )
451 ret = -1; 452 ret = -1;
452 else if ( t2.hasDueDate() ) 453 else if ( t2.hasDueDate() )
453 ret = 1; 454 ret = 1;
454 else 455 else
455 ret = 0; 456 ret = 0;
456 457
457 return ret; 458 return ret;
458 } 459 }
459 460
460}; 461};
461 462
462/* 463/*
463 * Returns: 464 * Returns:
464 * 0 if item1 == item2 465 * 0 if item1 == item2
465 * 466 *
466 * non-zero if item1 != item2 467 * non-zero if item1 != item2
467 * 468 *
468 * This function returns int rather than bool so that reimplementations 469 * This function returns int rather than bool so that reimplementations
469 * can return one of three values and use it to sort by: 470 * can return one of three values and use it to sort by:
470 * 471 *
471 * 0 if item1 == item2 472 * 0 if item1 == item2
472 * 473 *
473 * > 0 (positive integer) if item1 > item2 474 * > 0 (positive integer) if item1 > item2
474 * 475 *
475 * < 0 (negative integer) if item1 < item2 476 * < 0 (negative integer) if item1 < item2
476 * 477 *
477 */ 478 */
478class OTodoXMLVector : public QVector<OTodoXMLContainer> { 479class OTodoXMLVector : public QVector<OTodoXMLContainer> {
479public: 480public:
480 OTodoXMLVector(int size, bool asc, int sort) 481 OTodoXMLVector(int size, bool asc, int sort)
481 : QVector<OTodoXMLContainer>( size ) 482 : QVector<OTodoXMLContainer>( size )
482 { 483 {
483 setAutoDelete( true ); 484 setAutoDelete( true );
484 m_asc = asc; 485 m_asc = asc;
485 m_sort = sort; 486 m_sort = sort;
486 } 487 }
487 /* return the summary/description */ 488 /* return the summary/description */
488 QString string( const OTodo& todo) { 489 QString string( const OTodo& todo) {
489 return todo.summary().isEmpty() ? 490 return todo.summary().isEmpty() ?
490 todo.description().left(20 ) : 491 todo.description().left(20 ) :
491 todo.summary(); 492 todo.summary();
492 } 493 }
493 /** 494 /**
494 * we take the sortorder( switch on it ) 495 * we take the sortorder( switch on it )
495 * 496 *
496 */ 497 */
497 int compareItems( Item d1, Item d2 ) { 498 int compareItems( Item d1, Item d2 ) {
498 bool seComp, sePrio, seDesc, seDeadline; 499 bool seComp, sePrio, seDesc, seDeadline;
499 seComp = sePrio = seDeadline = seDesc = false; 500 seComp = sePrio = seDeadline = seDesc = false;
500 int ret =0; 501 int ret =0;
501 OTodoXMLContainer* con1 = (OTodoXMLContainer*)d1; 502 OTodoXMLContainer* con1 = (OTodoXMLContainer*)d1;
502 OTodoXMLContainer* con2 = (OTodoXMLContainer*)d2; 503 OTodoXMLContainer* con2 = (OTodoXMLContainer*)d2;
503 504
504 /* same item */ 505 /* same item */
505 if ( con1->todo.uid() == con2->todo.uid() ) 506 if ( con1->todo.uid() == con2->todo.uid() )
506 return 0; 507 return 0;
507 508
508 switch ( m_sort ) { 509 switch ( m_sort ) {
509 /* completed */ 510 /* completed */
510 case 0: { 511 case 0: {
511 ret = completed( con1->todo, con2->todo ); 512 ret = completed( con1->todo, con2->todo );
512 seComp = TRUE; 513 seComp = TRUE;
513 break; 514 break;
514 } 515 }
515 /* priority */ 516 /* priority */
516 case 1: { 517 case 1: {
517 ret = priority( con1->todo, con2->todo ); 518 ret = priority( con1->todo, con2->todo );
518 sePrio = TRUE; 519 sePrio = TRUE;
519 break; 520 break;
520 } 521 }
521 /* description */ 522 /* description */
522 case 2: { 523 case 2: {
523 ret = description( con1->todo, con2->todo ); 524 ret = description( con1->todo, con2->todo );
524 seDesc = TRUE; 525 seDesc = TRUE;
525 break; 526 break;
526 } 527 }
527 /* deadline */ 528 /* deadline */
528 case 3: { 529 case 3: {
529 ret = deadline( con1->todo, con2->todo ); 530 ret = deadline( con1->todo, con2->todo );
530 seDeadline = TRUE; 531 seDeadline = TRUE;
531 break; 532 break;
532 } 533 }
533 default: 534 default:
534 ret = 0; 535 ret = 0;
535 break; 536 break;
536 }; 537 };
537 /* 538 /*
538 * FIXME do better sorting if the first sort criteria 539 * FIXME do better sorting if the first sort criteria
539 * ret equals 0 start with complete and so on... 540 * ret equals 0 start with complete and so on...
540 */ 541 */
541 542
542 /* twist it we're not ascending*/ 543 /* twist it we're not ascending*/
543 if (!m_asc) 544 if (!m_asc)
544 ret = ret * -1; 545 ret = ret * -1;
545 546
546 if ( ret ) 547 if ( ret )
547 return ret; 548 return ret;
548 549
549 // default did not gave difference let's try it other way around 550 // default did not gave difference let's try it other way around
550 /* 551 /*
551 * General try if already checked if not test 552 * General try if already checked if not test
552 * and return 553 * and return
553 * 1.Completed 554 * 1.Completed
554 * 2.Priority 555 * 2.Priority
555 * 3.Description 556 * 3.Description
556 * 4.DueDate 557 * 4.DueDate
557 */ 558 */
558 if (!seComp ) { 559 if (!seComp ) {
559 if ( (ret = completed( con1->todo, con2->todo ) ) ) { 560 if ( (ret = completed( con1->todo, con2->todo ) ) ) {
560 if (!m_asc ) ret *= -1; 561 if (!m_asc ) ret *= -1;
561 return ret; 562 return ret;
562 } 563 }
563 } 564 }
564 if (!sePrio ) { 565 if (!sePrio ) {
565 if ( (ret = priority( con1->todo, con2->todo ) ) ) { 566 if ( (ret = priority( con1->todo, con2->todo ) ) ) {
566 if (!m_asc ) ret *= -1; 567 if (!m_asc ) ret *= -1;
567 return ret; 568 return ret;
568 } 569 }
569 } 570 }
570 if (!seDesc ) { 571 if (!seDesc ) {
571 if ( (ret = description(con1->todo, con2->todo ) ) ) { 572 if ( (ret = description(con1->todo, con2->todo ) ) ) {
572 if (!m_asc) ret *= -1; 573 if (!m_asc) ret *= -1;
573 return ret; 574 return ret;
574 } 575 }
575 } 576 }
576 if (!seDeadline) { 577 if (!seDeadline) {
577 if ( (ret = deadline( con1->todo, con2->todo ) ) ) { 578 if ( (ret = deadline( con1->todo, con2->todo ) ) ) {
578 if (!m_asc) ret *= -1; 579 if (!m_asc) ret *= -1;
579 return ret; 580 return ret;
580 } 581 }
581 } 582 }
582 583
583 return 0; 584 return 0;
584 } 585 }
585 private: 586 private:
586 bool m_asc; 587 bool m_asc;
587 int m_sort; 588 int m_sort;
588 589
589}; 590};
590 591
591QArray<int> OTodoAccessXML::sorted( bool asc, int sortOrder, 592QArray<int> OTodoAccessXML::sorted( bool asc, int sortOrder,
592 int sortFilter, int cat ) { 593 int sortFilter, int cat ) {
593 qWarning("sorted! %d cat", cat); 594 qWarning("sorted! %d cat", cat);
594 OTodoXMLVector vector(m_events.count(), asc,sortOrder ); 595 OTodoXMLVector vector(m_events.count(), asc,sortOrder );
595 QMap<int, OTodo>::Iterator it; 596 QMap<int, OTodo>::Iterator it;
596 int item = 0; 597 int item = 0;
597 598
598 bool bCat = sortFilter & 1 ? true : false; 599 bool bCat = sortFilter & 1 ? true : false;
599 bool bOnly = sortFilter & 2 ? true : false; 600 bool bOnly = sortFilter & 2 ? true : false;
600 bool comp = sortFilter & 4 ? true : false; 601 bool comp = sortFilter & 4 ? true : false;
601 for ( it = m_events.begin(); it != m_events.end(); ++it ) { 602 for ( it = m_events.begin(); it != m_events.end(); ++it ) {
602 603
603 /* show category */ 604 /* show category */
604 if ( bCat && cat != 0) 605 if ( bCat && cat != 0)
605 if (!(*it).categories().contains( cat ) ) { 606 if (!(*it).categories().contains( cat ) ) {
606 qWarning("category mis match"); 607 qWarning("category mis match");
607 continue; 608 continue;
608 } 609 }
609 /* isOverdue but we should not show overdue - why?*/ 610 /* isOverdue but we should not show overdue - why?*/
610/* if ( (*it).isOverdue() && !bOnly ) { 611/* if ( (*it).isOverdue() && !bOnly ) {
611 qWarning("item is overdue but !bOnly"); 612 qWarning("item is overdue but !bOnly");
612 continue; 613 continue;
613 } 614 }
614*/ 615*/
615 if ( !(*it).isOverdue() && bOnly ) { 616 if ( !(*it).isOverdue() && bOnly ) {
616 qWarning("item is not overdue but bOnly checked"); 617 qWarning("item is not overdue but bOnly checked");
617 continue; 618 continue;
618 } 619 }
619 620
620 if ((*it).isCompleted() && comp ) { 621 if ((*it).isCompleted() && comp ) {
621 qWarning("completed continue!"); 622 qWarning("completed continue!");
622 continue; 623 continue;
623 } 624 }
624 625
625 626
626 OTodoXMLContainer* con = new OTodoXMLContainer(); 627 OTodoXMLContainer* con = new OTodoXMLContainer();
627 con->todo = (*it); 628 con->todo = (*it);
628 vector.insert(item, con ); 629 vector.insert(item, con );
629 item++; 630 item++;
630 } 631 }
631 qWarning("XXX %d Items added", item); 632 qWarning("XXX %d Items added", item);
632 vector.resize( item ); 633 vector.resize( item );
633 /* sort it now */ 634 /* sort it now */
634 vector.sort(); 635 vector.sort();
635 /* now get the uids */ 636 /* now get the uids */
636 QArray<int> array( vector.count() ); 637 QArray<int> array( vector.count() );
637 for (uint i= 0; i < vector.count(); i++ ) { 638 for (uint i= 0; i < vector.count(); i++ ) {
638 array[i] = ( vector.at(i) )->todo.uid(); 639 array[i] = ( vector.at(i) )->todo.uid();
639 } 640 }
640 qWarning("array count = %d %d", array.count(), vector.count() ); 641 qWarning("array count = %d %d", array.count(), vector.count() );
641 return array; 642 return array;
642}; 643};
643void OTodoAccessXML::removeAllCompleted() { 644void OTodoAccessXML::removeAllCompleted() {
644 for ( QMap<int, OTodo>::Iterator it = m_events.begin(); it != m_events.end(); ++it ) { 645 for ( QMap<int, OTodo>::Iterator it = m_events.begin(); it != m_events.end(); ++it ) {
645 if ( (*it).isCompleted() ) 646 if ( (*it).isCompleted() )
646 m_events.remove( it ); 647 m_events.remove( it );
647 } 648 }
648} 649}