将输入字段聚焦在React组件中 - 尝试创建ref时出现类型错误

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了将输入字段聚焦在React组件中 - 尝试创建ref时出现类型错误相关的知识,希望对你有一定的参考价值。

我有一个包含文本<input>元素的React组件。安装组件时,我希望在输入字段中设置文本光标,即我希望文本输入元素具有焦点。

在“传统的”javascript React组件中,我将通过ref获取输入字段的DOM元素,然后调用其focus方法。

我已经阅读了这篇文档,解释了如何在Reason-React中使用引用:https://github.com/reasonml/reason-react/blob/master/docs/react-ref.md

唉,这个页面上包含的代码示例是针对自定义组件的引用,它只提到它也适用于React DOM元素。

所以我试图将示例代码转换为React DOM元素,这是我到目前为止所尝试的:

type state = {
  text: string,
  inputElement: ref(option(Dom.element))
};

let valueFromEvent = (evt) : string => (
  evt
  |> ReactEventRe.Form.target
  |> ReactDOMRe.domElementToObj
)##value;

let component = ReasonReact.reducerComponent("EditTodoField");

let setInputElement = (theRef, {ReasonReact.state}) =>
  state.inputElement := Js.Nullable.to_opt(theRef);

let make = (~initialText, ~onSubmit, _) => {
  ...component,
  initialState: () => {text: initialText, inputElement: ref(None)},
  reducer: (newText, state) => ReasonReact.Update({...state, text: newText}),
  render: ({state: {text}, reduce, handle}) =>
    <input
      value=text
      _type="text"
      ref=(handle(setInputElement))
      placeholder="Todo description"
      onChange=(reduce((evt) => valueFromEvent(evt)))
      onKeyDown=(
        (evt) =>
          if (ReactEventRe.Keyboard.key(evt) == "Enter") {
            onSubmit(text);
            (reduce(() => ""))()
          } else if (ReactEventRe.Keyboard.key(evt) == "Escape") {
            onSubmit(initialText);
            (reduce(() => ""))()
          }
      )
    />
};

我得到的错误信息是这样的:

We've found a bug for you!
/Users/pahund/git/todo-list-reason-react/src/EditTodoField.re 21:11-35

19 ┆ value=text
20 ┆ _type="text"
21 ┆ ref=(handle(setInputElement))
22 ┆ placeholder="Todo description"
23 ┆ onChange=(reduce((evt) => valueFromEvent(evt)))

This has type:
  ReasonReact.Callback.t(Js.Nullable.t(Dom.element)) (defined as
    (Js.Nullable.t(Dom.element)) => unit)
But somewhere wanted:
  (Js.null(Dom.element)) => unit

The incompatible parts:
  Js.Nullable.t(Dom.element) (defined as Js.nullable(Dom.element))
  vs
  Js.null(Dom.element)

我知道问题可能在于我如何定义代码开头的状态类型,对于DOM元素而言,它与自定义组件的不同。

这里修复bug的正确类型定义是什么?

完整的项目可以在GitHub上找到:https://github.com/pahund/todo-list-reason-react/tree/ref-problem

答案

我认为你的理由 - 反应依赖已经过时了。 refs在0.3.0从Js.null(Dom.element)变为Js.nullable(Dom.element)。见https://github.com/reasonml/reason-react/blob/master/HISTORY.md#030

如果由于某种原因你不能或拒绝升级,你可以只使用Js.Null.to_opt而不是:)

(另外,如果你升级,你可以使用Js.toOption代替Js.Nullable.to_opt作为一个很好的捷径)

以上是关于将输入字段聚焦在React组件中 - 尝试创建ref时出现类型错误的主要内容,如果未能解决你的问题,请参考以下文章

如何在(Ionic + React.js App)中自动聚焦 IonInput/OtpInput 字段,我已经尝试了所有解决方案

在React Form无状态组件和状态完全Root组件之间传递多个输入字段

React 如何使用 refs 来聚焦/模糊?

如何使用 React refs 来聚焦 Redux Form 字段?

React Native:如何聚焦包装在自定义组件中的 TextInput?

如何在 React Js 中编辑输入值?