You are here

ContextFormBase.php in Context 8.4

File

modules/context_ui/src/Form/ContextFormBase.php
View source
<?php

namespace Drupal\context_ui\Form;

use Drupal\Core\Form\FormState;
use Drupal\context\ContextManager;
use Drupal\context\Entity\Context;
use Drupal\Core\Entity\EntityForm;
use Drupal\Core\Form\FormStateInterface;
use Drupal\Core\Plugin\ContextAwarePluginInterface;
use Drupal\Core\Plugin\Context\ContextRepositoryInterface;
use Symfony\Component\DependencyInjection\ContainerInterface;

/**
 * Provides a context form base.
 */
abstract class ContextFormBase extends EntityForm {

  /**
   * The entity being used by this form.
   *
   * @var \Drupal\context\ContextInterface
   */
  protected $entity;

  /**
   * The Context module context manager.
   *
   * @var \Drupal\context\ContextManager
   */
  protected $contextManager;

  /**
   * The Drupal context repository.
   *
   * @var \Drupal\context\Entity\ContextRepositoryInterface
   */
  protected $contextRepository;

  /**
   * Construct a new context form.
   *
   * @param \Drupal\context\ContextManager $contextManager
   *   The Context module context manager.
   * @param \Drupal\context\Entity\ContextRepositoryInterface $contextRepository
   *   The Drupal context repository.
   */
  public function __construct(ContextManager $contextManager, ContextRepositoryInterface $contextRepository) {
    $this->contextManager = $contextManager;
    $this->contextRepository = $contextRepository;
  }

  /**
   * {@inheritdoc}
   */
  public static function create(ContainerInterface $container) {
    return new static($container
      ->get('context.manager'), $container
      ->get('context.repository'));
  }

  /**
   * {@inheritdoc}
   */
  public function form(array $form, FormStateInterface $formState) {
    $form['general'] = [
      '#type' => 'fieldset',
      '#title' => $this
        ->t('General details'),
    ];
    $form['general']['label'] = [
      '#type' => 'textfield',
      '#title' => $this
        ->t('Label'),
      '#default_value' => $this->entity
        ->getLabel(),
      '#required' => TRUE,
      '#description' => $this
        ->t('Enter label for this context.'),
    ];
    $form['general']['name'] = [
      '#type' => 'machine_name',
      '#title' => $this
        ->t('Machine name'),
      '#default_value' => $this->entity
        ->getName(),
      '#machine_name' => [
        'source' => [
          'general',
          'label',
        ],
        'exists' => [
          $this,
          'contextExists',
        ],
      ],
    ];
    $form['general']['group'] = [
      '#type' => 'textfield',
      '#title' => $this
        ->t('Group'),
      '#default_value' => $this->entity
        ->getGroup(),
      '#description' => $this
        ->t('Enter name of the group this context should belong to.'),
      '#autocomplete_route_name' => 'context.groups.autocomplete',
    ];
    $form['general']['description'] = [
      '#type' => 'textfield',
      '#title' => $this
        ->t('Description'),
      '#default_value' => $this->entity
        ->getDescription(),
      '#description' => $this
        ->t('Enter a description for this context definition.'),
    ];
    if ($this->entity
      ->id()) {
      $form['general']['enable'] = [
        '#type' => 'checkbox',
        '#title' => $this
          ->t('Context enabled'),
        '#default_value' => !$this->entity
          ->disabled(),
        '#description' => $this
          ->t('Check it if want to let this context enabled.'),
      ];
    }
    return $form;
  }

  /**
   * {@inheritdoc}
   */
  public function validateForm(array &$form, FormStateInterface $form_state) {
    if ($form_state
      ->hasValue('conditions')) {
      $this
        ->validateConditions($form, $form_state);
    }
    if ($form_state
      ->hasValue('reactions')) {
      $this
        ->validateReactions($form, $form_state);
    }
  }

  /**
   * {@inheritdoc}
   */
  public function submitForm(array &$form, FormStateInterface $form_state) {

    // Enable or disable context.
    if ($this->entity
      ->disabled() == $form_state
      ->getValue('enable') && $this->entity
      ->id()) {
      $this->entity
        ->disable();
    }

    // Save entity values that the built in submit handler cant take care of.
    if ($form_state
      ->hasValue('require_all_conditions')) {
      $this->entity
        ->setRequireAllConditions($form_state
        ->getValue('require_all_conditions'));
    }
    if ($form_state
      ->hasValue('conditions')) {
      $this
        ->handleConditions($form, $form_state);
    }
    if ($form_state
      ->hasValue('reactions')) {
      $this
        ->handleReactions($form, $form_state);
    }

    // If the group is empty set it to the context no group value,
    // otherwise Drupal will save it as an empty string instead.
    if ($form_state
      ->hasValue('group') && empty($form_state
      ->getValue('group'))) {
      $form_state
        ->setValue('group', Context::CONTEXT_GROUP_NONE);
    }

    // Run the default submit method.
    parent::submitForm($form, $form_state);
  }

