1 // ContactDataObject.cpp - Client Data Object.
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/>
19 #include "ContactDataObject.h"
21 ContactLoadStatus ContactDataObject::LoadFile(wxString Filename){
23 if (!wxFileExists(Filename)){
25 return CONTACTLOAD_FILEMISSING;
31 if (!ContactFile.Open(Filename, wxFile::read, wxS_DEFAULT)){
33 return CONTACTLOAD_FILEERROR;
37 // Check that the vCard is a valid vCard 4.0 file.
39 vCard vCard4FormatCheck;
41 vCard4FormatCheck.LoadFile(Filename);
43 if (vCard4FormatCheck.Get("VERSION") != wxT("4.0")){
45 return CONTACTLOAD_FILEINVALIDFORMAT;
49 // Check that the vCard meets the base specification.
51 if (!vCard4FormatCheck.MeetBaseSpecification()){
53 return CONTACTLOAD_FILEBASESPECFAIL;
57 wxStringTokenizer wSTContactFileLines(vCard4FormatCheck.WriteString(), wxT("\r\n"));
59 std::map<int, wxString> ContactFileLines;
61 int ContactLineSeek = 0;
63 while (wSTContactFileLines.HasMoreTokens() == TRUE){
65 wxString ContactLine = wSTContactFileLines.GetNextToken();
66 ContactFileLines.insert(std::make_pair(ContactLineSeek, ContactLine));
71 wxString wxSPropertyNextLine;
73 bool ExtraLineSeek = TRUE;
74 bool QuoteMode = FALSE;
75 bool PropertyFind = TRUE;
76 bool KindProcessed = FALSE;
77 int ContactLineLen = 0;
78 int QuoteBreakPoint = 0;
82 wxString PropertyLine;
83 wxString PropertySeg1;
84 wxString PropertySeg2;
85 wxString PropertyNextLine;
88 for (std::map<int,wxString>::iterator iter = ContactFileLines.begin();
89 iter != ContactFileLines.end(); ++iter){
102 ContactLine = iter->second;
104 while (ExtraLineSeek == TRUE){
106 // Check if there is extra data on the next line
107 // (indicated by space or tab at the start) and add data.
111 if (iter == ContactFileLines.end()){
118 PropertyNextLine = iter->second;
120 if (PropertyNextLine.Mid(0, 1) == wxT(" ") || PropertyNextLine.Mid(0, 1) == wxT("\t")){
122 PropertyNextLine.Remove(0, 1);
123 ContactLine.Append(PropertyNextLine);
128 ExtraLineSeek = FALSE;
134 ContactLineLen = ContactLine.Len();
136 // Make sure we are not in quotation mode.
137 // Make sure colon does not have \ or \\ before it.
139 for (int i = 0; i <= ContactLineLen; i++){
141 if ((ContactLine.Mid(i, 1) == wxT(";") || ContactLine.Mid(i, 1) == wxT(":")) && PropertyFind == TRUE){
143 PropertyFind = FALSE;
145 } else if (PropertyFind == TRUE){
147 Property.Append(ContactLine.Mid(i, 1));
151 if (ContactLine.Mid(i, 1) == wxT("\"")){
153 if (QuoteMode == TRUE){
165 if (ContactLine.Mid(i, 1) == wxT(":") && ContactLine.Mid((i - 1), 1) != wxT("\\") && QuoteMode == FALSE){
174 // Split that line at the point into two variables (ignore the colon).
176 PropertySeg1 = ContactLine.Mid(0, QuoteBreakPoint);
177 PropertySeg2 = ContactLine.Mid((QuoteBreakPoint + 1));
179 if (Property == wxT("KIND") && KindProcessed == FALSE){
181 ProcessKind(PropertySeg2);
183 } else if (Property == wxT("MEMBER")){
185 ProcessMember(PropertySeg1, PropertySeg2, &GroupCount);
188 } else if (Property == wxT("FN")){
190 ProcessFN(PropertySeg1, PropertySeg2, &FNCount);
197 return CONTACTLOAD_OK;
201 void ContactDataObject::ProcessKind(wxString KindType){
203 if (KindType == wxT("individual")){
205 ContactKind = CONTACTKIND_INDIVIDUAL;
207 } else if (KindType == wxT("group")){
209 ContactKind = CONTACTKIND_GROUP;
211 } else if (KindType == wxT("org")){
213 ContactKind = CONTACTKIND_ORGANISATION;
215 } else if (KindType == wxT("location")){
217 ContactKind = CONTACTKIND_LOCATION;
221 ContactKind = CONTACTKIND_NONE;
226 void ContactDataObject::ProcessMember(wxString PropertySeg1, wxString PropertySeg2, int *GroupCount){
228 std::map<int, int> SplitPoints;
229 std::map<int, int> SplitLength;
231 int intPrevValue = 8;
235 SplitValues(&PropertySeg1, &SplitPoints, &SplitLength, intPrevValue);
239 wxString PropertyName;
240 wxString PropertyValue;
241 wxString PropertyData;
242 wxString PropertyTokens;
243 std::map<int,int>::iterator SLiter;
244 bool FirstToken = TRUE;
246 for (std::map<int, int>::iterator intiter = SplitPoints.begin();
247 intiter != SplitPoints.end(); ++intiter){
249 SLiter = SplitLength.find(intiter->first);
251 PropertyData = PropertySeg1.Mid(intPrevValue, (SLiter->second));
253 wxStringTokenizer PropertyElement (PropertyData, wxT("="));
254 PropertyName = PropertyElement.GetNextToken();
255 PropertyValue = PropertyElement.GetNextToken();
257 intPrevValue = intiter->second;
259 CaptureString(&PropertyValue, FALSE);
261 if (PropertyName == wxT("ALTID")){
263 GroupsListAltID.erase(*GroupCount);
264 GroupsListAltID.insert(std::make_pair(*GroupCount, PropertyValue));
266 } else if (PropertyName == wxT("PID")){
268 GroupsListPID.erase(*GroupCount);
269 GroupsListPID.insert(std::make_pair(*GroupCount, PropertyValue));
271 } else if (PropertyName == wxT("PREF")){
273 int PriorityNumber = 0;
274 bool ValidNumber = TRUE;
277 PriorityNumber = std::stoi(PropertyValue.ToStdString());
280 catch(std::invalid_argument &e){
284 if (ValidNumber == TRUE){
286 GroupsListPref.erase(*GroupCount);
287 GroupsListPref.insert(std::make_pair(*GroupCount, PriorityNumber));
291 } else if (PropertyName == wxT("MEDIATYPE")){
293 GroupsListMediaType.erase(*GroupCount);
294 GroupsListMediaType.insert(std::make_pair(*GroupCount, PropertyValue));
296 } else if (!PropertyName.IsEmpty() && !PropertyValue.IsEmpty()){
298 if (FirstToken == TRUE){
300 PropertyTokens.Append(PropertyName + wxT("=") + PropertyValue);
305 PropertyTokens.Append(wxT(";") + PropertyName + wxT("=") + PropertyValue);
313 GroupsList.insert(std::make_pair(*GroupCount, PropertySeg2));
315 if (!PropertyTokens.IsEmpty()){
317 GroupsListTokens.insert(std::make_pair(*GroupCount, PropertyTokens));
324 void ContactDataObject::ProcessFN(wxString PropertySeg1, wxString PropertySeg2, int *FNCount){
326 std::map<int, int> SplitPoints;
327 std::map<int, int> SplitLength;
329 int intPrevValue = 4;
333 SplitValues(&PropertySeg1, &SplitPoints, &SplitLength, intPrevValue);
337 wxString PropertyName;
338 wxString PropertyValue;
339 wxString PropertyData;
340 wxString PropertyTokens;
341 std::map<int,int>::iterator SLiter;
342 bool FirstToken = TRUE;
344 for (std::map<int, int>::iterator intiter = SplitPoints.begin();
345 intiter != SplitPoints.end(); ++intiter){
347 SLiter = SplitLength.find(intiter->first);
349 PropertyData = PropertySeg1.Mid(intPrevValue, (SLiter->second));
351 wxStringTokenizer PropertyElement (PropertyData, wxT("="));
352 PropertyName = PropertyElement.GetNextToken();
353 PropertyValue = PropertyElement.GetNextToken();
355 intPrevValue = intiter->second;
357 CaptureString(&PropertyValue, FALSE);
359 if (PropertyName == wxT("TYPE")){
361 if (!PropertyValue.IsEmpty() || PropertyValue == wxT("home") ||
362 PropertyValue == wxT("work") ){
364 FullNamesListType.erase(*FNCount);
365 FullNamesListType.insert(std::make_pair(*FNCount, PropertyValue));
369 } else if (PropertyName == wxT("LANGUAGE")){
371 FullNamesListLanguage.erase(*FNCount);
372 FullNamesListLanguage.insert(std::make_pair(*FNCount, PropertyValue));
374 } else if (PropertyName == wxT("ALTID")){
376 FullNamesListAltID.erase(*FNCount);
377 FullNamesListAltID.insert(std::make_pair(*FNCount, PropertyValue));
379 } else if (PropertyName == wxT("PID")){
381 FullNamesListPID.erase(*FNCount);
382 FullNamesListPID.insert(std::make_pair(*FNCount, PropertyValue));
384 } else if (PropertyName == wxT("PREF")){
386 int PriorityNumber = 0;
387 bool ValidNumber = TRUE;
390 PriorityNumber = std::stoi(PropertyValue.ToStdString());
393 catch(std::invalid_argument &e){
397 if (ValidNumber == TRUE){
399 FullNamesListPref.erase(*FNCount);
400 FullNamesListPref.insert(std::make_pair(*FNCount, PriorityNumber));
404 } else if (!PropertyName.IsEmpty() && !PropertyValue.IsEmpty()){
406 if (FirstToken == TRUE){
408 PropertyTokens.Append(PropertyName + wxT("=") + PropertyValue);
413 PropertyTokens.Append(wxT(";") + PropertyName + wxT("=") + PropertyValue);
421 FullNamesList.insert(std::make_pair(*FNCount, PropertySeg2));
423 if (!PropertyTokens.IsEmpty()){
425 FullNamesListTokens.insert(std::make_pair(*FNCount, PropertyTokens));
431 void SplitValues(wxString *PropertyLine,
432 std::map<int,int> *SplitPoints,
433 std::map<int,int> *SplitLength,
436 size_t intPropertyLen = PropertyLine->Len();
437 int intSplitsFound = 0;
438 int intSplitSize = 0;
439 int intSplitSeek = 0;
441 for (int i = intSize; i <= intPropertyLen; i++){
445 if (PropertyLine->Mid(i, 1) == wxT(";") &&
446 PropertyLine->Mid((i - 1), 1) != wxT("\\")){
448 if (intSplitsFound == 0){
450 SplitLength->insert(std::make_pair(intSplitsFound, (intSplitSize)));
454 SplitLength->insert(std::make_pair(intSplitsFound, (intSplitSize - 1)));
458 SplitPoints->insert(std::make_pair(intSplitsFound, (i + 1)));
468 if (intSplitsFound == 0){
470 SplitPoints->insert(std::make_pair(intSplitsFound, (8 + 1)));
471 SplitLength->insert(std::make_pair(intSplitsFound, intSplitSize));
475 SplitPoints->insert(std::make_pair(intSplitsFound, (intSplitSeek + 1)));
476 SplitLength->insert(std::make_pair(intSplitsFound, intSplitSize));