You are here

class Form in Zircon Profile 8

Same name in this branch
  1. 8 vendor/symfony/dom-crawler/Form.php \Symfony\Component\DomCrawler\Form
  2. 8 core/lib/Drupal/Core/Render/Element/Form.php \Drupal\Core\Render\Element\Form
Same name and namespace in other branches
  1. 8.0 vendor/symfony/dom-crawler/Form.php \Symfony\Component\DomCrawler\Form

Form represents an HTML form.

@author Fabien Potencier <fabien@symfony.com>

Hierarchy

  • class \Symfony\Component\DomCrawler\Link
    • class \Symfony\Component\DomCrawler\Form implements \Symfony\Component\DomCrawler\ArrayAccess

Expanded class hierarchy of Form

3 files declare their use of Form
BrowserKitDriver.php in vendor/behat/mink-browserkit-driver/src/BrowserKitDriver.php
Client.php in vendor/symfony/browser-kit/Client.php
FormTest.php in vendor/symfony/dom-crawler/Tests/FormTest.php
4 string references to 'Form'
ContactFormListBuilder::buildHeader in core/modules/contact/src/ContactFormListBuilder.php
Builds the header row for the entity listing.
ContactSitewideTest::testSiteWideContact in core/modules/contact/src/Tests/ContactSitewideTest.php
Tests configuration options and the site-wide contact form.
ContactStorageTest::testContactStorage in core/modules/contact/src/Tests/ContactStorageTest.php
Tests configuration options and the site-wide contact form.
EntityDisplayTest::assertDependencyHelper in core/modules/field_ui/src/Tests/EntityDisplayTest.php
Provides a helper for dependency assertions.

File

vendor/symfony/dom-crawler/Form.php, line 22

Namespace

Symfony\Component\DomCrawler
View source
class Form extends Link implements \ArrayAccess {

  /**
   * @var \DOMElement
   */
  private $button;

  /**
   * @var FormFieldRegistry
   */
  private $fields;

  /**
   * @var string
   */
  private $baseHref;

  /**
   * Constructor.
   *
   * @param \DOMElement $node       A \DOMElement instance
   * @param string      $currentUri The URI of the page where the form is embedded
   * @param string      $method     The method to use for the link (if null, it defaults to the method defined by the form)
   * @param string      $baseHref   The URI of the <base> used for relative links, but not for empty action
   *
   * @throws \LogicException if the node is not a button inside a form tag
   */
  public function __construct(\DOMElement $node, $currentUri, $method = null, $baseHref = null) {
    parent::__construct($node, $currentUri, $method);
    $this->baseHref = $baseHref;
    $this
      ->initialize();
  }

  /**
   * Gets the form node associated with this form.
   *
   * @return \DOMElement A \DOMElement instance
   */
  public function getFormNode() {
    return $this->node;
  }

  /**
   * Sets the value of the fields.
   *
   * @param array $values An array of field values
   *
   * @return Form
   */
  public function setValues(array $values) {
    foreach ($values as $name => $value) {
      $this->fields
        ->set($name, $value);
    }
    return $this;
  }

  /**
   * Gets the field values.
   *
   * The returned array does not include file fields (@see getFiles).
   *
   * @return array An array of field values.
   */
  public function getValues() {
    $values = array();
    foreach ($this->fields
      ->all() as $name => $field) {
      if ($field
        ->isDisabled()) {
        continue;
      }
      if (!$field instanceof Field\FileFormField && $field
        ->hasValue()) {
        $values[$name] = $field
          ->getValue();
      }
    }
    return $values;
  }

  /**
   * Gets the file field values.
   *
   * @return array An array of file field values.
   */
  public function getFiles() {
    if (!in_array($this
      ->getMethod(), array(
      'POST',
      'PUT',
      'DELETE',
      'PATCH',
    ))) {
      return array();
    }
    $files = array();
    foreach ($this->fields
      ->all() as $name => $field) {
      if ($field
        ->isDisabled()) {
        continue;
      }
      if ($field instanceof Field\FileFormField) {
        $files[$name] = $field
          ->getValue();
      }
    }
    return $files;
  }

