Skip to content

Commit 80686be

Browse files
committedOct 2, 2024·
refactor(@angular/build): avoid hydration warnings when RenderMode.Client is set
With the introduction of the `RenderMode` configuration for routes, some routes may be set to `RenderMode.Client` while still including the `provideClientHydration()` function in the provider list during bootstrap. This led to a false-positive warning in the console, incorrectly suggesting a hydration misconfiguration. This commit introduces a DOM marker that allows the framework to bypass these unnecessary checks, preventing the misleading warnings. See: angular/angular#58004
1 parent c33e862 commit 80686be

File tree

3 files changed

+81
-0
lines changed

3 files changed

+81
-0
lines changed
 

‎packages/angular/build/src/utils/index-file/index-html-generator.ts

+6
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@ import { addEventDispatchContract } from './add-event-dispatch-contract';
1414
import { CrossOriginValue, Entrypoint, FileInfo, augmentIndexHtml } from './augment-index-html';
1515
import { InlineCriticalCssProcessor } from './inline-critical-css';
1616
import { InlineFontsProcessor } from './inline-fonts';
17+
import { addNgcmAttribute } from './ngcm-attribute';
1718
import { addNonce } from './nonce';
1819

1920
type IndexHtmlGeneratorPlugin = (
@@ -82,6 +83,7 @@ export class IndexHtmlGenerator {
8283

8384
// SSR plugins
8485
if (options.generateDedicatedSSRContent) {
86+
this.csrPlugins.push(addNgcmAttributePlugin());
8587
this.ssrPlugins.push(addEventDispatchContractPlugin(), addNoncePlugin());
8688
}
8789
}
@@ -203,3 +205,7 @@ function postTransformPlugin({ options }: IndexHtmlGenerator): IndexHtmlGenerato
203205
function addEventDispatchContractPlugin(): IndexHtmlGeneratorPlugin {
204206
return (html) => addEventDispatchContract(html);
205207
}
208+
209+
function addNgcmAttributePlugin(): IndexHtmlGeneratorPlugin {
210+
return (html) => addNgcmAttribute(html);
211+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
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.dev/license
7+
*/
8+
9+
import { htmlRewritingStream } from './html-rewriting-stream';
10+
11+
/**
12+
* Defines a name of an attribute that is added to the `<body>` tag
13+
* in the `index.html` file in case a given route was configured
14+
* with `RenderMode.Client`. 'cm' is an abbreviation for "Client Mode".
15+
*
16+
* @see https://github.com/angular/angular/pull/58004
17+
*/
18+
const CLIENT_RENDER_MODE_FLAG = 'ngcm';
19+
20+
/**
21+
* Transforms the provided HTML by adding the `ngcm` attribute to the `<body>` tag.
22+
* This is used in the client-side rendered (CSR) version of `index.html` to prevent hydration warnings.
23+
*
24+
* @param html The HTML markup to be transformed.
25+
* @returns A promise that resolves to the transformed HTML string with the necessary modifications.
26+
*/
27+
export async function addNgcmAttribute(html: string): Promise<string> {
28+
const { rewriter, transformedContent } = await htmlRewritingStream(html);
29+
30+
rewriter.on('startTag', (tag) => {
31+
if (
32+
tag.tagName === 'body' &&
33+
!tag.attrs.some((attr) => attr.name === CLIENT_RENDER_MODE_FLAG)
34+
) {
35+
tag.attrs.push({ name: CLIENT_RENDER_MODE_FLAG, value: '' });
36+
}
37+
38+
rewriter.emitStartTag(tag);
39+
});
40+
41+
return transformedContent();
42+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
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.dev/license
7+
*/
8+
9+
import { addNgcmAttribute } from './ngcm-attribute';
10+
11+
describe('addNgcmAttribute', () => {
12+
it('should add the ngcm attribute to the <body> tag', async () => {
13+
const result = await addNgcmAttribute(`
14+
<html>
15+
<head></head>
16+
<body><p>hello world!</p></body>
17+
</html>
18+
`);
19+
20+
expect(result).toContain('<body ngcm=""><p>hello world!</p></body>');
21+
});
22+
23+
it('should not override an existing ngcm attribute', async () => {
24+
const result = await addNgcmAttribute(`
25+
<html>
26+
<head></head>
27+
<body ngcm="foo"><p>hello world!</p></body>
28+
</html>
29+
`);
30+
31+
expect(result).toContain('<body ngcm="foo"><p>hello world!</p></body>');
32+
});
33+
});

0 commit comments

Comments
 (0)
Please sign in to comment.