CssCollectionRenderer::render

public CssCollectionRenderer::render(array $css_assets)

Renders an asset collection.

Parameters

array $assets: An asset collection.

Return value

array A render array to render the asset collection.

Overrides AssetCollectionRendererInterface::render

File

core/lib/Drupal/Core/Asset/CssCollectionRenderer.php, line 76

Class

CssCollectionRenderer
Renders CSS assets.

Namespace

Drupal\Core\Asset

Code

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
public function render(array $css_assets) {
  $elements = array();
 
  // 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.
  $query_string = $this->state->get('system.css_js_query_string') ? : '0';
 
  // Defaults for LINK and STYLE elements.
  $link_element_defaults = array(
    '#type' => 'html_tag',
    '#tag' => 'link',
    '#attributes' => array(
      'rel' => 'stylesheet',
    ),
  );
  $style_element_defaults = array(
    '#type' => 'html_tag',
    '#tag' => 'style',
  );
 
  // For filthy IE hack.
  $current_ie_group_keys = NULL;
  $get_ie_group_key = function($css_asset) {
    return array($css_asset['type'], $css_asset['preprocess'], $css_asset['group'], $css_asset['media'], $css_asset['browsers']);
  };
 
  // Loop through all CSS assets, by key, to allow for the special IE
  // workaround.
  $css_assets_keys = array_keys($css_assets);
  for ($i = 0; $i < count($css_assets_keys); $i++) {
    $css_asset = $css_assets[$css_assets_keys[$i]];
    switch ($css_asset['type']) {
      // For file items, there are three possibilities.
      // - There are up to 31 CSS assets on the page (some of which may be
      //   aggregated). In this case, output a LINK tag for file CSS assets.
      // - There are more than 31 CSS assets on the page, yet we must stay
      //   below IE<10's limit of 31 total CSS inclusion tags, we handle this
      //   in two ways:
      //    - file CSS assets that are not eligible for aggregation (their
      //      'preprocess' flag has been set to FALSE): in this case, output a
      //      LINK tag.
      //    - file CSS assets that can be aggregated (and possibly have been):
      //      in this case, figure out which subsequent file CSS assets share
      //      the same key properties ('group', 'media' and 'browsers') and
      //      output this group into as few STYLE tags as possible (a STYLE
      //      tag may contain only 31 @import statements).
      case 'file':
        // The dummy query string needs to be added to the URL to control
        // browser-caching.
        $query_string_separator = (strpos($css_asset['data'], '?') !== FALSE) ? '&' : '?';
 
        // As long as the current page will not run into IE's limit for CSS
        // assets: output a LINK tag for a file CSS asset.
        if (count($css_assets) <= 31) {
          $element = $link_element_defaults;
          $element['#attributes']['href'] = file_url_transform_relative(file_create_url($css_asset['data'])) . $query_string_separator . $query_string;
          $element['#attributes']['media'] = $css_asset['media'];
          $element['#browsers'] = $css_asset['browsers'];
          $elements[] = $element;
        }
        // The current page will run into IE's limits for CSS assets: work
        // around these limits by performing a light form of grouping.
        // Once Drupal only needs to support IE10 and later, we can drop this.
        else {
          // The file CSS asset is ineligible for aggregation: output it in a
          // LINK tag.
          if (!$css_asset['preprocess']) {
            $element = $link_element_defaults;
            $element['#attributes']['href'] = file_url_transform_relative(file_create_url($css_asset['data'])) . $query_string_separator . $query_string;
            $element['#attributes']['media'] = $css_asset['media'];
            $element['#browsers'] = $css_asset['browsers'];
            $elements[] = $element;
          }
          // The file CSS asset can be aggregated, but hasn't been: combine
          // multiple items into as few STYLE tags as possible.
          else {
            $import = array();
            // Start with the current CSS asset, iterate over subsequent CSS
            // assets and find which ones have the same 'type', 'group',
            // 'preprocess', 'media' and 'browsers' properties.
            $j = $i;
            $next_css_asset = $css_asset;
            $current_ie_group_key = $get_ie_group_key($css_asset);
            do {
              // The dummy query string needs to be added to the URL to
              // control browser-caching. IE7 does not support a media type on
              // the @import statement, so we instead specify the media for
              // the group on the STYLE tag.
              $import[] = '@import url("' . Html::escape(file_url_transform_relative(file_create_url($next_css_asset['data'])) . '?' . $query_string) . '");';
              // Move the outer for loop skip the next item, since we
              // processed it here.
              $i = $j;
              // Retrieve next CSS asset, unless there is none: then break.
              if ($j + 1 < count($css_assets_keys)) {
                $j++;
                $next_css_asset = $css_assets[$css_assets_keys[$j]];
              }
              else {
                break;
              }
            } while ($get_ie_group_key($next_css_asset) == $current_ie_group_key);
 
            // In addition to IE's limit of 31 total CSS inclusion tags, it
            // also has a limit of 31 @import statements per STYLE tag.
            while (!empty($import)) {
              $import_batch = array_slice($import, 0, 31);
              $import = array_slice($import, 31);
              $element = $style_element_defaults;
              // This simplifies the JavaScript regex, allowing each line
              // (separated by \n) to be treated as a completely different
              // string. This means that we can use ^ and $ on one line at a
              // time, and not worry about style tags since they'll never
              // match the regex.
              $element['#value'] = "\n" . implode("\n", $import_batch) . "\n";
              $element['#attributes']['media'] = $css_asset['media'];
              $element['#browsers'] = $css_asset['browsers'];
              $elements[] = $element;
            }
          }
        }
        break;
 
        // Output a LINK tag for an external CSS asset. The asset's 'data'
        // property contains the full URL.
      case 'external':
        $element = $link_element_defaults;
        $element['#attributes']['href'] = $css_asset['data'];
        $element['#attributes']['media'] = $css_asset['media'];
        $element['#browsers'] = $css_asset['browsers'];
        $elements[] = $element;
        break;
 
      default:
        throw new \Exception('Invalid CSS asset type.');
    }
  }
 
  return $elements;
}
doc_Drupal
2016-10-29 08:59:16
Comments
Leave a Comment

Please login to continue.