| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392 |
- <?php
- namespace Sabre\VObject\Property;
- use Sabre\VObject\Component;
- use Sabre\VObject\Document;
- use Sabre\VObject\Parser\MimeDir;
- use Sabre\VObject\Property;
- use Sabre\Xml;
- /**
- * Text property.
- *
- * This object represents TEXT values.
- *
- * @copyright Copyright (C) fruux GmbH (https://fruux.com/)
- * @author Evert Pot (http://evertpot.com/)
- * @license http://sabre.io/license/ Modified BSD License
- */
- class Text extends Property
- {
- /**
- * In case this is a multi-value property. This string will be used as a
- * delimiter.
- *
- * @var string
- */
- public $delimiter = ',';
- /**
- * List of properties that are considered 'structured'.
- *
- * @var array
- */
- protected $structuredValues = [
- // vCard
- 'N',
- 'ADR',
- 'ORG',
- 'GENDER',
- 'CLIENTPIDMAP',
- // iCalendar
- 'REQUEST-STATUS',
- ];
- /**
- * Some text components have a minimum number of components.
- *
- * N must for instance be represented as 5 components, separated by ;, even
- * if the last few components are unused.
- *
- * @var array
- */
- protected $minimumPropertyValues = [
- 'N' => 5,
- 'ADR' => 7,
- ];
- /**
- * Creates the property.
- *
- * You can specify the parameters either in key=>value syntax, in which case
- * parameters will automatically be created, or you can just pass a list of
- * Parameter objects.
- *
- * @param Component $root The root document
- * @param string $name
- * @param string|array|null $value
- * @param array $parameters List of parameters
- * @param string $group The vcard property group
- */
- public function __construct(Component $root, $name, $value = null, array $parameters = [], $group = null)
- {
- // There's two types of multi-valued text properties:
- // 1. multivalue properties.
- // 2. structured value properties
- //
- // The former is always separated by a comma, the latter by semi-colon.
- if (in_array($name, $this->structuredValues)) {
- $this->delimiter = ';';
- }
- parent::__construct($root, $name, $value, $parameters, $group);
- }
- /**
- * Sets a raw value coming from a mimedir (iCalendar/vCard) file.
- *
- * This has been 'unfolded', so only 1 line will be passed. Unescaping is
- * not yet done, but parameters are not included.
- *
- * @param string $val
- */
- public function setRawMimeDirValue($val)
- {
- $this->setValue(MimeDir::unescapeValue($val, $this->delimiter));
- }
- /**
- * Sets the value as a quoted-printable encoded string.
- *
- * @param string $val
- */
- public function setQuotedPrintableValue($val)
- {
- $val = quoted_printable_decode($val);
- // Quoted printable only appears in vCard 2.1, and the only character
- // that may be escaped there is ;. So we are simply splitting on just
- // that.
- //
- // We also don't have to unescape \\, so all we need to look for is a ;
- // that's not preceded with a \.
- $regex = '# (?<!\\\\) ; #x';
- $matches = preg_split($regex, $val);
- $this->setValue($matches);
- }
- /**
- * Returns a raw mime-dir representation of the value.
- *
- * @return string
- */
- public function getRawMimeDirValue()
- {
- $val = $this->getParts();
- if (isset($this->minimumPropertyValues[$this->name])) {
- $val = array_pad($val, $this->minimumPropertyValues[$this->name], '');
- }
- foreach ($val as &$item) {
- if (!is_array($item)) {
- $item = [$item];
- }
- foreach ($item as &$subItem) {
- if (!is_null($subItem)) {
- $subItem = strtr(
- $subItem,
- [
- '\\' => '\\\\',
- ';' => '\;',
- ',' => '\,',
- "\n" => '\n',
- "\r" => '',
- ]
- );
- }
- }
- $item = implode(',', $item);
- }
- return implode($this->delimiter, $val);
- }
- /**
- * Returns the value, in the format it should be encoded for json.
- *
- * This method must always return an array.
- *
- * @return array
- */
- public function getJsonValue()
- {
- // Structured text values should always be returned as a single
- // array-item. Multi-value text should be returned as multiple items in
- // the top-array.
- if (in_array($this->name, $this->structuredValues)) {
- return [$this->getParts()];
- }
- return $this->getParts();
- }
- /**
- * Returns the type of value.
- *
- * This corresponds to the VALUE= parameter. Every property also has a
- * 'default' valueType.
- *
- * @return string
- */
- public function getValueType()
- {
- return 'TEXT';
- }
- /**
- * Turns the object back into a serialized blob.
- *
- * @return string
- */
- public function serialize()
- {
- // We need to kick in a special type of encoding, if it's a 2.1 vcard.
- if (Document::VCARD21 !== $this->root->getDocumentType()) {
- return parent::serialize();
- }
- $val = $this->getParts();
- if (isset($this->minimumPropertyValues[$this->name])) {
- $val = \array_pad($val, $this->minimumPropertyValues[$this->name], '');
- }
- // Imploding multiple parts into a single value, and splitting the
- // values with ;.
- if (\count($val) > 1) {
- foreach ($val as $k => $v) {
- $val[$k] = \str_replace(';', '\;', $v);
- }
- $val = \implode(';', $val);
- } else {
- $val = $val[0];
- }
- $str = $this->name;
- if ($this->group) {
- $str = $this->group.'.'.$this->name;
- }
- foreach ($this->parameters as $param) {
- if ('QUOTED-PRINTABLE' === $param->getValue()) {
- continue;
- }
- $str .= ';'.$param->serialize();
- }
- // If the resulting value contains a \n, we must encode it as
- // quoted-printable.
- if (false !== \strpos($val, "\n")) {
- $str .= ';ENCODING=QUOTED-PRINTABLE:';
- $lastLine = $str;
- $out = null;
- // The PHP built-in quoted-printable-encode does not correctly
- // encode newlines for us. Specifically, the \r\n sequence must in
- // vcards be encoded as =0D=OA and we must insert soft-newlines
- // every 75 bytes.
- for ($ii = 0; $ii < \strlen($val); ++$ii) {
- $ord = \ord($val[$ii]);
- // These characters are encoded as themselves.
- if ($ord >= 32 && $ord <= 126) {
- $lastLine .= $val[$ii];
- } else {
- $lastLine .= '='.\strtoupper(\bin2hex($val[$ii]));
- }
- if (\strlen($lastLine) >= 75) {
- // Soft line break
- $out .= $lastLine."=\r\n ";
- $lastLine = null;
- }
- }
- if (!\is_null($lastLine)) {
- $out .= $lastLine."\r\n";
- }
- return $out;
- } else {
- $str .= ':'.$val;
- $str = \preg_replace(
- '/(
- (?:^.)? # 1 additional byte in first line because of missing single space (see next line)
- .{1,74} # max 75 bytes per line (1 byte is used for a single space added after every CRLF)
- (?![\x80-\xbf]) # prevent splitting multibyte characters
- )/x',
- "$1\r\n ",
- $str
- );
- // remove single space after last CRLF
- return \substr($str, 0, -1);
- }
- }
- /**
- * This method serializes only the value of a property. This is used to
- * create xCard or xCal documents.
- *
- * @param Xml\Writer $writer XML writer
- */
- protected function xmlSerializeValue(Xml\Writer $writer)
- {
- $values = $this->getParts();
- $map = function ($items) use ($values, $writer) {
- foreach ($items as $i => $item) {
- $writer->writeElement(
- $item,
- !empty($values[$i]) ? $values[$i] : null
- );
- }
- };
- switch ($this->name) {
- // Special-casing the REQUEST-STATUS property.
- //
- // See:
- // http://tools.ietf.org/html/rfc6321#section-3.4.1.3
- case 'REQUEST-STATUS':
- $writer->writeElement('code', $values[0]);
- $writer->writeElement('description', $values[1]);
- if (isset($values[2])) {
- $writer->writeElement('data', $values[2]);
- }
- break;
- case 'N':
- $map([
- 'surname',
- 'given',
- 'additional',
- 'prefix',
- 'suffix',
- ]);
- break;
- case 'GENDER':
- $map([
- 'sex',
- 'text',
- ]);
- break;
- case 'ADR':
- $map([
- 'pobox',
- 'ext',
- 'street',
- 'locality',
- 'region',
- 'code',
- 'country',
- ]);
- break;
- case 'CLIENTPIDMAP':
- $map([
- 'sourceid',
- 'uri',
- ]);
- break;
- default:
- parent::xmlSerializeValue($writer);
- }
- }
- /**
- * Validates the node for correctness.
- *
- * The following options are supported:
- * - Node::REPAIR - If something is broken, and automatic repair may
- * be attempted.
- *
- * An array is returned with warnings.
- *
- * Every item in the array has the following properties:
- * * level - (number between 1 and 3 with severity information)
- * * message - (human readable message)
- * * node - (reference to the offending node)
- *
- * @param int $options
- *
- * @return array
- */
- public function validate($options = 0)
- {
- $warnings = parent::validate($options);
- if (isset($this->minimumPropertyValues[$this->name])) {
- $minimum = $this->minimumPropertyValues[$this->name];
- $parts = $this->getParts();
- if (count($parts) < $minimum) {
- $warnings[] = [
- 'level' => $options & self::REPAIR ? 1 : 3,
- 'message' => 'The '.$this->name.' property must have at least '.$minimum.' values. It only has '.count($parts),
- 'node' => $this,
- ];
- if ($options & self::REPAIR) {
- $parts = array_pad($parts, $minimum, '');
- $this->setParts($parts);
- }
- }
- }
- return $warnings;
- }
- }
|