<?php /** * Zend Framework (http://framework.zend.com/) * * @link http://github.com/zendframework/zf2 for the canonical source repository * @copyright Copyright (c) 2005-2014 Zend Technologies USA Inc. (http://www.zend.com) * @license http://framework.zend.com/license/new-bsd New BSD License */ namespace Zend\View\Helper; use stdClass; use Zend\View; use Zend\View\Exception; /** * Zend\View\Helper\HeadMeta * * @see http://www.w3.org/TR/xhtml1/dtds.html * * Allows the following 'virtual' methods: * @method HeadMeta appendName($keyValue, $content, $modifiers = array()) * @method HeadMeta offsetGetName($index, $keyValue, $content, $modifiers = array()) * @method HeadMeta prependName($keyValue, $content, $modifiers = array()) * @method HeadMeta setName($keyValue, $content, $modifiers = array()) * @method HeadMeta appendHttpEquiv($keyValue, $content, $modifiers = array()) * @method HeadMeta offsetGetHttpEquiv($index, $keyValue, $content, $modifiers = array()) * @method HeadMeta prependHttpEquiv($keyValue, $content, $modifiers = array()) * @method HeadMeta setHttpEquiv($keyValue, $content, $modifiers = array()) * @method HeadMeta appendProperty($keyValue, $content, $modifiers = array()) * @method HeadMeta offsetGetProperty($index, $keyValue, $content, $modifiers = array()) * @method HeadMeta prependProperty($keyValue, $content, $modifiers = array()) * @method HeadMeta setProperty($keyValue, $content, $modifiers = array()) */ class HeadMeta extends Placeholder\Container\AbstractStandalone { /** * Allowed key types * * @var array */ protected $typeKeys = array('name', 'http-equiv', 'charset', 'property', 'itemprop'); /** * Required attributes for meta tag * * @var array */ protected $requiredKeys = array('content'); /** * Allowed modifier keys * * @var array */ protected $modifierKeys = array('lang', 'scheme'); /** * Registry key for placeholder * * @var string */ protected $regKey = 'Zend_View_Helper_HeadMeta'; /** * Constructor * * Set separator to PHP_EOL * */ public function __construct() { parent::__construct(); $this->setSeparator(PHP_EOL); } /** * Retrieve object instance; optionally add meta tag * * @param string $content * @param string $keyValue * @param string $keyType * @param array $modifiers * @param string $placement * @return HeadMeta */ public function __invoke($content = null, $keyValue = null, $keyType = 'name', $modifiers = array(), $placement = Placeholder\Container\AbstractContainer::APPEND) { if ((null !== $content) && (null !== $keyValue)) { $item = $this->createData($keyType, $keyValue, $content, $modifiers); $action = strtolower($placement); switch ($action) { case 'append': case 'prepend': case 'set': $this->$action($item); break; default: $this->append($item); break; } } return $this; } /** * Overload method access * * @param string $method * @param array $args * @throws Exception\BadMethodCallException * @return HeadMeta */ public function __call($method, $args) { if (preg_match( '/^(?P<action>set|(pre|ap)pend|offsetSet)(?P<type>Name|HttpEquiv|Property|Itemprop)$/', $method, $matches) ) { $action = $matches['action']; $type = $this->normalizeType($matches['type']); $argc = count($args); $index = null; if ('offsetSet' == $action) { if (0 < $argc) { $index = array_shift($args); --$argc; } } if (2 > $argc) { throw new Exception\BadMethodCallException( 'Too few arguments provided; requires key value, and content' ); } if (3 > $argc) { $args[] = array(); } $item = $this->createData($type, $args[0], $args[1], $args[2]); if ('offsetSet' == $action) { return $this->offsetSet($index, $item); } $this->$action($item); return $this; } return parent::__call($method, $args); } /** * Render placeholder as string * * @param string|int $indent * @return string */ public function toString($indent = null) { $indent = (null !== $indent) ? $this->getWhitespace($indent) : $this->getIndent(); $items = array(); $this->getContainer()->ksort(); try { foreach ($this as $item) { $items[] = $this->itemToString($item); } } catch (Exception\InvalidArgumentException $e) { trigger_error($e->getMessage(), E_USER_WARNING); return ''; } return $indent . implode($this->escape($this->getSeparator()) . $indent, $items); } /** * Create data item for inserting into stack * * @param string $type * @param string $typeValue * @param string $content * @param array $modifiers * @return stdClass */ public function createData($type, $typeValue, $content, array $modifiers) { $data = new stdClass; $data->type = $type; $data->$type = $typeValue; $data->content = $content; $data->modifiers = $modifiers; return $data; } /** * Build meta HTML string * * @param stdClass $item * @throws Exception\InvalidArgumentException * @return string */ public function itemToString(stdClass $item) { if (!in_array($item->type, $this->typeKeys)) { throw new Exception\InvalidArgumentException(sprintf( 'Invalid type "%s" provided for meta', $item->type )); } $type = $item->type; $modifiersString = ''; foreach ($item->modifiers as $key => $value) { if ($this->view->plugin('doctype')->isHtml5() && $key == 'scheme' ) { throw new Exception\InvalidArgumentException( 'Invalid modifier "scheme" provided; not supported by HTML5' ); } if (!in_array($key, $this->modifierKeys)) { continue; } $modifiersString .= $key . '="' . $this->escape($value) . '" '; } $modifiersString = rtrim($modifiersString); if ('' != $modifiersString) { $modifiersString = ' ' . $modifiersString; } if (method_exists($this->view, 'plugin')) { if ($this->view->plugin('doctype')->isHtml5() && $type == 'charset' ) { $tpl = ($this->view->plugin('doctype')->isXhtml()) ? '<meta %s="%s"/>' : '<meta %s="%s">'; } elseif ($this->view->plugin('doctype')->isXhtml()) { $tpl = '<meta %s="%s" content="%s"%s />'; } else { $tpl = '<meta %s="%s" content="%s"%s>'; } } else { $tpl = '<meta %s="%s" content="%s"%s />'; } $meta = sprintf( $tpl, $type, $this->escape($item->$type), $this->escape($item->content), $modifiersString ); if (isset($item->modifiers['conditional']) && !empty($item->modifiers['conditional']) && is_string($item->modifiers['conditional'])) { $meta = '<!--[if ' . $this->escape($item->modifiers['conditional']) . ']>' . $meta . '<![endif]-->'; } return $meta; } /** * Normalize type attribute of meta * * @param string $type type in CamelCase * @throws Exception\DomainException * @return string */ protected function normalizeType($type) { switch ($type) { case 'Name': return 'name'; case 'HttpEquiv': return 'http-equiv'; case 'Property': return 'property'; case 'Itemprop': return 'itemprop'; default: throw new Exception\DomainException(sprintf( 'Invalid type "%s" passed to normalizeType', $type )); } } /** * Determine if item is valid * * @param mixed $item * @return bool */ protected function isValid($item) { if ((!$item instanceof stdClass) || !isset($item->type) || !isset($item->modifiers) ) { return false; } if (!isset($item->content) && (! $this->view->plugin('doctype')->isHtml5() || (! $this->view->plugin('doctype')->isHtml5() && $item->type !== 'charset')) ) { return false; } // <meta itemprop= ... /> is only supported with doctype html if (! $this->view->plugin('doctype')->isHtml5() && $item->type === 'itemprop' ) { return false; } // <meta property= ... /> is only supported with doctype RDFa if (!$this->view->plugin('doctype')->isRdfa() && $item->type === 'property' ) { return false; } return true; } /** * Append * * @param string $value * @return void * @throws Exception\InvalidArgumentException */ public function append($value) { if (!$this->isValid($value)) { throw new Exception\InvalidArgumentException( 'Invalid value passed to append; please use appendMeta()' ); } return $this->getContainer()->append($value); } /** * OffsetSet * * @param string|int $index * @param string $value * @throws Exception\InvalidArgumentException * @return void */ public function offsetSet($index, $value) { if (!$this->isValid($value)) { throw new Exception\InvalidArgumentException( 'Invalid value passed to offsetSet; please use offsetSetName() or offsetSetHttpEquiv()' ); } return $this->getContainer()->offsetSet($index, $value); } /** * OffsetUnset * * @param string|int $index * @throws Exception\InvalidArgumentException * @return void */ public function offsetUnset($index) { if (!in_array($index, $this->getContainer()->getKeys())) { throw new Exception\InvalidArgumentException('Invalid index passed to offsetUnset()'); } return $this->getContainer()->offsetUnset($index); } /** * Prepend * * @param string $value * @throws Exception\InvalidArgumentException * @return void */ public function prepend($value) { if (!$this->isValid($value)) { throw new Exception\InvalidArgumentException( 'Invalid value passed to prepend; please use prependMeta()' ); } return $this->getContainer()->prepend($value); } /** * Set * * @param string $value * @throws Exception\InvalidArgumentException * @return void */ public function set($value) { if (!$this->isValid($value)) { throw new Exception\InvalidArgumentException('Invalid value passed to set; please use setMeta()'); } $container = $this->getContainer(); foreach ($container->getArrayCopy() as $index => $item) { if ($item->type == $value->type && $item->{$item->type} == $value->{$value->type}) { $this->offsetUnset($index); } } return $this->append($value); } /** * Create an HTML5-style meta charset tag. Something like <meta charset="utf-8"> * * Not valid in a non-HTML5 doctype * * @param string $charset * @return HeadMeta Provides a fluent interface */ public function setCharset($charset) { $item = new stdClass; $item->type = 'charset'; $item->charset = $charset; $item->content = null; $item->modifiers = array(); $this->set($item); return $this; } }