  /**
   * Gets the field values as PHP.
   *
   * This method converts fields with the array notation
   * (like foo[bar] to arrays) like PHP does.
   *
   * @return array An array of field values.
   */
  public function getPhpValues() {
    $values = array();
    foreach ($this
      ->getValues() as $name => $value) {
      $qs = http_build_query(array(
        $name => $value,
      ), '', '&');
      if (!empty($qs)) {
        parse_str($qs, $expandedValue);
        $varName = substr($name, 0, strlen(key($expandedValue)));
        $values = array_replace_recursive($values, array(
          $varName => current($expandedValue),
        ));
      }
    }
    return $values;
  }

  /**
   * Gets the file field values as PHP.
   *
   * This method converts fields with the array notation
   * (like foo[bar] to arrays) like PHP does.
   *
   * @return array An array of field values.
   */
  public function getPhpFiles() {
    $values = array();
    foreach ($this
      ->getFiles() as $name => $value) {
      $qs = http_build_query(array(
        $name => $value,
      ), '', '&');
      if (!empty($qs)) {
        parse_str($qs, $expandedValue);
        $varName = substr($name, 0, strlen(key($expandedValue)));
        $values = array_replace_recursive($values, array(
          $varName => current($expandedValue),
        ));
      }
    }
    return $values;
  }

  /**
   * Gets the URI of the form.
   *
   * The returned URI is not the same as the form "action" attribute.
   * This method merges the value if the method is GET to mimics
   * browser behavior.
   *
   * @return string The URI
   */
  public function getUri() {
    $uri = parent::getUri();
    if (!in_array($this
      ->getMethod(), array(
      'POST',
      'PUT',
      'DELETE',
      'PATCH',
    ))) {
      $query = parse_url($uri, PHP_URL_QUERY);
      $currentParameters = array();
      if ($query) {
        parse_str($query, $currentParameters);
      }
      $queryString = http_build_query(array_merge($currentParameters, $this
        ->getValues()), null, '&');
      $pos = strpos($uri, '?');
      $base = false === $pos ? $uri : substr($uri, 0, $pos);
      $uri = rtrim($base . '?' . $queryString, '?');
    }
    return $uri;
  }
  protected function getRawUri() {
    return $this->node
      ->getAttribute('action');
  }

  /**
   * Gets the form method.
   *
   * If no method is defined in the form, GET is returned.
   *
   * @return string The method
   */
  public function getMethod() {
    if (null !== $this->method) {
      return $this->method;
    }
    return $this->node
      ->getAttribute('method') ? strtoupper($this->node
      ->getAttribute('method')) : 'GET';
  }

  /**
   * Returns true if the named field exists.
   *
   * @param string $name The field name
   *
   * @return bool true if the field exists, false otherwise
   */
  public function has($name) {
    return $this->fields
      ->has($name);
  }

  /**
   * Removes a field from the form.
   *
   * @param string $name The field name
   *
   * @throws \InvalidArgumentException when the name is malformed
   */
  public function remove($name) {
    $this->fields
      ->remove($name);
  }

  /**
   * Gets a named field.
   *
   * @param string $name The field name
   *
   * @return FormField The field instance
   *
   * @throws \InvalidArgumentException When field is not present in this form
   */
  public function get($name) {
    return $this->fields
      ->get($name);
  }

  /**
   * Sets a named field.
   *
   * @param FormField $field The field
   */
  public function set(FormField $field) {
    $this->fields
      ->add($field);
  }

  /**
   * Gets all fields.
   *
   * @return FormField[] An array of fields
   */
  public function all() {
    return $this->fields
      ->all();
  }

  /**
   * Returns true if the named field exists.
   *
   * @param string $name The field name
   *
   * @return bool true if the field exists, false otherwise
   */
  public function offsetExists($name) {
    return $this
      ->has($name);
  }

  /**
   * Gets the value of a field.
   *
   * @param string $name The field name
   *
   * @return FormField The associated Field instance
   *
   * @throws \InvalidArgumentException if the field does not exist
   */
  public function offsetGet($name) {
    return $this->fields
      ->get($name);
  }

