From e8e32f7a1194baabd6e158805f40396c667e6c56 Mon Sep 17 00:00:00 2001 From: Unrud Date: Fri, 10 Mar 2017 23:02:35 +0100 Subject: Add InfCloud --- radicale_web/web/infcloud/data_process.js | 7792 +++++++++++++++++++++++++++++ 1 file changed, 7792 insertions(+) create mode 100644 radicale_web/web/infcloud/data_process.js (limited to 'radicale_web/web/infcloud/data_process.js') diff --git a/radicale_web/web/infcloud/data_process.js b/radicale_web/web/infcloud/data_process.js new file mode 100644 index 0000000..d52f7dc --- /dev/null +++ b/radicale_web/web/infcloud/data_process.js @@ -0,0 +1,7792 @@ +/* +InfCloud - the open source CalDAV/CardDAV Web Client +Copyright (C) 2011-2015 + Jan Mate + Andrej Lezo + Matej Mihalik + +This program is free software: you can redistribute it and/or modify +it under the terms of the GNU Affero General Public License as +published by the Free Software Foundation, either version 3 of the +License, or (at your option) any later version. + +This program is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU Affero General Public License for more details. + +You should have received a copy of the GNU Affero General Public License +along with this program. If not, see . +*/ + +String.prototype.parseComnpactISO8601=function(uid) +{ + if(this.length>=15) + var formattedString=this.substring(0, 4)+'/'+this.substring(4, 6)+'/'+this.substring(6, 8)+' '+this.substring(9, 11)+':'+this.substring(11, 13)+':'+this.substring(13, 15); + else + var formattedString=this.substring(0, 4)+'/'+this.substring(4, 6)+'/'+this.substring(6, 8)+' '+'00:00:00'; + + var value=Date.parse(formattedString); + if(isNaN(value)) + return false + else + return new Date(value); +} + +function repeatStartCompare(objA,objB) +{ + var startA='',startB=''; + if(objA.rec_id!='') + startA=objA.rec_id.parseComnpactISO8601(); + else if(objA.start) + startA=new Date(objA.start.getTime()); + else if(objA.end) + startA=new Date(objA.end.getTime()); + else + startA=Infinity; + + if(objB.rec_id!='') + startB=objB.rec_id.parseComnpactISO8601(); + else if(objB.start) + startB=new Date(objB.start.getTime()); + else if(objB.end) + startB=new Date(objB.end.getTime()); + else + startB=Infinity; + + if(startAstartB) + return 1; + + return 0; +} + +function findWeek(weekNo,inDate,day) +{ + var distance = (day + 7 - inDate.getDay()) % 7; + var date = new Date(inDate.getTime()); + date.setDate(date.getDate() + distance); + if(date.getWeekNo() <= weekNo) + date.setDate(date.getDate() + 7*(weekNo-date.getWeekNo())); + else + { + var actualYearWeeks = new Date(date.getFullYear(),11,31,1,1,1).getWeekNo(); + date.setDate(date.getDate() + 7*(actualYearWeeks-date.getWeekNo())); + } + +} +String.prototype.getSecondsFromOffset=function() +{ + if(this.length>=5) + { + var hours=this.substring(1,3); + var minutes=this.substring(3,5); + var seconds='00'; + if(this.length>=7) + seconds=this.substring(5,7); + + var value=parseInt(hours,10)*60*60+parseInt(minutes,10)*60+parseInt(seconds,10); + if(this.charAt(0)=='-') + value=value*-1; + + if(!isNaN(value)) + return value + else + return 0; + } + else + return 0; +} +Array.prototype.indexElementOf=function(value) +{ + for(var i=0;i0) + { + for(var v=0;vnow) + { + var delay=aTime-now; + if(maxAlarmValue0) + { + for(var ir=0;ir0 && eventsArray[collections[i].uid][j].id!=eventsArray[collections[i].uid][j-1].id) + if(calEvent.alertTime.length>0) + { + for(var k=0; know) + { + var delay=aTime-now; + if(maxAlarmValue0 && todosArray[todoCollections[i].uid][j].id!=todosArray[todoCollections[i].uid][j-1].id) + if(todoEvent.alertTime.length>0) + { + if(todoEvent.end) + var showDate= new Date(todoEvent.end.getTime()); + else if(todoEvent.start) + var showDate= new Date(todoEvent.start.getTime()); + else + var showDate=new Date(); + for(var k=0; know) + { + var delay=aTime-now; + if(maxAlarmValuedate) + continue; + + if(checkRule && daylightComponents[i].startMonth) // is RRULE SET + { + objDayLight=daylightComponents[i]; + actualDaylightComponent=getDateFromDay(objDayLight, date,false,uid); + break; + } + else + { + for(var j=0;jdate) + continue; + + if(checkRule && standardComponents[i].startMonth) // is RRULE SET + { + objDayLight=standardComponents[i]; + actualStandardComponent=getDateFromDay(objDayLight, date); + break; + } + else + { + for(var j=0;jactualStandardComponent.startDate) + offset=actualDaylightComponent.offsetTo; + else + offset=actualStandardComponent.offsetTo; + } + else if(actualDaylightComponent) + offset=actualDaylightComponent.offsetTo; + else if(actualStandardComponent) + offset=actualStandardComponent.offsetTo; + } + else if(tZone == 'local') + offset = getStringLocalOffset(date); + return offset; +} + +function getStringLocalOffset(date) +{ + var offset = '+0000'; + var localOffset = date.getTimezoneOffset(); + if(localOffset>0) + { + var hours = Math.floor(localOffset/60); + var minutes = localOffset - hours*60; + offset = '-' + (hours<10 ? '0'+hours : hours); + offset += (minutes<10 ? '0'+minutes : minutes); + } + else if(localOffset<0) + { + localOffset = localOffset*-1; + var hours = Math.floor(localOffset/60); + var minutes = localOffset - hours*60; + offset = '+' + (hours<10 ? '0'+hours : hours); + offset += (minutes<10 ? '0'+minutes : minutes); + } + + return offset; +} + +function getDayLightObject(tzObject,t) +{ + var dayLightStartDate, dayLightEndDate, myDate=t; + dayLightStartDate=getDateFromDay(tzObject, t); + dayLightEndDate=getDateFromDay(tzObject, t); + + for(var i=0;idayLightEndDate) + { + if(myDate>dayLightStartDate) + dayLightEndDate.setFullYear(dayLightEndDate.getFullYear()+1); + else + dayLightStartDate.setFullYear(dayLightStartDate.getFullYear()-1); + } + + return {dayLightStartDate : dayLightStartDate, dayLightEndDate: dayLightEndDate}; +} + +function deleteEventFromArray(uid) +{ + var rid=uid.substring(0, uid.lastIndexOf('/')+1); + var count=0; + if(globalEventList.displayEventsArray[rid]!=null && typeof globalEventList.displayEventsArray[rid] != 'undefined') + for(var i=globalEventList.displayEventsArray[rid].length-1;i>=0;i--) + if(globalEventList.displayEventsArray[rid][i].id==uid) + { + count++; + for(var o=0;o=0;i--) + if(globalEventList.displayTodosArray[rid][i].id==uid) + { + for(var o=0;o0) + { + var rid=$('#uid').val().substring(0, $('#uid').val().lastIndexOf('/')+1); + if(rid) + if(globalEventList.events[rid][$('#uid').val()].uid!=undefined) + origVcalendarString=globalEventList.events[rid][$('#uid').val()].vcalendar; + while(origVcalendarString.match(vCalendar.pre['vevent'])!=null) + { + if(origVcalendarString.substring(origVcalendarString.indexOf('BEGIN:VEVENT')-2, origVcalendarString.indexOf('BEGIN:VEVENT'))=='\r\n') + { + var partEvent=origVcalendarString.substring(origVcalendarString.indexOf('BEGIN:VEVENT')-2,origVcalendarString.indexOf('END:VEVENT')+'END:VEVENT'.length); + origVcalendarString=origVcalendarString.replace(partEvent, ''); + } + else + { + var partEvent=origVcalendarString.substring(origVcalendarString.indexOf('BEGIN:VEVENT'),origVcalendarString.indexOf('END:VEVENT')+'END:VEVENT'.length); + origVcalendarString=origVcalendarString.replace(partEvent, ''); + partEvent+='\r\n'; + } + eventStringArray[eventStringArray.length]=partEvent; + } + } + var origTimezone = ''; + for(var iE=0;iE1 && $('#vcalendarHash').val()==hex_sha256(inputEvents[j].vcalendar)) + inputEvents[j].vcalendar=changeRuleForFuture(inputEvents[j], $('#futureStart').val().split(';')[0]); + + if(inputEvents[j].vcalendar.indexOf('\r\n')==0 && vCalendarText.lastIndexOf('\r\n')==(vCalendarText.length-2)) + vCalendarText+=inputEvents[j].vcalendar.substring(2,inputEvents[j].vcalendar.length); + else if((inputEvents[j].vcalendar.indexOf('\r\n')==0 && vCalendarText.lastIndexOf('\r\n')!=(vCalendarText.length-2)) || (inputEvents[j].vcalendar.indexOf('\r\n')!=0 && vCalendarText.lastIndexOf('\r\n')==(vCalendarText.length-2)) ) + vCalendarText+=inputEvents[j].vcalendar; + else + vCalendarText+='\r\n'+inputEvents[j].vcalendar; + futureMode=true; + } + else if(deleteMode && $('#futureStart').val().split(';')[0]!='' && $('#futureStart').val().split(';')[1]==inputEvents[j].start) + { + if($('#vcalendarHash').val()==hex_sha256(inputEvents[j].vcalendar)) + { + inputEvents[j].vcalendar=changeRuleForFuture(inputEvents[j], 2); + } + + if(inputEvents[j].vcalendar.indexOf('\r\n')==0 && vCalendarText.lastIndexOf('\r\n')==(vCalendarText.length-2)) + vCalendarText+=inputEvents[j].vcalendar.substring(2,inputEvents[j].vcalendar.length); + else if((inputEvents[j].vcalendar.indexOf('\r\n')==0 && vCalendarText.lastIndexOf('\r\n')!=(vCalendarText.length-2)) || (inputEvents[j].vcalendar.indexOf('\r\n')!=0 && vCalendarText.lastIndexOf('\r\n')==(vCalendarText.length-2)) ) + vCalendarText+=inputEvents[j].vcalendar; + else + vCalendarText+='\r\n'+inputEvents[j].vcalendar; + } + else + { + realEvent=inputEvents[j]; + } + } + vCalendarText=vCalendarText.replace(realEvent.vcalendar,''); + for(var ip=0; ip0) + // sel_option=inputEvents[0].timeZone; + //} + + if(sel_option=='UTC') + { + isUTC=true; + timeZoneAttr=''; + } + else if(sel_option=='local') + timeZoneAttr=''; + else if(sel_option=='custom') + timeZoneAttr=';'+vcalendarEscapeValue('TZID='+realEvent.timeZone); + else + timeZoneAttr=';'+vcalendarEscapeValue('TZID='+sel_option); + + var timezoneComponent=''; + if(globalSettings.rewritetimezonecomponent.value || !vCalendar.tplM['unprocessedVTIMEZONE']) + { + if(tzArray.indexOf(sel_option)==-1) + timezoneComponent=buildTimezoneComponent(sel_option); + } + else + timezoneComponent=vCalendar.tplM['unprocessedVTIMEZONE']; + + if(vCalendarText.lastIndexOf('\r\n')!=(vCalendarText.length-2)) + vCalendarText+='\r\n'; + + vCalendarText+=timezoneComponent; + } + // ---------------------------------- EVENT ---------------------------------- // + if(vCalendar.tplM['beginVEVENT']!=null && (process_elem=vCalendar.tplM['beginVEVENT'][0])!=undefined) + { + if(vCalendarText.lastIndexOf('\r\n')==(vCalendarText.length-2)) + vCalendarText+=vCalendar.tplM['beginVEVENT'][0]; + else + vCalendarText+='\r\n'+vCalendar.tplM['beginVEVENT'][0]; + vevent=true; + } + else + { + process_elem=vCalendar.tplC['beginVEVENT']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + + if(vCalendarText.lastIndexOf('\r\n')==(vCalendarText.length-2)) + vCalendarText+=process_elem; + else + vCalendarText+='\r\n'+process_elem; + vevent=true; + } + + var d, + utc, + d=new Date(); + + utc=d.getUTCFullYear()+(d.getUTCMonth()+1<10 ? '0' : '')+(d.getUTCMonth()+1)+(d.getUTCDate()<10 ? '0' : '')+d.getUTCDate()+'T'+(d.getUTCHours()<10 ? '0' : '')+d.getUTCHours()+(d.getUTCMinutes()<10 ? '0' : '')+d.getUTCMinutes()+(d.getUTCSeconds()<10 ? '0' : '')+d.getUTCSeconds()+'Z'; + var create=true; + + if($('#recurrenceID').val()=='') + var checkVal='orig'; + else + var checkVal=$('#recurrenceID').val(); + + var created=''; + for(vev in vCalendar.tplM['contentline_CREATED']) + { + if(vev==checkVal) + created=vCalendar.tplM['contentline_CREATED'][vev]; + } + if(created!='') + { + process_elem=created; + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_CREATED']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue(utc)); + } + vCalendarText+=process_elem; + + if(vCalendar.tplM['contentline_LM']!=null && (process_elem=vCalendar.tplM['contentline_LM'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_LM']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue(utc)); + vCalendarText+=process_elem; + + if(vCalendar.tplM['contentline_DTSTAMP']!=null && (process_elem=vCalendar.tplM['contentline_DTSTAMP'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_DTSTAMP']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue(utc)); + vCalendarText+=process_elem; + + // UID (required by RFC) + if($('#futureStart').val()=='' && (operation!='MOVE_IN'&& operation!='MOVE_OTHER') && (vCalendar.tplM['contentline_UID']!=null && (process_elem=vCalendar.tplM['contentline_UID'][0])!=undefined)) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_UID']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + // it is VERY small probability, that for 2 newly created vevents/vtodos the same UID is generated (but not impossible :( ...) + var newUID=globalEventList.getNewUID(); + process_elem=process_elem.replace('##:::##uid##:::##', newUID); + } + vCalendarText+=process_elem; + + if(vCalendar.tplM['contentline_SUMMARY']!=null && (process_elem=vCalendar.tplM['contentline_SUMMARY'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_SUMMARY']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue($('#name').val())); + //process_elem=process_elem.replace('##:::##value##:::##',vcalendarEscapeValue('zmena')); + vCalendarText+=process_elem; + + if($('#priority').val()!='0') + { + if(vCalendar.tplM['contentline_PRIORITY']!=null && (process_elem=vCalendar.tplM['contentline_PRIORITY'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_PRIORITY']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue($('#priority').val())); + vCalendarText+=process_elem; + } + + if(vevent) + { + if($('#repeat').val()!='no-repeat') + { + var interval=$("#repeat_interval_detail").val(); + var byDay=''; + var monthDay=''; + var bymonth=''; + var wkst=''; + var isCustom=false; + if(interval==1 || interval=='') + interval=''; + else interval=";INTERVAL="+$("#repeat_interval_detail").val(); + + var frequency=$('#repeat').val(); + if(frequency=='TWO_WEEKLY') + { + frequency='WEEKLY'; + interval=";INTERVAL="+2; + } + else if(frequency=='BUSINESS') + { + frequency='WEEKLY'; + byDay=';BYDAY='; + if(globalSettings.weekenddays.value.length>0) + { + for(var i=0;i<7;i++) + if(globalSettings.weekenddays.value.indexOf(i)==-1) + byDay+=i+','; + byDay=byDay.substring(0,byDay.length-1); + byDay=byDay.replace(1,'MO').replace(2,'TU').replace(3,'WE').replace(4,'TH').replace(5,'FR').replace(6,'SA').replace(0,'SU'); + } + else + { + byDay='SA,SU'; + } + interval=''; + } + else if(frequency=='WEEKEND') + { + frequency='WEEKLY'; + byDay=';BYDAY='; + if(globalSettings.weekenddays.value.length>0) + { + for(var i=0;i0) + { + byDay=';BYDAY='; + for(var ri=0;ri0) + { + monthDay=';BYMONTHDAY='; + for(var ri=0;ri0) + { + bymonth=';BYMONTH='; + for(var ri=0;ri0) + { + monthDay=';BYMONTHDAY='; + for(var ri=0;ri0) + { + var alarmText = ''; + if($(".alert[data-id="+(t+1)+"]").val()!='none') + { + if(vCalendar.tplM['beginVALARM']!=null && (process_elem=vCalendar.tplM['beginVALARM'][0])!=undefined) + alarmText+=vCalendar.tplM['beginVALARM'][0]; + else + { + process_elem=vCalendar.tplC['beginVALARM']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + alarmText+=process_elem; + vevent=true; + } + + if($(".alert[data-id="+(t+1)+"]").val()=='message') + { + if($(".alert_message_details[data-id="+(t+1)+"]").val()=='on_date') + { + if(vCalendar.tplM['contentline_TRIGGER']!=null && (process_elem=vCalendar.tplM['contentline_TRIGGER'][0])!=undefined) + { + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_TRIGGER']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + + var dateTo=$.datepicker.parseDate(globalSettings.datepickerformat.value, $(".message_date_input[data-id="+(t+1)+"]").val()); + var datetime_to=$.fullCalendar.formatDate(dateTo, 'yyyy-MM-dd'); + var aDate=new Date(Date.parse("01/02/1990, "+$(".message_time_input[data-id="+(t+1)+"]").val() )); + var time_to=$.fullCalendar.formatDate(aDate, 'HH:mm:ss'); + + var alarmDT=$.fullCalendar.parseDate(datetime_to+'T'+time_to); + + if(globalSettings.timezonesupport.value) + sel_option=$('#timezone').val(); + + if($('.timezone_row').css('display')=='none') + sel_option='local'; + + if(sel_option!='local') + { + var origValOffset=getOffsetByTZ(sel_option, alarmDT); + var origIntOffset = origValOffset.getSecondsFromOffset()*-1; + alarmDT = new Date(alarmDT.setSeconds(origIntOffset)); + } + + var newValue=$.fullCalendar.formatDate(alarmDT, "yyyyMMdd'T'HHmmss")+(sel_option!='local' ? 'Z' : ''); + + process_elem=process_elem.replace('##:::##VALUE=DATE-TIME##:::##', ';VALUE=DATE-TIME'); + process_elem=process_elem.replace('##:::##VALUE=DURATION##:::##', ''); + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue(newValue)); + alarmText+=process_elem; + } + else + { + var duration=''; + var before_after=$(".before_after_input[data-id="+(t+1)+"]").val(); + if($(".alert_message_details[data-id="+(t+1)+"]").val()=='minutes_before') + duration="-PT"+before_after+"M"; + if($(".alert_message_details[data-id="+(t+1)+"]").val()=='hours_before') + duration="-PT"+before_after+"H"; + if($(".alert_message_details[data-id="+(t+1)+"]").val()=='days_before') + duration="-P"+before_after+"D"; + if($(".alert_message_details[data-id="+(t+1)+"]").val()=='weeks_before') + duration="-P"+before_after+"W"; + if($(".alert_message_details[data-id="+(t+1)+"]").val()=='seconds_before') + duration="-PT"+before_after+"S"; + if($(".alert_message_details[data-id="+(t+1)+"]").val()=='minutes_after') + duration="PT"+before_after+"M"; + if($(".alert_message_details[data-id="+(t+1)+"]").val()=='hours_after') + duration="PT"+before_after+"H"; + if($(".alert_message_details[data-id="+(t+1)+"]").val()=='days_after') + duration="P"+before_after+"D"; + if($(".alert_message_details[data-id="+(t+1)+"]").val()=='weeks_after') + duration="P"+before_after+"W"; + if($(".alert_message_details[data-id="+(t+1)+"]").val()=='seconds_after') + duration="PT"+before_after+"S"; + if(vCalendar.tplM['contentline_TRIGGER']!=null && (process_elem=vCalendar.tplM['contentline_TRIGGER'][0])!=undefined) + { + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_TRIGGER']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##VALUE=DATE-TIME##:::##', ''); + process_elem=process_elem.replace('##:::##VALUE=DURATION##:::##', ';VALUE=DURATION'); + process_elem=process_elem.replace('##:::##value##:::##', duration); + alarmText+=process_elem; + } + + if(vCalendar.tplM['contentline_ACTION']!=null && (process_elem=vCalendar.tplM['contentline_ACTION'][0])!=undefined) + { + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_ACTION']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue('DISPLAY')); + alarmText+=process_elem; + var a=new Date(); + + if(vCalendar.tplM['contentline_DESCRIPTION']!=null && (process_elem=vCalendar.tplM['contentline_DESCRIPTION'][0])!=undefined) + { + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_DESCRIPTION']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue('Reminder')); + alarmText+=process_elem; + + } + if((typeof vCalendar.tplM['unprocessedVALARM']!='undefined' && typeof vCalendar.tplM['unprocessedVALARM'][t]!='undefined') && (vCalendar.tplM['unprocessedVALARM'][t]!='') && (vCalendar.tplM['unprocessedVALARM'][t]!=null)) + { + tmp=vCalendar.tplM['unprocessedVALARM'][t].replace(RegExp('^\r\n'), '').replace(RegExp('\r\n$'), ''); + if(tmp.indexOf('\r\n')==0) + tmp=tmp.substring(2, tmp.length); + if(tmp.lastIndexOf('\r\n')!=(tmp.length-2)) + tmp+='\r\n'; + alarmText+=tmp; + } + if(vCalendar.tplM['endVALARM']!=null && (process_elem=vCalendar.tplM['endVALARM'][0])!=undefined) + alarmText+=vCalendar.tplM['endVALARM'][0]; + else + { + process_elem=vCalendar.tplC['endVALARM']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + alarmText+=process_elem; + } + if(alarmUniqueArray.indexOf(alarmText)==-1) + { + alarmUniqueArray.push(alarmText); + vCalendarText+=alarmText; + } + } + } + } + vCalendar.tplM['unprocessedVALARM']=new Array(); + + if($('#avail').val()!='none') + { + if(vCalendar.tplM['contentline_TRANSP']!=null && (process_elem=vCalendar.tplM['contentline_TRANSP'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_TRANSP']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + if($('#avail').val()=='busy') + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue('OPAQUE')); + else + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue('TRANSPARENT')); + vCalendarText+=process_elem; + } + + if($('#url_EVENT').val()!='') + { + if(vCalendar.tplM['contentline_URL']!=null && (process_elem=vCalendar.tplM['contentline_URL'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_URL']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue($('#url_EVENT').val())); + vCalendarText+=process_elem; + } + + + + } + //DESCRIPTION + if($('#note').val()!='') + { + // NOTE + if(vCalendar.tplM['contentline_NOTE']!=null && (process_elem=vCalendar.tplM['contentline_NOTE'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_NOTE']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue($('#note').val())); + vCalendarText+=process_elem; + } + + if($('#status').val()!='NONE') + { + + //if((value=$('[id="vcalendar_editor"] [data-type="\\%note"]').find('textarea').val())!='') + //{ + if(vCalendar.tplM['contentline_STATUS']!=null && (process_elem=vCalendar.tplM['contentline_STATUS'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_STATUS']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue($('#status').val())); + vCalendarText+=process_elem; + } + + //CLASS + if($('#type').val()!='') + { + // CLASS + if(vCalendar.tplM['contentline_CLASS']!=null && (process_elem=vCalendar.tplM['contentline_CLASS'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_CLASS']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + if(typeof vCalendar.tplM['contentline_CLASS'] =='undefined' || vCalendar.tplM['contentline_CLASS']==null || vCalendar.tplM['contentline_CLASS'].length==0) + process_elem=''; + } + + if($('.row_type').css('display')!='none') + { + process_elem=vCalendar.tplC['contentline_CLASS']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue($('#type').val().toUpperCase())); + } + vCalendarText+=process_elem; + } + + //RECURRENCE-ID + if($('#recurrenceID').val()) + { + if(vCalendar.tplM['contentline_REC_ID']!=null && (process_elem=vCalendar.tplM['contentline_REC_ID'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_REC_ID']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + + var rec_id=$('#recurrenceID').val() + if(rec_id.indexOf('T')==-1) + { + process_elem=process_elem.replace('##:::##AllDay##:::##', ';'+vcalendarEscapeValue('VALUE=DATE')); + process_elem=process_elem.replace('##:::##TZID##:::##', vcalendarEscapeValue('')); + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue(rec_id)); + } + else + { + process_elem=process_elem.replace('##:::##AllDay##:::##', vcalendarEscapeValue('')); + + /*if((typeof vCalendar.tplM['unprocessed']!='undefined') && (vCalendar.tplM['unprocessed']!='') && (vCalendar.tplM['unprocessed']!=null)) + { + var checkTZID=vCalendar.tplM['unprocessed'].match(vCalendar.pre['contentline_TZID']); + if(checkTZID!=null) + { + parsed=checkTZID[0].match(vCalendar.pre['contentline_parse']); + process_elem=process_elem.replace('##:::##TZID##:::##', ';'+vcalendarEscapeValue("TZID="+parsed[4])); + } + else + process_elem=process_elem.replace('##:::##TZID##:::##', ';'+vcalendarEscapeValue("TZID="+ sel_option)); + } + else*/ + + process_elem=process_elem.replace('##:::##TZID##:::##',timeZoneAttr); + if(isUTC && rec_id.charAt(rec_id.length-1)!='Z') + rec_id+='Z'; + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue(rec_id)); + } + vCalendarText+=process_elem; + } + + if(vCalendar.tplM['contentline_E_DTSTART']!=null && (process_elem=vCalendar.tplM['contentline_E_DTSTART'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCalendar.re['group']+'\\.)?)', 'm')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.', '\\.'), 'mg'), '\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCalendar.tplC['contentline_E_DTSTART']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + + var datetime_from='', datetime_to=''; + var a=$.datepicker.parseDate(globalSettings.datepickerformat.value, $('#date_from').val()); + var a2=$.datepicker.parseDate(globalSettings.datepickerformat.value, $('#date_to').val()); + var b=new Date(1970, 1, 1, 0, 0, 0); + if(datetime_from=='') + datetime_from=$.fullCalendar.formatDate(a, 'yyyyMMdd'); + + if(datetime_to=='') + datetime_to=$.fullCalendar.formatDate(a2, 'yyyyMMdd'); + + var dateTo=$.datepicker.parseDate('yymmdd',datetime_to); + + if($('#allday').prop('checked')) + { + process_elem=process_elem.replace('##:::##AllDay##:::##', ';'+vcalendarEscapeValue('VALUE=DATE')); + process_elem=process_elem.replace('##:::##TZID##:::##', vcalendarEscapeValue('')); + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue(datetime_from)); + } + else + { + b=new Date(Date.parse("01/02/1990, "+$('#time_from').val() )); + var time_from=$.fullCalendar.formatDate(b, 'HHmmss'); + process_elem=process_elem.replace('##:::##AllDay##:::##', vcalendarEscapeValue('')); + + /*if((typeof vCalendar.tplM['unprocessed']!='undefined') && (vCalendar.tplM['unprocessed']!='') && (vCalendar.tplM['unprocessed']!=null)) + { + var checkTZID=vCalendar.tplM['unprocessed'].match(vCalendar.pre['contentline_TZID']); + if(checkTZID!=null) + { + parsed=checkTZID[0].match(vCalendar.pre['contentline_parse']); + process_elem=process_elem.replace('##:::##TZID##:::##', ';'+vcalendarEscapeValue("TZID="+parsed[4])); + } + else + process_elem=process_elem.replace('##:::##TZID##:::##', ';'+vcalendarEscapeValue("TZID="+ sel_option)); + } + else*/ + process_elem=process_elem.replace('##:::##TZID##:::##', timeZoneAttr); + process_elem=process_elem.replace('##:::##value##:::##', vcalendarEscapeValue(datetime_from+'T'+time_from+(isUTC ? 'Z' : ''))); + } + + vCalendarText+=process_elem; + + if(realEvent!='') + { + if(realEvent.type!='') + { + var repeatStart=realEvent.repeatStart; + a.setHours(b.getHours()); + a.setMinutes(b.getMinutes()); + a.setSeconds(b.getSeconds()); + var changeDate=a; + var offsetDate=changeDate-repeatStart; + var realEventUID=realEvent.vcalendar.match(vCalendar.pre['contentline_UID']); + + if(realEventUID!=null) + realEventUID=realEventUID[0].match(vCalendar.pre['contentline_parse'])[4]; + + if(offsetDate!=0) + { + var vcalendarOrig=vCalendarText; + var eventArray=new Array(),backupEventArray= new Array(); + while(vcalendarOrig.match(vCalendar.pre['vevent'])!=null) + { + if(vcalendarOrig.substring(vcalendarOrig.indexOf('BEGIN:VEVENT')-2, vcalendarOrig.indexOf('BEGIN:VEVENT'))=='\r\n') + { + var partEvent=vcalendarOrig.substring(vcalendarOrig.indexOf('BEGIN:VEVENT')-2,vcalendarOrig.indexOf('END:VEVENT')+'END:VEVENT'.length); + vcalendarOrig=vcalendarOrig.replace(partEvent, ''); + } + else + { + var partEvent=vcalendarOrig.substring(vcalendarOrig.indexOf('BEGIN:VEVENT'),vcalendarOrig.indexOf('END:VEVENT')+'END:VEVENT'.length); + vcalendarOrig=vcalendarOrig.replace(partEvent, ''); + partEvent+='\r\n'; + } + eventArray[eventArray.length]=partEvent; + backupEventArray[backupEventArray.length]=partEvent; + } + if(eventArray.length==0) + console.log("Error: '"+inputUID+"': unable to parse vEvent"); + + for(var it=0;it"name", [3]->";param;param", [4]->"value" + if((parsed=('\r\n'+vcalendar_full[vcalendar_full.length-2]+'\r\n').match(vCalendar.pre['contentline_parse']))==null) + return false; + // values not directly supported by the editor (old values are kept intact) + vCalendar.tplM['end'][0]=vCalendar.tplC['end'].replace(/##:::##group_wd##:::##/g, vcalendar_end_group=parsed[1]); + + if(vcalendar_begin_group!=vcalendar_end_group) + return false;// the vCalendar BEGIN and END "group" are different + // remove the vCalendar BEGIN and END + + vcalendar='\r\n'+vcalendar_full.slice(1, vcalendar_full.length-2).join('\r\n')+'\r\n'; + + /* + vcalendar_element=vcalendar.match(vCalendar.pre['tzone']); + if(vcalendar_element!=null) + vcalendar=vcalendar.replace(vcalendar_element[0],''); + */ + + //FIX TIMEZONE + var beginTimeZone=vcalendar.indexOf('BEGIN:VTIMEZONE'); + var startEndTimeZone=vcalendar.lastIndexOf('END:VTIMEZONE'); + var endTimeZone=0; + var vTimeZone=''; + + if(beginTimeZone!=-1 && startEndTimeZone!=-1) + { + for(i=(startEndTimeZone+2);i"group.", [2]->"name", [3]->";param;param", [4]->"value" + if((parsed=('\r\n'+vcalendar_full[vevent_full.length-2]+'\r\n').match(vCalendar.pre['contentline_parse']))==null) + return false; + // values not directly supported by the editor (old values are kept intact) + vCalendar.tplM['endVEVENT'][0]=vCalendar.tplC['endVEVENT'].replace(/##:::##group_wd##:::##/g, vcalendar_end_group=parsed[1]); + + if(vcalendar_begin_group!=vcalendar_end_group) + return false;// the vCalendar BEGIN and END "group" are different + + // remove the vCalendar BEGIN and END + + vevent='\r\n'+vevent_full.slice(2, vevent_full.length-1).join('\r\n')+'\r\n'; + //SUMMARY + vcalendar_element=vevent.match(RegExp('\r\n'+vCalendar.re['contentline_SUMMARY'], 'mi')); + if(vcalendar_element!=null) + { + parsed=vcalendar_element[0].match(vCalendar.pre['contentline_parse']); + //note=String(vcalendar_element).split(':')[1]; + title=vcalendarUnescapeValue(parsed[4]); + vCalendar.tplM['contentline_SUMMARY'][0]=vCalendar.tplC['contentline_SUMMARY']; + vCalendar.tplM['contentline_SUMMARY'][0]=vCalendar.tplM['contentline_SUMMARY'][0].replace(/##:::##group_wd##:::##/g, parsed[1]); + vCalendar.tplM['contentline_SUMMARY'][0]=vCalendar.tplM['contentline_SUMMARY'][0].replace(/##:::##params_wsc##:::##/g, parsed[3]); + vevent=vevent.replace(vcalendar_element[0], '\r\n'); + + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while ((vcalendar_element_related=vevent.match(re))!=null) + { + // append the parameter to its parent + vCalendar.tplM['contentline_SUMMARY'][0]+=vcalendar_element_related[0].substr(2); + // remove the processed parameter + vevent=vevent.replace(vcalendar_element_related[0], '\r\n'); + } + } + } + + vcalendar_element=vevent.match(RegExp('\r\n'+vCalendar.re['contentline_TRANSP'], 'mi')); + if(vcalendar_element!=null) + { + parsed=vcalendar_element[0].match(vCalendar.pre['contentline_parse']); + //note=String(vcalendar_element).split(':')[1]; + title=vcalendarUnescapeValue(parsed[4]); + vCalendar.tplM['contentline_TRANSP'][0]=vCalendar.tplC['contentline_TRANSP']; + vCalendar.tplM['contentline_TRANSP'][0]=vCalendar.tplM['contentline_TRANSP'][0].replace(/##:::##group_wd##:::##/g, parsed[1]); + vCalendar.tplM['contentline_TRANSP'][0]=vCalendar.tplM['contentline_TRANSP'][0].replace(/##:::##params_wsc##:::##/g, parsed[3]); + vevent=vevent.replace(vcalendar_element[0], '\r\n'); + + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while ((vcalendar_element_related=vevent.match(re))!=null) + { + // append the parameter to its parent + vCalendar.tplM['contentline_TRANSP'][0]+=vcalendar_element_related[0].substr(2); + // remove the processed parameter + vevent=vevent.replace(vcalendar_element_related[0], '\r\n'); + } + } + } + + vcalendar_element=vevent.match(RegExp('\r\n'+vCalendar.re['contentline_PRIORITY'], 'mi')); + if(vcalendar_element!=null) + { + parsed=vcalendar_element[0].match(vCalendar.pre['contentline_parse']); + + //note=String(vcalendar_element).split(':')[1]; + title=vcalendarUnescapeValue(parsed[4]); + + vCalendar.tplM['contentline_PRIORITY'][0]=vCalendar.tplC['contentline_PRIORITY']; + vCalendar.tplM['contentline_PRIORITY'][0]=vCalendar.tplM['contentline_PRIORITY'][0].replace(/##:::##group_wd##:::##/g, parsed[1]); + vCalendar.tplM['contentline_PRIORITY'][0]=vCalendar.tplM['contentline_PRIORITY'][0].replace(/##:::##params_wsc##:::##/g, parsed[3]); + + vevent=vevent.replace(vcalendar_element[0], '\r\n'); + + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while ((vcalendar_element_related=vevent.match(re))!=null) + { + // append the parameter to its parent + vCalendar.tplM['contentline_PRIORITY'][0]+=vcalendar_element_related[0].substr(2); + // remove the processed parameter + vevent=vevent.replace(vcalendar_element_related[0], '\r\n'); + } + } + } + + //LOCATION + vcalendar_element=vevent.match(RegExp('\r\n'+vCalendar.re['contentline_LOCATION'], 'mi')); + if(vcalendar_element!=null) + { + parsed=vcalendar_element[0].match(vCalendar.pre['contentline_parse']); + //note=String(vcalendar_element).split(':')[1]; + title=vcalendarUnescapeValue(parsed[4]); + vCalendar.tplM['contentline_LOCATION'][0]=vCalendar.tplC['contentline_LOCATION']; + vCalendar.tplM['contentline_LOCATION'][0]=vCalendar.tplM['contentline_LOCATION'][0].replace(/##:::##group_wd##:::##/g, parsed[1]); + vCalendar.tplM['contentline_LOCATION'][0]=vCalendar.tplM['contentline_LOCATION'][0].replace(/##:::##params_wsc##:::##/g, parsed[3]); + vevent=vevent.replace(vcalendar_element[0], '\r\n'); + + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while ((vcalendar_element_related=vevent.match(re))!=null) + { + // append the parameter to its parent + vCalendar.tplM['contentline_LOCATION'][0]+=vcalendar_element_related[0].substr(2); + // remove the processed parameter + vevent=vevent.replace(vcalendar_element_related[0], '\r\n'); + } + } + } + + //URL + vcalendar_element=vevent.match(RegExp('\r\n'+vCalendar.re['contentline_URL'], 'mi')); + if(vcalendar_element!=null) + { + parsed=vcalendar_element[0].match(vCalendar.pre['contentline_parse']); + //note=String(vcalendar_element).split(':')[1]; + title=vcalendarUnescapeValue(parsed[4]); + vCalendar.tplM['contentline_URL'][0]=vCalendar.tplC['contentline_URL']; + vCalendar.tplM['contentline_URL'][0]=vCalendar.tplM['contentline_URL'][0].replace(/##:::##group_wd##:::##/g, parsed[1]); + vCalendar.tplM['contentline_URL'][0]=vCalendar.tplM['contentline_URL'][0].replace(/##:::##params_wsc##:::##/g, parsed[3]); + vevent=vevent.replace(vcalendar_element[0], '\r\n'); + + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while ((vcalendar_element_related=vevent.match(re))!=null) + { + // append the parameter to its parent + vCalendar.tplM['contentline_URL'][0]+=vcalendar_element_related[0].substr(2); + // remove the processed parameter + vevent=vevent.replace(vcalendar_element_related[0], '\r\n'); + } + } + } + + // ------------------------------ VALARM ------------------------------ // + var valarm=vevent.match(vCalendar.pre['valarm']); + if(valarm!=null) + { + vevent=vevent.replace(valarm[0], ''); + var alarmString=''; + var alarmArray=new Array(); + + for(var i=0;i"group.", [2]->"name", [3]->";param;param", [4]->"value" + if((parsed=('\r\n'+valarm_full[valarm_full.length-2]+'\r\n').match(vCalendar.pre['contentline_parse']))==null) + return false; + + // values not directly supported by the editor (old values are kept intact) + vCalendar.tplM['endVALARM'][j]=vCalendar.tplC['endVALARM'].replace(/##:::##group_wd##:::##/g, vcalendar_end_group=parsed[1]); + + if(vcalendar_begin_group!=vcalendar_end_group) + return false;// the vCalendar BEGIN and END "group" are different + + // remove the vCalendar BEGIN and END + alarmArray[j]='\r\n'+valarm_full.slice(1, valarm_full.length-2).join('\r\n')+'\r\n'; + + trigger=alarmArray[j].match(vCalendar.pre['contentline_TRIGGER']); + + if(trigger!=null) + { + parsed=(trigger[0]+'\r\n').match(vCalendar.pre['contentline_parse']); + + vCalendar.tplM['contentline_TRIGGER'][j]=vCalendar.tplC['contentline_TRIGGER']; + vCalendar.tplM['contentline_TRIGGER'][j]=vCalendar.tplM['contentline_TRIGGER'][j].replace(/##:::##group_wd##:::##/g, parsed[1]); + var pars=vcalendarSplitParam(parsed[3]); + var parString=''; + for(var i=0;i0) + { + var daysUntilDaylight=(parseInt(daylightStartCount)-1)*7; + var dayLightStartDate=new Date(checkDate.getFullYear(), checkDate.getMonth(), checkDate.getDate()+daysUntilDaylight, daylightStartsHours, daylightStartsMinutes); + } + else + { + var tmpLastDay=21+checkDate.getDate(); + var checkTmpDay=new Date(t.getFullYear(),daylightStartsMonth,tmpLastDay+7,23,59,0); + + if(checkTmpDay.getMonth()!=daylightStartsMonth) + var lastDay=tmpLastDay; + else + var lastDay=tmpLastDay+7; + + var daysUntilDaylight=(daylightStartCount+1)*7; + var dayLightStartDate=new Date(checkDate.getFullYear(), checkDate.getMonth(), lastDay+daysUntilDaylight, daylightStartsHours, daylightStartsMinutes); + } + + if(dayLightStartDate>t && !disableRecursion) + dayLightStartDate=getDateFromDay(objComponent, t, true).startDate; + + return {offsetFrom:objComponent.tzOffsetFROM, offsetTo: objComponent.tzOffsetTO, startDate: dayLightStartDate}; +} + + function vcalendarToData(inputCollection, inputEvent, isNew) +{ + var vcalendarOrig=inputEvent.vcalendar; + var eventArray=new Array(); + + //CHECK CALSCALE + var elem=vcalendarOrig.match(vCalendar.pre['contentline_CALSCALE']); + if(elem!=null) + { + calscale=elem[0].match(vCalendar.pre['contentline_parse'])[4]; + if(calscale!='GREGORIAN') + { + console.log("Error:'"+inputEvent.uid+"': Unsupported calscale in:"+vcalendarOrig); + return false; + } + } + //CHECK VERSION + var elemV=vcalendarOrig.match(vCalendar.pre['contentline_VERSION']); + if(elemV!=null) + { + var ver=elemV[0].match(vCalendar.pre['contentline_parse'])[4]; + if(ver!='2.0') + { + console.log("Error:'"+inputEvent.uid+"': Unsupported version ("+ver+") in:"+vcalendarOrig); + return false; + } + } + + //FIX TIMEZONE + var beginTimeZone=vcalendarOrig.indexOf('BEGIN:VTIMEZONE'); + var startEndTimeZone=vcalendarOrig.lastIndexOf('END:VTIMEZONE'); + var endTimeZone=0; + + var rid=inputEvent.uid.substring(0, inputEvent.uid.lastIndexOf('/')+1); + var evid=inputEvent.uid.substring(inputEvent.uid.lastIndexOf('/')+1, inputEvent.uid.length); + + var isChange=false, + needReload=false; + + if(!isNew) + { + var events=findEventInArray(inputEvent.uid, true); + if(events!='') + { + if(events.etag!=inputEvent.etag) + { + for(var i=0; i'+name+" "+localization[globalInterfaceLanguage].repeatChangeTxt); + $('#repeatConfirmBoxQuestion').html(localization[globalInterfaceLanguage].repeatChangeTxtClose); + } + else + needReload=true; + } + } + isChange=true; + } + } + } + + if((beginTimeZone!=-1) && (startEndTimeZone!=-1)) + { + for(i=(startEndTimeZone+2);i0) + isRepeat=true; + for(var i=0;i1 &&(frequency=='MONTHLY'||frequency=='YEARLY')) + { + console.log("Error:'"+inputEvent.uid+"': Unsupported recurrence rule in event:"+vcalendar); + return false; + } + } + } + if(!returnForValue) + { + + continue; + } + if(!interval) + interval=1; + } + + var dayLightStartDate, dayLightEndDate, tzObject; + vcalendar_element=vcalendar.match(vCalendar.pre['contentline_DTSTART']); + if(vcalendar_element!=null) + { + parsed=vcalendar_element[0].match(vCalendar.pre['contentline_parse']); + + start=parsed[4]; + var help1=start; + + if(help1.indexOf("T")==-1) + { + help1=help1.substring(0, 4)+'-'+help1.substring(4, 6)+'-'+help1.substring(6, 8); + all=true; + } + else + { + help1=help1.substring(0, 4)+'-'+help1.substring(4, 6)+'-'+help1.substring(6, 8)+'T'+help1.substring(9, 11)+':'+help1.substring(11, 13)+':'+help1.substring(13, 15); + all=false; + } + + var t=$.fullCalendar.parseDate(help1); + if(t==null) + return false; + if(t.toString()=='Invalid Date') + return false; + + if(!all) + { + parsed_value=vcalendarSplitParam(parsed[3]); + for(h=1;h1 || tzName=='UTC') + { + if(tzName!='UTC') + tzName=$.trim(dtStartTimezone[1]); + var finTZ = checkTimezone(tzName); + if(finTZ!=null) + tzName = finTZ; + if(globalSettings.timezonesupport.value && tzName in timezones) + { + valOffsetFrom=getOffsetByTZ(tzName, t); + intOffset=(getLocalOffset(t)*-1*1000)-valOffsetFrom.getSecondsFromOffset()*1000; + } + } + else if(processedTimezones.indexOf(tzName)==-1) + { + if(timeZonesEnabled.indexOf(tzName)==-1) + timeZonesEnabled.push('local'); + processedTimezones.push('local'); + } + if(tzName!='' && tzName != 'local') + if(processedTimezones.indexOf(tzName)==-1) + { + if(timeZonesEnabled.indexOf(tzName)==-1) + timeZonesEnabled.push(tzName); + processedTimezones.push(tzName); + } + } + else + tzName = globalSessionTimeZone; + realStart=$.fullCalendar.parseDate(help1); + inputEvent.start=$.fullCalendar.parseDate(help1); + start=$.fullCalendar.parseDate(help1); + if(intOffset) + { + inputEvent.start.setTime(inputEvent.start.getTime()+intOffset); + start.setTime(start.getTime()+intOffset); + } + if(exDate_array!=null) + for(var j=0;j1 || tzNameA=='UTC') + { + if(tzNameA!='UTC' && dtStartTimezoneA[0]==';TZID') + tzNameA=$.trim(dtStartTimezoneA[1]); + var finTZ = checkTimezone(tzNameA); + if(finTZ!=null) + tzNameA = finTZ; + if(globalSettings.timezonesupport.value && tzNameA in timezones) + { + var valOffsetFromA=getOffsetByTZ(tzNameA, alarmTimeA); + intOffsetA=getOffsetByTZ(tzName, alarmTimeA).getSecondsFromOffset()*1000-valOffsetFromA.getSecondsFromOffset()*1000; + } + } + else if(processedTimezones.indexOf(tzName)==-1) + { + if(timeZonesEnabled.indexOf(tzName)==-1) + timeZonesEnabled.push('local'); + processedTimezones.push('local'); + } + if(tzNameA!='' && tzNameA != 'local') + if(processedTimezones.indexOf(tzNameA)==-1) + { + if(timeZonesEnabled.indexOf(tzNameA)==-1) + timeZonesEnabled.push(tzNameA); + processedTimezones.push(tzNameA); + } + if(intOffsetA!='') + alarmTimeA.setTime(alarmTimeA.getTime()+intOffsetA); + alertTime[j]=$.fullCalendar.formatDate(alarmTimeA,"yyyy-MM-dd'T'HH:mm:ss"); + } + else + { + alertTime[j]=0; + if(value.indexOf('W')!=-1) + alertTime[j]=parseAlarmWeek(value); + else if(value.indexOf('D')!=-1) + alertTime[j]=parseAlarmDay(value); + else if(value.indexOf('T')!=-1) + alertTime[j]=parseAlarmTime(value); + if(parsed[4].charAt(0)=="-") + alertTime[j]="-"+alertTime[j]; + else + alertTime[j]="+"+alertTime[j]; + } + } + } + else + break; + + noteA=alarmArray[j].match(vCalendar.pre['contentline_NOTE']); + if(noteA!=null) + { + parsed=noteA[0].match(vCalendar.pre['contentline_parse']); + alertNote[j]=parsed[4]; + } + else + alertNote[j]='Default note'; + } + } + } + + vcalendar_element=vcalendar.match(vCalendar.pre['contentline_LOCATION']); + if(vcalendar_element!=null) + { + parsed=vcalendar_element[0].match(vCalendar.pre['contentline_parse']); + location=vcalendarUnescapeValue(parsed[4]); + } + + vcalendar_element=vcalendar.match(vCalendar.pre['contentline_NOTE']); + if(vcalendar_element!=null) + { + parsed=vcalendar_element[0].match(vCalendar.pre['contentline_parse']); + note=vcalendarUnescapeValue(parsed[4]); + } + + vcalendar_element=vcalendar.match(vCalendar.pre['contentline_SUMMARY']); + if(vcalendar_element!=null) + { + parsed=vcalendar_element[0].match(vCalendar.pre['contentline_parse']); + title=vcalendarUnescapeValue(parsed[4]); + } + + vcalendar_element=vcalendar.match(vCalendar.pre['contentline_PRIORITY']); + if(vcalendar_element!=null) + { + parsed=vcalendar_element[0].match(vCalendar.pre['contentline_parse']); + priority=vcalendarUnescapeValue(parsed[4]); + } + + var index=0; + for(var p=0;p1 || tzName=='UTC') + { + if(tzName!='UTC') + tzName=$.trim(dtStartTimezone[1]); + var finTZ = checkTimezone(tzName); + if(finTZ!=null) + tzName = finTZ; + if(globalSettings.timezonesupport.value && tzName in timezones) + { + valOffsetFrom=getOffsetByTZ(tzName, t1); + intOffset=(getLocalOffset(t1)*-1*1000)-valOffsetFrom.getSecondsFromOffset()*1000; + } + } + else if(processedTimezones.indexOf(tzName)==-1) + { + if(timeZonesEnabled.indexOf(tzName)==-1) + timeZonesEnabled.push('local'); + processedTimezones.push('local'); + } + //realEnd=$.fullCalendar.parseDate(help); + //help1+=valOffsetFrom; + + if(tzName!='' && tzName != 'local') + if(processedTimezones.indexOf(tzName)==-1) + { + if(timeZonesEnabled.indexOf(tzName)==-1) + timeZonesEnabled.push(tzName); + processedTimezones.push(tzName); + } + } + else + tzName = globalSessionTimeZone; + + realEnd=$.fullCalendar.parseDate(help); + inputEvent.end=$.fullCalendar.parseDate(help); + end=$.fullCalendar.parseDate(help); + if(intOffset) + { + inputEvent.end.setTime(inputEvent.end.getTime()+intOffset); + end.setTime(end.getTime()+intOffset); + } + } + else + return false; + + if(globalVisibleCalDAVCollections.indexOf(rid)!=-1 || isChange || isNew) + { + if(isRepeat) + { + var futureRLimit = new Date(globalToLoadedLimit.getTime()) + futureRLimit.setDate(futureRLimit.getDate()+14); + var ruleString=vcalendar.match(vCalendar.pre['contentline_RRULE2'])[0].match(vCalendar.pre['contentline_parse'])[4]; + inputEvent.isRepeat=true; + if(realStart) + var varDate=new Date(realStart.getTime()); + else + var varDate=new Date(start.getTime()); + + if(realEnd) + var varEndDate=new Date(realEnd.getTime()); + else + var varEndDate=new Date(end.getTime()); + + var lastGenDate=''; + var repeatStart=new Date(varDate.getTime()); + var repeatEnd=new Date(varEndDate.getTime()); + var untilDate='',realUntilDate='',realUntil=''; + + if(until!=='') + { + if(isUntilDate) + { + if(until.indexOf('T')!=-1) + { + var uString = until.substring(0, 4)+'-'+until.substring(4, 6)+'-'+until.substring(6, 8)+'T'+until.substring(9, 11)+':'+until.substring(11, 13)+':'+until.substring(13, 15); + var ut=$.fullCalendar.parseDate(uString); + if(ut==null) + return false; + if(ut.toString()=='Invalid Date') + return false; + if(!all) + { + if(globalSettings.timezonesupport.value && tzName in timezones) + valOffsetFrom=getOffsetByTZ(tzName, ut); + if(valOffsetFrom) + { + var intOffset=valOffsetFrom.getSecondsFromOffset()*1000; + ut.setTime(ut.getTime()+intOffset); + } + } + untilDate = new Date(ut.getTime()); + } + else + { + untilDate=$.fullCalendar.parseDate(until.substring(0, 4)+'-'+until.substring(4, 6)+'-'+until.substring(6, 8)); + untilDate.setHours(realStart.getHours()); + untilDate.setMinutes(realStart.getMinutes()); + untilDate.setSeconds(realStart.getSeconds()); + } + + realUntil=''; + } + else + { + untilDate=''; + realUntil=until; + + } + realUntilDate=untilDate; + inputEvent.untilDate=untilDate; + } + else + { + untilDate=new Date(futureRLimit.getTime()); + realUntilDate=''; + inputEvent.untilDate='never'; + } + var repeatCount=0, realRepeatCount=0; + + if(!inputEvent.isDrawn) + { + if(alertTime.length>0) + { + var aTime=''; + var now=new Date(); + if(!inputCollection.ignoreAlarms) + alertTimeOut=setAlertTimeouts(false,alertTime, start, end, {allDay:all, title:title}, true, inputEvent.uid); + } + realRepeatCount++; + var checkRec=isInRecurrenceArray(varDate,stringUID,recurrence_id_array, tzName); + + if(exDates.length>0) + if(exDates.indexOf(varDate.toString())!=-1) + checkRec=true; + if(!checkRec) + { + repeatCount++; + var tmpObj=new items(inputEvent.etag, start, end, title, all, inputEvent.uid, rid, evid, note, inputEvent.displayValue, alertTime, alertNote, realUntilDate, frequency, interval, realUntil, repeatStart, repeatEnd, byMonthDay,repeatCount, realRepeatCount, vcalendar, location, alertTimeOut,tzName, realStart, realEnd, byDay, rec_id,wkst,classType, avail,hrefUrl, compareString,priority,status,ruleString); + globalEventList.displayEventsArray[rid].splice(globalEventList.displayEventsArray[rid].length, 0, tmpObj); + } + } + + var lastGenDate=generateRepeatInstances({ + untilDate:realUntilDate, + repeatStart:varDate, + futureRLimit:futureRLimit, + stringUID:stringUID, + recurrence_id_array:recurrence_id_array, + exDates:exDates, + alertTime:alertTime, + ignoreAlarms:inputCollection.ignoreAlarms, + items:new items(inputEvent.etag, varDate, varEndDate, title, all, inputEvent.uid, rid, evid, note, inputEvent.displayValue, alertTime, alertNote, realUntilDate, frequency, interval, realUntil, repeatStart, repeatEnd, byMonthDay, repeatCount, realRepeatCount, vcalendar, location, alertTimeOut, tzName, realStart, realEnd, byDay, rec_id,wkst,classType, avail,hrefUrl,compareString,priority,status,ruleString) + }); + } + else + { + if(!inputCollection.ignoreAlarms) + alertTimeOut=setAlertTimeouts(false,alertTime, start, end, {allDay:all, title:title},true,inputEvent.uid); + + var tmpObj=new items(inputEvent.etag, start, end, title, all, inputEvent.uid, rid, evid, note, inputEvent.displayValue, alertTime, alertNote, '', '', '', '', '', '', '', '', '', vcalendar, location, alertTimeOut, tzName, realStart, realEnd, byDay, rec_id,wkst,classType, avail,hrefUrl,compareString,priority,status,ruleString); + if(isChange) + { + if(needReload) + showEventForm(null, null, tmpObj, globalJsEvent, 'show', ''); + } + globalEventList.displayEventsArray[rid].splice(globalEventList.displayEventsArray[rid].length, 0, tmpObj); + } + } + } + inputEvent.isDrawn=true; +} + +function notRFCDataToRFCData(vcalendarString) +{ + // If vCalendar contains only '\n' instead of '\r\n' we correct this + if(vcalendarString.match(RegExp('\r', 'm'))==null) + vcalendarString=vcalendarString.replace(RegExp('\n', 'gm'), '\r\n'); + + // remove multiple empty lines + vcalendarString=vcalendarString.replace(RegExp('(\r\n)+','gm'),'\r\n'); + + // remove line folding + vcalendarString=vcalendarString.replace(RegExp('\r\n'+vCalendar.re['WSP'], 'gm'), ''); + + // append '\r\n' to the end of the vCalendar if missing + if(vcalendarString[vcalendarString.length-1]!='\n') + vcalendarString+='\r\n'; + + return vcalendarString; +} + +function vCalendarCleanup(vcalendarString) +{ + vcalendarString=notRFCDataToRFCData(vcalendarString); + return vcalendarString; +} +function dataToVcard(accountUID, inputUID, inputFilterUID, inputEtag) +{ + var vCardText=''; + var groupCounter=0; + var tmpvCardEditorRef=$('#vCardEditor'); + if(typeof globalDisabledContactAttributes=='undefined' || !(globalDisabledContactAttributes instanceof Array)) + globalDisabledContactAttributes=[]; + + // vCard BEGIN (required by RFC) + if(vCard.tplM['begin']!=null && (process_elem=vCard.tplM['begin'][0])!=undefined) + vCardText+=vCard.tplM['begin'][0]; + else + { + process_elem=vCard.tplC['begin']; + process_elem=process_elem.replace('##:::##group_wd##:::##',''); + vCardText+=process_elem; + } + +// VERSION (required by RFC) + if(vCard.tplM['contentline_VERSION']!=null && (process_elem=vCard.tplM['contentline_VERSION'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCard.tplC['contentline_VERSION']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##version##:::##', '3.0'); + } + vCardText+=process_elem; + +// UID (required by RFC) + var newUID=''; + if(vCard.tplM['contentline_UID']!=null && (process_elem=vCard.tplM['contentline_UID'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCard.tplC['contentline_UID']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + + newUID=globalAddressbookList.getNewUID(); + + // it is VERY small probability, that for 2 newly created contacts the same UID is generated (but not impossible :( ...) + process_elem=process_elem.replace('##:::##uid##:::##',newUID); + } + vCardText+=process_elem; + +// N (required by RFC) + if(vCard.tplM['contentline_N']!=null && (process_elem=vCard.tplM['contentline_N'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCard.tplC['contentline_N']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##family##:::##',vcardEscapeValue(tmpvCardEditorRef.find('[data-type="family"]').val())); + process_elem=process_elem.replace('##:::##given##:::##',vcardEscapeValue(tmpvCardEditorRef.find('[data-type="given"]').val())); + process_elem=process_elem.replace('##:::##middle##:::##',vcardEscapeValue(tmpvCardEditorRef.find('[data-type="middle"]').val())); + process_elem=process_elem.replace('##:::##prefix##:::##',vcardEscapeValue(tmpvCardEditorRef.find('[data-type="prefix"]').val())); + process_elem=process_elem.replace('##:::##suffix##:::##',vcardEscapeValue(tmpvCardEditorRef.find('[data-type="suffix"]').val())); + vCardText+=process_elem; + +// FN (extracted from newly created N [previous "process_elem"], required by RFC) + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=('\r\n'+process_elem).match(vCard.pre['contentline_parse']); + // parsed_value = [0]->Family, [1]->Given, [2]->Middle, [3]->Prefix, [4]->Suffix + parsed_value=vcardSplitValue(parsed[4],';'); + +// XXX toto je blbost, v settingsoch predsa musi byt jednoznacne ci sa uklada format A alebo B + /* backward compatibility for stupid users (remove it in future) */ + if(typeof globalSettings.contactstorefn.value=='string') + var tmp=globalSettings.contactstorefn.value.replace(RegExp(',', 'g'),', ').split(','); + else /* new configuration options (arrays) */ + var tmp=globalSettings.contactstorefn.value.slice(); // copy the configuration array + + var first_found=false; + for(var i=0;i we use the company name as FN + fn_value=vcardEscapeValue(tmpvCardEditorRef.find('[data-type="org"]').val()); + + if(vCard.tplM['contentline_FN']!=null && (process_elem=vCard.tplM['contentline_FN'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCard.tplC['contentline_FN']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##fn##:::##',fn_value); + vCardText+=process_elem; + +// CATEGORIES + if(globalDisabledContactAttributes.indexOf('CATEGORIES')==-1 && (value=tmpvCardEditorRef.find('[data-type="\\%categories"]').find('input[data-type="value"]').val())!='') + { + if(vCard.tplM['contentline_CATEGORIES']!=null && (process_elem=vCard.tplM['contentline_CATEGORIES'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCard.tplC['contentline_CATEGORIES']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##', value); // we do not need to escape the value here! + vCardText+=process_elem; + } + +// NOTE + if(globalDisabledContactAttributes.indexOf('NOTE')==-1 && (value=tmpvCardEditorRef.find('[data-type="\\%note"]').find('textarea').val())!='') + { + if(vCard.tplM['contentline_NOTE']!=null && (process_elem=vCard.tplM['contentline_NOTE'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCard.tplC['contentline_NOTE']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##',vcardEscapeValue(value)); + vCardText+=process_elem; + } + +// REV + if(vCard.tplM['contentline_REV']!=null && (process_elem=vCard.tplM['contentline_REV'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCard.tplC['contentline_REV']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + } + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + var d = new Date(); + var utc=d.getUTCFullYear()+(d.getUTCMonth()+1<10 ? '0':'')+(d.getUTCMonth()+1)+(d.getUTCDate()<10 ? '0':'')+d.getUTCDate()+'T'+(d.getUTCHours()<10 ? '0':'')+d.getUTCHours()+(d.getUTCMinutes()<10 ? '0':'')+d.getUTCMinutes()+(d.getUTCSeconds()<10 ? '0':'')+d.getUTCSeconds()+'Z'; + process_elem=process_elem.replace('##:::##value##:::##', utc); + vCardText+=process_elem; + +// NICKNAME + if(globalDisabledContactAttributes.indexOf('NICKNAME')==-1 && (value=tmpvCardEditorRef.find('[data-type="nickname"]').val())!='') + { + if(vCard.tplM['contentline_NICKNAME']!=null && (process_elem=vCard.tplM['contentline_NICKNAME'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCard.tplC['contentline_NICKNAME']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##', vcardEscapeValue(value)); + vCardText+=process_elem; + } + +// X-PHONETIC-FIRST-NAME + if(globalDisabledContactAttributes.indexOf('X-PHONETIC-FIRST-NAME')==-1 && (value=tmpvCardEditorRef.find('[data-type="ph_firstname"]').val())!='') + { + if(vCard.tplM['contentline_X-PHONETIC-FIRST-NAME']!=null && (process_elem=vCard.tplM['contentline_X-PHONETIC-FIRST-NAME'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCard.tplC['contentline_X-PHONETIC-FIRST-NAME']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##',vcardEscapeValue(value)); + vCardText+=process_elem; + } + +// X-PHONETIC-LAST-NAME + if(globalDisabledContactAttributes.indexOf('X-PHONETIC-LAST-NAME')==-1 && (value=tmpvCardEditorRef.find('[data-type="ph_lastname"]').val())!='') + { + if(vCard.tplM['contentline_X-PHONETIC-LAST-NAME']!=null && (process_elem=vCard.tplM['contentline_X-PHONETIC-LAST-NAME'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCard.tplC['contentline_X-PHONETIC-LAST-NAME']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##',vcardEscapeValue(value)); + vCardText+=process_elem; + } + +// BDAY + if(globalDisabledContactAttributes.indexOf('BDAY')==-1 && (value=tmpvCardEditorRef.find('[data-type="date_bday"]').val())!='') + { + var valid=true; + try {var date=$.datepicker.parseDate(globalSettings.datepickerformat.value, value)} + catch (e) {valid=false} + + if(valid==true) + { + if(vCard.tplM['contentline_BDAY']!=null && (process_elem=vCard.tplM['contentline_BDAY'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCard.tplC['contentline_BDAY']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ';VALUE=date'); + } + + process_elem=process_elem.replace('##:::##value##:::##',vcardEscapeValue($.datepicker.formatDate('yy-mm-dd', date))); + vCardText+=process_elem; + } + } + +// X-ABDATE + if(globalDisabledContactAttributes.indexOf('X-ABDATE')==-1) + { + tmpvCardEditorRef.find('[data-type="\\%date"]').each( + function (index,element) + { + if((value=$(element).find('[data-type="date_value"]').val())!='') + { + var valid=true; + try {var date=$.datepicker.parseDate(globalSettings.datepickerformat.value, value)} + catch (e) {valid=false} + + if(valid==true) + { + incGroupCounter=false; + if(vCard.tplM['contentline_X-ABDATE']!=null && (process_elem=vCard.tplM['contentline_X-ABDATE'][$(element).attr('data-id')])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + { + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter)+'.').substring(2); + incGroupCounter=true; + } + } + else + { + process_elem=vCard.tplC['contentline_X-ABDATE']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + } + var date_value=$.datepicker.formatDate('yy-mm-dd', date); + + var tmp_type=$(element).find('[data-type="date_type"] option').filter(':selected').attr('data-type'); + /* construct the "custom" type */ + if(tmp_type==':custom') + { + var tmp_cust_value=$(element).find('[data-type="custom_value"]').val(); + var tmp_cust_value_processed=tmp_cust_value.replace(RegExp('^\\s*|\\s*$','g'),'').replaceAll(' ',' '); + // if a custom type is already defined as standard type, use the standard definition + if((tmp_cust_already_exists=$(element).find('[data-type="date_type"] option').filter(function(){return $(this).html()==tmp_cust_value_processed;}).attr('data-type'))!=undefined) + tmp_type=tmp_cust_already_exists; + else // use custom type + tmp_type=':'+tmp_cust_value+':'; + } + + params_wsc=''; + tmp_normal_types=tmp_type.replace(RegExp(':.*:','g'),',').replaceAll(',,',',').replace(RegExp('^,|,$','g'),''); + if(tmp_normal_types!='') + params_wsc=';TYPE='+vcardEscapeValue(tmp_normal_types).toUpperCase().replace(RegExp('\\\\,','g'),';TYPE='); + + process_elem=process_elem.replace('##:::##params_wsc##:::##',params_wsc); + process_elem=process_elem.replace('##:::##value##:::##',vcardEscapeValue(date_value)); + + my_related=''; + tmp_related_type=tmp_type.match(RegExp(':(.*):')); // only one element of related (X-ABLabel) is supported + + if(tmp_related_type!=null && tmp_related_type[1]!='') + my_related='X-ABLabel:'+vcardEscapeValue((dataTypes['date_store_as'][tmp_related_type[1]]!=undefined ? dataTypes['date_store_as'][tmp_related_type[1]] : tmp_related_type[1]))+'\r\n'; + + if(my_related!='') + { + incGroupCounter=true; + parsed=('\r\n'+process_elem).match(vCard.pre['contentline_parse']); + if(parsed[1]!='') // if group is present, we use it, otherwise we create a new group + process_elem+=parsed[1]+my_related; + else + process_elem='item'+groupCounter+'.'+process_elem+'item'+groupCounter+'.'+my_related; + } + + if(incGroupCounter) groupCounter++; + + if(globalSettings.compatibility.value.anniversaryOutputFormat.indexOf('other')!=-1) + { + // X-ANNIVERSARY + if(tmp_type==':_$!!$_:') + { + if(globalSettings.compatibility.value.anniversaryOutputFormat.indexOf('apple')!=-1) + vCardText+=process_elem; + process_elem='X-ANNIVERSARY;VALUE=date:'+vcardEscapeValue(date_value)+'\r\n'; + } + + } + vCardText+=process_elem; + } + } + }); + } + +// TITLE + if(globalDisabledContactAttributes.indexOf('TITLE')==-1 && (value=tmpvCardEditorRef.find('[data-type="title"]').val())!='') + { + if(vCard.tplM['contentline_TITLE']!=null && (process_elem=vCard.tplM['contentline_TITLE'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCard.tplC['contentline_TITLE']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##value##:::##',vcardEscapeValue(value)); + vCardText+=process_elem; + } + +// ORG + if(globalDisabledContactAttributes.indexOf('ORG')==-1) + { + value=tmpvCardEditorRef.find('[data-type="org"]:visible:not([readonly])').val(); + value2=tmpvCardEditorRef.find('[data-type="department"]:visible:not([readonly])').val(); + if((value!=undefined && value!='') || (value2!=undefined && value2!='')) + { + if(vCard.tplM['contentline_ORG']!=null && (process_elem=vCard.tplM['contentline_ORG'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCard.tplC['contentline_ORG']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + process_elem=process_elem.replace('##:::##units_wsc##:::##', ''); + } + process_elem=process_elem.replace('##:::##org##:::##',vcardEscapeValue(value)+(value2!=undefined && value2!='' ? ';'+vcardEscapeValue(value2) : '')); + vCardText+=process_elem; + } + } + +// X-ABShowAs + if(globalDisabledContactAttributes.indexOf('X-ABShowAs')==-1 && tmpvCardEditorRef.find('[data-type="isorg"]').prop('checked')) + { + if(vCard.tplM['contentline_X-ABShowAs']!=null && (process_elem=vCard.tplM['contentline_X-ABShowAs'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + } + else + { + process_elem=vCard.tplC['contentline_X-ABShowAs']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##params_wsc##:::##', ''); + process_elem=process_elem.replace('##:::##value##:::##', 'COMPANY'); + } + vCardText+=process_elem; + } + +// PHOTO + if(globalDisabledContactAttributes.indexOf('PHOTO')==-1 && !tmpvCardEditorRef.find('#photo').hasClass('photo_blank')) + { + var value = $('#photoURLHidden').val() || tmpvCardEditorRef.find('#photo').get(0).toDataURL('image/png'); + if(vCard.tplM['contentline_PHOTO']!=null && (process_elem=vCard.tplM['contentline_PHOTO'][0])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+(groupCounter++)+'.').substring(2); + + process_elem=process_elem.replace('##:::##value##:::##',value); + } + else + { + process_elem=vCard.tplC['contentline_PHOTO']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + process_elem=process_elem.replace('##:::##value##:::##', value); + } + + // Data URL (non-remote) will always be a binary encoded png image + if($('#photoURLHidden').val()==='') { + process_elem=process_elem.replace('##:::##params_wsc##:::##', ';ENCODING=b;TYPE=png'); + } + // For remote URL, we can't reliably determine its type, so we just append the VALUE=URI param + else { + process_elem=process_elem.replace('##:::##params_wsc##:::##', ';VALUE=URI'); + } + + vCardText+=process_elem; + } + +// ADR + if(globalDisabledContactAttributes.indexOf('ADR')==-1) + { + tmpvCardEditorRef.find('[data-type="\\%address"]').each( + function (index,element) + { + // if data is present for the selected country's address fields + var found=0; + $(element).find('[data-addr-field]').each( + function(index,element) + { + if($(element).attr('data-addr-field')!='' && $(element).attr('data-addr-field')!='country' && $(element).val()!='') + { + found=1; + return false; + } + } + ); + if(found) + { + var incGroupCounter=false; + if(vCard.tplM['contentline_ADR']!=null && (process_elem=vCard.tplM['contentline_ADR'][$(element).attr('data-id')])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + { + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+groupCounter+'.').substring(2); + incGroupCounter=true; + } + } + else + { + process_elem=vCard.tplC['contentline_ADR']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + } + + tmp_type=$(element).find('[data-type="address_type"] option').filter(':selected').attr('data-type'); + + /* construct the "custom" type */ + if(tmp_type==':custom') + { + var tmp_cust_value=$(element).find('[data-type="custom_value"]').val(); + var tmp_cust_value_processed=tmp_cust_value.replace(RegExp('^\\s*|\\s*$','g'),'').replaceAll(' ',' '); + // if a custom type is already defined as standard type, use the standard definition + if((tmp_cust_already_exists=$(element).find('[data-type="address_type"] option').filter(function(){return $(this).html()==tmp_cust_value_processed;}).attr('data-type'))!=undefined) + tmp_type=tmp_cust_already_exists; + else // use custom type + tmp_type=':'+tmp_cust_value+':'; + } + + params_wsc=''; + tmp_normal_types=tmp_type.replace(RegExp(':.*:','g'),',').replaceAll(',,',',').replace(RegExp('^,|,$','g'),''); + if(tmp_normal_types!='') + params_wsc=';TYPE='+vcardEscapeValue(tmp_normal_types).toUpperCase().replace(RegExp('\\\\,','g'),';TYPE='); + + var streetVal = $(element).find('[data-addr-field="street"]').map(function() { + var val = $(this).val(); + + if(val) { + return val; + } + }).get().join('\n'); + + process_elem=process_elem.replace('##:::##params_wsc##:::##',params_wsc); + process_elem=process_elem.replace('##:::##pobox##:::##',vcardEscapeValue($(element).find('[data-addr-field="pobox"]').val())); + process_elem=process_elem.replace('##:::##extaddr##:::##',vcardEscapeValue($(element).find('[data-addr-field="extaddr"]').val())); + process_elem=process_elem.replace('##:::##street##:::##',vcardEscapeValue(streetVal)); + process_elem=process_elem.replace('##:::##locality##:::##',vcardEscapeValue($(element).find('[data-addr-field="locality"]').val())); + process_elem=process_elem.replace('##:::##region##:::##',vcardEscapeValue($(element).find('[data-addr-field="region"]').val())); + process_elem=process_elem.replace('##:::##code##:::##',vcardEscapeValue($(element).find('[data-addr-field="code"]').val())); + process_elem=process_elem.replace('##:::##country##:::##',vcardEscapeValue($(element).find('[data-type="country_type"] option').filter(':selected').attr('data-full-name'))); + + my_related='X-ABADR:'+vcardEscapeValue($(element).find('[data-type="country_type"] option').filter(':selected').attr('data-type'))+'\r\n'; + parsed=('\r\n'+process_elem).match(vCard.pre['contentline_parse']); + if(parsed[1]!='') // if group is present, we use it, otherwise we create a new group + process_elem+=parsed[1]+my_related; + else + process_elem='item'+groupCounter+'.'+process_elem+'item'+groupCounter+'.'+my_related; + incGroupCounter=true; // we always increate the group number, because the X-ABADR is always stored + + my_related=''; + tmp_related_type=tmp_type.match(RegExp(':(.*):')); // only one element of related (X-ABLabel) is supported + + if(tmp_related_type!=null && tmp_related_type[1]!='') + my_related='X-ABLabel:'+vcardEscapeValue((dataTypes['address_type_store_as'][tmp_related_type[1]]!=undefined ? dataTypes['address_type_store_as'][tmp_related_type[1]] : tmp_related_type[1]))+'\r\n'; + + if(my_related!='') + { + incGroupCounter=true; + parsed=('\r\n'+process_elem).match(vCard.pre['contentline_parse']); + if(parsed[1]!='') // if group is present, we use it, otherwise we create a new group + process_elem+=parsed[1]+my_related; + else + process_elem='item'+groupCounter+'.'+process_elem+'item'+groupCounter+'.'+my_related; + } + + if(incGroupCounter) groupCounter++; + vCardText+=process_elem; + } + } + ); + } + +// TEL + if(globalDisabledContactAttributes.indexOf('TEL')==-1) + { + tmpvCardEditorRef.find('[data-type="\\%phone"]').each( + function (index,element) + { + if((value=$(element).find('[data-type="value"]').val())!='') + { + var incGroupCounter=false; + if(vCard.tplM['contentline_TEL']!=null && (process_elem=vCard.tplM['contentline_TEL'][$(element).attr('data-id')])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + { + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+groupCounter+'.').substring(2); + incGroupCounter=true; + } + } + else + { + process_elem=vCard.tplC['contentline_TEL']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + } + tmp_type=$(element).find('[data-type="phone_type"] option').filter(':selected').attr('data-type'); + + /* construct the "custom" type */ + if(tmp_type==':custom') + { + var tmp_cust_value=$(element).find('[data-type="custom_value"]').val(); + var tmp_cust_value_processed=tmp_cust_value.replace(RegExp('^\\s*|\\s*$','g'),'').replaceAll(' ',' '); + // if a custom type is already defined as standard type, use the standard definition + if((tmp_cust_already_exists=$(element).find('[data-type="phone_type"] option').filter(function(){return $(this).html()==tmp_cust_value_processed;}).attr('data-type'))!=undefined) + tmp_type=tmp_cust_already_exists; + else // use custom type + tmp_type=':'+tmp_cust_value+':'; + } + + params_wsc=''; + tmp_normal_types=tmp_type.replace(RegExp(':.*:','g'),',').replaceAll(',,',',').replace(RegExp('^,|,$','g'),''); + + if(tmp_normal_types!='') + params_wsc=';TYPE='+vcardEscapeValue(tmp_normal_types).toUpperCase().replace(RegExp('\\\\,','g'),';TYPE='); + + process_elem=process_elem.replace('##:::##params_wsc##:::##',params_wsc); + process_elem=process_elem.replace('##:::##value##:::##',vcardEscapeValue(value)); + + my_related=''; + tmp_related_type=tmp_type.match(RegExp(':(.*):')); // only one element of related (X-ABLabel) is supported + + if(tmp_related_type!=null && tmp_related_type[1]!='') + my_related='X-ABLabel:'+vcardEscapeValue((dataTypes['phone_type_store_as'][tmp_related_type[1]]!=undefined ? dataTypes['phone_type_store_as'][tmp_related_type[1]] : tmp_related_type[1]))+'\r\n'; + + if(my_related!='') + { + incGroupCounter=true; + parsed=('\r\n'+process_elem).match(vCard.pre['contentline_parse']); + if(parsed[1]!='') // if group is present, we use it, otherwise we create a new group + process_elem+=parsed[1]+my_related; + else + process_elem='item'+groupCounter+'.'+process_elem+'item'+groupCounter+'.'+my_related; + } + + if(incGroupCounter) groupCounter++; + vCardText+=process_elem; + } + } + ); + } + +// EMAIL + if(globalDisabledContactAttributes.indexOf('EMAIL')==-1) + { + tmpvCardEditorRef.find('[data-type="\\%email"]').each( + function (index,element) + { + if((value=$(element).find('[data-type="value"]').val())!='') + { + incGroupCounter=false; + if(vCard.tplM['contentline_EMAIL']!=null && (process_elem=vCard.tplM['contentline_EMAIL'][$(element).attr('data-id')])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + { + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+groupCounter+'.').substring(2); + incGroupCounter=true; + } + } + else + { + process_elem=vCard.tplC['contentline_EMAIL']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + } + + tmp_type=$(element).find('[data-type="email_type"] option').filter(':selected').attr('data-type'); + + /* construct the "custom" type */ + if(tmp_type==':custom') + { + var tmp_cust_value=$(element).find('[data-type="custom_value"]').val(); + var tmp_cust_value_processed=tmp_cust_value.replace(RegExp('^\\s*|\\s*$','g'),'').replaceAll(' ',' '); + // if a custom type is already defined as standard type, use the standard definition + if((tmp_cust_already_exists=$(element).find('[data-type="email_type"] option').filter(function(){return $(this).html()==tmp_cust_value_processed;}).attr('data-type'))!=undefined) + tmp_type=tmp_cust_already_exists; + else // use custom type + tmp_type=':'+tmp_cust_value+':'; + } + + params_wsc=''; + tmp_normal_types=tmp_type.replace(RegExp(':.*:','g'),',').replaceAll(',,',',').replace(RegExp('^,|,$','g'),''); + if(tmp_normal_types!='') + params_wsc=';TYPE='+vcardEscapeValue(tmp_normal_types).toUpperCase().replace(RegExp('\\\\,','g'),';TYPE='); + + process_elem=process_elem.replace('##:::##params_wsc##:::##',params_wsc); + process_elem=process_elem.replace('##:::##value##:::##',vcardEscapeValue(value)); + + my_related=''; + tmp_related_type=tmp_type.match(RegExp(':(.*):')); // only one element of related (X-ABLabel) is supported + + if(tmp_related_type!=null && tmp_related_type[1]!='') + my_related='X-ABLabel:'+vcardEscapeValue((dataTypes['email_type_store_as'][tmp_related_type[1]]!=undefined ? dataTypes['email_type_store_as'][tmp_related_type[1]] : tmp_related_type[1]))+'\r\n'; + + if(my_related!='') + { + incGroupCounter=true; + parsed=('\r\n'+process_elem).match(vCard.pre['contentline_parse']); + if(parsed[1]!='') // if group is present, we use it, otherwise we create a new group + process_elem+=parsed[1]+my_related; + else + process_elem='item'+groupCounter+'.'+process_elem+'item'+groupCounter+'.'+my_related; + } + + if(incGroupCounter) groupCounter++; + vCardText+=process_elem; + } + } + ); + } + +// URL + if(globalDisabledContactAttributes.indexOf('URL')==-1) + { + tmpvCardEditorRef.find('[data-type="\\%url"]').each( + function (index,element) + { + if((value=$(element).find('[data-type="value"]').val())!='') + { + incGroupCounter=false; + if(vCard.tplM['contentline_URL']!=null && (process_elem=vCard.tplM['contentline_URL'][$(element).attr('data-id')])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + { + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+groupCounter+'.').substring(2); + incGroupCounter=true; + } + } + else + { + process_elem=vCard.tplC['contentline_URL']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + } + + tmp_type=$(element).find('[data-type="url_type"] option').filter(':selected').attr('data-type'); + + /* construct the "custom" type */ + if(tmp_type==':custom') + { + var tmp_cust_value=$(element).find('[data-type="custom_value"]').val(); + var tmp_cust_value_processed=tmp_cust_value.replace(RegExp('^\\s*|\\s*$','g'),'').replaceAll(' ',' '); + // if a custom type is already defined as standard type, use the standard definition + if((tmp_cust_already_exists=$(element).find('[data-type="url_type"] option').filter(function(){return $(this).html()==tmp_cust_value_processed;}).attr('data-type'))!=undefined) + tmp_type=tmp_cust_already_exists; + else // use custom type + tmp_type=':'+tmp_cust_value+':'; + } + + params_wsc=''; + tmp_normal_types=tmp_type.replace(RegExp(':.*:','g'),',').replaceAll(',,',',').replace(RegExp('^,|,$','g'),''); + if(tmp_normal_types!='') + params_wsc=';TYPE='+vcardEscapeValue(tmp_normal_types).toUpperCase().replace(RegExp('\\\\,','g'),';TYPE='); + + process_elem=process_elem.replace('##:::##params_wsc##:::##',params_wsc); + process_elem=process_elem.replace('##:::##value##:::##',vcardEscapeValue(value)); + + my_related=''; + tmp_related_type=tmp_type.match(RegExp(':(.*):')); // only one element of related (X-ABLabel) is supported + + if(tmp_related_type!=null && tmp_related_type[1]!='') + my_related='X-ABLabel:'+vcardEscapeValue((dataTypes['url_type_store_as'][tmp_related_type[1]]!=undefined ? dataTypes['url_type_store_as'][tmp_related_type[1]] : tmp_related_type[1]))+'\r\n'; + + if(my_related!='') + { + incGroupCounter=true; + parsed=('\r\n'+process_elem).match(vCard.pre['contentline_parse']); + if(parsed[1]!='') // if group is present, we use it, otherwise we create a new group + process_elem+=parsed[1]+my_related; + else + process_elem='item'+groupCounter+'.'+process_elem+'item'+groupCounter+'.'+my_related; + } + + if(incGroupCounter) groupCounter++; + vCardText+=process_elem; + } + } + ); + } + +// X-ABRELATEDNAMES + if(globalDisabledContactAttributes.indexOf('X-ABRELATEDNAMES')==-1) + { + tmpvCardEditorRef.find('[data-type="\\%person"]').each( + function (index,element) + { + if((value=$(element).find('[data-type="value"]').val())!='') + { + incGroupCounter=false; + if(vCard.tplM['contentline_X-ABRELATEDNAMES']!=null && (process_elem=vCard.tplM['contentline_X-ABRELATEDNAMES'][$(element).attr('data-id')])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + { + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+groupCounter+'.').substring(2); + incGroupCounter=true; + } + } + else + { + process_elem=vCard.tplC['contentline_X-ABRELATEDNAMES']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + } + + tmp_type=$(element).find('[data-type="person_type"] option').filter(':selected').attr('data-type'); + + /* construct the "custom" type */ + if(tmp_type==':custom') + { + var tmp_cust_value=$(element).find('[data-type="custom_value"]').val(); + var tmp_cust_value_processed=tmp_cust_value.replace(RegExp('^\\s*|\\s*$','g'),'').replaceAll(' ',' '); + // if a custom type is already defined as standard type, use the standard definition + if((tmp_cust_already_exists=$(element).find('[data-type="person_type"] option').filter(function(){return $(this).html()==tmp_cust_value_processed;}).attr('data-type'))!=undefined) + tmp_type=tmp_cust_already_exists; + else // use custom type + tmp_type=':'+tmp_cust_value+':'; + } + + params_wsc=''; + tmp_normal_types=tmp_type.replace(RegExp(':.*:','g'),',').replaceAll(',,',',').replace(RegExp('^,|,$','g'),''); + if(tmp_normal_types!='') + params_wsc=';TYPE='+vcardEscapeValue(tmp_normal_types).toUpperCase().replace(RegExp('\\\\,','g'),';TYPE='); + + process_elem=process_elem.replace('##:::##params_wsc##:::##',params_wsc); + process_elem=process_elem.replace('##:::##value##:::##',vcardEscapeValue(value)); + + my_related=''; + tmp_related_type=tmp_type.match(RegExp(':(.*):')); // only one element of related (X-ABLabel) is supported + + if(tmp_related_type!=null && tmp_related_type[1]!='') + my_related='X-ABLabel:'+vcardEscapeValue((dataTypes['person_type_store_as'][tmp_related_type[1]]!=undefined ? dataTypes['person_type_store_as'][tmp_related_type[1]] : tmp_related_type[1]))+'\r\n'; + + if(my_related!='') + { + incGroupCounter=true; + parsed=('\r\n'+process_elem).match(vCard.pre['contentline_parse']); + if(parsed[1]!='') // if group is present, we use it, otherwise we create a new group + process_elem+=parsed[1]+my_related; + else + process_elem='item'+groupCounter+'.'+process_elem+'item'+groupCounter+'.'+my_related; + } + + if(incGroupCounter) groupCounter++; + + if(tmp_related_type!=null && tmp_related_type[1]!='') + { + // In addition of the X-ABRELATEDNAMES attributes add also the old style X-* attributes + switch(tmp_related_type[1]) + { + case '_$!!$_': + process_elem+='X-ASSISTANT:'+vcardEscapeValue(value)+'\r\n'; + // process_elem+='X-EVOLUTION-ASSISTANT:'+vcardEscapeValue(value)+'\r\n'; + break; + case '_$!!$_': + process_elem+='X-MANAGER:'+vcardEscapeValue(value)+'\r\n'; + // process_elem+='X-EVOLUTION-MANAGER:'+vcardEscapeValue(value)+'\r\n'; + break; + case '_$!!$_': + process_elem+='X-SPOUSE:'+vcardEscapeValue(value)+'\r\n'; + // process_elem+='X-EVOLUTION-SPOUSE:'+vcardEscapeValue(value)+'\r\n'; + break; + default: + break; + } + } + + vCardText+=process_elem; + } + } + ); + } + +// IMPP + if(globalDisabledContactAttributes.indexOf('IMPP')==-1) + { + tmpvCardEditorRef.find('[data-type="\\%im"]').each( + function (index,element) + { + if((value=$(element).find('[data-type="value"]').val())!='') + { + incGroupCounter=false; + if(vCard.tplM['contentline_IMPP']!=null && (process_elem=vCard.tplM['contentline_IMPP'][$(element).attr('data-id')])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + { + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+groupCounter+'.').substring(2); + incGroupCounter=true; + } + } + else + { + process_elem=vCard.tplC['contentline_IMPP']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + } + + tmp_type=$(element).find('[data-type="im_type"] option').filter(':selected').attr('data-type'); + + /* construct the "custom" type */ + if(tmp_type==':custom') + { + var tmp_cust_value=$(element).find('[data-type="custom_value"]:first').val(); + var tmp_cust_value_processed=tmp_cust_value.replace(RegExp('^\\s*|\\s*$','g'),'').replaceAll(' ',' '); + // if a custom type is already defined as standard type, use the standard definition + if((tmp_cust_already_exists=$(element).find('[data-type="im_type"] option').filter(function(){return $(this).html()==tmp_cust_value_processed;}).attr('data-type'))!=undefined) + tmp_type=tmp_cust_already_exists; + else // use custom type + tmp_type=':'+tmp_cust_value+':'; + } + + params_wsc=params_wsc_old_repr=''; + tmp_normal_types=tmp_type.replace(RegExp(':.*:','g'),',').replaceAll(',,',',').replace(RegExp('^,|,$','g'),''); + if(tmp_normal_types!='') + params_wsc=params_wsc_old_repr=';TYPE='+vcardEscapeValue(tmp_normal_types).toUpperCase().replace(RegExp('\\\\,','g'),';TYPE='); + + tmp_service_type=$(element).find('[data-type="im_service_type"] option').filter(':selected').attr('data-type'); + + /* construct the "custom" type */ + if(tmp_service_type==':custom') + { + var tmp_cust_value=$(element).find('[data-type="custom_value"]:last').val(); + var tmp_cust_value_processed=tmp_cust_value.replace(RegExp('^\\s*|\\s*$','g'),'').replaceAll(' ',' '); + // if a custom type is already defined as standard type, use the standard definition + if((tmp_cust_already_exists=$(element).find('[data-type="im_service_type"] option').filter(function(){return $(this).html()==tmp_cust_value_processed;}).attr('data-type'))!=undefined) + tmp_service_type=tmp_cust_already_exists; + else // use custom type + tmp_service_type=':'+tmp_cust_value+':'; + } + + if(dataTypes['im_service_type_store_as'][tmp_service_type]!=undefined) + tmp_service_type=dataTypes['im_service_type_store_as'][tmp_service_type]; + params_wsc=';X-SERVICE-TYPE='+vcardEscapeValue(tmp_service_type)+params_wsc; + + process_elem=process_elem.replace('##:::##params_wsc##:::##',params_wsc); + switch(tmp_service_type.toLowerCase()) // RFC4770 + { + case 'aim': + im_value='aim:'+vcardEscapeValue(value); + break; + case 'facebook': + im_value='xmpp:'+vcardEscapeValue(value); + break; + case 'googletalk': + im_value='xmpp:'+vcardEscapeValue(value); + break; + case 'icq': + im_value='aim:'+vcardEscapeValue(value); + break; + case 'irc': + im_value='irc:'+vcardEscapeValue(value); + break; + case 'jabber': + im_value='xmpp:'+vcardEscapeValue(value); + break; + case 'msn': + im_value='msnim:'+vcardEscapeValue(value); + break; + case 'skype': + im_value='skype:'+vcardEscapeValue(value); + break; + case 'yahoo': + im_value='ymsgr:'+vcardEscapeValue(value); + break; + default: // 'gadugadu', 'qq', ... + im_value='x-apple:'+vcardEscapeValue(value); + break; + } + process_elem=process_elem.replace('##:::##value##:::##',im_value); + + my_related=''; + tmp_related_type=tmp_type.match(RegExp(':(.*):')); // only one element of related (X-ABLabel) is supported + + if(tmp_related_type!=null && tmp_related_type[1]!='') + my_related='X-ABLabel:'+vcardEscapeValue((dataTypes['im_type_store_as'][tmp_related_type[1]]!=undefined ? dataTypes['im_type_store_as'][tmp_related_type[1]] : tmp_related_type[1]))+'\r\n'; + + if(my_related!='') + { + incGroupCounter=true; + parsed=('\r\n'+process_elem).match(vCard.pre['contentline_parse']); + if(parsed[1]!='') // if group is present, we use it, otherwise we create a new group + process_elem+=parsed[1]+my_related; + else + process_elem='item'+groupCounter+'.'+process_elem+'item'+groupCounter+'.'+my_related; + } + if(incGroupCounter) groupCounter++; + + // In addition of the IMPP attributes add also the old style X-* attributes + process_elem_old_repr=''; + switch(tmp_service_type.toLowerCase()) + { + case 'aim': + new_group_wd=''; + if(incGroupCounter) + { + new_group_wd='item'+groupCounter+'.'; + process_elem_old_repr=('\r\n'+process_elem).replace(RegExp('\r\nitem'+(groupCounter-1)+'\\.','mg'),'\r\n'+new_group_wd); + groupCounter++; + } + else + process_elem_old_repr='\r\n'+process_elem; + process_elem+=process_elem_old_repr.replace('\r\n'+new_group_wd+'IMPP;X-SERVICE-TYPE='+ vcardEscapeValue(tmp_service_type),new_group_wd+'X-AIM').replace(im_value+'\r\n',vcardEscapeValue(value)+'\r\n'); + break; + case 'jabber': + new_group_wd=''; + if(incGroupCounter) + { + new_group_wd='item'+groupCounter+'.'; + process_elem_old_repr=('\r\n'+process_elem).replace(RegExp('\r\nitem'+(groupCounter-1)+'\\.','mg'),'\r\n'+new_group_wd); + groupCounter++; + } + else + process_elem_old_repr='\r\n'+process_elem; + process_elem+=process_elem_old_repr.replace('\r\n'+new_group_wd+'IMPP;X-SERVICE-TYPE='+ vcardEscapeValue(tmp_service_type),new_group_wd+'X-JABBER').replace(im_value+'\r\n',vcardEscapeValue(value)+'\r\n'); + break; + case 'msn': + new_group_wd=''; + if(incGroupCounter) + { + new_group_wd='item'+groupCounter+'.'; + process_elem_old_repr=('\r\n'+process_elem).replace(RegExp('\r\nitem'+(groupCounter-1)+'\\.','mg'),'\r\n'+new_group_wd); + groupCounter++; + } + else + process_elem_old_repr='\r\n'+process_elem; + process_elem+=process_elem_old_repr.replace('\r\n'+new_group_wd+'IMPP;X-SERVICE-TYPE='+ vcardEscapeValue(tmp_service_type),new_group_wd+'X-MSN').replace(im_value+'\r\n',vcardEscapeValue(value)+'\r\n'); + break; + case 'yahoo': + new_group_wd=''; + process_elem_tmp=process_elem; + if(incGroupCounter) + { + new_group_wd='item'+groupCounter+'.'; + process_elem_old_repr=('\r\n'+process_elem_tmp).replace(RegExp('\r\nitem'+(groupCounter-1)+'\\.','mg'),'\r\n'+new_group_wd); + groupCounter++; + } + else + process_elem_old_repr='\r\n'+process_elem; + process_elem+=process_elem_old_repr.replace('\r\n'+new_group_wd+'IMPP;X-SERVICE-TYPE='+ vcardEscapeValue(tmp_service_type),new_group_wd+'X-YAHOO').replace(im_value+'\r\n',vcardEscapeValue(value)+'\r\n'); + + new_group_wd=''; + if(incGroupCounter) + { + new_group_wd='item'+groupCounter+'.'; + process_elem_old_repr=('\r\n'+process_elem_tmp).replace(RegExp('\r\nitem'+(groupCounter-2)+'\\.','mg'),'\r\n'+new_group_wd); + groupCounter++; + } + else + process_elem_old_repr='\r\n'+process_elem; + process_elem+=process_elem_old_repr.replace('\r\n'+new_group_wd+'IMPP;X-SERVICE-TYPE='+ vcardEscapeValue(tmp_service_type),new_group_wd+'X-YAHOO-ID').replace(im_value+'\r\n',vcardEscapeValue(value)+'\r\n'); + break; + case 'icq': + new_group_wd=''; + if(incGroupCounter) + { + new_group_wd='item'+groupCounter+'.'; + process_elem_old_repr=('\r\n'+process_elem).replace(RegExp('\r\nitem'+(groupCounter-1)+'\\.','mg'),'\r\n'+new_group_wd); + groupCounter++; + } + else + process_elem_old_repr='\r\n'+process_elem; + process_elem+=process_elem_old_repr.replace('\r\n'+new_group_wd+'IMPP;X-SERVICE-TYPE='+ vcardEscapeValue(tmp_service_type),new_group_wd+'X-ICQ').replace(im_value+'\r\n',vcardEscapeValue(value)+'\r\n'); + break; + default: + break; + } + vCardText+=process_elem; + } + } + ); + } + +// X-SOCIALPROFILE + if(globalDisabledContactAttributes.indexOf('X-SOCIALPROFILE')==-1) + { + tmpvCardEditorRef.find('[data-type="\\%profile"]').each( + function (index,element) + { + if((value=$(element).find('[data-type="value"]').val())!='') + { + incGroupCounter=false; + if(vCard.tplM['contentline_X-SOCIALPROFILE']!=null && (process_elem=vCard.tplM['contentline_X-SOCIALPROFILE'][$(element).attr('data-id')])!=undefined) + { + // replace the object and related objects' group names (+ append the related objects after the processed) + parsed=('\r\n'+process_elem).match(RegExp('\r\n((?:'+vCard.re['group']+'\\.)?)','m')); + if(parsed[1]!='') // if group is present, replace the object and related objects' group names + { + process_elem=('\r\n'+process_elem).replace(RegExp('\r\n'+parsed[1].replace('.','\\.'),'mg'),'\r\nitem'+groupCounter+'.').substring(2); + incGroupCounter=true; + } + } + else + { + process_elem=vCard.tplC['contentline_X-SOCIALPROFILE']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + } + + tmp_type=$(element).find('[data-type="profile_type"] option').filter(':selected').attr('data-type'); + + /* construct the "custom" type */ + if(tmp_type==':custom') + { + var tmp_cust_value=$(element).find('[data-type="custom_value"]').val(); + var tmp_cust_value_processed=tmp_cust_value.replace(RegExp('^\\s*|\\s*$','g'),'').replaceAll(' ',' '); + // if a custom type is already defined as standard type, use the standard definition + if((tmp_cust_already_exists=$(element).find('[data-type="profile_type"] option').filter(function(){return $(this).html()==tmp_cust_value_processed;}).attr('data-type'))!=undefined) + tmp_type=tmp_cust_already_exists; + else // use custom type + tmp_type=':'+tmp_cust_value+':'; + } + + params_wsc=''; + tmp_normal_types=tmp_type.replace(RegExp(':.*:','g'),',').replaceAll(',,',',').replace(RegExp('^,|,$','g'),''); + if(tmp_normal_types!='') + params_wsc=';TYPE='+vcardEscapeValue(tmp_normal_types).toUpperCase().replace(RegExp('\\\\,','g'),';TYPE=')+';x-user='+vcardEscapeValue(tmp_type=='twitter' ? value.replace(/^@+/, '') : value); + + process_elem=process_elem.replace('##:::##params_wsc##:::##',params_wsc); + process_elem=process_elem.replace('##:::##value##:::##', vcardEscapeValue((globalSettings.urihandlerprofile.value[tmp_type]!=undefined ? globalSettings.urihandlerprofile.value[tmp_type] : 'x-apple:%u').replace('%u', (tmp_type=='twitter' ? value.replace(/^@+/, '') : value)))); + + if(incGroupCounter) groupCounter++; + vCardText+=process_elem; + } + } + ); + } + + // extension hook + if(typeof(globalContactsExtDataToVcard)=='function') + vCardText=globalContactsExtDataToVcard(tmpvCardEditorRef, vCardText); + + // PRODID + vCardText+='PRODID:-//Inf-IT//'+globalAppName+' '+globalVersion+'//EN\r\n'; + + if(typeof vCard.tplM['unprocessed_unrelated']!='undefined') + vCardText+=vCard.tplM['unprocessed_unrelated'].replace(RegExp('^\r\n'),''); + + // vCard END (required by RFC) + if(vCard.tplM['end']!=null && (process_elem=vCard.tplM['end'][0])!=undefined) + vCardText+=vCard.tplM['end'][0]; + else + { + process_elem=vCard.tplC['end']; + process_elem=process_elem.replace('##:::##group_wd##:::##', ''); + vCardText+=process_elem; + } + + // replace unsupported XML characters + vCardText=vCardText.replace(/[^\u0009\u000A\u000D\u0020-\uD7FF\uE000-\uFFFD]/g, ' '); + + // line folding (RFC2426 - section 2.6) - maximum of 75 octects (and cannot break + // multi-octet UTF8-characters) allowed on one line, excluding a line break (CRLF) + vCardText=vObjectLineFolding(vCardText); + + if(typeof(globalContactsExtPutVcardToCollectionOverload)=='function') + globalContactsExtPutVcardToCollectionOverload(accountUID, inputEtag, newUID, vCardText); + else + { + var selAddr = tmpvCardEditorRef.find('[data-attr-name="_DEST_"]').find('option:selected').attr('data-type') + //addressbook selectbox was changed + var orgAddr = $('#vCardEditor').attr('data-url').replace(RegExp('[^/]*$'),''); + if($('#ExtendedDest').length>0) + { + var putGroups=new Array(); + var removeGroups=new Array(); + var myGroups = new Array() + if(inputEtag!='') + { + myGroups=globalAddressbookList.getMyContactGroups($('#vCardEditor').attr('data-url')); + for(var gi=0; gi0) + { + tmp2.uiObjects={resource:globalRefAddContact.attr('data-filter-url')}; + if(putGroups.length>0) + tmp2.addToContactGroupUID=putGroups.slice(); + if(removeGroups.length>0) + tmp2.removeToContactGroupUID=removeGroups.slice(); + } + tmp2.formSave=true; + lockAndPerformToCollection(tmp2, globalRefAddContact.attr('data-filter-url'), 'IRM_DELETE'); + } + else + { + if(inputEtag=='') + inputUID=selAddr; + if($('#ExtendedDest').length>0 && (putGroups.length>0 || removeGroups.length>0)) + { + if(inputEtag!='') + var tmp2=globalAddressbookList.getContactByUID($('#vCardEditor').attr('data-url')); + else + var tmp2={accountUID: accountUID, uid: inputUID, etag: inputEtag}; + var vUID = $('#vCardEditor').attr('data-url').match(RegExp('[^/]*$')); + // here we generate the destination for MOVE (we don't use the old vCard file name to minimalize the possible conflict situations) + tmp2.vcard=vCardText; + tmp2.uiObjects={resource:globalRefAddContact.attr('data-filter-url')}; + tmp2.addToContactGroupUID=new Array(); + tmp2.removeToContactGroupUID=new Array(); + if(putGroups.length>0) + tmp2.addToContactGroupUID=putGroups.slice(); + if(removeGroups.length>0) + tmp2.removeToContactGroupUID=removeGroups.slice(); + tmp2.formSave=true; + lockAndPerformToCollection(tmp2, globalRefAddContact.attr('data-filter-url'), 'PUT'); + } + else + putVcardToCollectionMain({accountUID: accountUID, uid: inputUID, etag: inputEtag, vcard: vCardText}, inputFilterUID); + } + } +} + +function vcardToData(inputContact, inputIsReadonly, inputIsCompany, inputEditorMode) +{ + if(typeof globalDebug!='undefined' && globalDebug instanceof Array && globalDebug.indexOf('vcard')!=-1) + console.time('vcardToData timer'); + + if(inputContact.vcard==undefined) + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; + } + + var tmpvCardEditorRef=CardDAVeditor_cleanup(false, inputIsCompany); // editor initialization + + $('#ABContactColor').css('background-color', inputContact.color); + + if(typeof globalDisabledContactAttributes=='undefined' || !(globalDisabledContactAttributes instanceof Array)) + globalDisabledContactAttributes=[]; + + if(inputContact.vcard.match(vCard.pre['vcard'])) + { + // ------------------------------------------------------------------------------------- // + // BEGIN and END + vcard_full=inputContact.vcard.split('\r\n'); // vCard data to array + + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + if((parsed=('\r\n'+vcard_full[0]+'\r\n').match(vCard.pre['contentline_parse']))==null) + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; + } + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['begin'][0]=vCard.tplC['begin'].replace('##:::##group_wd##:::##', vcard_begin_group=parsed[1]); + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + if((parsed=('\r\n'+vcard_full[vcard_full.length-2]+'\r\n').match(vCard.pre['contentline_parse']))==null) + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; + } + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['end'][0]=vCard.tplC['end'].replace('##:::##group_wd##:::##', vcard_end_group=parsed[1]); + + if(vcard_begin_group!=vcard_end_group) + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; // the vCard BEGIN and END "group" are different + } + + // remove the vCard BEGIN and END + vcard='\r\n'+vcard_full.slice(1, vcard_full.length-2).join('\r\n')+'\r\n'; + +//console.time('VERSION timer'); + // ------------------------------------------------------------------------------------- // + // VERSION -> what to do if present more than once? + vcard_element=vcard.match(vCard.pre['contentline_VERSION']); + if(vcard_element!=null && vcard_element.length==1) // if the VERSION attribute is not present exactly once, vCard is considered invalid + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + if(parsed[3]=='' && parsed[4]=='3.0') // RFC requirement + { + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['contentline_VERSION'][0]=vCard.tplC['contentline_VERSION']; + vCard.tplM['contentline_VERSION'][0]=vCard.tplM['contentline_VERSION'][0].replace('##:::##group_wd##:::##', parsed[1]); + vCard.tplM['contentline_VERSION'][0]=vCard.tplM['contentline_VERSION'][0].replace('##:::##version##:::##', parsed[4]); + + // remove the processed parameter + vcard=vcard.replace(vcard_element[0], '\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while((vcard_element_related=vcard.match(re))!=null) + { + // append the parameter to its parent + vCard.tplM['contentline_VERSION'][0]+=vcard_element_related[0].substr(2); + // remove the processed parameter + vcard=vcard.replace(vcard_element_related[0],'\r\n'); + } + } + } + else + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; // invalid input for "VERSION" (we support only vCard 3.0) + } + } + else + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; // vcard "VERSION" not present or present more than once + } +//console.timeEnd('VERSION timer'); + +//console.time('UID timer'); + // ------------------------------------------------------------------------------------- // + // UID -> TODO: what to do if present more than once? + vcard_element=vcard.match(vCard.pre['contentline_UID']); + if(vcard_element!=null && vcard_element.length==1) // if the UID attribute is not present exactly once, vCard is considered invalid + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['contentline_UID'][0]=vCard.tplC['contentline_UID']; + vCard.tplM['contentline_UID'][0]=vCard.tplM['contentline_UID'][0].replace('##:::##group_wd##:::##', parsed[1]); + vCard.tplM['contentline_UID'][0]=vCard.tplM['contentline_UID'][0].replace('##:::##params_wsc##:::##', parsed[3]); + vCard.tplM['contentline_UID'][0]=vCard.tplM['contentline_UID'][0].replace('##:::##uid##:::##', parsed[4]); + + tmpvCardEditorRef.find('#vCardEditor').attr('data-vcard-uid', parsed[4]); // special hack; usually used by extension hooks + + // remove the processed parameter + vcard=vcard.replace(vcard_element[0], '\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while((vcard_element_related=vcard.match(re))!=null) + { + // append the parameter to its parent + vCard.tplM['contentline_UID'][0]+=vcard_element_related[0].substr(2); + // remove the processed parameter + vcard=vcard.replace(vcard_element_related[0],'\r\n'); + } + } + } +// Old not RFC vCards not contain UID - we ignore this error (UID is generated if vCard is changed) +// else +// { +// console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); +// return false; // vcard UID not present or present more than once +// } +//console.timeEnd('UID timer'); + +//console.time('FN timer'); + // ------------------------------------------------------------------------------------- // + // FN -> TODO: what to do if present more than once? + vcard_element=vcard.match(vCard.pre['contentline_FN']); + if(vcard_element!=null && vcard_element.length==1) // if the FN attribute is not present exactly once, vCard is considered invalid + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['contentline_FN'][0]=vCard.tplC['contentline_FN']; + vCard.tplM['contentline_FN'][0]=vCard.tplM['contentline_FN'][0].replace('##:::##group_wd##:::##', parsed[1]); + vCard.tplM['contentline_FN'][0]=vCard.tplM['contentline_FN'][0].replace('##:::##params_wsc##:::##', parsed[3]); + + // remove the processed parameter + vcard=vcard.replace(vcard_element[0],'\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while((vcard_element_related=vcard.match(re))!=null) + { + // append the parameter to its parent + vCard.tplM['contentline_FN'][0]+=vcard_element_related[0].substr(2); + // remove the processed parameter + vcard=vcard.replace(vcard_element_related[0],'\r\n'); + } + } + } + else + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; // vcard FN not present or present more than once + } +//console.timeEnd('FN timer'); + +//console.time('N timer'); + // ------------------------------------------------------------------------------------- // + // N -> TODO: what to do if present more than once? + vcard_element=vcard.match(vCard.pre['contentline_N']); + if(vcard_element!=null && vcard_element.length==1) // if the N attribute is not present exactly once, vCard is considered invalid + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + // parsed_value = [0]->Family, [1]->Given, [2]->Middle, [3]->Prefix, [4]->Suffix + parsed_value=vcardSplitValue(parsed[4],';'); + + if(parsed_value[0]!=undefined && parsed_value[0]!='') + tmpvCardEditorRef.find('[data-type="family"]').val(vcardUnescapeValue(parsed_value[0])); + if(parsed_value[1]!=undefined && parsed_value[1]!='') + tmpvCardEditorRef.find('[data-type="given"]').val(vcardUnescapeValue(parsed_value[1])); + if(parsed_value[2]!=undefined && parsed_value[2]!='') + tmpvCardEditorRef.find('[data-type="middle"]').val(vcardUnescapeValue(parsed_value[2])); + if(parsed_value[3]!=undefined && parsed_value[3]!='') + tmpvCardEditorRef.find('[data-type="prefix"]').val(vcardUnescapeValue(parsed_value[3])); + if(parsed_value[4]!=undefined && parsed_value[4]!='') + tmpvCardEditorRef.find('[data-type="suffix"]').val(vcardUnescapeValue(parsed_value[4])); + + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['contentline_N'][0]=vCard.tplC['contentline_N']; + vCard.tplM['contentline_N'][0]=vCard.tplM['contentline_N'][0].replace('##:::##group_wd##:::##', parsed[1]); + vCard.tplM['contentline_N'][0]=vCard.tplM['contentline_N'][0].replace('##:::##params_wsc##:::##', parsed[3]); + + // remove the processed parameter + vcard=vcard.replace(vcard_element[0],'\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while((vcard_element_related=vcard.match(re))!=null) + { + // append the parameter to its parent + vCard.tplM['contentline_N'][0]+=vcard_element_related[0].substr(2); + // remove the processed parameter + vcard=vcard.replace(vcard_element_related[0],'\r\n'); + } + } + } + else + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; // vcard N not present or present more than once + } +//console.timeEnd('N timer'); + +//console.time('CATEGORIES timer'); + // ------------------------------------------------------------------------------------- // + // CATEGORIES -> present max. once because of the CardDavMATE vCard transformations + if(globalDisabledContactAttributes.indexOf('CATEGORIES')==-1) + { + vcard_element=vcard.match(vCard.pre['contentline_CATEGORIES']); + if(vcard_element!=null && vcard_element.length==1) + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + + tmpvCardEditorRef.find('#tags').importTags(parsed[4]); // we do not need to unescape the value here! + + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['contentline_CATEGORIES'][0]=vCard.tplC['contentline_CATEGORIES']; + vCard.tplM['contentline_CATEGORIES'][0]=vCard.tplM['contentline_CATEGORIES'][0].replace('##:::##group_wd##:::##', parsed[1]); + vCard.tplM['contentline_CATEGORIES'][0]=vCard.tplM['contentline_CATEGORIES'][0].replace('##:::##params_wsc##:::##', parsed[3]); + + // remove the processed parameter + vcard=vcard.replace(vcard_element[0],'\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while((vcard_element_related=vcard.match(re))!=null) + { + // append the parameter to its parent + vCard.tplM['contentline_CATEGORIES'][0]+=vcard_element_related[0].substr(2); + // remove the processed parameter + vcard=vcard.replace(vcard_element_related[0],'\r\n'); + } + } + } + } +//console.timeEnd('CATEGORIES timer'); + +//console.time('NOTE timer'); + // ------------------------------------------------------------------------------------- // + // NOTE -> TODO: what to do if present more than once? + if(globalDisabledContactAttributes.indexOf('NOTE')==-1) + { + vcard_element=vcard.match(vCard.pre['contentline_NOTE']); + if(vcard_element!=null) + { + if(vcard_element.length==1) // if the NOTE attribute is present exactly once + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + + tmpvCardEditorRef.find('[data-type="\\%note"]').find('textarea').text(vcardUnescapeValue(parsed[4])).trigger('autosize.resize'); + + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['contentline_NOTE'][0]=vCard.tplC['contentline_NOTE']; + vCard.tplM['contentline_NOTE'][0]=vCard.tplM['contentline_NOTE'][0].replace('##:::##group_wd##:::##', parsed[1]); + vCard.tplM['contentline_NOTE'][0]=vCard.tplM['contentline_NOTE'][0].replace('##:::##params_wsc##:::##', parsed[3]); + + // remove the processed parameter + vcard=vcard.replace(vcard_element[0],'\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while((vcard_element_related=vcard.match(re))!=null) + { + // append the parameter to its parent + vCard.tplM['contentline_NOTE'][0]+=vcard_element_related[0].substr(2); + // remove the processed parameter + vcard=vcard.replace(vcard_element_related[0],'\r\n'); + } + } + } + else + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; // vcard NOTE present more than once + } + } + } +//console.timeEnd('NOTE timer'); + +//console.time('REV timer'); + // ------------------------------------------------------------------------------------- // + // REV -> what to do if present more than once? + vcard_element=vcard.match(vCard.pre['contentline_REV']); + if(vcard_element!=null) // if the REV attribute is exists + { + if(vcard_element.length==1) // and is present exactly once + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['contentline_REV'][0]=vCard.tplC['contentline_REV']; + vCard.tplM['contentline_REV'][0]=vCard.tplM['contentline_REV'][0].replace('##:::##group_wd##:::##', parsed[1]); + + // remove the processed parameter + vcard=vcard.replace(vcard_element[0],'\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while((vcard_element_related=vcard.match(re))!=null) + { + // append the parameter to its parent + vCard.tplM['contentline_REV'][0]+=vcard_element_related[0].substr(2); + // remove the processed parameter + vcard=vcard.replace(vcard_element_related[0],'\r\n'); + } + } + } + else + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; // vcard REV present more than once + } + } +//console.timeEnd('REV timer'); + +//console.time('NICKNAME timer'); + // ------------------------------------------------------------------------------------- // + // NICKNAME -> TODO: what to do if present more than once? + if(globalDisabledContactAttributes.indexOf('NICKNAME')==-1) + { + vcard_element=vcard.match(vCard.pre['contentline_NICKNAME']); + if(vcard_element!=null) + { + if(vcard_element.length!=1) // if the NICKNAME attribute is present more than once, vCard is considered invalid + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; + } + + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + + tmpvCardEditorRef.find('[data-type="nickname"]').val(vcardUnescapeValue(parsed[4])); + + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['contentline_NICKNAME'][0]=vCard.tplC['contentline_NICKNAME']; + vCard.tplM['contentline_NICKNAME'][0]=vCard.tplM['contentline_NICKNAME'][0].replace('##:::##group_wd##:::##', parsed[1]); + vCard.tplM['contentline_NICKNAME'][0]=vCard.tplM['contentline_NICKNAME'][0].replace('##:::##params_wsc##:::##', parsed[3]); + + // remove the processed parameter + vcard=vcard.replace(vcard_element[0],'\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while((vcard_element_related=vcard.match(re))!=null) + { + // append the parameter to its parent + vCard.tplM['contentline_NICKNAME'][0]+=vcard_element_related[0].substr(2); + // remove the processed parameter + vcard=vcard.replace(vcard_element_related[0],'\r\n'); + } + } + } + } +//console.timeEnd('NICKNAME timer'); + +//console.time('X-PHONETIC-FIST-NAME timer'); + // ------------------------------------------------------------------------------------- // + // X-PHONETIC-FIRST-NAME -> TODO: what to do if present more than once? + if(globalDisabledContactAttributes.indexOf('X-PHONETIC-FIRST-NAME')==-1) + { + vcard_element=vcard.match(vCard.pre['contentline_X-PHONETIC-FIRST-NAME']); + if(vcard_element!=null) + { + if(vcard_element.length!=1) // if the X-PHONETIC-FIRST-NAME attribute is present more than once, vCard is considered invalid + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; + } + + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + + tmpvCardEditorRef.find('[data-type="ph_firstname"]').val(vcardUnescapeValue(parsed[4])); + + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['contentline_X-PHONETIC-FIRST-NAME'][0]=vCard.tplC['contentline_X-PHONETIC-FIRST-NAME']; + vCard.tplM['contentline_X-PHONETIC-FIRST-NAME'][0]=vCard.tplM['contentline_X-PHONETIC-FIRST-NAME'][0].replace('##:::##group_wd##:::##', parsed[1]); + vCard.tplM['contentline_X-PHONETIC-FIRST-NAME'][0]=vCard.tplM['contentline_X-PHONETIC-FIRST-NAME'][0].replace('##:::##params_wsc##:::##', parsed[3]); + + // remove the processed parameter + vcard=vcard.replace(vcard_element[0],'\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while((vcard_element_related=vcard.match(re))!=null) + { + // append the parameter to its parent + vCard.tplM['contentline_X-PHONETIC-FIRST-NAME'][0]+=vcard_element_related[0].substr(2); + // remove the processed parameter + vcard=vcard.replace(vcard_element_related[0],'\r\n'); + } + } + } + } +//console.timeEnd('X-PHONETIC-FIST-NAME timer'); + +//console.time('X-PHONETIC-LAST-NAME timer'); + // ------------------------------------------------------------------------------------- // + // X-PHONETIC-LAST-NAME -> TODO: what to do if present more than once? + if(globalDisabledContactAttributes.indexOf('X-PHONETIC-LAST-NAME')==-1) + { + vcard_element=vcard.match(vCard.pre['contentline_X-PHONETIC-LAST-NAME']); + if(vcard_element!=null) + { + if(vcard_element.length!=1) // if the X-PHONETIC-LAST-NAME attribute is present more than once, vCard is considered invalid + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; + } + + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + + tmpvCardEditorRef.find('[data-type="ph_lastname"]').val(vcardUnescapeValue(parsed[4])); + + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['contentline_X-PHONETIC-LAST-NAME'][0]=vCard.tplC['contentline_X-PHONETIC-LAST-NAME']; + vCard.tplM['contentline_X-PHONETIC-LAST-NAME'][0]=vCard.tplM['contentline_X-PHONETIC-LAST-NAME'][0].replace('##:::##group_wd##:::##', parsed[1]); + vCard.tplM['contentline_X-PHONETIC-LAST-NAME'][0]=vCard.tplM['contentline_X-PHONETIC-LAST-NAME'][0].replace('##:::##params_wsc##:::##', parsed[3]); + + // remove the processed parameter + vcard=vcard.replace(vcard_element[0],'\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while((vcard_element_related=vcard.match(re))!=null) + { + // append the parameter to its parent + vCard.tplM['contentline_X-PHONETIC-LAST-NAME'][0]+=vcard_element_related[0].substr(2); + // remove the processed parameter + vcard=vcard.replace(vcard_element_related[0],'\r\n'); + } + } + } + } +//console.timeEnd('X-PHONETIC-LAST-NAME timer'); + +//console.time('BDAY timer'); + // ------------------------------------------------------------------------------------- // + // BDAY + if(globalDisabledContactAttributes.indexOf('BDAY')==-1) + { + vcard_element=vcard.match(vCard.pre['contentline_BDAY']); + if(vcard_element!=null) + { + if(vcard_element.length!=1) // if the BDAY attribute is present more than once, vCard is considered invalid + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; + } + + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + + var valid=true; + try {var date=$.datepicker.parseDate('yy-mm-dd', parsed[4])} + catch (e) {valid=false} + + if(valid==true) + { + tmpvCardEditorRef.find('[data-type="date_bday"]').val(vcardUnescapeValue($.datepicker.formatDate(globalSettings.datepickerformat.value, date))).change(); + + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['contentline_BDAY'][0]=vCard.tplC['contentline_BDAY']; + vCard.tplM['contentline_BDAY'][0]=vCard.tplM['contentline_BDAY'][0].replace('##:::##group_wd##:::##', parsed[1]); + vCard.tplM['contentline_BDAY'][0]=vCard.tplM['contentline_BDAY'][0].replace('##:::##params_wsc##:::##', parsed[3]); + + // remove the processed parameter + vcard=vcard.replace(vcard_element[0],'\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while((vcard_element_related=vcard.match(re))!=null) + { + // append the parameter to its parent + vCard.tplM['contentline_BDAY'][0]+=vcard_element_related[0].substr(2); + // remove the processed parameter + vcard=vcard.replace(vcard_element_related[0],'\r\n'); + } + } + } + else + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; // if the date value is invalid, vCard is considered invalid + } + } + } +//console.timeEnd('BDAY timer'); + +//console.time('X-ABDATE timer'); + // ------------------------------------------------------------------------------------- // + // X-ABDATE + if(globalDisabledContactAttributes.indexOf('X-ABDATE')==-1) + { + var element_i=0; + while((vcard_element=vcard.match(vCard.pre['contentline_X-ABDATE']))!=null) + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + var parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + + var valid=true; + try {var date=$.datepicker.parseDate('yy-mm-dd', parsed[4])} + catch (e) {valid=false} + + if(valid==true) + { + // parsed_value = [1..]->X-ABDATE-params + var parsed_value=vcardSplitParam(parsed[3]); + + // click to "add" button if not enought data rows present + var tmp_sel=tmpvCardEditorRef.find('[data-type="\\%date"]').last(); + if(tmp_sel.find('[data-type="date_value"]').val()!='') + tmp_sel.find('[data-type="\\%add"]').find('input[type="image"]').click(); + + // get the "TYPE=" values array + var pref=0; //by default there is no preferred date + var type_values=Array(); + var j=0; + for(var i=1; i array('HOME','INTERNET') -> 'home,internet' + var type_values_txt_label=type_values_us.join(' ').replace(vCard.pre['vcardToData_colon_begin_or_end'], ''); // TYPE=INTERNET;TYPE=INTERNET;TYPE=HOME; -> array('HOME','INTERNET') -> 'home internet' + if(type_values_txt=='') // if no person type defined, we use the 'other' type as default + type_values_txt=type_values_txt_label='other'; + + // get the default available types + var type_list=new Array(); + tmpvCardEditorRef.find('[data-type="\\%date"]:eq('+element_i+')').find('[data-type="date_type"]').children().each(function(index, element){type_list[type_list.length]=$(element).attr('data-type');}); + + // if an existing type regex matches the new type, use the old type + // and replace the old type definition with new type definition to comforn the server vCard type format + for(var i=0; i TODO: what to do if present more than once? + if(globalDisabledContactAttributes.indexOf('TITLE')==-1) + { + vcard_element=vcard.match(vCard.pre['contentline_TITLE']); + if(vcard_element!=null) + { + if(vcard_element.length!=1) // if the TITLE attribute is present more than once, vCard is considered invalid + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; + } + + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + + tmpvCardEditorRef.find('[data-type="title"]').val(vcardUnescapeValue(parsed[4])); + + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['contentline_TITLE'][0]=vCard.tplC['contentline_TITLE']; + vCard.tplM['contentline_TITLE'][0]=vCard.tplM['contentline_TITLE'][0].replace('##:::##group_wd##:::##', parsed[1]); + vCard.tplM['contentline_TITLE'][0]=vCard.tplM['contentline_TITLE'][0].replace('##:::##params_wsc##:::##', parsed[3]); + + // remove the processed parameter + vcard=vcard.replace(vcard_element[0],'\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while((vcard_element_related=vcard.match(re))!=null) + { + // append the parameter to its parent + vCard.tplM['contentline_TITLE'][0]+=vcard_element_related[0].substr(2); + // remove the processed parameter + vcard=vcard.replace(vcard_element_related[0],'\r\n'); + } + } + } + } +//console.timeEnd('TITLE timer'); + +//console.time('ORG timer'); + // ------------------------------------------------------------------------------------- // + // ORG -> TODO: what to do if present more than once? + if(globalDisabledContactAttributes.indexOf('ORG')==-1) + { + vcard_element=vcard.match(vCard.pre['contentline_ORG']); + if(vcard_element!=null) + { + if(vcard_element.length!=1) // if the ORG attribute is present more than once, vCard is considered invalid + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; + } + + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + // parsed_value = [0]->Org, [1..]->Org Units + parsed_value=vcardSplitValue(parsed[4], ';'); + + if(parsed_value[0]!=undefined && parsed_value[0]!='') + tmpvCardEditorRef.find('[data-type="org"]').val(vcardUnescapeValue(parsed_value[0])); + if(parsed_value[1]!=undefined && parsed_value[1]!='') + tmpvCardEditorRef.find('[data-type="department"]').val(vcardUnescapeValue(parsed_value[1])); + + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['contentline_ORG'][0]=vCard.tplC['contentline_ORG']; + vCard.tplM['contentline_ORG'][0]=vCard.tplM['contentline_ORG'][0].replace('##:::##group_wd##:::##', parsed[1]); + vCard.tplM['contentline_ORG'][0]=vCard.tplM['contentline_ORG'][0].replace('##:::##params_wsc##:::##', parsed[3]); + vCard.tplM['contentline_ORG'][0]=vCard.tplM['contentline_ORG'][0].replace('##:::##units_wsc##:::##', (parsed_value[2]==undefined ? '' : ';'+parsed_value.slice(2).join(';'))); + + // remove the processed parameter + vcard=vcard.replace(vcard_element[0],'\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while((vcard_element_related=vcard.match(re))!=null) + { + // append the parameter to its parent + vCard.tplM['contentline_ORG'][0]+=vcard_element_related[0].substr(2); + // remove the processed parameter + vcard=vcard.replace(vcard_element_related[0],'\r\n'); + } + } + } + } +//console.timeEnd('ORG timer'); + +//console.time('X-ABShowAs timer'); + // ------------------------------------------------------------------------------------- // + // X-ABShowAs -> TODO: what to do if present more than once? + var photo_show_org=false; + if(globalDisabledContactAttributes.indexOf('X-ABShowAs')==-1) + { + vcard_element=vcard.match(vCard.pre['X-ABShowAs']); + if(vcard_element!=null) + { + if(vcard_element.length>1) // if the X-ABShowAs attribute is present more than once, vCard is considered invalid + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; + } + + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + + if(vcardUnescapeValue(parsed[4]).match(RegExp('^company$','i'))) + { + tmpvCardEditorRef.find('[data-type="isorg"]').prop('checked', true); + photo_show_org=true; + } + + // values not directly supported by the editor (old values are kept intact) + vCard.tplM['contentline_X-ABShowAs'][0]=vCard.tplC['contentline_X-ABShowAs']; + vCard.tplM['contentline_X-ABShowAs'][0]=vCard.tplM['contentline_X-ABShowAs'][0].replace('##:::##group_wd##:::##', parsed[1]); + vCard.tplM['contentline_X-ABShowAs'][0]=vCard.tplM['contentline_X-ABShowAs'][0].replace('##:::##params_wsc##:::##', parsed[3]); + vCard.tplM['contentline_X-ABShowAs'][0]=vCard.tplM['contentline_X-ABShowAs'][0].replace('##:::##value##:::##', parsed[4]); + + // remove the processed parameter + vcard=vcard.replace(vcard_element[0],'\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'im'); + while((vcard_element_related=vcard.match(re))!=null) + { + // append the parameter to its parent + vCard.tplM['contentline_X-ABShowAs'][0]+=vcard_element_related[0].substr(2); + // remove the processed parameter + vcard=vcard.replace(vcard_element_related[0],'\r\n'); + } + } + } + } +//console.timeEnd('X-ABShowAs timer'); + +//console.time('PHOTO timer'); + // ------------------------------------------------------------------------------------- // + // PHOTO -> TODO: what to do if present more than once? + if(photo_show_org) + tmpvCardEditorRef.find('#photo').toggleClass('photo_user photo_company'); + + if(globalDisabledContactAttributes.indexOf('PHOTO')==-1) + { + vcard_element=vcard.match(vCard.pre['contentline_PHOTO']); + if(vcard_element!=null) // if the PHOTO attribute is present more than once, we use the first value + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + + var img_type = ''; + var custom_params = ''; + var typeRe = RegExp('TYPE=(.*)', 'i'); + var othersRe = RegExp('(?:ENCODING|VALUE)=.*', 'i'); + + parsed_value = vcardSplitParam(parsed[3]); + + for(i=1; i"group.", [2]->"name", [3]->";param;param", [4]->"value" + var parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + // parsed_param = [1..]->ADR-params + var parsed_param=vcardSplitParam(parsed[3]); + // parsed_value = [1..]->ADR elements + var parsed_value=vcardSplitValue(parsed[4],';'); + + // click to "add" button if not enought data rows present + var found=0; + tmpvCardEditorRef.find('[data-type="\\%address"]').last().find('[data-type="value"]').each( + function(index,element) + { + if($(element).val()!='') + { + found=1; + return false; + } + } + ); + if(found) + tmpvCardEditorRef.find('[data-type="\\%address"]').last().find('[data-type="\\%add"]').find('input[type="image"]').click(); + + // get the "TYPE=" values array + var pref=0; //by default there is no preferred address + var type_values=Array(); + var j=0; + for(var i=1; i array('FAX','HOME') -> 'fax,home' + var type_values_txt_label=type_values_us.join(' ').replace(vCard.pre['vcardToData_colon_begin_or_end'], ''); // TYPE=HOME;TYPE=HOME;TYPE=FAX; -> array('FAX','HOME') -> 'fax home' + if(type_values_txt=='') // if no address type defined, we use the 'work' type as default + type_values_txt=type_values_txt_label='work'; + + // get the default available types + var type_list=new Array(); + tmpvCardEditorRef.find('[data-type="\\%address"]:eq('+element_i+')').find('[data-type="address_type"]').children().each(function(index, element){type_list[type_list.length]=$(element).attr('data-type');}); + + // if an existing type regex matches the new type, use the old type + // and replace the old type definition with new type definition to comforn the server vCard type format + for(var i=0;i0 || (found=tmp.find('[data-type="country_type"]').children('[data-full-name="'+jqueryEscapeSelector(parsed_value[6])+'"]')).length>0) + found.prop('selected', true); + else if(globalSettings.addresscountryequivalence.value.length>0 && parsed_value[6]!=undefined) // unknown ADR format (country not detected) + { +// TODO: move regex object directly into config.js + for(var i=0; i"group.", [2]->"name", [3]->";param;param", [4]->"value" + var parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + // parsed_value = [1..]->TEL-params + var parsed_value=vcardSplitParam(parsed[3]); + + // click to "add" button if not enought data rows present + var tmp_sel=tmpvCardEditorRef.find('[data-type="\\%phone"]').last(); + if(tmp_sel.find('[data-type="value"]').val()!='') + tmp_sel.find('[data-type="\\%add"]').find('input[type="image"]').click(); + + // get the "TYPE=" values array + var pref=0; //by default there is no preferred phone number + var type_values=Array(); + var j=0; + for(var i=1; i array('FAX','HOME') -> 'fax,home' + var type_values_txt_label=type_values_us.join(' ').replace(vCard.pre['vcardToData_colon_begin_or_end'], ''); // TYPE=HOME;TYPE=HOME;TYPE=FAX; -> array('FAX','HOME') -> 'fax home' + if(type_values_txt=='') // if no phone type defined, we use the 'cell' type as default + type_values_txt=type_values_txt_label='cell'; + + // get the default available types (optimize in future) + var type_list=new Array(); + tmpvCardEditorRef.find('[data-type="\\%phone"]:eq('+element_i+')').find('[data-type="phone_type"]').children().each(function(index, element){type_list[type_list.length]=$(element).attr('data-type');}); + + // if an existing type regex matches the new type, use the old type + // and replace the old type definition with new type definition to comforn the current vCard type format + for(var i=0; i"group.", [2]->"name", [3]->";param;param", [4]->"value" + var parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + // parsed_value = [1..]->EMAIL-params + var parsed_value=vcardSplitParam(parsed[3]); + + // click to "add" button if not enought data rows present + var tmp_sel=tmpvCardEditorRef.find('[data-type="\\%email"]').last(); + if(tmp_sel.find('[data-type="value"]').val()!='') + tmp_sel.find('[data-type="\\%add"]').find('input[type="image"]').click(); + + // get the "TYPE=" values array + var pref=0; //by default there is no preferred email address + var type_values=Array(); + var j=0; + for(var i=1; i array('HOME','INTERNET') -> 'home,internet' + var type_values_txt_label=type_values_us.join(' ').replace(vCard.pre['vcardToData_colon_begin_or_end'], ''); // TYPE=INTERNET;TYPE=INTERNET;TYPE=HOME; -> array('HOME','INTERNET') -> 'home internet' + if(type_values_txt=='') // if no email type defined, we use the 'home' type as default + type_values_txt=type_values_txt_label='home,internet'; + + // get the default available types + var type_list=new Array(); + tmpvCardEditorRef.find('[data-type="\\%email"]:eq('+element_i+')').find('[data-type="email_type"]').children().each(function(index, element){type_list[type_list.length]=$(element).attr('data-type');}); + + // if an existing type regex matches the new type, use the old type + // and replace the old type definition with new type definition to comforn the server vCard type format + for(var i=0; i"group.", [2]->"name", [3]->";param;param", [4]->"value" + var parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + // parsed_value = [1..]->X-SOCIALPROFILE-params + var parsed_value=vcardSplitParam(parsed[3]); + + // click to "add" button if not enought data rows present + var tmp_sel=tmpvCardEditorRef.find('[data-type="\\%profile"]').last(); + if(tmp_sel.find('[data-type="value"]').val()!='') + tmp_sel.find('[data-type="\\%add"]').find('input[type="image"]').click(); + + // get the "TYPE=" values array + var pref=0; //by default there is no preferred X-SOCIALPROFILE + var type_values=Array(); + var j=0; + var social_user=''; + for(i=1;i array('B','A','C') -> 'a,b,c' + var type_values_txt_label=type_values_us.join(' ').replace(vCard.pre['vcardToData_colon_begin_or_end'], ''); // TYPE=B;TYPE=A;TYPE=C; -> array('B','A','C') -> 'a b c' + if(type_values_txt=='') // if no X-SOCIALPROFILE type defined, we use the 'twitter' type as default + type_values_txt=type_values_txt_label='twitter'; + + // get the default available types + var type_list=new Array(); + tmpvCardEditorRef.find('[data-type="\\%profile"]:eq('+element_i+')').find('[data-type="profile_type"]').children().each(function(index, element){type_list[type_list.length]=$(element).attr('data-type');}); + + // if an existing type regex matches the new type, use the old type + // and replace the old type definition with new type definition to comforn the server vCard type format + for(var i=0; i"group.", [2]->"name", [3]->";param;param", [4]->"value" + var parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + // parsed_value = [1..]->URL-params + var parsed_value=vcardSplitParam(parsed[3]); + + // click to "add" button if not enought data rows present + var tmp_sel=tmpvCardEditorRef.find('[data-type="\\%url"]').last(); + if(tmp_sel.find('[data-type="value"]').val()!='') + tmp_sel.find('[data-type="\\%add"]').find('input[type="image"]').click(); + + // get the "TYPE=" values array + var pref=0; //by default there is no preferred url address + var type_values=Array(); + var j=0; + for(var i=1; i array('HOME','WORK') -> 'home,work' + var type_values_txt_label=type_values_us.join(' ').replace(vCard.pre['vcardToData_colon_begin_or_end'], ''); // TYPE=WORK;TYPE=WORK;TYPE=HOME; -> array('HOME','WORK') -> 'home work' + if(type_values_txt=='') // if no url type defined, we use the 'homepage' type as default + type_values_txt=type_values_txt_label='homepage'; + + // get the default available types (optimize in future) + var type_list=new Array(); + tmpvCardEditorRef.find('[data-type="\\%url"]:eq('+element_i+')').find('[data-type="url_type"]').children().each(function(index, element){type_list[type_list.length]=$(element).attr('data-type');}); + + // if an existing type regex matches the new type, use the old type + // and replace the old type definition with new type definition to comforn the server vCard type format + for(var i=0; i"group.", [2]->"name", [3]->";param;param", [4]->"value" + var parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + // parsed_value = [1..]->X-ABRELATEDNAMES-params + var parsed_value=vcardSplitParam(parsed[3]); + + // click to "add" button if not enought data rows present + var tmp_sel=tmpvCardEditorRef.find('[data-type="\\%person"]').last(); + if(tmp_sel.find('[data-type="value"]').val()!='') + tmp_sel.find('[data-type="\\%add"]').find('input[type="image"]').click(); + + // get the "TYPE=" values array + var pref=0; //by default there is no preferred person + var type_values=Array(); + var j=0; + for(var i=1; i array('HOME','INTERNET') -> 'home,internet' + var type_values_txt_label=type_values_us.join(' ').replace(vCard.pre['vcardToData_colon_begin_or_end'], ''); // TYPE=INTERNET;TYPE=INTERNET;TYPE=HOME; -> array('HOME','INTERNET') -> 'home internet' + if(type_values_txt=='') // if no person type defined, we use the 'other' type as default + type_values_txt=type_values_txt_label='other'; + + // get the default available types + var type_list=new Array(); + tmpvCardEditorRef.find('[data-type="\\%person"]:eq('+element_i+')').find('[data-type="person_type"]').children().each(function(index, element){type_list[type_list.length]=$(element).attr('data-type');}); + + // if an existing type regex matches the new type, use the old type + // and replace the old type definition with new type definition to comforn the server vCard type format + for(var i=0; i"group.", [2]->"name", [3]->";param;param", [4]->"value" + var parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + // parsed_value = [1..]->IMPP-params + var parsed_value=vcardSplitParam(parsed[3]); + + // click to "add" button if not enought data rows present + var tmp_sel=tmpvCardEditorRef.find('[data-type="\\%im"]').last(); + if(tmp_sel.find('[data-type="value"]').val()!='') + tmp_sel.find('[data-type="\\%add"]').find('input[type="image"]').click(); + + // get the "TYPE=" & "X-SERVICE-TYPE" values array + var pref=0; //by default there is no preferred IM + var type_values=Array(); + var j=0; + var service_type_value=''; + for(var i=1; i array('HOME','INTERNET') -> 'home,internet' + type_values_txt_label=type_values_us.join(' ').replace(vCard.pre['vcardToData_colon_begin_or_end'], ''); // TYPE=INTERNET;TYPE=INTERNET;TYPE=HOME; -> array('HOME','INTERNET') -> 'home internet' + if(type_values_txt=='') // if no IMPP type defined, we use the 'other' type as default + type_values_txt=type_values_txt_label='other'; + + // get the default available types + var type_list=new Array(); + tmpvCardEditorRef.find('[data-type="\\%im"]:eq('+element_i+')').find('[data-type="im_type"]').children().each(function(index, element){type_list[type_list.length]=$(element).attr('data-type');}); + + // if an existing type regex matches the new type, use the old type + // and replace the old type definition with new type definition to comforn the server vCard type format + for(var i=0; i Cancel support /loading the previous active contact/) + if(inputContact.uid!=undefined) // occurs if loadContactByVcard is used (it also appends the UID of previous contact into 'data-id') + tmpvCardEditorRef.find('#vCardEditor').find('[data-type="cancel"]').attr('data-id', inputContact.uid); + + processEditorElements(tmpvCardEditorRef, inputEditorMode, inputIsReadonly, inputContact.isCompany); + + var tmp_optionslist=[]; + // create the list of available collections to the interface + for(var i=0; i').attr({'data-type': globalResourceCardDAVList.collections[i].uid, 'data-color': globalResourceCardDAVList.collections[i].color}).text(globalResourceCardDAVList.collections[i].displayvalue); + // add the list of available collections to the interface + tmpvCardEditorRef.find('[data-attr-name="_DEST_"]').append(tmp_optionslist); + // bind the change event (color change in the editor) + tmpvCardEditorRef.find('[data-attr-name="_DEST_"]').change(function(){ + var selColl=globalResourceCardDAVList.getCollectionByUID($(this).find('option:selected').attr('data-type')); + globalRefAddContact.attr('data-url', selColl.uid.replace(RegExp('[^/]+$'),'')); + globalRefAddContact.attr('data-filter-url',selColl.uid); // Set the current addressbook filter uid + globalRefAddContact.attr('data-account-uid',selColl.accountUID); + $('#ABContactColor').css('background-color', $(this).find('option:selected').attr('data-color')); + }); + + var collUID=''; + if(typeof inputContact.uid!='undefined') + collUID= inputContact.uid.replace(RegExp('[^/]*$'),''); + else + collUID = globalRefAddContact.attr('data-url'); + var select_elem=tmpvCardEditorRef.find('[data-attr-name="_DEST_"]').find('[data-type="'+jqueryEscapeSelector(collUID)+'"]'); + if(select_elem.length==1) + select_elem.prop('selected', true); + + if(typeof globalContactsExtVcardToData!='undefined' && !inputIsCompany) + tmpvCardEditorRef.find('[data-type="DEST"]').addClass('element_no_display'); + + // Unprocessed unrelated vCard elements + vCard.tplM['unprocessed_unrelated']=vcard; + + if(typeof globalDebug!='undefined' && globalDebug instanceof Array && globalDebug.indexOf('vcard')!=-1) + { + console.timeEnd('vcardToData timer'); + + if(vcard!='\r\n') + console.log('Warning: [vCard unprocessed unrelated]: '+vcard); + } + + //clean error message + $('#ABMessage').height('0'); + + $('#ABContact').empty().append(tmpvCardEditorRef); + + var foundGroup=0; + for(var adr in globalAddressbookList.vcard_groups) + { + if(globalAddressbookList.vcard_groups[adr].length>0) + { + foundGroup=1; + break; + } + } + + if(foundGroup) + { + if(typeof inputContact.uid!='undefined') + extendDestSelect(); + else + { + var selGroup = $('#ResourceCardDAVList').find('.contact_group').find(':input.resourceCardDAV_selected').attr('data-id'); + extendDestSelect(selGroup); + if(typeof selGroup!= 'undefined') + select_elem.text(localization[globalInterfaceLanguage].txtVcardGroupsTextSingle.replace('%coll%',globalResourceCardDAVList.getCollectionByUID(collUID).displayvalue)); + } + } + if(typeof inputContact.uid !='undefined') + checkForVcardGroups(inputContact.uid); + if(typeof(globalContactsSelectProcess)=='function') + globalContactsSelectProcess(tmpvCardEditorRef, inputContact); + + return true; + } + else + { + console.log("Error: '"+inputContact.uid+"': unable to parse vCard"); + return false; + } +} + +function basicRFCFixesAndCleanup(vcardString) +{ + // If vCard contains only '\n' instead of '\r\n' we fix it + if(vcardString.match(vCard.pre['basicRFCFixesAndCleanup_r-m'])==null) + vcardString=vcardString.replace(vCard.pre['basicRFCFixesAndCleanup_n-gm'], '\r\n'); + + // remove multiple empty lines + vcardString=vcardString.replace(vCard.pre['basicRFCFixesAndCleanup_rnp-gm'], '\r\n'); + + // append '\r\n' to the end of the vCard if missing + if(vcardString[vcardString.length-1]!='\n') + vcardString+='\r\n'; + + // remove line folding + vcardString=vcardString.replace(vCard.pre['basicRFCFixesAndCleanup_rnwsp-gm'], ''); + + // RFC-obsolete PHOTO fix + vcardString=vcardString.replace(vCard.pre['basicRFCFixesAndCleanup_photo-gim'], '\r\nPHOTO:'); + + // ------------------------------------------------------------------------------------- // + // begin CATEGORIES merge to one CATEGORIES attribute (sorry for related attributes) + // note: we cannot do this in additionalRFCFixes or normalizeVcard + var categoriesArr=[]; + var vcard_element=null; + var vcard_element_related=null; + while((vcard_element=vcardString.match(vCard.pre['contentline_CATEGORIES']))!=null) + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + var parsed=vcard_element[0].match(vCard.pre['contentline_parse']); + + categoriesArr[categoriesArr.length]=parsed[4]; + + // remove the processed parameter + vcardString=vcardString.replace(vcard_element[0],'\r\n'); + + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'm'); + while((vcard_element_related=vcardString.match(re))!=null) + // remove the processed parameter + vcardString=vcardString.replace(vcard_element_related[0],'\r\n'); + } + } + var categoriesTxt=categoriesArr.join(','); + + var tmp=vcardString.split('\r\n'); + tmp.splice(tmp.length-2,0,'CATEGORIES:'+categoriesTxt); + // end CATEGORIES cleanup + // ------------------------------------------------------------------------------------- // + + // ------------------------------------------------------------------------------------- // + // begin SoGo fixes (company vCards without N and FN attributes) + // we must perform vCard fixes here because the N and FN attributes are used in the collection list + + // if N attribute is missing we add it + if(vcardString.match(vCard.pre['contentline_N'])==null) + tmp.splice(1,0,'N:;;;;'); + + // if FN attribute is missing we add it + if(vcardString.match(vCard.pre['contentline_FN'])==null) + { + var fn_value=''; + var tmp2=null; + // if there is an ORG attribute defined, we use the company name as fn_value (instead of empty string) + if((tmp2=vcardString.match(vCard.pre['contentline_ORG']))!=null) + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + var parsed=tmp2[0].match(vCard.pre['contentline_parse']); + // parsed_value = [0]->Org, [1..]->Org Units + var parsed_value=vcardSplitValue(parsed[4],';'); + fn_value=parsed_value[0]; + } + tmp.splice(1,0,'FN:'+fn_value); + } + vcardString=tmp.join('\r\n'); + // end SoGo fixes + // ------------------------------------------------------------------------------------- // + + return {vcard: vcardString, categories: categoriesTxt}; +} + +function additionalRFCFixes(vcardString) +{ + // ------------------------------------------------------------------------------------- // + var tmp=vcardString.split('\r\n'); + + // update non-RFC attributes (special transformations) + for(var i=1;i"group.", [2]->"name", [3]->";param;param", [4]->"value" + var parsed=('\r\n'+tmp[i]+'\r\n').match(vCard.pre['contentline_parse']); + + if(parsed!=null) + { + switch(parsed[2]) + { + case 'TEL': + // remove the non-RFC params (Evolution bug) + var parsed_value=vcardSplitParam(parsed[3]); + for(var j=parsed_value.length-1;j>0;j--) + if(parsed_value[j].match(vCard.pre['additionalRFCFixes_tel-param'])==null) + parsed_value.splice(j,1); + + parsed[3]=parsed_value.join(';'); + tmp[i]=parsed[1]+parsed[2]+parsed[3]+':'+parsed[4]; + break; + case 'EMAIL': + // transform the params separated by ',' to 'TYPE=' params and remove the non-RFC params (Evolution bug) + var parsed_value=vcardSplitParam(parsed[3]); + for(var j=parsed_value.length-1;j>0;j--) + if(parsed_value[j].match(vCard.pre['additionalRFCFixes_email-param'])==null) + { + if((transformed=parsed_value[j].replace(vCard.pre['additionalRFCFixes_comma-g'], ';TYPE=')).match(vCard.pre['additionalRFCFixes_email-params'])!=null) + parsed_value[j]=transformed; + else + parsed_value.splice(j,1); + } + + parsed[3]=parsed_value.join(';'); + // add missing and required "internet" type (Sogo bug) + if(parsed[3].match(vCard.pre['additionalRFCFixes_type-internet'])==null) + parsed[3]+=';TYPE=INTERNET'; + + tmp[i]=parsed[1]+parsed[2]+parsed[3]+':'+parsed[4]; + break; +// the upcoming vCard 4.0 allows params for URL and many clients use it also in vCard 3.0 +// case 'URL': // no params allowed for URL (Evolution bug) +// tmp[i]=parsed[1]+parsed[2]+':'+parsed[4]; +// break; + default: + break; + } + } + } + vcardString=tmp.join('\r\n'); + // ------------------------------------------------------------------------------------- // + + return vcardString; +} + +// transform the vCard to the editor expected format +function normalizeVcard(vcardString) +{ + var parsed=null; + // remove the PRODID element (unusable for the editor) + while((parsed=vcardString.match(vCard.pre['contentline_PRODID']))!=null) + vcardString=vcardString.replace(parsed[0],'\r\n'); + + var tmp=vcardString.split('\r\n'); + var vcard_begin=tmp[0].replace(vCard.pre['normalizeVcard_group_w_dot'], 'item.')+'\r\n'; + var vcard_end=tmp[tmp.length-2].replace(vCard.pre['normalizeVcard_group_w_dot'], 'item.')+'\r\n'; + // remove the vCard BEGIN and END and all duplicate entries (usually created by other buggy clients) + vcardString='\r\n'+tmp.slice(1, tmp.length-2).join('\r\n')+'\r\n'; + + var vcard_out_grouped=new Array(); + while((parsed=vcardString.match(vCard.pre['contentline_parse']))!=null) + { + var additional_related=''; + var vcard_element_related=''; + var attr_name=''; + var params_swc=''; + var attr_value=''; + + // parsed = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + var params_array=vcardSplitParam(parsed[3]); + // we transform the old X-* IM attributes to new IMPP (internally used by editor) + switch(parsed[2]) + { + case 'X-ABDATE': + attr_name=parsed[2]; + params_swc=params_array.sort().join(';').toUpperCase(); // we need upper case here to remove duplicate values later + tmp=parsed[4].match(vCard.pre['normalizeVcard_date']); + attr_value=tmp[1]+'-'+tmp[2]+'-'+tmp[3]; // sorry, we support only date (no date-time support) + break; + case 'X-EVOLUTION-ANNIVERSARY': + case 'X-ANNIVERSARY': + attr_name='X-ABDATE'; + params_swc=''; + tmp=parsed[4].match(vCard.pre['normalizeVcard_date']); + attr_value=tmp[1]+'-'+tmp[2]+'-'+tmp[3]; // sorry, we support only date (no date-time support) + additional_related='X-ABLabel:_$!!$_\r\n'; + + // check for X-ABDATE attribute with the same value + var found=false; + var tmpVcardString=vcardString; + var tmp_vcard_element=null; + while((tmp_vcard_element=tmpVcardString.match(vCard.pre['contentline_X-ABDATE']))!=null) + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + var tmp_parsed=tmp_vcard_element[0].match(vCard.pre['contentline_parse']); + + if(tmp_parsed[4]==parsed[4] || tmp_parsed[4]==attr_value) + { + found=true; + break; + } + tmpVcardString=tmpVcardString.replace(tmp_vcard_element[0], '\r\n'); + } + + if(found==true) + { + // remove the processed element + vcardString=vcardString.replace(parsed[0], '\r\n'); + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.', '\\..*')+'\r\n', 'm'); + while((vcard_element_related=vcardString.match(re))!=null) + vcardString=vcardString.replace(vcard_element_related[0], '\r\n'); // remove the processed parameter + } + continue; + } + break; + case 'BDAY': + attr_name=parsed[2]; + params_swc=';VALUE=date'; + tmp=parsed[4].match(vCard.pre['normalizeVcard_date']); + attr_value=tmp[1]+'-'+tmp[2]+'-'+tmp[3]; // sorry, we support only date (no date-time support) + break; + case 'X-AIM': + case 'X-JABBER': + case 'X-MSN': + case 'X-YAHOO': + case 'X-YAHOO-ID': + case 'X-ICQ': + case 'X-SKYPE': + attr_name='IMPP'; + if(params_array.length==0) + params_array[0]=''; // after the join it generates ';' after the attribute name + params_array[params_array.length]='X-SERVICE-TYPE='+parsed[2].replace(vCard.pre['normalizeVcard_xb_or_ide'], ''); // extract the IM type + params_swc=params_array.sort().join(';'); + attr_value=parsed[4]; + + // check for IMPP attribute with the same value + var found=false; + var tmpVcardString=vcardString; + var tmp_vcard_element=null; + while((tmp_vcard_element=tmpVcardString.match(vCard.pre['contentline_IMPP']))!=null) + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + var tmp_parsed=tmp_vcard_element[0].match(vCard.pre['contentline_parse']); + + if(tmp_parsed[4].replace(vCard.pre['normalizeVcard_before_val'], '')==parsed[4]) + { + found=true; + break; + } + tmpVcardString=tmpVcardString.replace(tmp_vcard_element[0], '\r\n'); + } + + if(found==true) + { + // remove the processed element + vcardString=vcardString.replace(parsed[0], '\r\n'); + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.', '\\..*')+'\r\n', 'm'); + while((vcard_element_related=vcardString.match(re))!=null) + vcardString=vcardString.replace(vcard_element_related[0], '\r\n'); // remove the processed parameter + } + continue; + } + break; + case 'IMPP': + attr_name=parsed[2]; + params_swc=params_array.sort().join(';').toUpperCase(); // we need upper case here to remove duplicate values later + + // remove the '*:' from the '*:value' + // but we add them back during the vcard generation from the interface + attr_value=vcardSplitValue(parsed[4], ':').splice(1, 1).join('') + break; + case 'X-ASSISTANT': + case 'X-EVOLUTION-ASSISTANT': + attr_name='X-ABRELATEDNAMES'; + params_swc=''; + attr_value=parsed[4]; + additional_related='X-ABLabel:_$!!$_\r\n'; + + // check for X-ABRELATEDNAMES attribute with the same value + var found=false; + var tmpVcardString=vcardString; + var tmp_vcard_element=null; + while((tmp_vcard_element=tmpVcardString.match(vCard.pre['contentline_X-ABRELATEDNAMES']))!=null) + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + var tmp_parsed=tmp_vcard_element[0].match(vCard.pre['contentline_parse']); + + if(tmp_parsed[4]==parsed[4]) + { + found=true; + break; + } + tmpVcardString=tmpVcardString.replace(tmp_vcard_element[0], '\r\n'); + } + + if(found==true) + { + // remove the processed element + vcardString=vcardString.replace(parsed[0], '\r\n'); + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.', '\\..*')+'\r\n', 'm'); + while((vcard_element_related=vcardString.match(re))!=null) + vcardString=vcardString.replace(vcard_element_related[0], '\r\n'); // remove the processed parameter + } + continue; + } + break; + case 'X-MANAGER': + case 'X-EVOLUTION-MANAGER': + attr_name='X-ABRELATEDNAMES'; + params_swc=''; + attr_value=parsed[4]; + additional_related='X-ABLabel:_$!!$_\r\n'; + + // check for X-ABRELATEDNAMES attribute with the same value + var found=false; + var tmpVcardString=vcardString; + var tmp_vcard_element=null; + while((tmp_vcard_element=tmpVcardString.match(vCard.pre['contentline_X-ABRELATEDNAMES']))!=null) + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + var tmp_parsed=tmp_vcard_element[0].match(vCard.pre['contentline_parse']); + + if(tmp_parsed[4]==parsed[4]) + { + found=true; + break; + } + tmpVcardString=tmpVcardString.replace(tmp_vcard_element[0], '\r\n'); + } + + if(found==true) + { + // remove the processed element + vcardString=vcardString.replace(parsed[0], '\r\n'); + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.', '\\..*')+'\r\n', 'm'); + while((vcard_element_related=vcardString.match(re))!=null) + vcardString=vcardString.replace(vcard_element_related[0], '\r\n'); // remove the processed parameter + } + continue; + } + break; + case 'X-SPOUSE': + case 'X-EVOLUTION-SPOUSE': + attr_name='X-ABRELATEDNAMES'; + params_swc=''; + attr_value=parsed[4]; + additional_related='X-ABLabel:_$!!$_\r\n'; + + // check for X-ABRELATEDNAMES attribute with the same value + var found=false; + var tmpVcardString=vcardString; + var tmp_vcard_element=null; + while((tmp_vcard_element=tmpVcardString.match(vCard.pre['contentline_X-ABRELATEDNAMES']))!=null) + { + // parsed (contentline_parse) = [1]->"group.", [2]->"name", [3]->";param;param", [4]->"value" + var tmp_parsed=tmp_vcard_element[0].match(vCard.pre['contentline_parse']); + + if(tmp_parsed[4]==parsed[4]) + { + found=true; + break; + } + tmpVcardString=tmpVcardString.replace(tmp_vcard_element[0], '\r\n'); + } + + if(found==true) + { + // remove the processed element + vcardString=vcardString.replace(parsed[0], '\r\n'); + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.', '\\..*')+'\r\n', 'm'); + while((vcard_element_related=vcardString.match(re))!=null) + vcardString=vcardString.replace(vcard_element_related[0], '\r\n'); // remove the processed parameter + } + continue; + } + break; + default: + attr_name=parsed[2]; + params_swc=params_array.sort().join(';'); + attr_value=parsed[4]; + break; + } + // remove the processed element + vcardString=vcardString.replace(parsed[0],'\r\n'); + + if(attr_name!='FN' && attr_name!='N' && attr_value=='') // attributes with empty values are not supported and are removed here + { + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\..*')+'\r\n', 'm'); + while((vcard_element_related=vcardString.match(re))!=null) + // remove the processed parameter + vcardString=vcardString.replace(vcard_element_related[0], '\r\n'); + } + continue; + } + + // add the new element to output array (without group) + grouped_elem=new Array(); + grouped_elem[grouped_elem.length]=attr_name+params_swc+':'+attr_value+'\r\n'; + if(additional_related!='') // used if we manually add related items as a part of transformation + grouped_elem[grouped_elem.length]=additional_related; + // find the corresponding group data (if exists) + if(parsed[1]!='') + { + var re=RegExp('\r\n'+parsed[1].replace('.','\\.(.*)')+'\r\n', 'm'); + while((vcard_element_related=vcardString.match(re))!=null) + { + // add the related element to array + grouped_elem[grouped_elem.length]=vcard_element_related[1]+'\r\n'; + // remove the processed parameter + vcardString=vcardString.replace(vcard_element_related[0], '\r\n'); + } + } + // add the new grouped element to output + vcard_out_grouped[vcard_out_grouped.length]=grouped_elem.sort().join(''); + } +// +// after the transformation and grouping we remove all identical elements and preserve sorting + // (for example X-AIM and IMPP;X-SERVICE-TYPE=AIM, ...) + for(var i=vcard_out_grouped.length-1;i>=0;i--) + if(vcard_out_grouped.slice(0,i).indexOf(vcard_out_grouped[i])!=-1) + vcard_out_grouped.splice(i,1); + + // add new group names ... + elemCounter=0; + for(i=0;i1) + vcard_out_grouped[i]=(('\r\n'+vcard_out_grouped[i].substring(0, vcard_out_grouped[i].length-2)).replace(vCard.pre['normalizeVcard_rn-gm'], '\r\nitem'+(elemCounter++)+'.')+'\r\n').substring(2); + + vcard_out_grouped.unshift(vcard_begin); + vcard_out_grouped.push(vcard_end); + + return vcard_out_grouped.join(''); +} -- cgit