Skip to content

Commit 16357a2

Browse files
authored
Merge pull request #20645 from bertdeblock/fix-component-class-blueprint-in-ts-project
Make sure the `component-class` blueprint generates a signature in a TS project
2 parents 12106b7 + a362683 commit 16357a2

File tree

4 files changed

+52
-27
lines changed

4 files changed

+52
-27
lines changed

blueprints/-utils.js

+24
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,33 @@
11
const { dasherize } = require('ember-cli-string-utils');
2+
const { EOL } = require('os');
3+
4+
function generateComponentSignature(componentName) {
5+
let args = ` // The arguments accepted by the component${EOL} Args: {};`;
6+
7+
let blocks =
8+
` // Any blocks yielded by the component${EOL}` +
9+
` Blocks: {${EOL}` +
10+
` default: []${EOL}` +
11+
` };`;
12+
13+
let element =
14+
` // The element to which \`...attributes\` is applied in the component template${EOL}` +
15+
` Element: null;`;
16+
17+
return (
18+
`export interface ${componentName}Signature {${EOL}` +
19+
`${args}${EOL}` +
20+
`${blocks}${EOL}` +
21+
`${element}${EOL}` +
22+
`}${EOL}`
23+
);
24+
}
225

326
function modulePrefixForProject(project) {
427
return dasherize(project.config().modulePrefix);
528
}
629

730
module.exports = {
31+
generateComponentSignature,
832
modulePrefixForProject,
933
};
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
11
<%= importComponent %>
22
<%= importTemplate %>
3+
<%= componentSignature %>
34
export default <%= defaultExport %>

blueprints/component-class/index.js

+24-3
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ const getPathOption = require('ember-cli-get-component-path-option');
88
const normalizeEntityName = require('ember-cli-normalize-entity-name');
99
const { EOL } = require('os');
1010
const { has } = require('@ember/edition-utils');
11+
const { generateComponentSignature } = require('../-utils');
1112

1213
const maybePolyfillTypeScriptBlueprints = require('../-maybe-polyfill-typescript-blueprints');
1314

@@ -49,9 +50,17 @@ module.exports = {
4950
},
5051
],
5152

53+
/**
54+
Flag to let us correctly handle the case where we are running against a
55+
version of Ember CLI which does not support TS-based emit, and where we
56+
therefore *must* not emit a `defaultExport` local which includes a type
57+
parameter in the exported function call or class definition.
58+
*/
59+
_isUsingTS: false,
60+
5261
init() {
5362
this._super && this._super.init.apply(this, arguments);
54-
maybePolyfillTypeScriptBlueprints(this);
63+
this._isUsingTS = maybePolyfillTypeScriptBlueprints(this);
5564
let isOctane = has('octane');
5665

5766
this.availableOptions.forEach((option) => {
@@ -134,6 +143,7 @@ module.exports = {
134143
let importComponent = '';
135144
let importTemplate = '';
136145
let defaultExport = '';
146+
let componentSignature = '';
137147

138148
// if we're in an addon, build import statement
139149
if (options.project.isEmberCLIAddon() || (options.inRepoAddon && !options.inDummy)) {
@@ -161,17 +171,28 @@ module.exports = {
161171
break;
162172
case '@glimmer/component':
163173
importComponent = `import Component from '@glimmer/component';`;
164-
defaultExport = `class ${classifiedModuleName}Component extends Component {}`;
174+
if (this._isUsingTS) {
175+
componentSignature = generateComponentSignature(classifiedModuleName);
176+
defaultExport = `class ${classifiedModuleName}Component extends Component<${classifiedModuleName}Signature> {}`;
177+
} else {
178+
defaultExport = `class ${classifiedModuleName}Component extends Component {}`;
179+
}
165180
break;
166181
case '@ember/component/template-only':
167182
importComponent = `import templateOnly from '@ember/component/template-only';`;
168-
defaultExport = `templateOnly();`;
183+
if (this._isUsingTS) {
184+
componentSignature = generateComponentSignature(classifiedModuleName);
185+
defaultExport = `templateOnly<${classifiedModuleName}Signature>();`;
186+
} else {
187+
defaultExport = `templateOnly();`;
188+
}
169189
break;
170190
}
171191

172192
return {
173193
importTemplate,
174194
importComponent,
195+
componentSignature,
175196
defaultExport,
176197
path: getPathOption(options),
177198
componentClass,

blueprints/component/index.js

+3-24
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ const getPathOption = require('ember-cli-get-component-path-option');
99
const normalizeEntityName = require('ember-cli-normalize-entity-name');
1010
const { EOL } = require('os');
1111
const { has } = require('@ember/edition-utils');
12+
const { generateComponentSignature } = require('../-utils');
1213

1314
const maybePolyfillTypeScriptBlueprints = require('../-maybe-polyfill-typescript-blueprints');
1415

@@ -273,7 +274,7 @@ module.exports = {
273274
case '@glimmer/component':
274275
importComponent = `import Component from '@glimmer/component';`;
275276
if (this._isUsingTS) {
276-
componentSignature = signatureFor(classifiedModuleName);
277+
componentSignature = generateComponentSignature(classifiedModuleName);
277278
defaultExport = `class ${classifiedModuleName}Component extends Component<${classifiedModuleName}Signature> {}`;
278279
} else {
279280
defaultExport = `class ${classifiedModuleName}Component extends Component {}`;
@@ -282,7 +283,7 @@ module.exports = {
282283
case '@ember/component/template-only':
283284
importComponent = `import templateOnly from '@ember/component/template-only';`;
284285
if (this._isUsingTS) {
285-
componentSignature = signatureFor(classifiedModuleName);
286+
componentSignature = generateComponentSignature(classifiedModuleName);
286287
defaultExport = `templateOnly<${classifiedModuleName}Signature>();`;
287288
} else {
288289
defaultExport = `templateOnly();`;
@@ -300,25 +301,3 @@ module.exports = {
300301
};
301302
},
302303
};
303-
304-
function signatureFor(classifiedModuleName) {
305-
let args = ` // The arguments accepted by the component${EOL} Args: {};`;
306-
307-
let blocks =
308-
` // Any blocks yielded by the component${EOL}` +
309-
` Blocks: {${EOL}` +
310-
` default: []${EOL}` +
311-
` };`;
312-
313-
let element =
314-
` // The element to which \`...attributes\` is applied in the component template${EOL}` +
315-
` Element: null;`;
316-
317-
return (
318-
`interface ${classifiedModuleName}Signature {${EOL}` +
319-
`${args}${EOL}` +
320-
`${blocks}${EOL}` +
321-
`${element}${EOL}` +
322-
`}${EOL}`
323-
);
324-
}

0 commit comments

Comments
 (0)