You are here

YamlFormTableSelectSort.php in YAML Form 8

File

src/Element/YamlFormTableSelectSort.php
View source
<?php

namespace Drupal\yamlform\Element;

use Drupal\Core\Form\FormStateInterface;
use Drupal\Core\Render\Element;
use Drupal\Core\Render\Element\Table;
use Drupal\Core\StringTranslation\TranslatableMarkup;

/**
 * Provides a form element for a sortable tableselect element.
 *
 * @FormElement("yamlform_tableselect_sort")
 */
class YamlFormTableSelectSort extends Table {

  /**
   * {@inheritdoc}
   */
  public function getInfo() {
    $class = get_class($this);
    return [
      '#input' => TRUE,
      '#js_select' => TRUE,
      '#responsive' => TRUE,
      '#sticky' => FALSE,
      '#pre_render' => [
        [
          $class,
          'preRenderTable',
        ],
        [
          $class,
          'preRenderYamlFormTableSelectSort',
        ],
      ],
      '#process' => [
        [
          $class,
          'processYamlFormTableSelectSort',
        ],
      ],
      '#options' => [],
      '#empty' => '',
      '#theme' => 'table__tableselect_sort',
    ];
  }

  /**
   * {@inheritdoc}
   */
  public static function valueCallback(&$element, $input, FormStateInterface $form_state) {
    if ($input === FALSE) {
      $value = [];
      $element += [
        '#default_value' => [],
      ];
      foreach ($element['#default_value'] as $key => $flag) {
        if ($flag) {
          $value[$key] = $key;
        }
      }
      return $value;
    }
    else {
      if (is_array($input)) {
        uasort($input, [
          'Drupal\\Component\\Utility\\SortArray',
          'sortByWeightElement',
        ]);
        $values = [];
        foreach ($input as $key => $item) {
          if (!empty($item['checkbox'])) {
            $values[$item['checkbox']] = $item['checkbox'];
          }
        }
        return $values;
      }
      else {
        return [];
      }
    }
  }

  /**
   * Prepares a 'yamlform_tableselect_sort' #type element for rendering.
   *
   * @return array
   *   The processed element.
   */
  public static function preRenderYamlFormTableSelectSort($element) {
    $rows = [];
    $header = $element['#header'];
    if (!empty($element['#options'])) {

      // Generate a table row for each selectable item in #options.
      foreach (Element::children($element) as $key) {
        $row = [];
        $row['data'] = [];

        // Set the row to be draggable.
        $row['class'] = [
          'draggable',
        ];
        if (isset($element['#options'][$key]['#attributes'])) {
          $row += $element['#options'][$key]['#attributes'];
        }

        // Render the checkbox element.
        $row['data'][] = \Drupal::service('renderer')
          ->render($element[$key]['checkbox']);

        // As table.html.twig only maps header and row columns by order, create
        // the correct order by iterating over the header fields.
        foreach ($element['#header'] as $fieldname => $title) {

          // A row cell can span over multiple headers, which means less row
          // cells than headers could be present.
          if (isset($element['#options'][$key][$fieldname])) {

            // A header can span over multiple cells and in this case the cells
            // are passed in an array. The order of this array determines the
            // order in which they are added.
            if (is_array($element['#options'][$key][$fieldname]) && !isset($element['#options'][$key][$fieldname]['data'])) {
              foreach ($element['#options'][$key][$fieldname] as $cell) {
                $row['data'][] = $cell;
              }
            }
            else {
              $row['data'][] = $element['#options'][$key][$fieldname];
            }
          }
        }

        // Render the weight element.
        $row['data'][] = \Drupal::service('renderer')
          ->render($element[$key]['weight']);
        $rows[] = $row;
      }

      // Add an empty header or a "Select all" checkbox to provide room for the
      // checkboxes in the first table column.
      if ($element['#js_select']) {

        // Add a "Select all" checkbox.
        $element['#attached']['library'][] = 'core/drupal.tableselect';
        array_unshift($header, [
          'class' => [
            'select-all',
          ],
        ]);
      }
      else {

        // Add an empty header when radio buttons are displayed or a "Select all"
        // checkbox is not desired.
        array_unshift($header, '');
      }
    }

    // Append weight to header.
    $header[] = t('Weight');

    // Set header and rows.
    $element['#header'] = $header;
    $element['#rows'] = $rows;

    // Attach table sort.
    $element['#attributes']['class'][] = 'js-tableselect-sort';
    $element['#attributes']['class'][] = 'tableselect-sort';
    $element['#attached']['library'][] = 'yamlform/yamlform.element.tableselect_sort';
    return $element;
  }