  /**
   * Sets the value of a field.
   *
   * @param string       $name  The field name
   * @param string|array $value The value of the field
   *
   * @throws \InvalidArgumentException if the field does not exist
   */
  public function offsetSet($name, $value) {
    $this->fields
      ->set($name, $value);
  }

  /**
   * Removes a field from the form.
   *
   * @param string $name The field name
   */
  public function offsetUnset($name) {
    $this->fields
      ->remove($name);
  }

  /**
   * Disables validation.
   *
   * @return self
   */
  public function disableValidation() {
    foreach ($this->fields
      ->all() as $field) {
      if ($field instanceof Field\ChoiceFormField) {
        $field
          ->disableValidation();
      }
    }
    return $this;
  }

  /**
   * Sets the node for the form.
   *
   * Expects a 'submit' button \DOMElement and finds the corresponding form element, or the form element itself.
   *
   * @param \DOMElement $node A \DOMElement instance
   *
   * @throws \LogicException If given node is not a button or input or does not have a form ancestor
   */
  protected function setNode(\DOMElement $node) {
    $this->button = $node;
    if ('button' === $node->nodeName || 'input' === $node->nodeName && in_array(strtolower($node
      ->getAttribute('type')), array(
      'submit',
      'button',
      'image',
    ))) {
      if ($node
        ->hasAttribute('form')) {

        // if the node has the HTML5-compliant 'form' attribute, use it
        $formId = $node
          ->getAttribute('form');
        $form = $node->ownerDocument
          ->getElementById($formId);
        if (null === $form) {
          throw new \LogicException(sprintf('The selected node has an invalid form attribute (%s).', $formId));
        }
        $this->node = $form;
        return;
      }

      // we loop until we find a form ancestor
      do {
        if (null === ($node = $node->parentNode)) {
          throw new \LogicException('The selected node does not have a form ancestor.');
        }
      } while ('form' !== $node->nodeName);
    }
    elseif ('form' !== $node->nodeName) {
      throw new \LogicException(sprintf('Unable to submit on a "%s" tag.', $node->nodeName));
    }
    $this->node = $node;
  }

