Skip to content

Commit 18cfa04

Browse files
committed
feat(@angular-devkit/build-angular): add support to inline Adobe Fonts
With this change we add support to inline external Adobe fonts into the index html, we also add a `preconnect` hint which helps improve page load speed. Closes #21186
1 parent 47a1ccc commit 18cfa04

File tree

3 files changed

+230
-111
lines changed

3 files changed

+230
-111
lines changed

packages/angular_devkit/build_angular/src/browser/schema.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -108,7 +108,7 @@
108108
"properties": {
109109
"inline": {
110110
"type": "boolean",
111-
"description": "Reduce render blocking requests by inlining external Google fonts and icons CSS definitions in the application's HTML index file. This option requires internet access. `HTTPS_PROXY` environment variable can be used to specify a proxy server.",
111+
"description": "Reduce render blocking requests by inlining external Google Fonts and Adobe Fonts CSS definitions in the application's HTML index file. This option requires internet access. `HTTPS_PROXY` environment variable can be used to specify a proxy server.",
112112
"default": true
113113
}
114114
},

packages/angular_devkit/build_angular/src/utils/index-file/inline-fonts.ts

+130-59
Original file line numberDiff line numberDiff line change
@@ -25,18 +25,33 @@ const enum UserAgent {
2525
IE = 'Mozilla/5.0 (Windows NT 10.0; Trident/7.0; rv:11. 0) like Gecko',
2626
}
2727

28-
const SUPPORTED_PROVIDERS = ['fonts.googleapis.com'];
28+
interface FontProviderDetails {
29+
preconnectUrl: string;
30+
seperateRequestForWOFF: boolean;
31+
}
2932

3033
export interface InlineFontsOptions {
3134
minify?: boolean;
3235
WOFFSupportNeeded: boolean;
3336
}
3437