  /**
   * Handle submitting the condition plugins configuration forms.
   *
   * @param array $form
   *   The rendered form.
   * @param \Drupal\Core\Form\FormStateInterface $form_state
   *   The current form state.
   */
  private function handleConditions(array &$form, FormStateInterface $form_state) {
    $conditions = $form_state
      ->getValue('conditions', []);

    // Loop trough each condition and update the configuration values by
    // submitting the conditions form.
    foreach ($conditions as $condition_id => $configuration) {
      $condition = $this->entity
        ->getCondition($condition_id);
      $condition_values = (new FormState())
        ->setValues($configuration);
      $condition
        ->submitConfigurationForm($form, $condition_values);

      // If the condition is context aware then add context mapping to
      // the condition.
      if ($condition instanceof ContextAwarePluginInterface) {
        $condition
          ->setContextMapping($condition_values
          ->getValue('context_mapping', []));
      }
    }
  }

  /**
   * Handle submitting the context reaction plugins configuration forms.
   *
   * @param array $form
   *   The rendered form.
   * @param \Drupal\Core\Form\FormStateInterface $form_state
   *   The current form state.
   */
  private function handleReactions(array &$form, FormStateInterface $form_state) {
    $reactions = $form_state
      ->getValue('reactions', []);

    // Loop trough each reaction and update the configuration values by
    // submitting the reactions form.
    foreach ($reactions as $reaction_id => $configuration) {
      $reaction = $this->entity
        ->getReaction($reaction_id);
      $reaction_values = (new FormState())
        ->setValues($configuration);
      $reaction
        ->submitConfigurationForm($form, $reaction_values);
    }
  }

  /**
   * Validate the condition plugins configuration forms.
   *
   * @param array $form
   *   The rendered form.
   * @param \Drupal\Core\Form\FormStateInterface $form_state
   *   The current form state.
   */
  private function validateConditions(array &$form, FormStateInterface $form_state) {
    $conditions = $form_state
      ->getValue('conditions', []);

    // Loop trough each condition and update the configuration values by
    // validating the conditions form.
    foreach ($conditions as $condition_id => $configuration) {
      $condition = $this->entity
        ->getCondition($condition_id);
      $condition_values = (new FormState())
        ->setValues($configuration);
      $condition
        ->validateConfigurationForm($form, $condition_values);
      if ($condition_values
        ->hasAnyErrors()) {

        // In case of errors, copy them back from the dummy FormState to the
        // master form.
        foreach ($condition_values
          ->getErrors() as $element => $error) {
          $form_state
            ->setErrorByName("conditions][{$condition_id}][{$element}", $error);
        }
      }
    }
  }

  /**
   * Validate the context reaction plugins configuration forms.
   *
   * @param array $form
   *   The rendered form.
   * @param \Drupal\Core\Form\FormStateInterface $form_state
   *   The current form state.
   */
  private function validateReactions(array &$form, FormStateInterface $form_state) {
    $reactions = $form_state
      ->getValue('reactions', []);

    // Loop trough each reaction and update the configuration values by
    // validating the reactions form.
    foreach ($reactions as $reaction_id => $configuration) {
      $reaction = $this->entity
        ->getReaction($reaction_id);
      $reaction_values = (new FormState())
        ->setValues($configuration);
      $reaction
        ->validateConfigurationForm($form, $reaction_values);

      // Menu root should not be selected as the plugin does not exist.
      if ($reaction_id === 'menu') {
        $menu_items = $reaction_values
          ->getValue('menu_items');
        foreach ($menu_items as $menu_item) {
          $menu = strtok($menu_item, ':');
          $plugin_id = substr($menu_item, strlen($menu) + 1);
          if (!$plugin_id) {
            $plugin_title = $form['reactions']['reaction-menu']['options']['menu_items']['#options'][$menu_item];
            $error = $this
              ->t('Menu root @plugin_title cannot be selected.', [
              '@plugin_title' => $plugin_title,
            ]);
            $form_state
              ->setErrorByName("reactions][{$reaction_id}][{$menu_item}", $error);
          }
        }
      }
      if ($reaction_values
        ->hasAnyErrors()) {

        // In case of errors, copy them back from the dummy FormState to the
        // master form.
        foreach ($reaction_values
          ->getErrors() as $element => $error) {
          $form_state
            ->setErrorByName("reactions][{$reaction_id}][{$element}", $error);
        }
      }
    }
  }

  /**
   * Check to see if a context already exists with the specified name.
   *
   * @param string $name
   *   The machine name to check for.
   *
   * @return bool
   *   TRUE if context exists. FALSE if context doesn't exist.
   */
  public function contextExists($name) {
    return $this->contextManager
      ->contextExists($name);
  }

}

Classes

Namesort descending Description
ContextFormBase Provides a context form base.