  /**
   * Creates checkbox and weights to populate a 'yamlform_tableselect_order' table.
   *
   * @param array $element
   *   An associative array containing the properties and children of the
   *   yamlform_tableselect_order element.
   * @param \Drupal\Core\Form\FormStateInterface $form_state
   *   The current state of the form.
   * @param array $complete_form
   *   The complete form structure.
   *
   * @return array
   *   The processed element.
   */
  public static function processYamlFormTableSelectSort(&$element, FormStateInterface $form_state, &$complete_form) {
    $value = is_array($element['#value']) ? $element['#value'] : [];

    // Add validate callback that extracts the associative array of options.
    if (isset($element['#element_validate'])) {
      array_unshift($element['#element_validate'], [
        get_called_class(),
        'validateYamlFormTableSelectOrder',
      ]);
    }
    else {
      $element['#element_validate'][] = [
        get_called_class(),
        'validateYamlFormTableSelectOrder',
      ];
    }
    $element['#tree'] = TRUE;
    if (count($element['#options']) > 0) {
      if (!isset($element['#default_value']) || $element['#default_value'] === 0) {
        $element['#default_value'] = [];
      }

      // Place checked options first.
      $options = [];
      foreach ($value as $checked_option_key) {
        if (isset($element['#options'][$checked_option_key])) {
          $options[$checked_option_key] = $element['#options'][$checked_option_key];
          unset($element['#options'][$checked_option_key]);
        }
      }
      $options += $element['#options'];
      $element['#options'] = $options;

      // Set delta and default weight.
      $delta = count($element['#options']);
      $weight = 0;

      // Create a checkbox and weight for each item in #options in such a way
      // that the value of the yamlform_tableselect_order element behaves as if
      // it had been of type checkboxes with weight.
      foreach ($element['#options'] as $key => $choice) {

        // Do not overwrite manually created children.
        if (!isset($element[$key])) {
          $checkbox_title = '';
          $weight_title = '';
          if (isset($element['#options'][$key]['title']) && is_array($element['#options'][$key]['title'])) {
            if (!empty($element['#options'][$key]['title']['data']['#title'])) {
              $checkbox_title = new TranslatableMarkup('Update @title', [
                '@title' => $element['#options'][$key]['title']['data']['#title'],
              ]);
              $weight_title = new TranslatableMarkup('Weight for @title', [
                '@title' => $element['#options'][$key]['title']['data']['#title'],
              ]);
            }
          }
          $element[$key]['checkbox'] = [
            '#type' => 'checkbox',
            '#title' => $checkbox_title,
            '#title_display' => 'invisible',
            '#return_value' => $key,
            '#default_value' => isset($value[$key]) ? $key : NULL,
            '#attributes' => $element['#attributes'],
            '#ajax' => isset($element['#ajax']) ? $element['#ajax'] : NULL,
          ];
          $element[$key]['weight'] = [
            '#type' => 'weight',
            '#title' => $weight_title,
            '#title_display' => 'invisible',
            '#delta' => $delta,
            '#default_value' => $weight++,
            '#attributes' => [
              'class' => [
                'table-sort-weight',
              ],
            ],
          ];
        }
      }
    }
    else {
      $element['#value'] = [];
    }

    // Enable tabledrag.
    $element['#tabledrag'] = [
      [
        'action' => 'order',
        'relationship' => 'sibling',
        'group' => 'table-sort-weight',
      ],
    ];
    return $element;
  }

  /**
   * Validates yamlform_tableselect_other.
   */
  public static function validateYamlFormTableSelectOrder(&$element, FormStateInterface $form_state, &$complete_form) {

    // Get and sort checked values.
    $checked_values = [];
    foreach (Element::children($element) as $key) {
      if ($element[$key]['checkbox']['#value']) {
        $checked_values[] = [
          'value' => $element[$key]['checkbox']['#value'],
          'weight' => $element[$key]['weight']['#value'],
        ];
      }
    }
    uasort($checked_values, [
      'Drupal\\Component\\Utility\\SortArray',
      'sortByWeightElement',
    ]);

    // Set values.
    $values = [];
    foreach ($checked_values as $item) {
      $values[$item['value']] = $item['value'];
    }

    // Clear the element's value by setting it to NULL.
    $form_state
      ->setValueForElement($element, NULL);

    // Now, set the values as the element's value.
    $form_state
      ->setValueForElement($element, $values);
  }

}

Classes

Namesort descending Description
YamlFormTableSelectSort Provides a form element for a sortable tableselect element.