web-dev-qa-db-ja.com

onKeyDownイベントがReactのdivで機能しない

ReactのdivでkeyDownイベントを使用したい。私がやります:

  componentWillMount() {
      document.addEventListener("keydown", this.onKeyPressed.bind(this));
  }

  componentWillUnmount() {
      document.removeEventListener("keydown", this.onKeyPressed.bind(this));
  }      

  onKeyPressed(e) {
    console.log(e.keyCode);
  }

  render() {
    let player = this.props.boards.dungeons[this.props.boards.currentBoard].player;
    return (
      <div 
        className="player"
        style={{ position: "absolute" }}
        onKeyDown={this.onKeyPressed} // not working
      >
        <div className="light-circle">
          <div className="image-wrapper">
            <img src={IMG_URL+player.img} />
          </div>
        </div>
      </div>
    )
  }

それはうまくいきますが、私はもっとReactスタイルでやりたいです。私は試した

        onKeyDown={this.onKeyPressed}

コンポーネント上。しかし、反応しません。私が思い出すように、入力要素で動作します。

コードペン: http://codepen.io/lafisrap/pen/OmyBYG

どうすればいいですか?

53
Michael

ReactのdivでonKeyDownイベントをリッスンするには、tabIndex属性を使用する必要があります。 tabIndex = "0"を設定すると、ハンドラーが起動します。

90
burakhan alkan

このように書く必要があります

<div 
    className="player"
    style={{ position: "absolute" }}
    onKeyDown={this.onKeyPressed}
    tabIndex="0"
  >

onKeyPressedthisにバインドされていない場合は、矢印関数を使用して書き換えるか、コンポーネントconstructorにバインドしてください。

13
Panther

あなたは純粋なJavascriptで考えすぎています。これらのReactライフサイクルメソッドのリスナーを取り除き、event.keyの代わりにevent.keyCodeを使用します(これはJSイベントオブジェクトではないため、React SyntheticEvent です)。コンポーネント全体をこれと同じくらい単純にすることができます(コンストラクターでメソッドをバインドしていない場合)。

onKeyPressed(e) {
  console.log(e.key);
}

render() {
  let player = this.props.boards.dungeons[this.props.boards.currentBoard].player;
  return (
    <div 
      className="player"
      style={{ position: "absolute" }}
      onKeyDown={(e) => this.onKeyPressed(e)}
    >
      <div className="light-circle">
        <div className="image-wrapper">
          <img src={IMG_URL+player.img} />
        </div>
      </div>
    </div>
  )
}
2
steel