Skip to content

Commit 9904838

Browse files
authoredAug 17, 2022
Fix rewriting of components for custom elements
This commit particularly solves elements names that include dashes: custom elements, such as `custom-element-name`. These can be written in JSX like so: ```jsx <custom-element-name /> ``` …which is fine. But to support passing them in, we were previously rewriting such code to a member id, to take a key from an object, like so: ```jsx <_components.custom-element-name> ``` …which crashed. This commit solves that by taking the component from the object in a temporary variable, and using that valid variable name as a single component name. ```js const _component0 = _components['custom-element-name'] <_component0 /> ``` Closes GH-2100. Closes GH-2101. Co-authored-by: Titus Wormer <tituswormer@gmail.com> Reviewed-by: Titus Wormer <tituswormer@gmail.com>
1 parent 69a15b7 commit 9904838

File tree

2 files changed

+46
-11
lines changed

2 files changed

+46
-11
lines changed
 

‎packages/mdx/lib/plugin/recma-jsx-rewrite.js

+44-9
Original file line numberDiff line numberDiff line change
@@ -72,6 +72,8 @@ export function recmaJsxRewrite(options = {}) {
7272
let createErrorHelper
7373
/** @type {Scope|null} */
7474
let currentScope
75+
/** @type {Map<string | number, string>} */
76+
const idToInvalidComponentName = new Map()
7577

7678
walk(tree, {
7779
enter(_node) {
@@ -193,16 +195,24 @@ export function recmaJsxRewrite(options = {}) {
193195
fnScope.tags.push(id)
194196
}
195197

196-
node.openingElement.name = toJsxIdOrMemberExpression([
197-
'_components',
198-
id
199-
])
198+
/** @type {Array<string | number>} */
199+
let jsxIdExpression = ['_components', id]
200+
if (isIdentifierName(id) === false) {
201+
let invalidComponentName = idToInvalidComponentName.get(id)
202+
if (invalidComponentName === undefined) {
203+
invalidComponentName = `_component${idToInvalidComponentName.size}`
204+
idToInvalidComponentName.set(id, invalidComponentName)
205+
}
206+
207+
jsxIdExpression = [invalidComponentName]
208+
}
209+
210+
node.openingElement.name =
211+
toJsxIdOrMemberExpression(jsxIdExpression)
200212

201213
if (node.closingElement) {
202-
node.closingElement.name = toJsxIdOrMemberExpression([
203-
'_components',
204-
id
205-
])
214+
node.closingElement.name =
215+
toJsxIdOrMemberExpression(jsxIdExpression)
206216
}
207217
}
208218
}
@@ -259,7 +269,11 @@ export function recmaJsxRewrite(options = {}) {
259269
/** @type {Array<Statement>} */
260270
const statements = []
261271

262-
if (defaults.length > 0 || actual.length > 0) {
272+
if (
273+
defaults.length > 0 ||
274+
actual.length > 0 ||
275+
idToInvalidComponentName.size > 0
276+
) {
263277
if (providerImportSource) {
264278
importProvider = true
265279
parameters.push({
@@ -344,6 +358,27 @@ export function recmaJsxRewrite(options = {}) {
344358
componentsInit = {type: 'Identifier', name: '_components'}
345359
}
346360

361+
if (isNamedFunction(scope.node, '_createMdxContent')) {
362+
for (const [id, componentName] of idToInvalidComponentName) {
363+
// For JSX IDs that can’t be represented as JavaScript IDs (as in,
364+
// those with dashes, such as `custom-element`), generate a
365+
// separate variable that is a valid JS ID (such as `_component0`),
366+
// and takes it from components:
367+
// `const _component0 = _components['custom-element']`
368+
declarations.push({
369+
type: 'VariableDeclarator',
370+
id: {type: 'Identifier', name: componentName},
371+
init: {
372+
type: 'MemberExpression',
373+
object: {type: 'Identifier', name: '_components'},
374+
property: {type: 'Literal', value: id},
375+
computed: true,
376+
optional: false
377+
}
378+
})
379+
}
380+
}
381+
347382
if (componentsPattern) {
348383
declarations.push({
349384
type: 'VariableDeclarator',

‎packages/mdx/test/compile.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -860,8 +860,8 @@ test('jsx', async () => {
860860
'function _createMdxContent(props) {',
861861
' const _components = Object.assign({',
862862
' "a-b": "a-b"',
863-
' }, props.components);',
864-
' return <>{<_components.a-b></_components.a-b>}</>;',
863+
' }, props.components), _component0 = _components["a-b"];',
864+
' return <>{<_component0></_component0>}</>;',
865865
'}',
866866
'function MDXContent(props = {}) {',
867867
' const {wrapper: MDXLayout} = props.components || ({});',

0 commit comments

Comments
 (0)