From dd3a59762dccd7c269f14b9cb979ceffc5e5160c Mon Sep 17 00:00:00 2001
From: Nathaniel Catchpole <catch@35733.no-reply.drupal.org>
Date: Wed, 21 Jan 2015 15:21:06 +0000
Subject: [PATCH] Issue #2368797 by Wim Leers, dawehner, rteijeiro: Optimize
 ajaxPageState to keep Drupal 8 sites fast on high-latency networks, prevent
 CSS/JS aggregation from taking down sites and use HTTP GET for AJAX requests

---
 core/core.libraries.yml                       |   6 +
 core/core.services.yml                        |   6 +
 core/includes/common.inc                      | 746 +-----------------
 core/includes/theme.inc                       |  23 +-
 core/lib/Drupal/Core/Ajax/AjaxResponse.php    | 110 +--
 .../Drupal/Core/Ajax/OpenDialogCommand.php    |  17 -
 core/lib/Drupal/Core/Asset/AssetResolver.php  | 347 ++++++++
 .../Core/Asset/AssetResolverInterface.php     |  85 ++
 core/lib/Drupal/Core/Asset/AttachedAssets.php |  98 +++
 .../Core/Asset/AttachedAssetsInterface.php    |  85 ++
 .../Core/Asset/CssCollectionGrouper.php       |   2 +-
 .../Core/Asset/CssCollectionRenderer.php      |  43 +
 .../Drupal/Core/Asset/JsCollectionGrouper.php |   2 +-
 .../Core/Asset/JsCollectionRenderer.php       |  13 +-
 .../Core/Asset/LibraryDependencyResolver.php  | 100 +++
 .../LibraryDependencyResolverInterface.php    |  51 ++
 .../Drupal/Core/Render/Element/Scripts.php    |  67 --
 .../lib/Drupal/Core/Render/Element/Styles.php |  98 ---
 .../Core/Render/MainContent/AjaxRenderer.php  |   5 +-
 .../Render/MainContent/DialogRenderer.php     |   6 +-
 .../Core/Render/MainContent/ModalRenderer.php |   6 +-
 core/misc/ajax.js                             |  15 +-
 .../src/Tests/CKEditorLoadingTest.php         |  12 +-
 .../comment/src/Tests/CommentCSSTest.php      |   4 +-
 core/modules/editor/editor.api.php            |   4 +-
 .../editor/src/Form/EditorImageDialog.php     |   1 +
 .../editor/src/Form/EditorLinkDialog.php      |   1 +
 .../editor/src/Tests/EditorLoadingTest.php    |   2 +-
 .../Controller/FileWidgetAjaxController.php   |   6 +-
 .../modules/history/src/Tests/HistoryTest.php |   4 +-
 core/modules/locale/locale.module             |   5 +-
 .../src/Tests/LocaleLibraryAlterTest.php      |  10 +-
 .../src/Tests/MenuLinkContentUITest.php       |   4 +-
 .../node/src/Tests/NodeTranslationUITest.php  |   4 +-
 .../quickedit/src/QuickEditController.php     |  16 +-
 .../src/Tests/QuickEditLoadingTest.php        |  13 +-
 core/modules/simpletest/simpletest.module     |   3 +-
 core/modules/simpletest/src/WebTestBase.php   |   7 +-
 .../system/src/Tests/Ajax/DialogTest.php      |  24 +-
 .../system/src/Tests/Ajax/FrameworkTest.php   |  93 ++-
 .../src/Tests/Common/AttachedAssetsTest.php   | 180 +++--
 .../system/src/Tests/Theme/TableTest.php      |   8 +-
 core/modules/system/system.module             |  36 +-
 .../ajax_forms_test/ajax_forms_test.module    |  12 +-
 .../src/Form/AjaxFormsTestLazyLoadForm.php    |  15 -
 .../tests/modules/ajax_test/ajax_test.module  |  43 -
 .../src/Controller/AjaxTestController.php     |  31 +-
 .../ajax_test/src/Form/AjaxTestDialogForm.php |   6 +
 .../modules/common_test/common_test.module    |   4 +-
 core/modules/system/theme.api.php             |  22 +-
 core/modules/user/user.module                 |   3 +-
 .../src/Controller/ViewAjaxController.php     |  40 +-
 core/modules/views/src/Tests/ViewAjaxTest.php |   3 +-
 .../Controller/ViewAjaxControllerTest.php     |  23 +-
 .../views_ui/src/Form/Ajax/ViewsFormBase.php  |  14 +-
 core/modules/views_ui/src/Tests/RowUITest.php |  16 +-
 .../Asset/LibraryDependencyResolverTest.php   | 177 +++++
 .../Core/Controller/AjaxRendererTest.php      |   1 +
 58 files changed, 1479 insertions(+), 1299 deletions(-)
 create mode 100644 core/lib/Drupal/Core/Asset/AssetResolver.php
 create mode 100644 core/lib/Drupal/Core/Asset/AssetResolverInterface.php
 create mode 100644 core/lib/Drupal/Core/Asset/AttachedAssets.php
 create mode 100644 core/lib/Drupal/Core/Asset/AttachedAssetsInterface.php
 create mode 100644 core/lib/Drupal/Core/Asset/LibraryDependencyResolver.php
 create mode 100644 core/lib/Drupal/Core/Asset/LibraryDependencyResolverInterface.php
 delete mode 100644 core/lib/Drupal/Core/Render/Element/Scripts.php
 delete mode 100644 core/lib/Drupal/Core/Render/Element/Styles.php
 create mode 100644 core/tests/Drupal/Tests/Core/Asset/LibraryDependencyResolverTest.php

diff --git a/core/core.libraries.yml b/core/core.libraries.yml
index 1a362025bffc..ffa007346377 100644
--- a/core/core.libraries.yml
+++ b/core/core.libraries.yml
@@ -77,6 +77,12 @@ drupal.ajax:
   version: VERSION
   js:
     misc/ajax.js: {}
+  drupalSettings:
+    # These placeholder values will be set by system_js_settings_alter().
+    ajaxPageState:
+      libraries: null
+      theme: null
+      theme_token: null
   dependencies:
     - core/jquery
     - core/drupal
diff --git a/core/core.services.yml b/core/core.services.yml
index 73c78733e582..ee5c7f047d95 100644
--- a/core/core.services.yml
+++ b/core/core.services.yml
@@ -1115,6 +1115,12 @@ services:
   library.discovery.parser:
     class: Drupal\Core\Asset\LibraryDiscoveryParser
     arguments: ['@app.root', '@module_handler']
+  library.dependency_resolver:
+    class: Drupal\Core\Asset\LibraryDependencyResolver
+    arguments: ['@library.discovery']
+  asset.resolver:
+    class: Drupal\Core\Asset\AssetResolver
+    arguments: ['@library.discovery', '@library.dependency_resolver', '@module_handler', '@theme.manager']
   info_parser:
     class: Drupal\Core\Extension\InfoParser
   twig:
diff --git a/core/includes/common.inc b/core/includes/common.inc
index 1c915e9158f4..27c9c540d0b5 100644
--- a/core/includes/common.inc
+++ b/core/includes/common.inc
@@ -759,305 +759,6 @@ function _drupal_add_html_head_link($attributes, $header = FALSE) {
   _drupal_add_html_head($element, 'html_head_link:' . $attributes['rel'] . ':' . $href);
 }
 
