123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214 |
- <?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 = [];
- 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 = [];
- }
- $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 [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 [string] Fully qualified name => (mixed) value)
- */
- private function walk(array $array, $base = '', array &$output = [])
- {
- 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][]') = ['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 = [$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 [$name];
- }
- }
|