Skip to content

Commit

Permalink
Explicitly handle JSX tags in order (#486)
Browse files Browse the repository at this point in the history
Virtual code generation from estree expressions was based on an
assumption of the order in which JSXIdentifier nodes are visited.
`estree-walker` visits the nodes in property order. The parser generates
AST properties in an order that broke our code generation. As a result,
the string `_components.` was injected in an incorrect position in the
virtual code, yielding syntax errors.

This is now resolved by explicitly handling `JSXElement` nodes in both
the enter and leave methods, instead of just handling `JSXIdentifier`
nodes.

Closes #485
  • Loading branch information
remcohaszing authored Dec 23, 2024
1 parent d73050c commit b2e9706
Show file tree
Hide file tree
Showing 3 changed files with 67 additions and 17 deletions.
5 changes: 5 additions & 0 deletions .changeset/sweet-rabbits-lick.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
'@mdx-js/language-service': patch
---

Fix bug in virtual code generation for JSX closing elements
44 changes: 34 additions & 10 deletions packages/language-service/lib/virtual-code.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
/**
* @import {CodeMapping, VirtualCode} from '@volar/language-service'
* @import {ExportDefaultDeclaration, Program} from 'estree'
* @import {ExportDefaultDeclaration, JSXClosingElement, JSXOpeningElement, Program} from 'estree-jsx'
* @import {Nodes, Root} from 'mdast'
* @import {MdxjsEsm} from 'mdast-util-mdxjs-esm'
* @import {IScriptSnapshot} from 'typescript'
Expand Down Expand Up @@ -449,18 +449,32 @@ function getEmbeddedCodes(mdx, ast, checkMdx, jsxImportSource) {
function processJsxExpression(program, lastIndex) {
let newIndex = lastIndex
let functionNesting = 0

/**
* @param {JSXClosingElement | JSXOpeningElement} node
* @returns {undefined}
*/
function processJsxTag(node) {
const {name} = node

if (name.type !== 'JSXIdentifier') {
return
}

if (!isInjectableComponent(name.name, variables)) {
return
}

jsx =
addOffset(jsxMapping, mdx, jsx, newIndex, name.start) + '_components.'
newIndex = name.start
}

walk(program, {
enter(node) {
switch (node.type) {
case 'JSXIdentifier': {
if (!isInjectableComponent(node.name, variables)) {
return
}

jsx =
addOffset(jsxMapping, mdx, jsx, newIndex, node.start) +
'_components.'
newIndex = node.start
case 'JSXElement': {
processJsxTag(node.openingElement)
break
}

Expand Down Expand Up @@ -500,6 +514,16 @@ function getEmbeddedCodes(mdx, ast, checkMdx, jsxImportSource) {
break
}

case 'JSXElement': {
const {closingElement} = node

if (closingElement) {
processJsxTag(closingElement)
}

break
}

default:
}
}
Expand Down
35 changes: 28 additions & 7 deletions packages/language-service/test/language-plugin.js
Original file line number Diff line number Diff line change
Expand Up @@ -1752,8 +1752,12 @@ test('create virtual code w/ prefixed JSX expressions for mdxFlowExpression', ()
'{<div>{""}</div>}',
'{<Injected />}',
'{<Injected>{""}</Injected>}',
'{<Injected><Injected>{""}</Injected></Injected>}',
'{<Local />}',
'{<Local>{""}</Local>}'
'{<Local>{""}</Local>}',
'{<Local><Local>{""}</Local></Local>}',
'{<Local><Injected>{""}</Injected></Local>}',
'{<Injected><Local>{""}</Local></Injected>}'
)

const code = plugin.createVirtualCode?.('/test.mdx', 'mdx', snapshot, {
Expand Down Expand Up @@ -1799,9 +1803,18 @@ test('create virtual code w/ prefixed JSX expressions for mdxFlowExpression', ()
}
},
{
sourceOffsets: [28, 38, 56, 58, 71, 73, 88, 99, 111],
generatedOffsets: [843, 857, 879, 893, 910, 924, 951, 966, 982],
lengths: [9, 17, 2, 12, 2, 15, 10, 11, 21],
sourceOffsets: [
28, 38, 56, 58, 71, 73, 88, 99, 101, 111, 126, 137, 148, 160, 182,
219, 228, 243, 262, 264, 294
],
generatedOffsets: [
843, 857, 879, 893, 910, 924, 951, 966, 980, 1002, 1029, 1052, 1067,
1083, 1109, 1150, 1171, 1198, 1221, 1235, 1277
],
lengths: [
9, 17, 2, 12, 2, 15, 10, 2, 10, 15, 11, 10, 11, 21, 36, 9, 15, 18,
2, 30, 10
],
data: {
completion: true,
format: false,
Expand Down Expand Up @@ -1845,8 +1858,12 @@ test('create virtual code w/ prefixed JSX expressions for mdxFlowExpression', ()
' {<div>{""}</div>}',
' {<_components.Injected />}',
' {<_components.Injected>{""}</_components.Injected>}',
' {<_components.Injected><_components.Injected>{""}</_components.Injected></_components.Injected>}',
' {<Local />}',
' {<Local>{""}</Local>}',
' {<Local><Local>{""}</Local></Local>}',
' {<Local><_components.Injected>{""}</_components.Injected></Local>}',
' {<_components.Injected><Local>{""}</Local></_components.Injected>}',
' </>',
'}',
'',
Expand All @@ -1870,9 +1887,9 @@ test('create virtual code w/ prefixed JSX expressions for mdxFlowExpression', ()
languageId: 'markdown',
mappings: [
{
sourceOffsets: [26, 37, 55, 70, 98, 110],
generatedOffsets: [0, 9, 17, 25, 33, 41],
lengths: [2, 1, 1, 1, 1, 1],
sourceOffsets: [26, 37, 55, 70, 98, 147, 159, 181, 218, 261],
generatedOffsets: [0, 9, 17, 25, 33, 41, 49, 57, 65, 73],
lengths: [2, 1, 1, 1, 1, 1, 1, 1, 1, 1],
data: {
completion: true,
format: false,
Expand All @@ -1891,6 +1908,10 @@ test('create virtual code w/ prefixed JSX expressions for mdxFlowExpression', ()
'<!---->',
'<!---->',
'<!---->',
'<!---->',
'<!---->',
'<!---->',
'<!---->',
'<!---->'
)
}
Expand Down

0 comments on commit b2e9706

Please sign in to comment.