-rw-r--r-- | libopie/pim/otodoaccessxml.cpp | 1 | ||||
-rw-r--r-- | libopie2/opiepim/backend/otodoaccessxml.cpp | 1 |
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 | ||
22 | OTodoAccessXML::OTodoAccessXML( const QString& appName, | 22 | OTodoAccessXML::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 | } |
31 | OTodoAccessXML::~OTodoAccessXML() { | 31 | OTodoAccessXML::~OTodoAccessXML() { |
32 | 32 | ||
33 | } | 33 | } |
34 | bool OTodoAccessXML::load() { | 34 | bool 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 | } |
166 | bool OTodoAccessXML::reload() { | 166 | bool OTodoAccessXML::reload() { |
167 | m_events.clear(); | ||
167 | return load(); | 168 | return load(); |
168 | } | 169 | } |
169 | bool OTodoAccessXML::save() { | 170 | bool 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 | } |
220 | QArray<int> OTodoAccessXML::allRecords()const { | 221 | QArray<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 | } |
231 | QArray<int> OTodoAccessXML::queryByExample( const OTodo&, int ) { | 232 | QArray<int> OTodoAccessXML::queryByExample( const OTodo&, int ) { |
232 | QArray<int> ids(0); | 233 | QArray<int> ids(0); |
233 | return ids; | 234 | return ids; |
234 | } | 235 | } |
235 | OTodo OTodoAccessXML::find( int uid )const { | 236 | OTodo 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 | } |
244 | void OTodoAccessXML::clear() { | 245 | void 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 | } |
250 | bool OTodoAccessXML::add( const OTodo& todo ) { | 251 | bool 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 | } |
257 | bool OTodoAccessXML::remove( int uid ) { | 258 | bool 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 | } |
263 | bool OTodoAccessXML::replace( const OTodo& todo) { | 264 | bool 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 | } |
269 | QArray<int> OTodoAccessXML::effectiveToDos( const QDate& start, | 270 | QArray<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 | } |
291 | QArray<int> OTodoAccessXML::overDue() { | 292 | QArray<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 */ |
308 | void OTodoAccessXML::todo( QAsciiDict<int>* dict, OTodo& ev, | 309 | void 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 | } |
376 | QString OTodoAccessXML::toString( const OTodo& ev )const { | 377 | QString 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 | } |
414 | QString OTodoAccessXML::toString( const QArray<int>& ints ) const { | 415 | QString 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 | ||
423 | struct OTodoXMLContainer { | 424 | struct OTodoXMLContainer { |
424 | OTodo todo; | 425 | OTodo todo; |
425 | }; | 426 | }; |
426 | 427 | ||
427 | namespace { | 428 | namespace { |
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 | */ |
478 | class OTodoXMLVector : public QVector<OTodoXMLContainer> { | 479 | class OTodoXMLVector : public QVector<OTodoXMLContainer> { |
479 | public: | 480 | public: |
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 | ||
591 | QArray<int> OTodoAccessXML::sorted( bool asc, int sortOrder, | 592 | QArray<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 | }; |
643 | void OTodoAccessXML::removeAllCompleted() { | 644 | void 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 | ||
22 | OTodoAccessXML::OTodoAccessXML( const QString& appName, | 22 | OTodoAccessXML::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 | } |
31 | OTodoAccessXML::~OTodoAccessXML() { | 31 | OTodoAccessXML::~OTodoAccessXML() { |
32 | 32 | ||
33 | } | 33 | } |
34 | bool OTodoAccessXML::load() { | 34 | bool 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 | } |
166 | bool OTodoAccessXML::reload() { | 166 | bool OTodoAccessXML::reload() { |
167 | m_events.clear(); | ||
167 | return load(); | 168 | return load(); |
168 | } | 169 | } |
169 | bool OTodoAccessXML::save() { | 170 | bool 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 | } |
220 | QArray<int> OTodoAccessXML::allRecords()const { | 221 | QArray<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 | } |
231 | QArray<int> OTodoAccessXML::queryByExample( const OTodo&, int ) { | 232 | QArray<int> OTodoAccessXML::queryByExample( const OTodo&, int ) { |
232 | QArray<int> ids(0); | 233 | QArray<int> ids(0); |
233 | return ids; | 234 | return ids; |
234 | } | 235 | } |
235 | OTodo OTodoAccessXML::find( int uid )const { | 236 | OTodo 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 | } |
244 | void OTodoAccessXML::clear() { | 245 | void 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 | } |
250 | bool OTodoAccessXML::add( const OTodo& todo ) { | 251 | bool 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 | } |
257 | bool OTodoAccessXML::remove( int uid ) { | 258 | bool 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 | } |
263 | bool OTodoAccessXML::replace( const OTodo& todo) { | 264 | bool 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 | } |
269 | QArray<int> OTodoAccessXML::effectiveToDos( const QDate& start, | 270 | QArray<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 | } |
291 | QArray<int> OTodoAccessXML::overDue() { | 292 | QArray<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 */ |
308 | void OTodoAccessXML::todo( QAsciiDict<int>* dict, OTodo& ev, | 309 | void 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 | } |
376 | QString OTodoAccessXML::toString( const OTodo& ev )const { | 377 | QString 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 | } |
414 | QString OTodoAccessXML::toString( const QArray<int>& ints ) const { | 415 | QString 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 | ||
423 | struct OTodoXMLContainer { | 424 | struct OTodoXMLContainer { |
424 | OTodo todo; | 425 | OTodo todo; |
425 | }; | 426 | }; |
426 | 427 | ||
427 | namespace { | 428 | namespace { |
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 | */ |
478 | class OTodoXMLVector : public QVector<OTodoXMLContainer> { | 479 | class OTodoXMLVector : public QVector<OTodoXMLContainer> { |
479 | public: | 480 | public: |
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 | ||
591 | QArray<int> OTodoAccessXML::sorted( bool asc, int sortOrder, | 592 | QArray<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 | }; |
643 | void OTodoAccessXML::removeAllCompleted() { | 644 | void 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 | } |