| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213 | <?php/* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */namespace Symfony\Component\DomCrawler;use Symfony\Component\DomCrawler\Field\FormField;/** * This is an internal class that must not be used directly. * * @internal */class FormFieldRegistry{    private $fields = array();    private $base;    /**     * Adds a field to the registry.     */    public function add(FormField $field)    {        $segments = $this->getSegments($field->getName());        $target = &$this->fields;        while ($segments) {            if (!\is_array($target)) {                $target = array();            }            $path = array_shift($segments);            if ('' === $path) {                $target = &$target[];            } else {                $target = &$target[$path];            }        }        $target = $field;    }    /**     * Removes a field and its children from the registry.     *     * @param string $name The fully qualified name of the base field     */    public function remove($name)    {        $segments = $this->getSegments($name);        $target = &$this->fields;        while (\count($segments) > 1) {            $path = array_shift($segments);            if (!array_key_exists($path, $target)) {                return;            }            $target = &$target[$path];        }        unset($target[array_shift($segments)]);    }    /**     * Returns the value of the field and its children.     *     * @param string $name The fully qualified name of the field     *     * @return mixed The value of the field     *     * @throws \InvalidArgumentException if the field does not exist     */    public function &get($name)    {        $segments = $this->getSegments($name);        $target = &$this->fields;        while ($segments) {            $path = array_shift($segments);            if (!array_key_exists($path, $target)) {                throw new \InvalidArgumentException(sprintf('Unreachable field "%s"', $path));            }            $target = &$target[$path];        }        return $target;    }    /**     * Tests whether the form has the given field.     *     * @param string $name The fully qualified name of the field     *     * @return bool Whether the form has the given field     */    public function has($name)    {        try {            $this->get($name);            return true;        } catch (\InvalidArgumentException $e) {            return false;        }    }    /**     * Set the value of a field and its children.     *     * @param string $name  The fully qualified name of the field     * @param mixed  $value The value     *     * @throws \InvalidArgumentException if the field does not exist     */    public function set($name, $value)    {        $target = &$this->get($name);        if ((!\is_array($value) && $target instanceof Field\FormField) || $target instanceof Field\ChoiceFormField) {            $target->setValue($value);        } elseif (\is_array($value)) {            $fields = self::create($name, $value);            foreach ($fields->all() as $k => $v) {                $this->set($k, $v);            }        } else {            throw new \InvalidArgumentException(sprintf('Cannot set value on a compound field "%s".', $name));        }    }    /**     * Returns the list of field with their value.     *     * @return FormField[] The list of fields as array((string) Fully qualified name => (mixed) value)     */    public function all()    {        return $this->walk($this->fields, $this->base);    }    /**     * Creates an instance of the class.     *     * This function is made private because it allows overriding the $base and     * the $values properties without any type checking.     *     * @param string $base   The fully qualified name of the base field     * @param array  $values The values of the fields     *     * @return static     */    private static function create($base, array $values)    {        $registry = new static();        $registry->base = $base;        $registry->fields = $values;        return $registry;    }    /**     * Transforms a PHP array in a list of fully qualified name / value.     *     * @param array  $array  The PHP array     * @param string $base   The name of the base field     * @param array  $output The initial values     *     * @return array The list of fields as array((string) Fully qualified name => (mixed) value)     */    private function walk(array $array, $base = '', array &$output = array())    {        foreach ($array as $k => $v) {            $path = empty($base) ? $k : sprintf('%s[%s]', $base, $k);            if (\is_array($v)) {                $this->walk($v, $path, $output);            } else {                $output[$path] = $v;            }        }        return $output;    }    /**     * Splits a field name into segments as a web browser would do.     *     *     getSegments('base[foo][3][]') = array('base', 'foo, '3', '');     *     * @param string $name The name of the field     *     * @return string[] The list of segments     */    private function getSegments($name)    {        if (preg_match('/^(?P<base>[^[]+)(?P<extra>(\[.*)|$)/', $name, $m)) {            $segments = array($m['base']);            while (!empty($m['extra'])) {                $extra = $m['extra'];                if (preg_match('/^\[(?P<segment>.*?)\](?P<extra>.*)$/', $extra, $m)) {                    $segments[] = $m['segment'];                } else {                    $segments[] = $extra;                }            }            return $segments;        }        return array($name);    }}
 |