ConfigFormBase.php

Same filename in other branches
  1. 9 core/lib/Drupal/Core/Form/ConfigFormBase.php
  2. 8.9.x core/lib/Drupal/Core/Form/ConfigFormBase.php
  3. 10 core/lib/Drupal/Core/Form/ConfigFormBase.php

Namespace

Drupal\Core\Form

File

core/lib/Drupal/Core/Form/ConfigFormBase.php

View source
<?php

namespace Drupal\Core\Form;

use Drupal\Component\Utility\NestedArray;
use Drupal\Core\Config\Config;
use Drupal\Core\Config\ConfigFactoryInterface;
use Drupal\Core\Config\TypedConfigManagerInterface;
use Drupal\Core\Render\Element;
use Drupal\Component\Render\MarkupInterface;
use Drupal\Core\Render\Markup;
use Drupal\Core\Url;
use Symfony\Component\DependencyInjection\ContainerInterface;

/**
 * Base class for implementing system configuration forms.
 *
 * Subclasses of this form can choose to use config validation instead of form-
 * -specific validation logic. To do that, override copyFormValuesToConfig().
 */
abstract class ConfigFormBase extends FormBase {
    use ConfigFormBaseTrait;
    
    /**
     * The $form_state key which stores a map of config keys to form elements.
     *
     * This map is generated and stored by ::storeConfigKeyToFormElementMap(),
     * which is one of the form's #after_build callbacks.
     *
     * @see ::storeConfigKeyToFormElementMap()
     *
     * @var string
     */
    protected const CONFIG_KEY_TO_FORM_ELEMENT_MAP = 'config_targets';
    
    /**
     * Constructs a \Drupal\system\ConfigFormBase object.
     *
     * @param \Drupal\Core\Config\ConfigFactoryInterface $config_factory
     *   The factory for configuration objects.
     * @param \Drupal\Core\Config\TypedConfigManagerInterface $typedConfigManager
     *   The typed config manager.
     */
    public function __construct(ConfigFactoryInterface $config_factory, TypedConfigManagerInterface $typedConfigManager) {
        $this->setConfigFactory($config_factory);
    }
    
    /**
     * {@inheritdoc}
     */
    public static function create(ContainerInterface $container) {
        return new static($container->get('config.factory'), $container->get('config.typed'));
    }
    
    /**
     * Returns the typed config manager service.
     *
     * @return \Drupal\Core\Config\TypedConfigManagerInterface
     *   The typed config manager service.
     */
    protected function typedConfigManager() : TypedConfigManagerInterface {
        if ($this->typedConfigManager instanceof TypedConfigManagerInterface) {
            return $this->typedConfigManager;
        }
        return \Drupal::service('config.typed');
    }
    
    /**
     * {@inheritdoc}
     */
    public function buildForm(array $form, FormStateInterface $form_state) {
        $form['actions']['#type'] = 'actions';
        $form['actions']['submit'] = [
            '#type' => 'submit',
            '#value' => $this->t('Save configuration'),
            '#button_type' => 'primary',
        ];
        // By default, render the form using system-config-form.html.twig.
        $form['#theme'] = 'system_config_form';
        // Load default values from config into any element with a #config_target
        // property.
        $form['#process'][] = '::loadDefaultValuesFromConfig';
        $form['#after_build'][] = '::storeConfigKeyToFormElementMap';
        $form['#after_build'][] = '::checkConfigOverrides';
        return $form;
    }
    
    /**
     * Process callback to recursively load default values from #config_target.
     *
     * @param array $element
     *   The form element.
     *
     * @return array
     *   The form element, with its default value populated.
     */
    public function loadDefaultValuesFromConfig(array $element) : array {
        if (array_key_exists('#config_target', $element) && !array_key_exists('#default_value', $element)) {
            $target = $element['#config_target'];
            if (is_string($target)) {
                $target = ConfigTarget::fromString($target);
            }
            $config = $this->configFactory()
                ->getEditable($target->configName);
            $element['#default_value'] = $target->getValue($config);
        }
        foreach (Element::children($element) as $key) {
            $element[$key] = $this->loadDefaultValuesFromConfig($element[$key]);
        }
        return $element;
    }
    
    /**
     * #after_build callback which stores a map of element names to config keys.
     *
     * This will store an array in the form state whose keys are strings in the
     * form of `CONFIG_NAME:PROPERTY_PATH`, and whose values are instances of
     * \Drupal\Core\Form\ConfigTarget.
     *
     * This callback is run in the form's #after_build stage, rather than
     * #process, to guarantee that all of the form's elements have their final
     * #name and #parents properties set.
     *
     * @param array $element
     *   The element being processed.
     * @param \Drupal\Core\Form\FormStateInterface $form_state
     *   The current form state.
     *
     * @return array
     *   The processed element.
     *
     * @see \Drupal\Core\Form\ConfigFormBase::buildForm()
     */
    public function storeConfigKeyToFormElementMap(array $element, FormStateInterface $form_state) : array {
        // Empty the map to ensure the information is always correct after
        // rebuilding the form.
        $form_state->set(static::CONFIG_KEY_TO_FORM_ELEMENT_MAP, []);
        return $this->doStoreConfigMap($element, $form_state);
    }
    
