From 8a36ace53da36ef4880d921962e4736e5555845c Mon Sep 17 00:00:00 2001
From: Alex Pott <alex.a.pott@googlemail.com>
Date: Thu, 6 Oct 2022 11:47:59 +0100
Subject: [PATCH] Issue #2895352 by claudiu.cristea, neclimdul, rbosscher, Chi,
 alexpott: Allow tableselect element options to be disabled

---
 .../Core/Render/Element/Tableselect.php       | 10 +++-
 .../modules/form_test/form_test.routing.yml   |  8 ++++
 .../FormTestTableSelectDisabledRowsForm.php   | 44 +++++++++++++++++
 .../Form/ElementsTableSelectTest.php          | 48 +++++++++++++++++++
 4 files changed, 109 insertions(+), 1 deletion(-)
 create mode 100644 core/modules/system/tests/modules/form_test/src/Form/FormTestTableSelectDisabledRowsForm.php

diff --git a/core/lib/Drupal/Core/Render/Element/Tableselect.php b/core/lib/Drupal/Core/Render/Element/Tableselect.php
index 908e66044a74..6258500c0026 100644
--- a/core/lib/Drupal/Core/Render/Element/Tableselect.php
+++ b/core/lib/Drupal/Core/Render/Element/Tableselect.php
@@ -33,7 +33,9 @@
  * $options = [
  *   1 => ['color' => 'Red', 'shape' => 'Triangle'],
  *   2 => ['color' => 'Green', 'shape' => 'Square'],
- *   3 => ['color' => 'Blue', 'shape' => 'Hexagon'],
+ *   // Prevent users from selecting a row by adding a '#disabled' property set
+ *   // to TRUE.
+ *   3 => ['color' => 'Blue', 'shape' => 'Hexagon', '#disabled' => TRUE],
  * ];
  *
  * $form['table'] = array(
@@ -180,6 +182,9 @@ public static function preRenderTableselect($element) {
             }
           }
         }
+        if (!empty($element['#options'][$key]['#disabled'])) {
+          $row['class'][] = 'disabled';
+        }
         $rows[] = $row;
       }
       // Add an empty header or a "Select all" checkbox to provide room for the