-/**
- * Adds a cascading stylesheet to the stylesheet queue.
- *
- * Calling drupal_static_reset('_drupal_add_css') will clear all cascading
- * stylesheets added so far.
- *
- * If CSS aggregation/compression is enabled, all cascading style sheets added
- * with $options['preprocess'] set to TRUE will be merged into one aggregate
- * file and compressed by removing all extraneous white space.
- * Externally hosted stylesheets are never aggregated or compressed.
- *
- * The reason for aggregating the files is outlined quite thoroughly here:
- * http://www.die.net/musings/page_load_time/ "Load fewer external objects. Due
- * to request overhead, one bigger file just loads faster than two smaller ones
- * half its size."
- *
- * $options['preprocess'] should be only set to TRUE when a file is required for
- * all typical visitors and most pages of a site. It is critical that all
- * preprocessed files are added unconditionally on every page, even if the
- * files do not happen to be needed on a page.
- *
- * Non-preprocessed files should only be added to the page when they are
- * actually needed.
- *
- * @param $data
- *   (optional) The stylesheet data to be added, depending on what is passed
- *   through to the $options['type'] parameter:
- *   - 'file': The path to the CSS file relative to the base_path(), or a
- *     stream wrapper URI. For example: "modules/devel/devel.css" or
- *     "public://generated_css/stylesheet_1.css". Note that Modules should
- *     always prefix the names of their CSS files with the module name; for
- *     example, system-menus.css rather than simply menus.css. Themes can
- *     override module-supplied CSS files based on their filenames, and this
- *     prefixing helps prevent confusing name collisions for theme developers.
- *     See drupal_get_css() where the overrides are performed.
- *   - 'external': The absolute path to an external CSS file that is not hosted
- *     on the local server. These files will not be aggregated if CSS
- *     aggregation is enabled.
- * @param $options
- *   (optional) A string defining the 'type' of CSS that is being added in the
- *   $data parameter ('file' or 'external'), or an array which can have any or
- *   all of the following keys:
- *   - 'type': The type of stylesheet being added. Available options are 'file'
- *     or 'external'. Defaults to 'file'.
- *   - 'basename': Force a basename for the file being added. Modules are
- *     expected to use stylesheets with unique filenames, but integration of
- *     external libraries may make this impossible. The basename of
- *     'core/modules/node/node.css' is 'node.css'. If the external library
- *     "node.js" ships with a 'node.css', then a different, unique basename
- *     would be 'node.js.css'.
- *   - 'group': A number identifying the aggregation group in which to add the
- *     stylesheet. Available constants are:
- *     - CSS_AGGREGATE_DEFAULT: (default) Any module-layer CSS.
- *     - CSS_AGGREGATE_THEME: Any theme-layer CSS.
- *     The aggregate group number affects load order and the CSS cascade.
- *     Stylesheets in an aggregate with a lower group number will be output to
- *     the page before stylesheets in an aggregate with a higher group number,
- *     so CSS within higher aggregate groups can take precedence over CSS
- *     within lower aggregate groups.
- *   - 'every_page': For optimal front-end performance when aggregation is
- *     enabled, this should be set to TRUE if the stylesheet is present on every
- *     page of the website for users for whom it is present at all. This
- *     defaults to FALSE. It is set to TRUE for stylesheets added via module and
- *     theme .info.yml files. Modules that add stylesheets within
- *     hook_page_attachments() implementations, or from other code that ensures
- *     that the stylesheet is added to all website pages, should also set this flag
- *     to TRUE. All stylesheets within the same group that have the 'every_page'
- *     flag set to TRUE and do not have 'preprocess' set to FALSE are aggregated
- *     together into a single aggregate file, and that aggregate file can be
- *     reused across a user's entire site visit, leading to faster navigation
- *     between pages.
- *     However, stylesheets that are only needed on pages less frequently
- *     visited, can be added by code that only runs for those particular pages,
- *     and that code should not set the 'every_page' flag. This minimizes the
- *     size of the aggregate file that the user needs to download when first
- *     visiting the website. Stylesheets without the 'every_page' flag are
- *     aggregated into a separate aggregate file. This other aggregate file is
- *     likely to change from page to page, and each new aggregate file needs to
- *     be downloaded when first encountered, so it should be kept relatively
- *     small by ensuring that most commonly needed stylesheets are added to
- *     every page.
- *   - 'weight': The weight of the stylesheet specifies the order in which the
- *     CSS will appear relative to other stylesheets with the same aggregate
- *     group and 'every_page' flag. The exact ordering of stylesheets is as
- *     follows:
- *     - First by aggregate group.
- *     - Then by the 'every_page' flag, with TRUE coming before FALSE.
- *     - Then by weight.
- *     - Then by the order in which the CSS was added. For example, all else
- *       being the same, a stylesheet added by a call to _drupal_add_css() that
- *       happened later in the page request gets added to the page after one for
- *       which _drupal_add_css() happened earlier in the page request.
- *     Available constants are:
- *     - CSS_BASE: Styles for HTML elements ("base" styles).
- *     - CSS_LAYOUT: Styles that layout a page.
- *     - CSS_COMPONENT: Styles for design components (and their associated
- *       states and themes.)
- *     - CSS_STATE: Styles for states that are not included with components.
- *     - CSS_THEME: Styles for themes that are not included with components.
- *     The weight numbers follow the SMACSS convention of CSS categorization.
- *     See http://drupal.org/node/1887922
- *   - 'media': The media type for the stylesheet, e.g., all, print, screen.
- *     Defaults to 'all'. It is extremely important to leave this set to 'all'
- *     or it will negatively impact front-end performance. Instead add a @media
- *     block to the included CSS file.
- *   - 'preprocess': If TRUE and CSS aggregation/compression is enabled, the
- *     styles will be aggregated and compressed. Defaults to TRUE.
- *   - 'browsers': An array containing information specifying which browsers
- *     should load the CSS item. See
- *     \Drupal\Core\Render\Element\HtmlTag::preRenderConditionalComments() for
- *     details.
- *
- * @return
- *   An array of queued cascading stylesheets.
- *
- * @deprecated as of Drupal 8.0. Use the #attached key in render arrays instead.
- *
- * @see drupal_get_css()
- */
-function _drupal_add_css($data = NULL, $options = NULL) {
-  $css = &drupal_static(__FUNCTION__, array());
-
-  // Construct the options, taking the defaults into consideration.
-  if (isset($options)) {
-    if (!is_array($options)) {
-      $options = array('type' => $options);
-    }
-  }
-  else {
-    $options = array();
-  }
-
-  // Create an array of CSS files for each media type first, since each type needs to be served
-  // to the browser differently.
-  if (isset($data)) {
-    $options += array(
-      'type' => 'file',
-      'group' => CSS_AGGREGATE_DEFAULT,
-      'weight' => 0,
-      'every_page' => FALSE,
-      'media' => 'all',
-      'preprocess' => TRUE,
-      'data' => $data,
-      'browsers' => array(),
-    );
-    $options['browsers'] += array(
-      'IE' => TRUE,
-      '!IE' => TRUE,
-    );
-
-    // Files with a query string cannot be preprocessed.
-    if ($options['type'] === 'file' && $options['preprocess'] && strpos($options['data'], '?') !== FALSE) {
-      $options['preprocess'] = FALSE;
-    }
-
-    // Always add a tiny value to the weight, to conserve the insertion order.
-    $options['weight'] += count($css) / 1000;
-
-    // Add the data to the CSS array depending on the type.
-    switch ($options['type']) {
-      case 'file':
-        // Local CSS files are keyed by basename; if a file with the same
-        // basename is added more than once, it gets overridden.
-        // By default, take over the filename as basename.
-        if (!isset($options['basename'])) {
-          $options['basename'] = drupal_basename($data);
-        }
-        $css[$options['basename']] = $options;
-        break;
-
-      default:
-        // External files are keyed by their full URI, so the same CSS file is
-        // not added twice.
-        $css[$data] = $options;
-    }
-  }
-
-  return $css;
-}
-
-/**
- * Returns a themed representation of all stylesheets to attach to the page.
- *
- * It loads the CSS in order, with 'module' first, then 'theme' afterwards.
- * This ensures proper cascading of styles so themes can easily override
- * module styles through CSS selectors.
- *
- * Themes may replace module-defined CSS files by adding a stylesheet with the
- * same filename. For example, themes/bartik/system-menus.css would replace
- * modules/system/system-menus.css. This allows themes to override complete
- * CSS files, rather than specific selectors, when necessary.
- *
- * @param $css
- *   (optional) An array of CSS files. If no array is provided, the default
- *   stylesheets array is used instead.
- * @param $skip_alter
- *   (optional) If set to TRUE, this function skips calling
- *   \Drupal::moduleHandler->alter() on $css, useful when the calling function
- *   passes a $css array that has already been altered.
- *
- * @return
- *   A string of XHTML CSS tags.
- *
- * @see _drupal_add_css()
- */
-function drupal_get_css($css = NULL, $skip_alter = FALSE, $theme_add_css = TRUE) {
-  $theme_info = \Drupal::theme()->getActiveTheme();
-
-  if (!isset($css)) {
-    $css = _drupal_add_css();
-  }
-
-  // Allow modules and themes to alter the CSS items.
-  if (!$skip_alter) {
-    \Drupal::moduleHandler()->alter('css', $css);
-    \Drupal::theme()->alter('css', $css);
-  }
-
-  // Sort CSS items, so that they appear in the correct order.
-  uasort($css, 'drupal_sort_css_js');
-
-  // Allow themes to remove CSS files by basename.
-  if ($stylesheet_remove = $theme_info->getStyleSheetsRemove()) {
-    foreach ($css as $key => $options) {
-      if (isset($options['basename']) && isset($stylesheet_remove[$options['basename']])) {
-        unset($css[$key]);
-      }
-    }
-  }
-  // Allow themes to conditionally override CSS files by basename.
-  if ($stylesheet_override = $theme_info->getStyleSheetsOverride()) {
-    foreach ($css as $key => $options) {
-      if (isset($options['basename']) && isset($stylesheet_override[$options['basename']])) {
-        $css[$key]['data'] = $stylesheet_override[$options['basename']];
-      }
-    }
-  }
-
-  // Render the HTML needed to load the CSS.
-  $styles = array(
-    '#type' => 'styles',
-    '#items' => $css,
-  );
-
-  return drupal_render($styles);
-}
-
-/**
- * Sorts CSS and JavaScript resources.
- *
- * Callback for uasort() within:
- * - drupal_get_css()
- * - drupal_get_js()
- *
- * This sort order helps optimize front-end performance while providing modules
- * and themes with the necessary control for ordering the CSS and JavaScript
- * appearing on a page.
- *
- * @param $a
- *   First item for comparison. The compared items should be associative arrays
- *   of member items from _drupal_add_css() or _drupal_add_js().
- * @param $b
- *   Second item for comparison.
- *
- * @see _drupal_add_css()
- * @see _drupal_add_js()
- */
-function drupal_sort_css_js($a, $b) {
-  // First order by group, so that all items in the CSS_AGGREGATE_DEFAULT group
-  // appear before items in the CSS_AGGREGATE_THEME group. Modules may create
-  // additional groups by defining their own constants.
-  if ($a['group'] < $b['group']) {
-    return -1;
-  }
-  elseif ($a['group'] > $b['group']) {
-    return 1;
-  }
-  // Within a group, order all infrequently needed, page-specific files after
-  // common files needed throughout the website. Separating this way allows for
-  // the aggregate file generated for all of the common files to be reused
-  // across a site visit without being cut by a page using a less common file.
-  elseif ($a['every_page'] && !$b['every_page']) {
-    return -1;
-  }
-  elseif (!$a['every_page'] && $b['every_page']) {
-    return 1;
-  }
-  // Finally, order by weight.
-  elseif ($a['weight'] < $b['weight']) {
-    return -1;
-  }
-  elseif ($a['weight'] > $b['weight']) {
-    return 1;
-  }
-  else {
-    return 0;
-  }
-}
-
 /**
  * Deletes old cached CSS files.
  *
@@ -1124,200 +825,12 @@ function drupal_clean_id_identifier($id) {
 }
 
 /**
- * Adds a JavaScript file or setting to the page.
- *
- * The behavior of this function depends on the parameters it is called with.
- * Generally, it handles the addition of JavaScript to the page. The following
- * actions can be performed using this function:
- * - Add a file ('file'): Adds a reference to a JavaScript file to the page.
- * - Add external JavaScript ('external'): Allows the inclusion of external
- *   JavaScript files that are not hosted on the local server. Note that these
- *   external JavaScript references do not get aggregated when preprocessing is
- *   on.
- * - Add settings ('setting'): Adds settings to Drupal's global storage of
- *   JavaScript settings. Per-page settings are required by some modules to
- *   function properly. All settings will be accessible at drupalSettings.
- *
- * Examples:
- * @code
- *   _drupal_add_js('core/misc/collapse.js');
- *   _drupal_add_js('core/misc/collapse.js', 'file');
- *   _drupal_add_js('http://example.com/example.js', 'external');
- *   _drupal_add_js(array('myModule' => array('key' => 'value')), 'setting');
- * @endcode
- *
- * Calling drupal_static_reset('_drupal_add_js') will clear all JavaScript added
- * so far.
- *
- * If JavaScript aggregation is enabled, all JavaScript files added with
- * $options['preprocess'] set to TRUE will be merged into one aggregate file.
- * Externally hosted JavaScripts are never aggregated.
- *
- * The reason for aggregating the files is outlined quite thoroughly here:
- * http://www.die.net/musings/page_load_time/ "Load fewer external objects. Due
- * to request overhead, one bigger file just loads faster than two smaller ones
- * half its size."
- *
- * $options['preprocess'] should be only set to TRUE when a file is required for
- * all typical visitors and most pages of a site. It is critical that all
- * preprocessed files are added unconditionally on every page, even if the
- * files are not needed on a page.
- *
- * Non-preprocessed files should only be added to the page when they are
- * actually needed.
+ * Constructs an array of the defaults that are used for JavaScript assets.
  *
  * @param $data
- *   (optional) If given, the value depends on the $options parameter, or
- *   $options['type'] if $options is passed as an associative array:
- *   - 'file': Path to the file relative to base_path().
- *   - 'external': The absolute path to an external JavaScript file that is not
- *     hosted on the local server. These files will not be aggregated if
- *     JavaScript aggregation is enabled.
- *   - 'setting': An associative array with configuration options. The array is
- *     merged directly into drupalSettings. All modules should wrap their
- *     actual configuration settings in another variable to prevent conflicts in
- *     the drupalSettings namespace. Items added with a string key will replace
- *     existing settings with that key; items with numeric array keys will be
- *     added to the existing settings array.
- * @param $options
- *   (optional) A string defining the type of JavaScript that is being added in
- *   the $data parameter ('file'/'setting'/'external'), or an associative array.
- *   JavaScript settings should always pass the string 'setting' only. Other
- *   types can have the following elements in the array:
- *   - type: The type of JavaScript that is to be added to the page. Allowed
- *     values are 'file', 'external' or 'setting'. Defaults
- *     to 'file'.
- *   - scope: The location in which you want to place the script. Possible
- *     values are 'header' or 'footer'. If your theme implements different
- *     regions, you can also use these. Defaults to 'header'.
- *   - group: A number identifying the group in which to add the JavaScript.
- *     Available constants are:
- *     - JS_LIBRARY: Any libraries, settings, or jQuery plugins.
- *     - JS_DEFAULT: Any module-layer JavaScript.
- *     - JS_THEME: Any theme-layer JavaScript.
- *     The group number serves as a weight: JavaScript within a lower weight
- *     group is presented on the page before JavaScript within a higher weight
- *     group.
- *   - every_page: For optimal front-end performance when aggregation is
- *     enabled, this should be set to TRUE if the JavaScript is present on every
- *     page of the website for users for whom it is present at all. This
- *     defaults to FALSE. It is set to TRUE for JavaScript files that are added
- *     via module and theme .info.yml files. Modules that add JavaScript within
- *     hook_page_attachments() implementations, or from other code that ensures
- *     that the JavaScript is added to all website pages, should also set this
- *     flag to TRUE. All JavaScript files within the same group and that have the
- *     'every_page' flag set to TRUE and do not have 'preprocess' set to FALSE
- *     are aggregated together into a single aggregate file, and that aggregate
- *     file can be reused across a user's entire site visit, leading to faster
- *     navigation between pages. However, JavaScript that is only needed on
- *     pages less frequently visited, can be added by code that only runs for
- *     those particular pages, and that code should not set the 'every_page'
- *     flag. This minimizes the size of the aggregate file that the user needs
- *     to download when first visiting the website. JavaScript without the
- *     'every_page' flag is aggregated into a separate aggregate file. This
- *     other aggregate file is likely to change from page to page, and each new
- *     aggregate file needs to be downloaded when first encountered, so it
- *     should be kept relatively small by ensuring that most commonly needed
- *     JavaScript is added to every page.
- *   - weight: A number defining the order in which the JavaScript is added to
- *     the page relative to other JavaScript with the same 'scope', 'group',
- *     and 'every_page' value. In some cases, the order in which the JavaScript
- *     is presented on the page is very important. jQuery, for example, must be
- *     added to the page before any jQuery code is run, so jquery.js uses the
- *     JS_LIBRARY group and a weight of -20, jquery.once.js (a library drupal.js
- *     depends on) uses the JS_LIBRARY group and a weight of -19, drupal.js uses
- *     the JS_LIBRARY group and a weight of -1, other libraries use the
- *     JS_LIBRARY group and a weight of 0 or higher, and all other scripts use
- *     one of the other group constants. The exact ordering of JavaScript is as
- *     follows:
- *     - First by scope, with 'header' first, 'footer' last, and any other
- *       scopes provided by a custom theme coming in between, as determined by
- *       the theme.
- *     - Then by group.
- *     - Then by the 'every_page' flag, with TRUE coming before FALSE.
- *     - Then by weight.
- *     - Then by the order in which the JavaScript was added. For example, all
- *       else being the same, JavaScript added by a call to _drupal_add_js() that
- *       happened later in the page request gets added to the page after one for
- *       which _drupal_add_js() happened earlier in the page request.
- *   - cache: If set to FALSE, the JavaScript file is loaded anew on every page
- *     call; in other words, it is not cached. Used only when 'type' references
- *     a JavaScript file. Defaults to TRUE.
- *   - preprocess: If TRUE and JavaScript aggregation is enabled, the script
- *     file will be aggregated. Defaults to TRUE.
- *   - attributes: An associative array of attributes for the <script> tag. This
- *     may be used to add 'defer', 'async', or custom attributes. Note that
- *     setting any attributes will disable preprocessing as though the
- *     'preprocess' option was set to FALSE.
- *   - browsers: An array containing information specifying which browsers
- *     should load the JavaScript item. See
- *     \Drupal\Core\Render\Element\HtmlTag::preRenderConditionalComments() for
- *     details.
+ *   (optional) The default data parameter for the JavaScript asset array.
  *
- * @return
- *   The current array of JavaScript files, settings, and in-line code,
- *   including Drupal defaults, anything previously added with calls to
- *   _drupal_add_js(), and this function call's additions.
- *
- * @deprecated as of Drupal 8.0. Use the #attached key in render arrays instead.
- *
- * @see drupal_get_js()
- */
-function _drupal_add_js($data = NULL, $options = NULL) {
-  $javascript = &drupal_static(__FUNCTION__, array());
-
-  // Construct the options, taking the defaults into consideration.
-  if (isset($options)) {
-    if (!is_array($options)) {
-      $options = array('type' => $options);
-    }
-  }
-  else {
-    $options = array();
-  }
-  $options += drupal_js_defaults($data);
-
-  // Preprocess can only be set if caching is enabled and no attributes are set.
-  $options['preprocess'] = $options['cache'] && empty($options['attributes']) ? $options['preprocess'] : FALSE;
-
-  // Tweak the weight so that files of the same weight are included in the
-  // order of the calls to _drupal_add_js().
-  $options['weight'] += count($javascript) / 1000;
-  if (isset($data)) {
-    switch ($options['type']) {
-      case 'setting':
-        // If the setting array doesn't exist, add defaults values.
-        if (!isset($javascript['drupalSettings'])) {
-          $javascript['drupalSettings'] = array(
-            'type' => 'setting',
-            'scope' => 'header',
-            'group' => JS_SETTING,
-            'every_page' => TRUE,
-            'weight' => 0,
-            'browsers' => array(),
-            'data' => array(),
-          );
-        }
-        $javascript['drupalSettings']['data'] = NestedArray::mergeDeepArray([$javascript['drupalSettings']['data'], $data], TRUE);
-        break;
-
-      default: // 'file' and 'external'
-        // Local and external files must keep their name as the associative key
-        // so the same JavaScript file is not added twice.
-        $javascript[$options['data']] = $options;
-    }
-  }
-  return $javascript;
-}
-
-/**
- * Constructs an array of the defaults that are used for JavaScript items.
- *
- * @param $data
- *   (optional) The default data parameter for the JavaScript item array.
- *
- * @see drupal_get_js()
- * @see _drupal_add_js()
+ * @see hook_js_alter()
  */
 function drupal_js_defaults($data = NULL) {
   return array(
@@ -1335,135 +848,6 @@ function drupal_js_defaults($data = NULL) {
   );
 }
 
-/**
- * Returns a themed presentation of all JavaScript code for the current page.
- *
- * References to JavaScript files are placed in a certain order: first, all
- * 'core' files, then all 'module' and finally all 'theme' JavaScript files
- * are added to the page. Then, all settings are output. If running update.php,
- * all preprocessing is disabled.
- *
- * Note that hook_js_alter(&$javascript) is called during this function call
- * to allow alterations of the JavaScript during its presentation. Calls to
- * _drupal_add_js() from hook_js_alter() will not be added to the output
- * presentation. The correct way to add JavaScript during hook_js_alter()
- * is to add another element to the $javascript array, deriving from
- * drupal_js_defaults(). See locale_js_alter() for an example of this.
- *
- * @param $scope
- *   (optional) The scope for which the JavaScript rules should be returned.
- *   Defaults to 'header'.
- * @param $javascript
- *   (optional) An array with all JavaScript code. Defaults to the default
- *   JavaScript array for the given scope.
- * @param bool $skip_alter
- *   (optional) If set to TRUE, this function skips calling
- *   \Drupal::moduleHandler->alter() on $javascript, useful when the calling
- *   function passes a $javascript array that has already been altered.
- * @param bool $is_ajax
- *   (optional) If set to TRUE, this function is called from an Ajax request and
- *   adds javascript settings to update ajaxPageState values.
- *
- * @return
- *   All JavaScript code segments and includes for the scope as HTML tags.
- *
- * @see _drupal_add_js()
- * @see locale_js_alter()
- * @see drupal_js_defaults()
- */
-function drupal_get_js($scope = 'header', $javascript = NULL, $skip_alter = FALSE, $is_ajax = FALSE, $theme_add_js = TRUE) {
-  if (!isset($javascript)) {
-    $javascript = _drupal_add_js();
-  }
-  if (empty($javascript)) {
-    return '';
-  }
-
-  // Allow modules to alter the JavaScript.
-  if (!$skip_alter) {
-    \Drupal::moduleHandler()->alter('js', $javascript);
-    \Drupal::theme()->alter('js', $javascript);
-  }
-
-  // Filter out elements of the given scope.
-  $items = array();
-  foreach ($javascript as $key => $item) {
-    if ($item['scope'] == $scope) {
-      $items[$key] = $item;
-    }
-  }
-
-  if (!empty($items)) {
-    // Sort the JavaScript files so that they appear in the correct order.
-    uasort($items, 'drupal_sort_css_js');
-    // Don't add settings if there is no other JavaScript on the page, unless
-    // this is an AJAX request.
-    if (!empty($items['drupalSettings']) || $is_ajax) {
-      $theme_key = \Drupal::theme()->getActiveTheme()->getName();
-      // Provide the page with information about the theme that's used, so that
-      // a later AJAX request can be rendered using the same theme.
-      // @see \Drupal\Core\Theme\AjaxBasePageNegotiator
-      $ajaxPageState['theme'] = $theme_key;
-      // Checks that the DB is available before filling theme_token.
-      if (!defined('MAINTENANCE_MODE')) {
-        $ajaxPageState['theme_token'] = \Drupal::csrfToken()->get($theme_key);
-      }
-
-      // Provide the page with information about the individual JavaScript files
-      // used, information not otherwise available when aggregation is enabled.
-      $ajaxPageState['js'] = array_fill_keys(array_keys($javascript), 1);
-      unset($ajaxPageState['js']['drupalSettings']);
-
-      // Provide the page with information about the individual CSS files used,
-      // information not otherwise available when CSS aggregation is enabled.
-      // The setting is attached later in this function, but is set here, so
-      // that CSS files removed in drupal_process_attached() are still
-      // considered "used" and prevented from being added in a later AJAX
-      // request.
-      // Skip if no files were added to the page otherwise jQuery.extend() will
-      // overwrite the drupalSettings.ajaxPageState.css object with an empty
-      // array.
-      $css = _drupal_add_css();
-      if (!empty($css)) {
-        // Cast the array to an object to be on the safe side even if not empty.
-        $ajaxPageState['css'] = (object) array_fill_keys(array_keys($css), 1);
-      }
-
-      _drupal_add_js(['ajaxPageState' => $ajaxPageState], 'setting');
-
-      // If we're outputting the header scope, then this might be the final time
-      // that drupal_get_js() is running, so add the settings to this output as well
-      // as to the _drupal_add_js() cache. If $items['drupalSettings'] doesn't
-      // exist, it's because drupal_get_js() was intentionally passed a
-      // $javascript argument stripped of settings, potentially in order to
-      // override how settings get output, so in this case, do not add the
-      // setting to this output.
-      if ($scope == 'header' && isset($items['drupalSettings'])) {
-        $items['drupalSettings']['data']['ajaxPageState'] = $ajaxPageState;
-      }
-    }
-  }
-
-  // Process the 'drupalSettings' JavaScript asset, if any.
-  if (!empty($items['drupalSettings'])) {
-    $settings = $items['drupalSettings']['data'];
-
-    // Allow modules and themes to alter the JavaScript settings.
-    \Drupal::moduleHandler()->alter('js_settings', $settings);
-    \Drupal::theme()->alter('js_settings', $settings);
-
-    $items['drupalSettings']['data'] = $settings;
-  }
-
-  // Render the HTML needed to load the JavaScript.
-  $elements = array(
-    '#type' => 'scripts',
-    '#items' => $items,
-  );
-
-  return drupal_render($elements);
-}
-
 /**
  * Merges two #attached arrays.
  *
@@ -1517,7 +901,7 @@ function drupal_merge_attached(array $a, array $b) {
 }
 
 /**
- * Adds attachments to a render() structure.
+ * Processes non-asset attachments in a render() structure.
  *
  * Libraries, JavaScript settings, feeds, HTML <head> tags and HTML <head> links
  * are attached to elements using the #attached property. The #attached property
@@ -1528,6 +912,9 @@ function drupal_merge_attached(array $a, array $b) {
  * $build['#attached'] = [
  *   'library' => ['core/jquery']
  * ];
+ * $build['#attached']['http_header'] = [
+ *   ['Content-Type', 'application/rss+xml; charset=utf-8'],
+ * ];
  * @endcode
  *
  * The available keys are:
@@ -1538,58 +925,25 @@ function drupal_merge_attached(array $a, array $b) {
  * - 'html_head_link' (<link> tags in HTML <head>)
  * - 'http_header' (HTTP headers)
  *
- * For example:
- * @code
- * $build['#attached']['http_header'] = array(
- *   array('Content-Type', 'application/rss+xml; charset=utf-8'),
- * );
- * @endcode
+ * This function processes all non-asset attachments, to apply them to the
+ * current response (that means all keys except 'library' and 'drupalSettings').
  *
  * @param array $elements
  *   The structured array describing the data being rendered.
- * @param bool $dependency_check
- *   When TRUE, will exit if a given library's dependencies are missing. When
- *   set to FALSE, will continue to add the libraries, even though one or more
- *   dependencies are missing. Defaults to FALSE.
- *
- * @return bool
- *   FALSE if there were any missing library dependencies; TRUE if all library
- *   dependencies were met.
- *
- * @see _drupal_add_library()
- * @see _drupal_add_js()
- * @see _drupal_add_css()
+ *
  * @see drupal_render()
+ * @see \Drupal\Core\Asset\AssetResolver
+ *
+ * @throws LogicException
+ *   When attaching something of a non-existing attachment type.
  */
-function drupal_process_attached(array $elements, $dependency_check = FALSE) {
-  // Add defaults to the special attached structures that should be processed differently.
-  $elements['#attached'] += array(
-    'library' => array(),
-  );
-
-  // Add the libraries first.
-  $success = TRUE;
-  foreach ($elements['#attached']['library'] as $library) {
-    if (_drupal_add_library($library) === FALSE) {
-      $success = FALSE;
-      // Exit if the dependency is missing.
-      if ($dependency_check) {
-        return $success;
-      }
-    }
-  }
-  unset($elements['#attached']['library']);
-
-  // Convert every JavaScript settings asset into a regular JavaScript asset.
-  // @todo Clean this up in https://www.drupal.org/node/2368797
-  if (!empty($elements['#attached']['drupalSettings'])) {
-    _drupal_add_js($elements['#attached']['drupalSettings'], ['type' => 'setting']);
-    unset($elements['#attached']['drupalSettings']);
+function drupal_process_attached(array $elements) {
+  // Asset attachments are handled by \Drupal\Core\Asset\AssetResolver.
+  foreach (array('library', 'drupalSettings') as $type) {
+    unset($elements['#attached'][$type]);
   }
 
   // Add additional types of attachments specified in the render() structure.
-  // Libraries, JavaScript and CSS have been added already, as they require
-  // special handling.
   foreach ($elements['#attached'] as $callback => $options) {
     foreach ($elements['#attached'][$callback] as $args) {
       // Limit the amount allowed entries.
@@ -1617,8 +971,6 @@ function drupal_process_attached(array $elements, $dependency_check = FALSE) {
       }
     }
   }
-
-  return $success;
 }
 
 /**
@@ -1754,68 +1106,6 @@ function drupal_process_states(&$elements) {
   $elements[$key]['data-drupal-states'] = JSON::encode($elements['#states']);
 }
 
-/**
- * Adds multiple JavaScript or CSS files at the same time.
- *
- * A library defines a set of JavaScript and/or CSS files, optionally using
- * settings, and optionally requiring another library. For example, a library
- * can be a jQuery plugin, a JavaScript framework, or a CSS framework. This
- * function allows modules to load a library defined/shipped by itself or a
- * depending module, without having to add all files of the library separately.
- * Each library is only loaded once.
- *
- * @param $library_name
- *   The name of the library to add.
- * @param $every_page
- *   Set to TRUE if this library is added to every page on the site.
- *
- * @return
- *   TRUE if the library was successfully added; FALSE if the library or one of
- *   its dependencies could not be added.
- *
- * @see \Drupal\Core\Asset\LibraryDiscovery
- * @see hook_library_info_alter()
- *
- * @deprecated in Drupal 8.0.x, will be removed before Drupal 8.0.0
- *   Use #attached on render arrays.
- */
-function _drupal_add_library($library_name, $every_page = NULL) {
-  $added = &drupal_static(__FUNCTION__, array());
-  /** @var \Drupal\Core\Asset\LibraryDiscoveryInterface $library_discovery */
-  $library_discovery = \Drupal::service('library.discovery');
-
-  list($extension, $name) = explode('/', $library_name, 2);
-  // Only process the library if it exists and it was not added already.
-  if (!isset($added[$extension][$name])) {
-    if ($library = $library_discovery->getLibraryByName($extension, $name)) {
-      // Add all components within the library.
-      $elements['#attached'] = array(
-        'library' => $library['dependencies'],
-      );
-      if (isset($library['drupalSettings'])) {
-        $elements['#attached']['drupalSettings'] = $library['drupalSettings'];
-      }
-      $added[$extension][$name] = drupal_process_attached($elements, TRUE);
-
-      // Add both the JavaScript and the CSS.
-      // The parameters for _drupal_add_js() and _drupal_add_css() require special
-      // handling.
-      foreach (array('js', 'css') as $type) {
-        foreach ($library[$type] as $options) {
-          call_user_func('_drupal_add_' . $type, $options['data'], $options);
-        }
-        unset($elements['#attached'][$type]);
-      }
-    }
-    else {
-      // Requested library does not exist.
-      $added[$extension][$name] = FALSE;
-    }
-  }
-
-  return $added[$extension][$name];
-}
-
 /**
  * Assists in attaching the tableDrag JavaScript behavior to a themed table.
  *
diff --git a/core/includes/theme.inc b/core/includes/theme.inc
index 5f584196b184..2664a3472012 100644
--- a/core/includes/theme.inc
+++ b/core/includes/theme.inc
@@ -15,6 +15,7 @@
 use Drupal\Component\Utility\Unicode;
 use Drupal\Component\Utility\UrlHelper;
 use Drupal\Component\Utility\Xss;
+use Drupal\Core\Asset\AttachedAssets;
 use Drupal\Core\Config\Config;
 use Drupal\Core\Config\StorageException;
 use Drupal\Core\Extension\Extension;
@@ -1400,11 +1401,25 @@ function template_preprocess_html(&$variables) {
   // Render the attachments into HTML markup to be used directly in the template
   // for #type => html: html.html.twig.
   $all_attached = ['#attached' => $attached];
+  $assets = AttachedAssets::createFromRenderArray($all_attached);
+  // Take Ajax page state into account, to allow for something like Turbolinks
+  // to be implemented without altering core.
+  // @see https://github.com/rails/turbolinks/
+  $ajax_page_state = \Drupal::request()->request->get('ajax_page_state');
+  $assets->setAlreadyLoadedLibraries(isset($ajax_page_state) ? explode(',', $ajax_page_state['libraries']) : []);
+  // Optimize CSS/JS if necessary, but only during normal site operation.
+  $optimize_css = !defined('MAINTENANCE_MODE') && \Drupal::config('system.performance')->get('css.preprocess');
+  $optimize_js = !defined('MAINTENANCE_MODE') && \Drupal::config('system.performance')->get('js.preprocess');
+  // Render the asset collections.
+  $asset_resolver = \Drupal::service('asset.resolver');
+  $variables['styles'] = \Drupal::service('asset.css.collection_renderer')->render($asset_resolver->getCssAssets($assets, $optimize_css));
+  list($js_assets_header, $js_assets_footer) = $asset_resolver->getJsAssets($assets, $optimize_js);
+  $js_collection_renderer = \Drupal::service('asset.js.collection_renderer');
+  $variables['scripts'] = $js_collection_renderer->render($js_assets_header);
+  $variables['scripts_bottom'] = $js_collection_renderer->render($js_assets_footer);
+
+  // Handle all non-asset attachments.
   drupal_process_attached($all_attached);
-
-  $variables['styles'] = drupal_get_css();
-  $variables['scripts'] = drupal_get_js();
-  $variables['scripts_bottom'] = drupal_get_js('footer');
   $variables['head'] = drupal_get_html_head(FALSE);
 }
 
diff --git a/core/lib/Drupal/Core/Ajax/AjaxResponse.php b/core/lib/Drupal/Core/Ajax/AjaxResponse.php
index f69af8329d48..fcb45afbb174 100644
--- a/core/lib/Drupal/Core/Ajax/AjaxResponse.php
+++ b/core/lib/Drupal/Core/Ajax/AjaxResponse.php
@@ -7,6 +7,7 @@
 
 namespace Drupal\Core\Ajax;
 
+use Drupal\Core\Asset\AttachedAssets;
 use Symfony\Component\HttpFoundation\JsonResponse;
 use Symfony\Component\HttpFoundation\Request;
 use Symfony\Component\HttpFoundation\Response;
@@ -25,6 +26,32 @@ class AjaxResponse extends JsonResponse {
    */
   protected $commands = array();
 
+  /**
+   * The attachments for this Ajax response.
+   *
+   * @var array
+   */
+  protected $attachments = [
+    'library' => [],
+    'drupalSettings' => [],
+  ];
+
+  /**
+   * Sets attachments for this Ajax response.
+   *
+   * When this Ajax response is rendered, it will take care of generating the
+   * necessary Ajax commands, if any.
+   *
+   * @param array $attachments
+   *   An #attached array.
+   *
+   * @return $this
+   */
+  public function setAttachments(array $attachments) {
+    $this->attachments = $attachments;
+    return $this;
+  }
+
   /**
    * Add an AJAX command to the response.
    *
@@ -90,77 +117,54 @@ public function prepareResponse(Request $request) {
    *   An array of commands ready to be returned as JSON.
    */
   protected function ajaxRender(Request $request) {
-    // Ajax responses aren't rendered with html.html.twig, so we have to call
-    // drupal_get_css() and drupal_get_js() here, in order to have new files
-    // added during this request to be loaded by the page. We only want to send
-    // back files that the page hasn't already loaded, so we implement simple
-    // diffing logic using array_diff_key().
     $ajax_page_state = $request->request->get('ajax_page_state');
-    foreach (array('css', 'js') as $type) {
-      // It is highly suspicious if
-      // $request->request->get("ajax_page_state[$type]") is empty, since the
-      // base page ought to have at least one JS file and one CSS file loaded.
-      // It probably indicates an error, and rather than making the page reload
-      // all of the files, instead we return no new files.
-      if (empty($ajax_page_state[$type])) {
-        $items[$type] = array();
-      }
-      else {
-        $function = '_drupal_add_' . $type;
-        $items[$type] = $function();
-        \Drupal::moduleHandler()->alter($type, $items[$type]);
-        // @todo Inline CSS and JS items are indexed numerically. These can't be
-        //   reliably diffed with array_diff_key(), since the number can change
-        //   due to factors unrelated to the inline content, so for now, we
-        //   strip the inline items from Ajax responses, and can add support for
-        //   them when _drupal_add_css() and _drupal_add_js() are changed to use
-        //   a hash of the inline content as the array key.
-        foreach ($items[$type] as $key => $item) {
-          if (is_numeric($key)) {
-            unset($items[$type][$key]);
-          }
-        }
-        // Ensure that the page doesn't reload what it already has.
-        $items[$type] = array_diff_key($items[$type], $ajax_page_state[$type]);
-      }
-    }
+
+    // Aggregate CSS/JS if necessary, but only during normal site operation.
+    $config = \Drupal::config('system.performance');
+    $optimize_css = !defined('MAINTENANCE_MODE') && $config->get('css.preprocess');
+    $optimize_js = !defined('MAINTENANCE_MODE') && $config->get('js.preprocess');
+
+    // Resolve the attached libraries into asset collections.
+    $assets = new AttachedAssets();
+    $assets->setLibraries(isset($this->attachments['library']) ? $this->attachments['library'] : [])
+      ->setAlreadyLoadedLibraries(isset($ajax_page_state) ? explode(',', $ajax_page_state['libraries']) : [])
+      ->setSettings(isset($this->attachments['drupalSettings']) ? $this->attachments['drupalSettings'] : []);
+    $asset_resolver = \Drupal::service('asset.resolver');
+    $css_assets = $asset_resolver->getCssAssets($assets, $optimize_css);
+    list($js_assets_header, $js_assets_footer) = $asset_resolver->getJsAssets($assets, $optimize_js);
 
     // Render the HTML to load these files, and add AJAX commands to insert this
-    // HTML in the page. We pass TRUE as the $skip_alter argument to prevent the
-    // data from being altered again, as we already altered it above. Settings
-    // are handled separately, afterwards.
-    if (isset($items['js']['drupalSettings'])) {
-      unset($items['js']['drupalSettings']);
-    }
-    $styles = drupal_get_css($items['css'], TRUE);
-    $scripts_footer = drupal_get_js('footer', $items['js'], TRUE, TRUE);
-    $scripts_header = drupal_get_js('header', $items['js'], TRUE, TRUE);
+    // HTML in the page. Settings are handled separately, afterwards.
+    $settings = (isset($js_assets_header['drupalSettings'])) ? $js_assets_header['drupalSettings']['data'] : [];
+    unset($js_assets_header['drupalSettings']);
 
-    // Prepend commands to add the resources, preserving their relative order.
+    // Prepend commands to add the assets, preserving their relative order.
     $resource_commands = array();
-    if (!empty($styles)) {
-      $resource_commands[] = new AddCssCommand($styles);
+    $renderer = \Drupal::service('renderer');
+    if (!empty($css_assets)) {
+      $css_render_array = \Drupal::service('asset.css.collection_renderer')->render($css_assets);
+      $resource_commands[] = new AddCssCommand($renderer->render($css_render_array));
     }
-    if (!empty($scripts_header)) {
-      $resource_commands[] = new PrependCommand('head', $scripts_header);
+    if (!empty($js_assets_header)) {
+      $js_header_render_array = \Drupal::service('asset.js.collection_renderer')->render($js_assets_header);
+      $resource_commands[] = new PrependCommand('head', $renderer->render($js_header_render_array));
     }
-    if (!empty($scripts_footer)) {
-      $resource_commands[] = new AppendCommand('body', $scripts_footer);
+    if (!empty($js_assets_footer)) {
+      $js_footer_render_array = \Drupal::service('asset.js.collection_renderer')->render($js_assets_footer);
+      $resource_commands[] = new AppendCommand('body', $renderer->render($js_footer_render_array));
     }
     foreach (array_reverse($resource_commands) as $resource_command) {
       $this->addCommand($resource_command, TRUE);
     }
 
     // Prepend a command to merge changes and additions to drupalSettings.
-    $scripts = _drupal_add_js();
-    if (!empty($scripts['drupalSettings'])) {
-      $settings = $scripts['drupalSettings']['data'];
+    if (!empty($settings)) {
       // During Ajax requests basic path-specific settings are excluded from
       // new drupalSettings values. The original page where this request comes
       // from already has the right values. An Ajax request would update them
       // with values for the Ajax request and incorrectly override the page's
       // values.
-      // @see _drupal_add_js()
+      // @see system_js_settings_alter()
       unset($settings['path']);
       $this->addCommand(new SettingsCommand($settings, TRUE), TRUE);
     }
diff --git a/core/lib/Drupal/Core/Ajax/OpenDialogCommand.php b/core/lib/Drupal/Core/Ajax/OpenDialogCommand.php
index 89ae38cf11b1..9ee09cc8c991 100644
--- a/core/lib/Drupal/Core/Ajax/OpenDialogCommand.php
+++ b/core/lib/Drupal/Core/Ajax/OpenDialogCommand.php
@@ -125,9 +125,6 @@ public function setDialogTitle($title) {
    * Implements \Drupal\Core\Ajax\CommandInterface:render().
    */
   public function render() {
-    // Add the library for handling the dialog in the response.
-    $this->drupalAttachLibrary('core/drupal.dialog.ajax');
-
     // For consistency ensure the modal option is set to TRUE or FALSE.
     $this->dialogOptions['modal'] = isset($this->dialogOptions['modal']) && $this->dialogOptions['modal'];
     return array(
@@ -139,18 +136,4 @@ public function render() {
     );
   }
 
-  /**
-   * Wraps drupal_render.
-   *
-   * @param string $name
-   *   The name of the library.
-   *
-   * @todo Remove once drupal_render is converted to autoloadable code.
-   * @see https://drupal.org/node/2171071
-   */
-  protected function drupalAttachLibrary($name) {
-    $attached['#attached']['library'][] = $name;
-    drupal_process_attached($attached);
-  }
-
 }
diff --git a/core/lib/Drupal/Core/Asset/AssetResolver.php b/core/lib/Drupal/Core/Asset/AssetResolver.php
new file mode 100644
index 000000000000..1b02761907b1
--- /dev/null
+++ b/core/lib/Drupal/Core/Asset/AssetResolver.php
@@ -0,0 +1,347 @@
+<?php
+/**
+ * @file
+ * Contains \Drupal\Core\Asset\AssetResolver.
+ */
+
+namespace Drupal\Core\Asset;
+
+use Drupal\Component\Utility\NestedArray;
+use Drupal\Core\Extension\ModuleHandlerInterface;
+use Drupal\Core\Theme\ThemeManagerInterface;
+
+/**
+ * The default asset resolver.
+ */
+class AssetResolver implements AssetResolverInterface {
+
+  /**
+   * The library discovery service.
+   *
+   * @var \Drupal\Core\Asset\LibraryDiscoveryInterface
+   */
+  protected $libraryDiscovery;
+
+  /**
+   * The library dependency resolver.
+   *
+   * @var \Drupal\Core\Asset\LibraryDependencyResolverInterface
+   */
+  protected $libraryDependencyResolver;
+
+  /**
+   * The module handler.
+   *
+   * @var \Drupal\Core\Extension\ModuleHandlerInterface
+   */
+  protected $moduleHandler;
+
+  /**
+   * The theme manager.
+   *
+   * @var \Drupal\Core\Theme\ThemeManagerInterface
+   */
+  protected $themeManager;
+
+  /**
+   * Constructs a new AssetResolver instance.
+   *
+   * @param \Drupal\Core\Asset\LibraryDiscoveryInterface $library_discovery
+   *   The library discovery service.
+   * @param \Drupal\Core\Asset\LibraryDependencyResolverInterface $library_dependency_resolver
+   *   The library dependency resolver.
+   * @param \Drupal\Core\Extension\ModuleHandlerInterface $module_handler
+   *   The module handler.
+   * @param \Drupal\Core\Theme\ThemeManagerInterface $theme_manager
+   *   The theme manager.
+   */
+  public function __construct(LibraryDiscoveryInterface $library_discovery, LibraryDependencyResolverInterface $library_dependency_resolver, ModuleHandlerInterface $module_handler, ThemeManagerInterface $theme_manager) {
+    $this->libraryDiscovery = $library_discovery;
+    $this->libraryDependencyResolver = $library_dependency_resolver;
+    $this->moduleHandler = $module_handler;
+    $this->themeManager = $theme_manager;
+  }
+
+  /**
+   * Returns the libraries that need to be loaded.
+   *
+   * For example, with core/a depending on core/c and core/b on core/d:
+   * @code
+   * $assets = new AttachedAssets();
+   * $assets->setLibraries(['core/a', 'core/b', 'core/c']);
+   * $assets->setAlreadyLoadedLibraries(['core/c']);
+   * $resolver->getLibrariesToLoad($assets) === ['core/a', 'core/b', 'core/d']
+   * @endcode
+   *
+   * @param \Drupal\Core\Asset\AttachedAssetsInterface $assets
+   *   The assets attached to the current response.
+   *
+   * @return string[]
+   *   A list of libraries and their dependencies, in the order they should be
+   *   loaded, excluding any libraries that have already been loaded.
+   */
+  protected function getLibrariesToLoad(AttachedAssetsInterface $assets) {
+    return array_diff(
+      $this->libraryDependencyResolver->getLibrariesWithDependencies($assets->getLibraries()),
+      $this->libraryDependencyResolver->getLibrariesWithDependencies($assets->getAlreadyLoadedLibraries())
+    );
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getCssAssets(AttachedAssetsInterface $assets, $optimize) {
+    $theme_info = $this->themeManager->getActiveTheme();
+
+    $css = [];
+
+    foreach ($this->getLibrariesToLoad($assets) as $library) {
+      list($extension, $name) = explode('/', $library, 2);
+      $definition = $this->libraryDiscovery->getLibraryByName($extension, $name);
+      if (isset($definition['css'])) {
+        foreach ($definition['css'] as $options) {
+          $options += array(
+            'type' => 'file',
+            'group' => CSS_AGGREGATE_DEFAULT,
+            'weight' => 0,
+            'every_page' => FALSE,
+            'media' => 'all',
+            'preprocess' => TRUE,
+            'browsers' => array(),
+          );
+          $options['browsers'] += array(
+            'IE' => TRUE,
+            '!IE' => TRUE,
+          );
+
+          // Files with a query string cannot be preprocessed.
+          if ($options['type'] === 'file' && $options['preprocess'] && strpos($options['data'], '?') !== FALSE) {
+            $options['preprocess'] = FALSE;
+          }
+
+          // Always add a tiny value to the weight, to conserve the insertion
+          // order.
+          $options['weight'] += count($css) / 1000;
+
+          // Add the data to the CSS array depending on the type.
+          switch ($options['type']) {
+            case 'file':
+              // Local CSS files are keyed by basename; if a file with the same
+              // basename is added more than once, it gets overridden.
+              // By default, take over the filename as basename.
+              if (!isset($options['basename'])) {
+                $options['basename'] = drupal_basename($options['data']);
+              }
+              $css[$options['basename']] = $options;
+              break;
+
+            default:
+              // External files are keyed by their full URI, so the same CSS
+              // file is not added twice.
+              $css[$options['data']] = $options;
+          }
+        }
+      }
+    }
+
+    // Allow modules and themes to alter the CSS assets.
+    $this->moduleHandler->alter('css', $css, $assets);
+    $this->themeManager->alter('css', $css, $assets);
+
+    // Sort CSS items, so that they appear in the correct order.
+    uasort($css, 'static::sort');
+
+    // Allow themes to remove CSS files by basename.
+    if ($stylesheet_remove = $theme_info->getStyleSheetsRemove()) {
+      foreach ($css as $key => $options) {
+        if (isset($options['basename']) && isset($stylesheet_remove[$options['basename']])) {
+          unset($css[$key]);
+        }
+      }
+    }
+    // Allow themes to conditionally override CSS files by basename.
+    if ($stylesheet_override = $theme_info->getStyleSheetsOverride()) {
+      foreach ($css as $key => $options) {
+        if (isset($options['basename']) && isset($stylesheet_override[$options['basename']])) {
+          $css[$key]['data'] = $stylesheet_override[$options['basename']];
+        }
+      }
+    }
+
+    if ($optimize) {
+      $css = \Drupal::service('asset.css.collection_optimizer')->optimize($css);
+    }
+
+    return $css;
+  }
+
+  /**
+   * Returns the JavaScript settings assets for this response's libraries.
+   *
+   * Gathers all drupalSettings from all libraries in the attached assets
+   * collection and merges them, then it merges individual attached settings,
+   * and finally invokes hook_js_settings_alter() to allow alterations of
+   * JavaScript settings by modules and themes.
+   *
+   * @param \Drupal\Core\Asset\AttachedAssetsInterface $assets
+   *   The assets attached to the current response.
+   * @return array
+   *   A (possibly optimized) collection of JavaScript assets.
+   */
+  protected function getJsSettingsAssets(AttachedAssetsInterface $assets) {
+    $settings = [];
+
+    foreach ($this->getLibrariesToLoad($assets) as $library) {
+      list($extension, $name) = explode('/', $library, 2);
+      $definition = $this->libraryDiscovery->getLibraryByName($extension, $name);
+      if (isset($definition['drupalSettings'])) {
+        $settings = NestedArray::mergeDeepArray([$settings, $definition['drupalSettings']], TRUE);
+      }
+    }
+
+    // Attached settings win over settings in libraries.
+    $settings = NestedArray::mergeDeepArray([$settings, $assets->getSettings()], TRUE);
+
+    // Allow modules and themes to alter the JavaScript settings.
+    $this->moduleHandler->alter('js_settings', $settings, $assets);
+    $this->themeManager->alter('js_settings', $settings, $assets);
+
+    return $settings;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getJsAssets(AttachedAssetsInterface $assets, $optimize) {
+    $javascript = [];
+
+    foreach ($this->getLibrariesToLoad($assets) as $library) {
+      list($extension, $name) = explode('/', $library, 2);
+      $definition = $this->libraryDiscovery->getLibraryByName($extension, $name);
+      if (isset($definition['js'])) {
+        foreach ($definition['js'] as $options) {
+          $options += array(
+            'type' => 'file',
+            'group' => JS_DEFAULT,
+            'every_page' => FALSE,
+            'weight' => 0,
+            'scope' => 'header',
+            'cache' => TRUE,
+            'preprocess' => TRUE,
+            'attributes' => array(),
+            'version' => NULL,
+            'browsers' => array(),
+          );
+
+          // Preprocess can only be set if caching is enabled and no attributes
+          // are set.
+          $options['preprocess'] = $options['cache'] && empty($options['attributes']) ? $options['preprocess'] : FALSE;
+
+          // Always add a tiny value to the weight, to conserve the insertion
+          // order.
+          $options['weight'] += count($javascript) / 1000;
+
+          // Local and external files must keep their name as the associative
+          // key so the same JavaScript file is not added twice.
+          $javascript[$options['data']] = $options;
+        }
+      }
+    }
+
+    // Allow modules and themes to alter the JavaScript assets.
+    $this->moduleHandler->alter('js', $javascript, $assets);
+    $this->themeManager->alter('js', $javascript, $assets);
+
+    // Sort JavaScript assets, so that they appear in the correct order.
+    uasort($javascript, 'static::sort');
+
+    // Prepare the return value: filter JavaScript assets per scope.
+    $js_assets_header = [];
+    $js_assets_footer = [];
+    foreach ($javascript as $key => $item) {
+      if ($item['scope'] == 'header') {
+        $js_assets_header[$key] = $item;
+      }
+      elseif ($item['scope'] == 'footer') {
+        $js_assets_footer[$key] = $item;
+      }
+    }
+
+    // @todo Refactor this when the default scope is changed to 'footer' in
+    //    https://www.drupal.org/node/784626
+    // If the core/drupalSettings library is being loaded or is already loaded,
+    // get the JavaScript settings assets, and convert them into a single
+    // "regular" JavaScript asset.
+    $libraries_to_load = $this->getLibrariesToLoad($assets);
+    $settings_needed = in_array('core/drupalSettings', $libraries_to_load) || in_array('core/drupalSettings', $this->libraryDependencyResolver->getLibrariesWithDependencies($assets->getAlreadyLoadedLibraries()));
+    $settings_have_changed = count($libraries_to_load) > 0 || count($assets->getSettings()) > 0;
+    if ($settings_needed && $settings_have_changed) {
+      $settings = $this->getJsSettingsAssets($assets);
+      if (!empty($settings)) {
+        // Prepend to the list of JavaScript assets, to render it first.
+        $settings_as_inline_javascript = [
+          'type' => 'setting',
+          'group' => JS_SETTING,
+          'every_page' => TRUE,
+          'weight' => 0,
+          'browsers' => array(),
+          'data' => $settings,
+        ];
+        $js_assets_header = ['drupalSettings' => $settings_as_inline_javascript] + $js_assets_header;
+      }
+    }
+
+    return [
+      $js_assets_header,
+      $js_assets_footer,
+    ];
+  }
+
+  /**
+   * Sorts CSS and JavaScript resources.
+   *
+   * This sort order helps optimize front-end performance while providing
+   * modules and themes with the necessary control for ordering the CSS and
+   * JavaScript appearing on a page.
+   *
+   * @param $a
+   *   First item for comparison. The compared items should be associative
+   *   arrays of member items.
+   * @param $b
+   *   Second item for comparison.
+   *
+   * @return int
+   */
+  public static function sort($a, $b) {
+    // First order by group, so that all items in the CSS_AGGREGATE_DEFAULT
+    // group appear before items in the CSS_AGGREGATE_THEME group. Modules may
+    // create additional groups by defining their own constants.
+    if ($a['group'] < $b['group']) {
+      return -1;
+    }
+    elseif ($a['group'] > $b['group']) {
+      return 1;
+    }
+    // Within a group, order all infrequently needed, page-specific files after
+    // common files needed throughout the website. Separating this way allows
+    // for the aggregate file generated for all of the common files to be reused
+    // across a site visit without being cut by a page using a less common file.
+    elseif ($a['every_page'] && !$b['every_page']) {
+      return -1;
+    }
+    elseif (!$a['every_page'] && $b['every_page']) {
+      return 1;
+    }
+    // Finally, order by weight.
+    elseif ($a['weight'] < $b['weight']) {
+      return -1;
+    }
+    elseif ($a['weight'] > $b['weight']) {
+      return 1;
+    }
+    else {
+      return 0;
+    }
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Asset/AssetResolverInterface.php b/core/lib/Drupal/Core/Asset/AssetResolverInterface.php
new file mode 100644
index 000000000000..6857c87ffc01
--- /dev/null
+++ b/core/lib/Drupal/Core/Asset/AssetResolverInterface.php
@@ -0,0 +1,85 @@
+<?php
+/**
+ * @file
+ * Contains \Drupal\Core\Asset\AssetResolverInterface.
+ */
+
+namespace Drupal\Core\Asset;
+
+/**
+ * Resolves asset libraries into concrete CSS and JavaScript assets.
+ *
+ * Given an attached assets collection (to be loaded for the current response),
+ * the asset resolver can resolve those asset libraries into a list of concrete
+ * CSS and JavaScript assets.
+ *
+ * In other words: this allows developers to translate Drupal's asset
+ * abstraction (asset libraries) into concrete assets.
+ *
+ * @see \Drupal\Core\Asset\AttachedAssetsInterface
+ * @see \Drupal\Core\Asset\LibraryDependencyResolverInterface
+ */
+interface AssetResolverInterface {
+
+  /**
+   * Returns the CSS assets for the current response's libraries.
+   *
+   * It returns the CSS assets in order, according to the SMACSS categories
+   * specified in the assets' weights:
+   * 1. CSS_BASE
+   * 2. CSS_LAYOUT
+   * 3. CSS_COMPONENT
+   * 4. CSS_STATE
+   * 5. CSS_THEME
+   * @see https://www.drupal.org/node/1887918#separate-concerns
+   * This ensures proper cascading of styles so themes can easily override
+   * module styles through CSS selectors.
+   *
+   * Themes may replace module-defined CSS files by adding a stylesheet with the
+   * same filename. For example, themes/bartik/system-menus.css would replace
+   * modules/system/system-menus.css. This allows themes to override complete
+   * CSS files, rather than specific selectors, when necessary.
+   *
+   * Also invokes hook_css_alter(), to allow CSS assets to be altered.
+   *
+   * @param \Drupal\Core\Asset\AttachedAssetsInterface $assets
+   *   The assets attached to the current response.
+   * @param bool $optimize
+   *   Whether to apply the CSS asset collection optimizer, to return an
+   *   optimized CSS asset collection rather than an unoptimized one.
+   *
+   * @return array
+   *   A (possibly optimized) collection of CSS assets.
+   */
+  public function getCssAssets(AttachedAssetsInterface $assets, $optimize);
+
+  /**
+   * Returns the JavaScript assets for the current response's libraries.
+   *
+   * References to JavaScript files are placed in a certain order: first, all
+   * 'core' files, then all 'module' and finally all 'theme' JavaScript files
+   * are added to the page. Then, all settings are output, followed by 'inline'
+   * JavaScript code. If running update.php, all preprocessing is disabled.
+   *
+   * Note that hook_js_alter(&$javascript) is called during this function call
+   * to allow alterations of the JavaScript during its presentation. The correct
+   * way to add JavaScript during hook_js_alter() is to add another element to
+   * the $javascript array, deriving from drupal_js_defaults(). See
+   * locale_js_alter() for an example of this.
+   *
+   * @param \Drupal\Core\Asset\AttachedAssetsInterface $assets
+   *   The assets attached to the current response.
+   * @param bool $optimize
+   *   Whether to apply the JavaScript asset collection optimizer, to return
+   *   optimized JavaScript asset collections rather than an unoptimized ones.
+   *
+   * @return array
+   *   A nested array containing 2 values:
+   *   - at index zero: the (possibly optimized) collection of JavaScript assets
+   *     for the top of the page
+   *   - at index one: the (possibly optimized) collection of JavaScript assets
+   *     for the bottom of the page
+   */
+  public function getJsAssets(AttachedAssetsInterface $assets, $optimize);
+
+}
diff --git a/core/lib/Drupal/Core/Asset/AttachedAssets.php b/core/lib/Drupal/Core/Asset/AttachedAssets.php
new file mode 100644
index 000000000000..d94618e9898d
--- /dev/null
+++ b/core/lib/Drupal/Core/Asset/AttachedAssets.php
@@ -0,0 +1,98 @@
+<?php
+/**
+ * @file
+ * Contains \Drupal\Core\Asset\AttachedAssets.
+ */
+
+namespace Drupal\Core\Asset;
+
+/**
+ * The default attached assets collection.
+ */
+class AttachedAssets implements AttachedAssetsInterface {
+
+  /**
+   * The (ordered) list of asset libraries attached to the current response.
+   *
+   * @var string[]
+   */
+  public $libraries = [];
+
+  /**
+   * The JavaScript settings attached to the current response.
+   *
+   * @var array
+   */
+  public $settings = [];
+
+  /**
+   * The set of asset libraries that the client has already loaded.
+   *
+   * @var string[]
+   */
+  protected $alreadyLoadedLibraries = [];
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function createFromRenderArray(array $render_array) {
+    if (!isset($render_array['#attached'])) {
+      throw new \LogicException('The render array has not yet been rendered, hence not all attachments have been collected yet.');
+    }
+
+    $assets = new static();
+    if (isset($render_array['#attached']['library'])) {
+      $assets->setLibraries($render_array['#attached']['library']);
+    }
+    if (isset($render_array['#attached']['drupalSettings'])) {
+      $assets->setSettings($render_array['#attached']['drupalSettings']);
+    }
+    return $assets;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function setLibraries(array $libraries) {
+    $this->libraries = array_unique($libraries);
+    return $this;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getLibraries() {
+    return $this->libraries;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function setSettings(array $settings) {
+    $this->settings = $settings;
+    return $this;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getSettings() {
+    return $this->settings;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getAlreadyLoadedLibraries() {
+    return $this->alreadyLoadedLibraries;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function setAlreadyLoadedLibraries(array $libraries) {
+    $this->alreadyLoadedLibraries = $libraries;
+    return $this;
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Asset/AttachedAssetsInterface.php b/core/lib/Drupal/Core/Asset/AttachedAssetsInterface.php
new file mode 100644
index 000000000000..956cf6210417
--- /dev/null
+++ b/core/lib/Drupal/Core/Asset/AttachedAssetsInterface.php
@@ -0,0 +1,85 @@
+<?php
+/**
+ * @file
+ * Contains \Drupal\Core\Asset\AttachedAssetsInterface.
+ */
+
+namespace Drupal\Core\Asset;
+
+/**
+ * The attached assets collection for the current response.
+ *
+ * Allows for storage of:
+ * - an ordered list of asset libraries (to be loaded for the current response)
+ * - attached JavaScript settings (to be loaded for the current response)
+ * - a set of asset libraries that the client already has loaded (as indicated
+ *   in the request, to *not* be loaded for the current response)
+ *
+ * @see \Drupal\Core\Asset\AssetResolverInterface
+ */
+interface AttachedAssetsInterface {
+
+  /**
+   * Creates an AttachedAssetsInterface object from a render array.
+   *
+   * @param array $render_array
+   *   A render array.
+   *
+   * @return \Drupal\Core\Asset\AttachedAssetsInterface
+   *
+   * @throws \LogicException
+   */
+  public static function createFromRenderArray(array $render_array);
+
+  /**
+   * Sets the asset libraries attached to the current response.
+   *
+   * @param string[] $libraries
+   *   A list of libraries, in the order they should be loaded.
+   *
+   * @return $this
+   */
+  public function setLibraries(array $libraries);
+
+  /**
+   * Returns the asset libraries attached to the current response.
+   *
+   * @return string[]
+   */
+  public function getLibraries();
+
+  /**
+   * Sets the JavaScript settings that are attached to the current response.
+   *
+   * @param array $settings
+   *   The needed JavaScript settings.
+   *
+   * @return $this
+   */
+  public function setSettings(array $settings);
+
+  /**
+   * Returns the settings attached to the current response.
+   *
+   * @return array
+   */
+  public function getSettings();
+
+  /**
+   * Sets the asset libraries that the current request marked as already loaded.
+   *
+   * @param string[] $libraries
+   *   The set of already loaded libraries.
+   *
+   * @return $this
+   */
+  public function setAlreadyLoadedLibraries(array $libraries);
+
+  /**
+   * Returns the set of already loaded asset libraries.
+   *
+   * @return string[]
+   */
+  public function getAlreadyLoadedLibraries();
+
+}
diff --git a/core/lib/Drupal/Core/Asset/CssCollectionGrouper.php b/core/lib/Drupal/Core/Asset/CssCollectionGrouper.php
index a8f4355698a9..2f727e6f3cd0 100644
--- a/core/lib/Drupal/Core/Asset/CssCollectionGrouper.php
+++ b/core/lib/Drupal/Core/Asset/CssCollectionGrouper.php
@@ -58,7 +58,7 @@ public function group(array $css_assets) {
           // Group file items if their 'preprocess' flag is TRUE.
           // Help ensure maximum reuse of aggregate files by only grouping
           // together items that share the same 'group' value and 'every_page'
-          // flag. See _drupal_add_css() for details about that.
+          // flag.
           $group_keys = $item['preprocess'] ? array($item['type'], $item['group'], $item['every_page'], $item['media'], $item['browsers']) : FALSE;
           break;
 
diff --git a/core/lib/Drupal/Core/Asset/CssCollectionRenderer.php b/core/lib/Drupal/Core/Asset/CssCollectionRenderer.php
index d00fcb40f6eb..6d851d5f440d 100644
--- a/core/lib/Drupal/Core/Asset/CssCollectionRenderer.php
+++ b/core/lib/Drupal/Core/Asset/CssCollectionRenderer.php
@@ -11,6 +11,49 @@
 
 /**
  * Renders CSS assets.
+ *
+ * For production websites, LINK tags are preferable to STYLE tags with @import
+ * statements, because:
+ * - They are the standard tag intended for linking to a resource.
+ * - On Firefox 2 and perhaps other browsers, CSS files included with @import
+ *   statements don't get saved when saving the complete web page for offline
+ *   use: http://drupal.org/node/145218.
+ * - On IE, if only LINK tags and no @import statements are used, all the CSS
+ *   files are downloaded in parallel, resulting in faster page load, but if
+ *   @import statements are used and span across multiple STYLE tags, all the
+ *   ones from one STYLE tag must be downloaded before downloading begins for
+ *   the next STYLE tag. Furthermore, IE7 does not support media declaration on
+ *   the @import statement, so multiple STYLE tags must be used when different
+ *   files are for different media types. Non-IE browsers always download in
+ *   parallel, so this is an IE-specific performance quirk:
+ *   http://www.stevesouders.com/blog/2009/04/09/dont-use-import/.
+ *
+ * However, IE has an annoying limit of 31 total CSS inclusion tags
+ * (http://drupal.org/node/228818) and LINK tags are limited to one file per
+ * tag, whereas STYLE tags can contain multiple @import statements allowing
+ * multiple files to be loaded per tag. When CSS aggregation is disabled, a
+ * Drupal site can easily have more than 31 CSS files that need to be loaded, so
+ * using LINK tags exclusively would result in a site that would display
+ * incorrectly in IE. Depending on different needs, different strategies can be
+ * employed to decide when to use LINK tags and when to use STYLE tags.
+ *
+ * The strategy employed by this class is to use LINK tags for all aggregate
+ * files and for all files that cannot be aggregated (e.g., if 'preprocess' is
+ * set to FALSE or the type is 'external'), and to use STYLE tags for groups
+ * of files that could be aggregated together but aren't (e.g., if the site-wide
+ * aggregation setting is disabled). This results in all LINK tags when
+ * aggregation is enabled, a guarantee that as many or only slightly more tags
+ * are used with aggregation disabled than enabled (so that if the limit were to
+ * be crossed with aggregation enabled, the site developer would also notice the
+ * problem while aggregation is disabled), and an easy way for a developer to
+ * view HTML source while aggregation is disabled and know what files will be
+ * aggregated together when aggregation becomes enabled.
+ *
+ * This class evaluates the aggregation enabled/disabled condition on a group
+ * by group basis by testing whether an aggregate file has been made for the
+ * group rather than by testing the site-wide aggregation setting. This allows
+ * this class to work correctly even if modules have implemented custom
+ * logic for grouping and aggregating files.
  */
 class CssCollectionRenderer implements AssetCollectionRendererInterface {
 
diff --git a/core/lib/Drupal/Core/Asset/JsCollectionGrouper.php b/core/lib/Drupal/Core/Asset/JsCollectionGrouper.php
index 3547b5120e1e..33d27be23df3 100644
--- a/core/lib/Drupal/Core/Asset/JsCollectionGrouper.php
+++ b/core/lib/Drupal/Core/Asset/JsCollectionGrouper.php
@@ -45,7 +45,7 @@ public function group(array $js_assets) {
           // Group file items if their 'preprocess' flag is TRUE.
           // Help ensure maximum reuse of aggregate files by only grouping
           // together items that share the same 'group' value and 'every_page'
-          // flag. See _drupal_add_js() for details about that.
+          // flag.
           $group_keys = $item['preprocess'] ? array($item['type'], $item['group'], $item['every_page'], $item['browsers']) : FALSE;
           break;
 
diff --git a/core/lib/Drupal/Core/Asset/JsCollectionRenderer.php b/core/lib/Drupal/Core/Asset/JsCollectionRenderer.php
index 52e6fe1cd197..7db84abe3a86 100644
--- a/core/lib/Drupal/Core/Asset/JsCollectionRenderer.php
+++ b/core/lib/Drupal/Core/Asset/JsCollectionRenderer.php
@@ -22,7 +22,7 @@ class JsCollectionRenderer implements AssetCollectionRendererInterface {
   protected $state;
 
   /**
-   * Constructs a CssCollectionRenderer.
+   * Constructs a JsCollectionRenderer.
    *
    * @param \Drupal\Core\State\StateInterface
    *   The state key/value store.
@@ -33,6 +33,12 @@ public function __construct(StateInterface $state) {
 
   /**
    * {@inheritdoc}
+   *
+   * This class evaluates the aggregation enabled/disabled condition on a group
+   * by group basis by testing whether an aggregate file has been made for the
+   * group rather than by testing the site-wide aggregation setting. This allows
+   * this class to work correctly even if modules have implemented custom
+   * logic for grouping and aggregating files.
    */
   public function render(array $js_assets) {
     $elements = array();
@@ -40,9 +46,8 @@ public function render(array $js_assets) {
     // A dummy query-string is added to filenames, to gain control over
     // browser-caching. The string changes on every update or full cache
     // flush, forcing browsers to load a new copy of the files, as the
-    // URL changed. Files that should not be cached (see _drupal_add_js())
-    // get REQUEST_TIME as query-string instead, to enforce reload on every
-    // page request.
+    // URL changed. Files that should not be cached get REQUEST_TIME as
+    // query-string instead, to enforce reload on every page request.
     $default_query_string = $this->state->get('system.css_js_query_string') ?: '0';
 
     // For inline JavaScript to validate as XHTML, all JavaScript containing
diff --git a/core/lib/Drupal/Core/Asset/LibraryDependencyResolver.php b/core/lib/Drupal/Core/Asset/LibraryDependencyResolver.php
new file mode 100644
index 000000000000..34ffc0def8d0
--- /dev/null
+++ b/core/lib/Drupal/Core/Asset/LibraryDependencyResolver.php
@@ -0,0 +1,100 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Asset\LibraryDependencyResolver.
+ */
+
+namespace Drupal\Core\Asset;
+
+/**
+ * Resolves the dependencies of asset (CSS/JavaScript) libraries.
+ */
+class LibraryDependencyResolver implements LibraryDependencyResolverInterface {
+
+  /**
+   * The library discovery service.
+   *
+   * @var \Drupal\Core\Asset\LibraryDiscoveryInterface
+   */
+  protected $libraryDiscovery;
+
+  /**
+   * Constructs a new LibraryDependencyResolver instance.
+   *
+   * @param \Drupal\Core\Asset\LibraryDiscoveryInterface $library_discovery
+   *   The library discovery service.
+   */
+  public function __construct(LibraryDiscoveryInterface $library_discovery) {
+    $this->libraryDiscovery = $library_discovery;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getLibrariesWithDependencies(array $libraries) {
+    return $this->doGetDependencies($libraries);
+  }
+
+  /**
+   * Gets the given libraries with its dependencies.
+   *
+   * Helper method for ::getLibrariesWithDependencies().
+   *
+   * @param string[] $libraries_with_unresolved_dependencies
+   *   A list of libraries, with unresolved dependencies, in the order they
+   *   should be loaded.
+   * @param string[] $final_libraries
+   *   The final list of libraries (the return value) that is being built
+   *   recursively.
+   *
+   * @return string[]
+   *   A list of libraries, in the order they should be loaded, including their
+   *   dependencies.
+   */
+  protected function doGetDependencies(array $libraries_with_unresolved_dependencies, array $final_libraries = []) {
+    foreach ($libraries_with_unresolved_dependencies as $library) {
+      if (!in_array($library, $final_libraries)) {
+        list($extension, $name) = explode('/', $library, 2);
+        $definition = $this->libraryDiscovery->getLibraryByName($extension, $name);
+        if (!empty($definition['dependencies'])) {
+          $final_libraries = $this->doGetDependencies($definition['dependencies'], $final_libraries);
+        }
+        $final_libraries[] = $library;
+      }
+    }
+    return $final_libraries;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getMinimalRepresentativeSubset(array $libraries) {
+    $minimal = [];
+
+    // Determine each library's dependencies.
+    $with_deps = [];
+    foreach ($libraries as $library) {
+      $with_deps[$library] = $this->getLibrariesWithDependencies([$library]);
+    }
+
+    foreach ($libraries as $library) {
+      $exists = FALSE;
+      foreach ($with_deps as $other_library => $dependencies) {
+        if ($library == $other_library) {
+          continue;
+        }
+        if (in_array($library, $dependencies)) {
+          $exists = TRUE;
+          break;
+        }
+      }
+      if (!$exists) {
+        $minimal[] = $library;
+      }
+    }
+
+    return $minimal;
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Asset/LibraryDependencyResolverInterface.php b/core/lib/Drupal/Core/Asset/LibraryDependencyResolverInterface.php
new file mode 100644
index 000000000000..66362cc3d8e2
--- /dev/null
+++ b/core/lib/Drupal/Core/Asset/LibraryDependencyResolverInterface.php
@@ -0,0 +1,51 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Asset\LibraryDependencyResolverInterface.
+ */
+
+namespace Drupal\Core\Asset;
+
+/**
+ * Resolves the dependencies of asset (CSS/JavaScript) libraries.
+ */
+interface LibraryDependencyResolverInterface {
+
+  /**
+   * Gets the given libraries with their dependencies.
+   *
+   * Given ['core/a', 'core/b', 'core/c'], with core/a depending on core/c and
+   * core/b on core/d, returns ['core/a', 'core/b', 'core/c', 'core/d'].
+   *
+   * @param string[] $libraries
+   *   A list of libraries, in the order they should be loaded.
+   *
+   * @return string[]
+   *   A list of libraries, in the order they should be loaded, including their
+   *   dependencies.
+   */
+  public function getLibrariesWithDependencies(array $libraries);
+
+  /**
+   * Gets the minimal representative subset of the given libraries.
+   *
+   * A minimal representative subset means that any library in the given set of
+   * libraries that is a dependency of another library in the set, is removed.
+   *
+   * Hence a minimal representative subset is the most compact representation
+   * possible of a set of libraries.
+   *
+   * (Each asset library has dependencies and can therefore be seen as a tree.
+   * Hence the given list of libraries represent a forest. This function returns
+   * all roots of trees that are not a subtree of another tree in the forest.)
+   *
+   * @param string[] $libraries
+   *   A set of libraries.
+   *
+   * @return string[]
+   *   A representative subset of the given set of libraries.
+   */
+  public function getMinimalRepresentativeSubset(array $libraries);
+
+}
diff --git a/core/lib/Drupal/Core/Render/Element/Scripts.php b/core/lib/Drupal/Core/Render/Element/Scripts.php
deleted file mode 100644
index cc04edec76cb..000000000000
--- a/core/lib/Drupal/Core/Render/Element/Scripts.php
+++ /dev/null
@@ -1,67 +0,0 @@
-<?php
-
-/**
- * @file
- * Contains \Drupal\Core\Render\Element\Scripts.
- */
-
-namespace Drupal\Core\Render\Element;
-
-/**
- * Provides a render element for adding JavaScript to the HTML output.
- *
- * @see \Drupal\Core\Render\Element\Styles
- *
- * @RenderElement("scripts")
- */
-class Scripts extends RenderElement {
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getInfo() {
-    $class = get_class($this);
-    return array(
-      '#items' => array(),
-      '#pre_render' => array(
-        array($class, 'preRenderScripts'),
-      ),
-    );
-  }
-
-  /**
-   * #pre_render callback to add the elements needed for JavaScript tags to be rendered.
-   *
-   * This function evaluates the aggregation enabled/disabled condition on a group
-   * by group basis by testing whether an aggregate file has been made for the
-   * group rather than by testing the site-wide aggregation setting. This allows
-   * this function to work correctly even if modules have implemented custom
-   * logic for grouping and aggregating files.
-   *
-   * @param array $element
-   *   A render array containing:
-   *   - #items: The JavaScript items as returned by _drupal_add_js() and
-   *     altered by drupal_get_js().
-   *   - #group_callback: A function to call to group #items. Following
-   *     this function, #aggregate_callback is called to aggregate items within
-   *     the same group into a single file.
-   *   - #aggregate_callback: A function to call to aggregate the items within
-   *     the groups arranged by the #group_callback function.
-   *
-   * @return array
-   *   A render array that will render to a string of JavaScript tags.
-   *
-   * @see drupal_get_js()
-   */
-  public static function preRenderScripts($element) {
-    $js_assets = $element['#items'];
-
-    // Aggregate the JavaScript if necessary, but only during normal site
-    // operation.
-    if (!defined('MAINTENANCE_MODE') && \Drupal::config('system.performance')->get('js.preprocess')) {
-      $js_assets = \Drupal::service('asset.js.collection_optimizer')->optimize($js_assets);
-    }
-    return \Drupal::service('asset.js.collection_renderer')->render($js_assets);
-  }
-
-}
diff --git a/core/lib/Drupal/Core/Render/Element/Styles.php b/core/lib/Drupal/Core/Render/Element/Styles.php
deleted file mode 100644
index d285fac2243d..000000000000
--- a/core/lib/Drupal/Core/Render/Element/Styles.php
+++ /dev/null
@@ -1,98 +0,0 @@
-<?php
-
-/**
- * @file
- * Contains \Drupal\Core\Render\Element\Styles.
- */
-
-namespace Drupal\Core\Render\Element;
-
-/**
- * Provides a render element for adding CSS to the HTML output.
- *
- * @see \Drupal\Core\Render\Element\Scripts
- *
- * @RenderElement("styles")
- */
-class Styles extends RenderElement {
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getInfo() {
-    $class = get_class($this);
-    return array(
-      '#items' => array(),
-      '#pre_render' => array(
-        array($class, 'preRenderStyles'),
-      ),
-    );
-  }
-
-  /**
-   * Pre-render callback: Adds the elements needed for CSS tags to be rendered.
-   *
-   * For production websites, LINK tags are preferable to STYLE tags with @import
-   * statements, because:
-   * - They are the standard tag intended for linking to a resource.
-   * - On Firefox 2 and perhaps other browsers, CSS files included with @import
-   *   statements don't get saved when saving the complete web page for offline
-   *   use: http://drupal.org/node/145218.
-   * - On IE, if only LINK tags and no @import statements are used, all the CSS
-   *   files are downloaded in parallel, resulting in faster page load, but if
-   *   @import statements are used and span across multiple STYLE tags, all the
-   *   ones from one STYLE tag must be downloaded before downloading begins for
-   *   the next STYLE tag. Furthermore, IE7 does not support media declaration on
-   *   the @import statement, so multiple STYLE tags must be used when different
-   *   files are for different media types. Non-IE browsers always download in
-   *   parallel, so this is an IE-specific performance quirk:
-   *   http://www.stevesouders.com/blog/2009/04/09/dont-use-import/.
-   *
-   * However, IE has an annoying limit of 31 total CSS inclusion tags
-   * (http://drupal.org/node/228818) and LINK tags are limited to one file per
-   * tag, whereas STYLE tags can contain multiple @import statements allowing
-   * multiple files to be loaded per tag. When CSS aggregation is disabled, a
-   * Drupal site can easily have more than 31 CSS files that need to be loaded, so
-   * using LINK tags exclusively would result in a site that would display
-   * incorrectly in IE. Depending on different needs, different strategies can be
-   * employed to decide when to use LINK tags and when to use STYLE tags.
-   *
-   * The strategy employed by this function is to use LINK tags for all aggregate
-   * files and for all files that cannot be aggregated (e.g., if 'preprocess' is
-   * set to FALSE or the type is 'external'), and to use STYLE tags for groups
-   * of files that could be aggregated together but aren't (e.g., if the site-wide
-   * aggregation setting is disabled). This results in all LINK tags when
-   * aggregation is enabled, a guarantee that as many or only slightly more tags
-   * are used with aggregation disabled than enabled (so that if the limit were to
-   * be crossed with aggregation enabled, the site developer would also notice the
-   * problem while aggregation is disabled), and an easy way for a developer to
-   * view HTML source while aggregation is disabled and know what files will be
-   * aggregated together when aggregation becomes enabled.
-   *
-   * This function evaluates the aggregation enabled/disabled condition on a group
-   * by group basis by testing whether an aggregate file has been made for the
-   * group rather than by testing the site-wide aggregation setting. This allows
-   * this function to work correctly even if modules have implemented custom
-   * logic for grouping and aggregating files.
-   *
-   * @param array $element
-   *   A render array containing:
-   *   - '#items': The CSS items as returned by _drupal_add_css() and altered by
-   *     drupal_get_css().
-   *
-   * @return array
-   *   A render array that will render to a string of XHTML CSS tags.
-   *
-   * @see drupal_get_css()
-   */
-  public static function preRenderStyles($element) {
-    $css_assets = $element['#items'];
-
-    // Aggregate the CSS if necessary, but only during normal site operation.
-    if (!defined('MAINTENANCE_MODE') && \Drupal::config('system.performance')->get('css.preprocess')) {
-      $css_assets = \Drupal::service('asset.css.collection_optimizer')->optimize($css_assets);
-    }
-    return \Drupal::service('asset.css.collection_renderer')->render($css_assets);
-  }
-
-}
diff --git a/core/lib/Drupal/Core/Render/MainContent/AjaxRenderer.php b/core/lib/Drupal/Core/Render/MainContent/AjaxRenderer.php
index 766dc2dfd96a..6cddc0ceac85 100644
--- a/core/lib/Drupal/Core/Render/MainContent/AjaxRenderer.php
+++ b/core/lib/Drupal/Core/Render/MainContent/AjaxRenderer.php
@@ -65,6 +65,7 @@ public function renderResponse(array $main_content, Request $request, RouteMatch
     }
 
     $html = $this->drupalRenderRoot($main_content);
+    $response->setAttachments($main_content['#attached']);
 
     // The selector for the insert command is NULL as the new content will
     // replace the element making the Ajax call. The default 'replaceWith'
@@ -84,9 +85,7 @@ public function renderResponse(array $main_content, Request $request, RouteMatch
    * @todo: Remove as part of https://drupal.org/node/2182149
    */
   protected function drupalRenderRoot(&$elements) {
-    $output = drupal_render_root($elements);
-    drupal_process_attached($elements);
-    return $output;
+    return drupal_render_root($elements);
   }
 
 }
diff --git a/core/lib/Drupal/Core/Render/MainContent/DialogRenderer.php b/core/lib/Drupal/Core/Render/MainContent/DialogRenderer.php
index 9f599440ee03..7af9d93df1da 100644
--- a/core/lib/Drupal/Core/Render/MainContent/DialogRenderer.php
+++ b/core/lib/Drupal/Core/Render/MainContent/DialogRenderer.php
@@ -43,7 +43,11 @@ public function renderResponse(array $main_content, Request $request, RouteMatch
 
     // First render the main content, because it might provide a title.
     $content = drupal_render_root($main_content);
-    drupal_process_attached($main_content);
+
+    // Attach the library necessary for using the OpenDialogCommand and set the
+    // attachments for this Ajax response.
+    $main_content['#attached']['library'][] = 'core/drupal.dialog.ajax';
+    $response->setAttachments($main_content['#attached']);
 
     // Determine the title: use the title provided by the main content if any,
     // otherwise get it from the routing information.
diff --git a/core/lib/Drupal/Core/Render/MainContent/ModalRenderer.php b/core/lib/Drupal/Core/Render/MainContent/ModalRenderer.php
index c971eb3ad526..945a76c6778d 100644
--- a/core/lib/Drupal/Core/Render/MainContent/ModalRenderer.php
+++ b/core/lib/Drupal/Core/Render/MainContent/ModalRenderer.php
@@ -26,7 +26,11 @@ public function renderResponse(array $main_content, Request $request, RouteMatch
 
     // First render the main content, because it might provide a title.
     $content = drupal_render_root($main_content);
-    drupal_process_attached($main_content);
+
+    // Attach the library necessary for using the OpenModalDialogCommand and set
+    // the attachments for this Ajax response.
+    $main_content['#attached']['library'][] = 'core/drupal.dialog.ajax';
+    $response->setAttachments($main_content['#attached']);
 
     // If the main content doesn't provide a title, use the title resolver.
     $title = isset($main_content['#title']) ? $main_content['#title'] : $this->titleResolver->getTitle($request, $route_match->getRouteObject());
diff --git a/core/misc/ajax.js b/core/misc/ajax.js
index 40ab702057f9..ac902824a472 100644
--- a/core/misc/ajax.js
+++ b/core/misc/ajax.js
@@ -384,21 +384,12 @@
     // Allow Drupal to return new JavaScript and CSS files to load without
     // returning the ones already loaded.
     // @see \Drupal\Core\Theme\AjaxBasePageNegotiator
-    // @see drupal_get_css()
-    // @see drupal_get_js()
+    // @see \Drupal\Core\Asset\LibraryDependencyResolverInterface::getMinimalRepresentativeSubset()
+    // @see system_js_settings_alter()
     var pageState = drupalSettings.ajaxPageState;
     options.data['ajax_page_state[theme]'] = pageState.theme;
     options.data['ajax_page_state[theme_token]'] = pageState.theme_token;
-    for (var cssFile in pageState.css) {
-      if (pageState.css.hasOwnProperty(cssFile)) {
-        options.data['ajax_page_state[css][' + cssFile + ']'] = 1;
-      }
-    }
-    for (var jsFile in pageState.js) {
-      if (pageState.js.hasOwnProperty(jsFile)) {
-        options.data['ajax_page_state[js][' + jsFile + ']'] = 1;
-      }
-    }
+    options.data['ajax_page_state[libraries]'] = pageState.libraries;
   };
 
   /**
diff --git a/core/modules/ckeditor/src/Tests/CKEditorLoadingTest.php b/core/modules/ckeditor/src/Tests/CKEditorLoadingTest.php
index cddf23fa1145..dd53245e11ca 100644
--- a/core/modules/ckeditor/src/Tests/CKEditorLoadingTest.php
+++ b/core/modules/ckeditor/src/Tests/CKEditorLoadingTest.php
@@ -118,8 +118,7 @@ function testLoading() {
     $this->assertTrue(count($format_selector) === 1, 'A single text format selector exists on the page.');
     $specific_format_selector = $this->xpath('//select[contains(@class, "filter-list") and contains(@class, "editor") and @data-editor-for="edit-body-0-value"]');
     $this->assertTrue(count($specific_format_selector) === 1, 'A single text format selector exists on the page and has the "editor" class and a "data-editor-for" attribute with the correct value.');
-    $this->assertTrue(isset($settings['ajaxPageState']['js']['core/modules/ckeditor/js/ckeditor.js']), 'CKEditor glue JS is present.');
-    $this->assertTrue(isset($settings['ajaxPageState']['js']['core/assets/vendor/ckeditor/ckeditor.js']), 'CKEditor lib JS is present.');
+    $this->assertTrue(in_array('ckeditor/drupal.ckeditor', explode(',', $settings['ajaxPageState']['libraries'])), 'CKEditor glue library is present.');
 
     // Enable the ckeditor_test module, customize configuration. In this case,
     // there is additional CSS and JS to be loaded.
@@ -146,8 +145,7 @@ function testLoading() {
     $this->assertTrue($editor_settings_present, "Text Editor module's JavaScript settings are on the page.");
     $this->assertIdentical($expected, $settings['editor'], "Text Editor module's JavaScript settings on the page are correct.");
     $this->assertTrue($editor_js_present, 'Text Editor JavaScript is present.');
-    $this->assertTrue(isset($settings['ajaxPageState']['js']['core/modules/ckeditor/js/ckeditor.js']), 'CKEditor glue JS is present.');
-    $this->assertTrue(isset($settings['ajaxPageState']['js']['core/assets/vendor/ckeditor/ckeditor.js']), 'CKEditor lib JS is present.');
+    $this->assertTrue(in_array('ckeditor/drupal.ckeditor', explode(',', $settings['ajaxPageState']['libraries'])), 'CKEditor glue library is present.');
   }
 
   protected function getThingsToCheck() {
@@ -157,8 +155,10 @@ protected function getThingsToCheck() {
       $settings,
       // Editor.module's JS settings present.
       isset($settings['editor']),
-      // Editor.module's JS present.
-      isset($settings['ajaxPageState']['js']['core/modules/editor/js/editor.js']),
+      // Editor.module's JS present. Note: ckeditor/drupal.ckeditor depends on
+      // editor/drupal.editor, hence presence of the former implies presence of
+      // the latter.
+      isset($settings['ajaxPageState']) && in_array('ckeditor/drupal.ckeditor', explode(',', $settings['ajaxPageState']['libraries'])),
       // Body field.
       $this->xpath('//textarea[@id="edit-body-0-value"]'),
       // Format selector.
diff --git a/core/modules/comment/src/Tests/CommentCSSTest.php b/core/modules/comment/src/Tests/CommentCSSTest.php
index f6205515578c..36033d050e07 100644
--- a/core/modules/comment/src/Tests/CommentCSSTest.php
+++ b/core/modules/comment/src/Tests/CommentCSSTest.php
@@ -110,7 +110,7 @@ function testCommentClasses() {
         // user (the viewer) was the author of the comment. We do this in Java-
         // Script to prevent breaking the render cache.
         $this->assertIdentical(1, count($this->xpath('//*[contains(@class, "comment") and @data-comment-user-id="' . $case['comment_uid'] . '"]')), 'data-comment-user-id attribute is set on comment.');
-        $this->assertTrue(isset($settings['ajaxPageState']['js']['core/modules/comment/js/comment-by-viewer.js']), 'drupal.comment-by-viewer library is present.');
+        $this->assertRaw(drupal_get_path('module', 'comment') . '/js/comment-by-viewer.js', 'drupal.comment-by-viewer library is present.');
       }
 
       // Verify the unpublished class.
@@ -129,7 +129,7 @@ function testCommentClasses() {
       if ($case['comment_status'] == CommentInterface::PUBLISHED || $case['user'] == 'admin') {
         $this->assertIdentical(1, count($this->xpath('//*[contains(@class, "comment")]/*[@data-comment-timestamp="' . $comment->getChangedTime() . '"]')), 'data-comment-timestamp attribute is set on comment');
         $expectedJS = ($case['user'] !== 'anonymous');
-        $this->assertIdentical($expectedJS, isset($settings['ajaxPageState']['js']['core/modules/comment/js/comment-new-indicator.js']), 'drupal.comment-new-indicator library is present.');
+        $this->assertIdentical($expectedJS, isset($settings['ajaxPageState']) && in_array('comment/drupal.comment-new-indicator', explode(',', $settings['ajaxPageState']['libraries'])), 'drupal.comment-new-indicator library is present.');
       }
     }
   }
diff --git a/core/modules/editor/editor.api.php b/core/modules/editor/editor.api.php
index 66c5d2a0c166..2fd9fa41b91f 100644
--- a/core/modules/editor/editor.api.php
+++ b/core/modules/editor/editor.api.php
@@ -30,8 +30,8 @@ function hook_editor_info_alter(array &$editors) {
  * Modifies JavaScript settings that are added for text editors.
  *
  * @param array $settings
- *   All the settings that will be added to the page via _drupal_add_js() for
- *   the text formats to which a user has access.
+ *   All the settings that will be added to the page for the text formats to
+ *   which a user has access.
  */
 function hook_editor_js_settings_alter(array &$settings) {
   if (isset($settings['editor']['formats']['basic_html'])) {
diff --git a/core/modules/editor/src/Form/EditorImageDialog.php b/core/modules/editor/src/Form/EditorImageDialog.php
index a493deab5929..80d63fd1c049 100644
--- a/core/modules/editor/src/Form/EditorImageDialog.php
+++ b/core/modules/editor/src/Form/EditorImageDialog.php
@@ -228,6 +228,7 @@ public function submitForm(array &$form, FormStateInterface $form_state) {
       unset($form['#prefix'], $form['#suffix']);
       $status_messages = array('#theme' => 'status_messages');
       $output = drupal_render($form);
+      $response->setAttachments($form['#attached']);
       $output = '<div>' . drupal_render($status_messages) . $output . '</div>';
       $response->addCommand(new HtmlCommand('#editor-image-dialog-form', $output));
     }
diff --git a/core/modules/editor/src/Form/EditorLinkDialog.php b/core/modules/editor/src/Form/EditorLinkDialog.php
index 29ce81215962..39ff4fca1712 100644
--- a/core/modules/editor/src/Form/EditorLinkDialog.php
+++ b/core/modules/editor/src/Form/EditorLinkDialog.php
@@ -87,6 +87,7 @@ public function submitForm(array &$form, FormStateInterface $form_state) {
       unset($form['#prefix'], $form['#suffix']);
       $status_messages = array('#theme' => 'status_messages');
       $output = drupal_render($form);
+      $response->setAttachments($form['#attached']);
       $output = '<div>' . drupal_render($status_messages) . $output . '</div>';
       $response->addCommand(new HtmlCommand('#editor-link-dialog-form', $output));
     }
diff --git a/core/modules/editor/src/Tests/EditorLoadingTest.php b/core/modules/editor/src/Tests/EditorLoadingTest.php
index b619566f2ed1..e693a0ee2912 100644
--- a/core/modules/editor/src/Tests/EditorLoadingTest.php
+++ b/core/modules/editor/src/Tests/EditorLoadingTest.php
@@ -191,7 +191,7 @@ protected function getThingsToCheck() {
       // Editor.module's JS settings present.
       isset($settings['editor']),
       // Editor.module's JS present.
-      isset($settings['ajaxPageState']['js']['core/modules/editor/js/editor.js']),
+      strpos($this->getRawContent(), drupal_get_path('module', 'editor') . '/js/editor.js') !== FALSE,
       // Body field.
       $this->xpath('//textarea[@id="edit-body-0-value"]'),
       // Format selector.
diff --git a/core/modules/file/src/Controller/FileWidgetAjaxController.php b/core/modules/file/src/Controller/FileWidgetAjaxController.php
index 3adc43a58d31..e449caf15913 100644
--- a/core/modules/file/src/Controller/FileWidgetAjaxController.php
+++ b/core/modules/file/src/Controller/FileWidgetAjaxController.php
@@ -79,15 +79,13 @@ public function upload(Request $request) {
     $status_messages = array('#theme' => 'status_messages');
     $form['#prefix'] .= drupal_render($status_messages);
     $output = drupal_render($form);
-    drupal_process_attached($form);
-    $js = _drupal_add_js();
-    $settings = $js['drupalSettings']['data'];
 
     $response = new AjaxResponse();
+    $response->setAttachments($form['#attached']);
     foreach ($commands as $command) {
       $response->addCommand($command, TRUE);
     }
-    return $response->addCommand(new ReplaceCommand(NULL, $output, $settings));
+    return $response->addCommand(new ReplaceCommand(NULL, $output));
   }
 
   /**
diff --git a/core/modules/history/src/Tests/HistoryTest.php b/core/modules/history/src/Tests/HistoryTest.php
index 74e14427853e..a4aa1a5a5763 100644
--- a/core/modules/history/src/Tests/HistoryTest.php
+++ b/core/modules/history/src/Tests/HistoryTest.php
@@ -119,8 +119,8 @@ function testHistory() {
     $this->drupalGet('node/' . $nid);
     // JavaScript present to record the node read.
     $settings = $this->getDrupalSettings();
-    $this->assertTrue(isset($settings['ajaxPageState']['js']['core/modules/history/js/history.js']), 'history/api library is present.');
-    $this->assertTrue(isset($settings['ajaxPageState']['js']['core/modules/history/js/mark-as-read.js']), 'history/mark-as-read library is present.');
+    $libraries = explode(',', $settings['ajaxPageState']['libraries']);
+    $this->assertTrue(in_array('history/mark-as-read', $libraries), 'history/mark-as-read library is present.');
     $this->assertEqual([$nid => TRUE], $settings['history']['nodesToMarkAsRead'], 'drupalSettings to mark node as read are present.');
 
     // Simulate JavaScript: perform HTTP request to mark node as read.
diff --git a/core/modules/locale/locale.module b/core/modules/locale/locale.module
index c7cc9db33e95..3667f93287c3 100644
--- a/core/modules/locale/locale.module
+++ b/core/modules/locale/locale.module
@@ -15,6 +15,7 @@
 use Drupal\Component\Utility\UrlHelper;
 use Drupal\Component\Utility\Xss;
 use Drupal\Core\Url;
+use Drupal\Core\Asset\AttachedAssetsInterface;
 use Drupal\Core\Cache\Cache;
 use Drupal\Core\Form\FormStateInterface;
 use Drupal\Core\Language\Language;
@@ -445,7 +446,7 @@ function locale_cache_flush() {
 /**
  * Implements hook_js_alter().
  */
-function locale_js_alter(&$javascript) {
+function locale_js_alter(&$javascript, AttachedAssetsInterface $assets) {
   $files = array();
   foreach ($javascript as $item) {
     if (isset($item['type']) && $item['type'] == 'file') {
@@ -543,7 +544,7 @@ function locale_library_info_alter(array &$libraries, $module) {
  *
  * Generates the values for the altered core/jquery.ui.datepicker library.
  */
-function locale_js_settings_alter(&$settings) {
+function locale_js_settings_alter(&$settings, AttachedAssetsInterface $assets) {
   if (isset($settings['jquery']['ui']['datepicker'])) {
     $language_interface = \Drupal::languageManager()->getCurrentLanguage();
     $settings['jquery']['ui']['datepicker']['isRTL'] = $language_interface->getDirection() == LanguageInterface::DIRECTION_RTL;
diff --git a/core/modules/locale/src/Tests/LocaleLibraryAlterTest.php b/core/modules/locale/src/Tests/LocaleLibraryAlterTest.php
index 7b04c69f2dc5..5b7d6dfef44a 100644
--- a/core/modules/locale/src/Tests/LocaleLibraryAlterTest.php
+++ b/core/modules/locale/src/Tests/LocaleLibraryAlterTest.php
@@ -6,6 +6,7 @@
 
 namespace Drupal\locale\Tests;
 
+use Drupal\Core\Asset\AttachedAssets;
 use Drupal\simpletest\WebTestBase;
 
 /**
@@ -30,10 +31,9 @@ class LocaleLibraryAlterTest extends WebTestBase {
    * @see locale_library_alter()
    */
   public function testLibraryAlter() {
-    $attached['#attached']['library'][] = 'core/jquery.ui.datepicker';
-    drupal_render($attached);
-    drupal_process_attached($attached);
-    $scripts = drupal_get_js();
-    $this->assertTrue(strpos($scripts, 'locale.datepicker.js'), 'locale.datepicker.js added to scripts.');
+    $assets = new AttachedAssets();
+    $assets->setLibraries(['core/jquery.ui.datepicker']);
+    $js_assets = $this->container->get('asset.resolver')->getJsAssets($assets, FALSE)[0];
+    $this->assertTrue(array_key_exists('core/modules/locale/locale.datepicker.js', $js_assets), 'locale.datepicker.js added to scripts.');
   }
 }
diff --git a/core/modules/menu_link_content/src/Tests/MenuLinkContentUITest.php b/core/modules/menu_link_content/src/Tests/MenuLinkContentUITest.php
index 65404e85142e..a281583b0cec 100644
--- a/core/modules/menu_link_content/src/Tests/MenuLinkContentUITest.php
+++ b/core/modules/menu_link_content/src/Tests/MenuLinkContentUITest.php
@@ -75,9 +75,9 @@ function testTranslationLinkTheme() {
     $edit['admin_theme'] = 'seven';
     $this->drupalPostForm('admin/appearance', $edit, t('Save configuration'));
     $this->drupalGet('admin/structure/menu/item/' . $entityId . '/edit');
-    $this->assertRaw('"theme":"seven"', 'Edit uses admin theme.');
+    $this->assertRaw('core/themes/seven/css/base/elements.css', 'Edit uses admin theme.');
     $this->drupalGet('admin/structure/menu/item/' . $entityId . '/edit/translations');
-    $this->assertRaw('"theme":"seven"', 'Translation uses admin theme as well.');
+    $this->assertRaw('core/themes/seven/css/base/elements.css', 'Translation uses admin theme as well.');
   }
 
 }
diff --git a/core/modules/node/src/Tests/NodeTranslationUITest.php b/core/modules/node/src/Tests/NodeTranslationUITest.php
index c9014efd0e4d..54f913ed10be 100644
--- a/core/modules/node/src/Tests/NodeTranslationUITest.php
+++ b/core/modules/node/src/Tests/NodeTranslationUITest.php
@@ -175,13 +175,13 @@ function testTranslationLinkTheme() {
     $edit['use_admin_theme'] = TRUE;
     $this->drupalPostForm('admin/appearance', $edit, t('Save configuration'));
     $this->drupalGet('node/' . $article->id() . '/translations');
-    $this->assertRaw('"theme":"seven"', 'Translation uses admin theme if edit is admin.');
+    $this->assertRaw('core/themes/seven/css/base/elements.css', 'Translation uses admin theme if edit is admin.');
 
     // Turn off admin theme for editing, assert inheritance to translations.
     $edit['use_admin_theme'] = FALSE;
     $this->drupalPostForm('admin/appearance', $edit, t('Save configuration'));
     $this->drupalGet('node/' . $article->id() . '/translations');
-    $this->assertNoRaw('"theme":"seven"', 'Translation uses frontend theme if edit is frontend.');
+    $this->assertNoRaw('core/themes/seven/css/base/elements.css', 'Translation uses frontend theme if edit is frontend.');
 
     // Assert presence of translation page itself (vs. DisabledBundle below).
     $this->assertResponse(200);
diff --git a/core/modules/quickedit/src/QuickEditController.php b/core/modules/quickedit/src/QuickEditController.php
index 0e6d497e610f..339eab4efc30 100644
--- a/core/modules/quickedit/src/QuickEditController.php
+++ b/core/modules/quickedit/src/QuickEditController.php
@@ -142,8 +142,7 @@ public function attachments(Request $request) {
       throw new NotFoundHttpException();
     }
 
-    $elements['#attached'] = $this->editorSelector->getEditorAttachments($editors);
-    drupal_process_attached($elements);
+    $response->setAttachments($this->editorSelector->getEditorAttachments($editors));
 
     return $response;
   }
@@ -205,7 +204,12 @@ public function fieldForm(EntityInterface $entity, $field_name, $langcode, $view
       $response->addCommand(new FieldFormSavedCommand($output, $other_view_modes));
     }
     else {
-      $response->addCommand(new FieldFormCommand(drupal_render($form)));
+      $output = drupal_render($form);
+      // When working with a hidden form, we don't want its CSS/JS to be loaded.
+      if ($request->request->get('nocssjs') !== 'true') {
+        $response->setAttachments($form['#attached']);
+      }
+      $response->addCommand(new FieldFormCommand($output));
 
       $errors = $form_state->getErrors();
       if (count($errors)) {
@@ -216,12 +220,6 @@ public function fieldForm(EntityInterface $entity, $field_name, $langcode, $view
       }
     }
 
-    // When working with a hidden form, we don't want any CSS or JS to be loaded.
-    if ($request->request->get('nocssjs') === 'true') {
-      drupal_static_reset('_drupal_add_css');
-      drupal_static_reset('_drupal_add_js');
-    }
-
     return $response;
   }
 
diff --git a/core/modules/quickedit/src/Tests/QuickEditLoadingTest.php b/core/modules/quickedit/src/Tests/QuickEditLoadingTest.php
index a2875155fe26..0cb2c63d0582 100644
--- a/core/modules/quickedit/src/Tests/QuickEditLoadingTest.php
+++ b/core/modules/quickedit/src/Tests/QuickEditLoadingTest.php
@@ -87,9 +87,8 @@ public function testUserWithoutPermission() {
     $this->drupalGet('node/1');
 
     // Library and in-place editors.
-    $settings = $this->getDrupalSettings();
-    $this->assertFalse(isset($settings['ajaxPageState']['js']['core/modules/quickedit/js/quickedit.js']), 'Quick Edit library not loaded.');
-    $this->assertFalse(isset($settings['ajaxPageState']['js']['core/modules/quickedit/js/editors/formEditor.js']), "'form' in-place editor not loaded.");
+    $this->assertNoRaw('core/modules/quickedit/js/quickedit.js',  'Quick Edit library not loaded.');
+    $this->assertNoRaw('core/modules/quickedit/js/editors/formEditor.js', "'form' in-place editor not loaded.");
 
     // HTML annotation must always exist (to not break the render cache).
     $this->assertRaw('data-quickedit-entity-id="node/1"');
@@ -140,8 +139,9 @@ public function testUserWithPermission() {
 
     // Library and in-place editors.
     $settings = $this->getDrupalSettings();
-    $this->assertTrue(isset($settings['ajaxPageState']['js']['core/modules/quickedit/js/quickedit.js']), 'Quick Edit library loaded.');
-    $this->assertFalse(isset($settings['ajaxPageState']['js']['core/modules/quickedit/js/editors/formEditor.js']), "'form' in-place editor not loaded.");
+    $libraries = explode(',', $settings['ajaxPageState']['libraries']);
+    $this->assertTrue(in_array('quickedit/quickedit', $libraries), 'Quick Edit library loaded.');
+    $this->assertFalse(in_array('quickedit/quickedit.inPlaceEditor.form', $libraries), "'form' in-place editor not loaded.");
 
     // HTML annotation must always exist (to not break the render cache).
     $this->assertRaw('data-quickedit-entity-id="node/1"');
@@ -182,7 +182,7 @@ public function testUserWithPermission() {
     $this->assertIdentical('settings', $ajax_commands[0]['command'], 'The first AJAX command is a settings command.');
     // Second command: insert libraries into DOM.
     $this->assertIdentical('insert', $ajax_commands[1]['command'], 'The second AJAX command is an append command.');
-    $this->assertTrue(in_array('core/modules/quickedit/js/editors/formEditor.js', array_keys($ajax_commands[0]['settings']['ajaxPageState']['js'])), 'The quickedit.inPlaceEditor.form library is loaded.');
+    $this->assertTrue(in_array('quickedit/quickedit.inPlaceEditor.form', explode(',', $ajax_commands[0]['settings']['ajaxPageState']['libraries'])), 'The quickedit.inPlaceEditor.form library is loaded.');
 
     // Retrieving the form for this field should result in a 200 response,
     // containing only a quickeditFieldForm command.
@@ -481,6 +481,7 @@ public function testConcurrentEdit() {
 
     if ($form_tokens_found) {
       $post = array(
+        'nocssjs' => 'true',
         'form_id' => 'quickedit_field_form',
         'form_token' => $token_match[1],
         'form_build_id' => $build_id_match[1],
diff --git a/core/modules/simpletest/simpletest.module b/core/modules/simpletest/simpletest.module
index 273bf3c06e37..052e0b75ad9e 100644
--- a/core/modules/simpletest/simpletest.module
+++ b/core/modules/simpletest/simpletest.module
@@ -1,5 +1,6 @@
 <?php
 
+use Drupal\Core\Asset\AttachedAssetsInterface;
 use Drupal\Core\Database\Database;
 use Drupal\Core\Extension\ExtensionDiscovery;
 use Drupal\Core\Render\Element;
@@ -50,7 +51,7 @@ function simpletest_theme() {
 /**
  * Implements hook_js_alter().
  */
-function simpletest_js_alter(&$javascript) {
+function simpletest_js_alter(&$javascript, AttachedAssetsInterface $assets) {
   // Since SimpleTest is a special use case for the table select, stick the
   // SimpleTest JavaScript above the table select.
   $simpletest = drupal_get_path('module', 'simpletest') . '/simpletest.js';
diff --git a/core/modules/simpletest/src/WebTestBase.php b/core/modules/simpletest/src/WebTestBase.php
index 7b12fbfbc312..98253a449407 100644
--- a/core/modules/simpletest/src/WebTestBase.php
+++ b/core/modules/simpletest/src/WebTestBase.php
@@ -1993,12 +1993,7 @@ protected function getAjaxPageStatePostData() {
     if (isset($drupal_settings['ajaxPageState'])) {
       $post['ajax_page_state[theme]'] = $drupal_settings['ajaxPageState']['theme'];
       $post['ajax_page_state[theme_token]'] = $drupal_settings['ajaxPageState']['theme_token'];
-      foreach ($drupal_settings['ajaxPageState']['css'] as $key => $value) {
-        $post["ajax_page_state[css][$key]"] = 1;
-      }
-      foreach ($drupal_settings['ajaxPageState']['js'] as $key => $value) {
-        $post["ajax_page_state[js][$key]"] = 1;
-      }
+      $post['ajax_page_state[libraries]'] = $drupal_settings['ajaxPageState']['libraries'];
     }
     return $post;
   }
diff --git a/core/modules/system/src/Tests/Ajax/DialogTest.php b/core/modules/system/src/Tests/Ajax/DialogTest.php
index 4e295679f32f..f68425123e2b 100644
--- a/core/modules/system/src/Tests/Ajax/DialogTest.php
+++ b/core/modules/system/src/Tests/Ajax/DialogTest.php
@@ -94,7 +94,7 @@ public function testDialog() {
 
     // Emulate going to the JS version of the page and check the JSON response.
     $ajax_result = $this->drupalGetAJAX('ajax-test/dialog-contents', array(), array('Accept: application/vnd.drupal-modal'));
-    $this->assertEqual($modal_expected_response, $ajax_result[1], 'Modal dialog JSON response matches.');
+    $this->assertEqual($modal_expected_response, $ajax_result[3], 'Modal dialog JSON response matches.');
 
     // Check that requesting a "normal" dialog without JS goes to a page.
     $this->drupalGet('ajax-test/dialog-contents');
@@ -136,9 +136,13 @@ public function testDialog() {
     $ajax_result = $this->drupalPostAjaxForm('ajax-test/dialog', array(), 'button1');
 
     // Check that CSS and JavaScript are "added" to the page dynamically.
-    $this->assertTrue(in_array('dialog.css', array_keys($ajax_result[0]['settings']['ajaxPageState']['css'])), 'jQuery UI dialog CSS added to the page.');
-    $this->assertTrue(in_array('core/assets/vendor/jquery.ui/ui/dialog-min.js', array_keys($ajax_result[0]['settings']['ajaxPageState']['js'])), 'jQuery UI dialog JS added to the page.');
-    $this->assertTrue(in_array('core/misc/dialog/dialog.ajax.js', array_keys($ajax_result[0]['settings']['ajaxPageState']['js'])), 'Drupal dialog JS added to the page.');
+    $this->assertTrue(in_array('core/drupal.dialog.ajax', explode(',', $ajax_result[0]['settings']['ajaxPageState']['libraries'])), 'core/drupal.dialog.ajax library is added to the page.');
+    $dialog_css_exists = strpos($ajax_result[1]['data'], 'dialog.css') !== FALSE;
+    $this->assertTrue($dialog_css_exists, 'jQuery UI dialog CSS added to the page.');
+    $dialog_js_exists = strpos($ajax_result[2]['data'], 'dialog-min.js') !== FALSE;
+    $this->assertTrue($dialog_js_exists, 'jQuery UI dialog JS added to the page.');
+    $dialog_js_exists = strpos($ajax_result[2]['data'], 'dialog.ajax.js') !== FALSE;
+    $this->assertTrue($dialog_js_exists, 'Drupal dialog JS added to the page.');
 
     // Check that the response matches the expected value.
     $this->assertEqual($modal_expected_response, $ajax_result[3], 'POST request modal dialog JSON response matches.');
@@ -169,12 +173,12 @@ public function testDialog() {
       ],
     ];
     $this->assertEqual($expected_ajax_settings, $ajax_result[0]['settings']['ajax']);
-    $this->drupalSetContent($ajax_result[1]['data']);
+    $this->drupalSetContent($ajax_result[3]['data']);
     // Remove the data, the form build id and token will never match.
-    unset($ajax_result[1]['data']);
+    unset($ajax_result[3]['data']);
     $form = $this->xpath("//form[@id='ajax-test-form']");
     $this->assertTrue(!empty($form), 'Modal dialog JSON contains form.');
-    $this->assertEqual($form_expected_response, $ajax_result[1]);
+    $this->assertEqual($form_expected_response, $ajax_result[3]);
 
     // Check that requesting an entity form dialog without JS goes to a page.
     $this->drupalGet('admin/structure/contact/add');
@@ -185,12 +189,12 @@ public function testDialog() {
 
     // Emulate going to the JS version of the form and check the JSON response.
     $ajax_result = $this->drupalGetAJAX('admin/structure/contact/add', array(), array('Accept: application/vnd.drupal-modal'));
-    $this->drupalSetContent($ajax_result[1]['data']);
+    $this->drupalSetContent($ajax_result[3]['data']);
     // Remove the data, the form build id and token will never match.
-    unset($ajax_result[1]['data']);
+    unset($ajax_result[3]['data']);
     $form = $this->xpath("//form[@id='contact-form-add-form']");
     $this->assertTrue(!empty($form), 'Modal dialog JSON contains entity form.');
-    $this->assertEqual($entity_form_expected_response, $ajax_result[1]);
+    $this->assertEqual($entity_form_expected_response, $ajax_result[3]);
   }
 
 }
diff --git a/core/modules/system/src/Tests/Ajax/FrameworkTest.php b/core/modules/system/src/Tests/Ajax/FrameworkTest.php
index 5b4d2a3b814a..4e0118934a6e 100644
--- a/core/modules/system/src/Tests/Ajax/FrameworkTest.php
+++ b/core/modules/system/src/Tests/Ajax/FrameworkTest.php
@@ -13,6 +13,7 @@
 use Drupal\Core\Ajax\HtmlCommand;
 use Drupal\Core\Ajax\PrependCommand;
 use Drupal\Core\Ajax\SettingsCommand;
+use Drupal\Core\Asset\AttachedAssets;
 
 /**
  * Performs tests on AJAX framework functions.
@@ -24,11 +25,10 @@ class FrameworkTest extends AjaxTestBase {
    * Ensures \Drupal\Core\Ajax\AjaxResponse::ajaxRender() returns JavaScript settings from the page request.
    */
   public function testAJAXRender() {
-    // Verify that settings command is generated when JavaScript settings are
-    // set via _drupal_add_js().
+    // Verify that settings command is generated if JavaScript settings exist.
     $commands = $this->drupalGetAJAX('ajax-test/render');
     $expected = new SettingsCommand(array('ajax' => 'test'), TRUE);
-    $this->assertCommand($commands, $expected->render(), '\Drupal\Core\Ajax\AjaxResponse::ajaxRender() loads settings added with _drupal_add_js().');
+    $this->assertCommand($commands, $expected->render(), '\Drupal\Core\Ajax\AjaxResponse::ajaxRender() loads JavaScript settings.');
   }
 
   /**
@@ -38,16 +38,22 @@ public function testOrder() {
     $expected_commands = array();
 
     // Expected commands, in a very specific order.
+    $asset_resolver = \Drupal::service('asset.resolver');
+    $css_collection_renderer = \Drupal::service('asset.css.collection_renderer');
+    $js_collection_renderer = \Drupal::service('asset.js.collection_renderer');
+    $renderer = \Drupal::service('renderer');
     $expected_commands[0] = new SettingsCommand(array('ajax' => 'test'), TRUE);
-    drupal_static_reset('_drupal_add_css');
     $build['#attached']['library'][] = 'ajax_test/order-css-command';
-    drupal_process_attached($build);
-    $expected_commands[1] = new AddCssCommand(drupal_get_css(_drupal_add_css(), TRUE));
-    drupal_static_reset('_drupal_add_js');
+    $assets = AttachedAssets::createFromRenderArray($build);
+    $css_render_array = $css_collection_renderer->render($asset_resolver->getCssAssets($assets, FALSE));
+    $expected_commands[1] = new AddCssCommand($renderer->render($css_render_array));
     $build['#attached']['library'][] = 'ajax_test/order-js-command';
-    drupal_process_attached($build);
-    $expected_commands[2] = new PrependCommand('head', drupal_get_js('header', _drupal_add_js(), TRUE));
-    $expected_commands[3] = new AppendCommand('body', drupal_get_js('footer', _drupal_add_js(), TRUE));
+    $assets = AttachedAssets::createFromRenderArray($build);
+    list($js_assets_header, $js_assets_footer) = $asset_resolver->getJsAssets($assets, FALSE);
+    $js_header_render_array = $js_collection_renderer->render($js_assets_header);
+    $js_footer_render_array = $js_collection_renderer->render($js_assets_footer);
+    $expected_commands[2] = new PrependCommand('head', $renderer->render($js_header_render_array));
+    $expected_commands[3] = new AppendCommand('body', $renderer->render($js_footer_render_array));
     $expected_commands[4] = new HtmlCommand('body', 'Hello, world!');
 
     // Load any page with at least one CSS file, at least one JavaScript file
@@ -88,50 +94,52 @@ public function testAJAXRenderError() {
    * Tests that new JavaScript and CSS files are lazy-loaded on an AJAX request.
    */
   public function testLazyLoad() {
+    $asset_resolver = \Drupal::service('asset.resolver');
+    $css_collection_renderer = \Drupal::service('asset.css.collection_renderer');
+    $js_collection_renderer = \Drupal::service('asset.js.collection_renderer');
+    $renderer = \Drupal::service('renderer');
+
     $expected = array(
       'setting_name' => 'ajax_forms_test_lazy_load_form_submit',
       'setting_value' => 'executed',
-      'css' => drupal_get_path('module', 'system') . '/css/system.admin.css',
-      'js' => drupal_get_path('module', 'system') . '/system.js',
+      'library_1' => 'system/admin',
+      'library_2' => 'system/drupal.system',
     );
-    // CSS files are stored by basename, see _drupal_add_css().
-    $expected_css_basename = drupal_basename($expected['css']);
-
-    // @todo D8: Add a drupal_css_defaults() helper function.
-    $expected_css_html = drupal_get_css(array($expected_css_basename => array(
-      'type' => 'file',
-      'group' => CSS_AGGREGATE_DEFAULT,
-      'weight' => 0,
-      'every_page' => FALSE,
-      'media' => 'all',
-      'preprocess' => TRUE,
-      'data' => $expected['css'],
-      'browsers' => array('IE' => TRUE, '!IE' => TRUE),
-    )), TRUE);
-    $expected_js_html = drupal_get_js('header', array($expected['js'] => ['version' => \Drupal::VERSION] + drupal_js_defaults($expected['js'])), TRUE);
 
     // Get the base page.
     $this->drupalGet('ajax_forms_test_lazy_load_form');
     $original_settings = $this->getDrupalSettings();
-    $original_css = $original_settings['ajaxPageState']['css'];
-    $original_js = $original_settings['ajaxPageState']['js'];
+    $original_libraries = explode(',', $original_settings['ajaxPageState']['libraries']);
 
     // Verify that the base page doesn't have the settings and files that are to
     // be lazy loaded as part of the next requests.
     $this->assertTrue(!isset($original_settings[$expected['setting_name']]), format_string('Page originally lacks the %setting, as expected.', array('%setting' => $expected['setting_name'])));
-    $this->assertTrue(!isset($original_css[$expected['css']]), format_string('Page originally lacks the %css file, as expected.', array('%css' => $expected['css'])));
-    $this->assertTrue(!isset($original_js[$expected['js']]), format_string('Page originally lacks the %js file, as expected.', array('%js' => $expected['js'])));
+    $this->assertTrue(!in_array($expected['library_1'], $original_libraries), format_string('Page originally lacks the %library library, as expected.', array('%library' => $expected['library_1'])));
+    $this->assertTrue(!in_array($expected['library_2'], $original_libraries), format_string('Page originally lacks the %library library, as expected.', array('%library' => $expected['library_2'])));
+
+    // Calculate the expected CSS and JS.
+    $assets = new AttachedAssets();
+    $assets->setLibraries([$expected['library_1']])
+      ->setAlreadyLoadedLibraries($original_libraries);
+    $css_render_array = $css_collection_renderer->render($asset_resolver->getCssAssets($assets, FALSE));
+    $expected_css_html = $renderer->render($css_render_array);
+
+    $assets->setLibraries([$expected['library_2']])
+      ->setAlreadyLoadedLibraries($original_libraries);
+    $js_assets = $asset_resolver->getJsAssets($assets, FALSE)[0];
+    unset($js_assets['drupalSettings']);
+    $js_render_array = $js_collection_renderer->render($js_assets);
+    $expected_js_html = $renderer->render($js_render_array);
 
     // Submit the AJAX request without triggering files getting added.
     $commands = $this->drupalPostAjaxForm(NULL, array('add_files' => FALSE), array('op' => t('Submit')));
     $new_settings = $this->getDrupalSettings();
-    $new_css = $new_settings['ajaxPageState']['css'];
-    $new_js = $new_settings['ajaxPageState']['js'];
+    $new_libraries = explode(',', $new_settings['ajaxPageState']['libraries']);
 
     // Verify the setting was not added when not expected.
     $this->assertTrue(!isset($new_settings[$expected['setting_name']]), format_string('Page still lacks the %setting, as expected.', array('%setting' => $expected['setting_name'])));
-    $this->assertTrue(!isset($new_css[$expected['css']]), format_string('Page still lacks the %css file, as expected.', array('%css' => $expected['css'])));
-    $this->assertTrue(!isset($new_js[$expected['js']]), format_string('Page still lacks the %js file, as expected.', array('%js' => $expected['js'])));
+    $this->assertTrue(!in_array($expected['library_1'], $new_libraries), format_string('Page still lacks the %library library, as expected.', array('%library' => $expected['library_1'])));
+    $this->assertTrue(!in_array($expected['library_2'], $new_libraries), format_string('Page still lacks the %library library, as expected.', array('%library' => $expected['library_2'])));
     // Verify a settings command does not add CSS or scripts to drupalSettings
     // and no command inserts the corresponding tags on the page.
     $found_settings_command = FALSE;
@@ -144,14 +152,13 @@ public function testLazyLoad() {
         $found_markup_command = TRUE;
       }
     }
-    $this->assertFalse($found_settings_command, format_string('Page state still lacks the %css and %js files, as expected.', array('%css' => $expected['css'], '%js' => $expected['js'])));
-    $this->assertFalse($found_markup_command, format_string('Page still lacks the %css and %js files, as expected.', array('%css' => $expected['css'], '%js' => $expected['js'])));
+    $this->assertFalse($found_settings_command, format_string('Page state still lacks the %library_1 and %library_2 libraries, as expected.', array('%library_1' => $expected['library_1'], '%library_2' => $expected['library_2'])));
+    $this->assertFalse($found_markup_command, format_string('Page still lacks the %library_1 and %library_2 libraries, as expected.', array('%library_1' => $expected['library_1'], '%library_2' => $expected['library_2'])));
 
     // Submit the AJAX request and trigger adding files.
     $commands = $this->drupalPostAjaxForm(NULL, array('add_files' => TRUE), array('op' => t('Submit')));
     $new_settings = $this->getDrupalSettings();
-    $new_css = $new_settings['ajaxPageState']['css'];
-    $new_js = $new_settings['ajaxPageState']['js'];
+    $new_libraries = explode(',', $new_settings['ajaxPageState']['libraries']);
 
     // Verify the expected setting was added, both to drupalSettings, and as
     // the first AJAX command.
@@ -161,8 +168,8 @@ public function testLazyLoad() {
 
     // Verify the expected CSS file was added, both to drupalSettings, and as
     // the second AJAX command for inclusion into the HTML.
-    $this->assertEqual($new_css, $original_css + array($expected_css_basename => 1), format_string('Page state now has the %css file.', array('%css' => $expected['css'])));
-    $this->assertCommand(array_slice($commands, 1, 1), array('data' => $expected_css_html), format_string('Page now has the %css file.', array('%css' => $expected['css'])));
+    $this->assertTrue(in_array($expected['library_1'], $new_libraries), format_string('Page state now has the %library library.', array('%library' => $expected['library_1'])));
+    $this->assertCommand(array_slice($commands, 1, 1), array('data' => $expected_css_html), format_string('Page now has the %library library.', array('%library' => $expected['library_1'])));
 
     // Verify the expected JS file was added, both to drupalSettings, and as
     // the third AJAX command for inclusion into the HTML. By testing for an
@@ -170,8 +177,8 @@ public function testLazyLoad() {
     // unexpected JavaScript code, such as a jQuery.extend() that would
     // potentially clobber rather than properly merge settings, didn't
     // accidentally get added.
-    $this->assertEqual($new_js, $original_js + array($expected['js'] => 1), format_string('Page state now has the %js file.', array('%js' => $expected['js'])));
-    $this->assertCommand(array_slice($commands, 2, 1), array('data' => $expected_js_html), format_string('Page now has the %js file.', array('%js' => $expected['js'])));
+    $this->assertTrue(in_array($expected['library_2'], $new_libraries), format_string('Page state now has the %library library.', array('%library' => $expected['library_2'])));
+    $this->assertCommand(array_slice($commands, 2, 1), array('data' => $expected_js_html), format_string('Page now has the %library library.', array('%library' => $expected['library_2'])));
   }
 
   /**
diff --git a/core/modules/system/src/Tests/Common/AttachedAssetsTest.php b/core/modules/system/src/Tests/Common/AttachedAssetsTest.php
index 8401acfbfbf5..7cd97b384806 100644
--- a/core/modules/system/src/Tests/Common/AttachedAssetsTest.php
+++ b/core/modules/system/src/Tests/Common/AttachedAssetsTest.php
@@ -10,6 +10,7 @@
 use Drupal\Component\Serialization\Json;
 use Drupal\Component\Utility\Unicode;
 use Drupal\Component\Utility\Crypt;
+use Drupal\Core\Asset\AttachedAssets;
 use Drupal\simpletest\KernelTestBase;
 
 /**
@@ -27,6 +28,20 @@
  */
 class AttachedAssetsTest extends KernelTestBase {
 
+  /**
+   * The asset resolver service.
+   *
+   * @var \Drupal\Core\Asset\AssetResolverInterface
+   */
+  protected $assetResolver;
+
+  /**
+   * The renderer service.
+   *
+   * @var \Drupal\Core\Render\RendererInterface
+   */
+  protected $renderer;
+
   /**
    * {@inheritdoc}
    */
@@ -38,29 +53,19 @@ class AttachedAssetsTest extends KernelTestBase {
   protected function setUp() {
     parent::setUp();
 
-    // Disable preprocessing.
-    $this->config('system.performance')
-      ->set('css.preprocess', FALSE)
-      ->set('js.preprocess', FALSE)
-      ->save();
-
-    // Reset _drupal_add_css() and _drupal_add_js() statics before each test.
-    drupal_static_reset('_drupal_add_css');
-    drupal_static_reset('_drupal_add_js');
-
-    $this->installSchema('system', 'router');
-    \Drupal::service('router.builder')->rebuild();
+    $this->assetResolver = $this->container->get('asset.resolver');
+    $this->renderer = $this->container->get('renderer');
   }
 
   /**
    * Tests that default CSS and JavaScript is empty.
    */
   function testDefault() {
-    $build['#attached'] = [];
-    drupal_process_attached($build);
-
-    $this->assertEqual(array(), _drupal_add_css(), 'Default CSS is empty.');
-    $this->assertEqual(array(), _drupal_add_js(), 'Default JavaScript is empty.');
+    $assets = new AttachedAssets();
+    $this->assertEqual(array(), $this->assetResolver->getCssAssets($assets, FALSE), 'Default CSS is empty.');
+    list($js_assets_header, $js_assets_footer) = $this->assetResolver->getJsAssets($assets, FALSE);
+    $this->assertEqual(array(), $js_assets_header, 'Default header JavaScript is empty.');
+    $this->assertEqual(array(), $js_assets_footer, 'Default footer JavaScript is empty.');
   }
 
   /**
@@ -68,10 +73,9 @@ function testDefault() {
    */
   function testLibraryUnknown() {
     $build['#attached']['library'][] = 'unknown/unknown';
-    drupal_process_attached($build);
+    $assets = AttachedAssets::createFromRenderArray($build);
 
-    $scripts = drupal_get_js();
-    $this->assertTrue(strpos($scripts, 'unknown') === FALSE, 'Unknown library was not added to the page.');
+    $this->assertIdentical([], $this->assetResolver->getJsAssets($assets, FALSE)[0], 'Unknown library was not added to the page.');
   }
 
   /**
@@ -79,15 +83,17 @@ function testLibraryUnknown() {
    */
   function testAddFiles() {
     $build['#attached']['library'][] = 'common_test/files';
-    drupal_process_attached($build);
+    $assets = AttachedAssets::createFromRenderArray($build);
 
-    $css = _drupal_add_css();
-    $js = _drupal_add_js();
+    $css = $this->assetResolver->getCssAssets($assets, FALSE);
+    $js = $this->assetResolver->getJsAssets($assets, FALSE)[0];
     $this->assertTrue(array_key_exists('bar.css', $css), 'CSS files are correctly added.');
     $this->assertTrue(array_key_exists('core/modules/system/tests/modules/common_test/foo.js', $js), 'JavaScript files are correctly added.');
 
-    $rendered_css = drupal_get_css();
-    $rendered_js = drupal_get_js();
+    $css_render_array = \Drupal::service('asset.css.collection_renderer')->render($css);
+    $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js);
+    $rendered_css = $this->renderer->render($css_render_array);
+    $rendered_js = $this->renderer->render($js_render_array);
     $query_string = $this->container->get('state')->get('system.css_js_query_string') ?: '0';
     $this->assertNotIdentical(strpos($rendered_css, '<link rel="stylesheet" href="' . file_create_url('core/modules/system/tests/modules/common_test/bar.css') . '?' . $query_string . '" media="all" />'), FALSE, 'Rendering an external CSS file.');
     $this->assertNotIdentical(strpos($rendered_js, '<script src="' . file_create_url('core/modules/system/tests/modules/common_test/foo.js') . '?' . $query_string . '"></script>'), FALSE, 'Rendering an external JavaScript file.');
@@ -99,12 +105,13 @@ function testAddFiles() {
   function testAddJsSettings() {
     // Add a file in order to test default settings.
     $build['#attached']['library'][] = 'core/drupalSettings';
-    drupal_process_attached($build);
+    $assets = AttachedAssets::createFromRenderArray($build);
 
-    $javascript = _drupal_add_js();
+    $javascript = $this->assetResolver->getJsAssets($assets, FALSE)[0];
     $this->assertTrue(array_key_exists('currentPath', $javascript['drupalSettings']['data']['path']), 'The current path JavaScript setting is set correctly.');
 
-    $javascript = _drupal_add_js(array('drupal' => 'rocks', 'dries' => 280342800), 'setting');
+    $assets->setSettings(['drupal' => 'rocks', 'dries' => 280342800]);
+    $javascript = $this->assetResolver->getJsAssets($assets, FALSE)[0];
     $this->assertEqual(280342800, $javascript['drupalSettings']['data']['dries'], 'JavaScript setting is set correctly.');
     $this->assertEqual('rocks', $javascript['drupalSettings']['data']['drupal'], 'The other JavaScript setting is set correctly.');
   }
@@ -114,15 +121,17 @@ function testAddJsSettings() {
    */
   function testAddExternalFiles() {
     $build['#attached']['library'][] = 'common_test/external';
-    drupal_process_attached($build);
+    $assets = AttachedAssets::createFromRenderArray($build);
 
-    $css = _drupal_add_css();
-    $js = _drupal_add_js();
+    $css = $this->assetResolver->getCssAssets($assets, FALSE);
+    $js = $this->assetResolver->getJsAssets($assets, FALSE)[0];
     $this->assertTrue(array_key_exists('http://example.com/stylesheet.css', $css), 'External CSS files are correctly added.');
     $this->assertTrue(array_key_exists('http://example.com/script.js', $js), 'External JavaScript files are correctly added.');
 
-    $rendered_css = drupal_get_css();
-    $rendered_js = drupal_get_js();
+    $css_render_array = \Drupal::service('asset.css.collection_renderer')->render($css);
+    $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js);
+    $rendered_css = $this->renderer->render($css_render_array);
+    $rendered_js = $this->renderer->render($js_render_array);
     $this->assertNotIdentical(strpos($rendered_css, '<link rel="stylesheet" href="http://example.com/stylesheet.css" media="all" />'), FALSE, 'Rendering an external CSS file.');
     $this->assertNotIdentical(strpos($rendered_js, '<script src="http://example.com/script.js"></script>'), FALSE, 'Rendering an external JavaScript file.');
   }
@@ -132,9 +141,11 @@ function testAddExternalFiles() {
    */
   function testAttributes() {
     $build['#attached']['library'][] = 'common_test/js-attributes';
-    drupal_process_attached($build);
+    $assets = AttachedAssets::createFromRenderArray($build);
 
-    $rendered_js = drupal_get_js();
+    $js = $this->assetResolver->getJsAssets($assets, FALSE)[0];
+    $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js);
+    $rendered_js = $this->renderer->render($js_render_array);
     $expected_1 = '<script src="http://example.com/deferred-external.js" foo="bar" defer></script>';
     $expected_2 = '<script src="' . file_create_url('core/modules/system/tests/modules/common_test/deferred-internal.js') . '?v=1" defer bar="foo"></script>';
     $this->assertNotIdentical(strpos($rendered_js, $expected_1), FALSE, 'Rendered external JavaScript with correct defer and random attributes.');
@@ -145,13 +156,12 @@ function testAttributes() {
    * Tests that attributes are maintained when JS aggregation is enabled.
    */
   function testAggregatedAttributes() {
-    // Enable aggregation.
-    $this->config('system.performance')->set('js.preprocess', 1)->save();
-
     $build['#attached']['library'][] = 'common_test/js-attributes';
-    drupal_process_attached($build);
+    $assets = AttachedAssets::createFromRenderArray($build);
 
-    $rendered_js = drupal_get_js();
+    $js = $this->assetResolver->getJsAssets($assets, TRUE)[0];
+    $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js);
+    $rendered_js = $this->renderer->render($js_render_array);
     $expected_1 = '<script src="http://example.com/deferred-external.js" foo="bar" defer></script>';
     $expected_2 = '<script src="' . file_create_url('core/modules/system/tests/modules/common_test/deferred-internal.js') . '?v=1" defer bar="foo"></script>';
     $this->assertNotIdentical(strpos($rendered_js, $expected_1), FALSE, 'Rendered external JavaScript with correct defer and random attributes.');
@@ -166,9 +176,11 @@ function testHeaderSetting() {
     $build['#attached']['library'][] = 'core/drupalSettings';
     // Nonsensical value to verify if it's possible to override path settings.
     $build['#attached']['drupalSettings']['path']['pathPrefix'] = 'yarhar';
-    drupal_process_attached($build);
+    $assets = AttachedAssets::createFromRenderArray($build);
 
-    $rendered_js = drupal_get_js('header');
+    $js = $this->assetResolver->getJsAssets($assets, FALSE)[0];
+    $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js);
+    $rendered_js = $this->renderer->render($js_render_array);
 
     // Parse the generated drupalSettings <script> back to a PHP representation.
     $startToken = 'drupalSettings = ';
@@ -199,21 +211,23 @@ function testHeaderSetting() {
    */
   function testFooterHTML() {
     $build['#attached']['library'][] = 'common_test/js-footer';
-    drupal_process_attached($build);
+    $assets = AttachedAssets::createFromRenderArray($build);
 
-    $rendered_js = drupal_get_js('footer');
+    $js = $this->assetResolver->getJsAssets($assets, FALSE)[1];
+    $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js);
+    $rendered_js = $this->renderer->render($js_render_array);
     $query_string = $this->container->get('state')->get('system.css_js_query_string') ?: '0';
     $this->assertNotIdentical(strpos($rendered_js, '<script src="' . file_create_url('core/modules/system/tests/modules/common_test/footer.js') . '?' . $query_string . '"></script>'), FALSE, 'Rendering an external JavaScript file.');
   }
 
   /**
-   * Tests _drupal_add_js() sets preprocess to FALSE when cache is also FALSE.
+   * Tests that for assets with cache = FALSE, Drupal sets preprocess = FALSE.
    */
   function testNoCache() {
     $build['#attached']['library'][] = 'common_test/no-cache';
-    drupal_process_attached($build);
+    $assets = AttachedAssets::createFromRenderArray($build);
 
-    $js = _drupal_add_js();
+    $js = $this->assetResolver->getJsAssets($assets, FALSE)[0];
     $this->assertFalse($js['core/modules/system/tests/modules/common_test/nocache.js']['preprocess'], 'Setting cache to FALSE sets preprocess to FALSE when adding JavaScript.');
   }
 
@@ -226,14 +240,16 @@ function testBrowserConditionalComments() {
     $default_query_string = $this->container->get('state')->get('system.css_js_query_string') ?: '0';
 
     $build['#attached']['library'][] = 'common_test/browsers';
-    drupal_process_attached($build);
+    $assets = AttachedAssets::createFromRenderArray($build);
 
-    $js = drupal_get_js();
+    $js = $this->assetResolver->getJsAssets($assets, FALSE)[0];
+    $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js);
+    $rendered_js = $this->renderer->render($js_render_array);
     $expected_1 = "<!--[if lte IE 8]>\n" . '<script src="' . file_create_url('core/modules/system/tests/modules/common_test/old-ie.js') . '?' . $default_query_string . '"></script>' . "\n<![endif]-->";
     $expected_2 = "<!--[if !IE]><!-->\n" . '<script src="' . file_create_url('core/modules/system/tests/modules/common_test/no-ie.js') . '?' . $default_query_string . '"></script>' . "\n<!--<![endif]-->";
 
-    $this->assertNotIdentical(strpos($js, $expected_1), FALSE, 'Rendered JavaScript within downlevel-hidden conditional comments.');
-    $this->assertNotIdentical(strpos($js, $expected_2), FALSE, 'Rendered JavaScript within downlevel-revealed conditional comments.');
+    $this->assertNotIdentical(strpos($rendered_js, $expected_1), FALSE, 'Rendered JavaScript within downlevel-hidden conditional comments.');
+    $this->assertNotIdentical(strpos($rendered_js, $expected_2), FALSE, 'Rendered JavaScript within downlevel-revealed conditional comments.');
   }
 
   /**
@@ -242,10 +258,12 @@ function testBrowserConditionalComments() {
   function testVersionQueryString() {
     $build['#attached']['library'][] = 'core/backbone';
     $build['#attached']['library'][] = 'core/domready';
-    drupal_process_attached($build);
+    $assets = AttachedAssets::createFromRenderArray($build);
 
-    $js = drupal_get_js();
-    $this->assertTrue(strpos($js, 'core/assets/vendor/backbone/backbone-min.js?v=1.1.2') > 0 && strpos($js, 'core/assets/vendor/domready/ready.min.js?v=1.0.7') > 0 , 'JavaScript version identifiers correctly appended to URLs');
+    $js = $this->assetResolver->getJsAssets($assets, FALSE)[0];
+    $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js);
+    $rendered_js = $this->renderer->render($js_render_array);
+    $this->assertTrue(strpos($rendered_js, 'core/assets/vendor/backbone/backbone-min.js?v=1.1.2') > 0 && strpos($rendered_js, 'core/assets/vendor/domready/ready.min.js?v=1.0.7') > 0 , 'JavaScript version identifiers correctly appended to URLs');
   }
 
   /**
@@ -253,7 +271,7 @@ function testVersionQueryString() {
    */
   function testRenderOrder() {
     $build['#attached']['library'][] = 'common_test/order';
-    drupal_process_attached($build);
+    $assets = AttachedAssets::createFromRenderArray($build);
 
     // Construct the expected result from the regex.
     $expected_order_js = [
@@ -270,7 +288,9 @@ function testRenderOrder() {
     ];
 
     // Retrieve the rendered JavaScript and test against the regex.
-    $rendered_js = drupal_get_js();
+    $js = $this->assetResolver->getJsAssets($assets, FALSE)[0];
+    $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js);
+    $rendered_js = $this->renderer->render($js_render_array);
     $matches = array();
     if (preg_match_all('/weight_([-0-9]+_[0-9]+)/', $rendered_js, $matches)) {
       $result = $matches[1];
@@ -309,8 +329,10 @@ function testRenderOrder() {
       'theme_weight_0_2',
     ];
 
-    // Retrieve the rendered JavaScript and test against the regex.
-    $rendered_css = drupal_get_css();
+    // Retrieve the rendered CSS and test against the regex.
+    $css = $this->assetResolver->getCssAssets($assets, FALSE);
+    $css_render_array = \Drupal::service('asset.css.collection_renderer')->render($css);
+    $rendered_css = $this->renderer->render($css_render_array);
     $matches = array();
     if (preg_match_all('/([a-z]+)_weight_([-0-9]+_[0-9]+)/', $rendered_css, $matches)) {
       $result = $matches[0];
@@ -329,12 +351,14 @@ function testRenderDifferentWeight() {
     // still make itself appear first by defining a lower weight.
     $build['#attached']['library'][] = 'core/jquery';
     $build['#attached']['library'][] = 'common_test/weight';
-    drupal_process_attached($build);
-
-    $js = drupal_get_js();
-    $this->assertTrue(strpos($js, 'lighter.css') < strpos($js, 'first.js'), 'Lighter CSS assets are rendered first.');
-    $this->assertTrue(strpos($js, 'lighter.js') < strpos($js, 'first.js'), 'Lighter JavaScript assets are rendered first.');
-    $this->assertTrue(strpos($js, 'before-jquery.js') < strpos($js, 'core/assets/vendor/jquery/jquery.min.js'), 'Rendering a JavaScript file above jQuery.');
+    $assets = AttachedAssets::createFromRenderArray($build);
+
+    $js = $this->assetResolver->getJsAssets($assets, FALSE)[0];
+    $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js);
+    $rendered_js = $this->renderer->render($js_render_array);
+    $this->assertTrue(strpos($rendered_js, 'lighter.css') < strpos($rendered_js, 'first.js'), 'Lighter CSS assets are rendered first.');
+    $this->assertTrue(strpos($rendered_js, 'lighter.js') < strpos($rendered_js, 'first.js'), 'Lighter JavaScript assets are rendered first.');
+    $this->assertTrue(strpos($rendered_js, 'before-jquery.js') < strpos($rendered_js, 'core/assets/vendor/jquery/jquery.min.js'), 'Rendering a JavaScript file above jQuery.');
   }
 
   /**
@@ -346,13 +370,15 @@ function testAlter() {
     // Add both tableselect.js and simpletest.js.
     $build['#attached']['library'][] = 'core/drupal.tableselect';
     $build['#attached']['library'][] = 'simpletest/drupal.simpletest';
-    drupal_process_attached($build);
+    $assets = AttachedAssets::createFromRenderArray($build);
 
     // Render the JavaScript, testing if simpletest.js was altered to be before
     // tableselect.js. See simpletest_js_alter() to see where this alteration
     // takes place.
-    $js = drupal_get_js();
-    $this->assertTrue(strpos($js, 'simpletest.js') < strpos($js, 'core/misc/tableselect.js'), 'Altering JavaScript weight through the alter hook.');
+    $js = $this->assetResolver->getJsAssets($assets, FALSE)[0];
+    $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js);
+    $rendered_js = $this->renderer->render($js_render_array);
+    $this->assertTrue(strpos($rendered_js, 'simpletest.js') < strpos($rendered_js, 'core/misc/tableselect.js'), 'Altering JavaScript weight through the alter hook.');
   }
 
   /**
@@ -369,9 +395,11 @@ function testLibraryAlter() {
 
     // common_test_library_info_alter() also added a dependency on jQuery Form.
     $build['#attached']['library'][] = 'core/jquery.farbtastic';
-    drupal_process_attached($build);
-    $scripts = drupal_get_js();
-    $this->assertTrue(strpos($scripts, 'core/assets/vendor/jquery-form/jquery.form.js'), 'Altered library dependencies are added to the page.');
+    $assets = AttachedAssets::createFromRenderArray($build);
+    $js = $this->assetResolver->getJsAssets($assets, FALSE)[0];
+    $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js);
+    $rendered_js = $this->renderer->render($js_render_array);
+    $this->assertTrue(strpos($rendered_js, 'core/assets/vendor/jquery-form/jquery.form.js'), 'Altered library dependencies are added to the page.');
   }
 
   /**
@@ -413,15 +441,17 @@ function testLibraryNameConflicts() {
    */
   function testAddJsFileWithQueryString() {
     $build['#attached']['library'][] = 'common_test/querystring';
-    drupal_process_attached($build);
+    $assets = AttachedAssets::createFromRenderArray($build);
 
-    $css = _drupal_add_css();
-    $js = _drupal_add_js();
+    $css = $this->assetResolver->getCssAssets($assets, FALSE);
+    $js = $this->assetResolver->getJsAssets($assets, FALSE)[0];
     $this->assertTrue(array_key_exists('querystring.css?arg1=value1&arg2=value2', $css), 'CSS file with query string is correctly added.');
     $this->assertTrue(array_key_exists('core/modules/system/tests/modules/common_test/querystring.js?arg1=value1&arg2=value2', $js), 'JavaScript file with query string is correctly added.');
 
-    $rendered_css = drupal_get_css();
-    $rendered_js = drupal_get_js();
+    $css_render_array = \Drupal::service('asset.css.collection_renderer')->render($css);
+    $rendered_css = $this->renderer->render($css_render_array);
+    $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js);
+    $rendered_js = $this->renderer->render($js_render_array);
     $query_string = $this->container->get('state')->get('system.css_js_query_string') ?: '0';
     $this->assertNotIdentical(strpos($rendered_css, '<link rel="stylesheet" href="' . str_replace('&', '&amp;', file_create_url('core/modules/system/tests/modules/common_test/querystring.css?arg1=value1&arg2=value2')) . '&amp;' . $query_string . '" media="all" />'), FALSE, 'CSS file with query string gets version query string correctly appended..');
     $this->assertNotIdentical(strpos($rendered_js, '<script src="' . str_replace('&', '&amp;', file_create_url('core/modules/system/tests/modules/common_test/querystring.js?arg1=value1&arg2=value2')) . '&amp;' . $query_string . '"></script>'), FALSE, 'JavaScript file with query string gets version query string correctly appended.');
diff --git a/core/modules/system/src/Tests/Theme/TableTest.php b/core/modules/system/src/Tests/Theme/TableTest.php
index 412cb8e2386b..88aa25fa59f0 100644
--- a/core/modules/system/src/Tests/Theme/TableTest.php
+++ b/core/modules/system/src/Tests/Theme/TableTest.php
@@ -46,10 +46,8 @@ function testThemeTableStickyHeaders() {
       '#sticky' => TRUE,
     );
     $this->render($table);
-    $js = _drupal_add_js();
-    $this->assertTrue(isset($js['core/misc/tableheader.js']), 'tableheader.js found.');
+    $this->assertTrue(in_array('core/drupal.tableheader', $table['#attached']['library']), 'tableheader asset library found.');
     $this->assertRaw('sticky-enabled');
-    drupal_static_reset('_drupal_add_js');
   }
 
   /**
@@ -71,10 +69,8 @@ function testThemeTableNoStickyHeaders() {
       '#sticky' => FALSE,
     );
     $this->render($table);
-    $js = _drupal_add_js();
-    $this->assertFalse(isset($js['core/misc/tableheader.js']), 'tableheader.js not found.');
+    $this->assertFalse(in_array('core/drupal.tableheader', $table['#attached']['library']), 'tableheader asset library not found.');
     $this->assertNoRaw('sticky-enabled');
-    drupal_static_reset('_drupal_add_js');
   }
 
   /**
diff --git a/core/modules/system/system.module b/core/modules/system/system.module
index 458815422b60..c75cf18901c5 100644
--- a/core/modules/system/system.module
+++ b/core/modules/system/system.module
@@ -6,6 +6,7 @@
  */
 
 use Drupal\Component\Utility\UrlHelper;
+use Drupal\Core\Asset\AttachedAssetsInterface;
 use Drupal\Core\Cache\Cache;
 use Drupal\Core\Extension\Extension;
 use Drupal\Core\Extension\ExtensionDiscovery;
@@ -618,9 +619,9 @@ function system_page_attachments(array &$page) {
 /**
  * Implements hook_js_settings_alter().
  *
- * Generates the values for the core/drupalSettings library.
+ * Sets values for the core/drupalSettings and core/drupal.ajax libraries.
  */
-function system_js_settings_alter(&$settings) {
+function system_js_settings_alter(&$settings, AttachedAssetsInterface $assets) {
   // url() generates the script and prefix using hook_url_outbound_alter().
   // Instead of running the hook_url_outbound_alter() again here, extract
   // them from url().
@@ -656,6 +657,37 @@ function system_js_settings_alter(&$settings) {
   if (!isset($settings['locale']['pluralDelimiter'])) {
     $settings['locale']['pluralDelimiter'] = LOCALE_PLURAL_DELIMITER;
   }
+
+  // Now generate the values for the core/drupal.ajax library.
+  // We need to send ajaxPageState settings for core/drupal.ajax if:
+  // - ajaxPageState is being loaded in this Response, in which case it will
+  //   already exist at $settings['ajaxPageState'] (because the core/drupal.ajax
+  //   library definition specifies a placeholder 'ajaxPageState' setting).
+  // - core/drupal.ajax already has been loaded and hence this is an AJAX
+  //   Response in which we must send the list of extra asset libraries that are
+  //   being added in this AJAX Response.
+  /** @var \Drupal\Core\Asset\LibraryDependencyResolver $library_dependency_resolver */
+  $library_dependency_resolver = \Drupal::service('library.dependency_resolver');
+  if (isset($settings['ajaxPageState']) || in_array('core/drupal.ajax', $library_dependency_resolver->getLibrariesWithDependencies($assets->getAlreadyLoadedLibraries()))) {
+    // Provide the page with information about the theme that's used, so that
+    // a later AJAX request can be rendered using the same theme.
+    // @see \Drupal\Core\Theme\AjaxBasePageNegotiator
+    $theme_key = \Drupal::theme()->getActiveTheme()->getName();
+    $settings['ajaxPageState']['theme'] = $theme_key;
+    // Checks that the DB is available before filling theme_token.
+    if (!defined('MAINTENANCE_MODE')) {
+      $settings['ajaxPageState']['theme_token'] = \Drupal::csrfToken()->get($theme_key);
+    }
+
+    // Provide the page with information about the individual asset libraries
+    // used, information not otherwise available when aggregation is enabled.
+    $minimal_libraries = $library_dependency_resolver->getMinimalRepresentativeSubset(array_merge(
+      $assets->getLibraries(),
+      $assets->getAlreadyLoadedLibraries()
+    ));
+    sort($minimal_libraries);
+    $settings['ajaxPageState']['libraries'] = implode(',', $minimal_libraries);
+  }
 }
 
 /**
diff --git a/core/modules/system/tests/modules/ajax_forms_test/ajax_forms_test.module b/core/modules/system/tests/modules/ajax_forms_test/ajax_forms_test.module
index 2665c30d9953..3e470884db09 100644
--- a/core/modules/system/tests/modules/ajax_forms_test/ajax_forms_test.module
+++ b/core/modules/system/tests/modules/ajax_forms_test/ajax_forms_test.module
@@ -184,5 +184,15 @@ function ajax_forms_test_validation_number_form_callback($form, FormStateInterfa
  * AJAX form callback: Selects for the ajax_forms_test_lazy_load_form() form.
  */
 function ajax_forms_test_lazy_load_form_ajax($form, FormStateInterface $form_state) {
-  return array('#markup' => 'new content');
+  $build = [
+    '#markup' => 'new content',
+  ];
+
+  if ($form_state->getValue('add_files')) {
+    $build['#attached']['library'][] = 'system/admin';
+    $build['#attached']['library'][] = 'system/drupal.system';
+    $build['#attached']['drupalSettings']['ajax_forms_test_lazy_load_form_submit'] = 'executed';
+  }
+
+  return $build;
 }
diff --git a/core/modules/system/tests/modules/ajax_forms_test/src/Form/AjaxFormsTestLazyLoadForm.php b/core/modules/system/tests/modules/ajax_forms_test/src/Form/AjaxFormsTestLazyLoadForm.php
index 41e8499bd59a..fb4e010c3c9b 100644
--- a/core/modules/system/tests/modules/ajax_forms_test/src/Form/AjaxFormsTestLazyLoadForm.php
+++ b/core/modules/system/tests/modules/ajax_forms_test/src/Form/AjaxFormsTestLazyLoadForm.php
@@ -53,21 +53,6 @@ public function buildForm(array $form, FormStateInterface $form_state) {
    * {@inheritdoc}
    */
   public function submitForm(array &$form, FormStateInterface $form_state) {
-    if ($form_state->getValue('add_files')) {
-      $attached = [
-        '#attached' => [
-          'library' => [
-            'system/admin',
-            'system/drupal.system',
-          ],
-          'drupalSettings' => [
-            'ajax_forms_test_lazy_load_form_submit' => 'executed',
-          ],
-        ],
-      ];
-      drupal_render($attached);
-      drupal_process_attached($attached);
-    }
     $form_state->setRebuild();
   }
 
diff --git a/core/modules/system/tests/modules/ajax_test/ajax_test.module b/core/modules/system/tests/modules/ajax_test/ajax_test.module
index d81da66ec830..d1faba3e4769 100644
--- a/core/modules/system/tests/modules/ajax_test/ajax_test.module
+++ b/core/modules/system/tests/modules/ajax_test/ajax_test.module
@@ -10,51 +10,8 @@
 use Drupal\Core\Ajax\OpenDialogCommand;
 use Drupal\Core\Ajax\OpenModalDialogCommand;
 use Drupal\Core\Ajax\CloseDialogCommand;
-use Drupal\Core\Ajax\HtmlCommand;
 use Drupal\Core\Url;
 
-/**
- * Menu callback: Returns an element suitable for use by
- * \Drupal\Core\Ajax\AjaxResponse::ajaxRender().
- *
- * Additionally ensures that \Drupal\Core\Ajax\AjaxResponse::ajaxRender()
- * incorporates JavaScript settings generated during the page request by
- * invoking _drupal_add_js() with a dummy setting.
- *
- * @deprecated \Drupal\ajax_test\Controller\AjaxTestController::render()
- */
-function ajax_test_render() {
-  $attached = array(
-    '#attached' => array(
-      'drupalSettings' => array(
-        'ajax' => 'test',
-      ),
-    ),
-  );
-  // @todo Why is this being tested via an explicit drupal_render() call?
-  drupal_render($attached);
-  drupal_process_attached($attached);
-  $response = new AjaxResponse();
-  return $response;
-}
-
-/**
- * Menu callback: Returns an AjaxResponse; settings command set last.
- *
- * Helps verifying AjaxResponse reorders commands to ensure correct execution.
- *
- * @deprecated \Drupal\ajax_test\Controller\AjaxTestController::order()
- */
-function ajax_test_order() {
-  $response = new AjaxResponse();
-  // HTML insertion command.
-  $response->addCommand(new HtmlCommand('body', 'Hello, world!'));
-  $build['#attached']['library'][] = 'ajax_test/order';
-  drupal_process_attached($build);
-
-  return $response;
-}
-
 /**
  * Menu callback: Returns AJAX element with #error property set.
  *
diff --git a/core/modules/system/tests/modules/ajax_test/src/Controller/AjaxTestController.php b/core/modules/system/tests/modules/ajax_test/src/Controller/AjaxTestController.php
index ab25fdee277e..c4d40bdec9c9 100644
--- a/core/modules/system/tests/modules/ajax_test/src/Controller/AjaxTestController.php
+++ b/core/modules/system/tests/modules/ajax_test/src/Controller/AjaxTestController.php
@@ -7,6 +7,8 @@
 
 namespace Drupal\ajax_test\Controller;
 
+use Drupal\Core\Ajax\AjaxResponse;
+use Drupal\Core\Ajax\HtmlCommand;
 use Drupal\Core\Url;
 
 /**
@@ -23,17 +25,38 @@ public function dialogContents() {
   }
 
   /**
-   * @todo Remove ajax_test_render().
+   * Returns a render array that will be rendered by AjaxRenderer.
+   *
+   * Ensures that \Drupal\Core\Ajax\AjaxResponse::ajaxRender()
+   * incorporates JavaScript settings generated during the page request by
+   * adding a dummy setting.
    */
   public function render() {
-    return ajax_test_render();
+    return [
+      '#attached' => [
+        'library' => [
+          'core/drupalSettings',
+        ],
+        'drupalSettings' => [
+          'ajax' => 'test',
+        ],
+      ],
+    ];
   }
 
   /**
-   * @todo Remove ajax_test_order().
+   * Returns an AjaxResponse; settings command set last.
+   *
+   * Helps verifying AjaxResponse reorders commands to ensure correct execution.
    */
   public function order() {
-    return ajax_test_order();
+    $response = new AjaxResponse();
+    // HTML insertion command.
+    $response->addCommand(new HtmlCommand('body', 'Hello, world!'));
+    $build['#attached']['library'][] = 'ajax_test/order';
+    $response->setAttachments($build['#attached']);
+
+    return $response;
   }
 
   /**
diff --git a/core/modules/system/tests/modules/ajax_test/src/Form/AjaxTestDialogForm.php b/core/modules/system/tests/modules/ajax_test/src/Form/AjaxTestDialogForm.php
index 946504d06746..c2a6a079108e 100644
--- a/core/modules/system/tests/modules/ajax_test/src/Form/AjaxTestDialogForm.php
+++ b/core/modules/system/tests/modules/ajax_test/src/Form/AjaxTestDialogForm.php
@@ -101,6 +101,12 @@ protected function dialog($is_modal = FALSE) {
     $response = new AjaxResponse();
     $title = $this->t('AJAX Dialog contents');
     $html = drupal_render($content);
+
+    // Attach the library necessary for using the Open(Modal)DialogCommand and
+    // set the attachments for this Ajax response.
+    $content['#attached']['library'][] = 'core/drupal.dialog.ajax';
+    $response->setAttachments($content['#attached']);
+
     if ($is_modal) {
       $response->addCommand(new OpenModalDialogCommand($title, $html));
     }
diff --git a/core/modules/system/tests/modules/common_test/common_test.module b/core/modules/system/tests/modules/common_test/common_test.module
index c463f20474b0..5c3585619286 100644
--- a/core/modules/system/tests/modules/common_test/common_test.module
+++ b/core/modules/system/tests/modules/common_test/common_test.module
@@ -5,6 +5,8 @@
  * Helper module for the Common tests.
  */
 
+use \Drupal\Core\Asset\AttachedAssetsInterface;
+
 /**
  * Applies #printed to an element to help test #pre_render.
  */
@@ -347,7 +349,7 @@ function common_test_page_attachments_alter(array &$page) {
  *
  * @see \Drupal\system\Tests\Common\JavaScriptTest::testHeaderSetting()
  */
-function common_test_js_settings_alter(&$settings) {
+function common_test_js_settings_alter(&$settings, AttachedAssetsInterface $assets) {
   // Modify an existing setting.
   if (array_key_exists('pluralDelimiter', $settings['locale'])) {
     $settings['locale']['pluralDelimiter'] = '☃';
diff --git a/core/modules/system/theme.api.php b/core/modules/system/theme.api.php
index 82e9e096d73e..0ed5728bfcfb 100644
--- a/core/modules/system/theme.api.php
+++ b/core/modules/system/theme.api.php
@@ -711,12 +711,13 @@ function hook_element_info_alter(array &$types) {
  *
  * @param $javascript
  *   An array of all JavaScript being presented on the page.
+ * @param \Drupal\Core\Asset\AttachedAssetsInterface $assets
+ *   The assets attached to the current response.
  *
- * @see _drupal_add_js()
- * @see drupal_get_js()
  * @see drupal_js_defaults()
+ * @see \Drupal\Core\Asset\AssetResolver
  */
-function hook_js_alter(&$javascript) {
+function hook_js_alter(&$javascript, \Drupal\Core\Asset\AttachedAssetsInterface $assets) {
   // Swap out jQuery to use an updated version of the library.
   $javascript['core/assets/vendor/jquery/jquery.min.js']['data'] = drupal_get_path('module', 'jquery_update') . '/jquery.js';
 }
@@ -791,12 +792,12 @@ function hook_library_info_build() {
  * @param array &$settings
  *   An array of all JavaScript settings (drupalSettings) being presented on the
  *   page.
+ * @param \Drupal\Core\Asset\AttachedAssetsInterface $assets
+ *   The assets attached to the current response.
  *
- * @see _drupal_add_js()
- * @see drupal_get_js()
- * @see drupal_js_defaults()
+ * @see \Drupal\Core\Asset\AssetResolver
  */
-function hook_js_settings_alter(array &$settings) {
+function hook_js_settings_alter(array &$settings, \Drupal\Core\Asset\AttachedAssetsInterface $assets) {
   // Add settings.
   $settings['user']['uid'] = \Drupal::currentUser();
 
@@ -856,11 +857,12 @@ function hook_library_info_alter(&$libraries, $module) {
  *
  * @param $css
  *   An array of all CSS items (files and inline CSS) being requested on the page.
+ * @param \Drupal\Core\Asset\AttachedAssetsInterface $assets
+ *   The assets attached to the current response.
  *
- * @see _drupal_add_css()
- * @see drupal_get_css()
+ * @see Drupal\Core\Asset\LibraryResolverInterface::getCssAssets()
  */
-function hook_css_alter(&$css) {
+function hook_css_alter(&$css, \Drupal\Core\Asset\AttachedAssetsInterface $assets) {
   // Remove defaults.css file.
   unset($css[drupal_get_path('module', 'system') . '/defaults.css']);
 }
diff --git a/core/modules/user/user.module b/core/modules/user/user.module
index d241b922fcf9..86c063d436f8 100644
--- a/core/modules/user/user.module
+++ b/core/modules/user/user.module
@@ -3,6 +3,7 @@
 use Drupal\Component\Utility\Crypt;
 use Drupal\Component\Utility\String;
 use Drupal\Component\Utility\Unicode;
+use Drupal\Core\Asset\AttachedAssetsInterface;
 use Drupal\Core\Entity\Display\EntityViewDisplayInterface;
 use Drupal\Core\Field\BaseFieldDefinition;
 use Drupal\Core\Form\FormStateInterface;
@@ -104,7 +105,7 @@ function user_theme() {
 /**
  * Implements hook_js_settings_alter().
  */
-function user_js_settings_alter(&$settings) {
+function user_js_settings_alter(&$settings, AttachedAssetsInterface $assets) {
   // Provide the user ID in drupalSettings to allow JavaScript code to customize
   // the experience for the end user, rather than the server side, which would
   // break the render cache.
diff --git a/core/modules/views/src/Controller/ViewAjaxController.php b/core/modules/views/src/Controller/ViewAjaxController.php
index 401b0f96f1a5..e938ef81b623 100644
--- a/core/modules/views/src/Controller/ViewAjaxController.php
+++ b/core/modules/views/src/Controller/ViewAjaxController.php
@@ -8,9 +8,11 @@
 namespace Drupal\views\Controller;
 
 use Drupal\Component\Utility\UrlHelper;
+use Drupal\Core\Ajax\AjaxResponse;
 use Drupal\Core\Ajax\ReplaceCommand;
 use Drupal\Core\DependencyInjection\ContainerInjectionInterface;
 use Drupal\Core\Entity\EntityStorageInterface;
+use Drupal\Core\Render\RendererInterface;
 use Drupal\views\Ajax\ScrollTopCommand;
 use Drupal\views\Ajax\ViewAjaxResponse;
 use Drupal\views\ViewExecutableFactory;
@@ -38,6 +40,13 @@ class ViewAjaxController implements ContainerInjectionInterface {
    */
   protected $executableFactory;
 
+  /**
+   * The renderer.
+   *
+   * @var \Drupal\Core\Render\RendererInterface
+   */
+  protected $renderer;
+
   /**
    * Constructs a ViewAjaxController object.
    *
@@ -45,10 +54,13 @@ class ViewAjaxController implements ContainerInjectionInterface {
    *   The entity storage for views.
    * @param \Drupal\views\ViewExecutableFactory $executable_factory
    *   The factory to load a view executable with.
+   * @param \Drupal\Core\Render\RendererInterface $renderer
+   *   The renderer.
    */
-  public function __construct(EntityStorageInterface $storage, ViewExecutableFactory $executable_factory) {
+  public function __construct(EntityStorageInterface $storage, ViewExecutableFactory $executable_factory, RendererInterface $renderer) {
     $this->storage = $storage;
     $this->executableFactory = $executable_factory;
+    $this->renderer = $renderer;
   }
 
   /**
@@ -57,7 +69,8 @@ public function __construct(EntityStorageInterface $storage, ViewExecutableFacto
   public static function create(ContainerInterface $container) {
     return new static(
       $container->get('entity.manager')->getStorage('view'),
-      $container->get('views.executable')
+      $container->get('views.executable'),
+      $container->get('renderer')
     );
   }
 
@@ -96,7 +109,7 @@ public function ajaxView(Request $request) {
 
       // Remove all of this stuff from the query of the request so it doesn't
       // end up in pagers and tablesort URLs.
-      foreach (array('view_name', 'view_display_id', 'view_args', 'view_path', 'view_dom_id', 'pager_element', 'view_base_path', 'ajax_html_ids', 'ajax_page_state') as $key) {
+      foreach (array('view_name', 'view_display_id', 'view_args', 'view_path', 'view_dom_id', 'pager_element', 'view_base_path', 'ajax_html_ids') as $key) {
         $request->query->remove($key);
         $request->request->remove($key);
       }
@@ -138,7 +151,8 @@ public function ajaxView(Request $request) {
         $view->dom_id = $dom_id;
 
         if ($preview = $view->preview($display_id, $args)) {
-          $response->addCommand(new ReplaceCommand(".view-dom-id-$dom_id", $this->drupalRender($preview)));
+          $response->addCommand(new ReplaceCommand(".view-dom-id-$dom_id", $this->renderer->render($preview)));
+          $response->setAttachments($preview['#attached']);
         }
         return $response;
       }
@@ -151,22 +165,4 @@ public function ajaxView(Request $request) {
     }
   }
 
-  /**
-   * Wraps drupal_render.
-   *
-   * @param array $elements
-   *   The structured array describing the data to be rendered.
-   *
-   * @return string
-   *   The rendered HTML.
-   *
-   * @todo Remove once drupal_render is converted to autoloadable code.
-   * @see https://drupal.org/node/2171071
-   */
-  protected function drupalRender(array $elements) {
-    $output = drupal_render($elements);
-    drupal_process_attached($elements);
-    return $output;
-  }
-
 }
diff --git a/core/modules/views/src/Tests/ViewAjaxTest.php b/core/modules/views/src/Tests/ViewAjaxTest.php
index 0420a0a4d84e..5eb004d4b194 100644
--- a/core/modules/views/src/Tests/ViewAjaxTest.php
+++ b/core/modules/views/src/Tests/ViewAjaxTest.php
@@ -50,7 +50,8 @@ public function testAjaxView() {
       'view_name' => 'test_ajax_view',
       'view_display_id' => 'page_1',
     );
-    $response = $this->drupalPost('views/ajax', 'application/json', $post);
+    $post += $this->getAjaxPageStatePostData();
+    $response = $this->drupalPost('views/ajax', 'application/vnd.drupal-ajax', $post);
     $data = Json::decode($response);
 
     // Ensure that the view insert command is part of the result.
diff --git a/core/modules/views/tests/src/Unit/Controller/ViewAjaxControllerTest.php b/core/modules/views/tests/src/Unit/Controller/ViewAjaxControllerTest.php
index 996ac1a40f90..7e09cb49ae5f 100644
--- a/core/modules/views/tests/src/Unit/Controller/ViewAjaxControllerTest.php
+++ b/core/modules/views/tests/src/Unit/Controller/ViewAjaxControllerTest.php
@@ -45,8 +45,15 @@ protected function setUp() {
     $this->executableFactory = $this->getMockBuilder('Drupal\views\ViewExecutableFactory')
       ->disableOriginalConstructor()
       ->getMock();
-
-    $this->viewAjaxController = new TestViewAjaxController($this->viewStorage, $this->executableFactory);
+    $this->renderer = $this->getMock('\Drupal\Core\Render\RendererInterface');
+    $this->renderer->expects($this->any())
+      ->method('render')
+      ->will($this->returnCallback(function(array &$elements) {
+        $elements['#attached'] = [];
+        return isset($elements['#markup']) ? $elements['#markup'] : '';
+      }));
+
+    $this->viewAjaxController = new ViewAjaxController($this->viewStorage, $this->executableFactory, $this->renderer);
   }
 
   /**
@@ -289,18 +296,6 @@ protected function assertViewResultCommand(ViewAjaxResponse $response, $position
 
 }
 
-/**
- * Overrides ViewAjaxController::drupalRender to protect the parent method.
- */
-class TestViewAjaxController extends ViewAjaxController {
-
-  // @todo Remove once drupal_render is converted to autoloadable code.
-  protected function drupalRender(array $elements) {
-    return isset($elements['#markup']) ? $elements['#markup'] : '';
-  }
-
-}
-
 }
 
 namespace {
diff --git a/core/modules/views_ui/src/Form/Ajax/ViewsFormBase.php b/core/modules/views_ui/src/Form/Ajax/ViewsFormBase.php
index de3890b255d7..6dd609e3c876 100644
--- a/core/modules/views_ui/src/Form/Ajax/ViewsFormBase.php
+++ b/core/modules/views_ui/src/Form/Ajax/ViewsFormBase.php
@@ -119,14 +119,6 @@ public function getForm(ViewEntityInterface $view, $display_id, $js) {
       unset($view->form_cache);
     }
 
-    // With the below logic, we may end up rendering a form twice (or two forms
-    // each sharing the same element ids), potentially resulting in
-    // _drupal_add_js() being called twice to add the same setting. drupal_get_js()
-    // is ok with that, but until \Drupal\Core\Ajax\AjaxResponse::ajaxRender()
-    // is (http://drupal.org/node/208611), reset the _drupal_add_js() static
-    // before rendering the second time.
-    $drupal_add_js_original = _drupal_add_js();
-    $drupal_add_js = &drupal_static('_drupal_add_js');
     $form_class = get_class($form_state->getFormObject());
     $response = $this->ajaxFormWrapper($form_class, $form_state);
 
@@ -137,7 +129,6 @@ public function getForm(ViewEntityInterface $view, $display_id, $js) {
 
     // Sometimes we need to re-generate the form for multi-step type operations.
     if (!empty($view->stack)) {
-      $drupal_add_js = $drupal_add_js_original;
       $stack = $view->stack;
       $top = array_shift($stack);
 
@@ -223,6 +214,11 @@ protected function ajaxFormWrapper($form_class, FormStateInterface &$form_state)
       // Ajax command list to execute.
       $response = new AjaxResponse();
 
+      // Attach the library necessary for using the OpenModalDialogCommand and
+      // set the attachments for this Ajax response.
+      $form['#attached']['library'][] = 'core/drupal.dialog.ajax';
+      $response->setAttachments($form['#attached']);
+
       $display = '';
       $status_messages = array('#theme' => 'status_messages');
       if ($messages = drupal_render($status_messages)) {
diff --git a/core/modules/views_ui/src/Tests/RowUITest.php b/core/modules/views_ui/src/Tests/RowUITest.php
index 67e415d3ee7d..eceb3bf9a414 100644
--- a/core/modules/views_ui/src/Tests/RowUITest.php
+++ b/core/modules/views_ui/src/Tests/RowUITest.php
@@ -60,8 +60,20 @@ public function testRowUI() {
     $this->assertEqual($row['options']['test_option'], $random_name, 'Make sure that the custom settings field got saved as expected.');
 
     // Change the row plugin to fields using ajax.
-    $this->drupalPostAjaxForm($row_plugin_url, array('row[type]' => 'fields'), array('op' => 'Apply'), str_replace('/nojs/', '/ajax/', $row_plugin_url));
-    $this->drupalPostAjaxForm(NULL, array(), array('op' => 'Apply'), str_replace('/nojs/', '/ajax/', $row_plugin_url));
+    // Note: this is the best approximation we can achieve, because we cannot
+    // simulate the 'openDialog' command in
+    // WebTestBase::drupalProcessAjaxResponse(), hence we have to make do.
+    $row_plugin_url_ajax = str_replace('/nojs/', '/ajax/', $row_plugin_url);
+    $ajax_settings = [
+      'accepts' => 'application/vnd.drupal-ajax',
+      'submit' => [
+        '_triggering_element_name' => 'op',
+        '_triggering_element_value' => 'Apply',
+      ],
+      'url' => $row_plugin_url_ajax,
+    ];
+    $this->drupalPostAjaxForm($row_plugin_url, ['row[type]' => 'fields'], NULL, $row_plugin_url_ajax, [], [], NULL, $ajax_settings);
+    $this->drupalGet($row_plugin_url);
     $this->assertResponse(200);
     $this->assertFieldByName('row[type]', 'fields', 'Make sure that the fields got saved as used row plugin.');
   }
diff --git a/core/tests/Drupal/Tests/Core/Asset/LibraryDependencyResolverTest.php b/core/tests/Drupal/Tests/Core/Asset/LibraryDependencyResolverTest.php
new file mode 100644
index 000000000000..e10adce0b39c
--- /dev/null
+++ b/core/tests/Drupal/Tests/Core/Asset/LibraryDependencyResolverTest.php
@@ -0,0 +1,177 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Tests\Core\Asset\LibraryDependencyResolverTest.
+ */
+
+namespace Drupal\Tests\Core\Asset;
+
+use Drupal\Core\Asset\LibraryDependencyResolver;
+use Drupal\Tests\UnitTestCase;
+
+/**
+ * @coversDefaultClass \Drupal\Core\Asset\LibraryDependencyResolver
+ * @group Asset
+ */
+class LibraryDependencyResolverTest extends UnitTestCase {
+
+  /**
+   * The tested library dependency resolver.
+   *
+   * @var \Drupal\Core\Asset\LibraryDependencyResolver
+   */
+  protected $libraryDependencyResolver;
+
+  /**
+   * The mocked library discovery service.
+   *
+   * @var \Drupal\Core\Asset\LibraryDiscoveryInterface|\PHPUnit_Framework_MockObject_MockObject
+   */
+  protected $libraryDiscovery;
+
+  /**
+   * The mocked module handler.
+   *
+   * @var \Drupal\Core\Extension\ModuleHandlerInterface|\PHPUnit_Framework_MockObject_MockObject
+   */
+  protected $moduleHandler;
+
+  /**
+   * Test library data.
+   *
+   * @var array
+   */
+  protected $libraryData = array(
+    'no_deps_a' => ['js' => [], 'css' => []],
+    'no_deps_b' => ['js' => [], 'css' => []],
+    'no_deps_c' => ['js' => [], 'css' => []],
+    'deps_a' => ['js' => [], 'css' => [], 'dependencies' => ['test/no_deps_a']],
+    'deps_b' => ['js' => [], 'css' => [], 'dependencies' => ['test/no_deps_a', 'test/no_deps_b']],
+    'deps_c' => ['js' => [], 'css' => [], 'dependencies' => ['test/no_deps_b', 'test/no_deps_a']],
+    'nested_deps_a' => ['js' => [], 'css' => [], 'dependencies' => ['test/deps_a']],
+    'nested_deps_b' => ['js' => [], 'css' => [], 'dependencies' => ['test/nested_deps_a']],
+    'nested_deps_c' => ['js' => [], 'css' => [], 'dependencies' => ['test/nested_deps_b']],
+  );
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function setUp() {
+    $this->libraryDiscovery = $this->getMockBuilder('Drupal\Core\Asset\LibraryDiscovery')
+      ->disableOriginalConstructor()
+      ->setMethods(['getLibrariesByExtension'])
+      ->getMock();
+    $this->libraryDiscovery->expects($this->any())
+      ->method('getLibrariesByExtension')
+      ->with('test')
+      ->will($this->returnValue($this->libraryData));
+    $this->libraryDependencyResolver= new LibraryDependencyResolver($this->libraryDiscovery);
+  }
+
+
+  /**
+   * Provides test data for ::testGetLibrariesWithDependencies().
+   */
+  public function providerTestGetLibrariesWithDependencies() {
+    return [
+      // Empty list of libraries.
+      [[], []],
+      // Without dependencies.
+      [['test/no_deps_a'], ['test/no_deps_a']],
+      [['test/no_deps_a', 'test/no_deps_b'], ['test/no_deps_a', 'test/no_deps_b']],
+      [['test/no_deps_b', 'test/no_deps_a'], ['test/no_deps_b', 'test/no_deps_a']],
+      // Single-level (direct) dependencies.
+      [['test/deps_a'], ['test/no_deps_a', 'test/deps_a']],
+      [['test/deps_b'], ['test/no_deps_a', 'test/no_deps_b', 'test/deps_b']],
+      [['test/deps_c'], ['test/no_deps_b', 'test/no_deps_a', 'test/deps_c']],
+      [['test/deps_a', 'test/deps_b'], ['test/no_deps_a', 'test/deps_a', 'test/no_deps_b', 'test/deps_b']],
+      [['test/deps_a', 'test/deps_c'], ['test/no_deps_a', 'test/deps_a', 'test/no_deps_b', 'test/deps_c']],
+      [['test/deps_a', 'test/deps_b', 'test/deps_c'], ['test/no_deps_a', 'test/deps_a', 'test/no_deps_b', 'test/deps_b', 'test/deps_c']],
+      [['test/deps_b', 'test/deps_a'], ['test/no_deps_a', 'test/no_deps_b', 'test/deps_b', 'test/deps_a']],
+      [['test/deps_b', 'test/deps_c'], ['test/no_deps_a', 'test/no_deps_b', 'test/deps_b', 'test/deps_c']],
+      [['test/deps_c', 'test/deps_b'], ['test/no_deps_b', 'test/no_deps_a', 'test/deps_c', 'test/deps_b']],
+      // Multi-level (indirect) dependencies.
+      [['test/nested_deps_a'], ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a']],
+      [['test/nested_deps_b'], ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b']],
+      [['test/nested_deps_c'], ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b', 'test/nested_deps_c']],
+      [['test/nested_deps_a', 'test/nested_deps_b'], ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b']],
+      [['test/nested_deps_b', 'test/nested_deps_a'], ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b']],
+      [['test/nested_deps_a', 'test/nested_deps_c'],                       ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b', 'test/nested_deps_c']],
+      [['test/nested_deps_b', 'test/nested_deps_c'],                       ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b', 'test/nested_deps_c']],
+      [['test/nested_deps_c', 'test/nested_deps_a'],                       ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b', 'test/nested_deps_c']],
+      [['test/nested_deps_a', 'test/nested_deps_b', 'test/nested_deps_c'], ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b', 'test/nested_deps_c']],
+      [['test/nested_deps_a', 'test/nested_deps_c', 'test/nested_deps_b'], ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b', 'test/nested_deps_c']],
+      [['test/nested_deps_b', 'test/nested_deps_a', 'test/nested_deps_c'], ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b', 'test/nested_deps_c']],
+      [['test/nested_deps_b', 'test/nested_deps_c', 'test/nested_deps_a'], ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b', 'test/nested_deps_c']],
+      [['test/nested_deps_c', 'test/nested_deps_a', 'test/nested_deps_b'], ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b', 'test/nested_deps_c']],
+      [['test/nested_deps_c', 'test/nested_deps_b', 'test/nested_deps_a'], ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b', 'test/nested_deps_c']],
+      // Complex dependencies, combining the above, with many intersections.
+      [['test/deps_c', 'test/nested_deps_b'],                   ['test/no_deps_b', 'test/no_deps_a', 'test/deps_c', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b']],
+      [['test/no_deps_a', 'test/deps_c', 'test/nested_deps_b'], ['test/no_deps_a', 'test/no_deps_b', 'test/deps_c', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b']],
+      [['test/nested_deps_b', 'test/deps_c', 'test/no_deps_c'], ['test/no_deps_a', 'test/deps_a', 'test/nested_deps_a', 'test/nested_deps_b', 'test/no_deps_b', 'test/deps_c', 'test/no_deps_c']],
+    ];
+  }
+
+  /**
+   * @covers ::getLibrariesWithDependencies()
+   *
+   * @dataProvider providerTestGetLibrariesWithDependencies
+   */
+  public function testGetLibrariesWithDependencies(array $libraries, array $expected) {
+    $this->assertEquals($expected, $this->libraryDependencyResolver->getLibrariesWithDependencies($libraries));
+  }
+
+  /**
+   * Provides test data for ::testGetMinimalRepresentativeSubset().
+   */
+  public function providerTestGetMinimalRepresentativeSubset() {
+    return [
+      // Empty list of libraries.
+      [[], []],
+      // Without dependencies.
+      [['test/no_deps_a'], ['test/no_deps_a']],
+      [['test/no_deps_a', 'test/no_deps_b'], ['test/no_deps_a', 'test/no_deps_b']],
+      [['test/no_deps_b', 'test/no_deps_a'], ['test/no_deps_b', 'test/no_deps_a']],
+      // Single-level (direct) dependencies.
+      [['test/deps_a'], ['test/deps_a']],
+      [['test/deps_b'], ['test/deps_b']],
+      [['test/deps_c'], ['test/deps_c']],
+      [['test/deps_a', 'test/deps_b'], ['test/deps_a', 'test/deps_b']],
+      [['test/deps_a', 'test/deps_c'], ['test/deps_a', 'test/deps_c']],
+      [['test/deps_a', 'test/deps_b', 'test/deps_c'], ['test/deps_a', 'test/deps_b', 'test/deps_c']],
+      [['test/deps_b', 'test/deps_a'], ['test/deps_b', 'test/deps_a']],
+      [['test/deps_b', 'test/deps_c'], ['test/deps_b', 'test/deps_c']],
+      [['test/deps_c', 'test/deps_b'], ['test/deps_c', 'test/deps_b']],
+      // Multi-level (indirect) dependencies.
+      [['test/nested_deps_a'], ['test/nested_deps_a']],
+      [['test/nested_deps_b'], ['test/nested_deps_b']],
+      [['test/nested_deps_c'], ['test/nested_deps_c']],
+      [['test/nested_deps_a', 'test/nested_deps_b'], ['test/nested_deps_b']],
+      [['test/nested_deps_b', 'test/nested_deps_a'], ['test/nested_deps_b']],
+      [['test/nested_deps_a', 'test/nested_deps_c'],                       ['test/nested_deps_c']],
+      [['test/nested_deps_b', 'test/nested_deps_c'],                       ['test/nested_deps_c']],
+      [['test/nested_deps_c', 'test/nested_deps_a'],                       ['test/nested_deps_c']],
+      [['test/nested_deps_a', 'test/nested_deps_b', 'test/nested_deps_c'], ['test/nested_deps_c']],
+      [['test/nested_deps_a', 'test/nested_deps_c', 'test/nested_deps_b'], ['test/nested_deps_c']],
+      [['test/nested_deps_b', 'test/nested_deps_a', 'test/nested_deps_c'], ['test/nested_deps_c']],
+      [['test/nested_deps_b', 'test/nested_deps_c', 'test/nested_deps_a'], ['test/nested_deps_c']],
+      [['test/nested_deps_c', 'test/nested_deps_a', 'test/nested_deps_b'], ['test/nested_deps_c']],
+      [['test/nested_deps_c', 'test/nested_deps_b', 'test/nested_deps_a'], ['test/nested_deps_c']],
+      // Complex dependencies, combining the above, with many intersections.
+      [['test/deps_c', 'test/nested_deps_b'],                   ['test/deps_c', 'test/nested_deps_b']],
+      [['test/no_deps_a', 'test/deps_c', 'test/nested_deps_b'], ['test/deps_c', 'test/nested_deps_b']],
+      [['test/nested_deps_b', 'test/deps_c', 'test/no_deps_c'], ['test/nested_deps_b', 'test/deps_c', 'test/no_deps_c']],
+    ];
+  }
+
+  /**
+   * @covers ::getMinimalRepresentativeSubset()
+   *
+   * @dataProvider providerTestGetMinimalRepresentativeSubset
+   */
+  public function testGetMinimalRepresentativeSubset(array $libraries, array $expected) {
+    $this->assertEquals($expected, $this->libraryDependencyResolver->getMinimalRepresentativeSubset($libraries));
+  }
+
+}
diff --git a/core/tests/Drupal/Tests/Core/Controller/AjaxRendererTest.php b/core/tests/Drupal/Tests/Core/Controller/AjaxRendererTest.php
index bb5b823b54cc..bac29641794d 100644
--- a/core/tests/Drupal/Tests/Core/Controller/AjaxRendererTest.php
+++ b/core/tests/Drupal/Tests/Core/Controller/AjaxRendererTest.php
@@ -70,6 +70,7 @@ class TestAjaxRenderer extends AjaxRenderer {
    * {@inheritdoc}
    */
   protected function drupalRenderRoot(&$elements, $is_root_call = FALSE) {
+    $elements += ['#attached' => []];
     if (isset($elements['#markup'])) {
       return $elements['#markup'];
     }
-- 
GitLab