    /**
     * Helper method for #after_build callback ::storeConfigKeyToFormElementMap().
     *
     * @param array $element
     *   The element being processed.
     * @param \Drupal\Core\Form\FormStateInterface $form_state
     *   The current form state.
     *
     * @return array
     *   The processed element.
     *
     * @see \Drupal\Core\Form\ConfigFormBase::storeConfigKeyToFormElementMap()
     */
    protected function doStoreConfigMap(array $element, FormStateInterface $form_state) : array {
        if (array_key_exists('#config_target', $element)) {
            $map = $form_state->get(static::CONFIG_KEY_TO_FORM_ELEMENT_MAP) ?? [];
            
            /** @var \Drupal\Core\Form\ConfigTarget|string $target */
            $target = $element['#config_target'];
            if (is_string($target)) {
                $target = ConfigTarget::fromString($target);
            }
            elseif ($target->toConfig instanceof \Closure || $target->fromConfig instanceof \Closure) {
                // If the form is using closures as toConfig or fromConfig callables
                // then form cannot be cached.
                $form_state->disableCache();
            }
            foreach ($target->propertyPaths as $property_path) {
                if (isset($map[$target->configName][$property_path])) {
                    throw new \LogicException(sprintf('Two #config_targets both target "%s" in the "%s" config: `%s` and `%s`.', $property_path, $target->configName, '$form[\'' . implode("']['", $map[$target->configName][$property_path]) . '\']', '$form[\'' . implode("']['", $element['#array_parents']) . '\']'));
                }
                $map[$target->configName][$property_path] = $element['#array_parents'];
            }
            $form_state->set(static::CONFIG_KEY_TO_FORM_ELEMENT_MAP, $map);
        }
        foreach (Element::children($element) as $key) {
            $element[$key] = $this->doStoreConfigMap($element[$key], $form_state);
        }
        return $element;
    }
    
    /**
     * {@inheritdoc}
     */
    public function validateForm(array &$form, FormStateInterface $form_state) {
        $map = $form_state->get(static::CONFIG_KEY_TO_FORM_ELEMENT_MAP) ?? [];
        foreach (array_keys($map) as $config_name) {
            $config = $this->configFactory()
                ->getEditable($config_name);
            static::copyFormValuesToConfig($config, $form_state, $form);
            $typed_config = $this->typedConfigManager()
                ->createFromNameAndData($config_name, $config->getRawData());
            $violations = $typed_config->validate();
            // Rather than immediately applying all violation messages to the
            // corresponding form elements, first collect the messages. The structure
            // of the form may cause a single form element to contain multiple config
            // property paths thanks to `type: sequence`. Common example: a <textarea>
            // with one line per sequence item.
            // @see \Drupal\Core\Config\Schema\Sequence
            // @see \Drupal\Core\Config\Schema\SequenceDataDefinition
            $violations_per_form_element = [];
            
            /** @var \Symfony\Component\Validator\ConstraintViolationInterface $violation */
            foreach ($violations as $violation) {
                $property_path = $violation->getPropertyPath();
                // Default to index 0.
                $index = 0;
                // Detect if this is a sequence item property path, and if so, attempt
                // to fall back to the containing sequence's property path.
                if (!isset($map[$config_name][$property_path]) && preg_match("/.*\\.(\\d+)\$/", $property_path, $matches) === 1) {
                    $index = intval($matches[1]);
                    // The property path as known in the config key-to-form element map
                    // will not have the sequence index in it.
                    $property_path = rtrim($property_path, '0123456789.');
                }
                if (isset($map[$config_name][$property_path])) {
                    $config_target = ConfigTarget::fromForm($map[$config_name][$property_path], $form);
                    $form_element_name = implode('][', $config_target->elementParents);
                }
                else {
                    // We cannot determine where to place the violation. The only option
                    // is the entire form.
                    $form_element_name = '';
                }
                $violations_per_form_element[$form_element_name][$index] = $violation;
            }
            // Now that we know how many constraint violation messages exist per form
            // element, set them. This is crucial because FormState::setErrorByName()
            // only allows a single validation error message per form element.
            // @see \Drupal\Core\Form\FormState::setErrorByName()
            foreach ($violations_per_form_element as $form_element_name => $violations) {
                // When only a single message exists, just set it.
                if (count($violations) === 1) {
                    $form_state->setErrorByName($form_element_name, reset($violations)->getMessage());
                    continue;
                }
                // However, if multiple exist, that implies it's a single form element
                // containing a `type: sequence`.
                $form_state->setErrorByName($form_element_name, $this->formatMultipleViolationsMessage($form_element_name, $violations));
            }
        }
    }
    
