1 // frmContactEditor-Load.cpp - frmContactEditor load contact subroutines.
3 // (c) 2012-2015 Xestia Software Development.
5 // This file is part of Xestia Address Book.
7 // Xestia Address Book is free software: you can redistribute it and/or modify
8 // it under the terms of the GNU General Public License as published by the
9 // Free Software Foundation, version 3 of the license.
11 // Xestia Address Book is distributed in the hope that it will be useful,
12 // but WITHOUT ANY WARRANTY; without even the implied warranty of
13 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 // GNU General Public License for more details.
16 // You should have received a copy of the GNU General Public License along
17 // with Xestia Address Book. If not, see <http://www.gnu.org/licenses/>
22 #include <wx/tokenzr.h>
23 #include <wx/datetime.h>
26 #include "frmContactEditor.h"
29 #include "../version.h"
30 #include "../vcard/vcard.h"
31 #include "../common/textprocessing.h"
32 #include "../common/dirs.h"
33 #include "cdo/ContactDataObject.h"
35 bool frmContactEditor::LoadContact(wxString Filename){
37 // Load the contact into the contact editor.
40 wxString wxSContactString;
44 if (StartupEditMode == FALSE){
45 XVMData = MainPtr->GetViewMode();
48 wxSContactFilename = Filename;
50 // Check if we are using wxWidgets version 2.8 or less and
51 // execute the required command accordingly.
53 #if wxABI_VERSION < 20900
54 ContactFile.Open(Filename.c_str(), wxT("r"));
56 ContactFile.Open(Filename, wxT("r"));
59 if (ContactFile.IsOpened() == FALSE){
65 ContactEditorData.LoadFile(Filename);
67 ContactFile.ReadAll(&wxSContactString, wxConvAuto());
71 std::map<int, wxString> ContactFileLines;
72 std::map<int, wxString>::iterator striter;
74 wxStringTokenizer wSTContactFileLines(wxSContactString, wxT("\r\n"));
76 int ContactLineSeek = 0;
78 while (wSTContactFileLines.HasMoreTokens() == TRUE){
80 ContactLine = wSTContactFileLines.GetNextToken();
81 ContactFileLines.insert(std::make_pair(ContactLineSeek, ContactLine));
88 bool QuoteMode = FALSE;
89 bool PropertyFind = TRUE;
90 bool HasExtraNicknames = FALSE;
91 bool IgnoreGender = FALSE;
92 bool ExtraLineSeek = TRUE;
93 //bool BirthdayProcessed = FALSE;
94 //bool AnniversaryProcessed = FALSE;
95 bool FNProcessed = FALSE;
96 bool GenderProcessed = FALSE;
97 bool NameProcessed = FALSE;
98 //bool UIDProcessed = FALSE;
99 //bool KindProcessed = FALSE;
100 bool ETagFound = FALSE;
101 bool ETagOrigFound = FALSE;
102 bool VersionProcessed = FALSE;
103 int intExtraNickname = 0;
104 wxString wxSProperty;
105 wxString wxSPropertySeg1;
106 wxString wxSPropertySeg2;
107 wxString wxSPropertyNextLine;
108 size_t ContactLineLen = 0;
109 int QuoteBreakPoint = 0;
112 int NicknameCount = 0;
121 int RelatedCount = 0;
126 int CategoryCount = 0;
132 int CalReqAdrCount = 0;
133 int FreeBusyCount = 0;
137 //int intValueSeek = 1;
139 // Process the unique ID (UID)
141 UIDToken = ContactEditorData.UIDToken;
143 // Process the contact type (KIND) (frmContactEditor-LoadGroup.cpp)
145 LoadKind(&ContactEditorData.ContactKind);
147 // Process the Birthday (BDAY) (frmContactEditor-LoadBADays.cpp)
149 LoadBirthday(&ContactEditorData.Birthday, &ContactEditorData.BirthdayText);
151 // Process the Anniversary (ANNIVERSARY) (frmContactEditor-LoadBADays.cpp)
153 LoadAnniversary(&ContactEditorData.Anniversary, &ContactEditorData.AnniversaryText);
155 // Process the Gender (GENDER) (frmContactEditor-LoadGender.cpp)
157 LoadGender(&ContactEditorData.Gender, &ContactEditorData.GenderDetails);
159 for (std::map<int,wxString>::iterator iter = ContactFileLines.begin();
160 iter != ContactFileLines.end(); ++iter){
162 // Find the colon which splits the start bit from the data part.
164 ContactLine = iter->second;
166 while (ExtraLineSeek == TRUE){
168 // Check if there is extra data on the next line
169 // (indicated by space or tab at the start) and add data.
173 if (iter == ContactFileLines.end()){
180 wxSPropertyNextLine = iter->second;
183 if (wxSPropertyNextLine.Mid(0, 1) == wxT(" ") || wxSPropertyNextLine.Mid(0, 1) == wxT("\t")){
185 wxSPropertyNextLine.Remove(0, 1);
186 //wxSPropertyNextLine.Trim(FALSE);
187 //ContactLine.Trim();
188 ContactLine.Append(wxSPropertyNextLine);
193 ExtraLineSeek = FALSE;
199 ContactLineLen = ContactLine.Len();
201 // Make sure we are not in quotation mode.
202 // Make sure colon does not have \ or \\ before it.
204 for (int i = 0; i <= ContactLineLen; i++){
206 if ((ContactLine.Mid(i, 1) == wxT(";") || ContactLine.Mid(i, 1) == wxT(":")) && PropertyFind == TRUE){
208 PropertyFind = FALSE;
210 } else if (PropertyFind == TRUE){
212 wxSProperty.Append(ContactLine.Mid(i, 1));
216 if (ContactLine.Mid(i, 1) == wxT("\"")){
218 if (QuoteMode == TRUE){
230 if (ContactLine.Mid(i, 1) == wxT(":") && ContactLine.Mid((i - 1), 1) != wxT("\\") && QuoteMode == FALSE){
239 // Split that line at the point into two variables (ignore the colon).
241 wxSPropertySeg1 = ContactLine.Mid(0, QuoteBreakPoint);
242 wxSPropertySeg2 = ContactLine.Mid((QuoteBreakPoint + 1));
244 // Add the data into the contact editor depending on what it is.
246 if (wxSProperty == wxT("VERSION") && VersionProcessed == FALSE){
248 // Check if version is 4.0, otherwise don't
251 if (wxSPropertySeg2 != wxT("4.0")){
252 wxMessageBox(_("This file is not a vCard 4.0 contact and is not supported under Xestia Address Book."),
253 _("Contact not supported"), wxICON_ERROR);
258 VersionProcessed = TRUE;
260 }/* if (wxSProperty == wxT("KIND") && KindProcessed == FALSE){
262 // See frmContactEditor-LoadGroup.cpp
264 LoadKind(wxSPropertySeg2);
266 }*/ else if (wxSProperty == wxT("MEMBER")){
268 // See frmContactEditor-LoadGroup.cpp
270 LoadMember(wxSPropertySeg2, &GroupCount);
272 } else if (wxSProperty == wxT("FN")){
274 // See frmContactEditor-LoadName.cpp
276 LoadFN(wxSPropertySeg1, wxSPropertySeg2, &FNCount, &FNProcessed, &ContactData);
278 } else if (wxSProperty == wxT("N") && NameProcessed == FALSE){
280 // See frmContactEditor-LoadName.cpp
282 LoadN(wxSPropertySeg1, wxSPropertySeg2, &NameProcessed, &ContactData);
284 } else if (wxSProperty == wxT("NICKNAME")){
286 // See frmContactEditor-LoadNickname.cpp
288 LoadNickname(wxSPropertySeg1, wxSPropertySeg2, &NicknameCount, &ContactData);
290 }/* else if (wxSProperty == wxT("GENDER") && GenderProcessed == FALSE){
292 // See frmContactEditor-LoadGender.cpp
294 LoadGender(wxSPropertySeg1, wxSPropertySeg2, &GenderProcessed, &ContactData);
296 } else if (wxSProperty == wxT("BDAY") && BirthdayProcessed == FALSE){
298 // See frmContactEditor-LoadBADays.cpp
300 LoadBDay(wxSPropertySeg1, wxSPropertySeg2, &BirthdayProcessed);
302 } else if (wxSProperty == wxT("ANNIVERSARY") && AnniversaryProcessed == FALSE){
304 // See frmContactEditor-LoadBADays.cpp
306 LoadAnniversary(wxSPropertySeg1, wxSPropertySeg2, &AnniversaryProcessed);
308 }*/ else if (wxSProperty == wxT("TZ")){
310 // See frmContactEditor-LoadTimeZone.cpp
312 LoadTimeZone(wxSPropertySeg1, wxSPropertySeg2, &TZCount);
314 } else if (wxSProperty == wxT("ADR")){
316 // See frmContactEditor-LoadAddress.cpp
318 LoadADR(wxSPropertySeg1, wxSPropertySeg2, &ADRCount);
320 } else if (wxSProperty == wxT("EMAIL")){
322 // See frmContactEditor-LoadEmail.cpp
324 LoadEmail(wxSPropertySeg1, wxSPropertySeg2, &EmailCount);
326 } else if (wxSProperty == wxT("IMPP")){
328 // See frmContactEditor-LoadIM.cpp
330 LoadIM(wxSPropertySeg1, wxSPropertySeg2, &IMPPCount);
332 } else if (wxSProperty == wxT("TEL")){
334 // See frmContactEditor-LoadTelephone.cpp
336 LoadTelephone(wxSPropertySeg1, wxSPropertySeg2, &TelCount);
338 } else if (wxSProperty == wxT("LANG")){
340 // See frmContactEditor-LoadLanguage.cpp
342 LoadLanguage(wxSPropertySeg1, wxSPropertySeg2, &LangCount);
344 } else if (wxSProperty == wxT("GEO")){
346 // See frmContactEditor-LoadGeo.cpp
348 LoadGeo(wxSPropertySeg1, wxSPropertySeg2, &GeoCount);
350 } else if (wxSProperty == wxT("RELATED")){
352 // See fromContactEditor-LoadRelated.cpp
354 LoadRelated(wxSPropertySeg1, wxSPropertySeg2, &RelatedCount);
356 } else if (wxSProperty == wxT("URL")){
358 // See frmContactEditor-LoadURL.cpp
360 LoadURL(wxSPropertySeg1, wxSPropertySeg2, &URLCount);
362 } else if (wxSProperty == wxT("TITLE")) {
364 // See frmContactEditor-LoadTitle.cpp
366 LoadTitle(wxSPropertySeg1, wxSPropertySeg2, &TitleCount);
368 } else if (wxSProperty == wxT("ROLE")) {
370 // See frmContactEditor-LoadRole.cpp
372 LoadRole(wxSPropertySeg1, wxSPropertySeg2, &RoleCount);
374 } else if (wxSProperty == wxT("ORG")) {
376 // See frmContactEditor-LoadOrg.cpp
378 LoadOrg(wxSPropertySeg1, wxSPropertySeg2, &OrgCount);
380 } else if (wxSProperty == wxT("NOTE")) {
382 // See frmContactEditor-LoadNote.cpp
384 LoadNote(wxSPropertySeg1, wxSPropertySeg2, &NoteCount);
386 } else if (wxSProperty == wxT("CATEGORIES")) {
388 // See frmContactEditor-LoadCategory.cpp
390 LoadCategory(wxSPropertySeg1, wxSPropertySeg2, &CategoryCount);
392 } else if (wxSProperty == wxT("PHOTO")) {
394 // See frmContactEditor-LoadPhoto.cpp
396 LoadPhoto(wxSPropertySeg1, wxSPropertySeg2, &PhotoCount);
398 } else if (wxSProperty == wxT("LOGO")) {
400 // See frmContactEditor-LoadLogo.cpp
402 LoadLogo(wxSPropertySeg1, wxSPropertySeg2, &LogoCount);
404 } else if (wxSProperty == wxT("SOUND")) {
406 // See frmContactEditor-LoadSound.cpp
408 LoadSound(wxSPropertySeg1, wxSPropertySeg2, &SoundCount);
410 } else if (wxSProperty == wxT("CALURI")){
412 // See frmContactEditor-LoadCalendar.cpp
414 LoadCalURI(wxSPropertySeg1, wxSPropertySeg2, &CalAdrCount);
416 } else if (wxSProperty == wxT("CALADRURI")){
418 // See frmContactEditor-LoadCalendar.cpp
420 LoadCalAdrURI(wxSPropertySeg1, wxSPropertySeg2, &CalReqAdrCount);
422 } else if (wxSProperty == wxT("FBURL")){
424 // See frmContactEditor-LoadCalendar.cpp
426 LoadCalFreeBusy(wxSPropertySeg1, wxSPropertySeg2, &FreeBusyCount);
428 } else if (wxSProperty == wxT("KEY")){
430 // See frmContactEditor-LoadKey.cpp
432 LoadKey(wxSPropertySeg1, wxSPropertySeg2, &KeyCount);
434 }/* else if (wxSProperty == wxT("UID") && UIDProcessed == FALSE){
436 UIDToken = wxSPropertySeg2;
439 }*/ else if (wxSProperty.Mid(0, 3) == wxT("VND")){
441 // Split the Vendor three ways.
443 wxStringTokenizer wSTVendorDetails(wxSPropertySeg1, wxT("-"));
446 wxString wxSVNDPropName;
449 while (wSTVendorDetails.HasMoreTokens() == TRUE){
451 wSTVendorDetails.GetNextToken();
452 wxSVNDID = wSTVendorDetails.GetNextToken();
453 wxSVNDPropName = wSTVendorDetails.GetNextToken();
458 if (!wxSVNDID.IsEmpty() && !wxSVNDPropName.IsEmpty()){
460 // Setup the values for later processing.
462 VendorList.insert(std::make_pair(intValueSeek, wxSPropertySeg2));
463 VendorListPEN.insert(std::make_pair(intValueSeek, wxSVNDID));
464 VendorListElement.insert(std::make_pair(intValueSeek, wxSVNDPropName));
466 // Add the data to the vendor variables.
470 coldata.SetId(intValueSeek);
471 coldata.SetData(intValueSeek);
472 coldata.SetText(wxSVNDID + wxT("-") + wxSVNDPropName);
474 ListCtrlIndex = lboVendorNamespace->InsertItem(coldata);
476 VendorList.erase(intValueSeek);
477 VendorListPEN.erase(intValueSeek);
478 VendorListElement.erase(intValueSeek);
480 VendorList.insert(std::make_pair(intValueSeek, wxSPropertySeg2));
481 VendorListPEN.insert(std::make_pair(intValueSeek, wxSVNDID));
482 VendorListElement.insert(std::make_pair(intValueSeek, wxSVNDPropName));
489 } else if (wxSProperty.Mid(0, 2) == wxT("X-")){
493 XTokenList.insert(std::make_pair(intValueSeek, wxSPropertySeg2));
494 XTokenListTokens.insert(std::make_pair(intValueSeek, wxSPropertySeg1.Mid(2)));
500 coldata.SetId(intValueSeek);
501 coldata.SetData(intValueSeek);
502 coldata.SetText(wxSPropertySeg1.Mid(2));
504 ListCtrlIndex = lboXToken->InsertItem(coldata);
512 // Reset the variables.
516 ExtraLineSeek = TRUE;
524 FMTimer.SetFilename(Filename);
525 FMTimer.Start(10000, FALSE);
532 void frmContactEditor::SplitValues(wxString *PropertyLine,
533 std::map<int,int> *SplitPoints,
534 std::map<int,int> *SplitLength,
537 size_t intPropertyLen = PropertyLine->Len();
538 int intSplitsFound = 0;
539 int intSplitSize = 0;
540 int intSplitSeek = 0;
542 for (int i = intSize; i <= intPropertyLen; i++){
546 if (PropertyLine->Mid(i, 1) == wxT(";") &&
547 PropertyLine->Mid((i - 1), 1) != wxT("\\")){
549 if (intSplitsFound == 0){
551 SplitLength->insert(std::make_pair(intSplitsFound, (intSplitSize)));
555 SplitLength->insert(std::make_pair(intSplitsFound, (intSplitSize - 1)));
559 SplitPoints->insert(std::make_pair(intSplitsFound, (i + 1)));
569 if (intSplitsFound == 0){
571 SplitPoints->insert(std::make_pair(intSplitsFound, (8 + 1)));
572 SplitLength->insert(std::make_pair(intSplitsFound, intSplitSize));
576 SplitPoints->insert(std::make_pair(intSplitsFound, (intSplitSeek + 1)));
577 SplitLength->insert(std::make_pair(intSplitsFound, intSplitSize));