# 介绍
[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`标识符是写死的硬编码,所以你必须保证React(大写的R)是可用的。*
# `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. 无状态函数组件 (SFC)
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 }) =>