From c0881f91e2a74f0617acd968f081722dd8d2e97a Mon Sep 17 00:00:00 2001
From: Alex Pott <alex.a.pott@googlemail.com>
Date: Mon, 7 Oct 2019 09:42:00 +0100
Subject: [PATCH] Issue #109493 by drunken monkey, gnuget, robertDouglass,
 dhirendra.mishra, jhodgdon, jibran, chaby, aleevas, alexpott, DanChadwick,
 Dries, catch: tablesort should allow rendered tables to have columns that
 default to DESC when their header is clicked

---
 core/includes/theme.inc                       |   5 +-
 core/lib/Drupal/Core/Utility/TableSort.php    |  19 ++-
 .../Render/Element/TableSortExtenderTest.php  | 116 +++++++++++++++++-
 3 files changed, 132 insertions(+), 8 deletions(-)

diff --git a/core/includes/theme.inc b/core/includes/theme.inc
index 5fec94c6920f..d2d54979b7be 100644
--- a/core/includes/theme.inc
+++ b/core/includes/theme.inc
@@ -844,6 +844,8 @@ function template_preprocess_image(&$variables) {
  *     - sort: A default sort order for this column ("asc" or "desc"). Only
  *       one column should be given a default sort order because table sorting
  *       only applies to one column at a time.
+ *     - initial_click_sort: Set the initial sort of the column when clicked.
+ *       Defaults to "asc".
  *     - class: An array of values for the 'class' attribute. In particular,
  *       the least important columns that can be hidden on narrow and medium
  *       width screens should have a 'priority-low' class, referenced with the
@@ -997,7 +999,8 @@ function template_preprocess_table(&$variables) {
 
         TableSort::header($cell_content, $cell, $variables['header'], $ts);
 
-        // TableSort::header() removes the 'sort' and 'field' keys.
+        // TableSort::header() removes the 'sort', 'initial_click_sort' and
+        // 'field' keys.
         $cell_attributes = new Attribute($cell);
       }
       $variables['header'][$col_key] = [];
diff --git a/core/lib/Drupal/Core/Utility/TableSort.php b/core/lib/Drupal/Core/Utility/TableSort.php
index 0f9ac41bb8b0..7e02c14c3259 100644
--- a/core/lib/Drupal/Core/Utility/TableSort.php
+++ b/core/lib/Drupal/Core/Utility/TableSort.php
@@ -73,9 +73,11 @@ public static function header(&$cell_content, array &$cell_attributes, array $he
         $image = \Drupal::service('renderer')->render($tablesort_indicator);
       }
       else {
-        // If the user clicks a different header, we want to sort ascending
-        // initially.
-        $context['sort'] = self::ASC;
+        // This determines the sort order when the column gets first clicked by
+        // the user. It is "asc" by default but the sort can be changed if
+        // $cell['initial_click_sort'] is defined. The possible values are "asc"
+        // or "desc".
+        $context['sort'] = $cell_attributes['initial_click_sort'] ?? self::ASC;
         $image = '';
       }
       $cell_content = Link::createFromRoute(new FormattableMarkup('@cell_content@image', ['@cell_content' => $cell_content, '@image' => $image]), '<current>', [], [
@@ -86,7 +88,7 @@ public static function header(&$cell_content, array &$cell_attributes, array $he
         ]),
       ]);
 
-      unset($cell_attributes['field'], $cell_attributes['sort']);
+      unset($cell_attributes['field'], $cell_attributes['sort'], $cell_attributes['initial_click_sort']);
     }
   }
 
