summaryrefslogtreecommitdiffabout
Unidiff
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--bin/kdepim/WhatsNew.txt22
-rw-r--r--kmicromail/libetpan/maildir/maildir.c25
-rw-r--r--kmicromail/libetpan/mh/mailmh.c12
-rw-r--r--korganizer/calendarview.cpp4
-rw-r--r--korganizer/kotodoview.cpp1
-rw-r--r--korganizer/kotodoviewitem.cpp2
-rw-r--r--libkcal/sharpformat.cpp7
-rw-r--r--version2
8 files changed, 60 insertions, 15 deletions
diff --git a/bin/kdepim/WhatsNew.txt b/bin/kdepim/WhatsNew.txt
index 4905631..eb7cf13 100644
--- a/bin/kdepim/WhatsNew.txt
+++ b/bin/kdepim/WhatsNew.txt
@@ -1,514 +1,536 @@
1Info about the changes in new versions of KDE-Pim/Pi 1Info about the changes in new versions of KDE-Pim/Pi
2 2
3********** VERSION 1.9.14 ************
4
5Fixed some problems with the dialog sizes when switching
6portrait/landscape mode on 640x480 PDA display.
7
8Fixed some other small bugs.
9
10Fixed an ugly bug in KOpieMail:
11KOpieMail was not able to write files (mails) to MSDOS file system,
12like on an usual preformatted SD card. That should work now.
13To save your mail data on the Sd card do the following:
14Create a dir on the SD card:
15mkdir /mnt/card/localmail
16Go to your home dir:
17cd
18Go to kopiemail data storage dir:
19cd kdepim/apps/kopiemail
20Create a symlink to the SD card:
21ls -s /mnt/card/localmail
22Now KOpieMail will store all mails on the SD card.
23
24
3********** VERSION 1.9.13 ************ 25********** VERSION 1.9.13 ************
4 26
5Fixed nasty PwM/Pi file reading bug, when 27Fixed nasty PwM/Pi file reading bug, when
6the used hash algo of file is different then the global 28the used hash algo of file is different then the global
7hash algo. 29hash algo.
8 30
9Added KA/Pi support for opie mailit mailapplication. 31Added KA/Pi support for opie mailit mailapplication.
10 32
11Fixed some bugs in OM/Pi. 33Fixed some bugs in OM/Pi.
12Now character conversion tables are available for the Zaurus 34Now character conversion tables are available for the Zaurus
13to make OM/Pi working properly. 35to make OM/Pi working properly.
14To get the character conversion in OM/Pi working, please download 36To get the character conversion in OM/Pi working, please download
15at the sourceforge project site the package 37at the sourceforge project site the package
16sr-character-conversion_SharpROM_arm.ipk.zip 38sr-character-conversion_SharpROM_arm.ipk.zip
17(or oz-character-conversion_OZ-gcc3xx_arm.ipk.zip for OZ roms) 39(or oz-character-conversion_OZ-gcc3xx_arm.ipk.zip for OZ roms)
18from the section "general files for KDE/Pim" 40from the section "general files for KDE/Pim"
19Instructions how to install this package are in a ReadMe in this file. 41Instructions how to install this package are in a ReadMe in this file.
20 42
21 43
22Fixed the orientation change problem in KA/Pi when switching 44Fixed the orientation change problem in KA/Pi when switching
23portrait/landscape mode. 45portrait/landscape mode.
24 46
25French translation available for KA/Pi and OM/Pi. 47French translation available for KA/Pi and OM/Pi.
26 48
27Fixed some problems with categories in KO/Pi in DTM sync. 49Fixed some problems with categories in KO/Pi in DTM sync.
28 50
29Added selection dialog for export to phone in KA/Pi. 51Added selection dialog for export to phone in KA/Pi.
30 52
31If in KO/Pi is an attendee selected to add to a meeting and this 53If in KO/Pi is an attendee selected to add to a meeting and this
32attendee is already in the list of attendees, this person is not added 54attendee is already in the list of attendees, this person is not added
33again. 55again.
34 56
35Some menu cleanup in KA/Pi. 57Some menu cleanup in KA/Pi.
36 58
37********** VERSION 1.9.12 ************ 59********** VERSION 1.9.12 ************
38 60
39Fix for the bug in KO/Pi What's Next view of version 1.9.11. 61Fix for the bug in KO/Pi What's Next view of version 1.9.11.
40 62
41Bugfix: Licence file is now shown again. 63Bugfix: Licence file is now shown again.
42 64
43OM/Pi now supports Unicode (utf8 charset). 65OM/Pi now supports Unicode (utf8 charset).
44Fixed some bugs in OM/Pi. 66Fixed some bugs in OM/Pi.
45 67
46KA/Pi has more German translation. 68KA/Pi has more German translation.
47 69
48 70
49********** VERSION 1.9.11 ************ 71********** VERSION 1.9.11 ************
50 72
51Fixed several problems in PWM/Pi, like 73Fixed several problems in PWM/Pi, like
52asking the user, if unsaved changed are pending 74asking the user, if unsaved changed are pending
53when closing the app. 75when closing the app.
54And PwM/Pi handles now different texts for the 76And PwM/Pi handles now different texts for the
55fields Description, Username, Password, configurable per category. 77fields Description, Username, Password, configurable per category.
56 78
57Fixed a crash in KO/Pi , when importing/loading vcs files 79Fixed a crash in KO/Pi , when importing/loading vcs files
58which have an entry with an attendee with state: 80which have an entry with an attendee with state:
59NEEDS ACTION 81NEEDS ACTION
60 82
61Fixed some problems in the German translation of OM/Pi, 83Fixed some problems in the German translation of OM/Pi,
62which makes some dialogs not fitting on the screen 84which makes some dialogs not fitting on the screen
63of the Z 5500. 85of the Z 5500.
64 86
65Fixed Qtopia crash, when disabling/deinstalling 87Fixed Qtopia crash, when disabling/deinstalling
66KO/Pi alarm applet. 88KO/Pi alarm applet.
67 89
68Implemented direct KDE<->KA/Pi sync for KA/Pi running 90Implemented direct KDE<->KA/Pi sync for KA/Pi running
69on Linux desktop. 91on Linux desktop.
70 92
71Added feature "remove sync info" to sync menu. 93Added feature "remove sync info" to sync menu.
72 94
73Tweaked the KO/Pi What's next view a bit, added 95Tweaked the KO/Pi What's next view a bit, added
74setting to hide events that are done. 96setting to hide events that are done.
75 97
76Disabled "beam receive enabled" on startup to 98Disabled "beam receive enabled" on startup to
77avoid problems if Fastload is enabled. 99avoid problems if Fastload is enabled.
78Please set "beam receive enabled", 100Please set "beam receive enabled",
79if you want to receive data via IR. 101if you want to receive data via IR.
80 102
81Fixed bug in direct KDE<->KO/Pi sync for KO/Pi running 103Fixed bug in direct KDE<->KO/Pi sync for KO/Pi running
82on Linux desktop. 104on Linux desktop.
83 105
84Made in KA/Pi scrolling possible, if details view is selected. 106Made in KA/Pi scrolling possible, if details view is selected.
85(The keyboard focus is set automatically to the search line) 107(The keyboard focus is set automatically to the search line)
86 108
87Fixed a bug in DMT sync, that a new entry in DTM was added 109Fixed a bug in DMT sync, that a new entry in DTM was added
88on every sync to Kx/Pi. 110on every sync to Kx/Pi.
89 111
90Fixed missing writing of KA/Pi categories to DMT entries when syncing. 112Fixed missing writing of KA/Pi categories to DMT entries when syncing.
91 113
92Fixed a bug in DMT sync with todos created in KO/Pi containing 114Fixed a bug in DMT sync with todos created in KO/Pi containing
93non-latin1 characters. 115non-latin1 characters.
94 116
95Rearranged package contents of Sharp-ipks and made all 117Rearranged package contents of Sharp-ipks and made all
96packages installable on SD again. 118packages installable on SD again.
97 119
98Fixed the writing of addressbook data in DTM sync. 120Fixed the writing of addressbook data in DTM sync.
99Empty fields in KA/Pi were not removed. 121Empty fields in KA/Pi were not removed.
100 122
101Added better category handling in KA/Pi: 123Added better category handling in KA/Pi:
102Added item 124Added item
103Edit Categories and 125Edit Categories and
104Manage new categories 126Manage new categories
105to the settings menu. 127to the settings menu.
106Possible to configure a view to display categories. 128Possible to configure a view to display categories.
107 129
108Added detailed "KDE Sync Howto" and "Multi Sync Howto" to Help menu. 130Added detailed "KDE Sync Howto" and "Multi Sync Howto" to Help menu.
109 131
110Fixed displaying of "free" days and time in KO Monthview and Agendaview. 132Fixed displaying of "free" days and time in KO Monthview and Agendaview.
111 133
112... and many other bugfixes. 134... and many other bugfixes.
113 135
114********** VERSION 1.9.10 ************ 136********** VERSION 1.9.10 ************
115 137
116Many internal small bugfixes. 138Many internal small bugfixes.
117And fix of the "big" bug in KO/Pi, 139And fix of the "big" bug in KO/Pi,
118that after Syncing the appointments had an offset by several hours. 140that after Syncing the appointments had an offset by several hours.
119That was a problem with the internal timezone setting, 141That was a problem with the internal timezone setting,
120introduced by the changed timezone configuration settings. 142introduced by the changed timezone configuration settings.
121 143
122German translation for OM/Pi is now available. 144German translation for OM/Pi is now available.
123 145
124 146
125********** VERSION 1.9.9 ************ 147********** VERSION 1.9.9 ************
126 148
127KDE-Pim/Pi has a new Member! 149KDE-Pim/Pi has a new Member!
128It is called PWM/Pi (Passwordmanager/platform-independent) 150It is called PWM/Pi (Passwordmanager/platform-independent)
129and it is available for the Zaurus. 151and it is available for the Zaurus.
130It is planned, that it will be available later for Windows. 152It is planned, that it will be available later for Windows.
131(And for Linux, of course). 153(And for Linux, of course).
132It is a port of the Passwordmanager of KDE. 154It is a port of the Passwordmanager of KDE.
133It will need the MicroKDElibs to run. 155It will need the MicroKDElibs to run.
134 156
135Made loading of addressbooks in KA/Pi up to 7 times faster! 157Made loading of addressbooks in KA/Pi up to 7 times faster!
136The bigger your addressbook file, the more starting speed 158The bigger your addressbook file, the more starting speed
137will you gain. (relatively) 159will you gain. (relatively)
138 160
139The Qtopia addressbook connector is now platform independend 161The Qtopia addressbook connector is now platform independend
140as well and should work on any platform for importing/exporting 162as well and should work on any platform for importing/exporting
141Qtopia and Opie XML files. 163Qtopia and Opie XML files.
142 164
143Added a +30min feature to the timezone settings to make 165Added a +30min feature to the timezone settings to make
144KDE-Pim/Pi useable in Australia and other parts on the 166KDE-Pim/Pi useable in Australia and other parts on the
145world with strange timezones ;-) 167world with strange timezones ;-)
146 168
147German "Umlaute" should now be sorted correctly on the Z in KA/Pi. 169German "Umlaute" should now be sorted correctly on the Z in KA/Pi.
148 170
149It is now possible to disable the 171It is now possible to disable the
150"receive data via infrared" feature, such that syncing with 172"receive data via infrared" feature, such that syncing with
151Outlook is now possible again with Kx/Pi runing. 173Outlook is now possible again with Kx/Pi runing.
152Please disable it, before syncing Sharp DTM with Outlook. 174Please disable it, before syncing Sharp DTM with Outlook.
153For your convenience, the "receive data via infrared" feature 175For your convenience, the "receive data via infrared" feature
154is disabled automatically, if you sync Kx/Pi with DTM. 176is disabled automatically, if you sync Kx/Pi with DTM.
155You have to enable it again manually after syncing. 177You have to enable it again manually after syncing.
156Enabling this feature makes it impossible to start the 178Enabling this feature makes it impossible to start the
157Sharp DTM apps. If this feature is enabled, you will only get the 179Sharp DTM apps. If this feature is enabled, you will only get the
158alarm notification from KO/Pi and not from the Sharp calendar. 180alarm notification from KO/Pi and not from the Sharp calendar.
159This is very useful if you sync KO/Pi with Sharp DTM, 181This is very useful if you sync KO/Pi with Sharp DTM,
160because after syncing you usually would get notified about 182because after syncing you usually would get notified about
161an alarm by KO/Pi and the Sharp Calendar. 183an alarm by KO/Pi and the Sharp Calendar.
162 184
163Together with the Linux desktop version of KO/Pi 185Together with the Linux desktop version of KO/Pi
164it is now possible to sync KO/Pi on the Zaurus 186it is now possible to sync KO/Pi on the Zaurus
165with the complete KDE-desktop (3.3 or later) 187with the complete KDE-desktop (3.3 or later)
166calendar data easily. 188calendar data easily.
167That makes it possible to sync the Z with one 189That makes it possible to sync the Z with one
168click of a mouse with the KDE-Desktop. 190click of a mouse with the KDE-Desktop.
169This feature it available for all Zaurus platforms KO/Pi 191This feature it available for all Zaurus platforms KO/Pi
170is running on. 192is running on.
171The only thing needed is a running KO/Pi on Linux and 193The only thing needed is a running KO/Pi on Linux and
172a compiled version of the small 194a compiled version of the small
173KDE-Pim/Pi<->KDE-Desktop access command line program, 195KDE-Pim/Pi<->KDE-Desktop access command line program,
174which is in the KDE-Pim/Pi sources available. 196which is in the KDE-Pim/Pi sources available.
175 197
176The "KDE-desktop" syncing feature for KA/Pi will follow 198The "KDE-desktop" syncing feature for KA/Pi will follow
177in the next releases. 199in the next releases.
178 200
179Fixed the vcard export bug, which had the version 1.9.8. 201Fixed the vcard export bug, which had the version 1.9.8.
180 202
181Added missing GERMAN translation to KO/Pi. 203Added missing GERMAN translation to KO/Pi.
182Hi PsionX, could you add the missing french translation?Thx! 204Hi PsionX, could you add the missing french translation?Thx!
183 205
184Translation files for KA/Pi are available as well. 206Translation files for KA/Pi are available as well.
185GERMAN translation will be available in the next release. 207GERMAN translation will be available in the next release.
186PsionX ( yres, you again ;-) ), could you start translating 208PsionX ( yres, you again ;-) ), could you start translating
187KA/Pi? Thx! 209KA/Pi? Thx!
188 210
189You can download the version 1.9.9 at 211You can download the version 1.9.9 at
190 212
191http://sourceforge.net/project/showfiles.php?group_id=104103&package_id=112604 213http://sourceforge.net/project/showfiles.php?group_id=104103&package_id=112604
192 214
193Note: 215Note:
194To run the mail program OM/Pi you need libopenssl. 216To run the mail program OM/Pi you need libopenssl.
195A link to a download loaction is available at 217A link to a download loaction is available at
196ZSI at www.killefiz.de 218ZSI at www.killefiz.de
197 219
198 220
199********** VERSION 1.9.8 ************ 221********** VERSION 1.9.8 ************
200 222
201Fixed character decoding in OM/Pi. 223Fixed character decoding in OM/Pi.
202(e.g. German "Umlaute" were not displayed properly.) 224(e.g. German "Umlaute" were not displayed properly.)
203 225
204Made is possible to reparent todos in KO/Pi. 226Made is possible to reparent todos in KO/Pi.
205Use contextmenu or keys (look at Help-Keys + Colors) for that. 227Use contextmenu or keys (look at Help-Keys + Colors) for that.
206 228
207Added the missing Sync-Howto and WhatsNew to the packages. 229Added the missing Sync-Howto and WhatsNew to the packages.
208 230
209KO/Pi on Linux desktop can now sync with KDE desktop. 231KO/Pi on Linux desktop can now sync with KDE desktop.
210That means: When using KO/Pi on Linux desktop for syncing with 232That means: When using KO/Pi on Linux desktop for syncing with
211KDE desktop and the Zaurus, the Zaurus can be synced now 233KDE desktop and the Zaurus, the Zaurus can be synced now
212with all KDE-Calendar resources, not only with one local file. 234with all KDE-Calendar resources, not only with one local file.
213(That makes it possible to sync the Zaurus with the 235(That makes it possible to sync the Zaurus with the
214calendar data on a Kolab server) 236calendar data on a Kolab server)
215 237
216KA/Pi syncing with KDE desktop will be available in the next version. 238KA/Pi syncing with KDE desktop will be available in the next version.
217 239
218 240
219********** VERSION 1.9.7 ************ 241********** VERSION 1.9.7 ************
220 242
221KO/Pi - KA/Pi on Windows: 243KO/Pi - KA/Pi on Windows:
222Now a directory can be defined by the user, where the 244Now a directory can be defined by the user, where the
223application/config data should be saved. 245application/config data should be saved.
224 Define your desired path in the evironment variable 246 Define your desired path in the evironment variable
225 MICROKDEHOME 247 MICROKDEHOME
226 before starting KO/Pi or KA/Pi. 248 before starting KO/Pi or KA/Pi.
227 249
228An easy Kx/Pi to Kx/Pi syncing is now possible 250An easy Kx/Pi to Kx/Pi syncing is now possible
229(it is called Pi-Sync) via network. 251(it is called Pi-Sync) via network.
230Please look at the Sync Howto. 252Please look at the Sync Howto.
231 253
232Exporting of calendar data and contacts to mobile phones is now possible. 254Exporting of calendar data and contacts to mobile phones is now possible.
233The SyncHowto is updated with information howto 255The SyncHowto is updated with information howto
234access/sync mobile phones. 256access/sync mobile phones.
235Please look at the Sync Howto. 257Please look at the Sync Howto.
236 258
237Now KO/Pi and KA/Pi on the Zaurus can receive data via infrared directly. 259Now KO/Pi and KA/Pi on the Zaurus can receive data via infrared directly.
238Please disable Fastload for the original contact/calendar applications 260Please disable Fastload for the original contact/calendar applications
239and close them. 261and close them.
240KO/Pi and KA/Pi must be running in order to receive the data. 262KO/Pi and KA/Pi must be running in order to receive the data.
241(KO/Pi and KA/Pi are always running if Fastload for them is enabled!) 263(KO/Pi and KA/Pi are always running if Fastload for them is enabled!)
242 264
243In the KA/Pi details view are now the preferred tel. numbers displayed on top 265In the KA/Pi details view are now the preferred tel. numbers displayed on top
244of the other data ( emails/tel.numbers/addresses) 266of the other data ( emails/tel.numbers/addresses)
245 267
246Fixed some syncing problems in KA/Pi. 268Fixed some syncing problems in KA/Pi.
247 269
248Added font settings for the KA/Pi details view. 270Added font settings for the KA/Pi details view.
249Added fields "children's name" and "gender" to KA/Pi. 271Added fields "children's name" and "gender" to KA/Pi.
250 272
251Made searching in KA/Pi better: 273Made searching in KA/Pi better:
252Now the first item in a view is selected after a search automatically and 274Now the first item in a view is selected after a search automatically and
253the views can be scrolled up/down when the search input field has the keyboard focus. 275the views can be scrolled up/down when the search input field has the keyboard focus.
254 276
255And, of course, fixed a bunch of reported bugs in KO/Pi and KA/Pi. 277And, of course, fixed a bunch of reported bugs in KO/Pi and KA/Pi.
256 278
257 279
258********** VERSION 1.9.6 ************ 280********** VERSION 1.9.6 ************
259 281
260Changes in the external application communication on the Zaurus 282Changes in the external application communication on the Zaurus
261in order to use less RAM when the apps are running. 283in order to use less RAM when the apps are running.
262First syncing of addressbooks (KA/Pi) is possible. 284First syncing of addressbooks (KA/Pi) is possible.
263 285
264 286
265********** VERSION 1.9.5a ************ 287********** VERSION 1.9.5a ************
266 288
267Fixed a bug in KO/Pi in the SharpDTM sync of version 1.9.5. 289Fixed a bug in KO/Pi in the SharpDTM sync of version 1.9.5.
268Fixed some small bugs. 290Fixed some small bugs.
269KA/Pi shows now the birthday in summary view. 291KA/Pi shows now the birthday in summary view.
270Now OM/Pi and KA/Pi are using the date format defined in KO/Pi 292Now OM/Pi and KA/Pi are using the date format defined in KO/Pi
271for displaying dates. 293for displaying dates.
272 294
273 295
274********** VERSION 1.9.5 ************ 296********** VERSION 1.9.5 ************
275 297
276There is still no Addressbook syncing! 298There is still no Addressbook syncing!
277 299
278New in 1.9.5: 300New in 1.9.5:
279 301
280Many bugfixes. 302Many bugfixes.
281Better searching in KA/Pi. 303Better searching in KA/Pi.
282You can configure in KA/Pi if you want to search only after 304You can configure in KA/Pi if you want to search only after
283<return> key pressed. 305<return> key pressed.
284 306
285Better mail downloading in OM/Pi. 307Better mail downloading in OM/Pi.
286 308
287First experimental alpha version of sync of KO/Pi with mobile phones. 309First experimental alpha version of sync of KO/Pi with mobile phones.
288See gammu documentation for supported phones. 310See gammu documentation for supported phones.
289You need to install the package kammu_1.9.5_arm.ipk for sync of KO/Pi with mobile phones. kammu_1.9.5_arm.ipk needs libbluetooth and libsdp. 311You need to install the package kammu_1.9.5_arm.ipk for sync of KO/Pi with mobile phones. kammu_1.9.5_arm.ipk needs libbluetooth and libsdp.
290Quick hint how to use: 312Quick hint how to use:
291NOTE: MOBILE PHONE SYNC IS EXPERIMENTAL! 313NOTE: MOBILE PHONE SYNC IS EXPERIMENTAL!
292Install kammu_1.9.5_arm.ipk , libbluetooth and libsdp. 314Install kammu_1.9.5_arm.ipk , libbluetooth and libsdp.
293Create syncprofile - mobile device 315Create syncprofile - mobile device
294Remove entry for model. (Leave empty ). 316Remove entry for model. (Leave empty ).
295Enable infrared on Zaurus and your Phone. 317Enable infrared on Zaurus and your Phone.
296Sync. 318Sync.
297To get a more detailed log, start kopi from konsole. 319To get a more detailed log, start kopi from konsole.
298 320
299********** VERSION 1.9.4 ************ 321********** VERSION 1.9.4 ************
300 322
301This is the version 1.9.4 of KDE-Pim/Pi for the Zaurus. 323This is the version 1.9.4 of KDE-Pim/Pi for the Zaurus.
302 324
303WARNING: 325WARNING:
304PLEASE BACKUP ALL YOUR DATA! 326PLEASE BACKUP ALL YOUR DATA!
305We have changed a lot and maybe there are some unknown problems. 327We have changed a lot and maybe there are some unknown problems.
306 328
307SYNC HANDLING HAS CHANGED! 329SYNC HANDLING HAS CHANGED!
308Such that, if you sync now with an already synded device, you will duplicated entries after the first sync. 330Such that, if you sync now with an already synded device, you will duplicated entries after the first sync.
309(This change was introduced to make it possible to sync with mobile phones, which will be available later (maybe in 4 weeks). 331(This change was introduced to make it possible to sync with mobile phones, which will be available later (maybe in 4 weeks).
310 332
311You need the kmicrokdelibs_1.9.4_arm.ipk as a base for the other programs. 333You need the kmicrokdelibs_1.9.4_arm.ipk as a base for the other programs.
312If you get the error: "Install only possible in main memory", just try it again to install it on SD card. That worked for me. And it was reported that rebooting Qtopia did help in this case as well. 334If you get the error: "Install only possible in main memory", just try it again to install it on SD card. That worked for me. And it was reported that rebooting Qtopia did help in this case as well.
313 335
314As programs are available: 336As programs are available:
315KO/Pi (korganizer ipk) - a calendar program. 337KO/Pi (korganizer ipk) - a calendar program.
316KA/Pi (kaddressbook ipk ) - an addressbook 338KA/Pi (kaddressbook ipk ) - an addressbook
317OM/Pi (kopiemail ipk ) an email program with pop/smtp and IMAP support. 339OM/Pi (kopiemail ipk ) an email program with pop/smtp and IMAP support.
318 340
319An alarm notification program ( korganizer-alarm ipk ) for KO/Pi that notifies you about alarms, even if the Zaurus is in suspend mode. 341An alarm notification program ( korganizer-alarm ipk ) for KO/Pi that notifies you about alarms, even if the Zaurus is in suspend mode.
320(If you do not see an icon in the taskbar after installing korganizer-alarm, please restart Qtopia) 342(If you do not see an icon in the taskbar after installing korganizer-alarm, please restart Qtopia)
321 343
322All the applications are installed in a "Pim" TAB. 344All the applications are installed in a "Pim" TAB.
323If this TAB is new on your system, you can get an icon in this TAB by installing pim_TAB_icon_1.9.4_arm.ipk 345If this TAB is new on your system, you can get an icon in this TAB by installing pim_TAB_icon_1.9.4_arm.ipk
324 346
325All the application are integrated. 347All the application are integrated.
326Such that you can choose in KO/Pi the attendees of a meeting from the addresses in KA/Pi. When you click in KA/Pi on the email address, OM/Pi is started to write the mail. 348Such that you can choose in KO/Pi the attendees of a meeting from the addresses in KA/Pi. When you click in KA/Pi on the email address, OM/Pi is started to write the mail.
327 349
328HINT: 350HINT:
329If you install KPhone/Pi 0.9.7, it will be called, if you click in KA/Pi on a phone number. 351If you install KPhone/Pi 0.9.7, it will be called, if you click in KA/Pi on a phone number.
330 352
331What's new? 353What's new?
332 354
333SYNC HANDLING HAS CHANGED! 355SYNC HANDLING HAS CHANGED!
334Such that, if you sync now with an already synded device, you will duplicated entries after the first sync. 356Such that, if you sync now with an already synded device, you will duplicated entries after the first sync.
335(This change was introduced to make it possible to sync with mobile phones, which will be available later (maybe in 4 weeks). 357(This change was introduced to make it possible to sync with mobile phones, which will be available later (maybe in 4 weeks).
336 358
337New in OM/Pi: 359New in OM/Pi:
338When copying(i.e. downloading mails) , you can specify, that only mails of a given size should be downloaded. Added mail copy possibility for selected mails. 360When copying(i.e. downloading mails) , you can specify, that only mails of a given size should be downloaded. Added mail copy possibility for selected mails.
339 361
340New in KO/Pi: 362New in KO/Pi:
341French is now available for KO/Pi. 363French is now available for KO/Pi.
342Choose menu:Actions - Configure:TAB locale 364Choose menu:Actions - Configure:TAB locale
343Syncing has changed. 365Syncing has changed.
344Phone sync available soon. 366Phone sync available soon.
345Not much changes, I cannot remember them ... 367Not much changes, I cannot remember them ...
346 368
347New in KA/Pi: 369New in KA/Pi:
348Beaming possible. 370Beaming possible.
349Sharp DTM readonly access possible( create a new DTM resource ); 371Sharp DTM readonly access possible( create a new DTM resource );
350Better searching possible. 372Better searching possible.
351Search is performed only after pressing the return key. 373Search is performed only after pressing the return key.
352Use wildcard * to specify parts of a name. 374Use wildcard * to specify parts of a name.
353 375
354Better name/email selection dialog (called from KO/Pi or OM/Pi). In this dialog, now searching is possible. Like in KA/Pi, use return key and wildcard * . 376Better name/email selection dialog (called from KO/Pi or OM/Pi). In this dialog, now searching is possible. Like in KA/Pi, use return key and wildcard * .
355 377
356A big improvement is the new management of the contact access. 378A big improvement is the new management of the contact access.
357In version 1.9.3, every application was using their own addressbook access data. 379In version 1.9.3, every application was using their own addressbook access data.
358That means, the addressbook was loaded up to three times in the memory, when accessed by KA/Pi, KO/Pi and OM/Pi. 380That means, the addressbook was loaded up to three times in the memory, when accessed by KA/Pi, KO/Pi and OM/Pi.
359That was wasting of memory, if you had several hundreds of contacts. 381That was wasting of memory, if you had several hundreds of contacts.
360 382
361Now only KA/Pi accesses the addressbook. 383Now only KA/Pi accesses the addressbook.
362If KO/Pi or OM/Pi want to get some name/email data, they request KA/Pi to open the name/email selection dialog and send it back to them. 384If KO/Pi or OM/Pi want to get some name/email data, they request KA/Pi to open the name/email selection dialog and send it back to them.
363If you click on an attendee in a meeting, its contact data is displayed in KA/Pi directly. 385If you click on an attendee in a meeting, its contact data is displayed in KA/Pi directly.
364That means, if KO/Pi or OM/Pi want to access contact data, KA/Pi is started first. 386That means, if KO/Pi or OM/Pi want to access contact data, KA/Pi is started first.
365 387
366New in the KO/Pi alarm applet: 388New in the KO/Pi alarm applet:
367Configure your own timer popup menu! 389Configure your own timer popup menu!
368(Text and minutes for timer countdown) 390(Text and minutes for timer countdown)
369Just edit the file 391Just edit the file
370(yourhomedir)/.kopialarmtimerrc 392(yourhomedir)/.kopialarmtimerrc
371and start/stop a timer to get a new menu with the data of this file. 393and start/stop a timer to get a new menu with the data of this file.
372 394
373********** VERSION 1.9.3 ************ 395********** VERSION 1.9.3 ************
3741) 3961)
375Now KO/Pi on Windows imports directly the calendar data of 397Now KO/Pi on Windows imports directly the calendar data of
376an installed Outlook. Should work with OL version >= 2000. 398an installed Outlook. Should work with OL version >= 2000.
377 399
378********** VERSION 1.9.2 ************ 400********** VERSION 1.9.2 ************
3791) 4011)
380KDE-Pim/Pi has got a new member: 402KDE-Pim/Pi has got a new member:
381KmicroMail (KM/Pi) is a mail program, 403KmicroMail (KM/Pi) is a mail program,
382which can handle IMAP and POP mail access. 404which can handle IMAP and POP mail access.
383It is based on Opie-Mail v3. 405It is based on Opie-Mail v3.
384All dependencies to the Opie libraries ar removed, 406All dependencies to the Opie libraries ar removed,
385such that no additional Opie lib is needed. 407such that no additional Opie lib is needed.
386It is already integrated in KO/Pi and KA/Pi. 408It is already integrated in KO/Pi and KA/Pi.
387It it now available for the Zaurus,probably it 409It it now available for the Zaurus,probably it
388will be available for other platforms later. 410will be available for other platforms later.
389Hint: 411Hint:
390Create your own contact (name + email) 412Create your own contact (name + email)
391in KA/Pi, select this contact and choose menu: 413in KA/Pi, select this contact and choose menu:
392Settings - Set Who Am I. 414Settings - Set Who Am I.
393Now the settings of this contact are used as 415Now the settings of this contact are used as
394the sender data in KM/Pi. 416the sender data in KM/Pi.
3952) 4172)
396KDE-Pim/Pi is split up in five different 418KDE-Pim/Pi is split up in five different
397packages now precompiled for Sharp Zaurus: 419packages now precompiled for Sharp Zaurus:
398--kmicrokdelibs_1.9.2_arm.ipk 420--kmicrokdelibs_1.9.2_arm.ipk
399The libs are needed for any 421The libs are needed for any
400of the following programs: 422of the following programs:
401--kaddressbook_1.9.2_arm.ipk 423--kaddressbook_1.9.2_arm.ipk
402--kmicromail_1.9.2_arm.ipk 424--kmicromail_1.9.2_arm.ipk
403--korganizer_1.9.2_arm.ipk 425--korganizer_1.9.2_arm.ipk
404Independ from that, there is the alarm applet 426Independ from that, there is the alarm applet
405available for KO/Pi, which also offers 427available for KO/Pi, which also offers
406quick access for a new mail or 428quick access for a new mail or
407showing the addressbook.: 429showing the addressbook.:
408--korganizer-alarm_1.9.2_arm.ipk 430--korganizer-alarm_1.9.2_arm.ipk
409Independend means, that the alarm applet 431Independend means, that the alarm applet
410does not need any of the libs or programs above to run. 432does not need any of the libs or programs above to run.
411But it would be quite useless without these programs. 433But it would be quite useless without these programs.
412NOTE: 434NOTE:
413If you get a 435If you get a
414"This application depends on other programs" 436"This application depends on other programs"
415during installation of 437during installation of
416--kmicrokdelibs_1.9.2_arm.ipk 438--kmicrokdelibs_1.9.2_arm.ipk
417you probably do not have to care about that. 439you probably do not have to care about that.
418kmicrokdelibs_1.9.2 will come with some 440kmicrokdelibs_1.9.2 will come with some
419resource plugins, which needs additional libraries. 441resource plugins, which needs additional libraries.
420(E.g. libopie1, if you want to use the 442(E.g. libopie1, if you want to use the
421opie resource connector in KA/Pi). 443opie resource connector in KA/Pi).
422If you do not have this libraries installed, 444If you do not have this libraries installed,
423you simply cannot use the resource. 445you simply cannot use the resource.
424To make it clear: 446To make it clear:
425If the libraries are missing, the applications 447If the libraries are missing, the applications
426using kmicrokdelibs will start, 448using kmicrokdelibs will start,
427because the resources are plugins. 449because the resources are plugins.
4283) 4503)
429KO/Pi and friends are now installable on SD-Card! 451KO/Pi and friends are now installable on SD-Card!
430It is recommended to install all libs and apps 452It is recommended to install all libs and apps
431on the SD card or all in the internal storage. 453on the SD card or all in the internal storage.
432There may be problems, if this is mixed. 454There may be problems, if this is mixed.
4334) 4554)
434Fixed two bugs in the alarm notification on Windows. 456Fixed two bugs in the alarm notification on Windows.
4355) 4575)
436Great improvement! 458Great improvement!
437KO/Pi uses now the latest version of libical. 459KO/Pi uses now the latest version of libical.
438Libical is the library which actually reads 460Libical is the library which actually reads
439the calendar files and extract the data from it. 461the calendar files and extract the data from it.
440With the old version, there were problems 462With the old version, there were problems
441(crashes or program hangs) when licical did read 463(crashes or program hangs) when licical did read
442files, which were not stored from KO/Pi. 464files, which were not stored from KO/Pi.
443I do not know, if the new libical works perfect, 465I do not know, if the new libical works perfect,
444but actually it works much better than 466but actually it works much better than
445the old version. 467the old version.
446There are no problems with compatibility with 468There are no problems with compatibility with
447old calendar files of KO/Pi, of course! 469old calendar files of KO/Pi, of course!
4486) 4706)
449New in KA/Pi: 471New in KA/Pi:
450Opie addressbook resource connector available! 472Opie addressbook resource connector available!
451You will need libopie1 and the opie addressbook, 473You will need libopie1 and the opie addressbook,
452of course. 474of course.
453With the Opie addressbook resource connector, 475With the Opie addressbook resource connector,
454you can access the Opie addressbook readonly in KA/Pi. 476you can access the Opie addressbook readonly in KA/Pi.
455If you want to edit or import the data into KA/Pi, 477If you want to edit or import the data into KA/Pi,
456do this: 478do this:
457a) Create an Opie resource. 479a) Create an Opie resource.
458 (Menu: Settings-Configure Resources). 480 (Menu: Settings-Configure Resources).
459After configuration and restarting KA/Pi you should see 481After configuration and restarting KA/Pi you should see
460the Opie contacts in KA/Pi. 482the Opie contacts in KA/Pi.
461b) Select some or all Opie contacts. 483b) Select some or all Opie contacts.
462(NOTE: +++++ 484(NOTE: +++++
463To know exactly, what contacts are Opie contacts, 485To know exactly, what contacts are Opie contacts,
464do this: Choose menu: 486do this: Choose menu:
465View-Modify View - TAB Fields. 487View-Modify View - TAB Fields.
466Select in the above list "Resource" and click 488Select in the above list "Resource" and click
467down arrow to add it to the "Selected fields". 489down arrow to add it to the "Selected fields".
468Click OK. 490Click OK.
469Now you have a new column "Resource" in your list, 491Now you have a new column "Resource" in your list,
470where you can see, what an Opie resource is. 492where you can see, what an Opie resource is.
471++++ NOTE end.) 493++++ NOTE end.)
472Ok, we do have now some Opie contacts seleted. 494Ok, we do have now some Opie contacts seleted.
473(Use SHIFT or CTRL key in order to multiple select). 495(Use SHIFT or CTRL key in order to multiple select).
474c) Choose menu: Edit-Copy. 496c) Choose menu: Edit-Copy.
475d) Choose menu: Edit-Paste. 497d) Choose menu: Edit-Paste.
476e) Select the resource, you want to add the contacts to. 498e) Select the resource, you want to add the contacts to.
477Congrats! Now you have read/write access to the copied 499Congrats! Now you have read/write access to the copied
478opie contacts as "real" KA/Pi contacts. 500opie contacts as "real" KA/Pi contacts.
479 501
480 502
481********** VERSION 1.9.1 ************ 503********** VERSION 1.9.1 ************
4821) 5041)
483 +++ IMPORTANT 1 +++ 505 +++ IMPORTANT 1 +++
484 506
485The storing place of the default calendar 507The storing place of the default calendar
486file has changed! 508file has changed!
487The default calendar file was 509The default calendar file was
488Applications/korganizer/mycalendar.ics 510Applications/korganizer/mycalendar.ics
489on Zaurus and 511on Zaurus and
490(yourHomeDir)/korganizer/mycalendar.ics 512(yourHomeDir)/korganizer/mycalendar.ics
491on Windows/Linux desktop. Now it is 513on Windows/Linux desktop. Now it is
492(yourHomeDir)/kdepim/korganizer/mycalendar.ics 514(yourHomeDir)/kdepim/korganizer/mycalendar.ics
493on Zaurus, Windows and Linux. 515on Zaurus, Windows and Linux.
494To load the old file, choose menu 516To load the old file, choose menu
495File-Load calendar backup. 517File-Load calendar backup.
496(It should be loaded automatically 518(It should be loaded automatically
497at startup with a warning message displayed). 519at startup with a warning message displayed).
498The place of the configuration file has changed too. 520The place of the configuration file has changed too.
499If you want to use your old KO/Pi config, 521If you want to use your old KO/Pi config,
500copy it to 522copy it to
501(yourHomeDir)/kdepim/config/korganizerrc 523(yourHomeDir)/kdepim/config/korganizerrc
502Please read VERSION 1.9.0 - topic 3) as well! 524Please read VERSION 1.9.0 - topic 3) as well!
503 525
504 +++ IMPORTANT 2 +++ 526 +++ IMPORTANT 2 +++
505 527
506Because of the new paths, you will need 528Because of the new paths, you will need
507a new version of the KO/Pi alarm applet 529a new version of the KO/Pi alarm applet
508for Zaurus. 530for Zaurus.
509Use version >= 1.9.1 531Use version >= 1.9.1
510 532
5112) 5332)
512Now the QWhat'sThis Icon works for items 534Now the QWhat'sThis Icon works for items
513in the month view as well. 535in the month view as well.
514(See VERSION 1.7.8 Topic 1) ). 536(See VERSION 1.7.8 Topic 1) ).
diff --git a/kmicromail/libetpan/maildir/maildir.c b/kmicromail/libetpan/maildir/maildir.c
index 0e038b1..1ef0b7a 100644
--- a/kmicromail/libetpan/maildir/maildir.c
+++ b/kmicromail/libetpan/maildir/maildir.c
@@ -1,729 +1,740 @@
1/* 1/*
2 * libEtPan! -- a mail stuff library 2 * libEtPan! -- a mail stuff library
3 * 3 *
4 * Copyright (C) 2001 - 2003 - DINH Viet Hoa 4 * Copyright (C) 2001 - 2003 - DINH Viet Hoa
5 * All rights reserved. 5 * All rights reserved.
6 * 6 *
7 * Redistribution and use in source and binary forms, with or without 7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions 8 * modification, are permitted provided that the following conditions
9 * are met: 9 * are met:
10 * 1. Redistributions of source code must retain the above copyright 10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer. 11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright 12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the 13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution. 14 * documentation and/or other materials provided with the distribution.
15 * 3. Neither the name of the libEtPan! project nor the names of its 15 * 3. Neither the name of the libEtPan! project nor the names of its
16 * contributors may be used to endorse or promote products derived 16 * contributors may be used to endorse or promote products derived
17 * from this software without specific prior written permission. 17 * from this software without specific prior written permission.
18 * 18 *
19 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 19 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
20 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 20 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 21 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 22 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
23 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 23 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
24 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 24 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
25 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 25 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
26 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 26 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
27 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 27 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
28 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 28 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29 * SUCH DAMAGE. 29 * SUCH DAMAGE.
30 */ 30 */
31 31
32/* 32/*
33 * $Id$ 33 * $Id$
34 */ 34 */
35 35
36#include "maildir.h" 36#include "maildir.h"
37 37
38#include <string.h> 38#include <string.h>
39#include <stdlib.h> 39#include <stdlib.h>
40#include <stdio.h> 40#include <stdio.h>
41#include <unistd.h> 41#include <unistd.h>
42#include <sys/types.h> 42#include <sys/types.h>
43#include <dirent.h> 43#include <dirent.h>
44#include <time.h> 44#include <time.h>
45#include <sys/stat.h> 45#include <sys/stat.h>
46#include <sys/mman.h> 46#include <sys/mman.h>
47 47
48#ifdef LIBETPAN_SYSTEM_BASENAME 48#ifdef LIBETPAN_SYSTEM_BASENAME
49#include <libgen.h> 49#include <libgen.h>
50#endif 50#endif
51 51
52/* 52/*
53 We suppose the maildir mailbox remains on one unique filesystem. 53 We suppose the maildir mailbox remains on one unique filesystem.
54*/ 54*/
55 55
56struct maildir * maildir_new(const char * path) 56struct maildir * maildir_new(const char * path)
57{ 57{
58 struct maildir * md; 58 struct maildir * md;
59 59
60 md = malloc(sizeof(* md)); 60 md = malloc(sizeof(* md));
61 if (md == NULL) 61 if (md == NULL)
62 goto err; 62 goto err;
63 63
64 md->mdir_counter = 0; 64 md->mdir_counter = 0;
65 md->mdir_mtime_new = (time_t) -1; 65 md->mdir_mtime_new = (time_t) -1;
66 md->mdir_mtime_cur = (time_t) -1; 66 md->mdir_mtime_cur = (time_t) -1;
67 67
68 md->mdir_pid = getpid(); 68 md->mdir_pid = getpid();
69 gethostname(md->mdir_hostname, sizeof(md->mdir_hostname)); 69 gethostname(md->mdir_hostname, sizeof(md->mdir_hostname));
70 strncpy(md->mdir_path, path, sizeof(md->mdir_path)); 70 strncpy(md->mdir_path, path, sizeof(md->mdir_path));
71 md->mdir_path[PATH_MAX - 1] = '\0'; 71 md->mdir_path[PATH_MAX - 1] = '\0';
72 72
73 md->mdir_msg_list = carray_new(128); 73 md->mdir_msg_list = carray_new(128);
74 if (md->mdir_msg_list == NULL) 74 if (md->mdir_msg_list == NULL)
75 goto free; 75 goto free;
76 76
77 md->mdir_msg_hash = chash_new(CHASH_DEFAULTSIZE, CHASH_COPYNONE); 77 md->mdir_msg_hash = chash_new(CHASH_DEFAULTSIZE, CHASH_COPYNONE);
78 if (md->mdir_msg_hash == NULL) 78 if (md->mdir_msg_hash == NULL)
79 goto free_msg_list; 79 goto free_msg_list;
80 80
81 return md; 81 return md;
82 82
83 free_msg_list: 83 free_msg_list:
84 carray_free(md->mdir_msg_list); 84 carray_free(md->mdir_msg_list);
85 free: 85 free:
86 free(md); 86 free(md);
87 err: 87 err:
88 return NULL; 88 return NULL;
89} 89}
90 90
91static void maildir_flush(struct maildir * md, int msg_new); 91static void maildir_flush(struct maildir * md, int msg_new);
92 92
93void maildir_free(struct maildir * md) 93void maildir_free(struct maildir * md)
94{ 94{
95 maildir_flush(md, 0); 95 maildir_flush(md, 0);
96 maildir_flush(md, 1); 96 maildir_flush(md, 1);
97 chash_free(md->mdir_msg_hash); 97 chash_free(md->mdir_msg_hash);
98 carray_free(md->mdir_msg_list); 98 carray_free(md->mdir_msg_list);
99 free(md); 99 free(md);
100} 100}
101 101
102#define MAX_TRY_ALLOC 32 102#define MAX_TRY_ALLOC 32
103 103
104static char * maildir_get_new_message_filename(struct maildir * md, 104static char * maildir_get_new_message_filename(struct maildir * md,
105 char * tmpfile) 105 char * tmpfile)
106{ 106{
107 char filename[PATH_MAX]; 107 char filename[PATH_MAX];
108 char basename[PATH_MAX]; 108 char basename[PATH_MAX];
109 int k; 109 int k;
110 time_t now; 110 time_t now;
111 111 struct stat f_stat;
112 now = time(NULL); 112 now = time(NULL);
113 k = 0; 113 k = 0;
114
115 fprintf(stderr,"maildir_get_new_message_filename: %s \n", tmpfile);
114 while (k < MAX_TRY_ALLOC) { 116 while (k < MAX_TRY_ALLOC) {
115 snprintf(basename, sizeof(basename), "%lu.%u_%u.%s", 117 snprintf(basename, sizeof(basename), "%lu.%u_%u.%s",
116 (unsigned long) now, md->mdir_pid, md->mdir_counter, md->mdir_hostname); 118 (unsigned long) now, md->mdir_pid, md->mdir_counter, md->mdir_hostname);
117 snprintf(filename, sizeof(filename), "%s/tmp/%s", 119 snprintf(filename, sizeof(filename), "%s/tmp/%s",
118 md->mdir_path, basename); 120 md->mdir_path, basename);
119 121 fprintf(stderr,"filename %s \n", filename);
120 if (link(tmpfile, filename) == 0) { 122 // LR changed following lines
123 if ( stat( filename, &f_stat ) == -1 ) {
124 //if (link(tmpfile, filename) == 0) {
121 char * dup_filename; 125 char * dup_filename;
122 126
123 dup_filename = strdup(filename); 127 dup_filename = strdup(filename);
124 if (dup_filename == NULL) { 128 if (dup_filename == NULL) {
125 unlink(filename); 129 //unlink(filename);
126 return NULL; 130 return NULL;
127 } 131 }
128 132 fprintf(stderr,"filename %s %s \n", tmpfile,dup_filename);
129 unlink(tmpfile); 133 //unlink(tmpfile);
134 rename (tmpfile,dup_filename );
130 md->mdir_counter ++; 135 md->mdir_counter ++;
131 136
132 return dup_filename; 137 return dup_filename;
133 } 138 }
134 139
135 md->mdir_counter ++; 140 md->mdir_counter ++;
136 k ++; 141 k ++;
137 } 142 }
138 143
139 return NULL; 144 return NULL;
140} 145}
141 146
142 147
143static void msg_free(struct maildir_msg * msg) 148static void msg_free(struct maildir_msg * msg)
144{ 149{
145 free(msg->msg_uid); 150 free(msg->msg_uid);
146 free(msg->msg_filename); 151 free(msg->msg_filename);
147 free(msg); 152 free(msg);
148} 153}
149 154
150/* 155/*
151 msg_new() 156 msg_new()
152 157
153 filename is given without path 158 filename is given without path
154*/ 159*/
155 160
156static struct maildir_msg * msg_new(char * filename, int new_msg) 161static struct maildir_msg * msg_new(char * filename, int new_msg)
157{ 162{
158 struct maildir_msg * msg; 163 struct maildir_msg * msg;
159 char * p; 164 char * p;
160 int flags; 165 int flags;
161 size_t uid_len; 166 size_t uid_len;
162 char * begin_uid; 167 char * begin_uid;
163 168
164 /* name of file : xxx-xxx_xxx-xxx:2,SRFT */ 169 /* name of file : xxx-xxx_xxx-xxx:2,SRFT */
165 170
166 msg = malloc(sizeof(* msg)); 171 msg = malloc(sizeof(* msg));
167 if (msg == NULL) 172 if (msg == NULL)
168 goto err; 173 goto err;
169 174
170 msg->msg_filename = strdup(filename); 175 msg->msg_filename = strdup(filename);
171 if (msg->msg_filename == NULL) 176 if (msg->msg_filename == NULL)
172 goto free; 177 goto free;
173 178
174 begin_uid = filename; 179 begin_uid = filename;
175 180
176 uid_len = strlen(begin_uid); 181 uid_len = strlen(begin_uid);
177 182
178 flags = 0; 183 flags = 0;
179 p = strstr(filename, ":2,"); 184 p = strstr(filename, ":2,");
180 if (p != NULL) { 185 if (p != NULL) {
181 uid_len = p - begin_uid; 186 uid_len = p - begin_uid;
182 187
183 p += 3; 188 p += 3;
184 189
185 /* parse flags */ 190 /* parse flags */
186 while (* p != '\0') { 191 while (* p != '\0') {
187 switch (* p) { 192 switch (* p) {
188 case 'S': 193 case 'S':
189 flags |= MAILDIR_FLAG_SEEN; 194 flags |= MAILDIR_FLAG_SEEN;
190 break; 195 break;
191 case 'R': 196 case 'R':
192 flags |= MAILDIR_FLAG_REPLIED; 197 flags |= MAILDIR_FLAG_REPLIED;
193 break; 198 break;
194 case 'F': 199 case 'F':
195 flags |= MAILDIR_FLAG_FLAGGED; 200 flags |= MAILDIR_FLAG_FLAGGED;
196 break; 201 break;
197 case 'T': 202 case 'T':
198 flags |= MAILDIR_FLAG_TRASHED; 203 flags |= MAILDIR_FLAG_TRASHED;
199 break; 204 break;
200 } 205 }
201 p ++; 206 p ++;
202 } 207 }
203 } 208 }
204 209
205 if (new_msg) 210 if (new_msg)
206 flags |= MAILDIR_FLAG_NEW; 211 flags |= MAILDIR_FLAG_NEW;
207 212
208 msg->msg_flags = flags; 213 msg->msg_flags = flags;
209 214
210 msg->msg_uid = malloc(uid_len + 1); 215 msg->msg_uid = malloc(uid_len + 1);
211 if (msg->msg_uid == NULL) 216 if (msg->msg_uid == NULL)
212 goto free_filename; 217 goto free_filename;
213 218
214 strncpy(msg->msg_uid, begin_uid, uid_len); 219 strncpy(msg->msg_uid, begin_uid, uid_len);
215 msg->msg_uid[uid_len] = '\0'; 220 msg->msg_uid[uid_len] = '\0';
216 221
217 return msg; 222 return msg;
218 223
219 free_filename: 224 free_filename:
220 free(msg->msg_filename); 225 free(msg->msg_filename);
221 free: 226 free:
222 free(msg); 227 free(msg);
223 err: 228 err:
224 return NULL; 229 return NULL;
225} 230}
226 231
227static void maildir_flush(struct maildir * md, int msg_new) 232static void maildir_flush(struct maildir * md, int msg_new)
228{ 233{
229 unsigned int i; 234 unsigned int i;
230 235
231 i = 0; 236 i = 0;
232 while (i < carray_count(md->mdir_msg_list)) { 237 while (i < carray_count(md->mdir_msg_list)) {
233 struct maildir_msg * msg; 238 struct maildir_msg * msg;
234 int delete; 239 int delete;
235 240
236 msg = carray_get(md->mdir_msg_list, i); 241 msg = carray_get(md->mdir_msg_list, i);
237 242
238 if (msg_new) { 243 if (msg_new) {
239 delete = 0; 244 delete = 0;
240 if ((msg->msg_flags & MAILDIR_FLAG_NEW) != 0) 245 if ((msg->msg_flags & MAILDIR_FLAG_NEW) != 0)
241 delete = 1; 246 delete = 1;
242 } 247 }
243 else { 248 else {
244 delete = 1; 249 delete = 1;
245 if ((msg->msg_flags & MAILDIR_FLAG_NEW) != 0) 250 if ((msg->msg_flags & MAILDIR_FLAG_NEW) != 0)
246 delete = 0; 251 delete = 0;
247 } 252 }
248 253
249 if (delete) { 254 if (delete) {
250 chashdatum key; 255 chashdatum key;
251 256
252 key.data = msg->msg_uid; 257 key.data = msg->msg_uid;
253 key.len = strlen(msg->msg_uid); 258 key.len = strlen(msg->msg_uid);
254 chash_delete(md->mdir_msg_hash, &key, NULL); 259 chash_delete(md->mdir_msg_hash, &key, NULL);
255 260
256 carray_delete(md->mdir_msg_list, i); 261 carray_delete(md->mdir_msg_list, i);
257 msg_free(msg); 262 msg_free(msg);
258 } 263 }
259 else { 264 else {
260 i ++; 265 i ++;
261 } 266 }
262 } 267 }
263} 268}
264 269
265static int add_message(struct maildir * md, 270static int add_message(struct maildir * md,
266 char * filename, int is_new) 271 char * filename, int is_new)
267{ 272{
268 struct maildir_msg * msg; 273 struct maildir_msg * msg;
269 chashdatum key; 274 chashdatum key;
270 chashdatum value; 275 chashdatum value;
271 unsigned int i; 276 unsigned int i;
272 int res; 277 int res;
273 int r; 278 int r;
274 279
280 fprintf(stderr,"add_message filename: %s \n", filename);
275 msg = msg_new(filename, is_new); 281 msg = msg_new(filename, is_new);
276 if (msg == NULL) { 282 if (msg == NULL) {
277 res = MAILDIR_ERROR_MEMORY; 283 res = MAILDIR_ERROR_MEMORY;
278 goto err; 284 goto err;
279 } 285 }
280 286
281 r = carray_add(md->mdir_msg_list, msg, &i); 287 r = carray_add(md->mdir_msg_list, msg, &i);
282 if (r < 0) { 288 if (r < 0) {
283 res = MAILDIR_ERROR_MEMORY; 289 res = MAILDIR_ERROR_MEMORY;
284 goto free_msg; 290 goto free_msg;
285 } 291 }
286 292
287 key.data = msg->msg_uid; 293 key.data = msg->msg_uid;
288 key.len = strlen(msg->msg_uid); 294 key.len = strlen(msg->msg_uid);
289 value.data = msg; 295 value.data = msg;
290 value.len = 0; 296 value.len = 0;
291 297
292 r = chash_set(md->mdir_msg_hash, &key, &value, NULL); 298 r = chash_set(md->mdir_msg_hash, &key, &value, NULL);
293 if (r < 0) { 299 if (r < 0) {
294 res = MAILDIR_ERROR_MEMORY; 300 res = MAILDIR_ERROR_MEMORY;
295 goto delete; 301 goto delete;
296 } 302 }
297 303
298 return MAILDIR_NO_ERROR; 304 return MAILDIR_NO_ERROR;
299 305
300 delete: 306 delete:
301 carray_delete(md->mdir_msg_list, i); 307 carray_delete(md->mdir_msg_list, i);
302 free_msg: 308 free_msg:
303 msg_free(msg); 309 msg_free(msg);
304 err: 310 err:
305 return res; 311 return res;
306} 312}
307 313
308static int add_directory(struct maildir * md, char * path, int is_new) 314static int add_directory(struct maildir * md, char * path, int is_new)
309{ 315{
310 DIR * d; 316 DIR * d;
311 struct dirent * entry; 317 struct dirent * entry;
312 int res; 318 int res;
313 int r; 319 int r;
314 char filename[PATH_MAX]; 320 char filename[PATH_MAX];
315 321
316 d = opendir(path); 322 d = opendir(path);
317 if (d == NULL) { 323 if (d == NULL) {
318 res = MAILDIR_ERROR_DIRECTORY; 324 res = MAILDIR_ERROR_DIRECTORY;
319 goto err; 325 goto err;
320 } 326 }
321 327
322 while ((entry = readdir(d)) != NULL) { 328 while ((entry = readdir(d)) != NULL) {
323 struct stat stat_info; 329 struct stat stat_info;
324 330
325 snprintf(filename, sizeof(filename), "%s/%s", path, entry->d_name); 331 snprintf(filename, sizeof(filename), "%s/%s", path, entry->d_name);
326 r = stat(filename, &stat_info); 332 r = stat(filename, &stat_info);
327 if (r < 0) 333 if (r < 0)
328 continue; 334 continue;
329 335
330 if (S_ISDIR(stat_info.st_mode)) 336 if (S_ISDIR(stat_info.st_mode))
331 continue; 337 continue;
332 338
333 r = add_message(md, entry->d_name, is_new); 339 r = add_message(md, entry->d_name, is_new);
334 if (r != MAILDIR_NO_ERROR) { 340 if (r != MAILDIR_NO_ERROR) {
335 /* ignore errors */ 341 /* ignore errors */
336 } 342 }
337 } 343 }
338 344
339 closedir(d); 345 closedir(d);
340 346
341 return MAILDIR_NO_ERROR; 347 return MAILDIR_NO_ERROR;
342 348
343 err: 349 err:
344 return res; 350 return res;
345} 351}
346 352
347int maildir_update(struct maildir * md) 353int maildir_update(struct maildir * md)
348{ 354{
349 struct stat stat_info; 355 struct stat stat_info;
350 char path_new[PATH_MAX]; 356 char path_new[PATH_MAX];
351 char path_cur[PATH_MAX]; 357 char path_cur[PATH_MAX];
352 int r; 358 int r;
353 int res; 359 int res;
354 360
355 snprintf(path_new, sizeof(path_new), "%s/new", md->mdir_path); 361 snprintf(path_new, sizeof(path_new), "%s/new", md->mdir_path);
356 snprintf(path_cur, sizeof(path_cur), "%s/cur", md->mdir_path); 362 snprintf(path_cur, sizeof(path_cur), "%s/cur", md->mdir_path);
357 363
358 /* did new/ changed ? */ 364 /* did new/ changed ? */
359 365
360 r = stat(path_new, &stat_info); 366 r = stat(path_new, &stat_info);
361 if (r < 0) { 367 if (r < 0) {
362 res = MAILDIR_ERROR_DIRECTORY; 368 res = MAILDIR_ERROR_DIRECTORY;
363 goto free; 369 goto free;
364 } 370 }
365 371
366 if (md->mdir_mtime_new != stat_info.st_mtime) { 372 if (md->mdir_mtime_new != stat_info.st_mtime) {
367 md->mdir_mtime_new = stat_info.st_mtime; 373 md->mdir_mtime_new = stat_info.st_mtime;
368 374
369 maildir_flush(md, 1); 375 maildir_flush(md, 1);
370 376
371 /* messages in new */ 377 /* messages in new */
372 r = add_directory(md, path_new, 1); 378 r = add_directory(md, path_new, 1);
373 if (r != MAILDIR_NO_ERROR) { 379 if (r != MAILDIR_NO_ERROR) {
374 res = r; 380 res = r;
375 goto free; 381 goto free;
376 } 382 }
377 } 383 }
378 384
379 /* did cur/ changed ? */ 385 /* did cur/ changed ? */
380 386
381 r = stat(path_cur, &stat_info); 387 r = stat(path_cur, &stat_info);
382 if (r < 0) { 388 if (r < 0) {
383 res = MAILDIR_ERROR_DIRECTORY; 389 res = MAILDIR_ERROR_DIRECTORY;
384 goto free; 390 goto free;
385 } 391 }
386 392
387 if (md->mdir_mtime_cur != stat_info.st_mtime) { 393 if (md->mdir_mtime_cur != stat_info.st_mtime) {
388 md->mdir_mtime_cur = stat_info.st_mtime; 394 md->mdir_mtime_cur = stat_info.st_mtime;
389 395
390 maildir_flush(md, 0); 396 maildir_flush(md, 0);
391 397
392 /* messages in cur */ 398 /* messages in cur */
393 r = add_directory(md, path_cur, 0); 399 r = add_directory(md, path_cur, 0);
394 if (r != MAILDIR_NO_ERROR) { 400 if (r != MAILDIR_NO_ERROR) {
395 res = r; 401 res = r;
396 goto free; 402 goto free;
397 } 403 }
398 } 404 }
399 405
400 return MAILDIR_NO_ERROR; 406 return MAILDIR_NO_ERROR;
401 407
402 free: 408 free:
403 maildir_flush(md, 0); 409 maildir_flush(md, 0);
404 maildir_flush(md, 1); 410 maildir_flush(md, 1);
405 md->mdir_mtime_cur = (time_t) -1; 411 md->mdir_mtime_cur = (time_t) -1;
406 md->mdir_mtime_new = (time_t) -1; 412 md->mdir_mtime_new = (time_t) -1;
407 return res; 413 return res;
408} 414}
409 415
410#ifndef LIBETPAN_SYSTEM_BASENAME 416#ifndef LIBETPAN_SYSTEM_BASENAME
411static char * libetpan_basename(char * filename) 417static char * libetpan_basename(char * filename)
412{ 418{
413 char * next; 419 char * next;
414 char * p; 420 char * p;
415 421
416 p = filename; 422 p = filename;
417 next = strchr(p, '/'); 423 next = strchr(p, '/');
418 424
419 while (next != NULL) { 425 while (next != NULL) {
420 p = next; 426 p = next;
421 next = strchr(p + 1, '/'); 427 next = strchr(p + 1, '/');
422 } 428 }
423 429
424 if (p == filename) 430 if (p == filename)
425 return filename; 431 return filename;
426 else 432 else
427 return p + 1; 433 return p + 1;
428} 434}
429#else 435#else
430#define libetpan_basename(a) basename(a) 436#define libetpan_basename(a) basename(a)
431#endif 437#endif
432 438
433int maildir_message_add_uid(struct maildir * md, 439int maildir_message_add_uid(struct maildir * md,
434 const char * message, size_t size, 440 const char * message, size_t size,
435 char * uid, size_t max_uid_len) 441 char * uid, size_t max_uid_len)
436{ 442{
437 char path_new[PATH_MAX]; 443 char path_new[PATH_MAX];
438 char tmpname[PATH_MAX]; 444 char tmpname[PATH_MAX];
439 int fd; 445 int fd;
440 int r; 446 int r;
441 char * mapping; 447 char * mapping;
442 char * delivery_tmp_name; 448 char * delivery_tmp_name;
443 char * delivery_tmp_basename; 449 char * delivery_tmp_basename;
444 char delivery_new_name[PATH_MAX]; 450 char delivery_new_name[PATH_MAX];
445 char * delivery_new_basename; 451 char * delivery_new_basename;
446 int res; 452 int res;
447 struct stat stat_info; 453 struct stat stat_info;
448 454
455 fprintf(stderr,"maildir_message_add_uid for uid: %s \n", uid);
449 r = maildir_update(md); 456 r = maildir_update(md);
450 if (r != MAILDIR_NO_ERROR) { 457 if (r != MAILDIR_NO_ERROR) {
451 res = r; 458 res = r;
452 goto err; 459 goto err;
453 } 460 }
454 461
455 /* write to tmp/ with a classic temporary file */ 462 /* write to tmp/ with a classic temporary file */
456 463
457 snprintf(tmpname, sizeof(tmpname), "%s/tmp/etpan-maildir-XXXXXX", 464 snprintf(tmpname, sizeof(tmpname), "%s/tmp/etpan-maildir-XXXXXX",
458 md->mdir_path); 465 md->mdir_path);
459 fd = mkstemp(tmpname); 466 fd = mkstemp(tmpname);
460 if (fd < 0) { 467 if (fd < 0) {
461 res = MAILDIR_ERROR_FILE; 468 res = MAILDIR_ERROR_FILE;
462 goto err; 469 goto err;
463 } 470 }
464 471
465 r = ftruncate(fd, size); 472 r = ftruncate(fd, size);
466 if (r < 0) { 473 if (r < 0) {
467 res = MAILDIR_ERROR_FILE; 474 res = MAILDIR_ERROR_FILE;
468 goto close; 475 goto close;
469 } 476 }
470 477
471 mapping = mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0); 478 mapping = mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
472 if (mapping == MAP_FAILED) { 479 if (mapping == MAP_FAILED) {
473 res = MAILDIR_ERROR_FILE; 480 res = MAILDIR_ERROR_FILE;
474 goto close; 481 goto close;
475 } 482 }
476 483
477 memcpy(mapping, message, size); 484 memcpy(mapping, message, size);
478 485
479 msync(mapping, size, MS_SYNC); 486 msync(mapping, size, MS_SYNC);
480 munmap(mapping, size); 487 munmap(mapping, size);
481 488
482 close(fd); 489 close(fd);
483 490
484 /* write to tmp/ with maildir standard name */ 491 /* write to tmp/ with maildir standard name */
485 492
486 delivery_tmp_name = maildir_get_new_message_filename(md, tmpname); 493 delivery_tmp_name = maildir_get_new_message_filename(md, tmpname);
487 if (delivery_tmp_name == NULL) { 494 if (delivery_tmp_name == NULL) {
488 res = MAILDIR_ERROR_FILE; 495 res = MAILDIR_ERROR_FILE;
489 goto unlink; 496 goto unlink;
490 } 497 }
491 498
492 /* write to new/ with maildir standard name */ 499 /* write to new/ with maildir standard name */
493 500
494 strncpy(tmpname, delivery_tmp_name, sizeof(tmpname)); 501 strncpy(tmpname, delivery_tmp_name, sizeof(tmpname));
495 tmpname[sizeof(tmpname) - 1] = '\0'; 502 tmpname[sizeof(tmpname) - 1] = '\0';
496 503
497 delivery_tmp_basename = libetpan_basename(tmpname); 504 delivery_tmp_basename = libetpan_basename(tmpname);
498 505
499 snprintf(delivery_new_name, sizeof(delivery_new_name), "%s/new/%s", 506 snprintf(delivery_new_name, sizeof(delivery_new_name), "%s/new/%s",
500 md->mdir_path, delivery_tmp_basename); 507 md->mdir_path, delivery_tmp_basename);
501 508
502 r = link(delivery_tmp_name, delivery_new_name); 509 r = link(delivery_tmp_name, delivery_new_name);
503 if (r < 0) { 510 if (r < 0) {
504 res = MAILDIR_ERROR_FILE; 511 res = MAILDIR_ERROR_FILE;
505 goto unlink_tmp; 512 goto unlink_tmp;
506 } 513 }
507 514
508 snprintf(path_new, sizeof(path_new), "%s/new", md->mdir_path); 515 snprintf(path_new, sizeof(path_new), "%s/new", md->mdir_path);
509 r = stat(path_new, &stat_info); 516 r = stat(path_new, &stat_info);
510 if (r < 0) { 517 if (r < 0) {
511 unlink(delivery_new_name); 518 unlink(delivery_new_name);
512 res = MAILDIR_ERROR_FILE; 519 res = MAILDIR_ERROR_FILE;
513 goto unlink_tmp; 520 goto unlink_tmp;
514 } 521 }
515 522
516 md->mdir_mtime_new = stat_info.st_mtime; 523 md->mdir_mtime_new = stat_info.st_mtime;
517 524
518 delivery_new_basename = libetpan_basename(delivery_new_name); 525 delivery_new_basename = libetpan_basename(delivery_new_name);
519 526
520 r = add_message(md, delivery_new_basename, 1); 527 r = add_message(md, delivery_new_basename, 1);
521 if (r != MAILDIR_NO_ERROR) { 528 if (r != MAILDIR_NO_ERROR) {
522 unlink(delivery_new_name); 529 unlink(delivery_new_name);
523 res = MAILDIR_ERROR_FILE; 530 res = MAILDIR_ERROR_FILE;
524 goto unlink_tmp; 531 goto unlink_tmp;
525 } 532 }
526 533
527 if (uid != NULL) 534 if (uid != NULL)
528 strncpy(uid, delivery_new_basename, max_uid_len); 535 strncpy(uid, delivery_new_basename, max_uid_len);
529 536
530 unlink(delivery_tmp_name); 537 unlink(delivery_tmp_name);
531 free(delivery_tmp_name); 538 free(delivery_tmp_name);
532 539
533 return MAILDIR_NO_ERROR; 540 return MAILDIR_NO_ERROR;
534 541
535 unlink_tmp: 542 unlink_tmp:
536 unlink(delivery_tmp_name); 543 unlink(delivery_tmp_name);
537 free(delivery_tmp_name); 544 free(delivery_tmp_name);
538 goto err; 545 goto err;
539 close: 546 close:
540 close(fd); 547 close(fd);
541 unlink: 548 unlink:
542 unlink(tmpname); 549 unlink(tmpname);
543 err: 550 err:
544 return res; 551 return res;
545} 552}
546 553
547int maildir_message_add(struct maildir * md, 554int maildir_message_add(struct maildir * md,
548 const char * message, size_t size) 555 const char * message, size_t size)
549{ 556{
550 return maildir_message_add_uid(md, message, size, 557 return maildir_message_add_uid(md, message, size,
551 NULL, 0); 558 NULL, 0);
552} 559}
553 560
554int maildir_message_add_file_uid(struct maildir * md, int fd, 561int maildir_message_add_file_uid(struct maildir * md, int fd,
555 char * uid, size_t max_uid_len) 562 char * uid, size_t max_uid_len)
556{ 563{
557 char * message; 564 char * message;
558 struct stat buf; 565 struct stat buf;
559 int r; 566 int r;
560 567
568 fprintf(stderr,"maildir_message_add_file_uid: %s \n", uid);
561 if (fstat(fd, &buf) == -1) 569 if (fstat(fd, &buf) == -1)
562 return MAILDIR_ERROR_FILE; 570 return MAILDIR_ERROR_FILE;
563 571
564 message = mmap(NULL, buf.st_size, PROT_READ, MAP_PRIVATE, fd, 0); 572 message = mmap(NULL, buf.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
565 if (message == MAP_FAILED) 573 if (message == MAP_FAILED)
566 return MAILDIR_ERROR_FILE; 574 return MAILDIR_ERROR_FILE;
567 575
568 r = maildir_message_add_uid(md, message, buf.st_size, uid, max_uid_len); 576 r = maildir_message_add_uid(md, message, buf.st_size, uid, max_uid_len);
569 577
570 munmap(message, buf.st_size); 578 munmap(message, buf.st_size);
571 579
572 return r; 580 return r;
573} 581}
574 582
575int maildir_message_add_file(struct maildir * md, int fd) 583int maildir_message_add_file(struct maildir * md, int fd)
576{ 584{
585 fprintf(stderr,"maildir_message_add_file \n");
577 return maildir_message_add_file_uid(md, fd, 586 return maildir_message_add_file_uid(md, fd,
578 NULL, 0); 587 NULL, 0);
579} 588}
580 589
581char * maildir_message_get(struct maildir * md, const char * uid) 590char * maildir_message_get(struct maildir * md, const char * uid)
582{ 591{
583 chashdatum key; 592 chashdatum key;
584 chashdatum value; 593 chashdatum value;
585 char filename[PATH_MAX]; 594 char filename[PATH_MAX];
586 char * dup_filename; 595 char * dup_filename;
587 struct maildir_msg * msg; 596 struct maildir_msg * msg;
588 char * dir; 597 char * dir;
589 int r; 598 int r;
590 599
600 fprintf(stderr,"maildir_message_get for uid: %s \n", uid);
591 key.data = (void *) uid; 601 key.data = (void *) uid;
592 key.len = strlen(uid); 602 key.len = strlen(uid);
593 r = chash_get(md->mdir_msg_hash, &key, &value); 603 r = chash_get(md->mdir_msg_hash, &key, &value);
594 if (r < 0) 604 if (r < 0)
595 return NULL; 605 return NULL;
596 606
597 msg = value.data; 607 msg = value.data;
598 if ((msg->msg_flags & MAILDIR_FLAG_NEW) != 0) 608 if ((msg->msg_flags & MAILDIR_FLAG_NEW) != 0)
599 dir = "new"; 609 dir = "new";
600 else 610 else
601 dir = "cur"; 611 dir = "cur";
602 612
603 snprintf(filename, sizeof(filename), "%s/%s/%s", 613 snprintf(filename, sizeof(filename), "%s/%s/%s",
604 md->mdir_path, dir, msg->msg_filename); 614 md->mdir_path, dir, msg->msg_filename);
605 615
606 dup_filename = strdup(filename); 616 dup_filename = strdup(filename);
607 if (dup_filename == NULL) 617 if (dup_filename == NULL)
608 return NULL; 618 return NULL;
609 619
610 return dup_filename; 620 return dup_filename;
611} 621}
612 622
613int maildir_message_remove(struct maildir * md, const char * uid) 623int maildir_message_remove(struct maildir * md, const char * uid)
614{ 624{
615 chashdatum key; 625 chashdatum key;
616 chashdatum value; 626 chashdatum value;
617 char filename[PATH_MAX]; 627 char filename[PATH_MAX];
618 struct maildir_msg * msg; 628 struct maildir_msg * msg;
619 char * dir; 629 char * dir;
620 int r; 630 int r;
621 int res; 631 int res;
622 632
633 fprintf(stderr,"maildir_message_remove for uid: %s \n", uid);
623 key.data = (void *) uid; 634 key.data = (void *) uid;
624 key.len = strlen(uid); 635 key.len = strlen(uid);
625 r = chash_get(md->mdir_msg_hash, &key, &value); 636 r = chash_get(md->mdir_msg_hash, &key, &value);
626 if (r < 0) { 637 if (r < 0) {
627 res = MAILDIR_ERROR_NOT_FOUND; 638 res = MAILDIR_ERROR_NOT_FOUND;
628 goto err; 639 goto err;
629 } 640 }
630 641
631 msg = value.data; 642 msg = value.data;
632 if ((msg->msg_flags & MAILDIR_FLAG_NEW) != 0) 643 if ((msg->msg_flags & MAILDIR_FLAG_NEW) != 0)
633 dir = "new"; 644 dir = "new";
634 else 645 else
635 dir = "cur"; 646 dir = "cur";
636 647
637 snprintf(filename, sizeof(filename), "%s/%s/%s", 648 snprintf(filename, sizeof(filename), "%s/%s/%s",
638 md->mdir_path, dir, msg->msg_filename); 649 md->mdir_path, dir, msg->msg_filename);
639 650
640 r = unlink(filename); 651 r = unlink(filename);
641 if (r < 0) { 652 if (r < 0) {
642 res = MAILDIR_ERROR_FILE; 653 res = MAILDIR_ERROR_FILE;
643 goto err; 654 goto err;
644 } 655 }
645 656
646 return MAILDIR_NO_ERROR; 657 return MAILDIR_NO_ERROR;
647 658
648 err: 659 err:
649 return res; 660 return res;
650} 661}
651 662
652int maildir_message_change_flags(struct maildir * md, 663int maildir_message_change_flags(struct maildir * md,
653 const char * uid, int new_flags) 664 const char * uid, int new_flags)
654{ 665{
655 chashdatum key; 666 chashdatum key;
656 chashdatum value; 667 chashdatum value;
657 char filename[PATH_MAX]; 668 char filename[PATH_MAX];
658 struct maildir_msg * msg; 669 struct maildir_msg * msg;
659 char * dir; 670 char * dir;
660 int r; 671 int r;
661 char new_filename[PATH_MAX]; 672 char new_filename[PATH_MAX];
662 char flag_str[5]; 673 char flag_str[5];
663 size_t i; 674 size_t i;
664 int res; 675 int res;
665 676 fprintf(stderr,"maildir_message_change_flags for uid: %s \n", uid);
666 key.data = (void *) uid; 677 key.data = (void *) uid;
667 key.len = strlen(uid); 678 key.len = strlen(uid);
668 r = chash_get(md->mdir_msg_hash, &key, &value); 679 r = chash_get(md->mdir_msg_hash, &key, &value);
669 if (r < 0) { 680 if (r < 0) {
670 res = MAILDIR_ERROR_NOT_FOUND; 681 res = MAILDIR_ERROR_NOT_FOUND;
671 goto err; 682 goto err;
672 } 683 }
673 684
674 msg = value.data; 685 msg = value.data;
675 if ((msg->msg_flags & MAILDIR_FLAG_NEW) != 0) 686 if ((msg->msg_flags & MAILDIR_FLAG_NEW) != 0)
676 dir = "new"; 687 dir = "new";
677 else 688 else
678 dir = "cur"; 689 dir = "cur";
679 690
680 snprintf(filename, sizeof(filename), "%s/%s/%s", 691 snprintf(filename, sizeof(filename), "%s/%s/%s",
681 md->mdir_path, dir, msg->msg_filename); 692 md->mdir_path, dir, msg->msg_filename);
682 693
683 if ((new_flags & MAILDIR_FLAG_NEW) != 0) 694 if ((new_flags & MAILDIR_FLAG_NEW) != 0)
684 dir = "new"; 695 dir = "new";
685 else 696 else
686 dir = "cur"; 697 dir = "cur";
687 698
688 i = 0; 699 i = 0;
689 if ((new_flags & MAILDIR_FLAG_SEEN) != 0) { 700 if ((new_flags & MAILDIR_FLAG_SEEN) != 0) {
690 flag_str[i] = 'S'; 701 flag_str[i] = 'S';
691 i ++; 702 i ++;
692 } 703 }
693 if ((new_flags & MAILDIR_FLAG_REPLIED) != 0) { 704 if ((new_flags & MAILDIR_FLAG_REPLIED) != 0) {
694 flag_str[i] = 'R'; 705 flag_str[i] = 'R';
695 i ++; 706 i ++;
696 } 707 }
697 if ((new_flags & MAILDIR_FLAG_FLAGGED) != 0) { 708 if ((new_flags & MAILDIR_FLAG_FLAGGED) != 0) {
698 flag_str[i] = 'F'; 709 flag_str[i] = 'F';
699 i ++; 710 i ++;
700 } 711 }
701 if ((new_flags & MAILDIR_FLAG_TRASHED) != 0) { 712 if ((new_flags & MAILDIR_FLAG_TRASHED) != 0) {
702 flag_str[i] = 'T'; 713 flag_str[i] = 'T';
703 i ++; 714 i ++;
704 } 715 }
705 flag_str[i] = 0; 716 flag_str[i] = 0;
706 717
707 if (flag_str[0] == '\0') 718 if (flag_str[0] == '\0')
708 snprintf(new_filename, sizeof(new_filename), "%s/%s/%s", 719 snprintf(new_filename, sizeof(new_filename), "%s/%s/%s",
709 md->mdir_path, dir, msg->msg_uid); 720 md->mdir_path, dir, msg->msg_uid);
710 else 721 else
711 snprintf(new_filename, sizeof(new_filename), "%s/%s/%s:2,%s", 722 snprintf(new_filename, sizeof(new_filename), "%s/%s/%s:2,%s",
712 md->mdir_path, dir, msg->msg_uid, flag_str); 723 md->mdir_path, dir, msg->msg_uid, flag_str);
713 724
714 if (strcmp(filename, new_filename) == 0) 725 if (strcmp(filename, new_filename) == 0)
715 return MAILDIR_NO_ERROR; 726 return MAILDIR_NO_ERROR;
716 727
717 r = link(filename, new_filename); 728 r = link(filename, new_filename);
718 if (r < 0) { 729 if (r < 0) {
719 res = MAILDIR_ERROR_FILE; 730 res = MAILDIR_ERROR_FILE;
720 goto err; 731 goto err;
721 } 732 }
722 733
723 unlink(filename); 734 unlink(filename);
724 735
725 return MAILDIR_NO_ERROR; 736 return MAILDIR_NO_ERROR;
726 737
727 err: 738 err:
728 return res; 739 return res;
729} 740}
diff --git a/kmicromail/libetpan/mh/mailmh.c b/kmicromail/libetpan/mh/mailmh.c
index 119f217..5e2b4cc 100644
--- a/kmicromail/libetpan/mh/mailmh.c
+++ b/kmicromail/libetpan/mh/mailmh.c
@@ -98,884 +98,888 @@ struct mailmh_msg_info * mailmh_msg_info_new(uint32_t index, size_t size,
98 msg_info->msg_index = index; 98 msg_info->msg_index = index;
99 msg_info->msg_size = size; 99 msg_info->msg_size = size;
100 msg_info->msg_mtime = mtime; 100 msg_info->msg_mtime = mtime;
101 101
102 msg_info->msg_array_index = 0; 102 msg_info->msg_array_index = 0;
103 103
104 return msg_info; 104 return msg_info;
105} 105}
106 106
107void mailmh_msg_info_free(struct mailmh_msg_info * msg_info) 107void mailmh_msg_info_free(struct mailmh_msg_info * msg_info)
108{ 108{
109 free(msg_info); 109 free(msg_info);
110} 110}
111 111
112struct mailmh_folder * mailmh_folder_new(struct mailmh_folder * parent, 112struct mailmh_folder * mailmh_folder_new(struct mailmh_folder * parent,
113 const char * name) 113 const char * name)
114{ 114{
115 char * filename; 115 char * filename;
116 char * parent_filename; 116 char * parent_filename;
117 117
118 struct mailmh_folder * folder; 118 struct mailmh_folder * folder;
119 119
120 folder = malloc(sizeof(* folder)); 120 folder = malloc(sizeof(* folder));
121 if (folder == NULL) 121 if (folder == NULL)
122 goto err; 122 goto err;
123 123
124 if (parent == NULL) { 124 if (parent == NULL) {
125 filename = strdup(name); 125 filename = strdup(name);
126 if (filename == NULL) 126 if (filename == NULL)
127 goto free_folder; 127 goto free_folder;
128 } 128 }
129 else { 129 else {
130 parent_filename = parent->fl_filename; 130 parent_filename = parent->fl_filename;
131 filename = malloc(strlen(parent_filename) + strlen(name) + 2); 131 filename = malloc(strlen(parent_filename) + strlen(name) + 2);
132 if (filename == NULL) 132 if (filename == NULL)
133 goto free_folder; 133 goto free_folder;
134 134
135 strcpy(filename, parent_filename); 135 strcpy(filename, parent_filename);
136 strcat(filename, MAIL_DIR_SEPARATOR_S); 136 strcat(filename, MAIL_DIR_SEPARATOR_S);
137 strcat(filename, name); 137 strcat(filename, name);
138 } 138 }
139 139
140 folder->fl_filename = filename; 140 folder->fl_filename = filename;
141 141
142 folder->fl_name = strdup(name); 142 folder->fl_name = strdup(name);
143 if (folder->fl_name == NULL) 143 if (folder->fl_name == NULL)
144 goto free_filename; 144 goto free_filename;
145 145
146 folder->fl_msgs_tab = carray_new(128); 146 folder->fl_msgs_tab = carray_new(128);
147 if (folder->fl_msgs_tab == NULL) 147 if (folder->fl_msgs_tab == NULL)
148 goto free_name; 148 goto free_name;
149 149
150#if 0 150#if 0
151 folder->fl_msgs_hash = cinthash_new(128); 151 folder->fl_msgs_hash = cinthash_new(128);
152 if (folder->fl_msgs_hash == NULL) 152 if (folder->fl_msgs_hash == NULL)
153 goto free_msgs_tab; 153 goto free_msgs_tab;
154#endif 154#endif
155 folder->fl_msgs_hash = chash_new(CHASH_DEFAULTSIZE, CHASH_COPYKEY); 155 folder->fl_msgs_hash = chash_new(CHASH_DEFAULTSIZE, CHASH_COPYKEY);
156 if (folder->fl_msgs_hash == NULL) 156 if (folder->fl_msgs_hash == NULL)
157 goto free_msgs_tab; 157 goto free_msgs_tab;
158 158
159 folder->fl_subfolders_tab = carray_new(128); 159 folder->fl_subfolders_tab = carray_new(128);
160 if (folder->fl_subfolders_tab == NULL) 160 if (folder->fl_subfolders_tab == NULL)
161 goto free_msgs_hash; 161 goto free_msgs_hash;
162 162
163 folder->fl_subfolders_hash = chash_new(128, CHASH_COPYNONE); 163 folder->fl_subfolders_hash = chash_new(128, CHASH_COPYNONE);
164 if (folder->fl_subfolders_hash == NULL) 164 if (folder->fl_subfolders_hash == NULL)
165 goto free_subfolders_tab; 165 goto free_subfolders_tab;
166 166
167 folder->fl_mtime = 0; 167 folder->fl_mtime = 0;
168 folder->fl_parent = parent; 168 folder->fl_parent = parent;
169 folder->fl_max_index = 0; 169 folder->fl_max_index = 0;
170 170
171 return folder; 171 return folder;
172 172
173 free_subfolders_tab: 173 free_subfolders_tab:
174 carray_free(folder->fl_subfolders_tab); 174 carray_free(folder->fl_subfolders_tab);
175 free_msgs_hash: 175 free_msgs_hash:
176#if 0 176#if 0
177 cinthash_free(folder->fl_msgs_hash); 177 cinthash_free(folder->fl_msgs_hash);
178#endif 178#endif
179 chash_free(folder->fl_msgs_hash); 179 chash_free(folder->fl_msgs_hash);
180 free_msgs_tab: 180 free_msgs_tab:
181 carray_free(folder->fl_msgs_tab); 181 carray_free(folder->fl_msgs_tab);
182 free_name: 182 free_name:
183 free(folder->fl_name); 183 free(folder->fl_name);
184 free_filename: 184 free_filename:
185 free(folder->fl_filename); 185 free(folder->fl_filename);
186 free_folder: 186 free_folder:
187 free(folder); 187 free(folder);
188 err: 188 err:
189 return NULL; 189 return NULL;
190} 190}
191 191
192void mailmh_folder_free(struct mailmh_folder * folder) 192void mailmh_folder_free(struct mailmh_folder * folder)
193{ 193{
194 unsigned int i; 194 unsigned int i;
195 195
196 for(i = 0 ; i < carray_count(folder->fl_subfolders_tab) ; i++) { 196 for(i = 0 ; i < carray_count(folder->fl_subfolders_tab) ; i++) {
197 struct mailmh_folder * subfolder; 197 struct mailmh_folder * subfolder;
198 198
199 subfolder = carray_get(folder->fl_subfolders_tab, i); 199 subfolder = carray_get(folder->fl_subfolders_tab, i);
200 if (subfolder != NULL) 200 if (subfolder != NULL)
201 mailmh_folder_free(subfolder); 201 mailmh_folder_free(subfolder);
202 } 202 }
203 carray_free(folder->fl_subfolders_tab); 203 carray_free(folder->fl_subfolders_tab);
204 chash_free(folder->fl_subfolders_hash); 204 chash_free(folder->fl_subfolders_hash);
205 205
206 for(i = 0 ; i < carray_count(folder->fl_msgs_tab) ; i++) { 206 for(i = 0 ; i < carray_count(folder->fl_msgs_tab) ; i++) {
207 struct mailmh_msg_info * msg_info; 207 struct mailmh_msg_info * msg_info;
208 208
209 msg_info = carray_get(folder->fl_msgs_tab, i); 209 msg_info = carray_get(folder->fl_msgs_tab, i);
210 if (msg_info != NULL) 210 if (msg_info != NULL)
211 mailmh_msg_info_free(msg_info); 211 mailmh_msg_info_free(msg_info);
212 } 212 }
213 carray_free(folder->fl_msgs_tab); 213 carray_free(folder->fl_msgs_tab);
214 chash_free(folder->fl_msgs_hash); 214 chash_free(folder->fl_msgs_hash);
215#if 0 215#if 0
216 cinthash_free(folder->fl_msgs_hash); 216 cinthash_free(folder->fl_msgs_hash);
217#endif 217#endif
218 218
219 free(folder->fl_filename); 219 free(folder->fl_filename);
220 free(folder->fl_name); 220 free(folder->fl_name);
221 221
222 free(folder); 222 free(folder);
223} 223}
224 224
225struct mailmh_folder * mailmh_folder_find(struct mailmh_folder * root, 225struct mailmh_folder * mailmh_folder_find(struct mailmh_folder * root,
226 const char * filename) 226 const char * filename)
227{ 227{
228 int r; 228 int r;
229 char pathname[PATH_MAX]; 229 char pathname[PATH_MAX];
230 char * p; 230 char * p;
231 chashdatum key; 231 chashdatum key;
232 chashdatum data; 232 chashdatum data;
233 struct mailmh_folder * folder; 233 struct mailmh_folder * folder;
234 char * start; 234 char * start;
235 235
236 if (strcmp(root->fl_filename, filename) == 0) 236 if (strcmp(root->fl_filename, filename) == 0)
237 return root; 237 return root;
238 238
239#if 0 239#if 0
240 r = mailmh_folder_update(root); 240 r = mailmh_folder_update(root);
241 if (r != MAILMH_NO_ERROR) 241 if (r != MAILMH_NO_ERROR)
242 return NULL; 242 return NULL;
243#endif 243#endif
244 244
245#if 0 245#if 0
246 for(i = 0 ; i < root->fl_subfolders_tab->len ; i++) { 246 for(i = 0 ; i < root->fl_subfolders_tab->len ; i++) {
247 struct mailmh_folder * subfolder; 247 struct mailmh_folder * subfolder;
248 248
249 subfolder = carray_get(root->fl_subfolders_tab, i); 249 subfolder = carray_get(root->fl_subfolders_tab, i);
250 if (subfolder != NULL) 250 if (subfolder != NULL)
251 if (strncmp(subfolder->fl_filename, filename, 251 if (strncmp(subfolder->fl_filename, filename,
252 strlen(subfolder->fl_filename)) == 0) 252 strlen(subfolder->fl_filename)) == 0)
253 return mailmh_folder_find(subfolder, filename); 253 return mailmh_folder_find(subfolder, filename);
254 } 254 }
255#endif 255#endif
256 strncpy(pathname, filename, PATH_MAX); 256 strncpy(pathname, filename, PATH_MAX);
257 pathname[PATH_MAX - 1] = 0; 257 pathname[PATH_MAX - 1] = 0;
258 start = pathname + strlen(root->fl_filename) + 1; 258 start = pathname + strlen(root->fl_filename) + 1;
259 259
260 p = strchr(start, MAIL_DIR_SEPARATOR); 260 p = strchr(start, MAIL_DIR_SEPARATOR);
261 if (p != NULL) { 261 if (p != NULL) {
262 * p = 0; 262 * p = 0;
263 263
264 root = mailmh_folder_find(root, pathname); 264 root = mailmh_folder_find(root, pathname);
265 if (root != NULL) { 265 if (root != NULL) {
266 folder = mailmh_folder_find(root, filename); 266 folder = mailmh_folder_find(root, filename);
267 if (folder == NULL) 267 if (folder == NULL)
268 return NULL; 268 return NULL;
269 return folder; 269 return folder;
270 } 270 }
271 271
272 return NULL; 272 return NULL;
273 } 273 }
274 else { 274 else {
275 key.data = pathname; 275 key.data = pathname;
276 key.len = strlen(pathname); 276 key.len = strlen(pathname);
277 r = chash_get(root->fl_subfolders_hash, &key, &data); 277 r = chash_get(root->fl_subfolders_hash, &key, &data);
278 if (r < 0) 278 if (r < 0)
279 return NULL; 279 return NULL;
280 280
281 return data.data; 281 return data.data;
282 } 282 }
283} 283}
284 284
285int mailmh_folder_update(struct mailmh_folder * folder) 285int mailmh_folder_update(struct mailmh_folder * folder)
286{ 286{
287 DIR * d; 287 DIR * d;
288 struct dirent * ent; 288 struct dirent * ent;
289 struct stat buf; 289 struct stat buf;
290 char * mh_seq; 290 char * mh_seq;
291 char filename[PATH_MAX]; 291 char filename[PATH_MAX];
292 int res; 292 int res;
293 int r; 293 int r;
294 uint32_t max_index; 294 uint32_t max_index;
295#if 0 295#if 0
296 int add_folder; 296 int add_folder;
297#endif 297#endif
298 unsigned int i; 298 unsigned int i;
299 299
300 if (stat(folder->fl_filename, &buf) == -1) { 300 if (stat(folder->fl_filename, &buf) == -1) {
301 res = MAILMH_ERROR_FOLDER; 301 res = MAILMH_ERROR_FOLDER;
302 goto err; 302 goto err;
303 } 303 }
304 304
305 if (folder->fl_mtime == buf.st_mtime) { 305 if (folder->fl_mtime == buf.st_mtime) {
306 res = MAILMH_NO_ERROR; 306 res = MAILMH_NO_ERROR;
307 goto err; 307 goto err;
308 } 308 }
309 309
310 folder->fl_mtime = buf.st_mtime; 310 folder->fl_mtime = buf.st_mtime;
311 311
312 d = opendir(folder->fl_filename); 312 d = opendir(folder->fl_filename);
313 if (d == NULL) { 313 if (d == NULL) {
314 res = MAILMH_ERROR_FOLDER; 314 res = MAILMH_ERROR_FOLDER;
315 goto err; 315 goto err;
316 } 316 }
317 317
318 max_index = 0; 318 max_index = 0;
319 319
320#if 0 320#if 0
321 if (folder->fl_subfolders_tab->len == 0) 321 if (folder->fl_subfolders_tab->len == 0)
322 add_folder = 1; 322 add_folder = 1;
323 else 323 else
324 add_folder = 0; 324 add_folder = 0;
325#endif 325#endif
326 326
327 /* clear the message list */ 327 /* clear the message list */
328 328
329 for(i = 0 ; i < carray_count(folder->fl_msgs_tab) ; i ++) { 329 for(i = 0 ; i < carray_count(folder->fl_msgs_tab) ; i ++) {
330 struct mailmh_msg_info * msg_info; 330 struct mailmh_msg_info * msg_info;
331 chashdatum key; 331 chashdatum key;
332 332
333 msg_info = carray_get(folder->fl_msgs_tab, i); 333 msg_info = carray_get(folder->fl_msgs_tab, i);
334 if (msg_info == NULL) 334 if (msg_info == NULL)
335 continue; 335 continue;
336 336
337#if 0 337#if 0
338 cinthash_remove(folder->fl_msgs_hash, msg_info->msg_index); 338 cinthash_remove(folder->fl_msgs_hash, msg_info->msg_index);
339#endif 339#endif
340 key.data = &msg_info->msg_index; 340 key.data = &msg_info->msg_index;
341 key.len = sizeof(msg_info->msg_index); 341 key.len = sizeof(msg_info->msg_index);
342 chash_delete(folder->fl_msgs_hash, &key, NULL); 342 chash_delete(folder->fl_msgs_hash, &key, NULL);
343 343
344 mailmh_msg_info_free(msg_info); 344 mailmh_msg_info_free(msg_info);
345 } 345 }
346 346
347 carray_set_size(folder->fl_msgs_tab, 0); 347 carray_set_size(folder->fl_msgs_tab, 0);
348 348
349 do { 349 do {
350 uint32_t index; 350 uint32_t index;
351 351
352 ent = readdir(d); 352 ent = readdir(d);
353 353
354 if (ent != NULL) { 354 if (ent != NULL) {
355 355
356 snprintf(filename, PATH_MAX, 356 snprintf(filename, PATH_MAX,
357 "%s%c%s", folder->fl_filename, MAIL_DIR_SEPARATOR, ent->d_name); 357 "%s%c%s", folder->fl_filename, MAIL_DIR_SEPARATOR, ent->d_name);
358 358
359 if (stat(filename, &buf) == -1) 359 if (stat(filename, &buf) == -1)
360 continue; 360 continue;
361 361
362 if (S_ISREG(buf.st_mode)) { 362 if (S_ISREG(buf.st_mode)) {
363 index = strtoul(ent->d_name, NULL, 10); 363 index = strtoul(ent->d_name, NULL, 10);
364 if (index != 0) { 364 if (index != 0) {
365 struct mailmh_msg_info * msg_info; 365 struct mailmh_msg_info * msg_info;
366 unsigned int array_index; 366 unsigned int array_index;
367 chashdatum key; 367 chashdatum key;
368 chashdatum data; 368 chashdatum data;
369 369
370 msg_info = mailmh_msg_info_new(index, buf.st_size, buf.st_mtime); 370 msg_info = mailmh_msg_info_new(index, buf.st_size, buf.st_mtime);
371 if (msg_info == NULL) { 371 if (msg_info == NULL) {
372 res = MAILMH_ERROR_MEMORY; 372 res = MAILMH_ERROR_MEMORY;
373 goto closedir; 373 goto closedir;
374 } 374 }
375 375
376 r = carray_add(folder->fl_msgs_tab, msg_info, &array_index); 376 r = carray_add(folder->fl_msgs_tab, msg_info, &array_index);
377 if (r < 0) { 377 if (r < 0) {
378 mailmh_msg_info_free(msg_info); 378 mailmh_msg_info_free(msg_info);
379 res = MAILMH_ERROR_MEMORY; 379 res = MAILMH_ERROR_MEMORY;
380 goto closedir; 380 goto closedir;
381 } 381 }
382 msg_info->msg_array_index = array_index; 382 msg_info->msg_array_index = array_index;
383 383
384 if (index > max_index) 384 if (index > max_index)
385 max_index = index; 385 max_index = index;
386 386
387#if 0 387#if 0
388 r = cinthash_add(folder->fl_msgs_hash, msg_info->msg_index, msg_info); 388 r = cinthash_add(folder->fl_msgs_hash, msg_info->msg_index, msg_info);
389#endif 389#endif
390 key.data = &msg_info->msg_index; 390 key.data = &msg_info->msg_index;
391 key.len = sizeof(msg_info->msg_index); 391 key.len = sizeof(msg_info->msg_index);
392 data.data = msg_info; 392 data.data = msg_info;
393 data.len = 0; 393 data.len = 0;
394 394
395 r = chash_set(folder->fl_msgs_hash, &key, &data, NULL); 395 r = chash_set(folder->fl_msgs_hash, &key, &data, NULL);
396 if (r < 0) { 396 if (r < 0) {
397 carray_delete_fast(folder->fl_msgs_tab, msg_info->msg_array_index); 397 carray_delete_fast(folder->fl_msgs_tab, msg_info->msg_array_index);
398 mailmh_msg_info_free(msg_info); 398 mailmh_msg_info_free(msg_info);
399 res = MAILMH_ERROR_MEMORY; 399 res = MAILMH_ERROR_MEMORY;
400 goto closedir; 400 goto closedir;
401 } 401 }
402 } 402 }
403 } 403 }
404 else if (S_ISDIR(buf.st_mode)) { 404 else if (S_ISDIR(buf.st_mode)) {
405 struct mailmh_folder * subfolder; 405 struct mailmh_folder * subfolder;
406 unsigned int array_index; 406 unsigned int array_index;
407 chashdatum key; 407 chashdatum key;
408 chashdatum data; 408 chashdatum data;
409 409
410 if (ent->d_name[0] == '.') { 410 if (ent->d_name[0] == '.') {
411 if (ent->d_name[1] == 0) 411 if (ent->d_name[1] == 0)
412 continue; 412 continue;
413 if ((ent->d_name[1] == '.') && (ent->d_name[2] == 0)) 413 if ((ent->d_name[1] == '.') && (ent->d_name[2] == 0))
414 continue; 414 continue;
415 } 415 }
416 416
417 key.data = ent->d_name; 417 key.data = ent->d_name;
418 key.len = strlen(ent->d_name); 418 key.len = strlen(ent->d_name);
419 r = chash_get(folder->fl_subfolders_hash, &key, &data); 419 r = chash_get(folder->fl_subfolders_hash, &key, &data);
420 if (r < 0) { 420 if (r < 0) {
421 subfolder = mailmh_folder_new(folder, ent->d_name); 421 subfolder = mailmh_folder_new(folder, ent->d_name);
422 if (subfolder == NULL) { 422 if (subfolder == NULL) {
423 res = MAILMH_ERROR_MEMORY; 423 res = MAILMH_ERROR_MEMORY;
424 goto closedir; 424 goto closedir;
425 } 425 }
426 426
427 r = carray_add(folder->fl_subfolders_tab, subfolder, &array_index); 427 r = carray_add(folder->fl_subfolders_tab, subfolder, &array_index);
428 if (r < 0) { 428 if (r < 0) {
429 mailmh_folder_free(subfolder); 429 mailmh_folder_free(subfolder);
430 res = MAILMH_ERROR_MEMORY; 430 res = MAILMH_ERROR_MEMORY;
431 goto closedir; 431 goto closedir;
432 } 432 }
433 subfolder->fl_array_index = array_index; 433 subfolder->fl_array_index = array_index;
434 434
435 key.data = subfolder->fl_filename; 435 key.data = subfolder->fl_filename;
436 key.len = strlen(subfolder->fl_filename); 436 key.len = strlen(subfolder->fl_filename);
437 data.data = subfolder; 437 data.data = subfolder;
438 data.len = 0; 438 data.len = 0;
439 r = chash_set(folder->fl_subfolders_hash, &key, &data, NULL); 439 r = chash_set(folder->fl_subfolders_hash, &key, &data, NULL);
440 if (r < 0) { 440 if (r < 0) {
441 carray_delete_fast(folder->fl_subfolders_tab, subfolder->fl_array_index); 441 carray_delete_fast(folder->fl_subfolders_tab, subfolder->fl_array_index);
442 mailmh_folder_free(subfolder); 442 mailmh_folder_free(subfolder);
443 res = MAILMH_ERROR_MEMORY; 443 res = MAILMH_ERROR_MEMORY;
444 goto closedir; 444 goto closedir;
445 } 445 }
446 } 446 }
447 } 447 }
448 } 448 }
449 } 449 }
450 while (ent != NULL); 450 while (ent != NULL);
451 451
452 folder->fl_max_index = max_index; 452 folder->fl_max_index = max_index;
453 453
454 mh_seq = malloc(strlen(folder->fl_filename) + 2 + sizeof(".mh_sequences")); 454 mh_seq = malloc(strlen(folder->fl_filename) + 2 + sizeof(".mh_sequences"));
455 if (mh_seq == NULL) { 455 if (mh_seq == NULL) {
456 res = MAILMH_ERROR_MEMORY; 456 res = MAILMH_ERROR_MEMORY;
457 goto closedir; 457 goto closedir;
458 } 458 }
459 strcpy(mh_seq, folder->fl_filename); 459 strcpy(mh_seq, folder->fl_filename);
460 strcat(mh_seq, MAIL_DIR_SEPARATOR_S); 460 strcat(mh_seq, MAIL_DIR_SEPARATOR_S);
461 strcat(mh_seq, ".mh_sequences"); 461 strcat(mh_seq, ".mh_sequences");
462 462
463 if (stat(mh_seq, &buf) == -1) { 463 if (stat(mh_seq, &buf) == -1) {
464 int fd; 464 int fd;
465 465
466 fd = creat(mh_seq, S_IRUSR | S_IWUSR); 466 fd = creat(mh_seq, S_IRUSR | S_IWUSR);
467 if (fd != -1) 467 if (fd != -1)
468 close(fd); 468 close(fd);
469 } 469 }
470 free(mh_seq); 470 free(mh_seq);
471 471
472 closedir(d); 472 closedir(d);
473 473
474 return MAILMH_NO_ERROR; 474 return MAILMH_NO_ERROR;
475 475
476 closedir: 476 closedir:
477 closedir(d); 477 closedir(d);
478 err: 478 err:
479 return res; 479 return res;
480} 480}
481 481
482int mailmh_folder_add_subfolder(struct mailmh_folder * parent, 482int mailmh_folder_add_subfolder(struct mailmh_folder * parent,
483 const char * name) 483 const char * name)
484{ 484{
485 char * foldername; 485 char * foldername;
486 int r; 486 int r;
487 struct mailmh_folder * folder; 487 struct mailmh_folder * folder;
488 unsigned int array_index; 488 unsigned int array_index;
489 chashdatum key; 489 chashdatum key;
490 chashdatum data; 490 chashdatum data;
491 491
492 foldername = malloc(strlen(parent->fl_filename) + strlen(name) + 2); 492 foldername = malloc(strlen(parent->fl_filename) + strlen(name) + 2);
493 if (foldername == NULL) 493 if (foldername == NULL)
494 return MAILMH_ERROR_MEMORY; 494 return MAILMH_ERROR_MEMORY;
495 strcpy(foldername, parent->fl_filename); 495 strcpy(foldername, parent->fl_filename);
496 strcat(foldername, MAIL_DIR_SEPARATOR_S); 496 strcat(foldername, MAIL_DIR_SEPARATOR_S);
497 strcat(foldername, name); 497 strcat(foldername, name);
498 498
499 r = mkdir(foldername, 0700); 499 r = mkdir(foldername, 0700);
500 free(foldername); 500 free(foldername);
501 501
502 if (r < 0) 502 if (r < 0)
503 return MAILMH_ERROR_FOLDER; 503 return MAILMH_ERROR_FOLDER;
504 504
505 folder = mailmh_folder_new(parent, name); 505 folder = mailmh_folder_new(parent, name);
506 if (folder == NULL) 506 if (folder == NULL)
507 return MAILMH_ERROR_MEMORY; 507 return MAILMH_ERROR_MEMORY;
508 508
509 r = carray_add(parent->fl_subfolders_tab, folder, &array_index); 509 r = carray_add(parent->fl_subfolders_tab, folder, &array_index);
510 if (r < 0) { 510 if (r < 0) {
511 mailmh_folder_free(folder); 511 mailmh_folder_free(folder);
512 return MAILMH_ERROR_MEMORY; 512 return MAILMH_ERROR_MEMORY;
513 } 513 }
514 folder->fl_array_index = array_index; 514 folder->fl_array_index = array_index;
515 515
516 key.data = folder->fl_filename; 516 key.data = folder->fl_filename;
517 key.len = strlen(folder->fl_filename); 517 key.len = strlen(folder->fl_filename);
518 data.data = folder; 518 data.data = folder;
519 data.len = 0; 519 data.len = 0;
520 520
521 r = chash_set(parent->fl_subfolders_hash, &key, &data, NULL); 521 r = chash_set(parent->fl_subfolders_hash, &key, &data, NULL);
522 if (r < 0) { 522 if (r < 0) {
523 carray_delete_fast(folder->fl_subfolders_tab, folder->fl_array_index); 523 carray_delete_fast(folder->fl_subfolders_tab, folder->fl_array_index);
524 mailmh_folder_free(folder); 524 mailmh_folder_free(folder);
525 return MAILMH_ERROR_MEMORY; 525 return MAILMH_ERROR_MEMORY;
526 } 526 }
527 527
528 return MAILMH_NO_ERROR; 528 return MAILMH_NO_ERROR;
529} 529}
530 530
531int mailmh_folder_remove_subfolder(struct mailmh_folder * folder) 531int mailmh_folder_remove_subfolder(struct mailmh_folder * folder)
532{ 532{
533 struct mailmh_folder * parent; 533 struct mailmh_folder * parent;
534 chashdatum key; 534 chashdatum key;
535 chashdatum data; 535 chashdatum data;
536 int r; 536 int r;
537 537
538 parent = folder->fl_parent; 538 parent = folder->fl_parent;
539 539
540 key.data = folder->fl_filename; 540 key.data = folder->fl_filename;
541 key.len = strlen(folder->fl_filename); 541 key.len = strlen(folder->fl_filename);
542 542
543 r = chash_get(parent->fl_subfolders_hash, &key, &data); 543 r = chash_get(parent->fl_subfolders_hash, &key, &data);
544 if (r < 0) 544 if (r < 0)
545 return MAILMH_ERROR_FOLDER; 545 return MAILMH_ERROR_FOLDER;
546 546
547 chash_delete(parent->fl_subfolders_hash, &key, NULL); 547 chash_delete(parent->fl_subfolders_hash, &key, NULL);
548 carray_delete_fast(parent->fl_subfolders_tab, folder->fl_array_index); 548 carray_delete_fast(parent->fl_subfolders_tab, folder->fl_array_index);
549 549
550 mailmh_folder_free(folder); 550 mailmh_folder_free(folder);
551 551
552 return MAILMH_NO_ERROR; 552 return MAILMH_NO_ERROR;
553 553
554} 554}
555 555
556int mailmh_folder_rename_subfolder(struct mailmh_folder * src_folder, 556int mailmh_folder_rename_subfolder(struct mailmh_folder * src_folder,
557 struct mailmh_folder * dst_folder, 557 struct mailmh_folder * dst_folder,
558 const char * new_name) 558 const char * new_name)
559{ 559{
560 int r; 560 int r;
561 struct mailmh_folder * folder; 561 struct mailmh_folder * folder;
562 struct mailmh_folder * parent; 562 struct mailmh_folder * parent;
563 char * new_foldername; 563 char * new_foldername;
564 564
565 parent = src_folder->fl_parent; 565 parent = src_folder->fl_parent;
566 if (parent == NULL) 566 if (parent == NULL)
567 return MAILMH_ERROR_RENAME; 567 return MAILMH_ERROR_RENAME;
568 568
569 new_foldername = malloc(strlen(dst_folder->fl_filename) + 2 + strlen(new_name)); 569 new_foldername = malloc(strlen(dst_folder->fl_filename) + 2 + strlen(new_name));
570 if (new_foldername == NULL) 570 if (new_foldername == NULL)
571 return MAILMH_ERROR_MEMORY; 571 return MAILMH_ERROR_MEMORY;
572 572
573 strcpy(new_foldername, dst_folder->fl_filename); 573 strcpy(new_foldername, dst_folder->fl_filename);
574 strcat(new_foldername, MAIL_DIR_SEPARATOR_S); 574 strcat(new_foldername, MAIL_DIR_SEPARATOR_S);
575 strcat(new_foldername, new_name); 575 strcat(new_foldername, new_name);
576 576
577 r = rename(src_folder->fl_filename, new_foldername); 577 r = rename(src_folder->fl_filename, new_foldername);
578 free(new_foldername); 578 free(new_foldername);
579 if (r < 0) 579 if (r < 0)
580 return MAILMH_ERROR_RENAME; 580 return MAILMH_ERROR_RENAME;
581 581
582 r = mailmh_folder_remove_subfolder(src_folder); 582 r = mailmh_folder_remove_subfolder(src_folder);
583 if (r != MAILMH_NO_ERROR) 583 if (r != MAILMH_NO_ERROR)
584 return r; 584 return r;
585 585
586 folder = mailmh_folder_new(dst_folder, new_name); 586 folder = mailmh_folder_new(dst_folder, new_name);
587 if (folder == NULL) 587 if (folder == NULL)
588 return MAILMH_ERROR_MEMORY; 588 return MAILMH_ERROR_MEMORY;
589 589
590 r = carray_add(parent->fl_subfolders_tab, folder, NULL); 590 r = carray_add(parent->fl_subfolders_tab, folder, NULL);
591 if (r < 0) { 591 if (r < 0) {
592 mailmh_folder_free(folder); 592 mailmh_folder_free(folder);
593 return MAILMH_ERROR_MEMORY; 593 return MAILMH_ERROR_MEMORY;
594 } 594 }
595 595
596 return MAILMH_NO_ERROR; 596 return MAILMH_NO_ERROR;
597} 597}
598 598
599#define MAX_TRY_ALLOC 32 599#define MAX_TRY_ALLOC 32
600 600
601/* initial file MUST be in the same directory */ 601/* initial file MUST be in the same directory */
602 602
603static int mailmh_folder_alloc_msg(struct mailmh_folder * folder, 603static int mailmh_folder_alloc_msg(struct mailmh_folder * folder,
604 char * filename, uint32_t * result) 604 char * filename, uint32_t * result)
605{ 605{
606 uint32_t max; 606 uint32_t max;
607 uint32_t k; 607 uint32_t k;
608 char * new_filename; 608 char * new_filename;
609 size_t len; 609 size_t len;
610 struct stat f_stat;
610 611
611 len = strlen(folder->fl_filename) + 20; 612 len = strlen(folder->fl_filename) + 20;
612 new_filename = malloc(len); 613 new_filename = malloc(len);
613 if (new_filename == NULL) 614 if (new_filename == NULL)
614 return MAILMH_ERROR_MEMORY; 615 return MAILMH_ERROR_MEMORY;
615 616
616 max = folder->fl_max_index + 1; 617 max = folder->fl_max_index + 1;
617 618
619 //fprintf(stderr,"mailmh_folder_alloc_msg filename: %s \n", filename);
618 k = 0; 620 k = 0;
619 while (k < MAX_TRY_ALLOC) { 621 while (k < MAX_TRY_ALLOC) {
620 snprintf(new_filename, len, "%s%c%lu", folder->fl_filename, 622 snprintf(new_filename, len, "%s%c%lu", folder->fl_filename,
621 MAIL_DIR_SEPARATOR, (unsigned long) (max + k)); 623 MAIL_DIR_SEPARATOR, (unsigned long) (max + k));
622 624 //fprintf(stderr,"mailmh_folder_alloc_msg new_filename: %s \n", new_filename);
623 if (link(filename, new_filename) == 0) { 625 if ( stat( new_filename, &f_stat ) == -1 ) {
626 // if (link(filename, new_filename) == 0) {
624 int r; 627 int r;
625 628 //fprintf(stderr,"filename found \n");
629 //unlink(filename);
630 rename (filename,new_filename );
626 free(new_filename); 631 free(new_filename);
627 unlink(filename);
628 632
629 if (k > MAX_TRY_ALLOC / 2) { 633 if (k > MAX_TRY_ALLOC / 2) {
630 r = mailmh_folder_update(folder); 634 r = mailmh_folder_update(folder);
631 /* ignore errors */ 635 /* ignore errors */
632 } 636 }
633 637
634 * result = max + k; 638 * result = max + k;
635 639
636 folder->fl_max_index = max + k; 640 folder->fl_max_index = max + k;
637 641
638 return MAILMH_NO_ERROR; 642 return MAILMH_NO_ERROR;
639 } 643 }
640 else if (errno == EXDEV) { 644 else if (errno == EXDEV) {
641 free(filename); 645 free(filename);
642 return MAILMH_ERROR_FOLDER; 646 return MAILMH_ERROR_FOLDER;
643 } 647 }
644 k ++; 648 k ++;
645 } 649 }
646 650
647 free(new_filename); 651 free(new_filename);
648 652
649 return MAILMH_ERROR_FOLDER; 653 return MAILMH_ERROR_FOLDER;
650} 654}
651 655
652int mailmh_folder_get_message_filename(struct mailmh_folder * folder, 656int mailmh_folder_get_message_filename(struct mailmh_folder * folder,
653 uint32_t index, char ** result) 657 uint32_t index, char ** result)
654{ 658{
655 char * filename; 659 char * filename;
656 int len; 660 int len;
657 661
658#if 0 662#if 0
659 r = mailmh_folder_update(folder); 663 r = mailmh_folder_update(folder);
660 if (r != MAILMH_NO_ERROR) 664 if (r != MAILMH_NO_ERROR)
661 return r; 665 return r;
662#endif 666#endif
663 667
664 len = strlen(folder->fl_filename) + 20; 668 len = strlen(folder->fl_filename) + 20;
665 filename = malloc(len); 669 filename = malloc(len);
666 if (filename == NULL) 670 if (filename == NULL)
667 return MAILMH_ERROR_MEMORY; 671 return MAILMH_ERROR_MEMORY;
668 672
669 snprintf(filename, len, "%s%c%lu", folder->fl_filename, MAIL_DIR_SEPARATOR, 673 snprintf(filename, len, "%s%c%lu", folder->fl_filename, MAIL_DIR_SEPARATOR,
670 (unsigned long) index); 674 (unsigned long) index);
671 675
672 * result = filename; 676 * result = filename;
673 677
674 return MAILMH_NO_ERROR;; 678 return MAILMH_NO_ERROR;;
675} 679}
676 680
677 681
678int mailmh_folder_get_message_fd(struct mailmh_folder * folder, 682int mailmh_folder_get_message_fd(struct mailmh_folder * folder,
679 uint32_t index, int flags, int * result) 683 uint32_t index, int flags, int * result)
680{ 684{
681 char * filename; 685 char * filename;
682 int fd; 686 int fd;
683 int r; 687 int r;
684 688
685#if 0 689#if 0
686 r = mailmh_folder_update(folder); 690 r = mailmh_folder_update(folder);
687 if (r != MAILMH_NO_ERROR) 691 if (r != MAILMH_NO_ERROR)
688 return r; 692 return r;
689#endif 693#endif
690 694
691 r = mailmh_folder_get_message_filename(folder, index, &filename); 695 r = mailmh_folder_get_message_filename(folder, index, &filename);
692 if (r != MAILMH_NO_ERROR) 696 if (r != MAILMH_NO_ERROR)
693 return r; 697 return r;
694 698
695 fd = open(filename, flags); 699 fd = open(filename, flags);
696 free(filename); 700 free(filename);
697 if (fd == -1) 701 if (fd == -1)
698 return MAILMH_ERROR_MSG_NOT_FOUND; 702 return MAILMH_ERROR_MSG_NOT_FOUND;
699 703
700 * result = fd; 704 * result = fd;
701 705
702 return MAILMH_NO_ERROR; 706 return MAILMH_NO_ERROR;
703} 707}
704 708
705int mailmh_folder_get_message_size(struct mailmh_folder * folder, 709int mailmh_folder_get_message_size(struct mailmh_folder * folder,
706 uint32_t index, size_t * result) 710 uint32_t index, size_t * result)
707{ 711{
708 int r; 712 int r;
709 char * filename; 713 char * filename;
710 struct stat buf; 714 struct stat buf;
711 715
712 r = mailmh_folder_get_message_filename(folder, index, &filename); 716 r = mailmh_folder_get_message_filename(folder, index, &filename);
713 if (r != MAILMH_NO_ERROR) 717 if (r != MAILMH_NO_ERROR)
714 return r; 718 return r;
715 719
716 r = stat(filename, &buf); 720 r = stat(filename, &buf);
717 free(filename); 721 free(filename);
718 if (r < 0) 722 if (r < 0)
719 return MAILMH_ERROR_FILE; 723 return MAILMH_ERROR_FILE;
720 724
721 * result = buf.st_size; 725 * result = buf.st_size;
722 726
723 return MAILMH_NO_ERROR; 727 return MAILMH_NO_ERROR;
724} 728}
725 729
726int mailmh_folder_add_message_uid(struct mailmh_folder * folder, 730int mailmh_folder_add_message_uid(struct mailmh_folder * folder,
727 const char * message, size_t size, 731 const char * message, size_t size,
728 uint32_t * pindex) 732 uint32_t * pindex)
729{ 733{
730 char * tmpname; 734 char * tmpname;
731 int fd; 735 int fd;
732 size_t namesize; 736 size_t namesize;
733 size_t left; 737 size_t left;
734 ssize_t res; 738 ssize_t res;
735 struct mailmh_msg_info * msg_info; 739 struct mailmh_msg_info * msg_info;
736 uint32_t index; 740 uint32_t index;
737 int error; 741 int error;
738 int r; 742 int r;
739 unsigned int array_index; 743 unsigned int array_index;
740 struct stat buf; 744 struct stat buf;
741 chashdatum key; 745 chashdatum key;
742 chashdatum data; 746 chashdatum data;
743 747
744#if 0 748#if 0
745 r = mailmh_folder_update(folder); 749 r = mailmh_folder_update(folder);
746 if (r != MAILMH_NO_ERROR) { 750 if (r != MAILMH_NO_ERROR) {
747 error = r; 751 error = r;
748 goto err; 752 goto err;
749 } 753 }
750#endif 754#endif
751 755
752 namesize = strlen(folder->fl_filename) + 20; 756 namesize = strlen(folder->fl_filename) + 20;
753 tmpname = malloc(namesize); 757 tmpname = malloc(namesize);
754 snprintf(tmpname, namesize, "%s%ctmpXXXXXX", 758 snprintf(tmpname, namesize, "%s%ctmpXXXXXX",
755 folder->fl_filename, MAIL_DIR_SEPARATOR); 759 folder->fl_filename, MAIL_DIR_SEPARATOR);
756 fd = mkstemp(tmpname); 760 fd = mkstemp(tmpname);
757 if (fd < 0) { 761 if (fd < 0) {
758 error = MAILMH_ERROR_FILE; 762 error = MAILMH_ERROR_FILE;
759 goto free; 763 goto free;
760 } 764 }
761 765
762 left = size; 766 left = size;
763 while (left > 0) { 767 while (left > 0) {
764 res = write(fd, message, left); 768 res = write(fd, message, left);
765 if (res == -1) { 769 if (res == -1) {
766 close(fd); 770 close(fd);
767 error = MAILMH_ERROR_FILE; 771 error = MAILMH_ERROR_FILE;
768 goto free; 772 goto free;
769 } 773 }
770 774
771 left -= res; 775 left -= res;
772 } 776 }
773 close(fd); 777 close(fd);
774 778
775 r = stat(tmpname, &buf); 779 r = stat(tmpname, &buf);
776 if (r < 0) { 780 if (r < 0) {
777 error = MAILMH_ERROR_FILE; 781 error = MAILMH_ERROR_FILE;
778 goto free; 782 goto free;
779 } 783 }
780 784
781 r = mailmh_folder_alloc_msg(folder, tmpname, &index); 785 r = mailmh_folder_alloc_msg(folder, tmpname, &index);
782 if (r != MAILMH_NO_ERROR) { 786 if (r != MAILMH_NO_ERROR) {
783 unlink(tmpname); 787 unlink(tmpname);
784 error = MAILMH_ERROR_COULD_NOT_ALLOC_MSG; 788 error = MAILMH_ERROR_COULD_NOT_ALLOC_MSG;
785 goto free; 789 goto free;
786 } 790 }
787 free(tmpname); 791 free(tmpname);
788 792
789 msg_info = mailmh_msg_info_new(index, size, buf.st_mtime); 793 msg_info = mailmh_msg_info_new(index, size, buf.st_mtime);
790 if (msg_info == NULL) { 794 if (msg_info == NULL) {
791 mailmh_folder_remove_message(folder, index); 795 mailmh_folder_remove_message(folder, index);
792 error = MAILMH_ERROR_MEMORY; 796 error = MAILMH_ERROR_MEMORY;
793 goto err; 797 goto err;
794 } 798 }
795 799
796 r = carray_add(folder->fl_msgs_tab, msg_info, &array_index); 800 r = carray_add(folder->fl_msgs_tab, msg_info, &array_index);
797 if (r < 0) { 801 if (r < 0) {
798 mailmh_folder_remove_message(folder, index); 802 mailmh_folder_remove_message(folder, index);
799 mailmh_msg_info_free(msg_info); 803 mailmh_msg_info_free(msg_info);
800 error = MAILMH_ERROR_MEMORY; 804 error = MAILMH_ERROR_MEMORY;
801 goto err; 805 goto err;
802 } 806 }
803 msg_info->msg_array_index = array_index; 807 msg_info->msg_array_index = array_index;
804 808
805#if 0 809#if 0
806 r = cinthash_add(folder->fl_msgs_hash, index, msg_info); 810 r = cinthash_add(folder->fl_msgs_hash, index, msg_info);
807#endif 811#endif
808 key.data = &index; 812 key.data = &index;
809 key.len = sizeof(index); 813 key.len = sizeof(index);
810 data.data = msg_info; 814 data.data = msg_info;
811 data.len = 0; 815 data.len = 0;
812 816
813 if (pindex != NULL) 817 if (pindex != NULL)
814 * pindex = index; 818 * pindex = index;
815 819
816 r = chash_set(folder->fl_msgs_hash, &key, &data, NULL); 820 r = chash_set(folder->fl_msgs_hash, &key, &data, NULL);
817 if (r < 0) { 821 if (r < 0) {
818 carray_delete_fast(folder->fl_msgs_tab, msg_info->msg_array_index); 822 carray_delete_fast(folder->fl_msgs_tab, msg_info->msg_array_index);
819 mailmh_msg_info_free(msg_info); 823 mailmh_msg_info_free(msg_info);
820 error = MAILMH_ERROR_MEMORY; 824 error = MAILMH_ERROR_MEMORY;
821 goto err; 825 goto err;
822 } 826 }
823 827
824 return MAILMH_NO_ERROR; 828 return MAILMH_NO_ERROR;
825 829
826 free: 830 free:
827 free(tmpname); 831 free(tmpname);
828 err: 832 err:
829 return error; 833 return error;
830} 834}
831 835
832int mailmh_folder_add_message(struct mailmh_folder * folder, 836int mailmh_folder_add_message(struct mailmh_folder * folder,
833 const char * message, size_t size) 837 const char * message, size_t size)
834{ 838{
835 return mailmh_folder_add_message_uid(folder, message, size, NULL); 839 return mailmh_folder_add_message_uid(folder, message, size, NULL);
836} 840}
837 841
838int mailmh_folder_add_message_file_uid(struct mailmh_folder * folder, 842int mailmh_folder_add_message_file_uid(struct mailmh_folder * folder,
839 int fd, uint32_t * pindex) 843 int fd, uint32_t * pindex)
840{ 844{
841 char * message; 845 char * message;
842 struct stat buf; 846 struct stat buf;
843 int r; 847 int r;
844 848
845#if 0 849#if 0
846 r = mailmh_folder_update(folder); 850 r = mailmh_folder_update(folder);
847 if (r != MAILMH_NO_ERROR) 851 if (r != MAILMH_NO_ERROR)
848 return r; 852 return r;
849#endif 853#endif
850 854
851 if (fstat(fd, &buf) == -1) 855 if (fstat(fd, &buf) == -1)
852 return MAILMH_ERROR_FILE; 856 return MAILMH_ERROR_FILE;
853 857
854 message = mmap(NULL, buf.st_size, PROT_READ, MAP_PRIVATE, fd, 0); 858 message = mmap(NULL, buf.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
855 if (message == MAP_FAILED) 859 if (message == MAP_FAILED)
856 return MAILMH_ERROR_FILE; 860 return MAILMH_ERROR_FILE;
857 861
858 r = mailmh_folder_add_message_uid(folder, message, buf.st_size, pindex); 862 r = mailmh_folder_add_message_uid(folder, message, buf.st_size, pindex);
859 863
860 munmap(message, buf.st_size); 864 munmap(message, buf.st_size);
861 865
862 return r; 866 return r;
863} 867}
864 868
865int mailmh_folder_add_message_file(struct mailmh_folder * folder, 869int mailmh_folder_add_message_file(struct mailmh_folder * folder,
866 int fd) 870 int fd)
867{ 871{
868 return mailmh_folder_add_message_file_uid(folder, fd, NULL); 872 return mailmh_folder_add_message_file_uid(folder, fd, NULL);
869} 873}
870 874
871int mailmh_folder_remove_message(struct mailmh_folder * folder, 875int mailmh_folder_remove_message(struct mailmh_folder * folder,
872 uint32_t index) 876 uint32_t index)
873{ 877{
874 char * filename; 878 char * filename;
875 struct mailmh_msg_info * msg_info; 879 struct mailmh_msg_info * msg_info;
876 int res; 880 int res;
877 int r; 881 int r;
878 chashdatum key; 882 chashdatum key;
879 chashdatum data; 883 chashdatum data;
880 884
881#if 0 885#if 0
882 r = mailmh_folder_update(folder); 886 r = mailmh_folder_update(folder);
883 if (r != MAILMH_NO_ERROR) { 887 if (r != MAILMH_NO_ERROR) {
884 res = r; 888 res = r;
885 goto err; 889 goto err;
886 } 890 }
887#endif 891#endif
888 892
889 r = mailmh_folder_get_message_filename(folder, index, &filename); 893 r = mailmh_folder_get_message_filename(folder, index, &filename);
890 if (filename == NULL) { 894 if (filename == NULL) {
891 res = r; 895 res = r;
892 goto err; 896 goto err;
893 } 897 }
894 898
895 if (unlink(filename) == -1) { 899 if (unlink(filename) == -1) {
896 res = MAILMH_ERROR_FILE; 900 res = MAILMH_ERROR_FILE;
897 goto free; 901 goto free;
898 } 902 }
899 903
900 key.data = &index; 904 key.data = &index;
901 key.len = sizeof(index); 905 key.len = sizeof(index);
902 r = chash_get(folder->fl_msgs_hash, &key, &data); 906 r = chash_get(folder->fl_msgs_hash, &key, &data);
903#if 0 907#if 0
904 msg_info = cinthash_find(folder->fl_msgs_hash, index); 908 msg_info = cinthash_find(folder->fl_msgs_hash, index);
905#endif 909#endif
906 if (r == 0) { 910 if (r == 0) {
907 msg_info = data.data; 911 msg_info = data.data;
908 912
909 carray_delete_fast(folder->fl_msgs_tab, msg_info->msg_array_index); 913 carray_delete_fast(folder->fl_msgs_tab, msg_info->msg_array_index);
910#if 0 914#if 0
911 cinthash_remove(folder->fl_msgs_hash, index); 915 cinthash_remove(folder->fl_msgs_hash, index);
912#endif 916#endif
913 chash_delete(folder->fl_msgs_hash, &key, NULL); 917 chash_delete(folder->fl_msgs_hash, &key, NULL);
914 } 918 }
915 919
916 return MAILMH_NO_ERROR; 920 return MAILMH_NO_ERROR;
917 921
918 free: 922 free:
919 free(filename); 923 free(filename);
920 err: 924 err:
921 return res; 925 return res;
922} 926}
923 927
924 928
925int mailmh_folder_move_message(struct mailmh_folder * dest_folder, 929int mailmh_folder_move_message(struct mailmh_folder * dest_folder,
926 struct mailmh_folder * src_folder, 930 struct mailmh_folder * src_folder,
927 uint32_t index) 931 uint32_t index)
928{ 932{
929 int fd; 933 int fd;
930 char * filename; 934 char * filename;
931 int r; 935 int r;
932 936
933#if 0 937#if 0
934 r = mailmh_folder_update(dest_folder); 938 r = mailmh_folder_update(dest_folder);
935 if (r != MAILMH_NO_ERROR) 939 if (r != MAILMH_NO_ERROR)
936 return r; 940 return r;
937 r = mailmh_folder_update(src_folder); 941 r = mailmh_folder_update(src_folder);
938 if (r != MAILMH_NO_ERROR) 942 if (r != MAILMH_NO_ERROR)
939 return r; 943 return r;
940#endif 944#endif
941 945
942 /* move on the same filesystem */ 946 /* move on the same filesystem */
943 r = mailmh_folder_get_message_filename(src_folder, index, &filename); 947 r = mailmh_folder_get_message_filename(src_folder, index, &filename);
944 if (r != MAILMH_NO_ERROR) 948 if (r != MAILMH_NO_ERROR)
945 return r; 949 return r;
946 950
947 r = mailmh_folder_alloc_msg(dest_folder, filename, &index); 951 r = mailmh_folder_alloc_msg(dest_folder, filename, &index);
948 free(filename); 952 free(filename);
949 if (r == MAILMH_NO_ERROR) 953 if (r == MAILMH_NO_ERROR)
950 return MAILMH_NO_ERROR; 954 return MAILMH_NO_ERROR;
951 955
952 /* move on the different filesystems */ 956 /* move on the different filesystems */
953 r = mailmh_folder_get_message_fd(src_folder, index, O_RDONLY, &fd); 957 r = mailmh_folder_get_message_fd(src_folder, index, O_RDONLY, &fd);
954 if (r != MAILMH_NO_ERROR) 958 if (r != MAILMH_NO_ERROR)
955 return r; 959 return r;
956 960
957 r = mailmh_folder_add_message_file(dest_folder, fd); 961 r = mailmh_folder_add_message_file(dest_folder, fd);
958 if (r != MAILMH_NO_ERROR) { 962 if (r != MAILMH_NO_ERROR) {
959 close(fd); 963 close(fd);
960 return r; 964 return r;
961 } 965 }
962 966
963 close(fd); 967 close(fd);
964 968
965 r = mailmh_folder_remove_message(src_folder, index); 969 r = mailmh_folder_remove_message(src_folder, index);
966 970
967 return MAILMH_NO_ERROR; 971 return MAILMH_NO_ERROR;
968} 972}
969 973
970unsigned int mailmh_folder_get_message_number(struct mailmh_folder * folder) 974unsigned int mailmh_folder_get_message_number(struct mailmh_folder * folder)
971{ 975{
972 unsigned int i; 976 unsigned int i;
973 unsigned int count; 977 unsigned int count;
974 978
975 count = 0; 979 count = 0;
976 for(i = 0 ; i < carray_count(folder->fl_msgs_tab) ; i ++) 980 for(i = 0 ; i < carray_count(folder->fl_msgs_tab) ; i ++)
977 if (carray_get(folder->fl_msgs_tab, i) != NULL) 981 if (carray_get(folder->fl_msgs_tab, i) != NULL)
978 count ++; 982 count ++;
979 983
980 return count; 984 return count;
981} 985}
diff --git a/korganizer/calendarview.cpp b/korganizer/calendarview.cpp
index 2321087..3908dbb 100644
--- a/korganizer/calendarview.cpp
+++ b/korganizer/calendarview.cpp
@@ -1938,1032 +1938,1036 @@ void CalendarView::readFilterSettings(KConfig *config)
1938 QStringList::ConstIterator end = filterList.end(); 1938 QStringList::ConstIterator end = filterList.end();
1939 while(it != end) { 1939 while(it != end) {
1940 // kdDebug() << " filter: " << (*it) << endl; 1940 // kdDebug() << " filter: " << (*it) << endl;
1941 1941
1942 CalFilter *filter; 1942 CalFilter *filter;
1943 filter = new CalFilter(*it); 1943 filter = new CalFilter(*it);
1944 config->setGroup("Filter_" + (*it)); 1944 config->setGroup("Filter_" + (*it));
1945 //qDebug("readFilterSettings %d ",config->readNumEntry("Criteria",0) ); 1945 //qDebug("readFilterSettings %d ",config->readNumEntry("Criteria",0) );
1946 filter->setCriteria(config->readNumEntry("Criteria",0)); 1946 filter->setCriteria(config->readNumEntry("Criteria",0));
1947 filter->setCategoryList(config->readListEntry("CategoryList")); 1947 filter->setCategoryList(config->readListEntry("CategoryList"));
1948 mFilters.append(filter); 1948 mFilters.append(filter);
1949 1949
1950 ++it; 1950 ++it;
1951 } 1951 }
1952 1952
1953 if (mFilters.count() == 0) { 1953 if (mFilters.count() == 0) {
1954 CalFilter *filter = new CalFilter(i18n("Default")); 1954 CalFilter *filter = new CalFilter(i18n("Default"));
1955 mFilters.append(filter); 1955 mFilters.append(filter);
1956 } 1956 }
1957 mFilterView->updateFilters(); 1957 mFilterView->updateFilters();
1958 config->setGroup("FilterView"); 1958 config->setGroup("FilterView");
1959 1959
1960 mFilterView->blockSignals(true); 1960 mFilterView->blockSignals(true);
1961 mFilterView->setFiltersEnabled(config->readBoolEntry("FilterEnabled")); 1961 mFilterView->setFiltersEnabled(config->readBoolEntry("FilterEnabled"));
1962 mFilterView->setSelectedFilter(config->readEntry("Current Filter")); 1962 mFilterView->setSelectedFilter(config->readEntry("Current Filter"));
1963 mFilterView->blockSignals(false); 1963 mFilterView->blockSignals(false);
1964 // We do it manually to avoid it being done twice by the above calls 1964 // We do it manually to avoid it being done twice by the above calls
1965 updateFilter(); 1965 updateFilter();
1966} 1966}
1967 1967
1968void CalendarView::writeFilterSettings(KConfig *config) 1968void CalendarView::writeFilterSettings(KConfig *config)
1969{ 1969{
1970 // kdDebug() << "CalendarView::writeFilterSettings()" << endl; 1970 // kdDebug() << "CalendarView::writeFilterSettings()" << endl;
1971 1971
1972 QStringList filterList; 1972 QStringList filterList;
1973 1973
1974 CalFilter *filter = mFilters.first(); 1974 CalFilter *filter = mFilters.first();
1975 while(filter) { 1975 while(filter) {
1976 // kdDebug() << " fn: " << filter->name() << endl; 1976 // kdDebug() << " fn: " << filter->name() << endl;
1977 filterList << filter->name(); 1977 filterList << filter->name();
1978 config->setGroup("Filter_" + filter->name()); 1978 config->setGroup("Filter_" + filter->name());
1979 config->writeEntry("Criteria",filter->criteria()); 1979 config->writeEntry("Criteria",filter->criteria());
1980 config->writeEntry("CategoryList",filter->categoryList()); 1980 config->writeEntry("CategoryList",filter->categoryList());
1981 filter = mFilters.next(); 1981 filter = mFilters.next();
1982 } 1982 }
1983 config->setGroup("General"); 1983 config->setGroup("General");
1984 config->writeEntry("CalendarFilters",filterList); 1984 config->writeEntry("CalendarFilters",filterList);
1985 1985
1986 config->setGroup("FilterView"); 1986 config->setGroup("FilterView");
1987 config->writeEntry("FilterEnabled",mFilterView->filtersEnabled()); 1987 config->writeEntry("FilterEnabled",mFilterView->filtersEnabled());
1988 config->writeEntry("Current Filter",mFilterView->selectedFilter()->name()); 1988 config->writeEntry("Current Filter",mFilterView->selectedFilter()->name());
1989} 1989}
1990 1990
1991 1991
1992void CalendarView::goToday() 1992void CalendarView::goToday()
1993{ 1993{
1994 mNavigator->selectToday(); 1994 mNavigator->selectToday();
1995} 1995}
1996 1996
1997void CalendarView::goNext() 1997void CalendarView::goNext()
1998{ 1998{
1999 mNavigator->selectNext(); 1999 mNavigator->selectNext();
2000} 2000}
2001 2001
2002void CalendarView::goPrevious() 2002void CalendarView::goPrevious()
2003{ 2003{
2004 mNavigator->selectPrevious(); 2004 mNavigator->selectPrevious();
2005} 2005}
2006void CalendarView::goNextMonth() 2006void CalendarView::goNextMonth()
2007{ 2007{
2008 mNavigator->selectNextMonth(); 2008 mNavigator->selectNextMonth();
2009} 2009}
2010 2010
2011void CalendarView::goPreviousMonth() 2011void CalendarView::goPreviousMonth()
2012{ 2012{
2013 mNavigator->selectPreviousMonth(); 2013 mNavigator->selectPreviousMonth();
2014} 2014}
2015void CalendarView::writeLocale() 2015void CalendarView::writeLocale()
2016{ 2016{
2017 //KPimGlobalPrefs::instance()->setGlobalConfig(); 2017 //KPimGlobalPrefs::instance()->setGlobalConfig();
2018#if 0 2018#if 0
2019 KGlobal::locale()->setHore24Format( !KOPrefs::instance()->mPreferredTime ); 2019 KGlobal::locale()->setHore24Format( !KOPrefs::instance()->mPreferredTime );
2020 KGlobal::locale()->setWeekStartMonday( !KOPrefs::instance()->mWeekStartsOnSunday ); 2020 KGlobal::locale()->setWeekStartMonday( !KOPrefs::instance()->mWeekStartsOnSunday );
2021 KGlobal::locale()->setIntDateFormat( (KLocale::IntDateFormat)KOPrefs::instance()->mPreferredDate ); 2021 KGlobal::locale()->setIntDateFormat( (KLocale::IntDateFormat)KOPrefs::instance()->mPreferredDate );
2022 KGlobal::locale()->setLanguage( KOPrefs::instance()->mPreferredLanguage ); 2022 KGlobal::locale()->setLanguage( KOPrefs::instance()->mPreferredLanguage );
2023 QString dummy = KOPrefs::instance()->mUserDateFormatLong; 2023 QString dummy = KOPrefs::instance()->mUserDateFormatLong;
2024 KGlobal::locale()->setDateFormat(dummy.replace( QRegExp("K"), QString(",") )); 2024 KGlobal::locale()->setDateFormat(dummy.replace( QRegExp("K"), QString(",") ));
2025 dummy = KOPrefs::instance()->mUserDateFormatShort; 2025 dummy = KOPrefs::instance()->mUserDateFormatShort;
2026 KGlobal::locale()->setDateFormatShort(dummy.replace( QRegExp("K"), QString(",") )); 2026 KGlobal::locale()->setDateFormatShort(dummy.replace( QRegExp("K"), QString(",") ));
2027 KGlobal::locale()->setDaylightSaving( KOPrefs::instance()->mUseDaylightsaving, 2027 KGlobal::locale()->setDaylightSaving( KOPrefs::instance()->mUseDaylightsaving,
2028 KOPrefs::instance()->mDaylightsavingStart, 2028 KOPrefs::instance()->mDaylightsavingStart,
2029 KOPrefs::instance()->mDaylightsavingEnd ); 2029 KOPrefs::instance()->mDaylightsavingEnd );
2030 KGlobal::locale()->setTimezone( KPimGlobalPrefs::instance()->mTimeZoneId ); 2030 KGlobal::locale()->setTimezone( KPimGlobalPrefs::instance()->mTimeZoneId );
2031#endif 2031#endif
2032} 2032}
2033void CalendarView::updateConfig() 2033void CalendarView::updateConfig()
2034{ 2034{
2035 writeLocale(); 2035 writeLocale();
2036 if ( KOPrefs::instance()->mUseAppColors ) 2036 if ( KOPrefs::instance()->mUseAppColors )
2037 QApplication::setPalette( QPalette (KOPrefs::instance()->mAppColor1, KOPrefs::instance()->mAppColor2), true ); 2037 QApplication::setPalette( QPalette (KOPrefs::instance()->mAppColor1, KOPrefs::instance()->mAppColor2), true );
2038 emit configChanged(); 2038 emit configChanged();
2039 mTodoList->updateConfig(); 2039 mTodoList->updateConfig();
2040 // mDateNavigator->setFont ( KOPrefs::instance()->mDateNavigatorFont); 2040 // mDateNavigator->setFont ( KOPrefs::instance()->mDateNavigatorFont);
2041 mCalendar->setTimeZoneId(KPimGlobalPrefs::instance()->mTimeZoneId); 2041 mCalendar->setTimeZoneId(KPimGlobalPrefs::instance()->mTimeZoneId);
2042 // To make the "fill window" configurations work 2042 // To make the "fill window" configurations work
2043 //mViewManager->raiseCurrentView(); 2043 //mViewManager->raiseCurrentView();
2044} 2044}
2045 2045
2046 2046
2047void CalendarView::eventChanged(Event *event) 2047void CalendarView::eventChanged(Event *event)
2048{ 2048{
2049 changeEventDisplay(event,KOGlobals::EVENTEDITED); 2049 changeEventDisplay(event,KOGlobals::EVENTEDITED);
2050 //updateUnmanagedViews(); 2050 //updateUnmanagedViews();
2051} 2051}
2052 2052
2053void CalendarView::eventAdded(Event *event) 2053void CalendarView::eventAdded(Event *event)
2054{ 2054{
2055 changeEventDisplay(event,KOGlobals::EVENTADDED); 2055 changeEventDisplay(event,KOGlobals::EVENTADDED);
2056} 2056}
2057 2057
2058void CalendarView::eventToBeDeleted(Event *) 2058void CalendarView::eventToBeDeleted(Event *)
2059{ 2059{
2060 kdDebug() << "CalendarView::eventToBeDeleted(): to be implemented" << endl; 2060 kdDebug() << "CalendarView::eventToBeDeleted(): to be implemented" << endl;
2061} 2061}
2062 2062
2063void CalendarView::eventDeleted() 2063void CalendarView::eventDeleted()
2064{ 2064{
2065 changeEventDisplay(0,KOGlobals::EVENTDELETED); 2065 changeEventDisplay(0,KOGlobals::EVENTDELETED);
2066} 2066}
2067void CalendarView::changeTodoDisplay(Todo *which, int action) 2067void CalendarView::changeTodoDisplay(Todo *which, int action)
2068{ 2068{
2069 changeIncidenceDisplay((Incidence *)which, action); 2069 changeIncidenceDisplay((Incidence *)which, action);
2070 mDateNavigator->updateView(); //LR 2070 mDateNavigator->updateView(); //LR
2071 //mDialogManager->updateSearchDialog(); 2071 //mDialogManager->updateSearchDialog();
2072 2072
2073 if (which) { 2073 if (which) {
2074 mViewManager->updateWNview(); 2074 mViewManager->updateWNview();
2075 //mTodoList->updateView(); 2075 //mTodoList->updateView();
2076 } 2076 }
2077 2077
2078} 2078}
2079 2079
2080void CalendarView::changeIncidenceDisplay(Incidence *which, int action) 2080void CalendarView::changeIncidenceDisplay(Incidence *which, int action)
2081{ 2081{
2082 updateUnmanagedViews(); 2082 updateUnmanagedViews();
2083 //qDebug(" CalendarView::changeIncidenceDisplay++++++++++++++++++++++++++ %d %d ",which, action ); 2083 //qDebug(" CalendarView::changeIncidenceDisplay++++++++++++++++++++++++++ %d %d ",which, action );
2084 if ( action == KOGlobals::EVENTDELETED ) { //delete 2084 if ( action == KOGlobals::EVENTDELETED ) { //delete
2085 mCalendar->checkAlarmForIncidence( 0, true ); 2085 mCalendar->checkAlarmForIncidence( 0, true );
2086 if ( mEventViewerDialog ) 2086 if ( mEventViewerDialog )
2087 mEventViewerDialog->hide(); 2087 mEventViewerDialog->hide();
2088 } 2088 }
2089 else 2089 else
2090 mCalendar->checkAlarmForIncidence( which , false ); 2090 mCalendar->checkAlarmForIncidence( which , false );
2091} 2091}
2092 2092
2093// most of the changeEventDisplays() right now just call the view's 2093// most of the changeEventDisplays() right now just call the view's
2094// total update mode, but they SHOULD be recoded to be more refresh-efficient. 2094// total update mode, but they SHOULD be recoded to be more refresh-efficient.
2095void CalendarView::changeEventDisplay(Event *which, int action) 2095void CalendarView::changeEventDisplay(Event *which, int action)
2096{ 2096{
2097 // kdDebug() << "CalendarView::changeEventDisplay" << endl; 2097 // kdDebug() << "CalendarView::changeEventDisplay" << endl;
2098 changeIncidenceDisplay((Incidence *)which, action); 2098 changeIncidenceDisplay((Incidence *)which, action);
2099 mDateNavigator->updateView(); 2099 mDateNavigator->updateView();
2100 //mDialogManager->updateSearchDialog(); 2100 //mDialogManager->updateSearchDialog();
2101 2101
2102 if (which) { 2102 if (which) {
2103 // If there is an event view visible update the display 2103 // If there is an event view visible update the display
2104 mViewManager->currentView()->changeEventDisplay(which,action); 2104 mViewManager->currentView()->changeEventDisplay(which,action);
2105 // TODO: check, if update needed 2105 // TODO: check, if update needed
2106 // if (which->getTodoStatus()) { 2106 // if (which->getTodoStatus()) {
2107 mTodoList->updateView(); 2107 mTodoList->updateView();
2108 // } 2108 // }
2109 } else { 2109 } else {
2110 mViewManager->currentView()->updateView(); 2110 mViewManager->currentView()->updateView();
2111 } 2111 }
2112} 2112}
2113 2113
2114 2114
2115void CalendarView::updateTodoViews() 2115void CalendarView::updateTodoViews()
2116{ 2116{
2117 2117
2118 mTodoList->updateView(); 2118 mTodoList->updateView();
2119 mViewManager->currentView()->updateView(); 2119 mViewManager->currentView()->updateView();
2120 2120
2121} 2121}
2122 2122
2123 2123
2124void CalendarView::updateView(const QDate &start, const QDate &end) 2124void CalendarView::updateView(const QDate &start, const QDate &end)
2125{ 2125{
2126 mTodoList->updateView(); 2126 mTodoList->updateView();
2127 mViewManager->updateView(start, end); 2127 mViewManager->updateView(start, end);
2128 //mDateNavigator->updateView(); 2128 //mDateNavigator->updateView();
2129} 2129}
2130 2130
2131void CalendarView::updateView() 2131void CalendarView::updateView()
2132{ 2132{
2133 DateList tmpList = mNavigator->selectedDates(); 2133 DateList tmpList = mNavigator->selectedDates();
2134 2134
2135 // We assume that the navigator only selects consecutive days. 2135 // We assume that the navigator only selects consecutive days.
2136 updateView( tmpList.first(), tmpList.last() ); 2136 updateView( tmpList.first(), tmpList.last() );
2137} 2137}
2138 2138
2139void CalendarView::updateUnmanagedViews() 2139void CalendarView::updateUnmanagedViews()
2140{ 2140{
2141 mDateNavigator->updateDayMatrix(); 2141 mDateNavigator->updateDayMatrix();
2142} 2142}
2143 2143
2144int CalendarView::msgItemDelete() 2144int CalendarView::msgItemDelete()
2145{ 2145{
2146 return KMessageBox::warningContinueCancel(this, 2146 return KMessageBox::warningContinueCancel(this,
2147 i18n("This item will be\npermanently deleted."), 2147 i18n("This item will be\npermanently deleted."),
2148 i18n("KO/Pi Confirmation"),i18n("Delete")); 2148 i18n("KO/Pi Confirmation"),i18n("Delete"));
2149} 2149}
2150 2150
2151 2151
2152void CalendarView::edit_cut() 2152void CalendarView::edit_cut()
2153{ 2153{
2154 Event *anEvent=0; 2154 Event *anEvent=0;
2155 2155
2156 Incidence *incidence = mViewManager->currentView()->selectedIncidences().first(); 2156 Incidence *incidence = mViewManager->currentView()->selectedIncidences().first();
2157 2157
2158 if (mViewManager->currentView()->isEventView()) { 2158 if (mViewManager->currentView()->isEventView()) {
2159 if ( incidence && incidence->type() == "Event" ) { 2159 if ( incidence && incidence->type() == "Event" ) {
2160 anEvent = static_cast<Event *>(incidence); 2160 anEvent = static_cast<Event *>(incidence);
2161 } 2161 }
2162 } 2162 }
2163 2163
2164 if (!anEvent) { 2164 if (!anEvent) {
2165 KNotifyClient::beep(); 2165 KNotifyClient::beep();
2166 return; 2166 return;
2167 } 2167 }
2168 DndFactory factory( mCalendar ); 2168 DndFactory factory( mCalendar );
2169 factory.cutEvent(anEvent); 2169 factory.cutEvent(anEvent);
2170 changeEventDisplay(anEvent, KOGlobals::EVENTDELETED); 2170 changeEventDisplay(anEvent, KOGlobals::EVENTDELETED);
2171} 2171}
2172 2172
2173void CalendarView::edit_copy() 2173void CalendarView::edit_copy()
2174{ 2174{
2175 Event *anEvent=0; 2175 Event *anEvent=0;
2176 2176
2177 Incidence *incidence = mViewManager->currentView()->selectedIncidences().first(); 2177 Incidence *incidence = mViewManager->currentView()->selectedIncidences().first();
2178 2178
2179 if (mViewManager->currentView()->isEventView()) { 2179 if (mViewManager->currentView()->isEventView()) {
2180 if ( incidence && incidence->type() == "Event" ) { 2180 if ( incidence && incidence->type() == "Event" ) {
2181 anEvent = static_cast<Event *>(incidence); 2181 anEvent = static_cast<Event *>(incidence);
2182 } 2182 }
2183 } 2183 }
2184 2184
2185 if (!anEvent) { 2185 if (!anEvent) {
2186 KNotifyClient::beep(); 2186 KNotifyClient::beep();
2187 return; 2187 return;
2188 } 2188 }
2189 DndFactory factory( mCalendar ); 2189 DndFactory factory( mCalendar );
2190 factory.copyEvent(anEvent); 2190 factory.copyEvent(anEvent);
2191} 2191}
2192 2192
2193void CalendarView::edit_paste() 2193void CalendarView::edit_paste()
2194{ 2194{
2195 QDate date = mNavigator->selectedDates().first(); 2195 QDate date = mNavigator->selectedDates().first();
2196 2196
2197 DndFactory factory( mCalendar ); 2197 DndFactory factory( mCalendar );
2198 Event *pastedEvent = factory.pasteEvent( date ); 2198 Event *pastedEvent = factory.pasteEvent( date );
2199 2199
2200 changeEventDisplay( pastedEvent, KOGlobals::EVENTADDED ); 2200 changeEventDisplay( pastedEvent, KOGlobals::EVENTADDED );
2201} 2201}
2202 2202
2203void CalendarView::edit_options() 2203void CalendarView::edit_options()
2204{ 2204{
2205 mDialogManager->showOptionsDialog(); 2205 mDialogManager->showOptionsDialog();
2206 //writeSettings(); 2206 //writeSettings();
2207} 2207}
2208 2208
2209void CalendarView::slotSelectPickerDate( QDate d) 2209void CalendarView::slotSelectPickerDate( QDate d)
2210{ 2210{
2211 mDateFrame->hide(); 2211 mDateFrame->hide();
2212 if ( mDatePickerMode == 1 ) { 2212 if ( mDatePickerMode == 1 ) {
2213 mNavigator->slotDaySelect( d ); 2213 mNavigator->slotDaySelect( d );
2214 } else if ( mDatePickerMode == 2 ) { 2214 } else if ( mDatePickerMode == 2 ) {
2215 if ( mMoveIncidence->type() == "Todo" ) { 2215 if ( mMoveIncidence->type() == "Todo" ) {
2216 Todo * to = (Todo *) mMoveIncidence; 2216 Todo * to = (Todo *) mMoveIncidence;
2217 QTime tim; 2217 QTime tim;
2218 if ( to->hasDueDate() ) 2218 if ( to->hasDueDate() )
2219 tim = to->dtDue().time(); 2219 tim = to->dtDue().time();
2220 else { 2220 else {
2221 tim = QTime ( 0,0,0 ); 2221 tim = QTime ( 0,0,0 );
2222 to->setFloats( true ); 2222 to->setFloats( true );
2223 to->setHasDueDate( true ); 2223 to->setHasDueDate( true );
2224 } 2224 }
2225 QDateTime dt ( d,tim ); 2225 QDateTime dt ( d,tim );
2226 to->setDtDue( dt ); 2226 to->setDtDue( dt );
2227 todoChanged( to ); 2227 todoChanged( to );
2228 } else { 2228 } else {
2229 QTime tim = mMoveIncidence->dtStart().time(); 2229 QTime tim = mMoveIncidence->dtStart().time();
2230 int secs = mMoveIncidence->dtStart().secsTo( mMoveIncidence->dtEnd()); 2230 int secs = mMoveIncidence->dtStart().secsTo( mMoveIncidence->dtEnd());
2231 QDateTime dt ( d,tim ); 2231 QDateTime dt ( d,tim );
2232 mMoveIncidence->setDtStart( dt ); 2232 mMoveIncidence->setDtStart( dt );
2233 ((Event*)mMoveIncidence)->setDtEnd( dt.addSecs( secs ) ); 2233 ((Event*)mMoveIncidence)->setDtEnd( dt.addSecs( secs ) );
2234 changeEventDisplay((Event*)mMoveIncidence, KOGlobals::EVENTEDITED); 2234 changeEventDisplay((Event*)mMoveIncidence, KOGlobals::EVENTEDITED);
2235 } 2235 }
2236 2236
2237 mMoveIncidence->setRevision( mMoveIncidence->revision()+1 ); 2237 mMoveIncidence->setRevision( mMoveIncidence->revision()+1 );
2238 } 2238 }
2239} 2239}
2240 2240
2241void CalendarView::removeCategories() 2241void CalendarView::removeCategories()
2242{ 2242{
2243 QPtrList<Incidence> incList = mCalendar->rawIncidences(); 2243 QPtrList<Incidence> incList = mCalendar->rawIncidences();
2244 QStringList catList = KOPrefs::instance()->mCustomCategories; 2244 QStringList catList = KOPrefs::instance()->mCustomCategories;
2245 QStringList catIncList; 2245 QStringList catIncList;
2246 QStringList newCatList; 2246 QStringList newCatList;
2247 Incidence* inc = incList.first(); 2247 Incidence* inc = incList.first();
2248 int i; 2248 int i;
2249 int count = 0; 2249 int count = 0;
2250 while ( inc ) { 2250 while ( inc ) {
2251 newCatList.clear(); 2251 newCatList.clear();
2252 catIncList = inc->categories() ; 2252 catIncList = inc->categories() ;
2253 for( i = 0; i< catIncList.count(); ++i ) { 2253 for( i = 0; i< catIncList.count(); ++i ) {
2254 if ( catList.contains (catIncList[i])) 2254 if ( catList.contains (catIncList[i]))
2255 newCatList.append( catIncList[i] ); 2255 newCatList.append( catIncList[i] );
2256 } 2256 }
2257 newCatList.sort(); 2257 newCatList.sort();
2258 inc->setCategories( newCatList.join(",") ); 2258 inc->setCategories( newCatList.join(",") );
2259 inc = incList.next(); 2259 inc = incList.next();
2260 } 2260 }
2261} 2261}
2262 2262
2263int CalendarView::addCategories() 2263int CalendarView::addCategories()
2264{ 2264{
2265 QPtrList<Incidence> incList = mCalendar->rawIncidences(); 2265 QPtrList<Incidence> incList = mCalendar->rawIncidences();
2266 QStringList catList = KOPrefs::instance()->mCustomCategories; 2266 QStringList catList = KOPrefs::instance()->mCustomCategories;
2267 QStringList catIncList; 2267 QStringList catIncList;
2268 Incidence* inc = incList.first(); 2268 Incidence* inc = incList.first();
2269 int i; 2269 int i;
2270 int count = 0; 2270 int count = 0;
2271 while ( inc ) { 2271 while ( inc ) {
2272 catIncList = inc->categories() ; 2272 catIncList = inc->categories() ;
2273 for( i = 0; i< catIncList.count(); ++i ) { 2273 for( i = 0; i< catIncList.count(); ++i ) {
2274 if ( !catList.contains (catIncList[i])) { 2274 if ( !catList.contains (catIncList[i])) {
2275 catList.append( catIncList[i] ); 2275 catList.append( catIncList[i] );
2276 //qDebug("add cat %s ", catIncList[i].latin1()); 2276 //qDebug("add cat %s ", catIncList[i].latin1());
2277 ++count; 2277 ++count;
2278 } 2278 }
2279 } 2279 }
2280 inc = incList.next(); 2280 inc = incList.next();
2281 } 2281 }
2282 catList.sort(); 2282 catList.sort();
2283 KOPrefs::instance()->mCustomCategories = catList; 2283 KOPrefs::instance()->mCustomCategories = catList;
2284 return count; 2284 return count;
2285} 2285}
2286 2286
2287void CalendarView::manageCategories() 2287void CalendarView::manageCategories()
2288{ 2288{
2289 KOCatPrefs* cp = new KOCatPrefs(); 2289 KOCatPrefs* cp = new KOCatPrefs();
2290 cp->show(); 2290 cp->show();
2291 int w =cp->sizeHint().width() ; 2291 int w =cp->sizeHint().width() ;
2292 int h = cp->sizeHint().height() ; 2292 int h = cp->sizeHint().height() ;
2293 int dw = QApplication::desktop()->width(); 2293 int dw = QApplication::desktop()->width();
2294 int dh = QApplication::desktop()->height(); 2294 int dh = QApplication::desktop()->height();
2295 cp->setGeometry( (dw-w)/2, (dh - h )/2 ,w,h ); 2295 cp->setGeometry( (dw-w)/2, (dh - h )/2 ,w,h );
2296 if ( !cp->exec() ) { 2296 if ( !cp->exec() ) {
2297 delete cp; 2297 delete cp;
2298 return; 2298 return;
2299 } 2299 }
2300 int count = 0; 2300 int count = 0;
2301 if ( cp->addCat() ) { 2301 if ( cp->addCat() ) {
2302 count = addCategories(); 2302 count = addCategories();
2303 if ( count ) { 2303 if ( count ) {
2304 topLevelWidget()->setCaption(QString::number( count )+ i18n(" Categories added to list! ")); 2304 topLevelWidget()->setCaption(QString::number( count )+ i18n(" Categories added to list! "));
2305 writeSettings(); 2305 writeSettings();
2306 } else 2306 } else
2307 topLevelWidget()->setCaption(QString::number( 0 )+ i18n(" Categories added to list! ")); 2307 topLevelWidget()->setCaption(QString::number( 0 )+ i18n(" Categories added to list! "));
2308 } else { 2308 } else {
2309 removeCategories(); 2309 removeCategories();
2310 updateView(); 2310 updateView();
2311 } 2311 }
2312 delete cp; 2312 delete cp;
2313} 2313}
2314 2314
2315void CalendarView::beamIncidence(Incidence * Inc) 2315void CalendarView::beamIncidence(Incidence * Inc)
2316{ 2316{
2317 QPtrList<Incidence> delSel ; 2317 QPtrList<Incidence> delSel ;
2318 delSel.append(Inc); 2318 delSel.append(Inc);
2319 beamIncidenceList( delSel ); 2319 beamIncidenceList( delSel );
2320} 2320}
2321void CalendarView::beamCalendar() 2321void CalendarView::beamCalendar()
2322{ 2322{
2323 QPtrList<Incidence> delSel = mCalendar->rawIncidences(); 2323 QPtrList<Incidence> delSel = mCalendar->rawIncidences();
2324 //qDebug("beamCalendar() "); 2324 //qDebug("beamCalendar() ");
2325 beamIncidenceList( delSel ); 2325 beamIncidenceList( delSel );
2326} 2326}
2327void CalendarView::beamFilteredCalendar() 2327void CalendarView::beamFilteredCalendar()
2328{ 2328{
2329 QPtrList<Incidence> delSel = mCalendar->incidences(); 2329 QPtrList<Incidence> delSel = mCalendar->incidences();
2330 //qDebug("beamFilteredCalendar() "); 2330 //qDebug("beamFilteredCalendar() ");
2331 beamIncidenceList( delSel ); 2331 beamIncidenceList( delSel );
2332} 2332}
2333void CalendarView::beamIncidenceList(QPtrList<Incidence> delSel ) 2333void CalendarView::beamIncidenceList(QPtrList<Incidence> delSel )
2334{ 2334{
2335 if ( beamDialog->exec () == QDialog::Rejected ) 2335 if ( beamDialog->exec () == QDialog::Rejected )
2336 return; 2336 return;
2337#ifdef DESKTOP_VERSION 2337#ifdef DESKTOP_VERSION
2338 QString fn = locateLocal( "tmp", "kopibeamfile" ); 2338 QString fn = locateLocal( "tmp", "kopibeamfile" );
2339#else 2339#else
2340 QString fn = "/tmp/kopibeamfile"; 2340 QString fn = "/tmp/kopibeamfile";
2341#endif 2341#endif
2342 QString mes; 2342 QString mes;
2343 bool createbup = true; 2343 bool createbup = true;
2344 if ( createbup ) { 2344 if ( createbup ) {
2345 QString description = "\n"; 2345 QString description = "\n";
2346 CalendarLocal* cal = new CalendarLocal(); 2346 CalendarLocal* cal = new CalendarLocal();
2347 if ( beamDialog->beamLocal() ) 2347 if ( beamDialog->beamLocal() )
2348 cal->setLocalTime(); 2348 cal->setLocalTime();
2349 else 2349 else
2350 cal->setTimeZoneId(KPimGlobalPrefs::instance()->mTimeZoneId); 2350 cal->setTimeZoneId(KPimGlobalPrefs::instance()->mTimeZoneId);
2351 Incidence *incidence = delSel.first(); 2351 Incidence *incidence = delSel.first();
2352 bool addText = false; 2352 bool addText = false;
2353 if ( delSel.count() < 10 ) 2353 if ( delSel.count() < 10 )
2354 addText = true; 2354 addText = true;
2355 else { 2355 else {
2356 description.sprintf(i18n(" %d items?"),delSel.count() ); 2356 description.sprintf(i18n(" %d items?"),delSel.count() );
2357 } 2357 }
2358 while ( incidence ) { 2358 while ( incidence ) {
2359 Incidence *in = incidence->clone(); 2359 Incidence *in = incidence->clone();
2360 if ( ! in->summary().isEmpty() ) { 2360 if ( ! in->summary().isEmpty() ) {
2361 in->setDescription(""); 2361 in->setDescription("");
2362 } else { 2362 } else {
2363 in->setSummary( in->description().left(20)); 2363 in->setSummary( in->description().left(20));
2364 in->setDescription(""); 2364 in->setDescription("");
2365 } 2365 }
2366 if ( addText ) 2366 if ( addText )
2367 description += in->summary() + "\n"; 2367 description += in->summary() + "\n";
2368 cal->addIncidence( in ); 2368 cal->addIncidence( in );
2369 incidence = delSel.next(); 2369 incidence = delSel.next();
2370 } 2370 }
2371 if ( beamDialog->beamVcal() ) { 2371 if ( beamDialog->beamVcal() ) {
2372 fn += ".vcs"; 2372 fn += ".vcs";
2373 FileStorage storage( cal, fn, new VCalFormat ); 2373 FileStorage storage( cal, fn, new VCalFormat );
2374 storage.save(); 2374 storage.save();
2375 } else { 2375 } else {
2376 fn += ".ics"; 2376 fn += ".ics";
2377 FileStorage storage( cal, fn, new ICalFormat( ) ); 2377 FileStorage storage( cal, fn, new ICalFormat( ) );
2378 storage.save(); 2378 storage.save();
2379 } 2379 }
2380 delete cal; 2380 delete cal;
2381 mes = i18n("KO/Pi: Ready for beaming"); 2381 mes = i18n("KO/Pi: Ready for beaming");
2382 topLevelWidget()->setCaption(mes); 2382 topLevelWidget()->setCaption(mes);
2383 KApplication::convert2latin1( fn ); 2383 KApplication::convert2latin1( fn );
2384#ifndef DESKTOP_VERSION 2384#ifndef DESKTOP_VERSION
2385 Ir *ir = new Ir( this ); 2385 Ir *ir = new Ir( this );
2386 connect( ir, SIGNAL( done( Ir * ) ), this, SLOT( beamDone( Ir * ) ) ); 2386 connect( ir, SIGNAL( done( Ir * ) ), this, SLOT( beamDone( Ir * ) ) );
2387 ir->send( fn, description, "text/x-vCalendar" ); 2387 ir->send( fn, description, "text/x-vCalendar" );
2388#endif 2388#endif
2389 } 2389 }
2390} 2390}
2391void CalendarView::beamDone( Ir *ir ) 2391void CalendarView::beamDone( Ir *ir )
2392{ 2392{
2393#ifndef DESKTOP_VERSION 2393#ifndef DESKTOP_VERSION
2394 delete ir; 2394 delete ir;
2395#endif 2395#endif
2396 topLevelWidget()->setCaption( i18n("KO/Pi: Beaming done.") ); 2396 topLevelWidget()->setCaption( i18n("KO/Pi: Beaming done.") );
2397 topLevelWidget()->raise(); 2397 topLevelWidget()->raise();
2398} 2398}
2399 2399
2400void CalendarView::moveIncidence(Incidence * inc ) 2400void CalendarView::moveIncidence(Incidence * inc )
2401{ 2401{
2402 if ( !inc ) return; 2402 if ( !inc ) return;
2403 // qDebug("showDatePickerForIncidence( ) "); 2403 // qDebug("showDatePickerForIncidence( ) ");
2404 if ( mDateFrame->isVisible() ) 2404 if ( mDateFrame->isVisible() )
2405 mDateFrame->hide(); 2405 mDateFrame->hide();
2406 else { 2406 else {
2407 int w =mDatePicker->sizeHint().width()+2*mDateFrame->lineWidth() ; 2407 int w =mDatePicker->sizeHint().width()+2*mDateFrame->lineWidth() ;
2408 int h = mDatePicker->sizeHint().height()+2*mDateFrame->lineWidth() ; 2408 int h = mDatePicker->sizeHint().height()+2*mDateFrame->lineWidth() ;
2409 int dw = QApplication::desktop()->width(); 2409 int dw = QApplication::desktop()->width();
2410 int dh = QApplication::desktop()->height(); 2410 int dh = QApplication::desktop()->height();
2411 mDateFrame->setGeometry( (dw-w)/2, (dh - h )/2 ,w,h ); 2411 mDateFrame->setGeometry( (dw-w)/2, (dh - h )/2 ,w,h );
2412 mDateFrame->show(); 2412 mDateFrame->show();
2413 } 2413 }
2414 mDatePickerMode = 2; 2414 mDatePickerMode = 2;
2415 mMoveIncidence = inc ; 2415 mMoveIncidence = inc ;
2416 QDate da; 2416 QDate da;
2417 if ( mMoveIncidence->type() == "Todo" ) { 2417 if ( mMoveIncidence->type() == "Todo" ) {
2418 Todo * to = (Todo *) mMoveIncidence; 2418 Todo * to = (Todo *) mMoveIncidence;
2419 if ( to->hasDueDate() ) 2419 if ( to->hasDueDate() )
2420 da = to->dtDue().date(); 2420 da = to->dtDue().date();
2421 else 2421 else
2422 da = QDate::currentDate(); 2422 da = QDate::currentDate();
2423 } else { 2423 } else {
2424 da = mMoveIncidence->dtStart().date(); 2424 da = mMoveIncidence->dtStart().date();
2425 } 2425 }
2426 mDatePicker->setDate( da ); 2426 mDatePicker->setDate( da );
2427} 2427}
2428void CalendarView::showDatePicker( ) 2428void CalendarView::showDatePicker( )
2429{ 2429{
2430 //qDebug("CalendarView::showDatePicker( ) "); 2430 //qDebug("CalendarView::showDatePicker( ) ");
2431 if ( mDateFrame->isVisible() ) 2431 if ( mDateFrame->isVisible() )
2432 mDateFrame->hide(); 2432 mDateFrame->hide();
2433 else { 2433 else {
2434 int w =mDatePicker->sizeHint().width() ; 2434 int w =mDatePicker->sizeHint().width() ;
2435 int h = mDatePicker->sizeHint().height() ; 2435 int h = mDatePicker->sizeHint().height() ;
2436 int dw = QApplication::desktop()->width(); 2436 int dw = QApplication::desktop()->width();
2437 int dh = QApplication::desktop()->height(); 2437 int dh = QApplication::desktop()->height();
2438 mDateFrame->setGeometry( (dw-w)/2, (dh - h )/2 ,w,h ); 2438 mDateFrame->setGeometry( (dw-w)/2, (dh - h )/2 ,w,h );
2439 mDateFrame->show(); 2439 mDateFrame->show();
2440 } 2440 }
2441 mDatePickerMode = 1; 2441 mDatePickerMode = 1;
2442 mDatePicker->setDate( mNavigator->selectedDates().first() ); 2442 mDatePicker->setDate( mNavigator->selectedDates().first() );
2443} 2443}
2444 2444
2445void CalendarView::showEventEditor() 2445void CalendarView::showEventEditor()
2446{ 2446{
2447#ifdef DESKTOP_VERSION 2447#ifdef DESKTOP_VERSION
2448 mEventEditor->show(); 2448 mEventEditor->show();
2449#else 2449#else
2450 if ( mEventEditor->width() != QApplication::desktop()->width() )
2451 mEventEditor->hide();
2450 mEventEditor->showMaximized(); 2452 mEventEditor->showMaximized();
2451#endif 2453#endif
2452} 2454}
2453void CalendarView::showTodoEditor() 2455void CalendarView::showTodoEditor()
2454{ 2456{
2455#ifdef DESKTOP_VERSION 2457#ifdef DESKTOP_VERSION
2456 mTodoEditor->show(); 2458 mTodoEditor->show();
2457#else 2459#else
2460 if ( mTodoEditor->width() != QApplication::desktop()->width() )
2461 mTodoEditor->hide();
2458 mTodoEditor->showMaximized(); 2462 mTodoEditor->showMaximized();
2459#endif 2463#endif
2460} 2464}
2461 2465
2462void CalendarView::cloneIncidence() 2466void CalendarView::cloneIncidence()
2463{ 2467{
2464 Incidence *incidence = currentSelection(); 2468 Incidence *incidence = currentSelection();
2465 if ( !incidence ) incidence = mTodoList->selectedIncidences().first(); 2469 if ( !incidence ) incidence = mTodoList->selectedIncidences().first();
2466 if ( incidence ) { 2470 if ( incidence ) {
2467 cloneIncidence(incidence); 2471 cloneIncidence(incidence);
2468 } 2472 }
2469} 2473}
2470void CalendarView::moveIncidence() 2474void CalendarView::moveIncidence()
2471{ 2475{
2472 Incidence *incidence = currentSelection(); 2476 Incidence *incidence = currentSelection();
2473 if ( !incidence ) incidence = mTodoList->selectedIncidences().first(); 2477 if ( !incidence ) incidence = mTodoList->selectedIncidences().first();
2474 if ( incidence ) { 2478 if ( incidence ) {
2475 moveIncidence(incidence); 2479 moveIncidence(incidence);
2476 } 2480 }
2477} 2481}
2478void CalendarView::beamIncidence() 2482void CalendarView::beamIncidence()
2479{ 2483{
2480 Incidence *incidence = currentSelection(); 2484 Incidence *incidence = currentSelection();
2481 if ( !incidence ) incidence = mTodoList->selectedIncidences().first(); 2485 if ( !incidence ) incidence = mTodoList->selectedIncidences().first();
2482 if ( incidence ) { 2486 if ( incidence ) {
2483 beamIncidence(incidence); 2487 beamIncidence(incidence);
2484 } 2488 }
2485} 2489}
2486void CalendarView::toggleCancelIncidence() 2490void CalendarView::toggleCancelIncidence()
2487{ 2491{
2488 Incidence *incidence = currentSelection(); 2492 Incidence *incidence = currentSelection();
2489 if ( !incidence ) incidence = mTodoList->selectedIncidences().first(); 2493 if ( !incidence ) incidence = mTodoList->selectedIncidences().first();
2490 if ( incidence ) { 2494 if ( incidence ) {
2491 cancelIncidence(incidence); 2495 cancelIncidence(incidence);
2492 } 2496 }
2493} 2497}
2494 2498
2495 2499
2496void CalendarView::cancelIncidence(Incidence * inc ) 2500void CalendarView::cancelIncidence(Incidence * inc )
2497{ 2501{
2498 inc->setCancelled( ! inc->cancelled() ); 2502 inc->setCancelled( ! inc->cancelled() );
2499 changeIncidenceDisplay( inc,KOGlobals::EVENTEDITED ); 2503 changeIncidenceDisplay( inc,KOGlobals::EVENTEDITED );
2500 updateView(); 2504 updateView();
2501} 2505}
2502void CalendarView::cloneIncidence(Incidence * orgInc ) 2506void CalendarView::cloneIncidence(Incidence * orgInc )
2503{ 2507{
2504 Incidence * newInc = orgInc->clone(); 2508 Incidence * newInc = orgInc->clone();
2505 newInc->recreate(); 2509 newInc->recreate();
2506 2510
2507 if ( newInc->type() == "Todo" ) { 2511 if ( newInc->type() == "Todo" ) {
2508 Todo* t = (Todo*) newInc; 2512 Todo* t = (Todo*) newInc;
2509 mTodoEditor->editTodo( t ); 2513 mTodoEditor->editTodo( t );
2510 showTodoEditor(); 2514 showTodoEditor();
2511 if ( mTodoEditor->exec() ) { 2515 if ( mTodoEditor->exec() ) {
2512 mCalendar->addTodo( t ); 2516 mCalendar->addTodo( t );
2513 updateView(); 2517 updateView();
2514 } else { 2518 } else {
2515 delete t; 2519 delete t;
2516 } 2520 }
2517 } 2521 }
2518 else { 2522 else {
2519 Event* e = (Event*) newInc; 2523 Event* e = (Event*) newInc;
2520 mEventEditor->editEvent( e ); 2524 mEventEditor->editEvent( e );
2521 showEventEditor(); 2525 showEventEditor();
2522 if ( mEventEditor->exec() ) { 2526 if ( mEventEditor->exec() ) {
2523 mCalendar->addEvent( e ); 2527 mCalendar->addEvent( e );
2524 updateView(); 2528 updateView();
2525 } else { 2529 } else {
2526 delete e; 2530 delete e;
2527 } 2531 }
2528 } 2532 }
2529} 2533}
2530 2534
2531void CalendarView::newEvent() 2535void CalendarView::newEvent()
2532{ 2536{
2533 // TODO: Replace this code by a common eventDurationHint of KOBaseView. 2537 // TODO: Replace this code by a common eventDurationHint of KOBaseView.
2534 KOAgendaView *aView = mViewManager->agendaView(); 2538 KOAgendaView *aView = mViewManager->agendaView();
2535 if (aView) { 2539 if (aView) {
2536 if (aView->selectionStart().isValid()) { 2540 if (aView->selectionStart().isValid()) {
2537 if (aView->selectedIsAllDay()) { 2541 if (aView->selectedIsAllDay()) {
2538 newEvent(aView->selectionStart(),aView->selectionEnd(),true); 2542 newEvent(aView->selectionStart(),aView->selectionEnd(),true);
2539 } else { 2543 } else {
2540 newEvent(aView->selectionStart(),aView->selectionEnd()); 2544 newEvent(aView->selectionStart(),aView->selectionEnd());
2541 } 2545 }
2542 return; 2546 return;
2543 } 2547 }
2544 } 2548 }
2545 2549
2546 QDate date = mNavigator->selectedDates().first(); 2550 QDate date = mNavigator->selectedDates().first();
2547 QDateTime current = QDateTime::currentDateTime(); 2551 QDateTime current = QDateTime::currentDateTime();
2548 if ( date <= current.date() ) { 2552 if ( date <= current.date() ) {
2549 int hour = current.time().hour() +1; 2553 int hour = current.time().hour() +1;
2550 newEvent( QDateTime( current.date(), QTime( hour, 0, 0 ) ), 2554 newEvent( QDateTime( current.date(), QTime( hour, 0, 0 ) ),
2551 QDateTime( current.date(), QTime( hour+ KOPrefs::instance()->mDefaultDuration, 0, 0 ) ) ); 2555 QDateTime( current.date(), QTime( hour+ KOPrefs::instance()->mDefaultDuration, 0, 0 ) ) );
2552 } else 2556 } else
2553 newEvent( QDateTime( date, QTime( KOPrefs::instance()->mStartTime, 0, 0 ) ), 2557 newEvent( QDateTime( date, QTime( KOPrefs::instance()->mStartTime, 0, 0 ) ),
2554 QDateTime( date, QTime( KOPrefs::instance()->mStartTime + 2558 QDateTime( date, QTime( KOPrefs::instance()->mStartTime +
2555 KOPrefs::instance()->mDefaultDuration, 0, 0 ) ) ); 2559 KOPrefs::instance()->mDefaultDuration, 0, 0 ) ) );
2556} 2560}
2557 2561
2558void CalendarView::newEvent(QDateTime fh) 2562void CalendarView::newEvent(QDateTime fh)
2559{ 2563{
2560 newEvent(fh, 2564 newEvent(fh,
2561 QDateTime(fh.addSecs(3600*KOPrefs::instance()->mDefaultDuration))); 2565 QDateTime(fh.addSecs(3600*KOPrefs::instance()->mDefaultDuration)));
2562} 2566}
2563 2567
2564void CalendarView::newEvent(QDate dt) 2568void CalendarView::newEvent(QDate dt)
2565{ 2569{
2566 newEvent(QDateTime(dt, QTime(0,0,0)), 2570 newEvent(QDateTime(dt, QTime(0,0,0)),
2567 QDateTime(dt, QTime(0,0,0)), true); 2571 QDateTime(dt, QTime(0,0,0)), true);
2568} 2572}
2569 2573
2570void CalendarView::newEvent(QDateTime fromHint, QDateTime toHint, bool allDay) 2574void CalendarView::newEvent(QDateTime fromHint, QDateTime toHint, bool allDay)
2571{ 2575{
2572 2576
2573 mEventEditor->newEvent(fromHint,toHint,allDay); 2577 mEventEditor->newEvent(fromHint,toHint,allDay);
2574 if ( mFilterView->filtersEnabled() ) { 2578 if ( mFilterView->filtersEnabled() ) {
2575 CalFilter *filter = mFilterView->selectedFilter(); 2579 CalFilter *filter = mFilterView->selectedFilter();
2576 if (filter && filter->showCategories()) { 2580 if (filter && filter->showCategories()) {
2577 mEventEditor->setCategories(filter->categoryList().join(",") ); 2581 mEventEditor->setCategories(filter->categoryList().join(",") );
2578 } 2582 }
2579 if ( filter ) 2583 if ( filter )
2580 mEventEditor->setSecrecy( filter->getSecrecy() ); 2584 mEventEditor->setSecrecy( filter->getSecrecy() );
2581 } 2585 }
2582 showEventEditor(); 2586 showEventEditor();
2583} 2587}
2584void CalendarView::todoAdded(Todo * t) 2588void CalendarView::todoAdded(Todo * t)
2585{ 2589{
2586 2590
2587 changeTodoDisplay ( t ,KOGlobals::EVENTADDED); 2591 changeTodoDisplay ( t ,KOGlobals::EVENTADDED);
2588 updateTodoViews(); 2592 updateTodoViews();
2589} 2593}
2590void CalendarView::todoChanged(Todo * t) 2594void CalendarView::todoChanged(Todo * t)
2591{ 2595{
2592 emit todoModified( t, 4 ); 2596 emit todoModified( t, 4 );
2593 // updateTodoViews(); 2597 // updateTodoViews();
2594} 2598}
2595void CalendarView::todoToBeDeleted(Todo *) 2599void CalendarView::todoToBeDeleted(Todo *)
2596{ 2600{
2597 //qDebug("todoToBeDeleted(Todo *) "); 2601 //qDebug("todoToBeDeleted(Todo *) ");
2598 updateTodoViews(); 2602 updateTodoViews();
2599} 2603}
2600void CalendarView::todoDeleted() 2604void CalendarView::todoDeleted()
2601{ 2605{
2602 //qDebug(" todoDeleted()"); 2606 //qDebug(" todoDeleted()");
2603 updateTodoViews(); 2607 updateTodoViews();
2604} 2608}
2605 2609
2606 2610
2607 2611
2608void CalendarView::newTodo() 2612void CalendarView::newTodo()
2609{ 2613{
2610 2614
2611 mTodoEditor->newTodo(QDateTime::currentDateTime().addDays(7),0,true); 2615 mTodoEditor->newTodo(QDateTime::currentDateTime().addDays(7),0,true);
2612 if ( mFilterView->filtersEnabled() ) { 2616 if ( mFilterView->filtersEnabled() ) {
2613 CalFilter *filter = mFilterView->selectedFilter(); 2617 CalFilter *filter = mFilterView->selectedFilter();
2614 if (filter && filter->showCategories()) { 2618 if (filter && filter->showCategories()) {
2615 mTodoEditor->setCategories(filter->categoryList().join(",") ); 2619 mTodoEditor->setCategories(filter->categoryList().join(",") );
2616 } 2620 }
2617 if ( filter ) 2621 if ( filter )
2618 mTodoEditor->setSecrecy( filter->getSecrecy() ); 2622 mTodoEditor->setSecrecy( filter->getSecrecy() );
2619 } 2623 }
2620 showTodoEditor(); 2624 showTodoEditor();
2621} 2625}
2622 2626
2623void CalendarView::newSubTodo() 2627void CalendarView::newSubTodo()
2624{ 2628{
2625 Todo *todo = selectedTodo(); 2629 Todo *todo = selectedTodo();
2626 if ( todo ) newSubTodo( todo ); 2630 if ( todo ) newSubTodo( todo );
2627} 2631}
2628 2632
2629void CalendarView::newSubTodo(Todo *parentEvent) 2633void CalendarView::newSubTodo(Todo *parentEvent)
2630{ 2634{
2631 2635
2632 mTodoEditor->newTodo(QDateTime::currentDateTime().addDays(7),parentEvent,true); 2636 mTodoEditor->newTodo(QDateTime::currentDateTime().addDays(7),parentEvent,true);
2633 showTodoEditor(); 2637 showTodoEditor();
2634} 2638}
2635 2639
2636void CalendarView::newFloatingEvent() 2640void CalendarView::newFloatingEvent()
2637{ 2641{
2638 DateList tmpList = mNavigator->selectedDates(); 2642 DateList tmpList = mNavigator->selectedDates();
2639 QDate date = tmpList.first(); 2643 QDate date = tmpList.first();
2640 2644
2641 newEvent( QDateTime( date, QTime( 12, 0, 0 ) ), 2645 newEvent( QDateTime( date, QTime( 12, 0, 0 ) ),
2642 QDateTime( date, QTime( 12, 0, 0 ) ), true ); 2646 QDateTime( date, QTime( 12, 0, 0 ) ), true );
2643} 2647}
2644 2648
2645 2649
2646void CalendarView::editEvent( Event *event ) 2650void CalendarView::editEvent( Event *event )
2647{ 2651{
2648 2652
2649 if ( !event ) return; 2653 if ( !event ) return;
2650 if ( event->isReadOnly() ) { 2654 if ( event->isReadOnly() ) {
2651 showEvent( event ); 2655 showEvent( event );
2652 return; 2656 return;
2653 } 2657 }
2654 mEventEditor->editEvent( event , mFlagEditDescription); 2658 mEventEditor->editEvent( event , mFlagEditDescription);
2655 showEventEditor(); 2659 showEventEditor();
2656} 2660}
2657void CalendarView::editJournal( Journal *jour ) 2661void CalendarView::editJournal( Journal *jour )
2658{ 2662{
2659 if ( !jour ) return; 2663 if ( !jour ) return;
2660 mDialogManager->hideSearchDialog(); 2664 mDialogManager->hideSearchDialog();
2661 mViewManager->showJournalView(); 2665 mViewManager->showJournalView();
2662 mNavigator->slotDaySelect( jour->dtStart().date() ); 2666 mNavigator->slotDaySelect( jour->dtStart().date() );
2663} 2667}
2664void CalendarView::editTodo( Todo *todo ) 2668void CalendarView::editTodo( Todo *todo )
2665{ 2669{
2666 if ( !todo ) return; 2670 if ( !todo ) return;
2667 2671
2668 if ( todo->isReadOnly() ) { 2672 if ( todo->isReadOnly() ) {
2669 showTodo( todo ); 2673 showTodo( todo );
2670 return; 2674 return;
2671 } 2675 }
2672 mTodoEditor->editTodo( todo ,mFlagEditDescription); 2676 mTodoEditor->editTodo( todo ,mFlagEditDescription);
2673 showTodoEditor(); 2677 showTodoEditor();
2674 2678
2675} 2679}
2676 2680
2677KOEventViewerDialog* CalendarView::getEventViewerDialog() 2681KOEventViewerDialog* CalendarView::getEventViewerDialog()
2678{ 2682{
2679 if ( !mEventViewerDialog ) { 2683 if ( !mEventViewerDialog ) {
2680 mEventViewerDialog = new KOEventViewerDialog(this); 2684 mEventViewerDialog = new KOEventViewerDialog(this);
2681 connect( mEventViewerDialog, SIGNAL( editIncidence( Incidence* )), this, SLOT(editIncidence( Incidence* ) ) ); 2685 connect( mEventViewerDialog, SIGNAL( editIncidence( Incidence* )), this, SLOT(editIncidence( Incidence* ) ) );
2682 connect( this, SIGNAL(configChanged()), mEventViewerDialog, SLOT(updateConfig())); 2686 connect( this, SIGNAL(configChanged()), mEventViewerDialog, SLOT(updateConfig()));
2683 connect( mEventViewerDialog, SIGNAL(jumpToTime( const QDate &)), 2687 connect( mEventViewerDialog, SIGNAL(jumpToTime( const QDate &)),
2684 dateNavigator(), SLOT( selectWeek( const QDate & ) ) ); 2688 dateNavigator(), SLOT( selectWeek( const QDate & ) ) );
2685 connect( mEventViewerDialog, SIGNAL(showAgendaView( bool ) ), 2689 connect( mEventViewerDialog, SIGNAL(showAgendaView( bool ) ),
2686 viewManager(), SLOT( showAgendaView( bool ) ) ); 2690 viewManager(), SLOT( showAgendaView( bool ) ) );
2687 mEventViewerDialog->resize( 640, 480 ); 2691 mEventViewerDialog->resize( 640, 480 );
2688 2692
2689 } 2693 }
2690 return mEventViewerDialog; 2694 return mEventViewerDialog;
2691} 2695}
2692void CalendarView::showEvent(Event *event) 2696void CalendarView::showEvent(Event *event)
2693{ 2697{
2694 getEventViewerDialog()->setEvent(event); 2698 getEventViewerDialog()->setEvent(event);
2695 getEventViewerDialog()->showMe(); 2699 getEventViewerDialog()->showMe();
2696} 2700}
2697 2701
2698void CalendarView::showTodo(Todo *event) 2702void CalendarView::showTodo(Todo *event)
2699{ 2703{
2700 getEventViewerDialog()->setTodo(event); 2704 getEventViewerDialog()->setTodo(event);
2701 getEventViewerDialog()->showMe(); 2705 getEventViewerDialog()->showMe();
2702} 2706}
2703void CalendarView::showJournal( Journal *jour ) 2707void CalendarView::showJournal( Journal *jour )
2704{ 2708{
2705 getEventViewerDialog()->setJournal(jour); 2709 getEventViewerDialog()->setJournal(jour);
2706 getEventViewerDialog()->showMe(); 2710 getEventViewerDialog()->showMe();
2707 2711
2708} 2712}
2709// void CalendarView::todoModified (Todo *event, int changed) 2713// void CalendarView::todoModified (Todo *event, int changed)
2710// { 2714// {
2711// // if (mDialogList.find (event) != mDialogList.end ()) { 2715// // if (mDialogList.find (event) != mDialogList.end ()) {
2712// // kdDebug() << "Todo modified and open" << endl; 2716// // kdDebug() << "Todo modified and open" << endl;
2713// // KOTodoEditor* temp = (KOTodoEditor *) mDialogList[event]; 2717// // KOTodoEditor* temp = (KOTodoEditor *) mDialogList[event];
2714// // temp->modified (changed); 2718// // temp->modified (changed);
2715 2719
2716// // } 2720// // }
2717 2721
2718// mViewManager->updateView(); 2722// mViewManager->updateView();
2719// } 2723// }
2720 2724
2721void CalendarView::appointment_show() 2725void CalendarView::appointment_show()
2722{ 2726{
2723 Event *anEvent = 0; 2727 Event *anEvent = 0;
2724 2728
2725 Incidence *incidence = mViewManager->currentView()->selectedIncidences().first(); 2729 Incidence *incidence = mViewManager->currentView()->selectedIncidences().first();
2726 2730
2727 if (mViewManager->currentView()->isEventView()) { 2731 if (mViewManager->currentView()->isEventView()) {
2728 if ( incidence && incidence->type() == "Event" ) { 2732 if ( incidence && incidence->type() == "Event" ) {
2729 anEvent = static_cast<Event *>(incidence); 2733 anEvent = static_cast<Event *>(incidence);
2730 } 2734 }
2731 } 2735 }
2732 2736
2733 if (!anEvent) { 2737 if (!anEvent) {
2734 KNotifyClient::beep(); 2738 KNotifyClient::beep();
2735 return; 2739 return;
2736 } 2740 }
2737 2741
2738 showEvent(anEvent); 2742 showEvent(anEvent);
2739} 2743}
2740 2744
2741void CalendarView::appointment_edit() 2745void CalendarView::appointment_edit()
2742{ 2746{
2743 Event *anEvent = 0; 2747 Event *anEvent = 0;
2744 2748
2745 Incidence *incidence = mViewManager->currentView()->selectedIncidences().first(); 2749 Incidence *incidence = mViewManager->currentView()->selectedIncidences().first();
2746 2750
2747 if (mViewManager->currentView()->isEventView()) { 2751 if (mViewManager->currentView()->isEventView()) {
2748 if ( incidence && incidence->type() == "Event" ) { 2752 if ( incidence && incidence->type() == "Event" ) {
2749 anEvent = static_cast<Event *>(incidence); 2753 anEvent = static_cast<Event *>(incidence);
2750 } 2754 }
2751 } 2755 }
2752 2756
2753 if (!anEvent) { 2757 if (!anEvent) {
2754 KNotifyClient::beep(); 2758 KNotifyClient::beep();
2755 return; 2759 return;
2756 } 2760 }
2757 2761
2758 editEvent(anEvent); 2762 editEvent(anEvent);
2759} 2763}
2760 2764
2761void CalendarView::appointment_delete() 2765void CalendarView::appointment_delete()
2762{ 2766{
2763 Event *anEvent = 0; 2767 Event *anEvent = 0;
2764 2768
2765 Incidence *incidence = mViewManager->currentView()->selectedIncidences().first(); 2769 Incidence *incidence = mViewManager->currentView()->selectedIncidences().first();
2766 2770
2767 if (mViewManager->currentView()->isEventView()) { 2771 if (mViewManager->currentView()->isEventView()) {
2768 if ( incidence && incidence->type() == "Event" ) { 2772 if ( incidence && incidence->type() == "Event" ) {
2769 anEvent = static_cast<Event *>(incidence); 2773 anEvent = static_cast<Event *>(incidence);
2770 } 2774 }
2771 } 2775 }
2772 2776
2773 if (!anEvent) { 2777 if (!anEvent) {
2774 KNotifyClient::beep(); 2778 KNotifyClient::beep();
2775 return; 2779 return;
2776 } 2780 }
2777 2781
2778 deleteEvent(anEvent); 2782 deleteEvent(anEvent);
2779} 2783}
2780 2784
2781void CalendarView::todo_resub( Todo * parent, Todo * sub ) 2785void CalendarView::todo_resub( Todo * parent, Todo * sub )
2782{ 2786{
2783 if (!sub) return; 2787 if (!sub) return;
2784 if (!parent) return; 2788 if (!parent) return;
2785 if ( sub->relatedTo() ) 2789 if ( sub->relatedTo() )
2786 sub->relatedTo()->removeRelation(sub); 2790 sub->relatedTo()->removeRelation(sub);
2787 sub->setRelatedTo(parent); 2791 sub->setRelatedTo(parent);
2788 sub->setRelatedToUid(parent->uid()); 2792 sub->setRelatedToUid(parent->uid());
2789 parent->addRelation(sub); 2793 parent->addRelation(sub);
2790 sub->updated(); 2794 sub->updated();
2791 parent->updated(); 2795 parent->updated();
2792 setModified(true); 2796 setModified(true);
2793 updateView(); 2797 updateView();
2794} 2798}
2795void CalendarView::todo_unsub(Todo *anTodo ) 2799void CalendarView::todo_unsub(Todo *anTodo )
2796{ 2800{
2797 // Todo *anTodo = selectedTodo(); 2801 // Todo *anTodo = selectedTodo();
2798 if (!anTodo) return; 2802 if (!anTodo) return;
2799 if (!anTodo->relatedTo()) return; 2803 if (!anTodo->relatedTo()) return;
2800 anTodo->relatedTo()->removeRelation(anTodo); 2804 anTodo->relatedTo()->removeRelation(anTodo);
2801 anTodo->setRelatedTo(0); 2805 anTodo->setRelatedTo(0);
2802 anTodo->updated(); 2806 anTodo->updated();
2803 anTodo->setRelatedToUid(""); 2807 anTodo->setRelatedToUid("");
2804 setModified(true); 2808 setModified(true);
2805 updateView(); 2809 updateView();
2806} 2810}
2807 2811
2808void CalendarView::deleteTodo(Todo *todo) 2812void CalendarView::deleteTodo(Todo *todo)
2809{ 2813{
2810 if (!todo) { 2814 if (!todo) {
2811 KNotifyClient::beep(); 2815 KNotifyClient::beep();
2812 return; 2816 return;
2813 } 2817 }
2814 if (KOPrefs::instance()->mConfirm) { 2818 if (KOPrefs::instance()->mConfirm) {
2815 switch (msgItemDelete()) { 2819 switch (msgItemDelete()) {
2816 case KMessageBox::Continue: // OK 2820 case KMessageBox::Continue: // OK
2817 if (!todo->relations().isEmpty()) { 2821 if (!todo->relations().isEmpty()) {
2818 KMessageBox::sorry(this,i18n("Cannot delete To-Do\nwhich has children."), 2822 KMessageBox::sorry(this,i18n("Cannot delete To-Do\nwhich has children."),
2819 i18n("Delete To-Do")); 2823 i18n("Delete To-Do"));
2820 } else { 2824 } else {
2821 checkExternalId( todo ); 2825 checkExternalId( todo );
2822 calendar()->deleteTodo(todo); 2826 calendar()->deleteTodo(todo);
2823 changeTodoDisplay( todo,KOGlobals::EVENTDELETED ); 2827 changeTodoDisplay( todo,KOGlobals::EVENTDELETED );
2824 updateView(); 2828 updateView();
2825 } 2829 }
2826 break; 2830 break;
2827 } // switch 2831 } // switch
2828 } else { 2832 } else {
2829 if (!todo->relations().isEmpty()) { 2833 if (!todo->relations().isEmpty()) {
2830 KMessageBox::sorry(this,i18n("Cannot delete To-Do\nwhich has children."), 2834 KMessageBox::sorry(this,i18n("Cannot delete To-Do\nwhich has children."),
2831 i18n("Delete To-Do")); 2835 i18n("Delete To-Do"));
2832 } else { 2836 } else {
2833 checkExternalId( todo ); 2837 checkExternalId( todo );
2834 mCalendar->deleteTodo(todo); 2838 mCalendar->deleteTodo(todo);
2835 changeTodoDisplay( todo,KOGlobals::EVENTDELETED ); 2839 changeTodoDisplay( todo,KOGlobals::EVENTDELETED );
2836 updateView(); 2840 updateView();
2837 } 2841 }
2838 } 2842 }
2839 emit updateSearchDialog(); 2843 emit updateSearchDialog();
2840} 2844}
2841void CalendarView::deleteJournal(Journal *jour) 2845void CalendarView::deleteJournal(Journal *jour)
2842{ 2846{
2843 if (!jour) { 2847 if (!jour) {
2844 KNotifyClient::beep(); 2848 KNotifyClient::beep();
2845 return; 2849 return;
2846 } 2850 }
2847 if (KOPrefs::instance()->mConfirm) { 2851 if (KOPrefs::instance()->mConfirm) {
2848 switch (msgItemDelete()) { 2852 switch (msgItemDelete()) {
2849 case KMessageBox::Continue: // OK 2853 case KMessageBox::Continue: // OK
2850 calendar()->deleteJournal(jour); 2854 calendar()->deleteJournal(jour);
2851 updateView(); 2855 updateView();
2852 break; 2856 break;
2853 } // switch 2857 } // switch
2854 } else { 2858 } else {
2855 calendar()->deleteJournal(jour);; 2859 calendar()->deleteJournal(jour);;
2856 updateView(); 2860 updateView();
2857 } 2861 }
2858 emit updateSearchDialog(); 2862 emit updateSearchDialog();
2859} 2863}
2860 2864
2861void CalendarView::deleteEvent(Event *anEvent) 2865void CalendarView::deleteEvent(Event *anEvent)
2862{ 2866{
2863 if (!anEvent) { 2867 if (!anEvent) {
2864 KNotifyClient::beep(); 2868 KNotifyClient::beep();
2865 return; 2869 return;
2866 } 2870 }
2867 2871
2868 if (anEvent->recurrence()->doesRecur()) { 2872 if (anEvent->recurrence()->doesRecur()) {
2869 QDate itemDate = mViewManager->currentSelectionDate(); 2873 QDate itemDate = mViewManager->currentSelectionDate();
2870 int km; 2874 int km;
2871 if (!itemDate.isValid()) { 2875 if (!itemDate.isValid()) {
2872 //kdDebug() << "Date Not Valid" << endl; 2876 //kdDebug() << "Date Not Valid" << endl;
2873 if (KOPrefs::instance()->mConfirm) { 2877 if (KOPrefs::instance()->mConfirm) {
2874 km = KMessageBox::warningContinueCancel(this,anEvent->summary() + 2878 km = KMessageBox::warningContinueCancel(this,anEvent->summary() +
2875 i18n("\nThis event recurs\nover multiple dates.\nAre you sure you want\nto delete this event\nand all its recurrences?"), 2879 i18n("\nThis event recurs\nover multiple dates.\nAre you sure you want\nto delete this event\nand all its recurrences?"),
2876 i18n("KO/Pi Confirmation"),i18n("Delete All")); 2880 i18n("KO/Pi Confirmation"),i18n("Delete All"));
2877 if ( km == KMessageBox::Continue ) 2881 if ( km == KMessageBox::Continue )
2878 km = KMessageBox::No; // No = all below 2882 km = KMessageBox::No; // No = all below
2879 } else 2883 } else
2880 km = KMessageBox::No; 2884 km = KMessageBox::No;
2881 } else { 2885 } else {
2882 km = KMessageBox::warningYesNoCancel(this,anEvent->summary() + 2886 km = KMessageBox::warningYesNoCancel(this,anEvent->summary() +
2883 i18n("\nThis event recurs\nover multiple dates.\nDo you want to delete\nall it's recurrences,\nor only the current one on:\n")+ 2887 i18n("\nThis event recurs\nover multiple dates.\nDo you want to delete\nall it's recurrences,\nor only the current one on:\n")+
2884 KGlobal::locale()->formatDate(itemDate)+i18n(" ?\n\nDelete:\n"), 2888 KGlobal::locale()->formatDate(itemDate)+i18n(" ?\n\nDelete:\n"),
2885 i18n("KO/Pi Confirmation"),i18n("Current"), 2889 i18n("KO/Pi Confirmation"),i18n("Current"),
2886 i18n("All")); 2890 i18n("All"));
2887 } 2891 }
2888 switch(km) { 2892 switch(km) {
2889 2893
2890 case KMessageBox::No: // Continue // all 2894 case KMessageBox::No: // Continue // all
2891 //qDebug("KMessageBox::No "); 2895 //qDebug("KMessageBox::No ");
2892 if (anEvent->organizer()==KOPrefs::instance()->email() && anEvent->attendeeCount()>0) 2896 if (anEvent->organizer()==KOPrefs::instance()->email() && anEvent->attendeeCount()>0)
2893 schedule(Scheduler::Cancel,anEvent); 2897 schedule(Scheduler::Cancel,anEvent);
2894 2898
2895 checkExternalId( anEvent); 2899 checkExternalId( anEvent);
2896 mCalendar->deleteEvent(anEvent); 2900 mCalendar->deleteEvent(anEvent);
2897 changeEventDisplay(anEvent,KOGlobals::EVENTDELETED); 2901 changeEventDisplay(anEvent,KOGlobals::EVENTDELETED);
2898 break; 2902 break;
2899 2903
2900 // Disabled because it does not work 2904 // Disabled because it does not work
2901 //#if 0 2905 //#if 0
2902 case KMessageBox::Yes: // just this one 2906 case KMessageBox::Yes: // just this one
2903 //QDate qd = mNavigator->selectedDates().first(); 2907 //QDate qd = mNavigator->selectedDates().first();
2904 //if (!qd.isValid()) { 2908 //if (!qd.isValid()) {
2905 // kdDebug() << "no date selected, or invalid date" << endl; 2909 // kdDebug() << "no date selected, or invalid date" << endl;
2906 // KNotifyClient::beep(); 2910 // KNotifyClient::beep();
2907 // return; 2911 // return;
2908 //} 2912 //}
2909 //while (!anEvent->recursOn(qd)) qd = qd.addDays(1); 2913 //while (!anEvent->recursOn(qd)) qd = qd.addDays(1);
2910 if (itemDate!=QDate(1,1,1) || itemDate.isValid()) { 2914 if (itemDate!=QDate(1,1,1) || itemDate.isValid()) {
2911 anEvent->addExDate(itemDate); 2915 anEvent->addExDate(itemDate);
2912 int duration = anEvent->recurrence()->duration(); 2916 int duration = anEvent->recurrence()->duration();
2913 if ( duration > 0 ) { 2917 if ( duration > 0 ) {
2914 anEvent->recurrence()->setDuration( duration - 1 ); 2918 anEvent->recurrence()->setDuration( duration - 1 );
2915 } 2919 }
2916 changeEventDisplay(anEvent, KOGlobals::EVENTEDITED); 2920 changeEventDisplay(anEvent, KOGlobals::EVENTEDITED);
2917 } 2921 }
2918 break; 2922 break;
2919 //#endif 2923 //#endif
2920 } // switch 2924 } // switch
2921 } else { 2925 } else {
2922 if (KOPrefs::instance()->mConfirm) { 2926 if (KOPrefs::instance()->mConfirm) {
2923 switch (KMessageBox::warningContinueCancel(this,anEvent->summary() + 2927 switch (KMessageBox::warningContinueCancel(this,anEvent->summary() +
2924 i18n("\nAre you sure you want\nto delete this event?"), 2928 i18n("\nAre you sure you want\nto delete this event?"),
2925 i18n("KO/Pi Confirmation"),i18n("Delete"))) { 2929 i18n("KO/Pi Confirmation"),i18n("Delete"))) {
2926 case KMessageBox::Continue: // OK 2930 case KMessageBox::Continue: // OK
2927 if (anEvent->organizer()==KOPrefs::instance()->email() && anEvent->attendeeCount()>0) 2931 if (anEvent->organizer()==KOPrefs::instance()->email() && anEvent->attendeeCount()>0)
2928 schedule(Scheduler::Cancel,anEvent); 2932 schedule(Scheduler::Cancel,anEvent);
2929 checkExternalId( anEvent); 2933 checkExternalId( anEvent);
2930 mCalendar->deleteEvent(anEvent); 2934 mCalendar->deleteEvent(anEvent);
2931 changeEventDisplay(anEvent, KOGlobals::EVENTDELETED); 2935 changeEventDisplay(anEvent, KOGlobals::EVENTDELETED);
2932 break; 2936 break;
2933 } // switch 2937 } // switch
2934 } else { 2938 } else {
2935 if (anEvent->organizer()==KOPrefs::instance()->email() && anEvent->attendeeCount()>0) 2939 if (anEvent->organizer()==KOPrefs::instance()->email() && anEvent->attendeeCount()>0)
2936 schedule(Scheduler::Cancel,anEvent); 2940 schedule(Scheduler::Cancel,anEvent);
2937 checkExternalId( anEvent); 2941 checkExternalId( anEvent);
2938 mCalendar->deleteEvent(anEvent); 2942 mCalendar->deleteEvent(anEvent);
2939 changeEventDisplay(anEvent, KOGlobals::EVENTDELETED); 2943 changeEventDisplay(anEvent, KOGlobals::EVENTDELETED);
2940 } 2944 }
2941 } // if-else 2945 } // if-else
2942 emit updateSearchDialog(); 2946 emit updateSearchDialog();
2943} 2947}
2944 2948
2945bool CalendarView::deleteEvent(const QString &uid) 2949bool CalendarView::deleteEvent(const QString &uid)
2946{ 2950{
2947 Event *ev = mCalendar->event(uid); 2951 Event *ev = mCalendar->event(uid);
2948 if (ev) { 2952 if (ev) {
2949 deleteEvent(ev); 2953 deleteEvent(ev);
2950 return true; 2954 return true;
2951 } else { 2955 } else {
2952 return false; 2956 return false;
2953 } 2957 }
2954} 2958}
2955 2959
2956/*****************************************************************************/ 2960/*****************************************************************************/
2957 2961
2958void CalendarView::action_mail() 2962void CalendarView::action_mail()
2959{ 2963{
2960#ifndef KORG_NOMAIL 2964#ifndef KORG_NOMAIL
2961 KOMailClient mailClient; 2965 KOMailClient mailClient;
2962 2966
2963 Incidence *incidence = currentSelection(); 2967 Incidence *incidence = currentSelection();
2964 2968
2965 if (!incidence) { 2969 if (!incidence) {
2966 KMessageBox::sorry(this,i18n("Can't generate mail:\nNo event selected.")); 2970 KMessageBox::sorry(this,i18n("Can't generate mail:\nNo event selected."));
2967 return; 2971 return;
2968 } 2972 }
2969 if(incidence->attendeeCount() == 0 ) { 2973 if(incidence->attendeeCount() == 0 ) {
diff --git a/korganizer/kotodoview.cpp b/korganizer/kotodoview.cpp
index a12acd1..9cafc60 100644
--- a/korganizer/kotodoview.cpp
+++ b/korganizer/kotodoview.cpp
@@ -1,888 +1,889 @@
1/* 1/*
2 This file is part of KOrganizer. 2 This file is part of KOrganizer.
3 Copyright (c) 2000,2001 Cornelius Schumacher <schumacher@kde.org> 3 Copyright (c) 2000,2001 Cornelius Schumacher <schumacher@kde.org>
4 4
5 This program is free software; you can redistribute it and/or modify 5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by 6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2 of the License, or 7 the Free Software Foundation; either version 2 of the License, or
8 (at your option) any later version. 8 (at your option) any later version.
9 9
10 This program is distributed in the hope that it will be useful, 10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of 11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details. 13 GNU General Public License for more details.
14 14
15 You should have received a copy of the GNU General Public License 15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software 16 along with this program; if not, write to the Free Software
17 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. 17 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
18 18
19 As a special exception, permission is given to link this program 19 As a special exception, permission is given to link this program
20 with any edition of Qt, and distribute the resulting executable, 20 with any edition of Qt, and distribute the resulting executable,
21 without including the source code for Qt in the source distribution. 21 without including the source code for Qt in the source distribution.
22*/ 22*/
23 23
24#include <qlayout.h> 24#include <qlayout.h>
25#include <qheader.h> 25#include <qheader.h>
26#include <qcursor.h> 26#include <qcursor.h>
27 27
28#include <qvbox.h> 28#include <qvbox.h>
29#include <kdebug.h> 29#include <kdebug.h>
30#include "koprefs.h" 30#include "koprefs.h"
31#include <klocale.h> 31#include <klocale.h>
32#include <kglobal.h> 32#include <kglobal.h>
33#include <kiconloader.h> 33#include <kiconloader.h>
34#include <kmessagebox.h> 34#include <kmessagebox.h>
35 35
36#include <libkcal/icaldrag.h> 36#include <libkcal/icaldrag.h>
37#include <libkcal/vcaldrag.h> 37#include <libkcal/vcaldrag.h>
38#include <libkcal/calfilter.h> 38#include <libkcal/calfilter.h>
39#include <libkcal/dndfactory.h> 39#include <libkcal/dndfactory.h>
40#include <libkcal/calendarresources.h> 40#include <libkcal/calendarresources.h>
41#include <libkcal/resourcecalendar.h> 41#include <libkcal/resourcecalendar.h>
42#include <kresources/resourceselectdialog.h> 42#include <kresources/resourceselectdialog.h>
43#ifndef DESKTOP_VERSION 43#ifndef DESKTOP_VERSION
44#include <qpe/qpeapplication.h> 44#include <qpe/qpeapplication.h>
45#else 45#else
46#include <qapplication.h> 46#include <qapplication.h>
47#endif 47#endif
48#ifndef KORG_NOPRINTER 48#ifndef KORG_NOPRINTER
49#include "calprinter.h" 49#include "calprinter.h"
50#endif 50#endif
51#include "docprefs.h" 51#include "docprefs.h"
52 52
53#include "kotodoview.h" 53#include "kotodoview.h"
54using namespace KOrg; 54using namespace KOrg;
55 55
56KOTodoListView::KOTodoListView(Calendar *calendar,QWidget *parent, 56KOTodoListView::KOTodoListView(Calendar *calendar,QWidget *parent,
57 const char *name) : 57 const char *name) :
58 KListView(parent,name) 58 KListView(parent,name)
59{ 59{
60 mName = QString ( name ); 60 mName = QString ( name );
61 mCalendar = calendar; 61 mCalendar = calendar;
62#ifndef DESKTOP_VERSION 62#ifndef DESKTOP_VERSION
63 QPEApplication::setStylusOperation(viewport(), QPEApplication::RightOnHold ); 63 QPEApplication::setStylusOperation(viewport(), QPEApplication::RightOnHold );
64#endif 64#endif
65 mOldCurrent = 0; 65 mOldCurrent = 0;
66 mMousePressed = false; 66 mMousePressed = false;
67 67
68 setAcceptDrops(true); 68 setAcceptDrops(true);
69 viewport()->setAcceptDrops(true); 69 viewport()->setAcceptDrops(true);
70 int size = 16; 70 int size = 16;
71 if (qApp->desktop()->width() < 300 ) 71 if (qApp->desktop()->width() < 300 )
72 size = 12; 72 size = 12;
73 setTreeStepSize( size + 6 ); 73 setTreeStepSize( size + 6 );
74 74
75} 75}
76 76
77void KOTodoListView::contentsDragEnterEvent(QDragEnterEvent *e) 77void KOTodoListView::contentsDragEnterEvent(QDragEnterEvent *e)
78{ 78{
79#ifndef KORG_NODND 79#ifndef KORG_NODND
80// kdDebug() << "KOTodoListView::contentsDragEnterEvent" << endl; 80// kdDebug() << "KOTodoListView::contentsDragEnterEvent" << endl;
81 if ( !ICalDrag::canDecode( e ) && !VCalDrag::canDecode( e ) && 81 if ( !ICalDrag::canDecode( e ) && !VCalDrag::canDecode( e ) &&
82 !QTextDrag::canDecode( e ) ) { 82 !QTextDrag::canDecode( e ) ) {
83 e->ignore(); 83 e->ignore();
84 return; 84 return;
85 } 85 }
86 86
87 mOldCurrent = currentItem(); 87 mOldCurrent = currentItem();
88#endif 88#endif
89} 89}
90 90
91 91
92void KOTodoListView::contentsDragMoveEvent(QDragMoveEvent *e) 92void KOTodoListView::contentsDragMoveEvent(QDragMoveEvent *e)
93{ 93{
94#ifndef KORG_NODND 94#ifndef KORG_NODND
95// kdDebug() << "KOTodoListView::contentsDragMoveEvent" << endl; 95// kdDebug() << "KOTodoListView::contentsDragMoveEvent" << endl;
96 96
97 if ( !ICalDrag::canDecode( e ) && !VCalDrag::canDecode( e ) && 97 if ( !ICalDrag::canDecode( e ) && !VCalDrag::canDecode( e ) &&
98 !QTextDrag::canDecode( e ) ) { 98 !QTextDrag::canDecode( e ) ) {
99 e->ignore(); 99 e->ignore();
100 return; 100 return;
101 } 101 }
102 102
103 e->accept(); 103 e->accept();
104#endif 104#endif
105} 105}
106 106
107void KOTodoListView::contentsDragLeaveEvent(QDragLeaveEvent *) 107void KOTodoListView::contentsDragLeaveEvent(QDragLeaveEvent *)
108{ 108{
109#ifndef KORG_NODND 109#ifndef KORG_NODND
110// kdDebug() << "KOTodoListView::contentsDragLeaveEvent" << endl; 110// kdDebug() << "KOTodoListView::contentsDragLeaveEvent" << endl;
111 111
112 setCurrentItem(mOldCurrent); 112 setCurrentItem(mOldCurrent);
113 setSelected(mOldCurrent,true); 113 setSelected(mOldCurrent,true);
114#endif 114#endif
115} 115}
116 116
117void KOTodoListView::contentsDropEvent(QDropEvent *e) 117void KOTodoListView::contentsDropEvent(QDropEvent *e)
118{ 118{
119#ifndef KORG_NODND 119#ifndef KORG_NODND
120// kdDebug() << "KOTodoListView::contentsDropEvent" << endl; 120// kdDebug() << "KOTodoListView::contentsDropEvent" << endl;
121 121
122 if ( !ICalDrag::canDecode( e ) && !VCalDrag::canDecode( e ) && 122 if ( !ICalDrag::canDecode( e ) && !VCalDrag::canDecode( e ) &&
123 !QTextDrag::canDecode( e ) ) { 123 !QTextDrag::canDecode( e ) ) {
124 e->ignore(); 124 e->ignore();
125 return; 125 return;
126 } 126 }
127 127
128 DndFactory factory( mCalendar ); 128 DndFactory factory( mCalendar );
129 Todo *todo = factory.createDropTodo(e); 129 Todo *todo = factory.createDropTodo(e);
130 130
131 if (todo) { 131 if (todo) {
132 e->acceptAction(); 132 e->acceptAction();
133 133
134 KOTodoViewItem *destination = 134 KOTodoViewItem *destination =
135 (KOTodoViewItem *)itemAt(contentsToViewport(e->pos())); 135 (KOTodoViewItem *)itemAt(contentsToViewport(e->pos()));
136 Todo *destinationEvent = 0; 136 Todo *destinationEvent = 0;
137 if (destination) destinationEvent = destination->todo(); 137 if (destination) destinationEvent = destination->todo();
138 138
139 Todo *existingTodo = mCalendar->todo(todo->uid()); 139 Todo *existingTodo = mCalendar->todo(todo->uid());
140 140
141 if(existingTodo) { 141 if(existingTodo) {
142// kdDebug() << "Drop existing Todo" << endl; 142// kdDebug() << "Drop existing Todo" << endl;
143 Incidence *to = destinationEvent; 143 Incidence *to = destinationEvent;
144 while(to) { 144 while(to) {
145 if (to->uid() == todo->uid()) { 145 if (to->uid() == todo->uid()) {
146 KMessageBox::sorry(this, 146 KMessageBox::sorry(this,
147 i18n("Cannot move To-Do to itself or a child of itself"), 147 i18n("Cannot move To-Do to itself or a child of itself"),
148 i18n("Drop To-Do")); 148 i18n("Drop To-Do"));
149 delete todo; 149 delete todo;
150 return; 150 return;
151 } 151 }
152 to = to->relatedTo(); 152 to = to->relatedTo();
153 } 153 }
154 existingTodo->setRelatedTo(destinationEvent); 154 existingTodo->setRelatedTo(destinationEvent);
155 emit todoDropped(todo); 155 emit todoDropped(todo);
156 delete todo; 156 delete todo;
157 } else { 157 } else {
158// kdDebug() << "Drop new Todo" << endl; 158// kdDebug() << "Drop new Todo" << endl;
159 todo->setRelatedTo(destinationEvent); 159 todo->setRelatedTo(destinationEvent);
160 mCalendar->addTodo(todo); 160 mCalendar->addTodo(todo);
161 161
162 emit todoDropped(todo); 162 emit todoDropped(todo);
163 } 163 }
164 } 164 }
165 else { 165 else {
166 QString text; 166 QString text;
167 if (QTextDrag::decode(e,text)) { 167 if (QTextDrag::decode(e,text)) {
168 //QListViewItem *qlvi = itemAt( contentsToViewport(e->pos()) ); 168 //QListViewItem *qlvi = itemAt( contentsToViewport(e->pos()) );
169 KOTodoViewItem *todoi = static_cast<KOTodoViewItem *>(itemAt( contentsToViewport(e->pos()) )); 169 KOTodoViewItem *todoi = static_cast<KOTodoViewItem *>(itemAt( contentsToViewport(e->pos()) ));
170 kdDebug() << "Dropped : " << text << endl; 170 kdDebug() << "Dropped : " << text << endl;
171 QStringList emails = QStringList::split(",",text); 171 QStringList emails = QStringList::split(",",text);
172 for(QStringList::ConstIterator it = emails.begin();it!=emails.end();++it) { 172 for(QStringList::ConstIterator it = emails.begin();it!=emails.end();++it) {
173 kdDebug() << " Email: " << (*it) << endl; 173 kdDebug() << " Email: " << (*it) << endl;
174 int pos = (*it).find("<"); 174 int pos = (*it).find("<");
175 QString name = (*it).left(pos); 175 QString name = (*it).left(pos);
176 QString email = (*it).mid(pos); 176 QString email = (*it).mid(pos);
177 if (!email.isEmpty() && todoi) { 177 if (!email.isEmpty() && todoi) {
178 todoi->todo()->addAttendee(new Attendee(name,email)); 178 todoi->todo()->addAttendee(new Attendee(name,email));
179 } 179 }
180 } 180 }
181 } 181 }
182 else { 182 else {
183 kdDebug() << "KOTodoListView::contentsDropEvent(): Todo from drop not decodable" << endl; 183 kdDebug() << "KOTodoListView::contentsDropEvent(): Todo from drop not decodable" << endl;
184 e->ignore(); 184 e->ignore();
185 } 185 }
186 } 186 }
187#endif 187#endif
188} 188}
189 189
190void KOTodoListView::contentsMousePressEvent(QMouseEvent* e) 190void KOTodoListView::contentsMousePressEvent(QMouseEvent* e)
191{ 191{
192 QListView::contentsMousePressEvent(e); 192 QListView::contentsMousePressEvent(e);
193#ifndef KORG_NODND 193#ifndef KORG_NODND
194 QPoint p(contentsToViewport(e->pos())); 194 QPoint p(contentsToViewport(e->pos()));
195 QListViewItem *i = itemAt(p); 195 QListViewItem *i = itemAt(p);
196 if (i) { 196 if (i) {
197 // if the user clicked into the root decoration of the item, don't 197 // if the user clicked into the root decoration of the item, don't
198 // try to start a drag! 198 // try to start a drag!
199 if (p.x() > header()->sectionPos(header()->mapToIndex(0)) + 199 if (p.x() > header()->sectionPos(header()->mapToIndex(0)) +
200 treeStepSize() * (i->depth() + (rootIsDecorated() ? 1 : 0)) + 200 treeStepSize() * (i->depth() + (rootIsDecorated() ? 1 : 0)) +
201 itemMargin() || 201 itemMargin() ||
202 p.x() < header()->sectionPos(header()->mapToIndex(0))) { 202 p.x() < header()->sectionPos(header()->mapToIndex(0))) {
203 if (e->button()==Qt::LeftButton) { 203 if (e->button()==Qt::LeftButton) {
204 mPressPos = e->pos(); 204 mPressPos = e->pos();
205 mMousePressed = true; 205 mMousePressed = true;
206 } 206 }
207 } 207 }
208 } 208 }
209#endif 209#endif
210} 210}
211 211
212void KOTodoListView::contentsMouseMoveEvent(QMouseEvent* e) 212void KOTodoListView::contentsMouseMoveEvent(QMouseEvent* e)
213{ 213{
214 214
215#ifndef KORG_NODND 215#ifndef KORG_NODND
216// kdDebug() << "KOTodoListView::contentsMouseMoveEvent()" << endl; 216// kdDebug() << "KOTodoListView::contentsMouseMoveEvent()" << endl;
217 QListView::contentsMouseMoveEvent(e); 217 QListView::contentsMouseMoveEvent(e);
218 if (mMousePressed && (mPressPos - e->pos()).manhattanLength() > 218 if (mMousePressed && (mPressPos - e->pos()).manhattanLength() >
219 QApplication::startDragDistance()) { 219 QApplication::startDragDistance()) {
220 mMousePressed = false; 220 mMousePressed = false;
221 QListViewItem *item = itemAt(contentsToViewport(mPressPos)); 221 QListViewItem *item = itemAt(contentsToViewport(mPressPos));
222 if (item) { 222 if (item) {
223// kdDebug() << "Start Drag for item " << item->text(0) << endl; 223// kdDebug() << "Start Drag for item " << item->text(0) << endl;
224 DndFactory factory( mCalendar ); 224 DndFactory factory( mCalendar );
225 ICalDrag *vd = factory.createDragTodo( 225 ICalDrag *vd = factory.createDragTodo(
226 ((KOTodoViewItem *)item)->todo(),viewport()); 226 ((KOTodoViewItem *)item)->todo(),viewport());
227 if (vd->drag()) { 227 if (vd->drag()) {
228 kdDebug() << "KOTodoListView::contentsMouseMoveEvent(): Delete drag source" << endl; 228 kdDebug() << "KOTodoListView::contentsMouseMoveEvent(): Delete drag source" << endl;
229 } 229 }
230/* 230/*
231 QString source = fullPath(item); 231 QString source = fullPath(item);
232 if ( QFile::exists(source) ) { 232 if ( QFile::exists(source) ) {
233 QUriDrag* ud = new QUriDrag(viewport()); 233 QUriDrag* ud = new QUriDrag(viewport());
234 ud->setFilenames( source ); 234 ud->setFilenames( source );
235 if ( ud->drag() ) 235 if ( ud->drag() )
236 QMessageBox::information( this, "Drag source", 236 QMessageBox::information( this, "Drag source",
237 QString("Delete ")+source, "Not implemented" ); 237 QString("Delete ")+source, "Not implemented" );
238*/ 238*/
239 } 239 }
240 } 240 }
241#endif 241#endif
242} 242}
243void KOTodoListView::keyPressEvent ( QKeyEvent * e ) 243void KOTodoListView::keyPressEvent ( QKeyEvent * e )
244{ 244{
245 245
246 QListViewItem* cn; 246 QListViewItem* cn;
247 if ( e->key() == Qt::Key_Return || e->key() == Qt::Key_Enter ) { 247 if ( e->key() == Qt::Key_Return || e->key() == Qt::Key_Enter ) {
248 cn = currentItem(); 248 cn = currentItem();
249 if ( cn ) { 249 if ( cn ) {
250 KOTodoViewItem* ci = (KOTodoViewItem*)( cn ); 250 KOTodoViewItem* ci = (KOTodoViewItem*)( cn );
251 if ( ci ){ 251 if ( ci ){
252 if ( e->state() == ShiftButton ) 252 if ( e->state() == ShiftButton )
253 ci->setOn( false ); 253 ci->setOn( false );
254 else 254 else
255 ci->setOn( true ); 255 ci->setOn( true );
256 cn = cn->nextSibling(); 256 cn = cn->nextSibling();
257 if ( cn ) { 257 if ( cn ) {
258 setCurrentItem ( cn ); 258 setCurrentItem ( cn );
259 ensureItemVisible ( cn ); 259 ensureItemVisible ( cn );
260 } 260 }
261 261
262 } 262 }
263 } 263 }
264 264
265 return; 265 return;
266 } 266 }
267 267
268 // qDebug("KOTodoListView::keyPressEvent "); 268 // qDebug("KOTodoListView::keyPressEvent ");
269 if ( e->state() == Qt::ControlButton || e->state() == Qt::ShiftButton || mName != "todolistsmall" ) { 269 if ( e->state() == Qt::ControlButton || e->state() == Qt::ShiftButton || mName != "todolistsmall" ) {
270 switch ( e->key() ) { 270 switch ( e->key() ) {
271 case Qt::Key_Down: 271 case Qt::Key_Down:
272 case Qt::Key_Up: 272 case Qt::Key_Up:
273 QListView::keyPressEvent ( e ); 273 QListView::keyPressEvent ( e );
274 break; 274 break;
275 case Qt::Key_Left: 275 case Qt::Key_Left:
276 case Qt::Key_Right: 276 case Qt::Key_Right:
277 QListView::keyPressEvent ( e ); 277 QListView::keyPressEvent ( e );
278 e->accept(); 278 e->accept();
279 return; 279 return;
280 break; 280 break;
281 default: 281 default:
282 e->ignore(); 282 e->ignore();
283 break; 283 break;
284 } 284 }
285 return; 285 return;
286 } 286 }
287 e->ignore(); 287 e->ignore();
288} 288}
289void KOTodoListView::contentsMouseReleaseEvent(QMouseEvent *e) 289void KOTodoListView::contentsMouseReleaseEvent(QMouseEvent *e)
290{ 290{
291 QListView::contentsMouseReleaseEvent(e); 291 QListView::contentsMouseReleaseEvent(e);
292 mMousePressed = false; 292 mMousePressed = false;
293} 293}
294 294
295void KOTodoListView::contentsMouseDoubleClickEvent(QMouseEvent *e) 295void KOTodoListView::contentsMouseDoubleClickEvent(QMouseEvent *e)
296{ 296{
297 if (!e) return; 297 if (!e) return;
298 298
299 QPoint vp = contentsToViewport(e->pos()); 299 QPoint vp = contentsToViewport(e->pos());
300 300
301 QListViewItem *item = itemAt(vp); 301 QListViewItem *item = itemAt(vp);
302 302
303 emit double_Clicked(item); 303 emit double_Clicked(item);
304 if (!item) return; 304 if (!item) return;
305 305
306 emit doubleClicked(item,vp,0); 306 emit doubleClicked(item,vp,0);
307} 307}
308 308
309///////////////////////////////////////////////////////////////////////////// 309/////////////////////////////////////////////////////////////////////////////
310 310
311KOQuickTodo::KOQuickTodo(QWidget *parent) : 311KOQuickTodo::KOQuickTodo(QWidget *parent) :
312 QLineEdit(parent) 312 QLineEdit(parent)
313{ 313{
314 setText(i18n("Click to add a new Todo")); 314 setText(i18n("Click to add a new Todo"));
315} 315}
316 316
317void KOQuickTodo::focusInEvent(QFocusEvent *ev) 317void KOQuickTodo::focusInEvent(QFocusEvent *ev)
318{ 318{
319 if ( text()==i18n("Click to add a new Todo") ) 319 if ( text()==i18n("Click to add a new Todo") )
320 setText(""); 320 setText("");
321 QLineEdit::focusInEvent(ev); 321 QLineEdit::focusInEvent(ev);
322} 322}
323 323
324void KOQuickTodo::focusOutEvent(QFocusEvent *ev) 324void KOQuickTodo::focusOutEvent(QFocusEvent *ev)
325{ 325{
326 setText(i18n("Click to add a new Todo")); 326 setText(i18n("Click to add a new Todo"));
327 QLineEdit::focusOutEvent(ev); 327 QLineEdit::focusOutEvent(ev);
328} 328}
329 329
330///////////////////////////////////////////////////////////////////////////// 330/////////////////////////////////////////////////////////////////////////////
331 331
332KOTodoView::KOTodoView(Calendar *calendar,QWidget* parent,const char* name) : 332KOTodoView::KOTodoView(Calendar *calendar,QWidget* parent,const char* name) :
333 KOrg::BaseView(calendar,parent,name) 333 KOrg::BaseView(calendar,parent,name)
334{ 334{
335 QBoxLayout *topLayout = new QVBoxLayout(this); 335 QBoxLayout *topLayout = new QVBoxLayout(this);
336 mName = QString ( name ); 336 mName = QString ( name );
337 mBlockUpdate = false; 337 mBlockUpdate = false;
338 mQuickAdd = new KOQuickTodo(this); 338 mQuickAdd = new KOQuickTodo(this);
339 topLayout->addWidget(mQuickAdd); 339 topLayout->addWidget(mQuickAdd);
340 340
341 if ( !KOPrefs::instance()->mEnableQuickTodo ) mQuickAdd->hide(); 341 if ( !KOPrefs::instance()->mEnableQuickTodo ) mQuickAdd->hide();
342 342
343 mTodoListView = new KOTodoListView(calendar,this, name ); 343 mTodoListView = new KOTodoListView(calendar,this, name );
344 topLayout->addWidget(mTodoListView); 344 topLayout->addWidget(mTodoListView);
345 //mTodoListView->header()->setMaximumHeight(30); 345 //mTodoListView->header()->setMaximumHeight(30);
346 mTodoListView->setRootIsDecorated(true); 346 mTodoListView->setRootIsDecorated(true);
347 mTodoListView->setAllColumnsShowFocus(true); 347 mTodoListView->setAllColumnsShowFocus(true);
348 348
349 mTodoListView->setShowSortIndicator(true); 349 mTodoListView->setShowSortIndicator(true);
350 350
351 mTodoListView->addColumn(i18n("Todo")); 351 mTodoListView->addColumn(i18n("Todo"));
352 mTodoListView->addColumn(i18n("Prio")); 352 mTodoListView->addColumn(i18n("Prio"));
353 mTodoListView->setColumnAlignment(1,AlignHCenter); 353 mTodoListView->setColumnAlignment(1,AlignHCenter);
354 mTodoListView->addColumn(i18n("Complete")); 354 mTodoListView->addColumn(i18n("Complete"));
355 mTodoListView->setColumnAlignment(2,AlignHCenter); 355 mTodoListView->setColumnAlignment(2,AlignHCenter);
356 mTodoListView->addColumn(i18n("Due Date")); 356 mTodoListView->addColumn(i18n("Due Date"));
357 mTodoListView->setColumnAlignment(3,AlignLeft); 357 mTodoListView->setColumnAlignment(3,AlignLeft);
358 mTodoListView->addColumn(i18n("Due Time")); 358 mTodoListView->addColumn(i18n("Due Time"));
359 mTodoListView->setColumnAlignment(4,AlignHCenter); 359 mTodoListView->setColumnAlignment(4,AlignHCenter);
360 mTodoListView->addColumn(i18n("Cancelled")); 360 mTodoListView->addColumn(i18n("Cancelled"));
361 mTodoListView->addColumn(i18n("Categories")); 361 mTodoListView->addColumn(i18n("Categories"));
362#if 0 362#if 0
363 mTodoListView->addColumn(i18n("Sort Id")); 363 mTodoListView->addColumn(i18n("Sort Id"));
364 mTodoListView->setColumnAlignment(4,AlignHCenter); 364 mTodoListView->setColumnAlignment(4,AlignHCenter);
365#endif 365#endif
366 366
367 mTodoListView->setMinimumHeight( 60 ); 367 mTodoListView->setMinimumHeight( 60 );
368 mTodoListView->setItemsRenameable( true ); 368 mTodoListView->setItemsRenameable( true );
369 mTodoListView->setRenameable( 0 ); 369 mTodoListView->setRenameable( 0 );
370 mTodoListView->setColumnWidth( 0, 120 ); 370 mTodoListView->setColumnWidth( 0, 120 );
371 mTodoListView->setColumnWidthMode(0, QListView::Manual); 371 mTodoListView->setColumnWidthMode(0, QListView::Manual);
372 mTodoListView->setColumnWidthMode(1, QListView::Manual); 372 mTodoListView->setColumnWidthMode(1, QListView::Manual);
373 mTodoListView->setColumnWidthMode(2, QListView::Manual); 373 mTodoListView->setColumnWidthMode(2, QListView::Manual);
374 mTodoListView->setColumnWidthMode(3, QListView::Manual); 374 mTodoListView->setColumnWidthMode(3, QListView::Manual);
375 mTodoListView->setColumnWidthMode(4, QListView::Manual); 375 mTodoListView->setColumnWidthMode(4, QListView::Manual);
376 mTodoListView->setColumnWidthMode(5, QListView::Manual); 376 mTodoListView->setColumnWidthMode(5, QListView::Manual);
377 mTodoListView->setColumnWidthMode(6, QListView::Manual);
377 mTodoListView->setColumnAlignment( 2, AlignCenter ); 378 mTodoListView->setColumnAlignment( 2, AlignCenter );
378#if 0 379#if 0
379 mTodoListView->setColumnWidthMode(6, QListView::Manual); 380 mTodoListView->setColumnWidthMode(6, QListView::Manual);
380#endif 381#endif
381 382
382 mPriorityPopupMenu = new QPopupMenu(this); 383 mPriorityPopupMenu = new QPopupMenu(this);
383 for (int i = 1; i <= 5; i++) { 384 for (int i = 1; i <= 5; i++) {
384 QString label = QString ("%1").arg (i); 385 QString label = QString ("%1").arg (i);
385 mPriority[mPriorityPopupMenu->insertItem (label)] = i; 386 mPriority[mPriorityPopupMenu->insertItem (label)] = i;
386 } 387 }
387 connect (mPriorityPopupMenu, SIGNAL(activated (int)), SLOT (setNewPriority(int))); 388 connect (mPriorityPopupMenu, SIGNAL(activated (int)), SLOT (setNewPriority(int)));
388 389
389 mPercentageCompletedPopupMenu = new QPopupMenu(this); 390 mPercentageCompletedPopupMenu = new QPopupMenu(this);
390 for (int i = 0; i <= 100; i+=20) { 391 for (int i = 0; i <= 100; i+=20) {
391 QString label = QString ("%1 %").arg (i); 392 QString label = QString ("%1 %").arg (i);
392 mPercentage[mPercentageCompletedPopupMenu->insertItem (label)] = i; 393 mPercentage[mPercentageCompletedPopupMenu->insertItem (label)] = i;
393 } 394 }
394 connect (mPercentageCompletedPopupMenu, SIGNAL (activated (int)), SLOT (setNewPercentage (int))); 395 connect (mPercentageCompletedPopupMenu, SIGNAL (activated (int)), SLOT (setNewPercentage (int)));
395 396
396 397
397 398
398 mItemPopupMenu = new QPopupMenu(this); 399 mItemPopupMenu = new QPopupMenu(this);
399 mItemPopupMenu->insertItem(i18n("Show..."), this, 400 mItemPopupMenu->insertItem(i18n("Show..."), this,
400 SLOT (showTodo())); 401 SLOT (showTodo()));
401 mItemPopupMenu->insertItem(i18n("Edit..."), this, 402 mItemPopupMenu->insertItem(i18n("Edit..."), this,
402 SLOT (editTodo())); 403 SLOT (editTodo()));
403 mItemPopupMenu->insertItem( i18n("Delete"), this, 404 mItemPopupMenu->insertItem( i18n("Delete"), this,
404 SLOT (deleteTodo())); 405 SLOT (deleteTodo()));
405 mItemPopupMenu->insertItem( i18n("Clone..."), this, 406 mItemPopupMenu->insertItem( i18n("Clone..."), this,
406 SLOT (cloneTodo())); 407 SLOT (cloneTodo()));
407 mItemPopupMenu->insertItem( i18n("Move..."), this, 408 mItemPopupMenu->insertItem( i18n("Move..."), this,
408 SLOT (moveTodo())); 409 SLOT (moveTodo()));
409 mItemPopupMenu->insertItem( i18n("Beam..."), this, 410 mItemPopupMenu->insertItem( i18n("Beam..."), this,
410 SLOT (beamTodo())); 411 SLOT (beamTodo()));
411 mItemPopupMenu->insertItem( i18n("Toggle Cancel"), this, 412 mItemPopupMenu->insertItem( i18n("Toggle Cancel"), this,
412 SLOT (cancelTodo())); 413 SLOT (cancelTodo()));
413 mItemPopupMenu->insertSeparator(); 414 mItemPopupMenu->insertSeparator();
414 415
415 mItemPopupMenu->insertItem( i18n("New Todo..."), this, 416 mItemPopupMenu->insertItem( i18n("New Todo..."), this,
416 SLOT (newTodo())); 417 SLOT (newTodo()));
417 mItemPopupMenu->insertItem(i18n("New Sub-Todo..."), this, 418 mItemPopupMenu->insertItem(i18n("New Sub-Todo..."), this,
418 SLOT (newSubTodo())); 419 SLOT (newSubTodo()));
419 mItemPopupMenu->insertItem(i18n("Unparent Todo"), this, 420 mItemPopupMenu->insertItem(i18n("Unparent Todo"), this,
420 SLOT (unparentTodo()),0,21); 421 SLOT (unparentTodo()),0,21);
421 mItemPopupMenu->insertItem(i18n("Reparent Todo"), this, 422 mItemPopupMenu->insertItem(i18n("Reparent Todo"), this,
422 SLOT (reparentTodo()),0,22); 423 SLOT (reparentTodo()),0,22);
423 mItemPopupMenu->insertSeparator(); 424 mItemPopupMenu->insertSeparator();
424 mItemPopupMenu->insertItem(i18n("Delete completed To-Dos","Purge Completed"), 425 mItemPopupMenu->insertItem(i18n("Delete completed To-Dos","Purge Completed"),
425 this, SLOT( purgeCompleted() ) ); 426 this, SLOT( purgeCompleted() ) );
426 mItemPopupMenu->insertItem(i18n("toggle completed To-Dos","Show Completed"), 427 mItemPopupMenu->insertItem(i18n("toggle completed To-Dos","Show Completed"),
427 this, SLOT( toggleCompleted() ),0, 33 ); 428 this, SLOT( toggleCompleted() ),0, 33 );
428 mItemPopupMenu->insertItem(i18n("toggle quick todo","Show Quick Todo"), 429 mItemPopupMenu->insertItem(i18n("toggle quick todo","Show Quick Todo"),
429 this, SLOT( toggleQuickTodo() ),0, 34 ); 430 this, SLOT( toggleQuickTodo() ),0, 34 );
430 431
431 mPopupMenu = new QPopupMenu(this); 432 mPopupMenu = new QPopupMenu(this);
432 mPopupMenu->insertItem(SmallIconSet("todo"), i18n("New Todo..."), this, 433 mPopupMenu->insertItem(SmallIconSet("todo"), i18n("New Todo..."), this,
433 SLOT (newTodo()),0,1); 434 SLOT (newTodo()),0,1);
434 mPopupMenu->insertItem(i18n("delete completed To-Dos","Purge Completed"), 435 mPopupMenu->insertItem(i18n("delete completed To-Dos","Purge Completed"),
435 this, SLOT(purgeCompleted()),0,2); 436 this, SLOT(purgeCompleted()),0,2);
436 mPopupMenu->insertItem(i18n("Show Completed"), 437 mPopupMenu->insertItem(i18n("Show Completed"),
437 this, SLOT( toggleCompleted() ),0,3 ); 438 this, SLOT( toggleCompleted() ),0,3 );
438 mPopupMenu->insertItem(i18n("toggle quick todo","Show Quick Todo"), 439 mPopupMenu->insertItem(i18n("toggle quick todo","Show Quick Todo"),
439 this, SLOT( toggleQuickTodo() ),0,4 ); 440 this, SLOT( toggleQuickTodo() ),0,4 );
440 mDocPrefs = new DocPrefs( name ); 441 mDocPrefs = new DocPrefs( name );
441 442
442 mPopupMenu->setCheckable( true ); 443 mPopupMenu->setCheckable( true );
443 mItemPopupMenu->setCheckable( true ); 444 mItemPopupMenu->setCheckable( true );
444 // Double clicking conflicts with opening/closing the subtree 445 // Double clicking conflicts with opening/closing the subtree
445 connect( mTodoListView, SIGNAL( doubleClicked( QListViewItem *) ), 446 connect( mTodoListView, SIGNAL( doubleClicked( QListViewItem *) ),
446 SLOT( editItem( QListViewItem *) ) ); 447 SLOT( editItem( QListViewItem *) ) );
447 /* 448 /*
448 connect( mTodoListView, SIGNAL( rightButtonClicked ( QListViewItem *, 449 connect( mTodoListView, SIGNAL( rightButtonClicked ( QListViewItem *,
449 const QPoint &,int ) ), 450 const QPoint &,int ) ),
450 SLOT( popupMenu( QListViewItem *, const QPoint & ,int) ) ); 451 SLOT( popupMenu( QListViewItem *, const QPoint & ,int) ) );
451 */ 452 */
452 connect( mTodoListView, SIGNAL( contextRequest ( QListViewItem *, 453 connect( mTodoListView, SIGNAL( contextRequest ( QListViewItem *,
453 const QPoint &,int ) ), 454 const QPoint &,int ) ),
454 SLOT( popupMenu( QListViewItem *, const QPoint & ,int) ) ); 455 SLOT( popupMenu( QListViewItem *, const QPoint & ,int) ) );
455 connect( mTodoListView, SIGNAL( clicked( QListViewItem * ) ), 456 connect( mTodoListView, SIGNAL( clicked( QListViewItem * ) ),
456 SLOT( itemClicked( QListViewItem * ) ) ); 457 SLOT( itemClicked( QListViewItem * ) ) );
457 connect( mTodoListView, SIGNAL( double_Clicked( QListViewItem * ) ), 458 connect( mTodoListView, SIGNAL( double_Clicked( QListViewItem * ) ),
458 SLOT( itemDoubleClicked( QListViewItem * ) ) ); 459 SLOT( itemDoubleClicked( QListViewItem * ) ) );
459 connect( mTodoListView, SIGNAL( todoDropped( Todo * ) ), 460 connect( mTodoListView, SIGNAL( todoDropped( Todo * ) ),
460 SLOT( updateView() ) ); 461 SLOT( updateView() ) );
461 connect( mTodoListView, SIGNAL( expanded( QListViewItem * ) ), 462 connect( mTodoListView, SIGNAL( expanded( QListViewItem * ) ),
462 SLOT( itemStateChanged( QListViewItem * ) ) ); 463 SLOT( itemStateChanged( QListViewItem * ) ) );
463 connect( mTodoListView, SIGNAL( collapsed( QListViewItem * ) ), 464 connect( mTodoListView, SIGNAL( collapsed( QListViewItem * ) ),
464 SLOT( itemStateChanged( QListViewItem * ) ) ); 465 SLOT( itemStateChanged( QListViewItem * ) ) );
465 466
466#if 0 467#if 0
467 connect(mTodoListView,SIGNAL(selectionChanged(QListViewItem *)), 468 connect(mTodoListView,SIGNAL(selectionChanged(QListViewItem *)),
468 SLOT(selectionChanged(QListViewItem *))); 469 SLOT(selectionChanged(QListViewItem *)));
469 connect(mTodoListView,SIGNAL(clicked(QListViewItem *)), 470 connect(mTodoListView,SIGNAL(clicked(QListViewItem *)),
470 SLOT(selectionChanged(QListViewItem *))); 471 SLOT(selectionChanged(QListViewItem *)));
471 connect(mTodoListView,SIGNAL(pressed(QListViewItem *)), 472 connect(mTodoListView,SIGNAL(pressed(QListViewItem *)),
472 SLOT(selectionChanged(QListViewItem *))); 473 SLOT(selectionChanged(QListViewItem *)));
473#endif 474#endif
474 connect( mTodoListView, SIGNAL(selectionChanged() ), 475 connect( mTodoListView, SIGNAL(selectionChanged() ),
475 SLOT( processSelectionChange() ) ); 476 SLOT( processSelectionChange() ) );
476 connect( mQuickAdd, SIGNAL( returnPressed () ), 477 connect( mQuickAdd, SIGNAL( returnPressed () ),
477 SLOT( addQuickTodo() ) ); 478 SLOT( addQuickTodo() ) );
478// if ( QApplication::desktop()->width() < 480 ) { 479// if ( QApplication::desktop()->width() < 480 ) {
479// setNarrow(); 480// setNarrow();
480 // mTodoListView->setColumnWidth( 0, 100 ); 481 // mTodoListView->setColumnWidth( 0, 100 );
481 482
482 // } 483 // }
483 484
484} 485}
485 486
486KOTodoView::~KOTodoView() 487KOTodoView::~KOTodoView()
487{ 488{
488 delete mDocPrefs; 489 delete mDocPrefs;
489} 490}
490 491
491void KOTodoView::jumpToDate () 492void KOTodoView::jumpToDate ()
492{ 493{
493 // if (mActiveItem) { 494 // if (mActiveItem) {
494// mActiveItem->todo()); 495// mActiveItem->todo());
495// if ( mActiveItem->todo()->hasDueDate() ) 496// if ( mActiveItem->todo()->hasDueDate() )
496// emit mActiveItem->todo()jumpToTime( mTodo->dtDue().date() ); 497// emit mActiveItem->todo()jumpToTime( mTodo->dtDue().date() );
497} 498}
498 499
499void KOTodoView::setNarrow() 500void KOTodoView::setNarrow()
500{ 501{
501 //mTodoListView->setColumnWidth( 0, 120 ); 502 //mTodoListView->setColumnWidth( 0, 120 );
502 mTodoListView->setColumnWidth( 1, 35 ); 503 mTodoListView->setColumnWidth( 1, 35 );
503 mTodoListView->setColumnWidth( 2, 40 ); 504 mTodoListView->setColumnWidth( 2, 40 );
504 mTodoListView->setColumnWidth( 3, 80 ); 505 mTodoListView->setColumnWidth( 3, 80 );
505 mTodoListView->setColumnWidth( 4, 40 ); 506 mTodoListView->setColumnWidth( 4, 40 );
506 mTodoListView->setColumnWidth( 5, 90 ); 507 mTodoListView->setColumnWidth( 5, 90 );
507 508
508} 509}
509void KOTodoView::updateView() 510void KOTodoView::updateView()
510{ 511{
511 pendingSubtodo = 0; 512 pendingSubtodo = 0;
512 if ( mBlockUpdate ) { 513 if ( mBlockUpdate ) {
513 //qDebug("blocked "); 514 //qDebug("blocked ");
514 return; 515 return;
515 } 516 }
516 //qDebug("update "); 517 //qDebug("update ");
517// kdDebug() << "KOTodoView::updateView()" << endl; 518// kdDebug() << "KOTodoView::updateView()" << endl;
518 QFont fo = KOPrefs::instance()->mTodoViewFont; 519 QFont fo = KOPrefs::instance()->mTodoViewFont;
519 mTodoListView->clear(); 520 mTodoListView->clear();
520 if ( mName == "todolistsmall" ) { 521 if ( mName == "todolistsmall" ) {
521 if ( KOPrefs::instance()->mTodoViewUsesSmallFont ) { 522 if ( KOPrefs::instance()->mTodoViewUsesSmallFont ) {
522 int ps = fo.pointSize() -2; 523 int ps = fo.pointSize() -2;
523 if ( ps > 12 ) 524 if ( ps > 12 )
524 ps -= 2; 525 ps -= 2;
525 fo.setPointSize( ps ); 526 fo.setPointSize( ps );
526 } 527 }
527 } 528 }
528 529
529 mTodoListView->setFont( fo ); 530 mTodoListView->setFont( fo );
530 // QFontMetrics fm ( KOPrefs::instance()->mTodoViewFont ); 531 // QFontMetrics fm ( KOPrefs::instance()->mTodoViewFont );
531 //mTodoListView->header()->setMaximumHeight(fm.height()); 532 //mTodoListView->header()->setMaximumHeight(fm.height());
532 QPtrList<Todo> todoList = calendar()->todos(); 533 QPtrList<Todo> todoList = calendar()->todos();
533 534
534/* 535/*
535 kdDebug() << "KOTodoView::updateView(): Todo List:" << endl; 536 kdDebug() << "KOTodoView::updateView(): Todo List:" << endl;
536 Event *t; 537 Event *t;
537 for(t = todoList.first(); t; t = todoList.next()) { 538 for(t = todoList.first(); t; t = todoList.next()) {
538 kdDebug() << " " << t->getSummary() << endl; 539 kdDebug() << " " << t->getSummary() << endl;
539 540
540 if (t->getRelatedTo()) { 541 if (t->getRelatedTo()) {
541 kdDebug() << " (related to " << t->getRelatedTo()->getSummary() << ")" << endl; 542 kdDebug() << " (related to " << t->getRelatedTo()->getSummary() << ")" << endl;
542 } 543 }
543 544
544 QPtrList<Event> l = t->getRelations(); 545 QPtrList<Event> l = t->getRelations();
545 Event *c; 546 Event *c;
546 for(c=l.first();c;c=l.next()) { 547 for(c=l.first();c;c=l.next()) {
547 kdDebug() << " - relation: " << c->getSummary() << endl; 548 kdDebug() << " - relation: " << c->getSummary() << endl;
548 } 549 }
549 } 550 }
550*/ 551*/
551 552
552 // Put for each Event a KOTodoViewItem in the list view. Don't rely on a 553 // Put for each Event a KOTodoViewItem in the list view. Don't rely on a
553 // specific order of events. That means that we have to generate parent items 554 // specific order of events. That means that we have to generate parent items
554 // recursively for proper hierarchical display of Todos. 555 // recursively for proper hierarchical display of Todos.
555 mTodoMap.clear(); 556 mTodoMap.clear();
556 Todo *todo; 557 Todo *todo;
557 todo = todoList.first();// todo; todo = todoList.next()) { 558 todo = todoList.first();// todo; todo = todoList.next()) {
558 while ( todo ) { 559 while ( todo ) {
559 bool next = true; 560 bool next = true;
560 // qDebug("todo %s ", todo->summary().latin1()); 561 // qDebug("todo %s ", todo->summary().latin1());
561 Incidence *incidence = todo->relatedTo(); 562 Incidence *incidence = todo->relatedTo();
562 while ( incidence ) { 563 while ( incidence ) {
563 if ( incidence->type() == "Todo") { 564 if ( incidence->type() == "Todo") {
564 //qDebug("related %s ",incidence->summary().latin1() ); 565 //qDebug("related %s ",incidence->summary().latin1() );
565 if ( !(todoList.contains ( ((Todo* )incidence ) ) )) { 566 if ( !(todoList.contains ( ((Todo* )incidence ) ) )) {
566 //qDebug("related not found "); 567 //qDebug("related not found ");
567 todoList.remove( ); 568 todoList.remove( );
568 todo = todoList.current(); 569 todo = todoList.current();
569 next = false; 570 next = false;
570 incidence = 0; 571 incidence = 0;
571 572
572 } else { 573 } else {
573 //qDebug("related found "); 574 //qDebug("related found ");
574 incidence = incidence->relatedTo(); 575 incidence = incidence->relatedTo();
575 } 576 }
576 } else 577 } else
577 incidence = 0; 578 incidence = 0;
578 } 579 }
579 if ( next ) 580 if ( next )
580 todo = todoList.next(); 581 todo = todoList.next();
581 } 582 }
582// qDebug("again .... "); 583// qDebug("again .... ");
583// for(todo = todoList.first(); todo; todo = todoList.next()) { 584// for(todo = todoList.first(); todo; todo = todoList.next()) {
584 585
585// qDebug("yytodo %s ", todo->summary().latin1()); 586// qDebug("yytodo %s ", todo->summary().latin1());
586// } 587// }
587 //qDebug("for "); 588 //qDebug("for ");
588 for(todo = todoList.first(); todo; todo = todoList.next()) { 589 for(todo = todoList.first(); todo; todo = todoList.next()) {
589 if (!mTodoMap.contains(todo) && ( KOPrefs::instance()->mShowCompletedTodo || !todo->isCompleted() ) ) 590 if (!mTodoMap.contains(todo) && ( KOPrefs::instance()->mShowCompletedTodo || !todo->isCompleted() ) )
590 { 591 {
591 insertTodoItem(todo); 592 insertTodoItem(todo);
592 } 593 }
593 } 594 }
594 //qDebug("for end "); 595 //qDebug("for end ");
595 // Restore opened/closed state 596 // Restore opened/closed state
596 mTodoListView->blockSignals( true ); 597 mTodoListView->blockSignals( true );
597 if( mDocPrefs ) restoreItemState( mTodoListView->firstChild() ); 598 if( mDocPrefs ) restoreItemState( mTodoListView->firstChild() );
598 mTodoListView->blockSignals( false ); 599 mTodoListView->blockSignals( false );
599 mTodoListView->setFocus(); 600 mTodoListView->setFocus();
600 processSelectionChange(); 601 processSelectionChange();
601} 602}
602 603
603void KOTodoView::restoreItemState( QListViewItem *item ) 604void KOTodoView::restoreItemState( QListViewItem *item )
604{ 605{
605 pendingSubtodo = 0; 606 pendingSubtodo = 0;
606 while( item ) { 607 while( item ) {
607 KOTodoViewItem *todoItem = (KOTodoViewItem *)item; 608 KOTodoViewItem *todoItem = (KOTodoViewItem *)item;
608 todoItem->setOpen( mDocPrefs->readBoolEntry( todoItem->todo()->uid() ) ); 609 todoItem->setOpen( mDocPrefs->readBoolEntry( todoItem->todo()->uid() ) );
609 if( item->childCount() > 0 ) restoreItemState( item->firstChild() ); 610 if( item->childCount() > 0 ) restoreItemState( item->firstChild() );
610 item = item->nextSibling(); 611 item = item->nextSibling();
611 } 612 }
612} 613}
613 614
614 615
615QMap<Todo *,KOTodoViewItem *>::ConstIterator 616QMap<Todo *,KOTodoViewItem *>::ConstIterator
616 KOTodoView::insertTodoItem(Todo *todo) 617 KOTodoView::insertTodoItem(Todo *todo)
617{ 618{
618// kdDebug() << "KOTodoView::insertTodoItem(): " << todo->getSummary() << endl; 619// kdDebug() << "KOTodoView::insertTodoItem(): " << todo->getSummary() << endl;
619 // TODO: Check, if dynmaic cast is necessary 620 // TODO: Check, if dynmaic cast is necessary
620 621
621 pendingSubtodo = 0; 622 pendingSubtodo = 0;
622 Incidence *incidence = todo->relatedTo(); 623 Incidence *incidence = todo->relatedTo();
623 if (incidence && incidence->type() == "Todo") { 624 if (incidence && incidence->type() == "Todo") {
624 Todo *relatedTodo = static_cast<Todo *>(incidence); 625 Todo *relatedTodo = static_cast<Todo *>(incidence);
625 626
626// kdDebug() << " has Related" << endl; 627// kdDebug() << " has Related" << endl;
627 QMap<Todo *,KOTodoViewItem *>::ConstIterator itemIterator; 628 QMap<Todo *,KOTodoViewItem *>::ConstIterator itemIterator;
628 itemIterator = mTodoMap.find(relatedTodo); 629 itemIterator = mTodoMap.find(relatedTodo);
629 if (itemIterator == mTodoMap.end()) { 630 if (itemIterator == mTodoMap.end()) {
630// kdDebug() << " related not yet in list" << endl; 631// kdDebug() << " related not yet in list" << endl;
631 itemIterator = insertTodoItem (relatedTodo); 632 itemIterator = insertTodoItem (relatedTodo);
632 } 633 }
633 // isn't this pretty stupid? We give one Todo to the KOTodoViewItem 634 // isn't this pretty stupid? We give one Todo to the KOTodoViewItem
634 // and one into the map. Sure finding is more easy but why? -zecke 635 // and one into the map. Sure finding is more easy but why? -zecke
635 KOTodoViewItem *todoItem = new KOTodoViewItem(*itemIterator,todo,this); 636 KOTodoViewItem *todoItem = new KOTodoViewItem(*itemIterator,todo,this);
636 return mTodoMap.insert(todo,todoItem); 637 return mTodoMap.insert(todo,todoItem);
637 } else { 638 } else {
638// kdDebug() << " no Related" << endl; 639// kdDebug() << " no Related" << endl;
639 // see above -zecke 640 // see above -zecke
640 KOTodoViewItem *todoItem = new KOTodoViewItem(mTodoListView,todo,this); 641 KOTodoViewItem *todoItem = new KOTodoViewItem(mTodoListView,todo,this);
641 return mTodoMap.insert(todo,todoItem); 642 return mTodoMap.insert(todo,todoItem);
642 } 643 }
643} 644}
644 645
645 646
646void KOTodoView::updateConfig() 647void KOTodoView::updateConfig()
647{ 648{
648 updateView(); 649 updateView();
649 mTodoListView->repaintContents(); 650 mTodoListView->repaintContents();
650} 651}
651 652
652QPtrList<Incidence> KOTodoView::selectedIncidences() 653QPtrList<Incidence> KOTodoView::selectedIncidences()
653{ 654{
654 QPtrList<Incidence> selected; 655 QPtrList<Incidence> selected;
655 656
656 KOTodoViewItem *item = (KOTodoViewItem *)(mTodoListView->selectedItem()); 657 KOTodoViewItem *item = (KOTodoViewItem *)(mTodoListView->selectedItem());
657// if (!item) item = mActiveItem; 658// if (!item) item = mActiveItem;
658 if (item) selected.append(item->todo()); 659 if (item) selected.append(item->todo());
659 660
660 return selected; 661 return selected;
661} 662}
662 663
663QPtrList<Todo> KOTodoView::selectedTodos() 664QPtrList<Todo> KOTodoView::selectedTodos()
664{ 665{
665 QPtrList<Todo> selected; 666 QPtrList<Todo> selected;
666 667
667 KOTodoViewItem *item = (KOTodoViewItem *)(mTodoListView->selectedItem()); 668 KOTodoViewItem *item = (KOTodoViewItem *)(mTodoListView->selectedItem());
668// if (!item) item = mActiveItem; 669// if (!item) item = mActiveItem;
669 if (item) selected.append(item->todo()); 670 if (item) selected.append(item->todo());
670 671
671 return selected; 672 return selected;
672} 673}
673 674
674void KOTodoView::changeEventDisplay(Event *, int) 675void KOTodoView::changeEventDisplay(Event *, int)
675{ 676{
676 updateView(); 677 updateView();
677} 678}
678 679
679void KOTodoView::showDates(const QDate &, const QDate &) 680void KOTodoView::showDates(const QDate &, const QDate &)
680{ 681{
681} 682}
682 683
683void KOTodoView::showEvents(QPtrList<Event>) 684void KOTodoView::showEvents(QPtrList<Event>)
684{ 685{
685 kdDebug() << "KOTodoView::selectEvents(): not yet implemented" << endl; 686 kdDebug() << "KOTodoView::selectEvents(): not yet implemented" << endl;
686} 687}
687 688
688void KOTodoView::printPreview(CalPrinter *calPrinter, const QDate &fd, 689void KOTodoView::printPreview(CalPrinter *calPrinter, const QDate &fd,
689 const QDate &td) 690 const QDate &td)
690{ 691{
691#ifndef KORG_NOPRINTER 692#ifndef KORG_NOPRINTER
692 calPrinter->preview(CalPrinter::Todolist, fd, td); 693 calPrinter->preview(CalPrinter::Todolist, fd, td);
693#endif 694#endif
694} 695}
695 696
696void KOTodoView::editItem(QListViewItem *item ) 697void KOTodoView::editItem(QListViewItem *item )
697{ 698{
698 // qDebug("editItem(QListViewItem *item ) "); 699 // qDebug("editItem(QListViewItem *item ) ");
699 emit editTodoSignal(((KOTodoViewItem *)item)->todo()); 700 emit editTodoSignal(((KOTodoViewItem *)item)->todo());
700} 701}
701 702
702void KOTodoView::showItem(QListViewItem *item,const QPoint &,int) 703void KOTodoView::showItem(QListViewItem *item,const QPoint &,int)
703{ 704{
704 emit showTodoSignal(((KOTodoViewItem *)item)->todo()); 705 emit showTodoSignal(((KOTodoViewItem *)item)->todo());
705} 706}
706 707
707void KOTodoView::popupMenu(QListViewItem *item,const QPoint &,int column) 708void KOTodoView::popupMenu(QListViewItem *item,const QPoint &,int column)
708{ 709{
709 pendingSubtodo = 0; 710 pendingSubtodo = 0;
710 mActiveItem = (KOTodoViewItem *)item; 711 mActiveItem = (KOTodoViewItem *)item;
711 if (item) { 712 if (item) {
712 switch (column){ 713 switch (column){
713 case 1: 714 case 1:
714 mPriorityPopupMenu->popup(QCursor::pos ()); break; 715 mPriorityPopupMenu->popup(QCursor::pos ()); break;
715 case 2: 716 case 2:
716 mPercentageCompletedPopupMenu->popup(QCursor::pos ()); break; 717 mPercentageCompletedPopupMenu->popup(QCursor::pos ()); break;
717 case 3: 718 case 3:
718 moveTodo(); 719 moveTodo();
719 break; 720 break;
720 case 6: 721 case 6:
721 getCategoryPopupMenu((KOTodoViewItem *)item)->popup(QCursor::pos ()); break; 722 getCategoryPopupMenu((KOTodoViewItem *)item)->popup(QCursor::pos ()); break;
722 default: 723 default:
723 mItemPopupMenu->popup(QCursor::pos()); 724 mItemPopupMenu->popup(QCursor::pos());
724 } 725 }
725 } else mPopupMenu->popup(QCursor::pos()); 726 } else mPopupMenu->popup(QCursor::pos());
726} 727}
727void KOTodoView::newTodo() 728void KOTodoView::newTodo()
728{ 729{
729 emit newTodoSignal(); 730 emit newTodoSignal();
730} 731}
731 732
732void KOTodoView::newSubTodo() 733void KOTodoView::newSubTodo()
733{ 734{
734 if (mActiveItem) { 735 if (mActiveItem) {
735 emit newSubTodoSignal(mActiveItem->todo()); 736 emit newSubTodoSignal(mActiveItem->todo());
736 } 737 }
737} 738}
738void KOTodoView::unparentTodo() 739void KOTodoView::unparentTodo()
739{ 740{
740 if (mActiveItem) { 741 if (mActiveItem) {
741 emit unparentTodoSignal(mActiveItem->todo()); 742 emit unparentTodoSignal(mActiveItem->todo());
742 } 743 }
743} 744}
744 745
745void KOTodoView::reparentTodo() 746void KOTodoView::reparentTodo()
746{ 747{
747 if (mActiveItem) { 748 if (mActiveItem) {
748 qDebug("KOTodoView::reparentTodo() "); 749 qDebug("KOTodoView::reparentTodo() ");
749 topLevelWidget()->setCaption(i18n("Click on new parent item")); 750 topLevelWidget()->setCaption(i18n("Click on new parent item"));
750 pendingSubtodo = mActiveItem; 751 pendingSubtodo = mActiveItem;
751 } 752 }
752} 753}
753void KOTodoView::editTodo() 754void KOTodoView::editTodo()
754{ 755{
755 if (mActiveItem) { 756 if (mActiveItem) {
756 emit editTodoSignal(mActiveItem->todo()); 757 emit editTodoSignal(mActiveItem->todo());
757 } 758 }
758} 759}
759void KOTodoView::cloneTodo() 760void KOTodoView::cloneTodo()
760{ 761{
761 if (mActiveItem) { 762 if (mActiveItem) {
762 emit cloneTodoSignal((Incidence*)mActiveItem->todo()); 763 emit cloneTodoSignal((Incidence*)mActiveItem->todo());
763 } 764 }
764} 765}
765void KOTodoView::cancelTodo() 766void KOTodoView::cancelTodo()
766{ 767{
767 if (mActiveItem) { 768 if (mActiveItem) {
768 emit cancelTodoSignal((Incidence*)mActiveItem->todo()); 769 emit cancelTodoSignal((Incidence*)mActiveItem->todo());
769 } 770 }
770} 771}
771void KOTodoView::moveTodo() 772void KOTodoView::moveTodo()
772{ 773{
773 if (mActiveItem) { 774 if (mActiveItem) {
774 emit moveTodoSignal((Incidence*)mActiveItem->todo()); 775 emit moveTodoSignal((Incidence*)mActiveItem->todo());
775 } 776 }
776} 777}
777void KOTodoView::beamTodo() 778void KOTodoView::beamTodo()
778{ 779{
779 if (mActiveItem) { 780 if (mActiveItem) {
780 emit beamTodoSignal((Incidence*)mActiveItem->todo()); 781 emit beamTodoSignal((Incidence*)mActiveItem->todo());
781 } 782 }
782} 783}
783 784
784 785
785void KOTodoView::showTodo() 786void KOTodoView::showTodo()
786{ 787{
787 if (mActiveItem) { 788 if (mActiveItem) {
788 emit showTodoSignal(mActiveItem->todo()); 789 emit showTodoSignal(mActiveItem->todo());
789 } 790 }
790} 791}
791 792
792void KOTodoView::deleteTodo() 793void KOTodoView::deleteTodo()
793{ 794{
794 if (mActiveItem) { 795 if (mActiveItem) {
795 if (mActiveItem->childCount()) { 796 if (mActiveItem->childCount()) {
796 KMessageBox::sorry(this,i18n("Cannot delete To-Do which has children."), 797 KMessageBox::sorry(this,i18n("Cannot delete To-Do which has children."),
797 i18n("Delete To-Do")); 798 i18n("Delete To-Do"));
798 } else { 799 } else {
799 emit deleteTodoSignal(mActiveItem->todo()); 800 emit deleteTodoSignal(mActiveItem->todo());
800 } 801 }
801 } 802 }
802} 803}
803 804
804void KOTodoView::setNewPriority(int index) 805void KOTodoView::setNewPriority(int index)
805{ 806{
806 if (mActiveItem && !mActiveItem->todo()->isReadOnly ()) { 807 if (mActiveItem && !mActiveItem->todo()->isReadOnly ()) {
807 mActiveItem->todo()->setPriority(mPriority[index]); 808 mActiveItem->todo()->setPriority(mPriority[index]);
808 mActiveItem->construct(); 809 mActiveItem->construct();
809 todoModified (mActiveItem->todo(), KOGlobals::PRIORITY_MODIFIED); 810 todoModified (mActiveItem->todo(), KOGlobals::PRIORITY_MODIFIED);
810 mActiveItem->todo()->setRevision( mActiveItem->todo()->revision()+1 ); 811 mActiveItem->todo()->setRevision( mActiveItem->todo()->revision()+1 );
811 } 812 }
812} 813}
813 814
814void KOTodoView::setNewPercentage(int index) 815void KOTodoView::setNewPercentage(int index)
815{ 816{
816 if (mActiveItem && !mActiveItem->todo()->isReadOnly ()) { 817 if (mActiveItem && !mActiveItem->todo()->isReadOnly ()) {
817 if (mPercentage[index] == 100) { 818 if (mPercentage[index] == 100) {
818 mActiveItem->todo()->setCompleted(QDateTime::currentDateTime()); 819 mActiveItem->todo()->setCompleted(QDateTime::currentDateTime());
819 } else { 820 } else {
820 mActiveItem->todo()->setCompleted(false); 821 mActiveItem->todo()->setCompleted(false);
821 } 822 }
822 mActiveItem->todo()->setPercentComplete(mPercentage[index]); 823 mActiveItem->todo()->setPercentComplete(mPercentage[index]);
823 mActiveItem->construct(); 824 mActiveItem->construct();
824 todoModified (mActiveItem->todo (), KOGlobals::COMPLETION_MODIFIED); 825 todoModified (mActiveItem->todo (), KOGlobals::COMPLETION_MODIFIED);
825 mActiveItem->todo()->setRevision( mActiveItem->todo()->revision()+1 ); 826 mActiveItem->todo()->setRevision( mActiveItem->todo()->revision()+1 );
826 } 827 }
827} 828}
828 829
829 830
830QPopupMenu * KOTodoView::getCategoryPopupMenu (KOTodoViewItem *todoItem) 831QPopupMenu * KOTodoView::getCategoryPopupMenu (KOTodoViewItem *todoItem)
831{ 832{
832 QPopupMenu* tempMenu = new QPopupMenu (this); 833 QPopupMenu* tempMenu = new QPopupMenu (this);
833 QStringList checkedCategories = todoItem->todo()->categories (); 834 QStringList checkedCategories = todoItem->todo()->categories ();
834 835
835 tempMenu->setCheckable (true); 836 tempMenu->setCheckable (true);
836 for (QStringList::Iterator it = KOPrefs::instance()->mCustomCategories.begin (); 837 for (QStringList::Iterator it = KOPrefs::instance()->mCustomCategories.begin ();
837 it != KOPrefs::instance()->mCustomCategories.end (); 838 it != KOPrefs::instance()->mCustomCategories.end ();
838 ++it) { 839 ++it) {
839 int index = tempMenu->insertItem (*it); 840 int index = tempMenu->insertItem (*it);
840 mCategory[index] = *it; 841 mCategory[index] = *it;
841 if (checkedCategories.find (*it) != checkedCategories.end ()) tempMenu->setItemChecked (index, true); 842 if (checkedCategories.find (*it) != checkedCategories.end ()) tempMenu->setItemChecked (index, true);
842 } 843 }
843 844
844 connect (tempMenu, SIGNAL (activated (int)), SLOT (changedCategories (int))); 845 connect (tempMenu, SIGNAL (activated (int)), SLOT (changedCategories (int)));
845 return tempMenu; 846 return tempMenu;
846 847
847 848
848} 849}
849void KOTodoView::changedCategories(int index) 850void KOTodoView::changedCategories(int index)
850{ 851{
851 if (mActiveItem && !mActiveItem->todo()->isReadOnly ()) { 852 if (mActiveItem && !mActiveItem->todo()->isReadOnly ()) {
852 QStringList categories = mActiveItem->todo()->categories (); 853 QStringList categories = mActiveItem->todo()->categories ();
853 if (categories.find (mCategory[index]) != categories.end ()) 854 if (categories.find (mCategory[index]) != categories.end ())
854 categories.remove (mCategory[index]); 855 categories.remove (mCategory[index]);
855 else 856 else
856 categories.insert (categories.end(), mCategory[index]); 857 categories.insert (categories.end(), mCategory[index]);
857 categories.sort (); 858 categories.sort ();
858 mActiveItem->todo()->setCategories (categories); 859 mActiveItem->todo()->setCategories (categories);
859 mActiveItem->construct(); 860 mActiveItem->construct();
860 mActiveItem->todo()->setRevision( mActiveItem->todo()->revision()+1 ); 861 mActiveItem->todo()->setRevision( mActiveItem->todo()->revision()+1 );
861 todoModified (mActiveItem->todo (), KOGlobals::CATEGORY_MODIFIED); 862 todoModified (mActiveItem->todo (), KOGlobals::CATEGORY_MODIFIED);
862 } 863 }
863} 864}
864void KOTodoView::itemDoubleClicked(QListViewItem *item) 865void KOTodoView::itemDoubleClicked(QListViewItem *item)
865{ 866{
866 if ( pendingSubtodo != 0 ) { 867 if ( pendingSubtodo != 0 ) {
867 topLevelWidget()->setCaption(i18n("Reparenting aborted!")); 868 topLevelWidget()->setCaption(i18n("Reparenting aborted!"));
868 } 869 }
869 pendingSubtodo = 0; 870 pendingSubtodo = 0;
870 if (!item) { 871 if (!item) {
871 newTodo(); 872 newTodo();
872 return; 873 return;
873 } 874 }
874 if ( KOPrefs::instance()->mEditOnDoubleClick ) 875 if ( KOPrefs::instance()->mEditOnDoubleClick )
875 editItem( item ); 876 editItem( item );
876 else 877 else
877 showItem( item , QPoint(), 0 ); 878 showItem( item , QPoint(), 0 );
878} 879}
879void KOTodoView::itemClicked(QListViewItem *item) 880void KOTodoView::itemClicked(QListViewItem *item)
880{ 881{
881 882
882 if (!item) { 883 if (!item) {
883 if ( pendingSubtodo != 0 ) { 884 if ( pendingSubtodo != 0 ) {
884 topLevelWidget()->setCaption(i18n("Reparenting aborted!")); 885 topLevelWidget()->setCaption(i18n("Reparenting aborted!"));
885 } 886 }
886 pendingSubtodo = 0; 887 pendingSubtodo = 0;
887 return; 888 return;
888 } 889 }
diff --git a/korganizer/kotodoviewitem.cpp b/korganizer/kotodoviewitem.cpp
index 85647b1..ae0b334 100644
--- a/korganizer/kotodoviewitem.cpp
+++ b/korganizer/kotodoviewitem.cpp
@@ -1,432 +1,432 @@
1/* 1/*
2 This file is part of KOrganizer. 2 This file is part of KOrganizer.
3 Copyright (c) 2000,2001 Cornelius Schumacher <schumacher@kde.org> 3 Copyright (c) 2000,2001 Cornelius Schumacher <schumacher@kde.org>
4 4
5 This program is free software; you can redistribute it and/or modify 5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by 6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2 of the License, or 7 the Free Software Foundation; either version 2 of the License, or
8 (at your option) any later version. 8 (at your option) any later version.
9 9
10 This program is distributed in the hope that it will be useful, 10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of 11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details. 13 GNU General Public License for more details.
14 14
15 You should have received a copy of the GNU General Public License 15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software 16 along with this program; if not, write to the Free Software
17 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. 17 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
18*/ 18*/
19 19
20#include <klocale.h> 20#include <klocale.h>
21#include <kdebug.h> 21#include <kdebug.h>
22#include <qapp.h> 22#include <qapp.h>
23 23
24#include <kiconloader.h> 24#include <kiconloader.h>
25#include "kotodoviewitem.h" 25#include "kotodoviewitem.h"
26#include "kotodoview.h" 26#include "kotodoview.h"
27#include "koprefs.h" 27#include "koprefs.h"
28 28
29KOTodoViewItem::KOTodoViewItem( QListView *parent, Todo *todo, KOTodoView *kotodo) 29KOTodoViewItem::KOTodoViewItem( QListView *parent, Todo *todo, KOTodoView *kotodo)
30 : QCheckListItem( parent , "", CheckBox ), mTodo( todo ), mTodoView( kotodo ) 30 : QCheckListItem( parent , "", CheckBox ), mTodo( todo ), mTodoView( kotodo )
31{ 31{
32 construct(); 32 construct();
33} 33}
34 34
35KOTodoViewItem::KOTodoViewItem( KOTodoViewItem *parent, Todo *todo, KOTodoView *kotodo ) 35KOTodoViewItem::KOTodoViewItem( KOTodoViewItem *parent, Todo *todo, KOTodoView *kotodo )
36 : QCheckListItem( parent, "", CheckBox ), mTodo( todo ), mTodoView( kotodo ) 36 : QCheckListItem( parent, "", CheckBox ), mTodo( todo ), mTodoView( kotodo )
37{ 37{
38 construct(); 38 construct();
39} 39}
40 40
41QString KOTodoViewItem::key(int column,bool) const 41QString KOTodoViewItem::key(int column,bool) const
42{ 42{
43 QMap<int,QString>::ConstIterator it = mKeyMap.find(column); 43 QMap<int,QString>::ConstIterator it = mKeyMap.find(column);
44 if (it == mKeyMap.end()) { 44 if (it == mKeyMap.end()) {
45 return text(column); 45 return text(column).lower();
46 } else { 46 } else {
47 return *it; 47 return *it;
48 } 48 }
49} 49}
50 50
51void KOTodoViewItem:: setup() 51void KOTodoViewItem:: setup()
52{ 52{
53 53
54 int h = 20; 54 int h = 20;
55 if ( listView () ) { 55 if ( listView () ) {
56 QFontMetrics fm ( listView ()->font () ); 56 QFontMetrics fm ( listView ()->font () );
57 h = fm.height(); 57 h = fm.height();
58 } 58 }
59 setHeight( h ); 59 setHeight( h );
60 60
61} 61}
62void KOTodoViewItem::setSortKey(int column,const QString &key) 62void KOTodoViewItem::setSortKey(int column,const QString &key)
63{ 63{
64 mKeyMap.insert(column,key); 64 mKeyMap.insert(column,key);
65} 65}
66 66
67#if QT_VERSION >= 300 67#if QT_VERSION >= 300
68void KOTodoViewItem::paintBranches(QPainter *p,const QColorGroup & cg,int w, 68void KOTodoViewItem::paintBranches(QPainter *p,const QColorGroup & cg,int w,
69 int y,int h) 69 int y,int h)
70{ 70{
71 QListViewItem::paintBranches(p,cg,w,y,h); 71 QListViewItem::paintBranches(p,cg,w,y,h);
72} 72}
73#else 73#else
74#endif 74#endif
75 75
76void KOTodoViewItem::construct() 76void KOTodoViewItem::construct()
77{ 77{
78 // qDebug("KOTodoViewItem::construct() "); 78 // qDebug("KOTodoViewItem::construct() ");
79 m_init = true; 79 m_init = true;
80 QString keyd = "=="; 80 QString keyd = "==";
81 QString keyt = "=="; 81 QString keyt = "==";
82 82
83 setOn(mTodo->isCompleted()); 83 setOn(mTodo->isCompleted());
84 setText(0,mTodo->summary()); 84 setText(0,mTodo->summary());
85 setText(1,QString::number(mTodo->priority())); 85 setText(1,QString::number(mTodo->priority()));
86 setText(2,i18n("%1 %").arg(QString::number(mTodo->percentComplete()))); 86 setText(2,i18n("%1 %").arg(QString::number(mTodo->percentComplete())));
87 if (mTodo->percentComplete()<100) { 87 if (mTodo->percentComplete()<100) {
88 if (mTodo->isCompleted()) setSortKey(2,QString::number(999)); 88 if (mTodo->isCompleted()) setSortKey(2,QString::number(999));
89 else setSortKey(2,QString::number(mTodo->percentComplete())); 89 else setSortKey(2,QString::number(mTodo->percentComplete()));
90 } 90 }
91 else { 91 else {
92 if (mTodo->isCompleted()) setSortKey(2,QString::number(999)); 92 if (mTodo->isCompleted()) setSortKey(2,QString::number(999));
93 else setSortKey(2,QString::number(99)); 93 else setSortKey(2,QString::number(99));
94 } 94 }
95 if (mTodo->hasDueDate()) { 95 if (mTodo->hasDueDate()) {
96 setText(3, mTodo->dtDueDateStr()); 96 setText(3, mTodo->dtDueDateStr());
97 QDate d = mTodo->dtDue().date(); 97 QDate d = mTodo->dtDue().date();
98 keyd.sprintf("%04d%02d%02d",d.year(),d.month(),d.day()); 98 keyd.sprintf("%04d%02d%02d",d.year(),d.month(),d.day());
99 setSortKey(3,keyd); 99 setSortKey(3,keyd);
100 if (mTodo->doesFloat()) { 100 if (mTodo->doesFloat()) {
101 setText(4,""); 101 setText(4,"");
102 } 102 }
103 else { 103 else {
104 setText(4,mTodo->dtDueTimeStr()); 104 setText(4,mTodo->dtDueTimeStr());
105 QTime t = mTodo->dtDue().time(); 105 QTime t = mTodo->dtDue().time();
106 keyt.sprintf("%02d%02d",t.hour(),t.minute()); 106 keyt.sprintf("%02d%02d",t.hour(),t.minute());
107 setSortKey(4,keyt); 107 setSortKey(4,keyt);
108 } 108 }
109 } else { 109 } else {
110 setText(3,""); 110 setText(3,"");
111 setText(4,""); 111 setText(4,"");
112 } 112 }
113 setSortKey(3,keyd); 113 setSortKey(3,keyd);
114 setSortKey(4,keyt); 114 setSortKey(4,keyt);
115 115
116 if (mTodo->isCompleted()) setSortKey(1,"6" + QString::number(mTodo->priority())+keyd+keyt); 116 if (mTodo->isCompleted()) setSortKey(1,"6" + QString::number(mTodo->priority())+keyd+keyt);
117 else setSortKey(1,QString::number(mTodo->priority())+keyd+keyt); 117 else setSortKey(1,QString::number(mTodo->priority())+keyd+keyt);
118 118
119 setText(5,mTodo->cancelled() ? i18n("Yes") : i18n("No")); 119 setText(5,mTodo->cancelled() ? i18n("Yes") : i18n("No"));
120 setText(6,mTodo->categoriesStr()); 120 setText(6,mTodo->categoriesStr());
121 121
122#if 0 122#if 0
123 // Find sort id in description. It's the text behind the last '#' character 123 // Find sort id in description. It's the text behind the last '#' character
124 // found in the description. White spaces are removed from beginning and end 124 // found in the description. White spaces are removed from beginning and end
125 // of sort id. 125 // of sort id.
126 int pos = mTodo->description().findRev('#'); 126 int pos = mTodo->description().findRev('#');
127 if (pos < 0) { 127 if (pos < 0) {
128 setText(6,""); 128 setText(6,"");
129 } else { 129 } else {
130 QString str = mTodo->description().mid(pos+1); 130 QString str = mTodo->description().mid(pos+1);
131 str.stripWhiteSpace(); 131 str.stripWhiteSpace();
132 setText(6,str); 132 setText(6,str);
133 } 133 }
134#endif 134#endif
135 135
136 m_known = false; 136 m_known = false;
137 m_init = false; 137 m_init = false;
138 138
139 setMyPixmap(); 139 setMyPixmap();
140 140
141} 141}
142void KOTodoViewItem::setMyPixmap() 142void KOTodoViewItem::setMyPixmap()
143{ 143{
144 int size = 5; 144 int size = 5;
145 QPixmap pixi = QPixmap( 1, 1 ); 145 QPixmap pixi = QPixmap( 1, 1 );
146 // if ( !mTodo->isCompleted() && mTodo->hasDueDate() && mTodo->dtDue() < QDateTime::currentDateTime() ) { 146 // if ( !mTodo->isCompleted() && mTodo->hasDueDate() && mTodo->dtDue() < QDateTime::currentDateTime() ) {
147// pixi = SmallIcon("redcross16"); 147// pixi = SmallIcon("redcross16");
148// } else { 148// } else {
149 QPainter p; 149 QPainter p;
150 150
151 int pixSize = 0; 151 int pixSize = 0;
152 QPixmap pPix = QPixmap( size, size ); 152 QPixmap pPix = QPixmap( size, size );
153 if ( mTodo->description().length() > 0 ) { 153 if ( mTodo->description().length() > 0 ) {
154 pixi.resize(size, pixSize+size); 154 pixi.resize(size, pixSize+size);
155 pPix.fill( Qt::darkGreen ); 155 pPix.fill( Qt::darkGreen );
156 p.begin( &pixi ); 156 p.begin( &pixi );
157 p. drawPixmap ( 0, pixSize, pPix); 157 p. drawPixmap ( 0, pixSize, pPix);
158 p.end(); 158 p.end();
159 pixSize += size; 159 pixSize += size;
160 } 160 }
161 if ( mTodo->isAlarmEnabled() ) { 161 if ( mTodo->isAlarmEnabled() ) {
162 pixi.resize(size, pixSize+size); 162 pixi.resize(size, pixSize+size);
163 pPix.fill( Qt::red ); 163 pPix.fill( Qt::red );
164 p.begin( &pixi ); 164 p.begin( &pixi );
165 p. drawPixmap ( 0, pixSize, pPix); 165 p. drawPixmap ( 0, pixSize, pPix);
166 p.end(); 166 p.end();
167 pixSize += size; 167 pixSize += size;
168 } 168 }
169 // } 169 // }
170 if ( pixi.width() > 1 ) { 170 if ( pixi.width() > 1 ) {
171 setPixmap ( 0,pixi ) ; 171 setPixmap ( 0,pixi ) ;
172 } 172 }
173 173
174 174
175} 175}
176void KOTodoViewItem::stateChange(bool state) 176void KOTodoViewItem::stateChange(bool state)
177{ 177{
178 // qDebug("KOTodoViewItem::stateChange "); 178 // qDebug("KOTodoViewItem::stateChange ");
179 // do not change setting on startup 179 // do not change setting on startup
180 if ( m_init ) return; 180 if ( m_init ) return;
181 181
182 kdDebug() << "State changed, modified " << state << endl; 182 kdDebug() << "State changed, modified " << state << endl;
183 QString keyd = "=="; 183 QString keyd = "==";
184 QString keyt = "=="; 184 QString keyt = "==";
185 185
186 if (state) mTodo->setCompleted(state); 186 if (state) mTodo->setCompleted(state);
187 else mTodo->setPercentComplete(0); 187 else mTodo->setPercentComplete(0);
188 if (isOn()!=state) { 188 if (isOn()!=state) {
189 setOn(state); 189 setOn(state);
190 } 190 }
191 191
192 if (mTodo->hasDueDate()) { 192 if (mTodo->hasDueDate()) {
193 setText(3, mTodo->dtDueDateStr()); 193 setText(3, mTodo->dtDueDateStr());
194 QDate d = mTodo->dtDue().date(); 194 QDate d = mTodo->dtDue().date();
195 keyd.sprintf("%04d%02d%02d",d.year(),d.month(),d.day()); 195 keyd.sprintf("%04d%02d%02d",d.year(),d.month(),d.day());
196 setSortKey(3,keyd); 196 setSortKey(3,keyd);
197 if (mTodo->doesFloat()) { 197 if (mTodo->doesFloat()) {
198 setText(4,""); 198 setText(4,"");
199 } 199 }
200 else { 200 else {
201 setText(4,mTodo->dtDueTimeStr()); 201 setText(4,mTodo->dtDueTimeStr());
202 QTime t = mTodo->dtDue().time(); 202 QTime t = mTodo->dtDue().time();
203 keyt.sprintf("%02d%02d",t.hour(),t.minute()); 203 keyt.sprintf("%02d%02d",t.hour(),t.minute());
204 setSortKey(4,keyt); 204 setSortKey(4,keyt);
205 } 205 }
206 } 206 }
207 if (mTodo->isCompleted()) setSortKey(1,QString::number(9)+keyd+keyt); 207 if (mTodo->isCompleted()) setSortKey(1,QString::number(9)+keyd+keyt);
208 else setSortKey(1,QString::number(mTodo->priority())+keyd+keyt); 208 else setSortKey(1,QString::number(mTodo->priority())+keyd+keyt);
209 209
210 setText(2,i18n("%1 %").arg(QString::number(mTodo->percentComplete()))); 210 setText(2,i18n("%1 %").arg(QString::number(mTodo->percentComplete())));
211 if (mTodo->percentComplete()<100) { 211 if (mTodo->percentComplete()<100) {
212 if (mTodo->isCompleted()) setSortKey(2,QString::number(999)); 212 if (mTodo->isCompleted()) setSortKey(2,QString::number(999));
213 else setSortKey(2,QString::number(mTodo->percentComplete())); 213 else setSortKey(2,QString::number(mTodo->percentComplete()));
214 } 214 }
215 else { 215 else {
216 if (mTodo->isCompleted()) setSortKey(2,QString::number(999)); 216 if (mTodo->isCompleted()) setSortKey(2,QString::number(999));
217 else setSortKey(2,QString::number(99)); 217 else setSortKey(2,QString::number(99));
218 } 218 }
219 QListViewItem * myChild = firstChild(); 219 QListViewItem * myChild = firstChild();
220 KOTodoViewItem *item; 220 KOTodoViewItem *item;
221 while( myChild ) { 221 while( myChild ) {
222 item = static_cast<KOTodoViewItem*>(myChild); 222 item = static_cast<KOTodoViewItem*>(myChild);
223 item->stateChange(state); 223 item->stateChange(state);
224 myChild = myChild->nextSibling(); 224 myChild = myChild->nextSibling();
225 } 225 }
226 mTodoView->modified(true); 226 mTodoView->modified(true);
227 mTodoView->setTodoModified( mTodo ); 227 mTodoView->setTodoModified( mTodo );
228 setMyPixmap(); 228 setMyPixmap();
229} 229}
230 230
231bool KOTodoViewItem::isAlternate() 231bool KOTodoViewItem::isAlternate()
232{ 232{
233#ifndef KORG_NOLVALTERNATION 233#ifndef KORG_NOLVALTERNATION
234 KOTodoListView *lv = static_cast<KOTodoListView *>(listView()); 234 KOTodoListView *lv = static_cast<KOTodoListView *>(listView());
235 if (lv && lv->alternateBackground().isValid()) 235 if (lv && lv->alternateBackground().isValid())
236 { 236 {
237 KOTodoViewItem *above = 0; 237 KOTodoViewItem *above = 0;
238 above = dynamic_cast<KOTodoViewItem *>(itemAbove()); 238 above = dynamic_cast<KOTodoViewItem *>(itemAbove());
239 m_known = above ? above->m_known : true; 239 m_known = above ? above->m_known : true;
240 if (m_known) 240 if (m_known)
241 { 241 {
242 m_odd = above ? !above->m_odd : false; 242 m_odd = above ? !above->m_odd : false;
243 } 243 }
244 else 244 else
245 { 245 {
246 KOTodoViewItem *item; 246 KOTodoViewItem *item;
247 bool previous = true; 247 bool previous = true;
248 if (QListViewItem::parent()) 248 if (QListViewItem::parent())
249 { 249 {
250 item = dynamic_cast<KOTodoViewItem *>(QListViewItem::parent()); 250 item = dynamic_cast<KOTodoViewItem *>(QListViewItem::parent());
251 if (item) 251 if (item)
252 previous = item->m_odd; 252 previous = item->m_odd;
253 item = dynamic_cast<KOTodoViewItem *>(QListViewItem::parent()->firstChild()); 253 item = dynamic_cast<KOTodoViewItem *>(QListViewItem::parent()->firstChild());
254 } 254 }
255 else 255 else
256 { 256 {
257 item = dynamic_cast<KOTodoViewItem *>(lv->firstChild()); 257 item = dynamic_cast<KOTodoViewItem *>(lv->firstChild());
258 } 258 }
259 259
260 while(item) 260 while(item)
261 { 261 {
262 item->m_odd = previous = !previous; 262 item->m_odd = previous = !previous;
263 item->m_known = true; 263 item->m_known = true;
264 item = dynamic_cast<KOTodoViewItem *>(item->nextSibling()); 264 item = dynamic_cast<KOTodoViewItem *>(item->nextSibling());
265 } 265 }
266 } 266 }
267 return m_odd; 267 return m_odd;
268 } 268 }
269 return false; 269 return false;
270#else 270#else
271 return false; 271 return false;
272#endif 272#endif
273} 273}
274 274
275void KOTodoViewItem::paintCell(QPainter *p, const QColorGroup &cg, int column, int width, int alignment) 275void KOTodoViewItem::paintCell(QPainter *p, const QColorGroup &cg, int column, int width, int alignment)
276{ 276{
277 QColorGroup _cg = cg; 277 QColorGroup _cg = cg;
278 QColorGroup::ColorRole role; 278 QColorGroup::ColorRole role;
279 if ( KOPrefs::instance()->mTodoViewUsesForegroundColor ) 279 if ( KOPrefs::instance()->mTodoViewUsesForegroundColor )
280 role = QColorGroup::Text; 280 role = QColorGroup::Text;
281 else 281 else
282 role = QColorGroup::Base; 282 role = QColorGroup::Base;
283 //#ifndef KORG_NOLVALTERNATION 283 //#ifndef KORG_NOLVALTERNATION
284 // if (isAlternate()) 284 // if (isAlternate())
285 // _cg.setColor(QColorGroup::Base, static_cast< KOTodoListView* >(listView())->alternateBackground()); 285 // _cg.setColor(QColorGroup::Base, static_cast< KOTodoListView* >(listView())->alternateBackground());
286 bool setColor = KOPrefs::instance()->mTodoViewUsesCatColors; 286 bool setColor = KOPrefs::instance()->mTodoViewUsesCatColors;
287 QColor colorToSet; 287 QColor colorToSet;
288 if ( setColor ) { 288 if ( setColor ) {
289 QStringList categories = mTodo->categories(); 289 QStringList categories = mTodo->categories();
290 QString cat = categories.first(); 290 QString cat = categories.first();
291 if ( !cat.isEmpty()) { 291 if ( !cat.isEmpty()) {
292 colorToSet = *(KOPrefs::instance()->categoryColor(cat) ); 292 colorToSet = *(KOPrefs::instance()->categoryColor(cat) );
293 } else 293 } else
294 setColor = false; 294 setColor = false;
295 } 295 }
296 if (mTodo->hasDueDate()) { 296 if (mTodo->hasDueDate()) {
297 if (mTodo->dtDue().date()==QDate::currentDate() && 297 if (mTodo->dtDue().date()==QDate::currentDate() &&
298 !mTodo->isCompleted()) { 298 !mTodo->isCompleted()) {
299 //_cg.setColor( role , KOPrefs::instance()->mTodoDueTodayColor); 299 //_cg.setColor( role , KOPrefs::instance()->mTodoDueTodayColor);
300 colorToSet = KOPrefs::instance()->mTodoDueTodayColor; 300 colorToSet = KOPrefs::instance()->mTodoDueTodayColor;
301 setColor = true; 301 setColor = true;
302 } 302 }
303 if (mTodo->dtDue().date() < QDate::currentDate() && 303 if (mTodo->dtDue().date() < QDate::currentDate() &&
304 !mTodo->isCompleted()) { 304 !mTodo->isCompleted()) {
305 //_cg.setColor( role, KOPrefs::instance()->mTodoOverdueColor); 305 //_cg.setColor( role, KOPrefs::instance()->mTodoOverdueColor);
306 colorToSet = KOPrefs::instance()->mTodoOverdueColor; 306 colorToSet = KOPrefs::instance()->mTodoOverdueColor;
307 setColor = true; 307 setColor = true;
308 } 308 }
309 } 309 }
310 310
311 if ( setColor ) { 311 if ( setColor ) {
312 _cg.setColor(role,colorToSet ); 312 _cg.setColor(role,colorToSet );
313 if ( role == QColorGroup::Base) { 313 if ( role == QColorGroup::Base) {
314 int rgb = colorToSet.red(); 314 int rgb = colorToSet.red();
315 rgb += colorToSet.blue()/2; 315 rgb += colorToSet.blue()/2;
316 rgb += colorToSet.green(); 316 rgb += colorToSet.green();
317 if ( rgb < 200 ) 317 if ( rgb < 200 )
318 _cg.setColor(QColorGroup::Text,Qt::white ); 318 _cg.setColor(QColorGroup::Text,Qt::white );
319 } 319 }
320 } 320 }
321 //#endif 321 //#endif
322 if ( column > 0 ){ 322 if ( column > 0 ){
323 if ( column == 2 && !KOPrefs::instance()->mTodoViewShowsPercentage ) { 323 if ( column == 2 && !KOPrefs::instance()->mTodoViewShowsPercentage ) {
324 p->save(); 324 p->save();
325 int progress = (int)(( (width-6)*mTodo->percentComplete())/100.0 + 0.5); 325 int progress = (int)(( (width-6)*mTodo->percentComplete())/100.0 + 0.5);
326 326
327 p->fillRect( 0, 0, width, height(), _cg.base() ); // background 327 p->fillRect( 0, 0, width, height(), _cg.base() ); // background
328 // p->setPen(Qt::black ); //border 328 // p->setPen(Qt::black ); //border
329 // p->setBrush( KOPrefs::instance()->mHighlightColorKGlobalSettings::baseColor() ); //filling 329 // p->setBrush( KOPrefs::instance()->mHighlightColorKGlobalSettings::baseColor() ); //filling
330 QColor fc = KOPrefs::instance()->mHighlightColor; 330 QColor fc = KOPrefs::instance()->mHighlightColor;
331 if ( mTodo->percentComplete() == 100 ) 331 if ( mTodo->percentComplete() == 100 )
332 fc = darkGreen; 332 fc = darkGreen;
333 p->drawRect( 2, 2, width-4, height()-4); 333 p->drawRect( 2, 2, width-4, height()-4);
334 p->fillRect( 3, 3, progress, height()-6, 334 p->fillRect( 3, 3, progress, height()-6,
335 fc ); 335 fc );
336 p->restore(); 336 p->restore();
337 } else { 337 } else {
338 QCheckListItem::paintCell(p, _cg, column, width, alignment); 338 QCheckListItem::paintCell(p, _cg, column, width, alignment);
339 } 339 }
340 return; 340 return;
341 } 341 }
342 342
343 int align = alignment; 343 int align = alignment;
344 344
345 if ( !p ) 345 if ( !p )
346 return; 346 return;
347 347
348 p->fillRect( 0, 0, width, height(), _cg.brush( QColorGroup::Base ) ); 348 p->fillRect( 0, 0, width, height(), _cg.brush( QColorGroup::Base ) );
349 349
350 QListView *lv = listView(); 350 QListView *lv = listView();
351 if ( !lv ) 351 if ( !lv )
352 return; 352 return;
353 int marg = 2;//lv->itemMargin(); 353 int marg = 2;//lv->itemMargin();
354 int r = 0; 354 int r = 0;
355 QCheckListItem::Type myType = QCheckListItem::CheckBox; 355 QCheckListItem::Type myType = QCheckListItem::CheckBox;
356 int BoxSize = 20; 356 int BoxSize = 20;
357 int boxOffset = 2; 357 int boxOffset = 2;
358 int xOffset = 2; 358 int xOffset = 2;
359 if (qApp->desktop()->width() < 300 ) { 359 if (qApp->desktop()->width() < 300 ) {
360 BoxSize = 14; 360 BoxSize = 14;
361 boxOffset = -1; 361 boxOffset = -1;
362 xOffset = 1; 362 xOffset = 1;
363 // marg = 0; 363 // marg = 0;
364 } 364 }
365 if ( height() < BoxSize ) { 365 if ( height() < BoxSize ) {
366 boxOffset = boxOffset - ((BoxSize - height())/2) ; 366 boxOffset = boxOffset - ((BoxSize - height())/2) ;
367 // qDebug("boxOffset %d height %d", boxOffset, height() ); 367 // qDebug("boxOffset %d height %d", boxOffset, height() );
368 BoxSize = height(); 368 BoxSize = height();
369 369
370 } 370 }
371 //bool winStyle = lv->style() == WindowsStyle; 371 //bool winStyle = lv->style() == WindowsStyle;
372 372
373 int lineStart = 5; 373 int lineStart = 5;
374 if ( myType == Controller ) { 374 if ( myType == Controller ) {
375 if ( !pixmap( 0 ) ) 375 if ( !pixmap( 0 ) )
376 r += BoxSize + 4; 376 r += BoxSize + 4;
377 } else { 377 } else {
378 ASSERT( lv ); //### 378 ASSERT( lv ); //###
379 //QFontMetrics fm( lv->font() ); 379 //QFontMetrics fm( lv->font() );
380 //int d = fm.height(); 380 //int d = fm.height();
381 int x = 0; 381 int x = 0;
382 int y = (height() - BoxSize) / 2; 382 int y = (height() - BoxSize) / 2;
383 //p->setPen( QPen( _cg.text(), winStyle ? 2 : 1 ) ); 383 //p->setPen( QPen( _cg.text(), winStyle ? 2 : 1 ) );
384 if ( myType == CheckBox ) { 384 if ( myType == CheckBox ) {
385 if ( isEnabled() ) 385 if ( isEnabled() )
386 p->setPen( QPen( _cg.text(), 1 ) ); 386 p->setPen( QPen( _cg.text(), 1 ) );
387 else 387 else
388 p->setPen( QPen( listView()->palette().color( QPalette::Disabled, QColorGroup::Text ), 1 ) ); 388 p->setPen( QPen( listView()->palette().color( QPalette::Disabled, QColorGroup::Text ), 1 ) );
389 p->drawRect( x+marg, y+2, BoxSize-4, BoxSize-4 ); 389 p->drawRect( x+marg, y+2, BoxSize-4, BoxSize-4 );
390 lineStart = x+marg; 390 lineStart = x+marg;
391 ///////////////////// 391 /////////////////////
392 x++; 392 x++;
393 y++; 393 y++;
394 if ( isOn() ) { 394 if ( isOn() ) {
395 QPointArray a( 7*2 ); 395 QPointArray a( 7*2 );
396 int i, xx, yy; 396 int i, xx, yy;
397 xx = x+xOffset+marg+(boxOffset/2); 397 xx = x+xOffset+marg+(boxOffset/2);
398 yy = y+5+boxOffset; 398 yy = y+5+boxOffset;
399 for ( i=0; i<3; i++ ) { 399 for ( i=0; i<3; i++ ) {
400 a.setPoint( 2*i, xx, yy ); 400 a.setPoint( 2*i, xx, yy );
401 a.setPoint( 2*i+1, xx, yy+2 ); 401 a.setPoint( 2*i+1, xx, yy+2 );
402 // qDebug(" "); 402 // qDebug(" ");
403 xx++; yy++; 403 xx++; yy++;
404 } 404 }
405 yy -= 2; 405 yy -= 2;
406 for ( i=3; i<7; i++ ) { 406 for ( i=3; i<7; i++ ) {
407 a.setPoint( 2*i, xx, yy ); 407 a.setPoint( 2*i, xx, yy );
408 a.setPoint( 2*i+1, xx, yy+2 ); 408 a.setPoint( 2*i+1, xx, yy+2 );
409 xx++; yy--; 409 xx++; yy--;
410 } 410 }
411 p->setPen( darkGreen ); 411 p->setPen( darkGreen );
412 p->drawLineSegments( a ); 412 p->drawLineSegments( a );
413 } 413 }
414 //////////////////////// 414 ////////////////////////
415 } 415 }
416 r += BoxSize + 4; 416 r += BoxSize + 4;
417 } 417 }
418 418
419 p->translate( r, 0 ); 419 p->translate( r, 0 );
420 p->setPen( QPen( _cg.text() ) ); 420 p->setPen( QPen( _cg.text() ) );
421 QListViewItem::paintCell( p, _cg, column, width - r, align ); 421 QListViewItem::paintCell( p, _cg, column, width - r, align );
422 if ( mTodo->cancelled () ) { 422 if ( mTodo->cancelled () ) {
423 p->setPen( black ); 423 p->setPen( black );
424 QRect br = p->boundingRect( 1,1,1,1,0,mTodo->summary() ); 424 QRect br = p->boundingRect( 1,1,1,1,0,mTodo->summary() );
425 int wid = br.width() +lineStart; 425 int wid = br.width() +lineStart;
426 if ( wid > width-3 ) 426 if ( wid > width-3 )
427 wid = width-3; 427 wid = width-3;
428 p->drawLine( lineStart, height()/2+1, wid, height()/2+1 ); 428 p->drawLine( lineStart, height()/2+1, wid, height()/2+1 );
429 429
430 } 430 }
431 431
432} 432}
diff --git a/libkcal/sharpformat.cpp b/libkcal/sharpformat.cpp
index 24b8349..c2ee2c9 100644
--- a/libkcal/sharpformat.cpp
+++ b/libkcal/sharpformat.cpp
@@ -1,770 +1,773 @@
1/* 1/*
2 This file is part of libkcal. 2 This file is part of libkcal.
3 3
4 Copyright (c) 2003 Cornelius Schumacher <schumacher@kde.org> 4 Copyright (c) 2003 Cornelius Schumacher <schumacher@kde.org>
5 5
6 This library is free software; you can redistribute it and/or 6 This library is free software; you can redistribute it and/or
7 modify it under the terms of the GNU Library General Public 7 modify it under the terms of the GNU Library General Public
8 License as published by the Free Software Foundation; either 8 License as published by the Free Software Foundation; either
9 version 2 of the License, or (at your option) any later version. 9 version 2 of the License, or (at your option) any later version.
10 10
11 This library is distributed in the hope that it will be useful, 11 This library is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of 12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 Library General Public License for more details. 14 Library General Public License for more details.
15 15
16 You should have received a copy of the GNU Library General Public License 16 You should have received a copy of the GNU Library General Public License
17 along with this library; see the file COPYING.LIB. If not, write to 17 along with this library; see the file COPYING.LIB. If not, write to
18 the Free Software Foundation, Inc., 59 Temple Place - Suite 330, 18 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
19 Boston, MA 02111-1307, USA. 19 Boston, MA 02111-1307, USA.
20*/ 20*/
21 21
22#include <qdatetime.h> 22#include <qdatetime.h>
23#include <qstring.h> 23#include <qstring.h>
24#include <qapplication.h> 24#include <qapplication.h>
25#include <qptrlist.h> 25#include <qptrlist.h>
26#include <qregexp.h> 26#include <qregexp.h>
27#include <qmessagebox.h> 27#include <qmessagebox.h>
28#include <qclipboard.h> 28#include <qclipboard.h>
29#include <qfile.h> 29#include <qfile.h>
30#include <qtextstream.h> 30#include <qtextstream.h>
31#include <qtextcodec.h> 31#include <qtextcodec.h>
32#include <qxml.h> 32#include <qxml.h>
33#include <qlabel.h> 33#include <qlabel.h>
34 34
35#include <kdebug.h> 35#include <kdebug.h>
36#include <klocale.h> 36#include <klocale.h>
37#include <kglobal.h> 37#include <kglobal.h>
38 38
39#include "calendar.h" 39#include "calendar.h"
40#include "alarm.h" 40#include "alarm.h"
41#include "recurrence.h" 41#include "recurrence.h"
42#include "calendarlocal.h" 42#include "calendarlocal.h"
43 43
44#include "sharpformat.h" 44#include "sharpformat.h"
45#include "syncdefines.h" 45#include "syncdefines.h"
46 46
47using namespace KCal; 47using namespace KCal;
48 48
49//CARDID,CATEGORY,DSRP,PLCE,MEM1,TIM1,TIM2,ADAY,ARON,ARMN,ARSD,RTYP,RFRQ,RPOS,RDYS,REND,REDT,ALSD,ALED,MDAY 49//CARDID,CATEGORY,DSRP,PLCE,MEM1,TIM1,TIM2,ADAY,ARON,ARMN,ARSD,RTYP,RFRQ,RPOS,RDYS,REND,REDT,ALSD,ALED,MDAY
50// 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 50// 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
51 51
52//ARSD silentalarm = 0 52//ARSD silentalarm = 0
53// 11 RTYP 225 no /0 dialy/ 1 weekly/ 3 month by date/ 2 month by day(pos)/ yearly 53// 11 RTYP 225 no /0 dialy/ 1 weekly/ 3 month by date/ 2 month by day(pos)/ yearly
54// 12 RFRQ 54// 12 RFRQ
55// 13 RPOS pos = 4. monday in month 55// 13 RPOS pos = 4. monday in month
56// 14 RDYS days: 1 mon/ 2 tue .. 64 sun 56// 14 RDYS days: 1 mon/ 2 tue .. 64 sun
57// 15 REND 0 = no end/ 1 = end 57// 15 REND 0 = no end/ 1 = end
58// 16 REDT rec end dt 58// 16 REDT rec end dt
59//ALSD 59//ALSD
60//ALED 60//ALED
61//MDAY 61//MDAY
62 62
63class SharpParser : public QObject 63class SharpParser : public QObject
64{ 64{
65 public: 65 public:
66 SharpParser( Calendar *calendar ) : mCalendar( calendar ) { 66 SharpParser( Calendar *calendar ) : mCalendar( calendar ) {
67 } 67 }
68 68
69 bool startElement( Calendar *existingCalendar, const QStringList & attList, QString qName ) 69 bool startElement( Calendar *existingCalendar, const QStringList & attList, QString qName )
70 { 70 {
71 int i = 1; 71 int i = 1;
72 bool skip = true; 72 bool skip = true;
73 int max = attList.count() -2; 73 int max = attList.count() -2;
74 while ( i < max ) { 74 while ( i < max ) {
75 if ( !attList[i].isEmpty() ) { 75 if ( !attList[i].isEmpty() ) {
76 skip = false; 76 skip = false;
77 break; 77 break;
78 } 78 }
79 ++i ; 79 ++i ;
80 } 80 }
81 if ( skip ) 81 if ( skip )
82 return false; 82 return false;
83 ulong cSum = SharpFormat::getCsum(attList ); 83 ulong cSum = SharpFormat::getCsum(attList );
84 84
85 if ( qName == "Event" ) { 85 if ( qName == "Event" ) {
86 Event *event; 86 Event *event;
87 event = existingCalendar->event( "Sharp_DTM",attList[0] ); 87 event = existingCalendar->event( "Sharp_DTM",attList[0] );
88 if ( event ) 88 if ( event )
89 event = (Event*)event->clone(); 89 event = (Event*)event->clone();
90 else 90 else
91 event = new Event; 91 event = new Event;
92 event->setID("Sharp_DTM", attList[0] ); 92 event->setID("Sharp_DTM", attList[0] );
93 event->setCsum( "Sharp_DTM", QString::number( cSum )); 93 event->setCsum( "Sharp_DTM", QString::number( cSum ));
94 event->setTempSyncStat(SYNC_TEMPSTATE_NEW_EXTERNAL ); 94 event->setTempSyncStat(SYNC_TEMPSTATE_NEW_EXTERNAL );
95 95
96 event->setSummary( attList[2] ); 96 event->setSummary( attList[2] );
97 event->setLocation( attList[3] ); 97 event->setLocation( attList[3] );
98 event->setDescription( attList[4] ); 98 event->setDescription( attList[4] );
99 if ( attList[7] == "1" ) { 99 if ( attList[7] == "1" ) {
100 event->setDtStart( QDateTime(fromString( attList[17]+"T000000", false ).date(),QTime(0,0,0 ) )); 100 event->setDtStart( QDateTime(fromString( attList[17]+"T000000", false ).date(),QTime(0,0,0 ) ));
101 event->setDtEnd( QDateTime(fromString( attList[18]+"T000000", false ).date(),QTime(0,0,0 ))); 101 event->setDtEnd( QDateTime(fromString( attList[18]+"T000000", false ).date(),QTime(0,0,0 )));
102 event->setFloats( true ); 102 event->setFloats( true );
103 } else { 103 } else {
104 event->setFloats( false ); 104 event->setFloats( false );
105 event->setDtStart( fromString( attList[5] ) ); 105 event->setDtStart( fromString( attList[5] ) );
106 event->setDtEnd( fromString( attList[6] )); 106 event->setDtEnd( fromString( attList[6] ));
107 } 107 }
108 108
109 QString rtype = attList[11]; 109 QString rtype = attList[11];
110 if ( rtype != "255" ) { 110 if ( rtype != "255" ) {
111 // qDebug("recurs "); 111 // qDebug("recurs ");
112 QDate startDate = event->dtStart().date(); 112 QDate startDate = event->dtStart().date();
113 113
114 QString freqStr = attList[12]; 114 QString freqStr = attList[12];
115 int freq = freqStr.toInt(); 115 int freq = freqStr.toInt();
116 116
117 QString hasEndDateStr = attList[15] ; 117 QString hasEndDateStr = attList[15] ;
118 bool hasEndDate = hasEndDateStr == "1"; 118 bool hasEndDate = hasEndDateStr == "1";
119 119
120 QString endDateStr = attList[16]; 120 QString endDateStr = attList[16];
121 QDate endDate = fromString( endDateStr ).date(); 121 QDate endDate = fromString( endDateStr ).date();
122 122
123 QString weekDaysStr = attList[14]; 123 QString weekDaysStr = attList[14];
124 uint weekDaysNum = weekDaysStr.toInt(); 124 uint weekDaysNum = weekDaysStr.toInt();
125 125
126 QBitArray weekDays( 7 ); 126 QBitArray weekDays( 7 );
127 int i; 127 int i;
128 int bb = 1; 128 int bb = 1;
129 for( i = 1; i <= 7; ++i ) { 129 for( i = 1; i <= 7; ++i ) {
130 weekDays.setBit( i - 1, ( bb & weekDaysNum )); 130 weekDays.setBit( i - 1, ( bb & weekDaysNum ));
131 bb = 2 << (i-1); 131 bb = 2 << (i-1);
132 //qDebug(" %d bit %d ",i-1,weekDays.at(i-1) ); 132 //qDebug(" %d bit %d ",i-1,weekDays.at(i-1) );
133 } 133 }
134 // qDebug("next "); 134 // qDebug("next ");
135 QString posStr = attList[13]; 135 QString posStr = attList[13];
136 int pos = posStr.toInt(); 136 int pos = posStr.toInt();
137 Recurrence *r = event->recurrence(); 137 Recurrence *r = event->recurrence();
138 138
139 if ( rtype == "0" ) { 139 if ( rtype == "0" ) {
140 if ( hasEndDate ) r->setDaily( freq, endDate ); 140 if ( hasEndDate ) r->setDaily( freq, endDate );
141 else r->setDaily( freq, -1 ); 141 else r->setDaily( freq, -1 );
142 } else if ( rtype == "1" ) { 142 } else if ( rtype == "1" ) {
143 if ( hasEndDate ) r->setWeekly( freq, weekDays, endDate ); 143 if ( hasEndDate ) r->setWeekly( freq, weekDays, endDate );
144 else r->setWeekly( freq, weekDays, -1 ); 144 else r->setWeekly( freq, weekDays, -1 );
145 } else if ( rtype == "3" ) { 145 } else if ( rtype == "3" ) {
146 if ( hasEndDate ) 146 if ( hasEndDate )
147 r->setMonthly( Recurrence::rMonthlyDay, freq, endDate ); 147 r->setMonthly( Recurrence::rMonthlyDay, freq, endDate );
148 else 148 else
149 r->setMonthly( Recurrence::rMonthlyDay, freq, -1 ); 149 r->setMonthly( Recurrence::rMonthlyDay, freq, -1 );
150 r->addMonthlyDay( startDate.day() ); 150 r->addMonthlyDay( startDate.day() );
151 } else if ( rtype == "2" ) { 151 } else if ( rtype == "2" ) {
152 if ( hasEndDate ) 152 if ( hasEndDate )
153 r->setMonthly( Recurrence::rMonthlyPos, freq, endDate ); 153 r->setMonthly( Recurrence::rMonthlyPos, freq, endDate );
154 else 154 else
155 r->setMonthly( Recurrence::rMonthlyPos, freq, -1 ); 155 r->setMonthly( Recurrence::rMonthlyPos, freq, -1 );
156 QBitArray days( 7 ); 156 QBitArray days( 7 );
157 days.fill( false ); 157 days.fill( false );
158 days.setBit( startDate.dayOfWeek() - 1 ); 158 days.setBit( startDate.dayOfWeek() - 1 );
159 r->addMonthlyPos( pos, days ); 159 r->addMonthlyPos( pos, days );
160 } else if ( rtype == "4" ) { 160 } else if ( rtype == "4" ) {
161 if ( hasEndDate ) 161 if ( hasEndDate )
162 r->setYearly( Recurrence::rYearlyMonth, freq, endDate ); 162 r->setYearly( Recurrence::rYearlyMonth, freq, endDate );
163 else 163 else
164 r->setYearly( Recurrence::rYearlyMonth, freq, -1 ); 164 r->setYearly( Recurrence::rYearlyMonth, freq, -1 );
165 r->addYearlyNum( startDate.month() ); 165 r->addYearlyNum( startDate.month() );
166 } 166 }
167 } else { 167 } else {
168 event->recurrence()->unsetRecurs(); 168 event->recurrence()->unsetRecurs();
169 } 169 }
170 170
171 QString categoryList = attList[1] ; 171 QString categoryList = attList[1] ;
172 event->setCategories( categoryList ); 172 event->setCategories( categoryList );
173 173
174 // strange 0 semms to mean: alarm enabled 174 // strange 0 semms to mean: alarm enabled
175 if ( attList[8] == "0" ) { 175 if ( attList[8] == "0" ) {
176 Alarm *alarm; 176 Alarm *alarm;
177 if ( event->alarms().count() > 0 ) 177 if ( event->alarms().count() > 0 )
178 alarm = event->alarms().first(); 178 alarm = event->alarms().first();
179 else { 179 else {
180 alarm = new Alarm( event ); 180 alarm = new Alarm( event );
181 event->addAlarm( alarm ); 181 event->addAlarm( alarm );
182 alarm->setType( Alarm::Audio ); 182 alarm->setType( Alarm::Audio );
183 } 183 }
184 //alarm->setType( Alarm::Audio ); 184 //alarm->setType( Alarm::Audio );
185 alarm->setEnabled( true ); 185 alarm->setEnabled( true );
186 int alarmOffset = attList[9].toInt(); 186 int alarmOffset = attList[9].toInt();
187 alarm->setStartOffset( alarmOffset * -60 ); 187 alarm->setStartOffset( alarmOffset * -60 );
188 } else { 188 } else {
189 Alarm *alarm; 189 Alarm *alarm;
190 if ( event->alarms().count() > 0 ) { 190 if ( event->alarms().count() > 0 ) {
191 alarm = event->alarms().first(); 191 alarm = event->alarms().first();
192 alarm->setType( Alarm::Audio ); 192 alarm->setType( Alarm::Audio );
193 alarm->setStartOffset( -60*15 ); 193 alarm->setStartOffset( -60*15 );
194 alarm->setEnabled( false ); 194 alarm->setEnabled( false );
195 } 195 }
196 } 196 }
197 197
198 mCalendar->addEvent( event); 198 mCalendar->addEvent( event);
199 } else if ( qName == "Todo" ) { 199 } else if ( qName == "Todo" ) {
200 Todo *todo; 200 Todo *todo;
201 201
202 todo = existingCalendar->todo( "Sharp_DTM", attList[0] ); 202 todo = existingCalendar->todo( "Sharp_DTM", attList[0] );
203 if (todo ) 203 if (todo )
204 todo = (Todo*)todo->clone(); 204 todo = (Todo*)todo->clone();
205 else 205 else
206 todo = new Todo; 206 todo = new Todo;
207 207
208//CARDID,CATEGORY,ETDY,LTDY,FNDY,MARK,PRTY,TITL,MEM1 208//CARDID,CATEGORY,ETDY,LTDY,FNDY,MARK,PRTY,TITL,MEM1
209// 0 1 2 3 4 5 6 7 8 209// 0 1 2 3 4 5 6 7 8
210//1,,,,,1,4,Loch zumachen,"" 210//1,,,,,1,4,Loch zumachen,""
211//3,Privat,20040317T000000,20040318T000000,20040319T000000,0,5,Call bbb,"notes123 bbb gggg ""bb "" " 211//3,Privat,20040317T000000,20040318T000000,20040319T000000,0,5,Call bbb,"notes123 bbb gggg ""bb "" "
212//2,"Familie,Freunde,Holiday",20040318T000000,20040324T000000,20040317T000000,1,2,tod2,notes 212//2,"Familie,Freunde,Holiday",20040318T000000,20040324T000000,20040317T000000,1,2,tod2,notes
213 213
214 todo->setID( "Sharp_DTM", attList[0]); 214 todo->setID( "Sharp_DTM", attList[0]);
215 todo->setCsum( "Sharp_DTM", QString::number( cSum )); 215 todo->setCsum( "Sharp_DTM", QString::number( cSum ));
216 todo->setTempSyncStat( SYNC_TEMPSTATE_NEW_EXTERNAL ); 216 todo->setTempSyncStat( SYNC_TEMPSTATE_NEW_EXTERNAL );
217 217
218 todo->setSummary( attList[7] ); 218 todo->setSummary( attList[7] );
219 todo->setDescription( attList[8]); 219 todo->setDescription( attList[8]);
220 220
221 int priority = attList[6].toInt(); 221 int priority = attList[6].toInt();
222 if ( priority == 0 ) priority = 3; 222 if ( priority == 0 ) priority = 3;
223 todo->setPriority( priority ); 223 todo->setPriority( priority );
224 224
225 QString categoryList = attList[1]; 225 QString categoryList = attList[1];
226 todo->setCategories( categoryList ); 226 todo->setCategories( categoryList );
227 227
228 228
229 229
230 QString hasDateStr = attList[3]; // due 230 QString hasDateStr = attList[3]; // due
231 if ( !hasDateStr.isEmpty() ) { 231 if ( !hasDateStr.isEmpty() ) {
232 if ( hasDateStr.right(6) == "000000" ) { 232 if ( hasDateStr.right(6) == "000000" ) {
233 todo->setDtDue( QDateTime(fromString( hasDateStr, false ).date(), QTime(0,0,0 )) ); 233 todo->setDtDue( QDateTime(fromString( hasDateStr, false ).date(), QTime(0,0,0 )) );
234 todo->setFloats( true ); 234 todo->setFloats( true );
235 } 235 }
236 else { 236 else {
237 todo->setDtDue( fromString( hasDateStr ) ); 237 todo->setDtDue( fromString( hasDateStr ) );
238 todo->setFloats( false ); 238 todo->setFloats( false );
239 } 239 }
240 240
241 todo->setHasDueDate( true ); 241 todo->setHasDueDate( true );
242 } 242 }
243 hasDateStr = attList[2];//start 243 hasDateStr = attList[2];//start
244 if ( !hasDateStr.isEmpty() ) { 244 if ( !hasDateStr.isEmpty() ) {
245 245
246 todo->setDtStart( fromString( hasDateStr ) ); 246 todo->setDtStart( fromString( hasDateStr ) );
247 todo->setHasStartDate( true); 247 todo->setHasStartDate( true);
248 } else 248 } else
249 todo->setHasStartDate( false ); 249 todo->setHasStartDate( false );
250 hasDateStr = attList[4];//completed 250 hasDateStr = attList[4];//completed
251 if ( !hasDateStr.isEmpty() ) { 251 if ( !hasDateStr.isEmpty() ) {
252 todo->setCompleted(fromString( hasDateStr ) ); 252 todo->setCompleted(fromString( hasDateStr ) );
253 } 253 }
254 QString completedStr = attList[5]; 254 QString completedStr = attList[5];
255 if ( completedStr == "0" ) 255 if ( completedStr == "0" )
256 todo->setCompleted( true ); 256 todo->setCompleted( true );
257 else 257 else {
258 todo->setCompleted( false ); 258 // do not change percent complete
259 if ( todo->isCompleted() )
260 todo->setCompleted( false );
261 }
259 mCalendar->addTodo( todo ); 262 mCalendar->addTodo( todo );
260 263
261 } else if ( qName == "Category" ) { 264 } else if ( qName == "Category" ) {
262 /* 265 /*
263 QString id = attributes.value( "id" ); 266 QString id = attributes.value( "id" );
264 QString name = attributes.value( "name" ); 267 QString name = attributes.value( "name" );
265 setCategory( id, name ); 268 setCategory( id, name );
266 */ 269 */
267 } 270 }
268 //qDebug("end "); 271 //qDebug("end ");
269 return true; 272 return true;
270 } 273 }
271 274
272 275
273 QDateTime fromString ( QString s, bool useTz = true ) { 276 QDateTime fromString ( QString s, bool useTz = true ) {
274 QDateTime dt; 277 QDateTime dt;
275 int y,m,t,h,min,sec; 278 int y,m,t,h,min,sec;
276 y = s.mid(0,4).toInt(); 279 y = s.mid(0,4).toInt();
277 m = s.mid(4,2).toInt(); 280 m = s.mid(4,2).toInt();
278 t = s.mid(6,2).toInt(); 281 t = s.mid(6,2).toInt();
279 h = s.mid(9,2).toInt(); 282 h = s.mid(9,2).toInt();
280 min = s.mid(11,2).toInt(); 283 min = s.mid(11,2).toInt();
281 sec = s.mid(13,2).toInt(); 284 sec = s.mid(13,2).toInt();
282 dt = QDateTime(QDate(y,m,t), QTime(h,min,sec)); 285 dt = QDateTime(QDate(y,m,t), QTime(h,min,sec));
283 int offset = KGlobal::locale()->localTimeOffset( dt ); 286 int offset = KGlobal::locale()->localTimeOffset( dt );
284 if ( useTz ) 287 if ( useTz )
285 dt = dt.addSecs ( offset*60); 288 dt = dt.addSecs ( offset*60);
286 return dt; 289 return dt;
287 290
288 } 291 }
289 protected: 292 protected:
290 QDateTime toDateTime( const QString &value ) 293 QDateTime toDateTime( const QString &value )
291 { 294 {
292 QDateTime dt; 295 QDateTime dt;
293 dt.setTime_t( value.toUInt() ); 296 dt.setTime_t( value.toUInt() );
294 297
295 return dt; 298 return dt;
296 } 299 }
297 300
298 private: 301 private:
299 Calendar *mCalendar; 302 Calendar *mCalendar;
300}; 303};
301 304
302 305
303SharpFormat::SharpFormat() 306SharpFormat::SharpFormat()
304{ 307{
305 308
306} 309}
307 310
308SharpFormat::~SharpFormat() 311SharpFormat::~SharpFormat()
309{ 312{
310} 313}
311ulong SharpFormat::getCsum( const QStringList & attList) 314ulong SharpFormat::getCsum( const QStringList & attList)
312{ 315{
313 int max = attList.count() -1; 316 int max = attList.count() -1;
314 ulong cSum = 0; 317 ulong cSum = 0;
315 int j,k,i; 318 int j,k,i;
316 int add; 319 int add;
317 for ( i = 1; i < max ; ++i ) { 320 for ( i = 1; i < max ; ++i ) {
318 QString s = attList[i]; 321 QString s = attList[i];
319 if ( ! s.isEmpty() ){ 322 if ( ! s.isEmpty() ){
320 j = s.length(); 323 j = s.length();
321 for ( k = 0; k < j; ++k ) { 324 for ( k = 0; k < j; ++k ) {
322 int mul = k +1; 325 int mul = k +1;
323 add = s[k].unicode (); 326 add = s[k].unicode ();
324 if ( k < 16 ) 327 if ( k < 16 )
325 mul = mul * mul; 328 mul = mul * mul;
326 add = add * mul *i*i*i; 329 add = add * mul *i*i*i;
327 cSum += add; 330 cSum += add;
328 } 331 }
329 } 332 }
330 } 333 }
331 return cSum; 334 return cSum;
332 335
333} 336}
334#include <stdlib.h> 337#include <stdlib.h>
335#define DEBUGMODE false 338#define DEBUGMODE false
336//#define DEBUGMODE true 339//#define DEBUGMODE true
337bool SharpFormat::load( Calendar *calendar, Calendar *existngCal ) 340bool SharpFormat::load( Calendar *calendar, Calendar *existngCal )
338{ 341{
339 342
340 343
341 bool debug = DEBUGMODE; 344 bool debug = DEBUGMODE;
342 QString text; 345 QString text;
343 QString codec = "utf8"; 346 QString codec = "utf8";
344 QLabel status ( i18n("Reading events ..."), 0 ); 347 QLabel status ( i18n("Reading events ..."), 0 );
345 348
346 int w = status.sizeHint().width()+20 ; 349 int w = status.sizeHint().width()+20 ;
347 if ( w < 200 ) w = 200; 350 if ( w < 200 ) w = 200;
348 int h = status.sizeHint().height()+20 ; 351 int h = status.sizeHint().height()+20 ;
349 int dw = QApplication::desktop()->width(); 352 int dw = QApplication::desktop()->width();
350 int dh = QApplication::desktop()->height(); 353 int dh = QApplication::desktop()->height();
351 status.setCaption(i18n("Reading DTM Data") ); 354 status.setCaption(i18n("Reading DTM Data") );
352 status.setGeometry( (dw-w)/2, (dh - h )/2 ,w,h ); 355 status.setGeometry( (dw-w)/2, (dh - h )/2 ,w,h );
353 status.show(); 356 status.show();
354 status.raise(); 357 status.raise();
355 qApp->processEvents(); 358 qApp->processEvents();
356 QString fileName; 359 QString fileName;
357 if ( ! debug ) { 360 if ( ! debug ) {
358 fileName = "/tmp/kopitempout"; 361 fileName = "/tmp/kopitempout";
359 QString command ="db2file datebook -r -c "+ codec + " > " + fileName; 362 QString command ="db2file datebook -r -c "+ codec + " > " + fileName;
360 system ( command.latin1() ); 363 system ( command.latin1() );
361 } else { 364 } else {
362 fileName = "/tmp/events.txt"; 365 fileName = "/tmp/events.txt";
363 366
364 } 367 }
365 QFile file( fileName ); 368 QFile file( fileName );
366 if (!file.open( IO_ReadOnly ) ) { 369 if (!file.open( IO_ReadOnly ) ) {
367 return false; 370 return false;
368 371
369 } 372 }
370 QTextStream ts( &file ); 373 QTextStream ts( &file );
371 ts.setCodec( QTextCodec::codecForName("utf8") ); 374 ts.setCodec( QTextCodec::codecForName("utf8") );
372 text = ts.read(); 375 text = ts.read();
373 file.close(); 376 file.close();
374 status.setText( i18n("Processing events ...") ); 377 status.setText( i18n("Processing events ...") );
375 status.raise(); 378 status.raise();
376 qApp->processEvents(); 379 qApp->processEvents();
377 fromString2Cal( calendar, existngCal, text, "Event" ); 380 fromString2Cal( calendar, existngCal, text, "Event" );
378 status.setText( i18n("Reading todos ...") ); 381 status.setText( i18n("Reading todos ...") );
379 qApp->processEvents(); 382 qApp->processEvents();
380 if ( ! debug ) { 383 if ( ! debug ) {
381 fileName = "/tmp/kopitempout"; 384 fileName = "/tmp/kopitempout";
382 QString command = "db2file todo -r -c " + codec+ " > " + fileName; 385 QString command = "db2file todo -r -c " + codec+ " > " + fileName;
383 system ( command.latin1() ); 386 system ( command.latin1() );
384 } else { 387 } else {
385 fileName = "/tmp/todo.txt"; 388 fileName = "/tmp/todo.txt";
386 } 389 }
387 file.setName( fileName ); 390 file.setName( fileName );
388 if (!file.open( IO_ReadOnly ) ) { 391 if (!file.open( IO_ReadOnly ) ) {
389 return false; 392 return false;
390 393
391 } 394 }
392 ts.setDevice( &file ); 395 ts.setDevice( &file );
393 text = ts.read(); 396 text = ts.read();
394 file.close(); 397 file.close();
395 398
396 status.setText( i18n("Processing todos ...") ); 399 status.setText( i18n("Processing todos ...") );
397 status.raise(); 400 status.raise();
398 qApp->processEvents(); 401 qApp->processEvents();
399 fromString2Cal( calendar, existngCal, text, "Todo" ); 402 fromString2Cal( calendar, existngCal, text, "Todo" );
400 return true; 403 return true;
401} 404}
402int SharpFormat::getNumFromRecord( QString answer, Incidence* inc ) 405int SharpFormat::getNumFromRecord( QString answer, Incidence* inc )
403{ 406{
404 int retval = -1; 407 int retval = -1;
405 QStringList templist; 408 QStringList templist;
406 QString tempString; 409 QString tempString;
407 int start = 0; 410 int start = 0;
408 int len = answer.length(); 411 int len = answer.length();
409 int end = answer.find ("\n",start)+1; 412 int end = answer.find ("\n",start)+1;
410 bool ok = true; 413 bool ok = true;
411 start = end; 414 start = end;
412 int ccc = 0; 415 int ccc = 0;
413 while ( start > 0 ) { 416 while ( start > 0 ) {
414 templist.clear(); 417 templist.clear();
415 ok = true; 418 ok = true;
416 int loopCount = 0; 419 int loopCount = 0;
417 while ( ok ) { 420 while ( ok ) {
418 ++loopCount; 421 ++loopCount;
419 if ( loopCount > 25 ) { 422 if ( loopCount > 25 ) {
420 qDebug("KO: Error in while loop"); 423 qDebug("KO: Error in while loop");
421 ok = false; 424 ok = false;
422 start = 0; 425 start = 0;
423 break; 426 break;
424 } 427 }
425 if ( ok ) 428 if ( ok )
426 tempString = getPart( answer, ok, start ); 429 tempString = getPart( answer, ok, start );
427 if ( start >= len || start == 0 ) { 430 if ( start >= len || start == 0 ) {
428 start = 0; 431 start = 0;
429 ok = false; 432 ok = false;
430 } 433 }
431 if ( tempString.right(1) =="\n" ) 434 if ( tempString.right(1) =="\n" )
432 tempString = tempString.left( tempString.length()-1); 435 tempString = tempString.left( tempString.length()-1);
433 436
434 templist.append( tempString ); 437 templist.append( tempString );
435 } 438 }
436 ++ccc; 439 ++ccc;
437 if ( ccc == 2 && loopCount < 25 ) { 440 if ( ccc == 2 && loopCount < 25 ) {
438 start = 0; 441 start = 0;
439 bool ok; 442 bool ok;
440 int newnum = templist[0].toInt( &ok ); 443 int newnum = templist[0].toInt( &ok );
441 if ( ok && newnum > 0) { 444 if ( ok && newnum > 0) {
442 retval = newnum; 445 retval = newnum;
443 inc->setID( "Sharp_DTM",templist[0] ); 446 inc->setID( "Sharp_DTM",templist[0] );
444 inc->setCsum( "Sharp_DTM", QString::number( getCsum( templist ) )); 447 inc->setCsum( "Sharp_DTM", QString::number( getCsum( templist ) ));
445 inc->setTempSyncStat( SYNC_TEMPSTATE_NEW_ID ); 448 inc->setTempSyncStat( SYNC_TEMPSTATE_NEW_ID );
446 } 449 }
447 if ( ok && newnum == -1 ) { 450 if ( ok && newnum == -1 ) {
448 qDebug("Error writing back %s ", inc->summary().latin1()); 451 qDebug("Error writing back %s ", inc->summary().latin1());
449 } 452 }
450 } 453 }
451 } 454 }
452 //qDebug("getNumFromRecord returning : %d ", retval); 455 //qDebug("getNumFromRecord returning : %d ", retval);
453 return retval; 456 return retval;
454} 457}
455bool SharpFormat::save( Calendar *calendar) 458bool SharpFormat::save( Calendar *calendar)
456{ 459{
457 460
458 QLabel status ( i18n("Processing/adding events ..."), 0 ); 461 QLabel status ( i18n("Processing/adding events ..."), 0 );
459 int w = status.sizeHint().width()+20 ; 462 int w = status.sizeHint().width()+20 ;
460 if ( w < 200 ) w = 200; 463 if ( w < 200 ) w = 200;
461 int h = status.sizeHint().height()+20 ; 464 int h = status.sizeHint().height()+20 ;
462 int dw = QApplication::desktop()->width(); 465 int dw = QApplication::desktop()->width();
463 int dh = QApplication::desktop()->height(); 466 int dh = QApplication::desktop()->height();
464 status.setCaption(i18n("Writing DTM Data") ); 467 status.setCaption(i18n("Writing DTM Data") );
465 status.setGeometry( (dw-w)/2, (dh - h )/2 ,w,h ); 468 status.setGeometry( (dw-w)/2, (dh - h )/2 ,w,h );
466 status.show(); 469 status.show();
467 status.raise(); 470 status.raise();
468 qApp->processEvents(); 471 qApp->processEvents();
469 bool debug = DEBUGMODE; 472 bool debug = DEBUGMODE;
470 QString codec = "utf8"; 473 QString codec = "utf8";
471 QString answer; 474 QString answer;
472 QString ePrefix = "CARDID,CATEGORY,DSRP,PLCE,MEM1,TIM1,TIM2,ADAY,ARON,ARMN,ARSD,RTYP,RFRQ,RPOS,RDYS,REND,REDT,ALSD,ALED,MDAY\n"; 475 QString ePrefix = "CARDID,CATEGORY,DSRP,PLCE,MEM1,TIM1,TIM2,ADAY,ARON,ARMN,ARSD,RTYP,RFRQ,RPOS,RDYS,REND,REDT,ALSD,ALED,MDAY\n";
473 QString tPrefix = "CARDID,CATEGORY,ETDY,LTDY,FNDY,MARK,PRTY,TITL,MEM1\n"; 476 QString tPrefix = "CARDID,CATEGORY,ETDY,LTDY,FNDY,MARK,PRTY,TITL,MEM1\n";
474 QString command; 477 QString command;
475 QPtrList<Event> er = calendar->rawEvents(); 478 QPtrList<Event> er = calendar->rawEvents();
476 Event* ev = er.first(); 479 Event* ev = er.first();
477 QString fileName = "/tmp/kopitempout"; 480 QString fileName = "/tmp/kopitempout";
478 int i = 0; 481 int i = 0;
479 QString changeString = ePrefix; 482 QString changeString = ePrefix;
480 QString deleteString = ePrefix; 483 QString deleteString = ePrefix;
481 bool deleteEnt = false; 484 bool deleteEnt = false;
482 bool changeEnt = false; 485 bool changeEnt = false;
483 QString message = i18n("Processing event # "); 486 QString message = i18n("Processing event # ");
484 int procCount = 0; 487 int procCount = 0;
485 while ( ev ) { 488 while ( ev ) {
486 //qDebug("i %d ", ++i); 489 //qDebug("i %d ", ++i);
487 if ( ev->tempSyncStat() != SYNC_TEMPSTATE_NEW_EXTERNAL ) { 490 if ( ev->tempSyncStat() != SYNC_TEMPSTATE_NEW_EXTERNAL ) {
488 status.setText ( message + QString::number ( ++procCount ) ); 491 status.setText ( message + QString::number ( ++procCount ) );
489 qApp->processEvents(); 492 qApp->processEvents();
490 QString eString = getEventString( ev ); 493 QString eString = getEventString( ev );
491 if ( ev->tempSyncStat() == SYNC_TEMPSTATE_DELETE ) { // delete 494 if ( ev->tempSyncStat() == SYNC_TEMPSTATE_DELETE ) { // delete
492 // deleting empty strings does not work. 495 // deleting empty strings does not work.
493 // we write first and x and then delete the record with the x 496 // we write first and x and then delete the record with the x
494 eString = eString.replace( QRegExp(",\"\""),",\"x\"" ); 497 eString = eString.replace( QRegExp(",\"\""),",\"x\"" );
495 changeString += eString + "\n"; 498 changeString += eString + "\n";
496 deleteString += eString + "\n"; 499 deleteString += eString + "\n";
497 deleteEnt = true; 500 deleteEnt = true;
498 changeEnt = true; 501 changeEnt = true;
499 } 502 }
500 else if ( ev->getID("Sharp_DTM").isEmpty() ) { // add new 503 else if ( ev->getID("Sharp_DTM").isEmpty() ) { // add new
501 QString fileNameIn = "/tmp/kopitempin"; 504 QString fileNameIn = "/tmp/kopitempin";
502 QFile fileIn( fileNameIn ); 505 QFile fileIn( fileNameIn );
503 if (!fileIn.open( IO_WriteOnly ) ) { 506 if (!fileIn.open( IO_WriteOnly ) ) {
504 return false; 507 return false;
505 } 508 }
506 QTextStream tsIn( &fileIn ); 509 QTextStream tsIn( &fileIn );
507 tsIn.setCodec( QTextCodec::codecForName("utf8") ); 510 tsIn.setCodec( QTextCodec::codecForName("utf8") );
508 tsIn << ePrefix << eString ; 511 tsIn << ePrefix << eString ;
509 fileIn.close(); 512 fileIn.close();
510 //command = "(echo \"" + ePrefix + eString + "\" ) | db2file datebook -w -g -c " + codec+ " > "+ fileName; 513 //command = "(echo \"" + ePrefix + eString + "\" ) | db2file datebook -w -g -c " + codec+ " > "+ fileName;
511 command = "(cat /tmp/kopitempin | db2file datebook -w -g -c " + codec+ ") > "+ fileName; 514 command = "(cat /tmp/kopitempin | db2file datebook -w -g -c " + codec+ ") > "+ fileName;
512 //qDebug("command ++++++++ "); 515 //qDebug("command ++++++++ ");
513 //qDebug("%s ",command.latin1()); 516 //qDebug("%s ",command.latin1());
514 //qDebug("command -------- "); 517 //qDebug("command -------- ");
515 system ( command.utf8() ); 518 system ( command.utf8() );
516 QFile file( fileName ); 519 QFile file( fileName );
517 if (!file.open( IO_ReadOnly ) ) { 520 if (!file.open( IO_ReadOnly ) ) {
518 return false; 521 return false;
519 522
520 } 523 }
521 QTextStream ts( &file ); 524 QTextStream ts( &file );
522 ts.setCodec( QTextCodec::codecForName("utf8") ); 525 ts.setCodec( QTextCodec::codecForName("utf8") );
523 answer = ts.read(); 526 answer = ts.read();
524 file.close(); 527 file.close();
525 //qDebug("answer \n%s ", answer.latin1()); 528 //qDebug("answer \n%s ", answer.latin1());
526 getNumFromRecord( answer, ev ) ; 529 getNumFromRecord( answer, ev ) ;
527 530
528 } 531 }
529 else { // change existing 532 else { // change existing
530 //qDebug("canging %d %d",ev->zaurusStat() ,ev->zaurusId() ); 533 //qDebug("canging %d %d",ev->zaurusStat() ,ev->zaurusId() );
531 //command = "(echo \"" + ePrefix + eString + "\" ) | db2file datebook -w -g -c " + codec+ " > "+ fileName; 534 //command = "(echo \"" + ePrefix + eString + "\" ) | db2file datebook -w -g -c " + codec+ " > "+ fileName;
532 changeString += eString + "\n"; 535 changeString += eString + "\n";
533 changeEnt = true; 536 changeEnt = true;
534 537
535 } 538 }
536 } 539 }
537 ev = er.next(); 540 ev = er.next();
538 } 541 }
539 status.setText ( i18n("Changing events ...") ); 542 status.setText ( i18n("Changing events ...") );
540 qApp->processEvents(); 543 qApp->processEvents();
541 //qDebug("changing... "); 544 //qDebug("changing... ");
542 if ( changeEnt ) { 545 if ( changeEnt ) {
543 QFile file( fileName ); 546 QFile file( fileName );
544 if (!file.open( IO_WriteOnly ) ) { 547 if (!file.open( IO_WriteOnly ) ) {
545 return false; 548 return false;
546 549
547 } 550 }
548 QTextStream ts( &file ); 551 QTextStream ts( &file );
549 ts.setCodec( QTextCodec::codecForName("utf8") ); 552 ts.setCodec( QTextCodec::codecForName("utf8") );
550 ts << changeString ; 553 ts << changeString ;
551 file.close(); 554 file.close();
552 command = "db2file datebook -w -g -c " + codec+ " < "+ fileName; 555 command = "db2file datebook -w -g -c " + codec+ " < "+ fileName;
553 system ( command.latin1() ); 556 system ( command.latin1() );
554 //qDebug("command %s file :\n%s ", command.latin1(), changeString.latin1()); 557 //qDebug("command %s file :\n%s ", command.latin1(), changeString.latin1());
555 558
556 } 559 }
557 status.setText ( i18n("Deleting events ...") ); 560 status.setText ( i18n("Deleting events ...") );
558 qApp->processEvents(); 561 qApp->processEvents();
559 //qDebug("deleting... "); 562 //qDebug("deleting... ");
560 if ( deleteEnt ) { 563 if ( deleteEnt ) {
561 QFile file( fileName ); 564 QFile file( fileName );
562 if (!file.open( IO_WriteOnly ) ) { 565 if (!file.open( IO_WriteOnly ) ) {
563 return false; 566 return false;
564 567
565 } 568 }
566 QTextStream ts( &file ); 569 QTextStream ts( &file );
567 ts.setCodec( QTextCodec::codecForName("utf8") ); 570 ts.setCodec( QTextCodec::codecForName("utf8") );
568 ts << deleteString; 571 ts << deleteString;
569 file.close(); 572 file.close();
570 command = "db2file datebook -d -c " + codec+ " < "+ fileName; 573 command = "db2file datebook -d -c " + codec+ " < "+ fileName;
571 system ( command.latin1() ); 574 system ( command.latin1() );
572 // qDebug("command %s file :\n%s ", command.latin1(), deleteString.latin1()); 575 // qDebug("command %s file :\n%s ", command.latin1(), deleteString.latin1());
573 } 576 }
574 577
575 578
576 changeString = tPrefix; 579 changeString = tPrefix;
577 deleteString = tPrefix; 580 deleteString = tPrefix;
578 status.setText ( i18n("Processing todos ...") ); 581 status.setText ( i18n("Processing todos ...") );
579 qApp->processEvents(); 582 qApp->processEvents();
580 QPtrList<Todo> tl = calendar->rawTodos(); 583 QPtrList<Todo> tl = calendar->rawTodos();
581 Todo* to = tl.first(); 584 Todo* to = tl.first();
582 i = 0; 585 i = 0;
583 message = i18n("Processing todo # "); 586 message = i18n("Processing todo # ");
584 procCount = 0; 587 procCount = 0;
585 while ( to ) { 588 while ( to ) {
586 if ( to->tempSyncStat() != SYNC_TEMPSTATE_NEW_EXTERNAL ) { 589 if ( to->tempSyncStat() != SYNC_TEMPSTATE_NEW_EXTERNAL ) {
587 status.setText ( message + QString::number ( ++procCount ) ); 590 status.setText ( message + QString::number ( ++procCount ) );
588 qApp->processEvents(); 591 qApp->processEvents();
589 QString eString = getTodoString( to ); 592 QString eString = getTodoString( to );
590 if ( to->tempSyncStat() == SYNC_TEMPSTATE_DELETE ) { // delete 593 if ( to->tempSyncStat() == SYNC_TEMPSTATE_DELETE ) { // delete
591 // deleting empty strings does not work. 594 // deleting empty strings does not work.
592 // we write first and x and then delete the record with the x 595 // we write first and x and then delete the record with the x
593 eString = eString.replace( QRegExp(",\"\""),",\"x\"" ); 596 eString = eString.replace( QRegExp(",\"\""),",\"x\"" );
594 changeString += eString + "\n"; 597 changeString += eString + "\n";
595 deleteString += eString + "\n"; 598 deleteString += eString + "\n";
596 deleteEnt = true; 599 deleteEnt = true;
597 changeEnt = true; 600 changeEnt = true;
598 } 601 }
599 else if ( to->getID("Sharp_DTM").isEmpty() ) { // add new 602 else if ( to->getID("Sharp_DTM").isEmpty() ) { // add new
600 603
601 604
602 605
603 QString fileNameIn = "/tmp/kopitempin"; 606 QString fileNameIn = "/tmp/kopitempin";
604 QFile fileIn( fileNameIn ); 607 QFile fileIn( fileNameIn );
605 if (!fileIn.open( IO_WriteOnly ) ) { 608 if (!fileIn.open( IO_WriteOnly ) ) {
606 return false; 609 return false;
607 } 610 }
608 QTextStream tsIn( &fileIn ); 611 QTextStream tsIn( &fileIn );
609 tsIn.setCodec( QTextCodec::codecForName("utf8") ); 612 tsIn.setCodec( QTextCodec::codecForName("utf8") );
610 tsIn << tPrefix << eString ; 613 tsIn << tPrefix << eString ;
611 fileIn.close(); 614 fileIn.close();
612 command = "(cat /tmp/kopitempin | db2file todo -w -g -c " + codec+ ") > "+ fileName; 615 command = "(cat /tmp/kopitempin | db2file todo -w -g -c " + codec+ ") > "+ fileName;
613 system ( command.utf8() ); 616 system ( command.utf8() );
614 QFile file( fileName ); 617 QFile file( fileName );
615 if (!file.open( IO_ReadOnly ) ) { 618 if (!file.open( IO_ReadOnly ) ) {
616 return false; 619 return false;
617 } 620 }
618 QTextStream ts( &file ); 621 QTextStream ts( &file );
619 ts.setCodec( QTextCodec::codecForName("utf8") ); 622 ts.setCodec( QTextCodec::codecForName("utf8") );
620 answer = ts.read(); 623 answer = ts.read();
621 file.close(); 624 file.close();
622 //qDebug("answer \n%s ", answer.latin1()); 625 //qDebug("answer \n%s ", answer.latin1());
623 getNumFromRecord( answer, to ) ; 626 getNumFromRecord( answer, to ) ;
624 627
625 } 628 }
626 else { // change existing 629 else { // change existing
627 //qDebug("canging %d %d",to->zaurusStat() ,to->zaurusId() ); 630 //qDebug("canging %d %d",to->zaurusStat() ,to->zaurusId() );
628 //command = "(echo \"" + ePrefix + eString + "\" ) | db2file datebook -w -g -c " + codec+ " > "+ fileName; 631 //command = "(echo \"" + ePrefix + eString + "\" ) | db2file datebook -w -g -c " + codec+ " > "+ fileName;
629 changeString += eString + "\n"; 632 changeString += eString + "\n";
630 changeEnt = true; 633 changeEnt = true;
631 634
632 } 635 }
633 } 636 }
634 637
635 to = tl.next(); 638 to = tl.next();
636 } 639 }
637 status.setText ( i18n("Changing todos ...") ); 640 status.setText ( i18n("Changing todos ...") );
638 qApp->processEvents(); 641 qApp->processEvents();
639 //qDebug("changing... "); 642 //qDebug("changing... ");
640 if ( changeEnt ) { 643 if ( changeEnt ) {
641 QFile file( fileName ); 644 QFile file( fileName );
642 if (!file.open( IO_WriteOnly ) ) { 645 if (!file.open( IO_WriteOnly ) ) {
643 return false; 646 return false;
644 647
645 } 648 }
646 QTextStream ts( &file ); 649 QTextStream ts( &file );
647 ts.setCodec( QTextCodec::codecForName("utf8") ); 650 ts.setCodec( QTextCodec::codecForName("utf8") );
648 ts << changeString ; 651 ts << changeString ;
649 file.close(); 652 file.close();
650 command = "db2file todo -w -g -c " + codec+ " < "+ fileName; 653 command = "db2file todo -w -g -c " + codec+ " < "+ fileName;
651 system ( command.latin1() ); 654 system ( command.latin1() );
652 //qDebug("command %s file :\n%s ", command.latin1(), changeString.latin1()); 655 //qDebug("command %s file :\n%s ", command.latin1(), changeString.latin1());
653 656
654 } 657 }
655 status.setText ( i18n("Deleting todos ...") ); 658 status.setText ( i18n("Deleting todos ...") );
656 qApp->processEvents(); 659 qApp->processEvents();
657 //qDebug("deleting... "); 660 //qDebug("deleting... ");
658 if ( deleteEnt ) { 661 if ( deleteEnt ) {
659 QFile file( fileName ); 662 QFile file( fileName );
660 if (!file.open( IO_WriteOnly ) ) { 663 if (!file.open( IO_WriteOnly ) ) {
661 return false; 664 return false;
662 665
663 } 666 }
664 QTextStream ts( &file ); 667 QTextStream ts( &file );
665 ts.setCodec( QTextCodec::codecForName("utf8") ); 668 ts.setCodec( QTextCodec::codecForName("utf8") );
666 ts << deleteString; 669 ts << deleteString;
667 file.close(); 670 file.close();
668 command = "db2file todo -d -c " + codec+ " < "+ fileName; 671 command = "db2file todo -d -c " + codec+ " < "+ fileName;
669 system ( command.latin1() ); 672 system ( command.latin1() );
670 // qDebug("command %s file :\n%s ", command.latin1(), deleteString.latin1()); 673 // qDebug("command %s file :\n%s ", command.latin1(), deleteString.latin1());
671 } 674 }
672 675
673 return true; 676 return true;
674} 677}
675QString SharpFormat::dtToString( const QDateTime& dti, bool useTZ ) 678QString SharpFormat::dtToString( const QDateTime& dti, bool useTZ )
676{ 679{
677 QString datestr; 680 QString datestr;
678 QString timestr; 681 QString timestr;
679 int offset = KGlobal::locale()->localTimeOffset( dti ); 682 int offset = KGlobal::locale()->localTimeOffset( dti );
680 QDateTime dt; 683 QDateTime dt;
681 if (useTZ) 684 if (useTZ)
682 dt = dti.addSecs ( -(offset*60)); 685 dt = dti.addSecs ( -(offset*60));
683 else 686 else
684 dt = dti; 687 dt = dti;
685 if(dt.date().isValid()){ 688 if(dt.date().isValid()){
686 const QDate& date = dt.date(); 689 const QDate& date = dt.date();
687 datestr.sprintf("%04d%02d%02d", 690 datestr.sprintf("%04d%02d%02d",
688 date.year(), date.month(), date.day()); 691 date.year(), date.month(), date.day());
689 } 692 }
690 if(dt.time().isValid()){ 693 if(dt.time().isValid()){
691 const QTime& time = dt.time(); 694 const QTime& time = dt.time();
692 timestr.sprintf("T%02d%02d%02d", 695 timestr.sprintf("T%02d%02d%02d",
693 time.hour(), time.minute(), time.second()); 696 time.hour(), time.minute(), time.second());
694 } 697 }
695 return datestr + timestr; 698 return datestr + timestr;
696} 699}
697QString SharpFormat::getEventString( Event* event ) 700QString SharpFormat::getEventString( Event* event )
698{ 701{
699 QStringList list; 702 QStringList list;
700 list.append( event->getID("Sharp_DTM") ); 703 list.append( event->getID("Sharp_DTM") );
701 list.append( event->categories().join(",") ); 704 list.append( event->categories().join(",") );
702 if ( !event->summary().isEmpty() ) 705 if ( !event->summary().isEmpty() )
703 list.append( event->summary() ); 706 list.append( event->summary() );
704 else 707 else
705 list.append("" ); 708 list.append("" );
706 if ( !event->location().isEmpty() ) 709 if ( !event->location().isEmpty() )
707 list.append( event->location() ); 710 list.append( event->location() );
708 else 711 else
709 list.append("" ); 712 list.append("" );
710 if ( !event->description().isEmpty() ) 713 if ( !event->description().isEmpty() )
711 list.append( event->description() ); 714 list.append( event->description() );
712 else 715 else
713 list.append( "" ); 716 list.append( "" );
714 if ( event->doesFloat () ) { 717 if ( event->doesFloat () ) {
715 list.append( dtToString( QDateTime(event->dtStart().date(), QTime(0,0,0)), false )); 718 list.append( dtToString( QDateTime(event->dtStart().date(), QTime(0,0,0)), false ));
716 list.append( dtToString( QDateTime(event->dtEnd().date(),QTime(23,59,59)), false )); //6 719 list.append( dtToString( QDateTime(event->dtEnd().date(),QTime(23,59,59)), false )); //6
717 list.append( "1" ); 720 list.append( "1" );
718 721
719 } 722 }
720 else { 723 else {
721 list.append( dtToString( event->dtStart()) ); 724 list.append( dtToString( event->dtStart()) );
722 list.append( dtToString( event->dtEnd()) ); //6 725 list.append( dtToString( event->dtEnd()) ); //6
723 list.append( "0" ); 726 list.append( "0" );
724 } 727 }
725 bool noAlarm = true; 728 bool noAlarm = true;
726 if ( event->alarms().count() > 0 ) { 729 if ( event->alarms().count() > 0 ) {
727 Alarm * al = event->alarms().first(); 730 Alarm * al = event->alarms().first();
728 if ( al->enabled() ) { 731 if ( al->enabled() ) {
729 noAlarm = false; 732 noAlarm = false;
730 list.append( "0" ); // yes, 0 == alarm 733 list.append( "0" ); // yes, 0 == alarm
731 list.append( QString::number( al->startOffset().asSeconds()/(-60) ) ); 734 list.append( QString::number( al->startOffset().asSeconds()/(-60) ) );
732 if ( al->type() == Alarm::Audio ) 735 if ( al->type() == Alarm::Audio )
733 list.append( "1" ); // type audio 736 list.append( "1" ); // type audio
734 else 737 else
735 list.append( "0" ); // type silent 738 list.append( "0" ); // type silent
736 } 739 }
737 } 740 }
738 if ( noAlarm ) { 741 if ( noAlarm ) {
739 list.append( "1" ); // yes, 1 == no alarm 742 list.append( "1" ); // yes, 1 == no alarm
740 list.append( "0" ); // no alarm offset 743 list.append( "0" ); // no alarm offset
741 list.append( "1" ); // type 744 list.append( "1" ); // type
742 } 745 }
743 // next is: 11 746 // next is: 11
744 // next is: 11-16 are recurrence 747 // next is: 11-16 are recurrence
745 Recurrence* rec = event->recurrence(); 748 Recurrence* rec = event->recurrence();
746 749
747 bool writeEndDate = false; 750 bool writeEndDate = false;
748 switch ( rec->doesRecur() ) 751 switch ( rec->doesRecur() )
749 { 752 {
750 case Recurrence::rDaily: // 0 753 case Recurrence::rDaily: // 0
751 list.append( "0" ); 754 list.append( "0" );
752 list.append( QString::number( rec->frequency() ));//12 755 list.append( QString::number( rec->frequency() ));//12
753 list.append( "0" ); 756 list.append( "0" );
754 list.append( "0" ); 757 list.append( "0" );
755 writeEndDate = true; 758 writeEndDate = true;
756 break; 759 break;
757 case Recurrence::rWeekly:// 1 760 case Recurrence::rWeekly:// 1
758 list.append( "1" ); 761 list.append( "1" );
759 list.append( QString::number( rec->frequency()) );//12 762 list.append( QString::number( rec->frequency()) );//12
760 list.append( "0" ); 763 list.append( "0" );
761 { 764 {
762 int days = 0; 765 int days = 0;
763 QBitArray weekDays = rec->days(); 766 QBitArray weekDays = rec->days();
764 int i; 767 int i;
765 for( i = 1; i <= 7; ++i ) { 768 for( i = 1; i <= 7; ++i ) {
766 if ( weekDays[i-1] ) { 769 if ( weekDays[i-1] ) {
767 days += 1 << (i-1); 770 days += 1 << (i-1);
768 } 771 }
769 } 772 }
770 list.append( QString::number( days ) ); 773 list.append( QString::number( days ) );
diff --git a/version b/version
index c4fe6e2..0343cbb 100644
--- a/version
+++ b/version
@@ -1 +1 @@
version = "1.9.13"; version = "1.9.14";