  /**
   * Adds form elements related to this form.
   *
   * Creates an internal copy of the submitted 'button' element and
   * the form node or the entire document depending on whether we need
   * to find non-descendant elements through HTML5 'form' attribute.
   */
  private function initialize() {
    $this->fields = new FormFieldRegistry();
    $xpath = new \DOMXPath($this->node->ownerDocument);

    // add submitted button if it has a valid name
    if ('form' !== $this->button->nodeName && $this->button
      ->hasAttribute('name') && $this->button
      ->getAttribute('name')) {
      if ('input' == $this->button->nodeName && 'image' == strtolower($this->button
        ->getAttribute('type'))) {
        $name = $this->button
          ->getAttribute('name');
        $this->button
          ->setAttribute('value', '0');

        // temporarily change the name of the input node for the x coordinate
        $this->button
          ->setAttribute('name', $name . '.x');
        $this
          ->set(new Field\InputFormField($this->button));

        // temporarily change the name of the input node for the y coordinate
        $this->button
          ->setAttribute('name', $name . '.y');
        $this
          ->set(new Field\InputFormField($this->button));

        // restore the original name of the input node
        $this->button
          ->setAttribute('name', $name);
      }
      else {
        $this
          ->set(new Field\InputFormField($this->button));
      }
    }

    // find form elements corresponding to the current form
    if ($this->node
      ->hasAttribute('id')) {

      // corresponding elements are either descendants or have a matching HTML5 form attribute
      $formId = Crawler::xpathLiteral($this->node
        ->getAttribute('id'));
      $fieldNodes = $xpath
        ->query(sprintf('descendant::input[@form=%s] | descendant::button[@form=%s] | descendant::textarea[@form=%s] | descendant::select[@form=%s] | //form[@id=%s]//input[not(@form)] | //form[@id=%s]//button[not(@form)] | //form[@id=%s]//textarea[not(@form)] | //form[@id=%s]//select[not(@form)]', $formId, $formId, $formId, $formId, $formId, $formId, $formId, $formId));
      foreach ($fieldNodes as $node) {
        $this
          ->addField($node);
      }
    }
    else {

      // do the xpath query with $this->node as the context node, to only find descendant elements
      // however, descendant elements with form attribute are not part of this form
      $fieldNodes = $xpath
        ->query('descendant::input[not(@form)] | descendant::button[not(@form)] | descendant::textarea[not(@form)] | descendant::select[not(@form)]', $this->node);
      foreach ($fieldNodes as $node) {
        $this
          ->addField($node);
      }
    }
    if ($this->baseHref && '' !== $this->node
      ->getAttribute('action')) {
      $this->currentUri = $this->baseHref;
    }
  }
  private function addField(\DOMElement $node) {
    if (!$node
      ->hasAttribute('name') || !$node
      ->getAttribute('name')) {
      return;
    }
    $nodeName = $node->nodeName;
    if ('select' == $nodeName || 'input' == $nodeName && 'checkbox' == strtolower($node
      ->getAttribute('type'))) {
      $this
        ->set(new Field\ChoiceFormField($node));
    }
    elseif ('input' == $nodeName && 'radio' == strtolower($node
      ->getAttribute('type'))) {

      // there may be other fields with the same name that are no choice
      // fields already registered (see https://github.com/symfony/symfony/issues/11689)
      if ($this
        ->has($node
        ->getAttribute('name')) && $this
        ->get($node
        ->getAttribute('name')) instanceof ChoiceFormField) {
        $this
          ->get($node
          ->getAttribute('name'))
          ->addChoice($node);
      }
      else {
        $this
          ->set(new Field\ChoiceFormField($node));
      }
    }
    elseif ('input' == $nodeName && 'file' == strtolower($node
      ->getAttribute('type'))) {
      $this
        ->set(new Field\FileFormField($node));
    }
    elseif ('input' == $nodeName && !in_array(strtolower($node
      ->getAttribute('type')), array(
      'submit',
      'button',
      'image',
    ))) {
      $this
        ->set(new Field\InputFormField($node));
    }
    elseif ('textarea' == $nodeName) {
      $this
        ->set(new Field\TextareaFormField($node));
    }
  }

}

Members

Namesort descending Modifiers Type Description Overrides
Form::$baseHref private property
Form::$button private property
Form::$fields private property
Form::addField private function
Form::all public function Gets all fields.
Form::disableValidation public function Disables validation.
Form::get public function Gets a named field.
Form::getFiles public function Gets the file field values.
Form::getFormNode public function Gets the form node associated with this form.
Form::getMethod public function Gets the form method. Overrides Link::getMethod
Form::getPhpFiles public function Gets the file field values as PHP.
Form::getPhpValues public function Gets the field values as PHP.
Form::getRawUri protected function Returns raw URI data. Overrides Link::getRawUri
Form::getUri public function Gets the URI of the form. Overrides Link::getUri
Form::getValues public function Gets the field values.
Form::has public function Returns true if the named field exists.
Form::initialize private function Adds form elements related to this form.
Form::offsetExists public function Returns true if the named field exists.
Form::offsetGet public function Gets the value of a field.
Form::offsetSet public function Sets the value of a field.
Form::offsetUnset public function Removes a field from the form.
Form::remove public function Removes a field from the form.
Form::set public function Sets a named field.
Form::setNode protected function Sets the node for the form. Overrides Link::setNode
Form::setValues public function Sets the value of the fields.
Form::__construct public function Constructor. Overrides Link::__construct
Link::$currentUri protected property
Link::$method protected property
Link::$node protected property
Link::canonicalizePath protected function Returns the canonicalized URI path (see RFC 3986, section 5.2.4).
Link::cleanupAnchor private function Remove the anchor from the uri.
Link::cleanupQuery private function Remove the query string from the uri.
Link::cleanupUri private function Removes the query string and the anchor from the given uri.
Link::getNode public function Gets the node associated with this link.