React Hooks forwardRef useImperativeHandle TypeScript类型 最佳实践

定义

官网文档

forwardRef : 转发 ref
useImperativeHandle : 用于自定义暴露给父组件的实例值

版本

"react": "^17.0.2"
"@types/react": "^17.0.44"

代码

父组件

import { FC, useEffect, useRef } from 'react';

// 导入子组件
import ChildComponent, { IRef } from './ChildComponent';

let MyConponent: FC = function () {
  // 指定泛型,对 ref 进行类型约束
  let testRef = useRef<IRef>(null);

  useEffect(() => {
    console.log(testRef);
    // 调用子组件的方法
    testRef.current?.MyTestFn();
  });

  return (
    <>
      <ChildComponent ref={testRef} />
    </>
  );
};

export default MyConponent;

子组件

import { useMemoizedFn } from 'ahooks';
import { forwardRef, memo, useImperativeHandle } from 'react';

export interface IRef {
  MyTestFn: () => void;
}

export interface IProps {
  myName?: string;
}

// 泛型第一个参数是 ref类型
// 第二个参数是 props 的类型 (可以不填写,默认是 { } )

// 指定上这两个类型, 享受 TypeScript 完美的开发体验
let ChildComponent = forwardRef<IRef, IProps>((props, forwardedRef) => {
  // 定义一个普通方法
  let test = useMemoizedFn(() => {
    alert('我是子组件');
  });

  // 暴露给 ref 的方法
  useImperativeHandle(
    forwardedRef,
    () => ({
      MyTestFn: test,
    }),
    [test], // 依赖项,  同 useEffect
  );

  // 由于 useImperativeHandle 的 @types 中写的是 "<T, R extends T>"
  // 返回值 extends 了 IRefProps 所以可以多写,ts也不会报错(比如下面的"a").
  // 由于interface中没有定义"a",所以使用的时候ts会报错.难道是作者写的 bug ???
  // 建议还是把"a"加到interface中获得更好的 ts 体验
  // 例子:
  // () => ({
  //   MyTestFn: test,
  //   a: "123ew"
  // }),

  return <p>子组件 {props.myName}</p>;
});

// 设置 props 的默认值
ChildComponent.defaultProps = {
  myName: '小明',
};

export default memo(ChildComponent);
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我了解了您的问题。对于这个问题,您需要使用 React Hooks 和 TypeScript 来结合 Ant Design 进行表单组件的二次封装和联动。 首先,您需要在 React 中定义一个表单组件,并使用 useState 钩来处理输入值的状态。例如: ``` import React, { useState } from 'react'; import { Form, Input, Button } from 'antd'; interface MyFormProps { initialValues?: MyFormValues; onSubmit?: (values: MyFormValues) => void; } interface MyFormValues { name: string; age: number; email: string; } const MyForm: React.FC<MyFormProps> = ({ initialValues = {}, onSubmit }) => { const [name, setName] = useState<string>(initialValues.name ?? ''); const [age, setAge] = useState<number>(initialValues.age ?? 0); const [email, setEmail] = useState<string>(initialValues.email ?? ''); const handleSubmit = (event: React.FormEvent) => { event.preventDefault(); if (onSubmit) { onSubmit({ name, age, email }); } }; const handleNameChange = (event: React.ChangeEvent<HTMLInputElement>) => { setName(event.target.value); }; const handleAgeChange = (event: React.ChangeEvent<HTMLInputElement>) => { setAge(parseInt(event.target.value, 10)); }; const handleEmailChange = (event: React.ChangeEvent<HTMLInputElement>) => { setEmail(event.target.value); }; return ( <Form layout="vertical" onSubmit={handleSubmit}> <Form.Item label="Name"> <Input value={name} onChange={handleNameChange} /> </Form.Item> <Form.Item label="Age"> <Input type="number" value={age} onChange={handleAgeChange} /> </Form.Item> <Form.Item label="Email"> <Input type="email" value={email} onChange={handleEmailChange} /> </Form.Item> <Form.Item> <Button type="primary" htmlType="submit"> Submit </Button> </Form.Item> </Form> ); }; export default MyForm; ``` 接下来,您需要在 Form.Item 组件中使用 getFieldDecorator 来封装每个表单项,并使用 onChange 钩来处理联动事件。例如: ``` import React from 'react'; import { Form, Input, Button } from 'antd'; interface MyFormProps { initialValues?: MyFormValues; onSubmit?: (values: MyFormValues) => void; } interface MyFormValues { name: string; age: number; email: string; } const MyForm: React.FC<MyFormProps> = ({ initialValues = {}, onSubmit }) => { const handleSubmit = (event: React.FormEvent) => { event.preventDefault(); if (onSubmit) { onSubmit(form.getFieldsValue()); } }; const form = Form.useForm()[0]; const handleNameChange = (event: React.ChangeEvent<HTMLInputElement>) => { form.setFieldsValue({ name: event.target.value }); }; const handleAgeChange = (event: React.ChangeEvent<HTMLInputElement>) => { form.setFieldsValue({ age: parseInt(event.target.value, 10) }); }; const handleEmailChange = (event: React.ChangeEvent<HTMLInputElement>) => { form.setFieldsValue({ email: event.target.value }); }; return ( <Form layout="vertical" onSubmit={handleSubmit} form={form}> <Form.Item label="Name"> {form.getFieldDecorator('name', { initialValue: initialValues.name, })(<Input onChange={handleNameChange} />)} </Form.Item> <Form.Item label="Age"> {form.getFieldDecorator('age', { initialValue: initialValues.age, })(<Input type="number" onChange={handleAgeChange} />)} </Form.Item> <Form.Item label="Email"> {form.getFieldDecorator('email', { initialValue: initialValues.email, })(<Input type="email" onChange={handleEmailChange} />)} </Form.Item> <Form.Item> <Button type="primary" htmlType="submit"> Submit </Button> </Form.Item> </Form> ); }; export default MyForm; ``` 如上所示,您可以使用 getFieldDecorator 来封装每个表单项,并使用 initialValue 属性来设置初始值。在 onChange 钩中,您可以使用 setFieldsValue 来联动更新其他表单项的值。 这就是使用 React Hooks 和 TypeScript 结合 Ant Design 进行表单组件的二次封装和联动的示例。希望对您有所帮助!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值