@@ -25,18 +25,33 @@ const enum UserAgent {
25
25
IE = 'Mozilla/5.0 (Windows NT 10.0; Trident/7.0; rv:11. 0) like Gecko' ,
26
26
}
27
27
28
- const SUPPORTED_PROVIDERS = [ 'fonts.googleapis.com' ] ;
28
+ interface FontProviderDetails {
29
+ preconnectUrl : string ;
30
+ seperateRequestForWOFF : boolean ;
31
+ }
29
32
30
33
export interface InlineFontsOptions {
31
34
minify ?: boolean ;
32
35
WOFFSupportNeeded : boolean ;
33
36
}
34
37
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
+
35
49
export class InlineFontsProcessor {
36
50
constructor ( private options : InlineFontsOptions ) { }
37
51
38
52
async process ( content : string ) : Promise < string > {
39
53
const hrefList : string [ ] = [ ] ;
54
+ const existingPreconnect = new Set < string > ( ) ;
40
55
41
56
// Collector link tags with href
42
57
const { rewriter : collectorStream } = await htmlRewritingStream ( content ) ;
@@ -48,20 +63,63 @@ export class InlineFontsProcessor {
48
63
return ;
49
64
}
50
65
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
+ }
58
94
}
59
95
} ) ;
60
96
61
97
await new Promise ( ( resolve ) => collectorStream . on ( 'finish' , resolve ) ) ;
62
98
63
99
// 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
+
65
123
if ( hrefsContent . size === 0 ) {
66
124
return content ;
67
125
}
@@ -71,21 +129,31 @@ export class InlineFontsProcessor {
71
129
rewriter . on ( 'startTag' , ( tag ) => {
72
130
const { tagName, attrs } = tag ;
73
131
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 ;
89
157
}
90
158
} ) ;
91
159
@@ -152,47 +220,50 @@ export class InlineFontsProcessor {
152
220
return data ;
153
221
}
154
222
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
+ }
157
228
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
+ }
166
234
167
- const url = new URL ( normalizedHref ) ;
168
- // Force HTTPS protocol
169
- url . protocol = 'https:' ;
235
+ cssContent += await this . getResponse ( url , UserAgent . Chrome ) ;
170
236
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
+ }
175
246
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
+ }
192
253
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 ;
194
261
}
195
262
196
- return hrefsContent ;
263
+ const url = new URL ( normalizedHref ) ;
264
+ // Force HTTPS protocol
265
+ url . protocol = 'https:' ;
266
+
267
+ return url ;
197
268
}
198
269
}
0 commit comments