在Typescript中反应事件类型

时间:2017-10-02 11:43:34

标签: reactjs typescript

当我在事件处理程序中使用MouseEvent类型时,我无法构建我的React / Typescript应用程序:

private ButtonClickHandler(event: MouseEvent): void
{
    ...
}

我明白了:

error TS2322: Type '{ onClick: (event: MouseEvent) => void; children: string; }' is not assignable to type 'DetailedHTMLProps<ButtonHTMLAttributes<HTMLButtonElement>, HTMLButtonElement>'.
  Type '{ onClick: (event: MouseEvent) => void; children: string; }' is not assignable to type 'ButtonHTMLAttributes<HTMLButtonElement>'.
    Types of property 'onClick' are incompatible.
      Type '(event: MouseEvent) => void' is not assignable to type 'EventHandler<MouseEvent<HTMLButtonElement>>'.
        Types of parameters 'event' and 'event' are incompatible.
          Type 'MouseEvent<HTMLButtonElement>' is not assignable to type 'MouseEvent'.
            Property 'fromElement' is missing in type 'MouseEvent<HTMLButtonElement>'.`

我也试过了MouseEvent<HTMLButtonElement>,但后来我得到了error TS2315: Type 'MouseEvent' is not generic.

为什么?

摘自package.json

 "devDependencies": {
    "@types/react": "^16.0.7",
    "@types/react-dom": "^15.5.5",
    "ts-loader": "^2.3.7",
    "typescript": "^2.5.3",
    "webpack": "^3.6.0"
  },
  "dependencies": {
    "react": "^16.0.0",
    "react-dom": "^16.0.0"
  }

编辑#1:

render()中的绑定:

<button onClick={ this.ButtonClickHandler }>Toggle</button>
constructor中的

this.ButtonClickHandler = this.ButtonClickHandler.bind(this);

3 个答案:

答案 0 :(得分:19)

我认为这里发生的事情是有两种不同的类型:来自jsx / lib / js / js / web.jsx的MouseEvent(没有类型参数)和{{1来自@ types / react / index.d.ts(它有一个类型参数React.MouseEvent<T>,它来自的元素的类型)。 Typescript使用structural typing,因此即使它们是不同的类型,它们也可以兼容。但是为了与T处理程序兼容,您需要使用React的版本,onClick的适当类型(T将起作用,或者您可以更具体如果你知道你将这个处理程序附加到哪个元素。)

因此HTMLElement失败,因为event: MouseEvent需要事件类型的专用版本。 onClick失败,因为它引用了错误的event: MouseEvent<HTMLElement>类型,没有类型参数的类型。 MouseEvent有效,因为您获得了正确的类型,为其提供了所需的参数,并且结果类型与React.MouseEvent<HTMLElement>处理程序兼容。

答案 1 :(得分:2)

你不相信它:将构造函数赋值移动到渲染函数就足够了:

render()
{
return (
<button onClick={ this.Button_Click.bind(this) }>Toggle</button>
)}

然后MouseEvent变得可用:

private Button_Click(event: MouseEvent): void

为什么会这样?

答案 2 :(得分:0)

我遇到了同样的问题,但是能够通过从MouseEvent显式导入React来解决此问题

我对此有误

import React from 'react';

但是已经解决了

import React, { MouseEvent } from 'react';