Skip to content

Commit dc2f659

Browse files
clydinalan-agius4
authored andcommitted
fix(@schematics/angular): generate component templates with a .ng.html file extension
To align with the updated style guide, Angular v20 will generate component templates with a `.ng.html` file extension instead of the previous `.html` by default. Projects will automatically use this new template extension. Projects can however opt-out of component generation using `.ng.html` by setting the `ngHtml` option to false for the component schematic. This can be done as a default in the `angular.json` or directly on the commandline via `--no-ng-html` when executing `ng generate`.
1 parent 312cc72 commit dc2f659

File tree

11 files changed

+48
-14
lines changed

11 files changed

+48
-14
lines changed

packages/schematics/angular/application/files/module-files/src/app/app.component.ts.template

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ import { Component } from '@angular/core';
99
%><router-outlet /><%
1010
} %>
1111
`,<% } else { %>
12-
templateUrl: './app.component.html',<% } %>
12+
templateUrl: './app.component.ng.html',<% } %>
1313
standalone: false,<% if(inlineStyle) { %>
1414
styles: []<% } else { %>
1515
styleUrl: './app.component.<%= style %>'<% } %>

packages/schematics/angular/application/files/standalone-files/src/app/app.component.ts.template

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ import { RouterOutlet } from '@angular/router';<% } %>
1111
%><router-outlet /><%
1212
} %>
1313
`,<% } else { %>
14-
templateUrl: './app.component.html',<% } if(inlineStyle) { %>
14+
templateUrl: './app.component.ng.html',<% } if(inlineStyle) { %>
1515
styles: [],<% } else { %>
1616
styleUrl: './app.component.<%= style %>'<% } %>
1717
})

