1 | /**
|
2 | * @license
|
3 | * Copyright Google LLC All Rights Reserved.
|
4 | *
|
5 | * Use of this source code is governed by an MIT-style license that can be
|
6 | * found in the LICENSE file at https://angular.io/license
|
7 | */
|
8 | import { DOCUMENT } from '@angular/common';
|
9 | import { BEFORE_APP_SERIALIZED } from '@angular/platform-server';
|
10 | import { BREAKPOINTS, CLASS_NAME, SERVER_TOKEN, ɵMatchMedia as MatchMedia, StylesheetMap, sortAscendingPriority, MediaMarshaller, } from '@angular/flex-layout/core';
|
11 | import { ServerMatchMedia } from './server-match-media';
|
12 | /**
|
13 | * Activate all the registered breakpoints in sequence, and then
|
14 | * retrieve the associated stylings from the virtual stylesheet
|
15 | * @param serverSheet the virtual stylesheet that stores styles for each
|
16 | * element
|
17 | * @param mediaController the MatchMedia service to activate/deactivate breakpoints
|
18 | * @param breakpoints the registered breakpoints to activate/deactivate
|
19 | * @param mediaMarshaller the MediaMarshaller service to disable fallback styles dynamically
|
20 | */
|
21 | export function generateStaticFlexLayoutStyles(serverSheet, mediaController, breakpoints, mediaMarshaller) {
|
22 | // Store the custom classes in the following map, that way only
|
23 | // one class gets allocated per HTMLElement, and each class can
|
24 | // be referenced in the static media queries
|
25 | const classMap = new Map();
|
26 | // Get the initial stylings for all the directives,
|
27 | // and initialize the fallback block of stylings.
|
28 | const defaultStyles = new Map(serverSheet.stylesheet);
|
29 | // Reset the class counter, otherwise class numbers will
|
30 | // increase with each server render.
|
31 | nextId = 0;
|
32 | let styleText = generateCss(defaultStyles, 'all', classMap);
|
33 | mediaMarshaller.useFallbacks = false;
|
34 | [...breakpoints].sort(sortAscendingPriority).forEach((bp) => {
|
35 | serverSheet.clearStyles();
|
36 | mediaController.activateBreakpoint(bp);
|
37 | const stylesheet = new Map(serverSheet.stylesheet);
|
38 | if (stylesheet.size > 0) {
|
39 | styleText += generateCss(stylesheet, bp.mediaQuery, classMap);
|
40 | }
|
41 | mediaController.deactivateBreakpoint(bp);
|
42 | });
|
43 | return styleText;
|
44 | }
|
45 | /**
|
46 | * Create a style tag populated with the dynamic stylings from Flex
|
47 | * components and attach it to the head of the DOM
|
48 | */
|
49 | export function FLEX_SSR_SERIALIZER_FACTORY(serverSheet, mediaController, _document, breakpoints, mediaMarshaller) {
|
50 | return () => {
|
51 | // This is the style tag that gets inserted into the head of the DOM,
|
52 | // populated with the manual media queries
|
53 | const styleTag = _document.createElement('style');
|
54 | const styleText = generateStaticFlexLayoutStyles(serverSheet, mediaController, breakpoints, mediaMarshaller);
|
55 | styleTag.classList.add(`${CLASS_NAME}ssr`);
|
56 | styleTag.textContent = styleText;
|
57 | _document.head.appendChild(styleTag);
|
58 | };
|
59 | }
|
60 | /**
|
61 | * Provider to set static styles on the server
|
62 | */
|
63 | export const SERVER_PROVIDERS = [
|
64 | {
|
65 | provide: BEFORE_APP_SERIALIZED,
|
66 | useFactory: FLEX_SSR_SERIALIZER_FACTORY,
|
67 | deps: [
|
68 | StylesheetMap,
|
69 | MatchMedia,
|
70 | DOCUMENT,
|
71 | BREAKPOINTS,
|
72 | MediaMarshaller,
|
73 | ],
|
74 | multi: true,
|
75 | },
|
76 | {
|
77 | provide: SERVER_TOKEN,
|
78 | useValue: true
|
79 | },
|
80 | {
|
81 | provide: MatchMedia,
|
82 | useClass: ServerMatchMedia
|
83 | }
|
84 | ];
|
85 | let nextId = 0;
|
86 | const IS_DEBUG_MODE = false;
|
87 | /**
|
88 | * create @media queries based on a virtual stylesheet
|
89 | * * Adds a unique class to each element and stores it
|
90 | * in a shared classMap for later reuse
|
91 | * @param stylesheet the virtual stylesheet that stores styles for each
|
92 | * element
|
93 | * @param mediaQuery the given @media CSS selector for the current breakpoint
|
94 | * @param classMap the map of HTML elements to class names to avoid duplications
|
95 | */
|
96 | function generateCss(stylesheet, mediaQuery, classMap) {
|
97 | let css = '';
|
98 | stylesheet.forEach((styles, el) => {
|
99 | let keyVals = '';
|
100 | let className = getClassName(el, classMap);
|
101 | styles.forEach((v, k) => {
|
102 | keyVals += v ? format(`${k}:${v};`) : '';
|
103 | });
|
104 | if (keyVals) {
|
105 | // Build list of CSS styles; each with a className
|
106 | css += format(`.${className} {`, keyVals, '}');
|
107 | }
|
108 | });
|
109 | // Group 1 or more styles (each with className) in a specific mediaQuery
|
110 | return format(`@media ${mediaQuery} {`, css, '}');
|
111 | }
|
112 | /**
|
113 | * For debugging purposes, prefix css segment with linefeed(s) for easy
|
114 | * debugging purposes.
|
115 | */
|
116 | function format(...list) {
|
117 | let result = '';
|
118 | list.forEach((css, i) => {
|
119 | result += IS_DEBUG_MODE ? formatSegment(css, i !== 0) : css;
|
120 | });
|
121 | return result;
|
122 | }
|
123 | function formatSegment(css, asPrefix = true) {
|
124 | return asPrefix ? `\n${css}` : `${css}\n`;
|
125 | }
|
126 | /**
|
127 | * Get className associated with CSS styling
|
128 | * If not found, generate global className and set
|
129 | * association.
|
130 | */
|
131 | function getClassName(element, classMap) {
|
132 | let className = classMap.get(element);
|
133 | if (!className) {
|
134 | className = `${CLASS_NAME}${nextId++}`;
|
135 | classMap.set(element, className);
|
136 | }
|
137 | element.classList.add(className);
|
138 | return className;
|
139 | }
|
140 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"server-provider.js","sourceRoot":"","sources":["../../../../../projects/libs/flex-layout/server/server-provider.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nimport {DOCUMENT} from '@angular/common';\nimport {BEFORE_APP_SERIALIZED} from '@angular/platform-server';\nimport {\n  BREAKPOINTS,\n  CLASS_NAME,\n  SERVER_TOKEN,\n  BreakPoint,\n  ɵMatchMedia as MatchMedia,\n  StylesheetMap,\n  sortAscendingPriority,\n  MediaMarshaller,\n} from '@angular/flex-layout/core';\n\nimport {ServerMatchMedia} from './server-match-media';\n\n/**\n * Activate all the registered breakpoints in sequence, and then\n * retrieve the associated stylings from the virtual stylesheet\n * @param serverSheet the virtual stylesheet that stores styles for each\n *        element\n * @param mediaController the MatchMedia service to activate/deactivate breakpoints\n * @param breakpoints the registered breakpoints to activate/deactivate\n * @param mediaMarshaller the MediaMarshaller service to disable fallback styles dynamically\n */\nexport function generateStaticFlexLayoutStyles(serverSheet: StylesheetMap,\n                                               mediaController: ServerMatchMedia,\n                                               breakpoints: BreakPoint[],\n                                               mediaMarshaller: MediaMarshaller) {\n  // Store the custom classes in the following map, that way only\n  // one class gets allocated per HTMLElement, and each class can\n  // be referenced in the static media queries\n  const classMap = new Map<HTMLElement, string>();\n\n  // Get the initial stylings for all the directives,\n  // and initialize the fallback block of stylings.\n  const defaultStyles = new Map(serverSheet.stylesheet);\n  // Reset the class counter, otherwise class numbers will\n  // increase with each server render.\n  nextId = 0;\n  let styleText = generateCss(defaultStyles, 'all', classMap);\n  mediaMarshaller.useFallbacks = false;\n\n  [...breakpoints].sort(sortAscendingPriority).forEach((bp) => {\n    serverSheet.clearStyles();\n    mediaController.activateBreakpoint(bp);\n    const stylesheet = new Map(serverSheet.stylesheet);\n    if (stylesheet.size > 0) {\n      styleText += generateCss(stylesheet, bp.mediaQuery, classMap);\n    }\n    mediaController.deactivateBreakpoint(bp);\n  });\n\n  return styleText;\n}\n\n/**\n * Create a style tag populated with the dynamic stylings from Flex\n * components and attach it to the head of the DOM\n */\nexport function FLEX_SSR_SERIALIZER_FACTORY(serverSheet: StylesheetMap,\n                                            mediaController: ServerMatchMedia,\n                                            _document: Document,\n                                            breakpoints: BreakPoint[],\n                                            mediaMarshaller: MediaMarshaller) {\n  return () => {\n    // This is the style tag that gets inserted into the head of the DOM,\n    // populated with the manual media queries\n    const styleTag = _document.createElement('style');\n    const styleText = generateStaticFlexLayoutStyles(\n      serverSheet, mediaController, breakpoints, mediaMarshaller);\n    styleTag.classList.add(`${CLASS_NAME}ssr`);\n    styleTag.textContent = styleText;\n    _document.head!.appendChild(styleTag);\n  };\n}\n\n/**\n *  Provider to set static styles on the server\n */\nexport const SERVER_PROVIDERS = [\n  {\n    provide: BEFORE_APP_SERIALIZED,\n    useFactory: FLEX_SSR_SERIALIZER_FACTORY,\n    deps: [\n      StylesheetMap,\n      MatchMedia,\n      DOCUMENT,\n      BREAKPOINTS,\n      MediaMarshaller,\n    ],\n    multi: true,\n  },\n  {\n    provide: SERVER_TOKEN,\n    useValue: true\n  },\n  {\n    provide: MatchMedia,\n    useClass: ServerMatchMedia\n  }\n];\n\n\nlet nextId = 0;\nconst IS_DEBUG_MODE = false;\n\nexport type StyleSheet = Map<HTMLElement, Map<string, string|number>>;\nexport type ClassMap = Map<HTMLElement, string>;\n\n/**\n * create @media queries based on a virtual stylesheet\n * * Adds a unique class to each element and stores it\n *   in a shared classMap for later reuse\n * @param stylesheet the virtual stylesheet that stores styles for each\n *        element\n * @param mediaQuery the given @media CSS selector for the current breakpoint\n * @param classMap the map of HTML elements to class names to avoid duplications\n */\nfunction generateCss(stylesheet: StyleSheet, mediaQuery: string, classMap: ClassMap) {\n  let css = '';\n  stylesheet.forEach((styles, el) => {\n    let keyVals = '';\n    let className = getClassName(el, classMap);\n\n    styles.forEach((v, k) => {\n      keyVals += v ? format(`${k}:${v};`) : '';\n    });\n\n    if (keyVals) {\n      // Build list of CSS styles; each with a className\n      css += format(`.${className} {`, keyVals, '}');\n    }\n  });\n\n  // Group 1 or more styles (each with className) in a specific mediaQuery\n  return format(`@media ${mediaQuery} {`, css, '}');\n}\n\n/**\n * For debugging purposes, prefix css segment with linefeed(s) for easy\n  * debugging purposes.\n */\nfunction format(...list: string[]): string {\n  let result = '';\n  list.forEach((css, i) => {\n    result += IS_DEBUG_MODE ? formatSegment(css, i !== 0) : css;\n  });\n  return result;\n}\n\nfunction formatSegment(css: string, asPrefix: boolean = true): string {\n  return asPrefix ? `\\n${css}` : `${css}\\n`;\n}\n\n/**\n * Get className associated with CSS styling\n * If not found, generate global className and set\n * association.\n */\nfunction getClassName(element: HTMLElement, classMap: Map<HTMLElement, string>) {\n  let className = classMap.get(element);\n  if (!className) {\n    className = `${CLASS_NAME}${nextId++}`;\n    classMap.set(element, className);\n  }\n  element.classList.add(className);\n\n  return className;\n}\n"]} |
\ | No newline at end of file |