@@ -238,6 +243,7 @@ public static function processTableselect(&$element, FormStateInterface $form_st
       foreach ($element['#options'] as $key => $choice) {
         // Do not overwrite manually created children.
         if (!isset($element[$key])) {
+          $disabled = !empty($element['#options'][$key]['#disabled']);
           if ($element['#multiple']) {
             $title = '';
             if (isset($element['#options'][$key]['title']) && is_array($element['#options'][$key]['title'])) {
@@ -254,6 +260,7 @@ public static function processTableselect(&$element, FormStateInterface $form_st
               '#return_value' => $key,
               '#default_value' => isset($value[$key]) ? $key : NULL,
               '#attributes' => $element['#attributes'],
+              '#disabled' => $disabled,
               '#ajax' => $element['#ajax'] ?? NULL,
             ];
           }
@@ -269,6 +276,7 @@ public static function processTableselect(&$element, FormStateInterface $form_st
               '#attributes' => $element['#attributes'],
               '#parents' => $element['#parents'],
               '#id' => HtmlUtility::getUniqueId('edit-' . implode('-', $parents_for_id)),
+              '#disabled' => $disabled,
               '#ajax' => $element['#ajax'] ?? NULL,
             ];
           }
diff --git a/core/modules/system/tests/modules/form_test/form_test.routing.yml b/core/modules/system/tests/modules/form_test/form_test.routing.yml
index 1f1379dd4417..a39d924d97de 100644
--- a/core/modules/system/tests/modules/form_test/form_test.routing.yml
+++ b/core/modules/system/tests/modules/form_test/form_test.routing.yml
@@ -164,6 +164,14 @@ form_test.tableselect_js:
   requirements:
     _access: 'TRUE'
 
+form_test.tableselect_disabled_rows:
+  path: '/form_test/tableselect/disabled-rows/{test_action}'
+  defaults:
+    _form: '\Drupal\form_test\Form\FormTestTableSelectDisabledRowsForm'
+    _title: 'Tableselect disabled rows tests'
+  requirements:
+    _access: 'TRUE'
+
 form_test.vertical_tabs:
   path: '/form_test/vertical-tabs'
   defaults:
diff --git a/core/modules/system/tests/modules/form_test/src/Form/FormTestTableSelectDisabledRowsForm.php b/core/modules/system/tests/modules/form_test/src/Form/FormTestTableSelectDisabledRowsForm.php
new file mode 100644
index 000000000000..8b00bfb7d43e
--- /dev/null
+++ b/core/modules/system/tests/modules/form_test/src/Form/FormTestTableSelectDisabledRowsForm.php
@@ -0,0 +1,44 @@
+<?php
+
+namespace Drupal\form_test\Form;
+
+use Drupal\Core\Form\FormStateInterface;
+
+/**
+ * Builds a form to test table select with disabled rows.
+ *
+ * @internal
+ */
+class FormTestTableSelectDisabledRowsForm extends FormTestTableSelectFormBase {
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getFormId() {
+    return '_form_test_tableselect_disabled_rows_form';
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function buildForm(array $form, FormStateInterface $form_state, $test_action = NULL) {
+    $multiple = ['multiple-true' => TRUE, 'multiple-false' => FALSE][$test_action];
+    $form = $this->tableselectFormBuilder($form, $form_state, [
+      '#multiple' => $multiple,
+      '#js_select' => TRUE,
+      '#ajax' => NULL,
+    ]);
+
+    // Disable the second row.
+    $form['tableselect']['#options']['row2']['#disabled'] = TRUE;
+
+    return $form;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function submitForm(array &$form, FormStateInterface $form_state) {
+  }
+
+}
diff --git a/core/modules/system/tests/src/FunctionalJavascript/Form/ElementsTableSelectTest.php b/core/modules/system/tests/src/FunctionalJavascript/Form/ElementsTableSelectTest.php
index c38627553bc9..2cfb76dbd0bf 100644
--- a/core/modules/system/tests/src/FunctionalJavascript/Form/ElementsTableSelectTest.php
+++ b/core/modules/system/tests/src/FunctionalJavascript/Form/ElementsTableSelectTest.php
@@ -60,4 +60,52 @@ public function testAjax() {
     }
   }
 
+  /**
+   * Tests table select with disabled rows.
+   */
+  public function testDisabledRows() {
+    // Asserts that a row number (1 based) is enabled.
+    $assert_row_enabled = function ($delta) {
+      $row = $this->assertSession()->elementExists('xpath', "//table/tbody/tr[$delta]");
+      $this->assertFalse($row->hasClass('disabled'));
+      $input = $row->find('css', 'input[value="row' . $delta . '"]');
+      $this->assertFalse($input->hasAttribute('disabled'));
+    };
+    // Asserts that a row number (1 based) is disabled.
+    $assert_row_disabled = function ($delta) {
+      $row = $this->assertSession()->elementExists('xpath', "//table/tbody/tr[$delta]");
+      $this->assertTrue($row->hasClass('disabled'));
+      $input = $row->find('css', 'input[value="row' . $delta . '"]');
+      $this->assertTrue($input->hasAttribute('disabled'));
+      $this->assertEquals('disabled', $input->getAttribute('disabled'));
+    };
+
+    // Test radios (#multiple == FALSE).
+    $this->drupalGet('form_test/tableselect/disabled-rows/multiple-false');
+
+    // Check that only 'row2' is disabled.
+    $assert_row_enabled(1);
+    $assert_row_disabled(2);
+    $assert_row_enabled(3);
+
+    // Test checkboxes (#multiple == TRUE).
+    $this->drupalGet('form_test/tableselect/disabled-rows/multiple-true');
+
+    // Check that only 'row2' is disabled.
+    $assert_row_enabled(1);
+    $assert_row_disabled(2);
+    $assert_row_enabled(3);
+
+    // Table select with checkboxes allow selection of all options.
+    $select_all_checkbox = $this->assertSession()->elementExists('xpath', '//table/thead/tr/th/input');
+    $select_all_checkbox->check();
+
+    // Check that the disabled option was not enabled or selected.
+    $page = $this->getSession()->getPage();
+    $page->hasCheckedField('row1');
+    $page->hasUncheckedField('row2');
+    $assert_row_disabled(2);
+    $page->hasCheckedField('row3');
+  }
+
 }
-- 
GitLab