diff --git a/core/includes/form.inc b/core/includes/form.inc
index 052c9ce87a9a3fd5e0404cf4444eab51715b8b0c..d248b7b5a9a9fefd572729d2daa9be146124ad5a 100644
--- a/core/includes/form.inc
+++ b/core/includes/form.inc
@@ -235,13 +235,13 @@ function template_preprocess_fieldset(&$variables) {
  * @param array $variables
  *   An associative array containing:
  *   - element: An associative array containing the properties of the element.
- *     Properties used: #attributes, #children, #open,
- *     #description, #id, #title, #value, #optional.
+ *     Properties used: #attributes, #children, #description, #required,
+ *     #summary_attributes, #title, #value.
  */
 function template_preprocess_details(&$variables) {
   $element = $variables['element'];
   $variables['attributes'] = $element['#attributes'];
-  $variables['summary_attributes'] = new Attribute();
+  $variables['summary_attributes'] = new Attribute($element['#summary_attributes']);
   if (!empty($element['#title'])) {
     $variables['summary_attributes']['role'] = 'button';
     if (!empty($element['#attributes']['id'])) {
diff --git a/core/lib/Drupal/Core/Render/Element/Details.php b/core/lib/Drupal/Core/Render/Element/Details.php
index 92bbb32bd1ea8a8d7a3409d48bb317ea43311874..e45131c8eaf927f384f7ab27957d96206b79f247 100644
--- a/core/lib/Drupal/Core/Render/Element/Details.php
+++ b/core/lib/Drupal/Core/Render/Element/Details.php
@@ -15,6 +15,8 @@
  * - #title: The title of the details container. Defaults to "Details".
  * - #open: Indicates whether the container should be open by default.
  *   Defaults to FALSE.
+ * - #summary_attributes: An array of attributes to apply to the <summary>
+ *   element.
  *
  * Usage example:
  * @code
@@ -43,6 +45,7 @@ public function getInfo() {
     $class = get_class($this);
     return [
       '#open' => FALSE,
+      '#summary_attributes' => [],
       '#value' => NULL,
       '#process' => [
         [$class, 'processGroup'],
diff --git a/core/modules/migrate_drupal_ui/src/Form/ReviewForm.php b/core/modules/migrate_drupal_ui/src/Form/ReviewForm.php
index 5fc6446945ff708a6e7d6cf9e08e3ca032d61659..403ef351f3b161df29bc52d026b1116b9964c9c0 100644
--- a/core/modules/migrate_drupal_ui/src/Form/ReviewForm.php
+++ b/core/modules/migrate_drupal_ui/src/Form/ReviewForm.php
@@ -260,12 +260,8 @@ public function buildForm(array $form, FormStateInterface $form_state) {
     $missing_module_list = [
       '#type' => 'details',
       '#open' => TRUE,
-      '#title' => [
-        '#type' => 'html_tag',
-        '#tag' => 'span',
-        '#value' => $this->t('Modules that will not be upgraded'),
-        '#attributes' => ['id' => ['error']],
-      ],
+      '#title' => $this->t('Modules that will not be upgraded'),
+      '#summary_attributes' => ['id' => ['error']],
       '#description' => $this->t('There are no modules installed on your new site to replace these modules. If you proceed with the upgrade now, configuration and/or content needed by these modules will not be available on your new site. For more information, see <a href=":review">Review the pre-upgrade analysis</a> in the <a href=":migrate">Upgrading to Drupal 8</a> handbook.', [':review' => 'https://www.drupal.org/docs/8/upgrade/upgrade-using-web-browser#pre-upgrade-analysis', ':migrate' => 'https://www.drupal.org/docs/8/upgrade']),
       '#weight' => 2,
     ];
@@ -301,12 +297,8 @@ public function buildForm(array $form, FormStateInterface $form_state) {
     // Available migrations.
     $available_module_list = [
       '#type' => 'details',
-      '#title' => [
-        '#type' => 'html_tag',
-        '#tag' => 'span',
-        '#value' => $this->t('Modules that will be upgraded'),
-        '#attributes' => ['id' => ['checked']],
-      ],
+      '#title' => $this->t('Modules that will be upgraded'),
+      '#summary_attributes' => ['id' => ['checked']],
       '#weight' => 3,
     ];
 
diff --git a/core/modules/system/tests/modules/form_test/src/Form/FormTestGroupDetailsForm.php b/core/modules/system/tests/modules/form_test/src/Form/FormTestGroupDetailsForm.php
index 12a05e12d2b1b84f6c57222b71bde4c8bd98cd57..32722fdb0660f4f9c72b5e6d9ed65ba583c13c86 100644
--- a/core/modules/system/tests/modules/form_test/src/Form/FormTestGroupDetailsForm.php
+++ b/core/modules/system/tests/modules/form_test/src/Form/FormTestGroupDetailsForm.php
@@ -39,6 +39,13 @@ public function buildForm(array $form, FormStateInterface $form_state, $required
       '#type' => 'textfield',
       '#title' => 'Nest in details element',
     ];
+    $form['summary_attributes'] = [
+      '#type' => 'details',
+      '#title' => 'Details element with summary attributes',
+      '#summary_attributes' => [
+        'data-summary-attribute' => 'test',
+      ],
+    ];
     return $form;
   }
 
diff --git a/core/modules/system/tests/src/Functional/Form/ElementTest.php b/core/modules/system/tests/src/Functional/Form/ElementTest.php
index 57da8daf278649a48d6b88d47c81700337845f80..6eefa484bddee173d67b01ae513ccf825aca9f99 100644
--- a/core/modules/system/tests/src/Functional/Form/ElementTest.php
+++ b/core/modules/system/tests/src/Functional/Form/ElementTest.php
@@ -223,4 +223,12 @@ public function testFormElementErrors() {
     $this->assertText('I am an error on the details element.');
   }
 
+  /**
+   * Tests summary attributes of details.
+   */
+  public function testDetailsSummaryAttributes() {
+    $this->drupalGet('form-test/group-details');
+    $this->assertTrue($this->cssSelect('summary[data-summary-attribute="test"]'));
+  }
+
 }