# 介绍 [JSX](https://facebook.github.io/jsx/)是一种嵌入式的类似XML的语法。 它可以被转换成合法的JavaScript,尽管转换的语义是依据不同的实现而定的。 JSX因[React](https://reactjs.org/)框架而流行,但也存在其它的实现。 TypeScript支持内嵌,类型检查以及将JSX直接编译为JavaScript。 # 基本用法 想要使用JSX必须做两件事: 1. 给文件一个`.tsx`扩展名 2. 启用`jsx`选项 TypeScript具有三种JSX模式:`preserve`,`react`和`react-native`。 这些模式只在代码生成阶段起作用 - 类型检查并不受影响。 在`preserve`模式下生成代码中会保留JSX以供后续的转换操作使用(比如:[Babel](https://babeljs.io/))。 另外,输出文件会带有`.jsx`扩展名。 `react`模式会生成`React.createElement`,在使用前不需要再进行转换操作了,输出文件的扩展名为`.js`。 `react-native`相当于`preserve`,它也保留了所有的JSX,但是输出文件的扩展名是`.js`。 模式 | 输入 | 输出 | 输出文件扩展名 ---------------|-----------|------------------------------|---------------------- `preserve` | `
` | `
` | `.jsx` `react` | `
` | `React.createElement("div")` | `.js` `react-native` | `
` | `
` | `.js` 你可以通过在命令行里使用`--jsx`标记或[tsconfig.json](./tsconfig.json.md)里的选项来指定模式。 > *注意:当了输出目标为`react JSX`时,你可以使用`--jsxFactory`指定JSX工厂函数(默认值为`React.createElement`) # `as`操作符 回想一下怎么写类型断言: ```ts var foo = bar; ``` 这里断言`bar`变量是`foo`类型的。 因为TypeScript也使用尖括号来表示类型断言,在结合JSX的语法后将带来解析上的困难。因此,TypeScript在`.tsx`文件里禁用了使用尖括号的类型断言。 由于不能够在`.tsx`文件里使用上述语法,因此我们应该使用另一个类型断言操作符:`as`。 上面的例子可以很容易地使用`as`操作符改写: ```ts var foo = bar as foo; ``` `as`操作符在`.ts`和`.tsx`里都可用,并且与尖括号类型断言行为是等价的。 # 类型检查 为了理解JSX的类型检查,你必须首先理解固有元素与基于值的元素之间的区别。 假设有这样一个JSX表达式``,`expr`可能引用环境自带的某些东西(比如,在DOM环境里的`div`或`span`)或者是你自定义的组件。 这是非常重要的,原因有如下两点: 1. 对于React,固有元素会生成字符串(`React.createElement("div")`),然而由你自定义的组件却不会生成(`React.createElement(MyComponent)`)。 2. 传入JSX元素里的属性类型的查找方式不同。 固有元素属性*本身*就支持,然而自定义的组件会自己去指定它们具有哪个属性。 TypeScript使用[与React相同的规范](http://facebook.github.io/react/docs/jsx-in-depth.html#html-tags-vs.-react-components) 来区别它们。 固有元素总是以一个小写字母开头,基于值的元素总是以一个大写字母开头。 ## 固有元素 固有元素使用特殊的接口`JSX.IntrinsicElements`来查找。 默认地,如果这个接口没有指定,会全部通过,不对固有元素进行类型检查。 然而,如果这个接口存在,那么固有元素的名字需要在`JSX.IntrinsicElements`接口的属性里查找。 例如: ```ts declare namespace JSX { interface IntrinsicElements { foo: any } } ; // 正确 ; // 错误 ``` 在上例中,``没有问题,但是``会报错,因为它没在`JSX.IntrinsicElements`里指定。 > 注意:你也可以在`JSX.IntrinsicElements`上指定一个用来捕获所有字符串索引: ```ts declare namespace JSX { interface IntrinsicElements { [elemName: string]: any; } } ``` ## 基于值的元素 基于值的元素会简单的在它所在的作用域里按标识符查找。 ```ts import MyComponent from "./myComponent"; ; // 正确 ; // 错误 ``` 有两种方式可以定义基于值的元素: 1. 函数组件 (FC) 2. 类组件 由于这两种基于值的元素在JSX表达式里无法区分,因此TypeScript首先会尝试将表达式做为函数组件进行解析。如果解析成功,那么TypeScript就完成了表达式到其声明的解析操作。如果按照函数组件解析失败,那么TypeScript会继续尝试以类组件的形式进行解析。如果依旧失败,那么将输出一个错误。 ### 函数组件 正如其名,组件被定义成JavaScript函数,它的第一个参数是`props`对象。 TypeScript会强制它的返回值可以赋值给`JSX.Element`。 ```ts interface FooProp { name: string; X: number; Y: number; } declare function AnotherComponent(prop: {name: string}); function ComponentFoo(prop: FooProp) { return ; } const Button = (prop: {value: string}, context: { color: string }) =>