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 // Process the Name (N) (frmContactEditor-LoadName.cpp)
161 LoadName(&ContactEditorData.NameTitle, &ContactEditorData.NameForename,
162 &ContactEditorData.NameSurname, &ContactEditorData.NameOtherNames,
163 &ContactEditorData.NameSuffix);
165 // Process the group members (MEMBER) (frmContactEditor-LoadGroup.cpp)
167 LoadMember(&ContactEditorData.GroupsList);
169 // Process the addresses (ADR) (frmContactEditor-LoadAddress.cpp)
171 LoadAddress(&ContactEditorData.GeneralAddressList,
172 &ContactEditorData.GeneralAddressListTown,
173 &ContactEditorData.GeneralAddressListCounty,
174 &ContactEditorData.GeneralAddressListPostCode,
175 &ContactEditorData.GeneralAddressListPref,
176 &ContactEditorData.HomeAddressList,
177 &ContactEditorData.HomeAddressListTown,
178 &ContactEditorData.HomeAddressListCounty,
179 &ContactEditorData.HomeAddressListPostCode,
180 &ContactEditorData.HomeAddressListPref,
181 &ContactEditorData.BusinessAddressList,
182 &ContactEditorData.BusinessAddressListTown,
183 &ContactEditorData.BusinessAddressListCounty,
184 &ContactEditorData.BusinessAddressListPostCode,
185 &ContactEditorData.BusinessAddressListPref,
188 // Process the timezones (TZ) (frmContactEditor-LoadTimeZone.cpp)
190 LoadTimeZone(&ContactEditorData.GeneralTZList,
191 &ContactEditorData.GeneralTZListPref,
192 &ContactEditorData.HomeTZList,
193 &ContactEditorData.HomeTZListPref,
194 &ContactEditorData.BusinessTZList,
195 &ContactEditorData.BusinessTZListPref,
198 // Process the emails (frmContactEditor-LoadEmail.cpp)
200 LoadEmail(&ContactEditorData.GeneralEmailList,
201 &ContactEditorData.GeneralEmailListPref,
202 &ContactEditorData.HomeEmailList,
203 &ContactEditorData.HomeEmailListPref,
204 &ContactEditorData.BusinessEmailList,
205 &ContactEditorData.BusinessEmailListPref,
208 for (std::map<int,wxString>::iterator iter = ContactFileLines.begin();
209 iter != ContactFileLines.end(); ++iter){
211 // Find the colon which splits the start bit from the data part.
213 ContactLine = iter->second;
215 while (ExtraLineSeek == TRUE){
217 // Check if there is extra data on the next line
218 // (indicated by space or tab at the start) and add data.
222 if (iter == ContactFileLines.end()){
229 wxSPropertyNextLine = iter->second;
232 if (wxSPropertyNextLine.Mid(0, 1) == wxT(" ") || wxSPropertyNextLine.Mid(0, 1) == wxT("\t")){
234 wxSPropertyNextLine.Remove(0, 1);
235 //wxSPropertyNextLine.Trim(FALSE);
236 //ContactLine.Trim();
237 ContactLine.Append(wxSPropertyNextLine);
242 ExtraLineSeek = FALSE;
248 ContactLineLen = ContactLine.Len();
250 // Make sure we are not in quotation mode.
251 // Make sure colon does not have \ or \\ before it.
253 for (int i = 0; i <= ContactLineLen; i++){
255 if ((ContactLine.Mid(i, 1) == wxT(";") || ContactLine.Mid(i, 1) == wxT(":")) && PropertyFind == TRUE){
257 PropertyFind = FALSE;
259 } else if (PropertyFind == TRUE){
261 wxSProperty.Append(ContactLine.Mid(i, 1));
265 if (ContactLine.Mid(i, 1) == wxT("\"")){
267 if (QuoteMode == TRUE){
279 if (ContactLine.Mid(i, 1) == wxT(":") && ContactLine.Mid((i - 1), 1) != wxT("\\") && QuoteMode == FALSE){
288 // Split that line at the point into two variables (ignore the colon).
290 wxSPropertySeg1 = ContactLine.Mid(0, QuoteBreakPoint);
291 wxSPropertySeg2 = ContactLine.Mid((QuoteBreakPoint + 1));
293 // Add the data into the contact editor depending on what it is.
295 if (wxSProperty == wxT("VERSION") && VersionProcessed == FALSE){
297 // Check if version is 4.0, otherwise don't
300 if (wxSPropertySeg2 != wxT("4.0")){
301 wxMessageBox(_("This file is not a vCard 4.0 contact and is not supported under Xestia Address Book."),
302 _("Contact not supported"), wxICON_ERROR);
307 VersionProcessed = TRUE;
309 }/* if (wxSProperty == wxT("KIND") && KindProcessed == FALSE){
311 // See frmContactEditor-LoadGroup.cpp
313 LoadKind(wxSPropertySeg2);
315 }/* else if (wxSProperty == wxT("MEMBER")){
317 // See frmContactEditor-LoadGroup.cpp
319 LoadMember(wxSPropertySeg2, &GroupCount);
321 }*/ else if (wxSProperty == wxT("FN")){
323 // See frmContactEditor-LoadName.cpp
325 LoadFN(wxSPropertySeg1, wxSPropertySeg2, &FNCount, &FNProcessed, &ContactData);
327 }/* else if (wxSProperty == wxT("N") && NameProcessed == FALSE){
329 // See frmContactEditor-LoadName.cpp
331 LoadN(wxSPropertySeg1, wxSPropertySeg2, &NameProcessed, &ContactData);
333 }*/ else if (wxSProperty == wxT("NICKNAME")){
335 // See frmContactEditor-LoadNickname.cpp
337 LoadNickname(wxSPropertySeg1, wxSPropertySeg2, &NicknameCount, &ContactData);
339 }/* else if (wxSProperty == wxT("GENDER") && GenderProcessed == FALSE){
341 // See frmContactEditor-LoadGender.cpp
343 LoadGender(wxSPropertySeg1, wxSPropertySeg2, &GenderProcessed, &ContactData);
345 } else if (wxSProperty == wxT("BDAY") && BirthdayProcessed == FALSE){
347 // See frmContactEditor-LoadBADays.cpp
349 LoadBDay(wxSPropertySeg1, wxSPropertySeg2, &BirthdayProcessed);
351 } else if (wxSProperty == wxT("ANNIVERSARY") && AnniversaryProcessed == FALSE){
353 // See frmContactEditor-LoadBADays.cpp
355 LoadAnniversary(wxSPropertySeg1, wxSPropertySeg2, &AnniversaryProcessed);
357 } else if (wxSProperty == wxT("TZ")){
359 // See frmContactEditor-LoadTimeZone.cpp
361 LoadTimeZone(wxSPropertySeg1, wxSPropertySeg2, &TZCount);
363 } else if (wxSProperty == wxT("ADR")){
365 // See frmContactEditor-LoadAddress.cpp
367 LoadADR(wxSPropertySeg1, wxSPropertySeg2, &ADRCount);
369 } else if (wxSProperty == wxT("EMAIL")){
371 // See frmContactEditor-LoadEmail.cpp
373 LoadEmail(wxSPropertySeg1, wxSPropertySeg2, &EmailCount);
375 }*/ else if (wxSProperty == wxT("IMPP")){
377 // See frmContactEditor-LoadIM.cpp
379 LoadIM(wxSPropertySeg1, wxSPropertySeg2, &IMPPCount);
381 } else if (wxSProperty == wxT("TEL")){
383 // See frmContactEditor-LoadTelephone.cpp
385 LoadTelephone(wxSPropertySeg1, wxSPropertySeg2, &TelCount);
387 } else if (wxSProperty == wxT("LANG")){
389 // See frmContactEditor-LoadLanguage.cpp
391 LoadLanguage(wxSPropertySeg1, wxSPropertySeg2, &LangCount);
393 } else if (wxSProperty == wxT("GEO")){
395 // See frmContactEditor-LoadGeo.cpp
397 LoadGeo(wxSPropertySeg1, wxSPropertySeg2, &GeoCount);
399 } else if (wxSProperty == wxT("RELATED")){
401 // See fromContactEditor-LoadRelated.cpp
403 LoadRelated(wxSPropertySeg1, wxSPropertySeg2, &RelatedCount);
405 } else if (wxSProperty == wxT("URL")){
407 // See frmContactEditor-LoadURL.cpp
409 LoadURL(wxSPropertySeg1, wxSPropertySeg2, &URLCount);
411 } else if (wxSProperty == wxT("TITLE")) {
413 // See frmContactEditor-LoadTitle.cpp
415 LoadTitle(wxSPropertySeg1, wxSPropertySeg2, &TitleCount);
417 } else if (wxSProperty == wxT("ROLE")) {
419 // See frmContactEditor-LoadRole.cpp
421 LoadRole(wxSPropertySeg1, wxSPropertySeg2, &RoleCount);
423 } else if (wxSProperty == wxT("ORG")) {
425 // See frmContactEditor-LoadOrg.cpp
427 LoadOrg(wxSPropertySeg1, wxSPropertySeg2, &OrgCount);
429 } else if (wxSProperty == wxT("NOTE")) {
431 // See frmContactEditor-LoadNote.cpp
433 LoadNote(wxSPropertySeg1, wxSPropertySeg2, &NoteCount);
435 } else if (wxSProperty == wxT("CATEGORIES")) {
437 // See frmContactEditor-LoadCategory.cpp
439 LoadCategory(wxSPropertySeg1, wxSPropertySeg2, &CategoryCount);
441 } else if (wxSProperty == wxT("PHOTO")) {
443 // See frmContactEditor-LoadPhoto.cpp
445 LoadPhoto(wxSPropertySeg1, wxSPropertySeg2, &PhotoCount);
447 } else if (wxSProperty == wxT("LOGO")) {
449 // See frmContactEditor-LoadLogo.cpp
451 LoadLogo(wxSPropertySeg1, wxSPropertySeg2, &LogoCount);
453 } else if (wxSProperty == wxT("SOUND")) {
455 // See frmContactEditor-LoadSound.cpp
457 LoadSound(wxSPropertySeg1, wxSPropertySeg2, &SoundCount);
459 } else if (wxSProperty == wxT("CALURI")){
461 // See frmContactEditor-LoadCalendar.cpp
463 LoadCalURI(wxSPropertySeg1, wxSPropertySeg2, &CalAdrCount);
465 } else if (wxSProperty == wxT("CALADRURI")){
467 // See frmContactEditor-LoadCalendar.cpp
469 LoadCalAdrURI(wxSPropertySeg1, wxSPropertySeg2, &CalReqAdrCount);
471 } else if (wxSProperty == wxT("FBURL")){
473 // See frmContactEditor-LoadCalendar.cpp
475 LoadCalFreeBusy(wxSPropertySeg1, wxSPropertySeg2, &FreeBusyCount);
477 } else if (wxSProperty == wxT("KEY")){
479 // See frmContactEditor-LoadKey.cpp
481 LoadKey(wxSPropertySeg1, wxSPropertySeg2, &KeyCount);
483 }/* else if (wxSProperty == wxT("UID") && UIDProcessed == FALSE){
485 UIDToken = wxSPropertySeg2;
488 }*/ else if (wxSProperty.Mid(0, 3) == wxT("VND")){
490 // Split the Vendor three ways.
492 wxStringTokenizer wSTVendorDetails(wxSPropertySeg1, wxT("-"));
495 wxString wxSVNDPropName;
498 while (wSTVendorDetails.HasMoreTokens() == TRUE){
500 wSTVendorDetails.GetNextToken();
501 wxSVNDID = wSTVendorDetails.GetNextToken();
502 wxSVNDPropName = wSTVendorDetails.GetNextToken();
507 if (!wxSVNDID.IsEmpty() && !wxSVNDPropName.IsEmpty()){
509 // Setup the values for later processing.
511 VendorList.insert(std::make_pair(intValueSeek, wxSPropertySeg2));
512 VendorListPEN.insert(std::make_pair(intValueSeek, wxSVNDID));
513 VendorListElement.insert(std::make_pair(intValueSeek, wxSVNDPropName));
515 // Add the data to the vendor variables.
519 coldata.SetId(intValueSeek);
520 coldata.SetData(intValueSeek);
521 coldata.SetText(wxSVNDID + wxT("-") + wxSVNDPropName);
523 ListCtrlIndex = lboVendorNamespace->InsertItem(coldata);
525 VendorList.erase(intValueSeek);
526 VendorListPEN.erase(intValueSeek);
527 VendorListElement.erase(intValueSeek);
529 VendorList.insert(std::make_pair(intValueSeek, wxSPropertySeg2));
530 VendorListPEN.insert(std::make_pair(intValueSeek, wxSVNDID));
531 VendorListElement.insert(std::make_pair(intValueSeek, wxSVNDPropName));
538 } else if (wxSProperty.Mid(0, 2) == wxT("X-")){
542 XTokenList.insert(std::make_pair(intValueSeek, wxSPropertySeg2));
543 XTokenListTokens.insert(std::make_pair(intValueSeek, wxSPropertySeg1.Mid(2)));
549 coldata.SetId(intValueSeek);
550 coldata.SetData(intValueSeek);
551 coldata.SetText(wxSPropertySeg1.Mid(2));
553 ListCtrlIndex = lboXToken->InsertItem(coldata);
561 // Reset the variables.
565 ExtraLineSeek = TRUE;
573 FMTimer.SetFilename(Filename);
574 FMTimer.Start(10000, FALSE);
581 void frmContactEditor::SplitValues(wxString *PropertyLine,
582 std::map<int,int> *SplitPoints,
583 std::map<int,int> *SplitLength,
586 size_t intPropertyLen = PropertyLine->Len();
587 int intSplitsFound = 0;
588 int intSplitSize = 0;
589 int intSplitSeek = 0;
591 for (int i = intSize; i <= intPropertyLen; i++){
595 if (PropertyLine->Mid(i, 1) == wxT(";") &&
596 PropertyLine->Mid((i - 1), 1) != wxT("\\")){
598 if (intSplitsFound == 0){
600 SplitLength->insert(std::make_pair(intSplitsFound, (intSplitSize)));
604 SplitLength->insert(std::make_pair(intSplitsFound, (intSplitSize - 1)));
608 SplitPoints->insert(std::make_pair(intSplitsFound, (i + 1)));
618 if (intSplitsFound == 0){
620 SplitPoints->insert(std::make_pair(intSplitsFound, (8 + 1)));
621 SplitLength->insert(std::make_pair(intSplitsFound, intSplitSize));
625 SplitPoints->insert(std::make_pair(intSplitsFound, (intSplitSeek + 1)));
626 SplitLength->insert(std::make_pair(intSplitsFound, intSplitSize));