38+
const SUPPORTED_PROVIDERS: Record<string, FontProviderDetails> = {
39+
'fonts.googleapis.com': {
40+
seperateRequestForWOFF: true,
41+
preconnectUrl: 'https://fonts.gstatic.com',
42+
},
43+
'use.typekit.net': {
44+
seperateRequestForWOFF: false,
45+
preconnectUrl: 'https://use.typekit.net',
46+
},
47+
};
48+
3549
export class InlineFontsProcessor {
3650
constructor(private options: InlineFontsOptions) {}
3751

3852
async process(content: string): Promise<string> {
3953
const hrefList: string[] = [];
54+
const existingPreconnect = new Set<string>();
4055

4156
// Collector link tags with href
4257
const { rewriter: collectorStream } = await htmlRewritingStream(content);
@@ -48,20 +63,63 @@ export class InlineFontsProcessor {
4863
return;
4964
}
5065

51-
// <link tag with rel="stylesheet" and a href.
52-
const href =
53-
attrs.find(({ name, value }) => name === 'rel' && value === 'stylesheet') &&
54-
attrs.find(({ name }) => name === 'href')?.value;
55-
56-
if (href) {
57-
hrefList.push(href);
66+
let hrefValue: string | undefined;
67+
let relValue: string | undefined;
68+
for (const { name, value } of attrs) {
69+
switch (name) {
70+
case 'rel':
71+
relValue = value;
72+
break;
73+
74+
case 'href':
75+
hrefValue = value;
76+
break;
77+
}
78+
79+
if (hrefValue && relValue) {
80+
switch (relValue) {
81+
case 'stylesheet':
82+
// <link rel="stylesheet" href="https://example.com/main.css">
83+
hrefList.push(hrefValue);
84+
break;
85+
86+
case 'preconnect':
87+
// <link rel="preconnect" href="https://example.com">
88+
existingPreconnect.add(hrefValue.replace(/\/$/, ''));
89+
break;
90+
}
91+
92+
return;
93+
}
5894
}
5995
});
6096

6197
await new Promise((resolve) => collectorStream.on('finish', resolve));
6298

6399
// Download stylesheets
64-
const hrefsContent = await this.processHrefs(hrefList);
100+
const hrefsContent = new Map<string, string>();
101+
const newPreconnectUrls = new Set<string>();
102+
103+
for (const hrefItem of hrefList) {
104+
const url = this.createNormalizedUrl(hrefItem);
105+
if (!url) {
106+
continue;
107+
}
108+
109+
const content = await this.processHref(url);
110+
if (content === undefined) {
111+
continue;
112+
}
113+
114+
hrefsContent.set(hrefItem, content);
115+
116+
// Add preconnect
117+
const preconnectUrl = this.getFontProviderDetails(url)?.preconnectUrl;
118+
if (preconnectUrl && !existingPreconnect.has(preconnectUrl)) {
119+
newPreconnectUrls.add(preconnectUrl);
120+
}
121+
}
122+
65123
if (hrefsContent.size === 0) {
66124
return content;
67125
}
@@ -71,21 +129,31 @@ export class InlineFontsProcessor {
71129
rewriter.on('startTag', (tag) => {
72130
const { tagName, attrs } = tag;
73131

74-
if (tagName !== 'link') {
75-
rewriter.emitStartTag(tag);
76-
77-
return;
78-
}
79-
80-
const hrefAttr =
81-
attrs.some(({ name, value }) => name === 'rel' && value === 'stylesheet') &&
82-
attrs.find(({ name, value }) => name === 'href' && hrefsContent.has(value));
83-
if (hrefAttr) {
84-
const href = hrefAttr.value;
85-
const cssContent = hrefsContent.get(href);
86-
rewriter.emitRaw(`<style type="text/css">${cssContent}</style>`);
87-
} else {
88-
rewriter.emitStartTag(tag);
132+
switch (tagName) {
133+
case 'head':
134+
rewriter.emitStartTag(tag);
135+
for (const url of newPreconnectUrls) {
136+
rewriter.emitRaw(`<link rel="preconnect" href="${url}" crossorigin>`);
137+
}
138+
break;
139+
140+
case 'link':
141+
const hrefAttr =
142+
attrs.some(({ name, value }) => name === 'rel' && value === 'stylesheet') &&
143+
attrs.find(({ name, value }) => name === 'href' && hrefsContent.has(value));
144+
if (hrefAttr) {
145+
const href = hrefAttr.value;
146+
const cssContent = hrefsContent.get(href);
147+
rewriter.emitRaw(`<style type="text/css">${cssContent}</style>`);
148+
} else {
149+
rewriter.emitStartTag(tag);
150+
}
151+
break;
152+
153+
default:
154+
rewriter.emitStartTag(tag);
155+
156+
break;
89157
}
90158
});
91159

@@ -152,47 +220,50 @@ export class InlineFontsProcessor {
152220
return data;
153221
}
154222

155-
private async processHrefs(hrefList: string[]): Promise<Map<string, string>> {
156-
const hrefsContent = new Map<string, string>();
223+
private async processHref(url: URL): Promise<string | undefined> {
224+
const provider = this.getFontProviderDetails(url);
225+
if (!provider) {
226+
return undefined;
227+
}
157228

158-
for (const hrefPath of hrefList) {
159-
// Need to convert '//' to 'https://' because the URL parser will fail with '//'.
160-
const normalizedHref = hrefPath.startsWith('//') ? `https:${hrefPath}` : hrefPath;
161-
if (!normalizedHref.startsWith('http')) {
162-
// Non valid URL.
163-
// Example: relative path styles.css.
164-
continue;
165-
}
229+
// The order IE -> Chrome is important as otherwise Chrome will load woff1.
230+
let cssContent = '';
231+
if (this.options.WOFFSupportNeeded && provider.seperateRequestForWOFF) {
232+
cssContent += await this.getResponse(url, UserAgent.IE);
233+
}
166234

167-
const url = new URL(normalizedHref);
168-
// Force HTTPS protocol
169-
url.protocol = 'https:';
235+
cssContent += await this.getResponse(url, UserAgent.Chrome);
170236

171-
if (!SUPPORTED_PROVIDERS.includes(url.hostname)) {
172-
// Provider not supported.
173-
continue;
174-
}
237+
if (this.options.minify) {
238+
cssContent = cssContent
239+
// Comments.
240+
.replace(/\/\*([\s\S]*?)\*\//g, '')
241+
// New lines.
242+
.replace(/\n/g, '')
243+
// Safe spaces.
244+
.replace(/\s?[\{\:\;]\s+/g, (s) => s.trim());
245+
}
175246

176-
// The order IE -> Chrome is important as otherwise Chrome will load woff1.
177-
let cssContent = '';
178-
if (this.options.WOFFSupportNeeded) {
179-
cssContent += await this.getResponse(url, UserAgent.IE);
180-
}
181-
cssContent += await this.getResponse(url, UserAgent.Chrome);
182-
183-
if (this.options.minify) {
184-
cssContent = cssContent
185-
// Comments.
186-
.replace(/\/\*([\s\S]*?)\*\//g, '')
187-
// New lines.
188-
.replace(/\n/g, '')
189-
// Safe spaces.
190-
.replace(/\s?[\{\:\;]\s+/g, (s) => s.trim());
191-
}
247+
return cssContent;
248+
}
249+
250+
private getFontProviderDetails(url: URL): FontProviderDetails | undefined {
251+
return SUPPORTED_PROVIDERS[url.hostname];
252+
}
192253

193-
hrefsContent.set(hrefPath, cssContent);
254+
private createNormalizedUrl(value: string): URL | undefined {
255+
// Need to convert '//' to 'https://' because the URL parser will fail with '//'.
256+
const normalizedHref = value.startsWith('//') ? `https:${value}` : value;
257+
if (!normalizedHref.startsWith('http')) {
258+
// Non valid URL.
259+
// Example: relative path styles.css.
260+
return undefined;
194261
}
195262

196-
return hrefsContent;
263+
const url = new URL(normalizedHref);
264+
// Force HTTPS protocol
265+
url.protocol = 'https:';
266+
267+
return url;
197268
}
198269
}

0 commit comments

Comments
 (0)