    /**
     * Formats multiple violation messages associated with a single form element.
     *
     * Validation constraints only know the internal data structure (the
     * configuration schema structure), but this need not be a disadvantage:
     * rather than informing the user some values are wrong, it is possible
     * guide them directly to the Nth entry in the sequence.
     *
     * To further improve the user experience, it is possible to override
     * method in subclasses to use specific knowledge about the structure of the
     * form and the nature of the data being validated, to instead generate more
     * precise and/or shortened violation messages.
     *
     * @param string $form_element_name
     *   The form element for which to format multiple violation messages.
     * @param \Symfony\Component\Validator\ConstraintViolationListInterface $violations
     *   The list of constraint violations that apply to this form element.
     *
     * @return \Drupal\Component\Render\MarkupInterface|\Stringable
     *   The rendered HTML.
     */
    protected function formatMultipleViolationsMessage(string $form_element_name, array $violations) : MarkupInterface|\Stringable {
        $transformed_message_parts = [];
        foreach ($violations as $index => $violation) {
            // Note that `@validation_error_message` (should) already contain a
            // trailing period, hence it is intentionally absent here.
            $transformed_message_parts[] = $this->t('Entry @human_index: @validation_error_message', [
                // Humans start counting from 1, not 0.
'@human_index' => $index + 1,
                // Translators may not necessarily know what "violation constraint
                // messages" are, but they definitely know "validation errors".
'@validation_error_message' => $violation->getMessage(),
            ]);
        }
        // We use \Drupal\Core\Render\Markup::create() here as it is safe,
        // rather than use t() because all input has been escaped by t().
        return Markup::create(implode("\n", $transformed_message_parts));
    }
    
    /**
     * {@inheritdoc}
     */
    public function submitForm(array &$form, FormStateInterface $form_state) {
        $map = $form_state->get(static::CONFIG_KEY_TO_FORM_ELEMENT_MAP) ?? [];
        foreach (array_keys($map) as $config_name) {
            $config = $this->configFactory()
                ->getEditable($config_name);
            static::copyFormValuesToConfig($config, $form_state, $form);
            $config->save();
        }
        $this->messenger()
            ->addStatus($this->t('The configuration options have been saved.'));
    }
    
    /**
     * Copies form values to Config keys.
     *
     * This should not change existing Config key-value pairs that are not being
     * edited by this form.
     *
     * @param \Drupal\Core\Config\Config $config
     *   The configuration being edited.
     * @param \Drupal\Core\Form\FormStateInterface $form_state
     *   The current state of the form.
     * @param array $form
     *   The form array.
     *
     * @see \Drupal\Core\Entity\EntityForm::copyFormValuesToEntity()
     */
    private static function copyFormValuesToConfig(Config $config, FormStateInterface $form_state, array $form) : void {
        $map = $form_state->get(static::CONFIG_KEY_TO_FORM_ELEMENT_MAP);
        foreach ($map[$config->getName()] as $array_parents) {
            $target = ConfigTarget::fromForm($array_parents, $form);
            if ($target->configName !== $config->getName()) {
                continue;
            }
            $value = $form_state->getValue($target->elementParents);
            $target->setValue($config, $value, $form_state);
        }
    }
    
    /**
     * Form #after_build callback: Adds message if overrides exist.
     */
    public function checkConfigOverrides(array $form, FormStateInterface $form_state) : array {
        // Determine which of those editable config keys have overrides.
        $override_links = [];
        $map = $form_state->get(static::CONFIG_KEY_TO_FORM_ELEMENT_MAP) ?? [];
        foreach ($map as $config_name => $config_keys) {
            $stored_config = $this->configFactory
                ->get($config_name);
            if (!$stored_config->hasOverrides()) {
                // The config has no overrides at all. Can be skipped.
                continue;
            }
            foreach ($config_keys as $key => $array_parents) {
                if ($stored_config->hasOverrides($key)) {
                    $element = NestedArray::getValue($form, $array_parents);
                    $override_links[] = [
                        'attributes' => [
                            'title' => $this->t("'@title' form element", [
                                '@title' => $element['#title'],
                            ]),
                        ],
                        'url' => Url::fromUri("internal:#{$element['#id']}"),
                        'title' => $element['#title'],
                    ];
                }
            }
        }
        if (!empty($override_links)) {
            $override_output = [
                '#theme' => 'links__config_overrides',
                '#heading' => [
                    'text' => $this->t('These values are overridden. Changes on this form will be saved, but overrides will take precedence. See <a href="https://www.drupal.org/docs/drupal-apis/configuration-api/configuration-override-system">configuration overrides documentation</a> for more information.'),
                    'level' => 'div',
                ],
                '#links' => $override_links,
            ];
            $form['config_override_status_messages'] = [
                'message' => [
                    '#theme' => 'status_messages',
                    '#message_list' => [
                        'status' => [
                            $override_output,
                        ],
                    ],
                    '#status_headings' => [
                        'status' => $this->t('Status message'),
                    ],
                ],
                // Ensure that the status message is at the top of the form.
'#weight' => array_reduce(Element::children($form), fn(int $carry, string $key) => min($form[$key]['#weight'] ?? 0, $carry), 0) - 1,
            ];
        }
        return $form;
    }

}

Classes

Title Deprecated Summary
ConfigFormBase Base class for implementing system configuration forms.

Buggy or inaccurate documentation? Please file an issue. Need support? Need help programming? Connect with the Drupal community.