* @copyright 1997-2007 The PHP Group * @license http://www.php.net/license/3_0.txt PHP License 3.0 * @version CVS: $Id: Build.php,v 1.10 2007/11/19 01:22:27 till Exp $ * @link http://pear.php.net/package/Contact_Vcard_Build */ // Part numbers for N components define('VCARD_N_FAMILY', 0); define('VCARD_N_GIVEN', 1); define('VCARD_N_ADDL', 2); define('VCARD_N_PREFIX', 3); define('VCARD_N_SUFFIX', 4); // Part numbers for ADR components define('VCARD_ADR_POB', 0); define('VCARD_ADR_EXTEND', 1); define('VCARD_ADR_STREET', 2); define('VCARD_ADR_LOCALITY', 3); define('VCARD_ADR_REGION', 4); define('VCARD_ADR_POSTCODE', 5); define('VCARD_ADR_COUNTRY', 6); // Part numbers for GEO components define('VCARD_GEO_LAT', 0); define('VCARD_GEO_LON', 1); /** * @ignore */ require_once 'PEAR.php'; /** * This class builds a single vCard (version 3.0 or 2.1). * * General note: we use the terms "set" "add" and "get" as function * prefixes. * * "Set" means there is only one iteration of a component, and it has * only one value repetition, so you set the whole thing at once. * * "Add" means eith multiple iterations of a component are allowed, or * that there is only one iteration allowed but there can be multiple * value repetitions, so you add iterations or repetitions to the current * stack. * * "Get" returns the full vCard line for a single iteration. * * @category File_Formats * @package Contact_Vcard_Build * @author Paul M. Jones * @copyright 1997-2007 The PHP Group * @license http://www.php.net/license/3_0.txt PHP License 3.0 * @version Release: 1.1.2 * @link http://pear.php.net/package/Contact_Vcard_Build * */ class Contact_Vcard_Build extends PEAR { /** * Values for vCard components. * * @access public * @var array */ var $value = array(); /** * Parameters for vCard components. * * @access public * @var array */ var $param = array(); /** * Tracks which component (N, ADR, TEL, etc) value was last set or added. * Used when adding parameters automatically to the proper component. * * @access public * @var string */ var $autoparam = null; /** * Constructor. * * @param string $version The vCard version to build; affects which * parameters are allowed and which components * are returned by fetch(). * * @access public * @return void * @see Contact_Vcard_Build::fetch() */ function __construct($version = '3.0') { parent::__construct(); $this->setErrorHandling(PEAR_ERROR_PRINT); $this->setVersion($version); } /** * Prepares a string so it may be safely used as vCard values. DO * NOT use this with binary encodings. Operates on text in-place; * does not return a value. Recursively descends into arrays. * * Escapes a string so that... * : => \; * ; => \; * , => \, * newline => literal \n * * @param mixed &$text The string or array or strings to escape. * * @access public * @return mixed Void on success, or a PEAR_Error object on failure. */ function escape(&$text) { if (is_object($text)) { $msg = 'The escape() method works only with string literals and arrays.'; return $this->raiseError($msg); } if (is_array($text)) { foreach ($text as $key => $val) { $this->escape($val); $text[$key] = $val; } return; } /* // escape colons not led by a backslash $regex = '(?addParam('TYPE', 'HOME', 'TEL', 0); * $vcard->addParam('TYPE', 'PREF', 'TEL', 0); * * @param string $param_name The parameter name, such as TYPE, VALUE, * or ENCODING. * @param string $param_value The parameter value. * @param string $comp The vCard component for which this is a * paramter (ADR, TEL, etc). If null, will be * the component that was last set or added-to. * @param mixed $iter An integer vCard component iteration that this * is a param for. E.g., if you have more than * one ADR component, 0 refers to the first ADR, * 1 to the second ADR, and so on. If null, the * parameter will be added to the last component * iteration available. * * @access public * @return mixed Void on success, or a PEAR_Error object on failure. */ function addParam($param_name, $param_value, $comp = null, $iter = null) { // if component is not specified, default to the last component // that was set or added. if ($comp === null) { $comp = $this->autoparam; } // using a null iteration means the param should be associated // with the latest/most-recent iteration of the selected // component. if ($iter === null) { $iter = $this->countIter($comp) - 1; } // massage the text arguments $comp = strtoupper(trim($comp)); $param_name = strtoupper(trim($param_name)); $param_value = trim($param_value); if (! is_integer($iter) || $iter < 0) { $msg = "$iter is not a valid iteration number for $comp; "; $msg .= "must be a positive integer."; return $this->raiseError($msg); } $result = $this->validateParam($param_name, $param_value, $comp, $iter); if (PEAR::isError($result)) { return $result; } $this->param[$comp][$iter][$param_name][] = $param_value; } /** * Validates parameter names and values based on the vCard version * (2.1 or 3.0). * * @param string $name The parameter name (e.g., TYPE or ENCODING). * @param string $text The parameter value (e.g., HOME or BASE64). * @param string $comp Optional, the component name (e.g., ADR or * PHOTO). Only used for error messaging. * @param string $iter Optional, the iteration of the component. * Only used for error messaging. * * @access public * @return mixed Boolean true if the parameter is valid, or a * PEAR_Error object if not. * */ function validateParam($name, $text, $comp = null, $iter = null) { $name = strtoupper($name); $text = strtoupper($text); switch ($comp) { case 'TEL': case 'TEL;PREF;CELL': case 'TEL;PREF;VOICE': case 'TEL;PREF;FAX': case 'EMAIL': case 'ADR': break; default: // all param values must have only the characters A-Z 0-9 and -. if (preg_match('/[^a-zA-Z0-9\-]/i', $text)) { $msg = "vCard [$comp] [$iter] [$name]: "; $msg .= 'The parameter value may contain only '; $msg .= 'a-z, A-Z, 0-9, and dashes (-).'; return $this->raiseError($msg); } break; } if ($this->value['VERSION'][0][0][0] == '2.1') { // Validate against version 2.1 (pretty strict) $types = array ( 'DOM', 'INTL', 'POSTAL', 'PARCEL','HOME', 'WORK', 'PREF', 'VOICE', 'FAX', 'MSG', 'CELL', 'PAGER', 'BBS', 'MODEM', 'CAR', 'ISDN', 'VIDEO', 'AOL', 'APPLELINK', 'ATTMAIL', 'CIS', 'EWORLD', 'INTERNET', 'IBMMAIL', 'MCIMAIL', 'POWERSHARE', 'PRODIGY', 'TLX', 'X400', 'GIF', 'CGM', 'WMF', 'BMP', 'MET', 'PMB', 'DIB', 'PICT', 'TIFF', 'PDF', 'PS', 'JPEG', 'QTIME', 'MPEG', 'MPEG2', 'AVI', 'WAVE', 'AIFF', 'PCM', 'X509', 'PGP' ); switch ($name) { case 'TYPE': if (! in_array($text, $types)) { $msg = "vCard 2.1 [$comp] [$iter]: "; $msg .= "$text is not a recognized TYPE."; $result = $this->raiseError($msg); } else { $result = true; } break; case 'ENCODING': if ($text != '7BIT' && $text != '8BIT' && $text != 'BASE64' && $text != 'QUOTED-PRINTABLE') { $msg = "vCard 2.1 [$comp] [$iter]: "; $msg .= "$text is not a recognized ENCODING."; $result = $this->raiseError($msg); } else { $result = true; } break; case 'CHARSET': // all charsets are OK $result = true; break; case 'LANGUAGE': // all languages are OK $result = true; break; case 'VALUE': if ($text != 'INLINE' && $text != 'CONTENT-ID' && $text != 'CID' && $text != 'URL' && $text != 'VCARD') { $msg = "vCard 2.1 [$comp] [$iter]: "; $msg .= "$text is not a recognized VALUE."; $result = $this->raiseError($msg); } else { $result = true; } break; default: $msg = "vCard 2.1 [$comp] [$iter]: "; $msg .= "$name is an unknown or invalid parameter name."; return $this->raiseError($msg); break; } } elseif ($this->value['VERSION'][0][0][0] == '3.0') { // Validate against version 3.0 (pretty lenient) switch ($name) { case 'TYPE': // all types are OK $result = true; break; case 'LANGUAGE': // all languages are OK $result = true; break; case 'ENCODING': if ($text != '8BIT' && $text != 'B') { $msg = "vCard 3.0 [$comp] [$iter]: "; $msg .= "The only allowed ENCODING"; $msg .= " parameters are 8BIT and B."; $result = $this->raiseError($msg); } else { $result = true; } break; case 'VALUE': if ($text != 'BINARY' && $text != 'PHONE-NUMBER' && $text != 'TEXT' && $text != 'URI' && $text != 'UTC-OFFSET' && $text != 'VCARD') { $msg = "vCard 3.0 [$comp] [$iter]: The only allowed VALUE "; $msg .= "parameters are "; $msg .= "BINARY, PHONE-NUMBER, TEXT, URI, UTC-OFFSET,"; $msg .= " and VCARD."; return $this->raiseError($msg); } $result = true; break; default: $msg = "vCard 3.0 [$comp] [$iter]: "; $msg .= "Unknown or invalid parameter name ($name)."; return $this->raiseError($msg); break; } } else { $msg = "[$comp] [$iter] Unknown vCard version number or other error."; return $this->raiseError($msg); } return $result; } /** * Gets back the parameter string for a given component. * * @param string $comp The component to get parameters for * (ADR, TEL, etc). * * @param int $iter The vCard component iteration to get the param * list for. E.g., if you have more than one ADR component, * 0 refers to the first ADR, 1 to the second ADR, and so on. * * @access public * @return string */ function getParam($comp, $iter = 0) { $comp = strtoupper($comp); $text = ''; if (isset($this->param[$comp][$iter]) && is_array($this->param[$comp][$iter])) { // loop through the array of parameters for // the component foreach ($this->param[$comp][$iter] as $param_name => $param_val) { // if there were previous parameter names, separate with // a semicolon if ($text != '') { $text .= ';'; } if ($param_val === null) { // no parameter value was specified, which is typical // for vCard version 2.1 -- the name is the value. $this->escape($param_name); $text .= $param_name; } else { // set the parameter name... $text .= strtoupper($param_name) . '='; // ...then escape and comma-separate the parameter // values. $this->escape($param_val); $text .= implode(',', $param_val); } } } // if there were no parameters, this will be blank. return $text; } /** * Resets the vCard values and params to be blank. * * @param string $version The vCard version to reset to ('2.1' or * '3.0' -- default is the same version as * previously set). * * @access public * @return void */ function reset($version = null) { $prev = $this->value['VERSION'][0][0][0]; $this->value = array(); $this->param = array(); $this->autoparam = null; if ($version === null) { $this->setVersion($prev); } else { $this->setVersion($version); } } /** * Generic, all-purpose method to store a string or array in * $this->value, in a way suitable for later output as a vCard * element. This forces the value to be the passed text or array * value, overriding any prior values. * * @param string $comp The component to set the value for ('N', * 'ADR', etc). * @param int $iter The component-iteration to set the value for. * @param int $part The part number of the component-iteration to set * the value for. * @param mixed $text A string or array; the set of repeated values * for this component-iteration part. * * @access public * @return void */ function setValue($comp, $iter, $part, $text) { $comp = strtoupper($comp); settype($text, 'array'); $this->value[$comp][$iter][$part] = $text; //mark this component as having been set. $this->autoparam = $comp; } /** * Generic, all-purpose method to add a repetition of a string or * array in $this->value, in a way suitable for later output as a * vCard element. This appends the value to be the passed text or * array value, leaving any prior values in place. * * @param string $comp The component to set the value for ('N', 'ADR', etc). * @param int $iter The component-iteration to set the value for. * @param int $part The part number of the component-iteration to set * the value for. * @param mixed $text A string or array; the set of repeated values * for this component-iteration part. * * @access public * @return void */ function addValue($comp, $iter, $part, $text) { $comp = strtoupper($comp); settype($text, 'array'); foreach ($text as $val) { $this->value[$comp][$iter][$part][] = $val; } $this->autoparam = $comp; } /** * Generic, all-purpose method to get back the data stored in $this->value. * * @param string $comp The component to set the value for ('N', * 'ADR', etc). * @param int $iter The component-iteration to set the value for. * @param int $part The part number of the component-iteration to get * the value for. * @param mixed $rept The repetition number within the part to get; * if null, get all repetitions of the part within the * iteration. * * @access public * @return string The value, escaped and delimited, of all * repetitions in the component-iteration part (or specific * repetition within the part). */ function getValue($comp, $iter = 0, $part = 0, $rept = null) { if ($rept === null && is_array($this->value[$comp][$iter][$part]) ) { // get all repetitions of a part $list = array(); foreach ($this->value[$comp][$iter][$part] as $key => $val) { $list[] = trim($val); } $this->escape($list); return implode(',', $list); } // get a specific repetition of a part $text = trim($this->value[$comp][$iter][$part][$rept]); $this->escape($text); return $text; } /** * Sets the full N component of the vCard. Will replace all other * values. There can only be one N component per vCard. * * @param mixed $family Single (string) or multiple (array) * family/last name. * @param mixed $given Single (string) or multiple (array) * given/first name. * @param mixed $addl Single (string) or multiple (array) * additional/middle name. * @param mixed $prefix Single (string) or multiple (array) honorific * prefix such as Mr., Miss, etc. * @param mixed $suffix Single (string) or multiple (array) honorific * suffix such as III, Jr., Ph.D., etc. * * @access public * @return void */ function setName($family, $given, $addl, $prefix, $suffix) { $this->autoparam = 'N'; $this->setValue('N', 0, VCARD_N_FAMILY, $family); $this->setValue('N', 0, VCARD_N_GIVEN, $given); $this->setValue('N', 0, VCARD_N_ADDL, $addl); $this->setValue('N', 0, VCARD_N_PREFIX, $prefix); $this->setValue('N', 0, VCARD_N_SUFFIX, $suffix); } /** * Gets back the full N component (first iteration only, since there * can only be one N component per vCard). * * @access public * @return string The first N component-interation of the vCard. */ function getName() { return $this->getMeta('N', 0) . $this->getValue('N', 0, VCARD_N_FAMILY) . ';' . $this->getValue('N', 0, VCARD_N_GIVEN) . ';' . $this->getValue('N', 0, VCARD_N_ADDL) . ';' . $this->getValue('N', 0, VCARD_N_PREFIX) . ';' . $this->getValue('N', 0, VCARD_N_SUFFIX); } /** * Sets the FN component of the card. If no text is passed as the * FN value, constructs an FN automatically from N components. There * is only one FN iteration per vCard. * * @param string $text Override the automatic generation of FN from N * elements with the specified text. * * @access public * @return mixed Void on success, or a PEAR_Error object on failure. */ function setFormattedName($text = null) { $this->autoparam = 'FN'; if ($text !== null) { $this->setValue('FN', 0, 0, $text); return; } // no text was specified for the FN, so build it // from the current N components if an N exists if (is_array($this->value['N'])) { // build from N. // first (given) name, first iteration, first repetition $text .= $this->getValue('N', 0, VCARD_N_GIVEN, 0); // add a space after, if there was text if ($text != '') { $text .= ' '; } // last (family) name, first iteration, first repetition $text .= $this->getValue('N', 0, VCARD_N_FAMILY, 0); // add a space after, if there was text if ($text != '') { $text .= ' '; } // last-name suffix, first iteration, first repetition $text .= $this->getValue('N', 0, VCARD_N_SUFFIX, 0); $this->setValue('FN', 0, 0, $text); return; } // no N exists, and no FN was set, so return. $msg = 'FN not specified and N not set; cannot set FN.'; return $this->raiseError($msg); } /** * Gets back the full FN component value. Only ever returns iteration * zero, because only one FN component is allowed per vCard. * * @access public * @uses self::getMeta() * @return string The FN value of the vCard. * @uses self::getMeta() */ function getFormattedName() { return $this->getMeta('FN', 0) . $this->getValue('FN', 0, 0); } /** * Sets the version of the the vCard. Only one iteration. * * @param string $text The text value of the verson text ('3.0' or '2.1'). * * @access public * @return mixed Void on success, or a PEAR_Error object on failure. */ function setVersion($text = '3.0') { $this->autoparam = 'VERSION'; if ($text != '3.0' && $text != '2.1') { return $this->raiseError('Version must be 3.0 or 2.1 to be valid.'); } $this->setValue('VERSION', 0, 0, $text); } /** * Gets back the version of the the vCard. Only one iteration. * * @access public * @uses self::getMeta() * @return string The data-source of the vCard. */ function getVersion() { return $this->getMeta('VERSION', 0) . $this->getValue('VERSION', 0); } /** * Sets the data-source of the the vCard. Only one iteration. * * @param string $text The text value of the data-source text. * * @access public * @return void */ function setSource($text) { $this->autoparam = 'SOURCE'; $this->setValue('SOURCE', 0, 0, $text); } /** * Gets back the data-source of the the vCard. Only one iteration. * * @access public * @return string The data-source of the vCard. */ function getSource() { return $this->getMeta('SOURCE', 0) . $this->getValue('SOURCE', 0, 0); } /** * Sets the displayed name of the vCard data-source. Only one iteration. * If no name is specified, copies the value of SOURCE. * * @param string $text The text value of the displayed data-source * name. If null, copies the value of SOURCE. * * @access public * @return mixed Void on success, or a PEAR_Error object on failure. */ function setSourceName($text = null) { $this->autoparam = 'NAME'; if ($text === null) { if (is_array($this->value['SOURCE'])) { $text = $this->getValue('SOURCE', 0, 0); } else { $msg = 'NAME not specified and SOURCE not set; cannot set NAME.'; return $this->raiseError($msg); } } $this->setValue('NAME', 0, 0, $text); } /** * Gets back the displayed data-source name of the the vCard. Only * one iteration. * * @access public * @return string The data-source name of the vCard. */ function getSourceName() { return $this->getMeta('NAME', 0) . $this->getValue('NAME', 0, 0); } /** * Sets the value of the PHOTO component. There is only one allowed * per vCard. * * @param string $text The value to set for this component. * @param boolean $url True or false, depending if $text is one. * * @access public * @return void */ function setPhoto($text, $url = false) { if ($url === false) { $this->autoparam = 'PHOTO'; $this->setValue('PHOTO', 0, 0, $text); return; } $this->autoparam = 'PHOTO;VALUE=URI'; $this->setValue('PHOTO;VALUE=URI', 0, 0, $text); } /** * Gets back the value of the PHOTO component. There is only one * allowed per vCard. * * @access public * @return string The value of this component or false if no photo is set. */ function getPhoto() { if (isset($this->value['PHOTO'])) { return $this->getMeta('PHOTO') . $this->getValue('PHOTO', 0, 0); } if (isset($this->value['PHOTO;VALUE=URI'])) { return $this->getMeta('PHOTO;VALUE=URI') . $this->getValue('PHOTO;VALUE=URI', 0, 0); } return false; } /** * Sets the value of the LOGO component. There is only one allowed * per vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function setLogo($text) { $this->autoparam = 'LOGO'; $this->setValue('LOGO', 0, 0, $text); } /** * Gets back the value of the LOGO component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getLogo() { return $this->getMeta('LOGO') . $this->getValue('LOGO', 0, 0); } /** * Sets the value of the SOUND component. There is only one allowed * per vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function setSound($text) { $this->autoparam = 'SOUND'; $this->setValue('SOUND', 0, 0, $text); } /** * Gets back the value of the SOUND component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getSound() { return $this->getMeta('SOUND') . $this->getValue('SOUND', 0, 0); } /** * Sets the value of the KEY component. There is only one allowed * per vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function setKey($text) { $this->autoparam = 'KEY'; $this->setValue('KEY', 0, 0, $text); } /** * Gets back the value of the KEY component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getKey() { return $this->getMeta('KEY') . $this->getValue('KEY', 0, 0); } /** * Sets the value of the BDAY component. There is only one allowed * per vCard. Date format is "yyyy-mm-dd[Thh:ii[:ss[Z|-06:00]]]". * * @param string $text The value to set for this component. * * @access public * @return void */ function setBirthday($text) { $this->autoparam = 'BDAY'; $this->setValue('BDAY', 0, 0, $text); } /** * Gets back the value of the BDAY component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getBirthday() { return $this->getMeta('BDAY') . $this->getValue('BDAY', 0, 0); } /** * Sets the value of the TZ component. There is only one allowed per * vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function setTZ($text) { $this->autoparam = 'TZ'; $this->setValue('TZ', 0, 0, $text); } /** * Gets back the value of the TZ component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getTZ() { return $this->getMeta('TZ') . $this->getValue('TZ', 0, 0); } /** * Sets the value of the MAILER component. There is only one allowed * per vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function setMailer($text) { $this->autoparam = 'MAILER'; $this->setValue('MAILER', 0, 0, $text); } /** * Gets back the value of the MAILER component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getMailer() { return $this->getMeta('MAILER') . $this->getValue('MAILER', 0, 0); } /** * Sets the value of the NOTE component. There is only one allowed * per vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function setNote($text) { $this->autoparam = 'NOTE'; $this->setValue('NOTE', 0, 0, $text); } /** * Gets back the value of the NOTE component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getNote() { return $this->getMeta('NOTE') . $this->getValue('NOTE', 0, 0); } /** * Sets the value of the TITLE component. There is only one allowed * per vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function setTitle($text) { $this->autoparam = 'TITLE'; $this->setValue('TITLE', 0, 0, $text); } /** * Gets back the value of the TITLE component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getTitle() { return $this->getMeta('TITLE') . $this->getValue('TITLE', 0, 0); } /** * Sets the value of the ROLE component. There is only one allowed * per vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function setRole($text) { $this->autoparam = 'ROLE'; $this->setValue('ROLE', 0, 0, $text); } /** * Gets back the value of the ROLE component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getRole() { return $this->getMeta('ROLE') . $this->getValue('ROLE', 0, 0); } /** * Sets the value of the URL component. There is only one allowed * per vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function setURL($text) { $this->autoparam = 'URL'; $this->setValue('URL', 0, 0, $text); } /** * Gets back the value of the URL component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getURL() { return $this->getMeta('URL') . $this->getValue('URL', 0, 0); } /** * Sets the value of the CLASS component. There is only one allowed * per vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function setClass($text) { $this->autoparam = 'CLASS'; $this->setValue('CLASS', 0, 0, $text); } /** * Gets back the value of the CLASS component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getClass() { return $this->getMeta('CLASS') . $this->getValue('CLASS', 0, 0); } /** * Sets the value of the SORT-STRING component. There is only one * allowed per vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function setSortString($text) { $this->autoparam = 'SORT-STRING'; $this->setValue('SORT-STRING', 0, 0, $text); } /** * Gets back the value of the SORT-STRING component. There is only * one allowed per vCard. * * @access public * @return string The value of this component. */ function getSortString() { return $this->getMeta('SORT-STRING') . $this->getValue('SORT-STRING', 0, 0); } /** * Sets the value of the PRODID component. There is only one allowed * per vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function setProductID($text) { $this->autoparam = 'PRODID'; $this->setValue('PRODID', 0, 0, $text); } /** * Gets back the value of the PRODID component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getProductID() { return $this->getMeta('PRODID') . $this->getValue('PRODID', 0, 0); } /** * Sets the value of the REV component. There is only one allowed * per vCard. This value is free and should reflect whatever you use * for control - e.g. a DATETIME value. * * @param string $text The value to set for this component. * * @access public * @return void */ function setRevision($text) { $this->autoparam = 'REV'; $this->setValue('REV', 0, 0, $text); } /** * Gets back the value of the REV component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getRevision() { return $this->getMeta('REV') . $this->getValue('REV', 0, 0); } /** * Sets the value of the UID component. There is only one allowed * per vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function setUniqueID($text) { $this->autoparam = 'UID'; $this->setValue('UID', 0, 0, $text); } /** * Gets back the value of the UID component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getUniqueID() { return $this->getMeta('UID') . $this->getValue('UID', 0, 0); } /** * Sets the value of the AGENT component. There is only one allowed * per vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function setAgent($text) { $this->autoparam = 'AGENT'; $this->setValue('AGENT', 0, 0, $text); } /** * Gets back the value of the AGENT component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getAgent() { return $this->getMeta('AGENT') . $this->getValue('AGENT', 0, 0); } /** * Sets the value of both parts of the GEO component. There is only * one GEO component allowed per vCard. * * @param string $lat The value to set for the longitude part * (decimal, + or -). * @param string $lon The value to set for the latitude part * (decimal, + or -). * * @access public * @return void */ function setGeo($lat, $lon) { $this->autoparam = 'GEO'; $this->setValue('GEO', 0, VCARD_GEO_LAT, $lat); $this->setValue('GEO', 0, VCARD_GEO_LON, $lon); } /** * Gets back the value of the GEO component. There is only one * allowed per vCard. * * @access public * @return string The value of this component. */ function getGeo() { return $this->getMeta('GEO', 0) . $this->getValue('GEO', 0, VCARD_GEO_LAT, 0) . ';' . $this->getValue('GEO', 0, VCARD_GEO_LON, 0); } /** * Sets the value of one entire ADR iteration. There can be zero, * one, or more ADR components in a vCard. * * @param mixed $pob String (one repetition) or array (multiple * repetitions) of the p.o. box part of the ADR * component iteration. * @param mixed $extend String (one repetition) or array (multiple * repetitions) of the "extended address" part * of the ADR component iteration. * @param mixed $street String (one repetition) or array (multiple * repetitions) of the street address part of * the ADR component iteration. * @param mixed $locality String (one repetition) or array (multiple * repetitions) of the locailty (e.g., city) part of the * ADR component iteration. * @param mixed $region String (one repetition) or array (multiple * repetitions) of the region (e.g., state, province, or * governorate) part of the ADR component iteration. * @param mixed $postcode String (one repetition) or array (multiple * repetitions) of the postal code (e.g., ZIP code) part * of the ADR component iteration. * @param mixed $country String (one repetition) or array (multiple * repetitions) of the country-name part of the ADR * component iteration. * * @access public * @return void */ function addAddress($pob, $extend, $street, $locality, $region, $postcode, $country) { $this->autoparam = 'ADR'; $iter = $this->countIter('ADR'); $this->setValue('ADR', $iter, VCARD_ADR_POB, $pob); $this->setValue('ADR', $iter, VCARD_ADR_EXTEND, $extend); $this->setValue('ADR', $iter, VCARD_ADR_STREET, $street); $this->setValue('ADR', $iter, VCARD_ADR_LOCALITY, $locality); $this->setValue('ADR', $iter, VCARD_ADR_REGION, $region); $this->setValue('ADR', $iter, VCARD_ADR_POSTCODE, $postcode); $this->setValue('ADR', $iter, VCARD_ADR_COUNTRY, $country); } /** * Gets back the value of one ADR component iteration. * * @param int $iter The component iteration-number to get the value * for. * * @access public * @return mixed The value of this component iteration, or a * PEAR_Error if the iteration is not valid. */ function getAddress($iter) { if (! is_integer($iter) || $iter < 0) { return $this->raiseError('ADR iteration number not valid.'); } return $this->getMeta('ADR', $iter) . $this->getValue('ADR', $iter, VCARD_ADR_POB) . ';' . $this->getValue('ADR', $iter, VCARD_ADR_EXTEND) . ';' . $this->getValue('ADR', $iter, VCARD_ADR_STREET) . ';' . $this->getValue('ADR', $iter, VCARD_ADR_LOCALITY) . ';' . $this->getValue('ADR', $iter, VCARD_ADR_REGION) . ';' . $this->getValue('ADR', $iter, VCARD_ADR_POSTCODE) . ';' . $this->getValue('ADR', $iter, VCARD_ADR_COUNTRY); } /** * Sets the value of one LABEL component iteration. There can be * zero, one, or more component iterations in a vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function addLabel($text) { $this->autoparam = 'LABEL'; $iter = $this->countIter('LABEL'); $this->setValue('LABEL', $iter, 0, $text); } /** * Gets back the value of one iteration of the LABEL component. * There can be zero, one, or more component iterations in a vCard. * * @param int $iter The component iteration-number to get the value * for. * * @access public * @return mixed The value of this component, or a PEAR_Error if * the iteration number is not valid. */ function getLabel($iter) { if (! is_integer($iter) || $iter < 0) { return $this->raiseError('LABEL iteration number not valid.'); } return $this->getMeta('LABEL', $iter) . $this->getValue('LABEL', $iter, 0); } /** * Sets the value of one component iteration. There can be zero, * one, or more component iterations in a vCard. This function also * takes a type argument to be able to add mobiles, fax and voice * as well. * * @param string $text The value to set for this component. * @param string $type The type: phone, mobile, fax or voice * * @access public * @return mixed void on success, PEAR_Error on failure * @uses self::_getTelephoneType() */ function addTelephone($text, $type = 'phone') { $autoparam = $this->_getTelephoneType($type); if (PEAR::isError($autoparam)) { return $autoparam; } $iter = $this->countIter($autoparam); $this->setValue($autoparam, $iter, 0, $text); $this->autoparam = $autoparam; } /** * Gets back the value of one iteration of the TEL component. There * can be zero, one, or more component iterations in a vCard. * * @param int $iter The component iteration-number to get the value * for. * @param string $type The type: phone, mobile, fax, voice * * @access public * @return mixed The value of this component, or a PEAR_Error if the * iteration number is not valid. * @uses self::_getTelephoneType() */ function getTelephone($iter, $type = 'phone') { $autoparam = $this->_getTelephoneType($type); if (PEAR::isError($autoparam)) { return $autoparam; } if (!is_integer($iter) || $iter < 0) { return $this->raiseError($autoparam . ' iteration number not valid.'); } return $this->getMeta($autoparam, $iter) . $this->getValue($autoparam, $iter, 0); } /** * Sets the value of one EMAIL component iteration. There can be zero, * one, or more component iterations in a vCard. * * @param string $text The value to set for this component. * * @access public * @return void */ function addEmail($text) { $this->autoparam = 'EMAIL'; $iter = $this->countIter('EMAIL'); $this->setValue('EMAIL', $iter, 0, $text); } /** * Gets back the value of one iteration of the EMAIL component. There can * be zero, one, or more component iterations in a vCard. * * @param int $iter The component iteration-number to get the value * for. * * @access public * * @return mixed The value of this component, or a PEAR_Error if the * iteration number is not valid. * */ function getEmail($iter) { if (! is_integer($iter) || $iter < 0) { return $this->raiseError('EMAIL iteration number not valid.'); } return $this->getMeta('EMAIL', $iter) . $this->getValue('EMAIL', $iter, 0); } /** * Sets the full value of the NICKNAME component. There is only one * component iteration allowed per vCard, but there may be multiple * value repetitions in the iteration. * * @param mixed $text String (one repetition) or array (multiple * reptitions) of the component iteration value. * * @access public * @return void */ function addNickname($text) { $this->autoparam = 'NICKNAME'; $this->addValue('NICKNAME', 0, 0, $text); } /** * Gets back the value of the NICKNAME component. There is only one * component allowed per vCard, but there may be multiple value * repetitions in the iteration. * * @access public * @return string The value of this component. */ function getNickname() { return $this->getMeta('NICKNAME') . $this->getValue('NICKNAME', 0, 0); } /** * Sets the full value of the CATEGORIES component. There is only * one component iteration allowed per vCard, but there may be * multiple value repetitions in the iteration. * * @param mixed $text String (one repetition) or array (multiple reptitions) * of the component iteration value. * * @access public * @return void */ function addCategories($text) { $this->autoparam = 'CATEGORIES'; $this->addValue('CATEGORIES', 0, 0, $text); } /** * Gets back the value of the CATEGORIES component. There is only * one component allowed per vCard, but there may be multiple value * repetitions in the iteration. * * @access public * @return string The value of this component. */ function getCategories() { return $this->getMeta('CATEGORIES', 0) . $this->getValue('CATEGORIES', 0, 0); } /** * Sets the full value of the ORG component. There can be only one * ORG component in a vCard. * * The ORG component can have one or more parts (as opposed to * repetitions of values within those parts). The first part is the * highest-level organization, the second part is the next-highest, * the third part is the third-highest, and so on. There can by any * number of parts in one ORG iteration. (This is different from * other components, such as NICKNAME, where an iteration has only * one part but may have many repetitions within that part.) * * @param mixed $text String (one ORG part) or array (of ORG parts) to * use as the value for the component iteration. * * @access public * @return void */ function addOrganization($text) { $this->autoparam = 'ORG'; settype($text, 'array'); $base = $this->countRept('ORG', 0); // start at the original base point, and add // new parts foreach ($text as $part => $val) { $this->setValue('ORG', 0, $base + $part, $val); } } /** * Gets back the value of the ORG component. * * @return string The value of this component. * @access public */ function getOrganization() { $text = $this->getMeta('ORG', 0); $k = $this->countRept('ORG', 0); $last = $k - 1; for ($part = 0; $part < $k; $part++) { $text .= $this->getValue('ORG', 0, $part); if ($part != $last) { $text .= ';'; } } return $text; } /** * Builds a vCard from a Contact_Vcard_Parse result array. Only send * one vCard from the parse-results. * * Usage (to build from first vCard in parsed results): * * $parse = new Contact_Vcard_Parse(); // new parser * $info = $parse->fromFile('sample.vcf'); // parse file * * $vcard = new Contact_Vcard_Build(); // new builder * $vcard->setFromArray($info[0]); // [0] is the first card * * @param array $src One vCard entry as parsed using * Contact_Vcard_Parse. * * @access public * @return void * @see Contact_Vcard_Parse::fromFile() * @see Contact_Vcard_Parse::fromText() */ function setFromArray($src) { // reset to a blank values and params $this->value = array(); $this->param = array(); // loop through components (N, ADR, TEL, etc) foreach ($src as $comp => $comp_val) { // set the autoparam property. not really needed, but let's // behave after an expected fashion, shall we? ;-) $this->autoparam = $comp; // iteration number of each component foreach ($comp_val as $iter => $iter_val) { // value or param? foreach ($iter_val as $kind => $kind_val) { // part number foreach ($kind_val as $part => $part_val) { // repetition number and text value foreach ($part_val as $rept => $text) { // ignore data when $kind is neither 'value' // nor 'param' if (strtolower($kind) == 'value') { $this->_setValue($comp, $iter, $part, $rept, $text); } elseif (strtolower($kind) == 'param') { $this->_setParam($comp, $iter, $part, $rept, $text); } } } } } } } /** * Fetches a full vCard text block based on $this->value and * $this->param. The order of the returned components is similar to * their order in RFC 2426. Honors the value of * $this->value['VERSION'] to determine which vCard components are * returned (2.1- or 3.0-compliant). * * @access public * @return string A properly formatted vCard text block. */ function fetch() { // vCard version is required if (! is_array($this->value['VERSION'])) { return $this->raiseError('VERSION not set (required).'); } // FN component is required if (! is_array($this->value['FN'])) { return $this->raiseError('FN component not set (required).'); } // N component is required if (! is_array($this->value['N'])) { return $this->raiseError('N component not set (required).'); } // initialize the vCard lines $lines = array(); // begin (required) $lines[] = "BEGIN:VCARD"; // version (required) // available in both 2.1 and 3.0 $lines[] = $this->getVersion(); // profile (3.0 only) if ($this->value['VERSION'][0][0][0] == '3.0') { $lines[] = "PROFILE:VCARD"; } // formatted name (required) // available in both 2.1 and 3.0 $lines[] = $this->getFormattedName(); // structured name (required) // available in both 2.1 and 3.0 $lines[] = $this->getName(); // displayed name of the data source (3.0 only) if (isset($this->value['NAME']) && $this->value['VERSION'][0][0][0] == '3.0') { $lines[] = $this->getSourceName(); } // data source (3.0 only) if (isset($this->value['SOURCE']) && $this->value['VERSION'][0][0][0] == '3.0') { $lines[] = $this->getSource(); } // nicknames (3.0 only) if (isset($this->value['NICKNAME']) && $this->value['VERSION'][0][0][0] == '3.0') { $lines[] = $this->getNickname(); } // personal photo // available in both 2.1 and 3.0 $_photo = $this->getPhoto(); if ($_photo !== false) { $lines[] = $_photo; } // bday // available in both 2.1 and 3.0 if (isset($this->value['BDAY'])) { $lines[] = $this->getBirthday(); } // adr // available in both 2.1 and 3.0 if (isset($this->value['ADR'])) { foreach ($this->value['ADR'] as $key => $val) { $lines[] = $this->getAddress($key); } } // label // available in both 2.1 and 3.0 if (isset($this->value['LABEL'])) { foreach ($this->value['LABEL'] as $key => $val) { $lines[] = $this->getLabel($key); } } // tel // available in both 2.1 and 3.0 if (isset($this->value['TEL'])) { foreach ($this->value['TEL'] as $key => $val) { $lines[] = $this->getTelephone($key); } } // mobiles if (isset($this->value['TEL;PREF;CELL'])) { foreach ($this->value['TEL;PREF;CELL'] as $key => $val) { $lines[] = $this->getTelephone($key, 'mobile'); } } // voice if (isset($this->value['TEL;PREF;VOICE'])) { foreach ($this->value['TEL;PREF;VOICE'] as $key => $val) { $lines[] = $this->getTelephone($key, 'voice'); } } // fax if (isset($this->value['TEL;PREF;FAX'])) { foreach ($this->value['TEL;PREF;FAX'] as $key => $val) { $lines[] = $this->getTelephone($key, 'fax'); } } // email // available in both 2.1 and 3.0 if (isset($this->value['EMAIL'])) { foreach ($this->value['EMAIL'] as $key => $val) { $lines[] = $this->getEmail($key); } } // mailer // available in both 2.1 and 3.0 if (isset($this->value['MAILER'])) { $lines[] = $this->getMailer(); } // tz // available in both 2.1 and 3.0 if (isset($this->value['TZ'])) { $lines[] = $this->getTZ(); } // geo // available in both 2.1 and 3.0 if (isset($this->value['GEO'])) { $lines[] = $this->getGeo(); } // title // available in both 2.1 and 3.0 if (isset($this->value['TITLE'])) { $lines[] = $this->getTitle(); } // role // available in both 2.1 and 3.0 if (isset($this->value['ROLE'])) { $lines[] = $this->getRole(); } // company logo // available in both 2.1 and 3.0 if (isset($this->value['LOGO'])) { $lines[] = $this->getLogo(); } // agent // available in both 2.1 and 3.0 if (isset($this->value['AGENT'])) { $lines[] = $this->getAgent(); } // org // available in both 2.1 and 3.0 if (isset($this->value['ORG'])) { $lines[] = $this->getOrganization(); } // categories (3.0 only) if (isset($this->value['CATEGORIES']) && $this->value['VERSION'][0][0][0] == '3.0') { $lines[] = $this->getCategories(); } // note // available in both 2.1 and 3.0 if (isset($this->value['NOTE'])) { $lines[] = $this->getNote(); } // prodid (3.0 only) if (isset($this->value['PRODID']) && $this->value['VERSION'][0][0][0] == '3.0') { $lines[] = $this->getProductID(); } // rev // available in both 2.1 and 3.0 if (isset($this->value['REV'])) { $lines[] = $this->getRevision(); } // sort-string (3.0 only) if (isset($this->value['SORT-STRING']) && $this->value['VERSION'][0][0][0] == '3.0') { $lines[] = $this->getSortString(); } // name-pronounciation sound // available in both 2.1 and 3.0 if (isset($this->value['SOUND'])) { $lines[] = $this->getSound(); } // uid // available in both 2.1 and 3.0 if (isset($this->value['UID'])) { $lines[] = $this->getUniqueID(); } // url // available in both 2.1 and 3.0 if (isset($this->value['URL'])) { $lines[] = $this->getURL(); } // class (3.0 only) if (isset($this->value['CLASS']) && $this->value['VERSION'][0][0][0] == '3.0') { $lines[] = $this->getClass(); } // key // available in both 2.1 and 3.0 if (isset($this->value['KEY'])) { $lines[] = $this->getKey(); } // required $lines[] = "END:VCARD"; // version 3.0 uses \n for new lines, // version 2.1 uses \r\n $newline = "\n"; if ($this->value['VERSION'][0][0][0] == '2.1') { $newline = "\r\n"; } // fold lines at 75 characters // commented out as a possible fix for Outlook compatibility // $regex = "(.{1,75})"; // foreach ($lines as $key => $val) { // if (strlen($val) > 75) { // // we trim to drop the last newline, which will be added // // again by the implode function at the end of fetch() // $lines[$key] = trim(preg_replace("/$regex/i", "\\1$newline ", $val)); // } // } // recode the lines from UTF-8 to lowest covering charset $lines = $this->charsetRecode($lines); // compile the array of lines into a single text block // and return (with a trailing newline) return implode($newline, $lines). $newline; } /** * Send the vCard as a downloadable file. * * @param string $filename The file name to give the vCard. * @param string $disposition How the file should be sent, either * 'inline' or 'attachment'. * @param string $charset The character set to use, defaults to * 'us-ascii'. * * @access public * @return void or Pear_Error in case of an error. */ function send($filename, $disposition = 'attachment', $charset = 'us-ascii') { $vcard = $this->fetch(); if (PEAR::isError($vcard)) { return $vcard; } header('Content-Type: application/directory; ' . 'profile="vcard"; ' . 'charset=' . $charset); header('Cache-Control: must-revalidate,post-check=0, pre-check=0'); header('Pragma: public'); header('Content-Length: ' . strlen($vcard)); header("Content-Disposition: $disposition; filename=\"$filename\""); echo $vcard; } /** * Gets the left-side/prefix/before-the-colon (metadata) part of a * vCard line, including the component identifier, the parameter * list, and a colon. * * @param string $comp The component to get metadata for (ADR, TEL, etc). * @param int $iter The vCard component iteration to get the metadata * for. E.g., if you have more than one ADR component, * 0 refers to the first ADR, 1 to the second ADR, * and so on. * * @access public * @return string The line prefix metadata. */ function getMeta($comp, $iter = 0) { $params = $this->getParam($comp, $iter); if (trim($params) == '') { // no parameters $text = $comp . ':'; } else { // has parameters. put an extra semicolon in. $text = $comp . ';' . $params . ':'; } return $text; } /** * Recode the lines to the 'smallest' encoding and add the charset info * * As, surprise surprise, Microsoft Outlook does not seem to be able to * handle properly tagged UTF-8 vCards, we try to represent the text in * smallest charset 'covering' the text in full (currently the test goes * 'US-ASCII' -> 'ISO-8859-1' -> 'CP1250' and falls back to 'UTF-8') * * @param array $lines the vCard's lines * * @return array the recoded and properly tagged lines */ function charsetRecode($lines) { $charsets = array('US-ASCII', 'ISO-8859-1', 'CP1250'); if ((function_exists('iconv_strlen') or function_exists('mb_strlen')) and function_exists('iconv')) { // for each line, get the character length of the UTF-8 string // and try to recode it to every charset in turn; iconv() returns // truncated strings on recode errors, so if the character length // after the recode equals the one before - we've found a valid // charset foreach ($lines as $number => $line) { $lineCharset = 'UTF-8'; if (function_exists('iconv_strlen')) { $strlen = iconv_strlen($line, 'UTF-8'); } else { $strlen = mb_strlen($line, 'UTF-8'); } foreach ($charsets as $charset) { $iconvd = iconv('UTF-8', $charset, $line); if (strlen($iconvd) == $strlen) { $lineCharset = $charset; $line = $iconvd; break; } } // tag the non-US-ASCII-only, recoded lines properly if ($lineCharset != 'US-ASCII') { $lines[$number] = preg_replace('/:/', ";ENCODING=8BIT;CHARSET=$lineCharset:", $line, 1); } } } else { // if there's no mb_strings or iconv support, // simply tag the non-US-ASCII-only lines as UTF-8 foreach ($lines as $number => $line) { if (preg_match('/[^\x00-\x7f]/', $line)) { $lines[$number] = preg_replace('/:/', ";ENCODING=8BIT;CHARSET=UTF-8:", $line, 1); } } } return $lines; } /** * Count the number of iterations for an element type. * * @param string $type The element type to count iterations for * (ADR, ORG, etc). * * @access public * @return int The number of iterations for that type. */ function countIter($type) { if (!isset($this->value[$type])) { return 0; } return count($this->value[$type]); } /** * Count the number of repetitions for an element type and iteration * number. * * @param string $type The element type to count iterations for * (ADR, ORG, etc). * @param int $rept The iteration number to count repetitions for. * * @access public * @return int The number of repetitions for that type and iteration. */ function countRept($type, $rept) { if (!isset($this->value[$type][$rept])) { return 0; } return count($this->value[$type][$rept]); } /** * Emulated destructor. * * @access private * @return boolean true */ function _Contact_Vcard_Build() { return true; } /** * _setValue * * @param string $comp The component to set the value for ('N', * 'ADR', etc). * @param int $iter The component-iteration to set the value for. * the value for. * @param int $part The part number of the component-iteration to set * @param mixed $rept The repetition number within the part to get; * if null, get all repetitions of the part within the * iteration. * @param mixed $text A string or array; the set of repeated values * for this component-iteration part. * * @access protected * @return void */ function _setValue($comp, $iter, $part, $rept, $text) { $this->value[strtoupper($comp)][$iter][$part][$rept] = $text; } /** * _setParam * * @param string $comp The component to set the param for. * @param int $iter The component-iteration to set the param for. * @param int $part The part number of the component-iteration to set * the param for. * @param mixed $rept The repetition number within the part to get; * if null, get all repetitions of the part within the * iteration. * @param mixed $text A string or array; the set of repeated values * for this component-iteration part. * * @access protected * @return void */ function _setParam($comp, $iter, $part, $rept, $text) { $this->param[strtoupper($comp)][$iter][$part][$rept] = $text; } /** * _getTelephoneType * * Resolves a HR friendly type to the internal * type used in the VCARD spec. * * @param string $type The type?! * * @access private * @return mixed string $type or Pear_Error * @see self::addTelephone() * @ses self::getTelephone() * @since 1.1.2 */ function _getTelephoneType($type) { switch ($type) { case 'phone': return 'TEL'; break; case 'mobile': return 'TEL;PREF;CELL'; break; case 'voice': return 'TEL;PREF;VOICE'; break; case 'fax': return 'TEL;PREF;FAX'; break; default: $msg = 'Type: ' . $type . ' is not yet implemented.'; return $this->raiseError($msg); break; } } } ?>