Skip to content

Commit bf5ccaa

Browse files
committed
Fix type error if JSX is not defined
React 19 no longer defined `JSX`. It prefers users to use `React.JSX` instead. Without it, it becomes impossible to infer what the types of props for particular components are, what the know tag names are, what the “result” of `jsx()` is. This PR is a patch to prevent errors when `JSX` is not there while still *allowing* `JSX` to exist and for this package to be typed nicely. Related-to GH-4. Closes GH-6.
1 parent 4a0148e commit bf5ccaa

9 files changed

+687
-542
lines changed

.gitignore

-1
Original file line numberDiff line numberDiff line change
@@ -10,4 +10,3 @@ yarn.lock
1010
!/lib/types.d.ts
1111
!/index.d.ts
1212
!/types-esmsh.d.ts
13-
!/types-jsx.d.ts

lib/index.js

+3-4
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@
66
* @import {MdxJsxFlowElementHast, MdxJsxTextElementHast} from 'mdast-util-mdx-jsx'
77
* @import {MdxjsEsmHast} from 'mdast-util-mdxjs-esm'
88
* @import {Position} from 'unist'
9-
* @import {Child, Create, Field, State, Style} from './types.js'
9+
* @import {Child, Create, Field, JsxElement, State, Style} from './types.js'
1010
*/
1111

1212
import {stringify as commas} from 'comma-separated-tokens'
@@ -53,7 +53,7 @@ const docs = 'https://github.com/syntax-tree/hast-util-to-jsx-runtime'
5353
* Tree to transform.
5454
* @param {Options} options
5555
* Configuration (required).
56-
* @returns {JSX.Element}
56+
* @returns {JsxElement}
5757
* JSX element.
5858
*/
5959

@@ -733,8 +733,7 @@ function findComponentFromName(state, name, allowExpression) {
733733
// Only literals can be passed in `components` currently.
734734
// No identifiers / member expressions.
735735
if (result.type === 'Literal') {
736-
const name = /** @type {keyof JSX.IntrinsicElements} */ (result.value)
737-
736+
const name = /** @type {PropertyKey} */ (result.value)
738737
return own.call(state.components, name) ? state.components[name] : name
739738
}
740739

lib/types.d.ts

+50-26
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ import type {Schema} from 'property-information'
99
/**
1010
* Child.
1111
*/
12-
export type Child = JSX.Element | string | null | undefined
12+
export type Child = JsxElement | string | null | undefined
1313

1414
/**
1515
* Possible components to use.
@@ -24,16 +24,16 @@ export type Child = JSX.Element | string | null | undefined
2424
// Note: this type has to be in `.ts` or `.d.ts`, otherwise TSC hardcodes
2525
// react into the `.d.ts` file.
2626
export type Components = {
27-
[TagName in keyof JSX.IntrinsicElements]:
28-
| Component<JSX.IntrinsicElements[TagName] & ExtraProps>
29-
| keyof JSX.IntrinsicElements
27+
[TagName in keyof JsxIntrinsicElements]:
28+
| Component<JsxIntrinsicElements[TagName] & ExtraProps>
29+
| keyof JsxIntrinsicElements
3030
}
3131

3232
/**
3333
* Function or class component.
3434
*
35-
* You can access props at `JSX.IntrinsicElements`.
36-
* For example, to find props for `a`, use `JSX.IntrinsicElements['a']`.
35+
* You can access props at `JsxIntrinsicElements`.
36+
* For example, to find props for `a`, use `JsxIntrinsicElements['a']`.
3737
*
3838
* @typeParam ComponentProps
3939
* Props type.
@@ -55,7 +55,7 @@ export type Create = (
5555
type: unknown,
5656
props: Props,
5757
key: string | undefined
58-
) => JSX.Element
58+
) => JsxElement
5959

6060
/**
6161
* Class component: given props, returns an instance.
@@ -69,7 +69,7 @@ export type Create = (
6969
*/
7070
type ClassComponent<ComponentProps> = new (
7171
props: ComponentProps
72-
) => JSX.ElementClass
72+
) => JsxElementClass
7373

