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 for (std::map<int,wxString>::iterator iter = ContactFileLines.begin();
166 iter != ContactFileLines.end(); ++iter){
168 // Find the colon which splits the start bit from the data part.
170 ContactLine = iter->second;
172 while (ExtraLineSeek == TRUE){
174 // Check if there is extra data on the next line
175 // (indicated by space or tab at the start) and add data.
179 if (iter == ContactFileLines.end()){
186 wxSPropertyNextLine = iter->second;
189 if (wxSPropertyNextLine.Mid(0, 1) == wxT(" ") || wxSPropertyNextLine.Mid(0, 1) == wxT("\t")){
191 wxSPropertyNextLine.Remove(0, 1);
192 //wxSPropertyNextLine.Trim(FALSE);
193 //ContactLine.Trim();
194 ContactLine.Append(wxSPropertyNextLine);
199 ExtraLineSeek = FALSE;
205 ContactLineLen = ContactLine.Len();
207 // Make sure we are not in quotation mode.
208 // Make sure colon does not have \ or \\ before it.
210 for (int i = 0; i <= ContactLineLen; i++){
212 if ((ContactLine.Mid(i, 1) == wxT(";") || ContactLine.Mid(i, 1) == wxT(":")) && PropertyFind == TRUE){
214 PropertyFind = FALSE;
216 } else if (PropertyFind == TRUE){
218 wxSProperty.Append(ContactLine.Mid(i, 1));
222 if (ContactLine.Mid(i, 1) == wxT("\"")){
224 if (QuoteMode == TRUE){
236 if (ContactLine.Mid(i, 1) == wxT(":") && ContactLine.Mid((i - 1), 1) != wxT("\\") && QuoteMode == FALSE){
245 // Split that line at the point into two variables (ignore the colon).
247 wxSPropertySeg1 = ContactLine.Mid(0, QuoteBreakPoint);
248 wxSPropertySeg2 = ContactLine.Mid((QuoteBreakPoint + 1));
250 // Add the data into the contact editor depending on what it is.
252 if (wxSProperty == wxT("VERSION") && VersionProcessed == FALSE){
254 // Check if version is 4.0, otherwise don't
257 if (wxSPropertySeg2 != wxT("4.0")){
258 wxMessageBox(_("This file is not a vCard 4.0 contact and is not supported under Xestia Address Book."),
259 _("Contact not supported"), wxICON_ERROR);
264 VersionProcessed = TRUE;
266 }/* if (wxSProperty == wxT("KIND") && KindProcessed == FALSE){
268 // See frmContactEditor-LoadGroup.cpp
270 LoadKind(wxSPropertySeg2);
272 }*/ else if (wxSProperty == wxT("MEMBER")){
274 // See frmContactEditor-LoadGroup.cpp
276 LoadMember(wxSPropertySeg2, &GroupCount);
278 } else if (wxSProperty == wxT("FN")){
280 // See frmContactEditor-LoadName.cpp
282 LoadFN(wxSPropertySeg1, wxSPropertySeg2, &FNCount, &FNProcessed, &ContactData);
284 }/* else if (wxSProperty == wxT("N") && NameProcessed == FALSE){
286 // See frmContactEditor-LoadName.cpp
288 LoadN(wxSPropertySeg1, wxSPropertySeg2, &NameProcessed, &ContactData);
290 }*/ else if (wxSProperty == wxT("NICKNAME")){
292 // See frmContactEditor-LoadNickname.cpp
294 LoadNickname(wxSPropertySeg1, wxSPropertySeg2, &NicknameCount, &ContactData);
296 }/* else if (wxSProperty == wxT("GENDER") && GenderProcessed == FALSE){
298 // See frmContactEditor-LoadGender.cpp
300 LoadGender(wxSPropertySeg1, wxSPropertySeg2, &GenderProcessed, &ContactData);
302 } else if (wxSProperty == wxT("BDAY") && BirthdayProcessed == FALSE){
304 // See frmContactEditor-LoadBADays.cpp
306 LoadBDay(wxSPropertySeg1, wxSPropertySeg2, &BirthdayProcessed);
308 } else if (wxSProperty == wxT("ANNIVERSARY") && AnniversaryProcessed == FALSE){
310 // See frmContactEditor-LoadBADays.cpp
312 LoadAnniversary(wxSPropertySeg1, wxSPropertySeg2, &AnniversaryProcessed);
314 }*/ else if (wxSProperty == wxT("TZ")){
316 // See frmContactEditor-LoadTimeZone.cpp
318 LoadTimeZone(wxSPropertySeg1, wxSPropertySeg2, &TZCount);
320 } else if (wxSProperty == wxT("ADR")){
322 // See frmContactEditor-LoadAddress.cpp
324 LoadADR(wxSPropertySeg1, wxSPropertySeg2, &ADRCount);
326 } else if (wxSProperty == wxT("EMAIL")){
328 // See frmContactEditor-LoadEmail.cpp
330 LoadEmail(wxSPropertySeg1, wxSPropertySeg2, &EmailCount);
332 } else if (wxSProperty == wxT("IMPP")){
334 // See frmContactEditor-LoadIM.cpp
336 LoadIM(wxSPropertySeg1, wxSPropertySeg2, &IMPPCount);
338 } else if (wxSProperty == wxT("TEL")){
340 // See frmContactEditor-LoadTelephone.cpp
342 LoadTelephone(wxSPropertySeg1, wxSPropertySeg2, &TelCount);
344 } else if (wxSProperty == wxT("LANG")){
346 // See frmContactEditor-LoadLanguage.cpp
348 LoadLanguage(wxSPropertySeg1, wxSPropertySeg2, &LangCount);
350 } else if (wxSProperty == wxT("GEO")){
352 // See frmContactEditor-LoadGeo.cpp
354 LoadGeo(wxSPropertySeg1, wxSPropertySeg2, &GeoCount);
356 } else if (wxSProperty == wxT("RELATED")){
358 // See fromContactEditor-LoadRelated.cpp
360 LoadRelated(wxSPropertySeg1, wxSPropertySeg2, &RelatedCount);
362 } else if (wxSProperty == wxT("URL")){
364 // See frmContactEditor-LoadURL.cpp
366 LoadURL(wxSPropertySeg1, wxSPropertySeg2, &URLCount);
368 } else if (wxSProperty == wxT("TITLE")) {
370 // See frmContactEditor-LoadTitle.cpp
372 LoadTitle(wxSPropertySeg1, wxSPropertySeg2, &TitleCount);
374 } else if (wxSProperty == wxT("ROLE")) {
376 // See frmContactEditor-LoadRole.cpp
378 LoadRole(wxSPropertySeg1, wxSPropertySeg2, &RoleCount);
380 } else if (wxSProperty == wxT("ORG")) {
382 // See frmContactEditor-LoadOrg.cpp
384 LoadOrg(wxSPropertySeg1, wxSPropertySeg2, &OrgCount);
386 } else if (wxSProperty == wxT("NOTE")) {
388 // See frmContactEditor-LoadNote.cpp
390 LoadNote(wxSPropertySeg1, wxSPropertySeg2, &NoteCount);
392 } else if (wxSProperty == wxT("CATEGORIES")) {
394 // See frmContactEditor-LoadCategory.cpp
396 LoadCategory(wxSPropertySeg1, wxSPropertySeg2, &CategoryCount);
398 } else if (wxSProperty == wxT("PHOTO")) {
400 // See frmContactEditor-LoadPhoto.cpp
402 LoadPhoto(wxSPropertySeg1, wxSPropertySeg2, &PhotoCount);
404 } else if (wxSProperty == wxT("LOGO")) {
406 // See frmContactEditor-LoadLogo.cpp
408 LoadLogo(wxSPropertySeg1, wxSPropertySeg2, &LogoCount);
410 } else if (wxSProperty == wxT("SOUND")) {
412 // See frmContactEditor-LoadSound.cpp
414 LoadSound(wxSPropertySeg1, wxSPropertySeg2, &SoundCount);
416 } else if (wxSProperty == wxT("CALURI")){
418 // See frmContactEditor-LoadCalendar.cpp
420 LoadCalURI(wxSPropertySeg1, wxSPropertySeg2, &CalAdrCount);
422 } else if (wxSProperty == wxT("CALADRURI")){
424 // See frmContactEditor-LoadCalendar.cpp
426 LoadCalAdrURI(wxSPropertySeg1, wxSPropertySeg2, &CalReqAdrCount);
428 } else if (wxSProperty == wxT("FBURL")){
430 // See frmContactEditor-LoadCalendar.cpp
432 LoadCalFreeBusy(wxSPropertySeg1, wxSPropertySeg2, &FreeBusyCount);
434 } else if (wxSProperty == wxT("KEY")){
436 // See frmContactEditor-LoadKey.cpp
438 LoadKey(wxSPropertySeg1, wxSPropertySeg2, &KeyCount);
440 }/* else if (wxSProperty == wxT("UID") && UIDProcessed == FALSE){
442 UIDToken = wxSPropertySeg2;
445 }*/ else if (wxSProperty.Mid(0, 3) == wxT("VND")){
447 // Split the Vendor three ways.
449 wxStringTokenizer wSTVendorDetails(wxSPropertySeg1, wxT("-"));
452 wxString wxSVNDPropName;
455 while (wSTVendorDetails.HasMoreTokens() == TRUE){
457 wSTVendorDetails.GetNextToken();
458 wxSVNDID = wSTVendorDetails.GetNextToken();
459 wxSVNDPropName = wSTVendorDetails.GetNextToken();
464 if (!wxSVNDID.IsEmpty() && !wxSVNDPropName.IsEmpty()){
466 // Setup the values for later processing.
468 VendorList.insert(std::make_pair(intValueSeek, wxSPropertySeg2));
469 VendorListPEN.insert(std::make_pair(intValueSeek, wxSVNDID));
470 VendorListElement.insert(std::make_pair(intValueSeek, wxSVNDPropName));
472 // Add the data to the vendor variables.
476 coldata.SetId(intValueSeek);
477 coldata.SetData(intValueSeek);
478 coldata.SetText(wxSVNDID + wxT("-") + wxSVNDPropName);
480 ListCtrlIndex = lboVendorNamespace->InsertItem(coldata);
482 VendorList.erase(intValueSeek);
483 VendorListPEN.erase(intValueSeek);
484 VendorListElement.erase(intValueSeek);
486 VendorList.insert(std::make_pair(intValueSeek, wxSPropertySeg2));
487 VendorListPEN.insert(std::make_pair(intValueSeek, wxSVNDID));
488 VendorListElement.insert(std::make_pair(intValueSeek, wxSVNDPropName));
495 } else if (wxSProperty.Mid(0, 2) == wxT("X-")){
499 XTokenList.insert(std::make_pair(intValueSeek, wxSPropertySeg2));
500 XTokenListTokens.insert(std::make_pair(intValueSeek, wxSPropertySeg1.Mid(2)));
506 coldata.SetId(intValueSeek);
507 coldata.SetData(intValueSeek);
508 coldata.SetText(wxSPropertySeg1.Mid(2));
510 ListCtrlIndex = lboXToken->InsertItem(coldata);
518 // Reset the variables.
522 ExtraLineSeek = TRUE;
530 FMTimer.SetFilename(Filename);
531 FMTimer.Start(10000, FALSE);
538 void frmContactEditor::SplitValues(wxString *PropertyLine,
539 std::map<int,int> *SplitPoints,
540 std::map<int,int> *SplitLength,
543 size_t intPropertyLen = PropertyLine->Len();
544 int intSplitsFound = 0;
545 int intSplitSize = 0;
546 int intSplitSeek = 0;
548 for (int i = intSize; i <= intPropertyLen; i++){
552 if (PropertyLine->Mid(i, 1) == wxT(";") &&
553 PropertyLine->Mid((i - 1), 1) != wxT("\\")){
555 if (intSplitsFound == 0){
557 SplitLength->insert(std::make_pair(intSplitsFound, (intSplitSize)));
561 SplitLength->insert(std::make_pair(intSplitsFound, (intSplitSize - 1)));
565 SplitPoints->insert(std::make_pair(intSplitsFound, (i + 1)));
575 if (intSplitsFound == 0){
577 SplitPoints->insert(std::make_pair(intSplitsFound, (8 + 1)));
578 SplitLength->insert(std::make_pair(intSplitsFound, intSplitSize));
582 SplitPoints->insert(std::make_pair(intSplitsFound, (intSplitSeek + 1)));
583 SplitLength->insert(std::make_pair(intSplitsFound, intSplitSize));