egroupware_official/addressbook/inc/class.vcaladdressbook.inc.php

463 lines
12 KiB
PHP
Raw Normal View History

2005-07-20 14:14:39 +02:00
<?php
/**************************************************************************\
* eGroupWare - iCalendar Parser *
* http://www.egroupware.org *
* Written by Lars Kneschke <lkneschke@egroupware.org> *
* -------------------------------------------- *
* This program is free software; you can redistribute it and/or modify it *
* under the terms of the GNU General Public License as published by the *
* Free Software Foundation; either version 2 of the License. *
\**************************************************************************/
2005-07-28 05:44:13 +02:00
/* $Id$ */
2005-07-20 14:14:39 +02:00
require_once EGW_SERVER_ROOT.'/addressbook/inc/class.bocontacts.inc.php';
require_once EGW_SERVER_ROOT.'/phpgwapi/inc/horde/Horde/iCalendar.php';
2005-07-20 14:14:39 +02:00
class vcaladdressbook extends bocontacts
2005-07-20 14:14:39 +02:00
{
#function vcaladdressbook()
#{
# $this->boaddressbook();
#}
/**
* @return int contact id
* @param string $_vcard the vcard
* @param int $_abID the internal addressbook id
* @param int $_vcardProfile profile id for mapping from vcard values to egw addressbook
* @desc import a vard into addressbook
*/
2006-01-16 10:49:04 +01:00
function addVCard($_vcard, $_abID)
2005-07-20 14:14:39 +02:00
{
if(!($contact = $this->vcardtoegw($_vcard)))
{
return false;
2005-07-20 14:14:39 +02:00
}
if($_abID > 0) $contact['ab_id'] = $_abID;
return $this->save($contact);
2005-07-20 14:14:39 +02:00
}
/**
* return a vcard
*
* @param int $_id the id of the contact
2005-07-20 14:14:39 +02:00
* @param int $_vcardProfile profile id for mapping from vcard values to egw addressbook
* @return string containing the vcard
2005-07-20 14:14:39 +02:00
*/
2006-01-16 10:49:04 +01:00
function getVCard($_id)
2005-07-20 14:14:39 +02:00
{
require_once(EGW_SERVER_ROOT.'/phpgwapi/inc/horde/Horde/iCalendar/vcard.php');
2005-07-20 14:14:39 +02:00
$vCard =& new Horde_iCalendar_vcard;
2005-07-28 05:44:13 +02:00
2006-01-16 10:49:04 +01:00
if(!is_array($this->supportedFields))
{
$this->setSupportedFields();
}
2005-07-28 05:44:13 +02:00
#_debug_array($fields);
2005-07-20 14:14:39 +02:00
if(($entry = $this->read($_id)))
2005-07-20 14:14:39 +02:00
{
#_debug_array($entry);
$sysCharSet = $GLOBALS['egw']->translation->charset();
2005-07-20 14:14:39 +02:00
2006-01-16 10:49:04 +01:00
foreach($this->supportedFields as $vcardField => $databaseFields)
2005-07-20 14:14:39 +02:00
{
$options = array();
$value = '';
foreach($databaseFields as $databaseField)
{
$tempVal = ';';
if(!empty($databaseField))
2005-07-28 05:44:13 +02:00
{
2005-07-20 14:14:39 +02:00
#$tempVal = trim('value').';';
$tempVal = trim($entry[0][$databaseField]).';';
2005-07-28 05:44:13 +02:00
}
2005-07-20 14:14:39 +02:00
$value .= $tempVal;
}
// remove the last ;
$value = substr($value, 0, -1);
2005-07-28 05:44:13 +02:00
2005-07-20 14:14:39 +02:00
switch($vcardField)
{
// TODO handle multiple categories
case 'CATEGORIES':
$catData = ExecMethod('phpgwapi.categories.return_single',$value);
$value = $catData[0]['name'];
break;
2006-01-16 10:49:04 +01:00
case 'CLASS':
$value = $value ? 'PRIVATE' : 'PUBLIC';
2006-01-16 10:49:04 +01:00
break;
2005-07-20 14:14:39 +02:00
case 'BDAY':
if(!empty($value))
{
list($y,$m,$d) = explode('-',$value);
$value = sprintf('%04d%02d%02dT000000Z',$y,$m,$d);
2005-07-20 14:14:39 +02:00
}
break;
}
$value = $GLOBALS['egw']->translation->convert($value,$sysCharSet,'utf-8');
2005-07-20 14:14:39 +02:00
// don't add the entry if it contains only ';'
if(strlen(str_replace(';','',$value)) != 0)
2005-07-28 05:44:13 +02:00
{
2005-07-20 14:14:39 +02:00
$vCard->setAttribute($vcardField, $value);
2005-07-28 05:44:13 +02:00
}
2005-07-20 14:14:39 +02:00
if(preg_match('/([\000-\012\015\016\020-\037\075])/',$value))
2005-07-28 05:44:13 +02:00
{
2005-07-20 14:14:39 +02:00
$options['ENCODING'] = 'QUOTED-PRINTABLE';
2005-07-28 05:44:13 +02:00
}
2005-07-20 14:14:39 +02:00
if(preg_match('/([\177-\377])/',$value))
2005-07-28 05:44:13 +02:00
{
2005-07-20 14:14:39 +02:00
$options['CHARSET'] = 'UTF-8';
2005-07-28 05:44:13 +02:00
}
2005-07-20 14:14:39 +02:00
$vCard->setParameter($vcardField, $options);
}
2005-07-28 05:44:13 +02:00
2005-07-20 14:14:39 +02:00
$result = $vCard->exportvCalendar();
2005-07-28 05:44:13 +02:00
2005-07-20 14:14:39 +02:00
return $result;
}
return False;
}
2006-01-16 10:49:04 +01:00
/**
* Search an exactly matching entry (used for slow sync)
*
* @param string $_vcard
* @return boolean/int/string contact-id or false, if not found
*/
function search($_vcard)
{
if(!($contact = $this->vcardtoegw($_vcard)))
{
return false;
}
if(($foundContacts = $this->search($contact)))
{
return $foundContacts[0]['id'];
}
return false;
}
2006-01-16 10:49:04 +01:00
function setSupportedFields($_productManufacturer='file', $_productName='')
{
$defaultFields[0] = array(
'ADR' => array('','','adr_one_street','adr_one_locality','adr_one_region',
'adr_one_postalcode','adr_one_countryname'),
'CATEGORIES' => array('cat_id'),
'CLASS' => array('private'),
2006-01-16 10:49:04 +01:00
'EMAIL' => array('email'),
'N' => array('n_family','n_given','','',''),
'NOTE' => array('note'),
'ORG' => array('org_name',''),
'TEL;CELL' => array('tel_cell'),
'TEL;FAX' => array('tel_fax'),
'TEL;HOME' => array('tel_home'),
'TEL;WORK' => array('tel_work'),
'TITLE' => array('title'),
);
$defaultFields[1] = array(
'ADR;WORK' => array('','','adr_one_street','adr_one_locality','adr_one_region',
'adr_one_postalcode','adr_one_countryname'),
'ADR;HOME' => array('','','adr_two_street','adr_two_locality','adr_two_region',
'adr_two_postalcode','adr_two_countryname'),
'BDAY' => array('bday'),
'CATEGORIES' => array('cat_id'),
'EMAIL;INTERNET;WORK' => array('email'),
'EMAIL;INTERNET;HOME' => array('email_home'),
'N' => array('n_family','n_given','n_middle','n_prefix','n_suffix'),
'NOTE' => array('note'),
'ORG' => array('org_name','org_unit'),
'TEL;CELL;WORK' => array('tel_cell'),
'TEL;FAX;WORK' => array('tel_fax'),
'TEL;HOME' => array('tel_home'),
'TEL;PAGER;WORK' => array('tel_pager'),
'TEL;WORK' => array('tel_work'),
'TITLE' => array('title'),
'URL;WORK' => array('url'),
);
$defaultFields[2] = array(
'ADR;HOME' => array('','','adr_one_street','adr_one_locality','adr_one_region',
'adr_one_postalcode','adr_one_countryname'),
'BDAY' => array('bday'),
'CATEGORIES' => array('cat_id'),
'CLASS' => array('private'),
2006-01-16 10:49:04 +01:00
'EMAIL' => array('email'),
'N' => array('n_family','n_given','','',''),
'NOTE' => array('note'),
'ORG' => array('org_name',''),
'TEL;CELL;WORK' => array('tel_cell'),
'TEL;FAX;WORK' => array('tel_fax'),
2006-01-16 10:49:04 +01:00
'TEL;HOME' => array('tel_home'),
'TEL;WORK' => array('tel_work'),
'TITLE' => array('title'),
'URL' => array('url'),
);
$defaultFields[3] = array(
'ADR;WORK' => array('','','adr_one_street','adr_one_locality','adr_one_region',
'adr_one_postalcode','adr_one_countryname'),
'ADR;HOME' => array('','','adr_two_street','adr_two_locality','adr_two_region',
'adr_two_postalcode','adr_two_countryname'),
'BDAY' => array('bday'),
'EMAIL;INTERNET;WORK' => array('email'),
'EMAIL;INTERNET;HOME' => array('email_home'),
'N' => array('n_family','n_given','','',''),
'NOTE' => array('note'),
'ORG' => array('org_name','org_unit'),
'TEL;CELL;WORK' => array('tel_cell'),
'TEL;FAX;WORK' => array('tel_fax'),
'TEL;HOME' => array('tel_home'),
'TEL;PAGER;WORK' => array('tel_pager'),
'TEL;WORK' => array('tel_work'),
'TITLE' => array('title'),
'URL;WORK' => array('url'),
);
switch(strtolower($_productManufacturer))
{
case 'nexthaus corporation':
switch(strtolower($_productName))
{
case 'syncje outlook edition':
default:
$this->supportedFields = $defaultFields[1];
break;
}
break;
// multisync does not provide anymore information then the manufacturer
// we suppose multisync with evolution
case 'the multisync project':
switch(strtolower($_productName))
{
default:
$this->supportedFields = $defaultFields[0];
break;
}
break;
case 'siemens':
switch(strtolower($_productName))
{
case 'sx1':
default:
$this->supportedFields = $defaultFields[3];
break;
}
break;
case 'sonyericsson':
switch(strtolower($_productName))
{
case 'd750i':
default:
$this->supportedFields = $defaultFields[2];
break;
}
break;
case 'synthesis ag':
switch(strtolower($_productName))
{
default:
$this->supportedFields = $defaultFields[0];
break;
}
break;
case 'file': // used outside of SyncML, eg. by the calendar itself ==> all possible fields
$this->supportedFields = $defaultFields[1];
break;
// the fallback for SyncML
default:
error_log("Client not found: $_productManufacturer $_productName");
$this->supportedFields = $defaultFields[0];
break;
}
}
function vcardtoegw($_vcard)
{
if(!is_array($this->supportedFields))
{
$this->setSupportedFields();
}
$this->supportedFields[0] = array(
);
require_once(EGW_SERVER_ROOT.'/phpgwapi/inc/horde/Horde/iCalendar.php');
$vCard = Horde_iCalendar::newComponent('vcard', $container);
if(!$vCard->parsevCalendar($_vcard,'VCARD'))
{
return False;
}
$vcardValues = $vCard->getAllAttributes();
#print "<pre>$_vcard</pre>";
#_debug_array($vcardValues);
foreach($vcardValues as $key => $vcardRow)
{
$rowName = $vcardRow['name'];
if(isset($vcardRow['params']['INTERNET']))
{
$rowName .= ";INTERNET";
}
if(isset($vcardRow['params']['CELL']))
{
$rowName .= ';CELL';
}
if(isset($vcardRow['params']['FAX']))
{
$rowName .= ';FAX';
}
if(isset($vcardRow['params']['PAGER']))
{
$rowName .= ';PAGER';
}
if(isset($vcardRow['params']['WORK']))
{
$rowName .= ';WORK';
}
if(isset($vcardRow['params']['HOME']))
{
$rowName .= ';HOME';
}
$rowNames[$rowName] = $key;
}
#error_log('rowNames: '.print_r($rowNames,true));
// now we have all rowNames the vcard provides
// we just need to map to the right addressbook fieldnames
// we need also to take care about ADR for example. we do not
// support this. We support only ADR;WORK or ADR;HOME
foreach($rowNames as $rowName => $vcardKey)
{
switch($rowName)
{
case 'ADR':
case 'TEL':
case 'TEL;FAX':
case 'TEL;CELL':
case 'TEL;PAGER':
if(!isset($rowNames[$rowName. ';WORK']))
{
$finalRowNames[$rowName. ';WORK'] = $vcardKey;
}
break;
case 'EMAIL':
case 'EMAIL;WORK':
if(!isset($rowNames['EMAIL;INTERNET;WORK']))
{
$finalRowNames['EMAIL;INTERNET;WORK'] = $vcardKey;
}
break;
case 'EMAIL;HOME':
if(!isset($rowNames['EMAIL;INTERNET;HOME']))
{
$finalRowNames['EMAIL;INTERNET;HOME'] = $vcardKey;
}
break;
case 'CATEGORIES':
#cat_id = 7,8
$vcardData['category'] = array();
if ($attributes['value'])
{
if (!is_object($this->cat))
{
if (!is_object($GLOBALS['egw']->categories))
{
$GLOBALS['egw']->categories =& CreateObject('phpgwapi.categories',$this->owner,'addressbook');
}
$this->cat =& $GLOBALS['egw']->categories;
}
foreach(explode(',',$attributes['value']) as $cat_name)
{
if (!($cat_id = $this->cat->name2id($cat_name)))
{
$cat_id = $this->cat->add( array('name' => $cat_name,'descr' => $cat_name ));
}
$vcardData['category'][] = $cat_id;
}
}
break;
case 'VERSION':
break;
default:
$finalRowNames[$rowName] = $vcardKey;
break;
}
}
#_debug_array($finalRowNames);
$contact = array();
foreach($finalRowNames as $key => $vcardKey)
{
if(isset($this->supportedFields[$key]))
{
$fieldNames = $this->supportedFields[$key];
foreach($fieldNames as $fieldKey => $fieldName)
{
if(!empty($fieldName))
{
switch($fieldName)
{
case 'private':
$contact[$fieldName] = $vcardValues[$vcardKey]['values'][$fieldKey] == 'PRIVATE';
break;
case 'cat_id':
if (!is_object($this->cat))
{
if (!is_object($GLOBALS['egw']->categories))
{
$GLOBALS['egw']->categories =& CreateObject('phpgwapi.categories',$GLOBALS['egw_info']['user']['account_id'],'addressbook');
}
$this->cat =& $GLOBALS['egw']->categories;
}
foreach(explode(',',$vcardValues[$vcardKey]['values'][$fieldKey]) as $cat_name)
{
if (!($cat_id = $this->cat->name2id($cat_name)))
{
$cat_id = $this->cat->add( array('name' => $cat_name,'descr' => $cat_name ));
}
$contact[$fieldName] = $cat_id;
}
break;
default:
$contact[$fieldName] = $vcardValues[$vcardKey]['values'][$fieldKey];
break;
}
}
}
}
}
#return true;
/* _debug_array($contact);exit; */
return $contact;
}
2005-07-20 14:14:39 +02:00
}