7474
/**
7575
* Casing to use for attribute names.
@@ -135,18 +135,31 @@ export type Fragment = unknown
135135
*/
136136
type FunctionComponent<ComponentProps> = (
137137
props: ComponentProps
138-
) => JSX.Element | string | null | undefined
138+
) => JsxElement | string | null | undefined
139139

140140
/**
141-
* Create a production element.
141+
* Conditional type for a class.
142142
*/
143-
export type Jsx = (
144-
// `any` because runtimes often have complex framework-specific types here.
145-
// type-coverage:ignore-next-line
146-
type: any,
147-
props: Props,
148-
key?: string | undefined
149-
) => JSX.Element
143+
// @ts-ignore: conditionally defined.
144+
export type JsxElementClass = any extends JSX.ElementClass
145+
? unknown
146+
: // @ts-ignore: conditionally defined.
147+
JSX.ElementClass
148+
149+
/**
150+
* Conditional type for a node object.
151+
*/
152+
// @ts-ignore: conditionally defined.
153+
export type JsxElement = any extends JSX.Element ? unknown : JSX.Element
154+
155+
/**
156+
* Conditional type for a record of tag names to corresponding props.
157+
*/
158+
// @ts-ignore: conditionally defined.
159+
export type JsxIntrinsicElements = any extends JSX.IntrinsicElements
160+
? Record<PropertyKey, any>
161+
: // @ts-ignore: conditionally defined.
162+
JSX.IntrinsicElements
150163

151164
/**
152165
* Create a development element.
@@ -160,7 +173,18 @@ export type JsxDev = (
160173
isStaticChildren: boolean,
161174
source: Source,
162175
self: undefined
163-
) => JSX.Element
176+
) => JsxElement
177+
178+
/**
179+
* Create a production element.
180+
*/
181+
export type Jsx = (
182+
// `any` because runtimes often have complex framework-specific types here.
183+
// type-coverage:ignore-next-line
184+
type: any,
185+
props: Props,
186+
key?: string | undefined
187+
) => JsxElement
164188

165189
/**
166190
* Configuration.
@@ -232,10 +256,6 @@ export interface OptionsDevelopment extends OptionsBase {
232256
* Whether to use `jsxDEV` (when on) or `jsx` and `jsxs` (when off).
233257
*/
234258
development: true
235-
/**
236-
* Dynamic JSX (optional).
237-
*/
238-
jsx?: Jsx | null | undefined
239259
/**
240260
* Development JSX.
241261
*/
@@ -244,6 +264,10 @@ export interface OptionsDevelopment extends OptionsBase {
244264
* Static JSX (optional).
245265
*/
246266
jsxs?: Jsx | null | undefined
267+
/**
268+
* Dynamic JSX (optional).
269+
*/
270+
jsx?: Jsx | null | undefined
247271
}
248272

249273
/**
@@ -258,10 +282,6 @@ export interface OptionsProduction extends OptionsBase {
258282
* Whether to use `jsxDEV` (when on) or `jsx` and `jsxs` (when off) (optional).
259283
*/
260284
development?: false | null | undefined
261-
/**
262-
* Dynamic JSX.
263-
*/
264-
jsx: Jsx
265285
/**
266286
* Development JSX (optional).
267287
*/
@@ -270,6 +290,10 @@ export interface OptionsProduction extends OptionsBase {
270290
* Static JSX.
271291
*/
272292
jsxs: Jsx
293+
/**
294+
* Dynamic JSX.
295+
*/
296+
jsx: Jsx
273297
}
274298