@@ -150,8 +152,13 @@ public static function getSort(array $headers, Request $request) {
     // Find out which header is currently being sorted.
     $order = static::getOrder($headers, $request);
     foreach ($headers as $header) {
-      if (is_array($header) && isset($header['data']) && $header['data'] == $order['name'] && isset($header['sort'])) {
-        return $header['sort'];
+      if (is_array($header) && isset($header['data']) && $header['data'] == $order['name']) {
+        if (isset($header['sort'])) {
+          return $header['sort'];
+        }
+        if (isset($header['initial_click_sort'])) {
+          return $header['initial_click_sort'];
+        }
       }
     }
     return self::ASC;
diff --git a/core/tests/Drupal/KernelTests/Core/Render/Element/TableSortExtenderTest.php b/core/tests/Drupal/KernelTests/Core/Render/Element/TableSortExtenderTest.php
index 7f3fab4653a0..2a7ea0410c84 100644
--- a/core/tests/Drupal/KernelTests/Core/Render/Element/TableSortExtenderTest.php
+++ b/core/tests/Drupal/KernelTests/Core/Render/Element/TableSortExtenderTest.php
@@ -136,7 +136,121 @@ public function testTableSortInit() {
     ];
     $ts = TableSort::getContextFromRequest($headers, $request);
     $this->verbose(strtr('$ts: <pre>!ts</pre>', ['!ts' => Html::escape(var_export($ts, TRUE))]));
-    $this->assertEqual($ts, $expected_ts, 'Complex table headers plus $_GET parameters sorted correctly.');
+    $this->assertEquals($expected_ts, $ts, 'Complex table headers plus $_GET parameters sorted correctly.');
+
+    // Test the initial_click_sort parameter.
+    $headers = [
+      'foo',
+      [
+        'data' => '1',
+        'field' => 'one',
+        'initial_click_sort' => 'desc',
+        'colspan' => 1,
+      ],
+      [
+        'data' => '2',
+        'field' => 'two',
+      ],
+      [
+        'data' => '3',
+        'field' => 'three',
+        'initial_click_sort' => 'desc',
+        'sort' => 'asc',
+      ],
+      [
+        'data' => '4',
+        'field' => 'four',
+        'initial_click_sort' => 'asc',
+      ],
+      [
+        'data' => '5',
+        'field' => 'five',
+        'initial_click_sort' => 'foo',
+      ],
+    ];
+    $request = Request::createFromGlobals();
+    $request->query->replace([
+      'order' => '1',
+    ]);
+    \Drupal::getContainer()->get('request_stack')->push($request);
+    $ts = TableSort::getContextFromRequest($headers, $request);
+    $this->verbose(strtr('$ts: <pre>!ts</pre>', ['!ts' => Html::escape(var_export($ts, TRUE))]));
+    $expected_ts = [
+      'name' => '1',
+      'sql' => 'one',
+      'sort' => 'desc',
+      'query' => [],
+    ];
+    $this->verbose(strtr('$ts: <pre>!ts</pre>', ['!ts' => Html::escape(var_export($ts, TRUE))]));
+    $this->assertEquals($expected_ts, $ts, 'Complex table headers using the initial_click_sort parameter are sorted correctly.');
+
+    // Test that if the initial_click_sort parameter is not defined, the default
+    // must be used instead (which is "asc").
+    $request = Request::createFromGlobals();
+    $request->query->replace([
+      'order' => '2',
+    ]);
+    \Drupal::getContainer()->get('request_stack')->push($request);
+    $ts = TableSort::getContextFromRequest($headers, $request);
+    $expected_ts = [
+      'name' => '2',
+      'sql' => 'two',
+      'sort' => 'asc',
+      'query' => [],
+    ];
+    $this->verbose(strtr('$ts: <pre>!ts</pre>', ['!ts' => Html::escape(var_export($ts, TRUE))]));
+    $this->assertEquals($expected_ts, $ts, 'Complex table headers without using the initial_click_sort parameter are sorted correctly.');
+
+    // Test that if the initial_click_sort parameter is defined, and the sort
+    // parameter is defined as well, the sort parameter has precedence.
+    $request = Request::createFromGlobals();
+    $request->query->replace([
+      'order' => '3',
+    ]);
+    \Drupal::getContainer()->get('request_stack')->push($request);
+    $ts = TableSort::getContextFromRequest($headers, $request);
+    $expected_ts = [
+      'name' => '3',
+      'sql' => 'three',
+      'sort' => 'asc',
+      'query' => [],
+    ];
+    $this->verbose(strtr('$ts: <pre>!ts</pre>', ['!ts' => Html::escape(var_export($ts, TRUE))]));
+    $this->assertEquals($expected_ts, $ts, 'Complex table headers using the initial_click_sort and sort parameters are sorted correctly.');
+
+    // Test that if the initial_click_sort parameter is defined and the value
+    // is "asc" it should be sorted correctly.
+    $request = Request::createFromGlobals();
+    $request->query->replace([
+      'order' => '4',
+    ]);
+    \Drupal::getContainer()->get('request_stack')->push($request);
+    $ts = TableSort::getContextFromRequest($headers, $request);
+    $expected_ts = [
+      'name' => '4',
+      'sql' => 'four',
+      'sort' => 'asc',
+      'query' => [],
+    ];
+    $this->verbose(strtr('$ts: <pre>!ts</pre>', ['!ts' => Html::escape(var_export($ts, TRUE))]));
+    $this->assertEquals($expected_ts, $ts, 'Complex table headers with the initial_click_sort set as ASC are sorted correctly.');
+
+    // Tests that if the initial_click_sort is defined with a non expected value
+    // that value will be passed as the "sort" value.
+    $request = Request::createFromGlobals();
+    $request->query->replace([
+      'order' => '5',
+    ]);
+    \Drupal::getContainer()->get('request_stack')->push($request);
+    $ts = TableSort::getContextFromRequest($headers, $request);
+    $expected_ts = [
+      'name' => '5',
+      'sql' => 'five',
+      'sort' => 'foo',
+      'query' => [],
+    ];
+    $this->verbose(strtr('$ts: <pre>!ts</pre>', ['!ts' => Html::escape(var_export($ts, TRUE))]));
+    $this->assertEquals($expected_ts, $ts, 'Complex table headers with the initial_click_sort set as foo are sorted correctly.');
   }
 
 }
-- 
GitLab