packages/schematics/angular/application/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -84,7 +84,7 @@ export default function (options: ApplicationOptions): Rule {
8484
? filter((path) => !path.endsWith('tsconfig.spec.json.template'))
8585
: noop(),
8686
componentOptions.inlineTemplate
87-
? filter((path) => !path.endsWith('component.html.template'))
87+
? filter((path) => !path.endsWith('.ng.html.template'))
8888
: noop(),
8989
applyTemplates({
9090
utils: strings,

packages/schematics/angular/application/index_spec.ts

+7-7
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@ describe('Application Schematic', () => {
5757
'/projects/foo/src/styles.css',
5858
'/projects/foo/src/app/app.module.ts',
5959
'/projects/foo/src/app/app.component.css',
60-
'/projects/foo/src/app/app.component.html',
60+
'/projects/foo/src/app/app.component.ng.html',
6161
'/projects/foo/src/app/app.component.spec.ts',
6262
'/projects/foo/src/app/app.component.ts',
6363
]),
@@ -267,7 +267,7 @@ describe('Application Schematic', () => {
267267
'/src/main.ts',
268268
'/src/styles.css',
269269
'/src/app/app.component.css',
270-
'/src/app/app.component.html',
270+
'/src/app/app.component.ng.html',
271271
'/src/app/app.component.spec.ts',
272272
'/src/app/app.component.ts',
273273
]),
@@ -448,7 +448,7 @@ describe('Application Schematic', () => {
448448
[
449449
'/projects/foo/tsconfig.spec.json',
450450
'/projects/foo/src/app/app.component.css',
451-
'/projects/foo/src/app/app.component.html',
451+
'/projects/foo/src/app/app.component.ng.html',
452452
'/projects/foo/src/app/app.component.spec.ts',
453453
].forEach((x) => expect(files).not.toContain(x));
454454

@@ -473,7 +473,7 @@ describe('Application Schematic', () => {
473473
'/projects/foo/tsconfig.spec.json',
474474
'/projects/foo/karma.conf.js',
475475
'/projects/foo/src/test.ts',
476-
'/projects/foo/src/app/app.component.html',
476+
'/projects/foo/src/app/app.component.ng.html',
477477
'/projects/foo/src/app/app.component.spec.ts',
478478
].forEach((x) => expect(files).not.toContain(x));
479479

@@ -510,7 +510,7 @@ describe('Application Schematic', () => {
510510
'/projects/foo/src/index.html',
511511
'/projects/foo/src/main.ts',
512512
'/projects/foo/src/styles.css',
513-
'/projects/foo/src/app/app.component.html',
513+
'/projects/foo/src/app/app.component.ng.html',
514514
'/projects/foo/src/app/app.component.ts',
515515
]),
516516
);
@@ -532,7 +532,7 @@ describe('Application Schematic', () => {
532532
'/projects/foo/src/styles.css',
533533
'/projects/foo/src/app/app.config.ts',
534534
'/projects/foo/src/app/app.component.css',
535-
'/projects/foo/src/app/app.component.html',
535+
'/projects/foo/src/app/app.component.ng.html',
536536
'/projects/foo/src/app/app.component.spec.ts',
537537
'/projects/foo/src/app/app.component.ts',
538538
]),
@@ -673,7 +673,7 @@ describe('Application Schematic', () => {
673673
'/projects/foo/src/app/app-routing.module.ts',
674674
'/projects/foo/src/app/app.module.ts',
675675
'/projects/foo/src/app/app.component.css',
676-
'/projects/foo/src/app/app.component.html',
676+
'/projects/foo/src/app/app.component.ng.html',
677677
'/projects/foo/src/app/app.component.spec.ts',
678678
'/projects/foo/src/app/app.component.ts',
679679
]),

packages/schematics/angular/component/files/__name@dasherize@if-flat__/__name@dasherize__.__type@dasherize__.ts.template

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ import { <% if(changeDetection !== 'Default') { %>ChangeDetectionStrategy, <% }%
99
<%= dasherize(name) %> works!
1010
</p>
1111
`<% } else { %>
12-
templateUrl: './<%= dasherize(name) %><%= type ? '.' + dasherize(type): '' %>.html'<% } if(inlineStyle) { %>,
12+
templateUrl: './<%= dasherize(name) %><%= type ? '.' + dasherize(type): '' %><%= ngext %>.html'<% } if(inlineStyle) { %>,
1313
styles: `<% if(displayBlock){ %>
1414
:host {
1515
display: block;

packages/schematics/angular/component/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -72,6 +72,7 @@ export default function (options: ComponentOptions): Rule {
7272
applyTemplates({
7373
...strings,
7474
'if-flat': (s: string) => (options.flat ? '' : s),
75+
'ngext': options.ngHtml ? '.ng' : '',
7576
...options,
7677
}),
7778
!options.type

packages/schematics/angular/component/index_spec.ts

+28
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@ describe('Component Schematic', () => {
3030
module: undefined,
3131
export: false,
3232
project: 'bar',
33+
ngHtml: false,
3334
};
3435

3536
const workspaceOptions: WorkspaceOptions = {
@@ -291,6 +292,33 @@ describe('Component Schematic', () => {
291292
expect(tree.files).toContain('/projects/bar/src/app/foo/foo.html');
292293
});
293294

295+
it('should not use `.ng.html` extension when ngHtml is false', async () => {
296+
const options = { ...defaultOptions, ngHtml: false };
297+
const tree = await schematicRunner.runSchematic('component', options, appTree);
298+
const content = tree.readContent('/projects/bar/src/app/foo/foo.component.ts');
299+
expect(content).toContain('foo.component.html');
300+
expect(tree.files).toContain('/projects/bar/src/app/foo/foo.component.css');
301+
expect(tree.files).toContain('/projects/bar/src/app/foo/foo.component.html');
302+
});
303+
304+
it('should use `.ng.html` extension when ngHtml is true', async () => {
305+
const options = { ...defaultOptions, ngHtml: true };
306+
const tree = await schematicRunner.runSchematic('component', options, appTree);
307+
const content = tree.readContent('/projects/bar/src/app/foo/foo.component.ts');
308+
expect(content).toContain('foo.component.ng.html');
309+
expect(tree.files).toContain('/projects/bar/src/app/foo/foo.component.css');
310+
expect(tree.files).toContain('/projects/bar/src/app/foo/foo.component.ng.html');
311+
});
312+
313+
it('should use `.ng.html` extension when ngHtml is not present', async () => {
314+
const options = { ...defaultOptions, ngHtml: undefined };
315+
const tree = await schematicRunner.runSchematic('component', options, appTree);
316+
const content = tree.readContent('/projects/bar/src/app/foo/foo.component.ts');
317+
expect(content).toContain('foo.component.ng.html');
318+
expect(tree.files).toContain('/projects/bar/src/app/foo/foo.component.css');
319+
expect(tree.files).toContain('/projects/bar/src/app/foo/foo.component.ng.html');
320+
});
321+
294322
it('should create the right selector with a path in the name', async () => {
295323
const options = { ...defaultOptions, name: 'sub/test' };
296324
appTree = await schematicRunner.runSchematic('component', options, appTree);

packages/schematics/angular/component/schema.json

+5
Original file line numberDiff line numberDiff line change
@@ -135,6 +135,11 @@
135135
"type": "boolean",
136136
"default": false,
137137
"description": "Use a default export for the component in its TypeScript file instead of a named export."
138+
},
139+
"ngHtml": {
140+
"type": "boolean",
141+
"default": true,
142+
"description": "Generate component template files with an '.ng.html' file extension instead of '.html'. The '.ng.html' file extension is recommended by the Angular style guide."
138143
}
139144
},
140145
"required": ["name", "project"]

packages/schematics/angular/module/index_spec.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -153,7 +153,7 @@ describe('Module Schematic', () => {
153153
'/projects/bar/src/app/foo/foo.module.ts',
154154
'/projects/bar/src/app/foo/foo-routing.module.ts',
155155
'/projects/bar/src/app/foo/foo.component.ts',
156-
'/projects/bar/src/app/foo/foo.component.html',
156+
'/projects/bar/src/app/foo/foo.component.ng.html',
157157
'/projects/bar/src/app/foo/foo.component.css',
158158
]),
159159
);
@@ -203,7 +203,7 @@ describe('Module Schematic', () => {
203203
expect(files).toContain('/projects/bar/src/app/foo/foo.module.ts');
204204
expect(files).not.toContain('/projects/bar/src/app/foo/foo-routing.module.ts');
205205
expect(files).toContain('/projects/bar/src/app/foo/foo.component.ts');
206-
expect(files).toContain('/projects/bar/src/app/foo/foo.component.html');
206+
expect(files).toContain('/projects/bar/src/app/foo/foo.component.ng.html');
207207
expect(files).toContain('/projects/bar/src/app/foo/foo.component.css');
208208

209209
const appModuleContent = tree.readContent('/projects/bar/src/app/app.module.ts');
@@ -231,7 +231,7 @@ describe('Module Schematic', () => {
231231
'/projects/bar/src/app/foo.module.ts',
232232
'/projects/bar/src/app/foo-routing.module.ts',
233233
'/projects/bar/src/app/foo.component.ts',
234-
'/projects/bar/src/app/foo.component.html',
234+
'/projects/bar/src/app/foo.component.ng.html',
235235
'/projects/bar/src/app/foo.component.css',
236236
]),
237237
);

0 commit comments

Comments
 (0)