275299
/**

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -113,6 +113,7 @@
113113
"default": "generic"
114114
}
115115
],
116+
"@typescript-eslint/ban-ts-comment": 0,
116117
"@typescript-eslint/ban-types": [
117118
"error",
118119
{

readme.md

+21-14
Original file line numberDiff line numberDiff line change
@@ -109,7 +109,7 @@ Yields:
109109
```
110110

111111
> **Note**:
112-
> To add proper type support in React 19,
112+
> to add better type support,
113113
> register a global JSX namespace:
114114
>
115115
> ```ts
@@ -163,13 +163,16 @@ with an automatic JSX runtime.
163163
164164
##### Returns
165165
166-
Result from your configured JSX runtime (`JSX.Element`).
166+
Result from your configured JSX runtime
167+
(`JSX.Element` if defined,
168+
otherwise `unknown` which you can cast yourself).
167169
168170
### `Components`
169171
170172
Possible components to use (TypeScript type).
171173
172-
Each key is a tag name typed in `JSX.IntrinsicElements`.
174+
Each key is a tag name typed in `JSX.IntrinsicElements`,
175+
if defined.
173176
Each value is either a different tag name
174177
or a component accepting the corresponding props
175178
(and an optional `node` prop if `passNode` is on).
@@ -184,14 +187,14 @@ use `JSX.IntrinsicElements['a']`.
184187
```ts
185188
import type {Element} from 'hast'
186189
190+
type ExtraProps = {node?: Element | undefined}
191+
187192
type Components = {
188193
[TagName in keyof JSX.IntrinsicElements]:
189194
| Component<JSX.IntrinsicElements[TagName] & ExtraProps>
190195
| keyof JSX.IntrinsicElements
191196
}
192197
193-
type ExtraProps = {node?: Element | undefined}
194-
195198
type Component<ComponentProps> =
196199
// Class component:
197200
| (new (props: ComponentProps) => JSX.ElementClass)
@@ -308,7 +311,9 @@ Create a production element (TypeScript type).
308311
309312
###### Returns
310313
311-
Element from your framework (`JSX.Element`).
314+
Element from your framework
315+
(`JSX.Element` if defined,
316+
otherwise `unknown` which you can cast yourself).
312317
313318
### `JsxDev`
314319
@@ -338,7 +343,9 @@ Create a development element (TypeScript type).
338343
339344
###### Returns
340345
341-
Element from your framework (`JSX.Element`).
346+
Element from your framework
347+
(`JSX.Element` if defined,
348+
otherwise `unknown` which you can cast yourself).
342349
343350
### `Options`
344351
@@ -348,12 +355,12 @@ Configuration (TypeScript type).
348355
349356
* `Fragment` ([`Fragment`][api-fragment], required)
350357
— fragment
351-
* `jsx` ([`Jsx`][api-jsx], required in production)
352-
— dynamic JSX
353-
* `jsxs` ([`Jsx`][api-jsx], required in production)
354-
— static JSX
355358
* `jsxDEV` ([`JsxDev`][api-jsx-dev], required in development)
356359
— development JSX
360+
* `jsxs` ([`Jsx`][api-jsx], required in production)
361+
— static JSX
362+
* `jsx` ([`Jsx`][api-jsx], required in production)
363+
— dynamic JSX
357364
* `components` ([`Partial<Components>`][api-components], optional)
358365
— components to use
359366
* `createEvaluater` ([`CreateEvaluater`][api-create-evaluater], optional)
@@ -574,7 +581,7 @@ render(
574581
)
575582
```
576583

577-
To add proper type support,
584+
To add better type support,
578585
register a global JSX namespace:
579586

580587
```ts
@@ -642,7 +649,7 @@ function Component() {
642649
}
643650
```
644651

645-
To add proper type support,
652+
To add better type support,
646653
register a global JSX namespace:
647654

648655
```ts
@@ -736,7 +743,7 @@ function Component() {
736743
}
737744
```
738745

739-
To add proper type support,
746+
To add better type support,
740747
register a global JSX namespace:
741748

742749
```ts

0 commit comments

Comments
 (0)