You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
162 lines
4.4 KiB
162 lines
4.4 KiB
2 years ago
|
<?php
|
||
|
/**
|
||
|
* Copyright 2012-2017 Horde LLC (http://www.horde.org/)
|
||
|
*
|
||
|
* See the enclosed file COPYING for license information (LGPL). If you
|
||
|
* did not receive this file, see http://www.horde.org/licenses/lgpl21.
|
||
|
*
|
||
|
* @category Horde
|
||
|
* @copyright 2012-2017 Horde LLC
|
||
|
* @license http://www.horde.org/licenses/lgpl21 LGPL 2.1
|
||
|
* @package Mime
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* This class parses a multipart/related MIME part (RFC 2387) to provide
|
||
|
* information on the part contents.
|
||
|
*
|
||
|
* @author Michael Slusarz <slusarz@horde.org>
|
||
|
* @category Horde
|
||
|
* @copyright 2012-2017 Horde LLC
|
||
|
* @license http://www.horde.org/licenses/lgpl21 LGPL 2.1
|
||
|
* @package Mime
|
||
|
*/
|
||
|
class Horde_Mime_Related implements IteratorAggregate
|
||
|
{
|
||
|
/**
|
||
|
* Content IDs.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
protected $_cids = array();
|
||
|
|
||
|
/**
|
||
|
* Start ID.
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $_start;
|
||
|
|
||
|
/**
|
||
|
* Constructor.
|
||
|
*
|
||
|
* @param Horde_Mime_Part $mime_part A MIME part object. Must be of
|
||
|
* type multipart/related.
|
||
|
*/
|
||
|
public function __construct(Horde_Mime_Part $mime_part)
|
||
|
{
|
||
|
if ($mime_part->getType() != 'multipart/related') {
|
||
|
throw new InvalidArgumentException('MIME part must be of type multipart/related');
|
||
|
}
|
||
|
|
||
|
$id = null;
|
||
|
$ids = array();
|
||
|
$related_id = $mime_part->getMimeId();
|
||
|
|
||
|
/* Build a list of parts -> CIDs. */
|
||
|
foreach ($mime_part->partIterator() as $val) {
|
||
|
$part_id = $val->getMimeId();
|
||
|
$ids[] = $part_id;
|
||
|
|
||
|
if ((strcmp($related_id, $part_id) !== 0) &&
|
||
|
($cid = $val->getContentId())) {
|
||
|
$this->_cids[$part_id] = $cid;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* Look at the 'start' parameter to determine which part to start
|
||
|
* with. If no 'start' parameter, use the first part (RFC 2387
|
||
|
* [3.1]). */
|
||
|
if ($start = $mime_part->getContentTypeParameter('start')) {
|
||
|
$id = $this->cidSearch(trim($start, '<> '));
|
||
|
}
|
||
|
|
||
|
if (empty($id)) {
|
||
|
reset($ids);
|
||
|
$id = next($ids);
|
||
|
}
|
||
|
|
||
|
$this->_start = $id;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Return the start ID.
|
||
|
*
|
||
|
* @return string The start ID.
|
||
|
*/
|
||
|
public function startId()
|
||
|
{
|
||
|
return $this->_start;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Search for a CID in the related part.
|
||
|
*
|
||
|
* @param string $cid The CID to search for.
|
||
|
*
|
||
|
* @return string The MIME ID or false if not found.
|
||
|
*/
|
||
|
public function cidSearch($cid)
|
||
|
{
|
||
|
return array_search($cid, $this->_cids);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Scan for CID strings in HTML data and replace with data returned from
|
||
|
* a callback method.
|
||
|
*
|
||
|
* @param mixed $text The HTML text (can be Horde_Domhtml object).
|
||
|
* @param callback $callback Callback method. Receives three arguments:
|
||
|
* MIME ID, the attribute name containing the
|
||
|
* content ID, and the node object. Expects
|
||
|
* return value of URL to display the data.
|
||
|
* @param string $charset HTML data charset.
|
||
|
*
|
||
|
* @return Horde_Domhtml A Horde_Domhtml object.
|
||
|
*/
|
||
|
public function cidReplace($text, $callback, $charset = 'UTF-8')
|
||
|
{
|
||
|
$dom = ($text instanceof Horde_Domhtml)
|
||
|
? $text
|
||
|
: new Horde_Domhtml($text, $charset);
|
||
|
|
||
|
foreach ($dom as $node) {
|
||
|
if ($node instanceof DOMElement) {
|
||
|
switch (Horde_String::lower($node->tagName)) {
|
||
|
case 'body':
|
||
|
case 'td':
|
||
|
$this->_cidReplace($node, 'background', $callback);
|
||
|
break;
|
||
|
|
||
|
case 'img':
|
||
|
$this->_cidReplace($node, 'src', $callback);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $dom;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
*/
|
||
|
protected function _cidReplace($node, $attribute, $callback)
|
||
|
{
|
||
|
if ($node->hasAttribute($attribute)) {
|
||
|
$val = $node->getAttribute($attribute);
|
||
|
if ((strpos($val, 'cid:') === 0) &&
|
||
|
($id = $this->cidSearch(substr($val, 4)))) {
|
||
|
$node->setAttribute($attribute, call_user_func($callback, $id, $attribute, $node));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* IteratorAggregate method. */
|
||
|
|
||
|
public function getIterator()
|
||
|
{
|
||
|
return new ArrayIterator($this->_cids);
|
||
|
}
|
||
|
|
||
|
}
|