<?php /** * Holds a bundle definition in a structure serializable to a native array. */ class Loco_config_ArrayModel extends Loco_config_Model { /** * {@inheritdoc} */ public function createDom(){ return new LocoConfigDocument( array('#document', array(), array() ) ); } /** * Construct model from serialized JSON * @return void */ public function loadJson( $json ){ $root = json_decode( $json, true ); if( ! $root || ! is_array($root) ){ throw new Loco_error_ParseException('Invalid JSON'); } $this->loadArray( $root ); } /** * Construct model from exported array * @return void */ public function loadArray( array $root ){ $dom = $this->getDom(); $dom->load( array('#document', array(), array($root) ) ); } /** * {@inheritdoc} * Emulates *very limited* XPath queries used by the XML DOM. */ public function query( $query, $context = null ){ $match = new LocoConfigNodeList; $query = explode('/', $query ); // absolute path always starts in document if( $absolute = empty($query[0]) ){ $match->append( $this->getDom() ); } // else start with base for relative path else if( $context instanceof LocoConfigNode ){ $match->append( $context ); } while( $query ){ $name = array_shift($query); // self references do nothing if( ! $name || '.' === $name ){ continue; } // match all current branches to produce new set of parents $next = new LocoConfigNodeList; foreach( $match as $parent ){ foreach( $parent->childNodes as $child ){ if( $name === $child->nodeName || ( '*' === $name && $child instanceof LocoConfigElement ) || ( 'text()' === $name && $child instanceof LocoConfigText) ){ $next->append( $child ); } } } $match = $next; } return $match; } } // The following classes are "private" to this file: // They partially implement the same interfaces as the core DOM classes and are used for code hints. // Interfaces are deliberately not used as the real DOM classes would not be able to implement them. /** * Node */ abstract class LocoConfigNode implements IteratorAggregate { /** * Raw data of internal format * @var array */ protected $data; /** * Child nodes once cast to node objects * @var LocoConfigNodeList */ protected $children; /** * @return mixed */ abstract public function export(); final public function __construct( $data ){ $this->data = $data; } protected function get_nodeName(){ return $this->data[0]; } /*protected function get_attributes(){ return $this->data[1]; }*/ protected function get_childNodes(){ return $this->getIterator(); } public function __get( $prop ){ $method = array( $this, 'get_'.$prop ); if( is_callable($method) ){ return call_user_func( $method ); } } /** @return LocoConfigNode */ public function appendChild( LocoConfigNode $child ){ $children = $this->getIterator(); $children->append( $child ); return $child; } /** @return bool */ public function hasChildNodes(){ return (bool) count( $this->getIterator() ); } /** * @return LocoConfigNodeList */ public function getIterator(){ if( ! $this->children ){ $raw = isset($this->data[2]) ? $this->data[2] : array(); $this->children = new LocoConfigNodeList( $this->data[2] ); } return $this->children; } public function get_textContent(){ $s = ''; foreach( $this as $child ){ $s .= $child->get_textContent(); } return $s; } } /** * NodeList */ class LocoConfigNodeList implements Iterator, Countable, ArrayAccess { private $nodes; private $i; private $n; public function __construct( array $nodes = array() ){ $this->nodes = $nodes; $this->n = count( $nodes ); } public function count(){ return $this->n; } public function rewind(){ $this->i = -1; $this->next(); } public function key(){ return $this->i; } public function current(){ return $this[ $this->i ]; } public function valid(){ return is_int($this->i); } public function next(){ if( ++$this->i === $this->n ){ $this->i = null; } } public function offsetExists( $i ){ return $i >= 0 && $i < $this->n; } public function offsetGet( $i ){ $node = $this->nodes[$i]; if( ! $node instanceof LocoConfigNode ){ if( is_array($node) ){ $node = new LocoConfigElement( $node ); } else { $node = new LocoConfigText( $node ); } $this->nodes[$i] = $node; } return $node; } /** * @codeCoverageIgnore */ public function offsetSet( $i, $value ){ throw new Exception('Use append'); } /** * @codeCoverageIgnore */ public function offsetUnset( $i ){ throw new Exception('Read only'); } public function append( LocoConfigNode $node ){ $this->nodes[] = $node; $this->n++; } /** * Revert nodes back to raw array form and return for exporting * @return array */ public function normalize(){ foreach( $this->nodes as $i => $node ){ if( $node instanceof LocoConfigNode ){ $this->nodes[$i] = $node->export(); } } return $this->nodes; } } /** * Document */ class LocoConfigDocument extends LocoConfigNode { /** * Rapidly set new data for document */ public function load( $data ){ $this->data = $data; $this->children = null; } /** * @return LocoConfigElement */ public function createElement( $name ){ return new LocoConfigElement( array( $name, array(), array() ) ); } /** * @return LocoConfigText */ public function createTextNode( $text ){ return new LocoConfigText( $text ); } /** * @return LocoConfigElement */ public function get_documentElement(){ $child = null; foreach( $this as $child ){ break; } return $child; } /** * {@inheritdoc} * Override to keep single element root */ public function export(){ if( $root = $this->get_documentElement() ){ return $root->export(); } } } /** * Element */ class LocoConfigElement extends LocoConfigNode { public function setAttribute( $prop, $value ){ $this->data[1][$prop] = $value; } public function removeAttribute( $prop ){ unset( $this->data[1][$prop] ); } public function getAttribute( $prop ){ if( isset($this->data[1][$prop]) ){ return $this->data[1][$prop]; } return ''; } public function hasAttribute( $prop ){ return isset($this->data[1][$prop]); } /** * {@inheritdoc} */ public function export(){ $raw = $this->data; // return any cast elements back to raw data if( $this->children ){ $raw[2] = $this->children->normalize(); } return $raw; } } /** * Text */ class LocoConfigText extends LocoConfigNode { protected function get_nodeName(){ return '#text'; } public function hasChildNodes(){ return false; } public function getIterator(){ return new ArrayIterator; } public function export(){ return (string) $this->data; } public function get_nodeValue(){ return (string) $this->data; } public function get_textContent(){ return (string) $this->data; } }