[React]基于Antd的FormModal的组件封装以及useFormModal的hooks封装

[React]基于Antd的FormModal的组件封装以及useFormModal的hooks封装

场景

很常见,打开弹窗输入表单等…

封装后,弹窗自行挂载到body上,只需关注表达逻辑和打开关闭逻辑,其它的已经帮你管理好了

源码

import React, { useRef, useMemo, memo, forwardRef, useCallback, useState, useImperativeHandle, useEffect } from 'react';
import { Modal, Form } from 'antd';
import type { ModalProps } from 'antd';
import { createPortal, render, unmountComponentAtNode } from 'react-dom';export const MyModal = memo(forwardRef((props: any, ref) => {useEffect(() => {console.log('modal had mounted')}, [])const [form] = Form.useForm();const [modalChildren, setModalChildren] = useState<React.ReactElement | null>(null);const [modalProps, setModalProps] = useState<ModalProps>({visible: false,...(props ?? {})});const typeRef = useRef<string>();const onFinish = useCallback((values: any) => {modalProps.onOk?.(values);}, [form, modalProps]);const onClose = useCallback(() => {if (typeRef.current === 'form') {form.resetFields();}setModalProps((source) => ({...source,visible: false,}));}, [form]);const onOpen = useCallback(() => {setModalProps((source) => ({...source,visible: true,}));}, [form]);useImperativeHandle(ref, () => ({injectChildren: (element) => {setModalChildren(element);},injectModalProps: (props) => {console.log(props)setModalProps((source) => {return {...source,...props,}});},open: () => {onOpen();},close: () => {onClose();},setFieldsValue: (values: any) => {form.setFieldsValue?.(values);},setType: (type: string) => {typeRef.current = type;}}), []);const handleOk = useCallback((e: any) => {if (typeRef.current === 'form') {form.submit();} else {modalProps.onOk?.(e);}}, [form, modalProps]);return (<Modal{...modalProps}onCancel={onClose}onOk={handleOk}>{modalChildren? React.cloneElement(modalChildren, typeRef.current === 'form'? {onFinish,form,onClose,}: { onClose }): null}</Modal>)
}));interface modalRefType {open: () => void;close: () => void;injectChildren: (child: React.ReactElement) => void;injectModalProps: (props: ModalProps) => void;setFieldsValue: (values: any) => void;setType: (type: string) => void;
}interface openArgType extends ModalProps {children?: React.ReactElement,type?: 'form' | 'default',initialValues?: {[key: string]: any;},
}const useMyModal = () => {const modalRef = useRef<modalRefType>();const handle = useMemo(() => {return {open: ({ children, type, initialValues, ...rest }: openArgType) => {console.log('modalRef.current: ', modalRef.current);modalRef.current?.setType(type ?? '');modalRef.current?.injectChildren(children ?? <div>111</div>);modalRef.current?.injectModalProps(rest);modalRef.current?.open();if (initialValues && type === 'form') {modalRef.current?.setFieldsValue?.(initialValues);}},close: () => {modalRef.current?.close();}};}, []);const containerRef = useRef<any>(document.createDocumentFragment())useEffect(() => {render(createPortal(<MyModal key="my-modal" ref={modalRef} />, document.body) as any, containerRef.current)return () => {unmountComponentAtNode(containerRef.current)}}, [])return [handle] as const;
}export default useMyModal

使用Demo

const [modalHandler] = useMyModal()// 表单组件内容
const AvailabilityForm = (props) => {return (<Formname="availability_form"{...props}><Form.Itemname="time"rules={REQUIRED_RULE}><DatePicker.RangePicker className='w-full' /></Form.Item></Form>)
}// 点击发布  
const handlePublish = useMemoizedFn(async () => {modalHandler.open({title: 'Available Date',type: 'form',initialValues: {},children: <AvailabilityForm></AvailabilityForm>,onOk: async (values: any) => {console.log('form vals', values);const timeRes: any[] = []console.log(Object.entries(values))Object.entries(values).forEach((item: any) => {const timeVal: any[] = item[1]if (Array.isArray(timeVal) && timeVal.length === 2) {timeRes.push({startTime: dayjs(timeVal[0]).utc().valueOf(),endTime: dayjs(timeVal[1]).utc().valueOf(),})}})try {const res = await netPublishListing(listingId, {availability: timeRes})if (res) {message.success('Success to publish')fetchItem()modalHandler.close();}console.log(res)} catch (err) {console.log('err: ', err);}},destroyOnClose: true,maskClosable: false});})

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/219301.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用 Timm 库替换 YOLOv8 主干网络 | 1000+ 主干融合YOLOv8

文章目录 前言版本差异说明替换方法parse_moedl( ) 方法_predict_once( ) 方法修改 yaml ,加载主干论文引用timm 是一个包含最先进计算机视觉模型、层、工具、优化器、调度器、数据加载器、数据增强和训练/评估脚本的库。 该库内置了 700 多个预训练模型,并且设计灵活易用。…

开发中需要防止用户哪些骚操作(测试鸭通关总结)

首先的话作为一个标准安全的后台认证和鉴权是一定要做好的&#xff0c;除此之外一下业务场景和常见问题也要进行考虑和预防。 一.收藏和点赞 网页前端和后端都要对点赞和收藏状态进行控制。简单说就是已经点赞和收藏的用户不能再进行点赞和收藏。 二.频繁请求某个接口 解决…

经典策略筛选-20231213

策略1&#xff1a; 龙头战法只做最强&#xff1a;国企改革 ----四川金顶 1、十日交易内出现 涨停或 &#xff08;涨幅大于7个点且量比大于3&#xff09; 2、JDK MACD RSI OBV LWR MTM 六指标共振 3、均线多头 4、 筹码峰 &#xff08;锁仓&#xff09; 5、现价> 五日均…

十二.镜头知识之镜头分辨率(解析力)

十二.镜头知识之镜头分辨率(解析力) 文章目录 十二.镜头知识之镜头分辨率(解析力)12.1 **分辨率与解像力的定义**12.1.0 分辨率定义12.1.1 解像力的定义12.1.2 解像力是分辨率的倒数12.1.3 Wavelength 对 物方分辨率的影响12.2 镜头分辨率跟相机分辨率如何才能匹配12.2.1 镜…

OkHttp: 拦截器和事件监听器

文章目录 1. 拦截器1. 拦截器链2. 实际案例1. 注册为应用拦截器2. 注册为网络拦截器 3. 如何选择用哪种拦截器1. 应用拦截器2. 网络层拦截器3. 重写请求4. 重写响应 4. 可用性 2. 事件监听器1. 请求的生命周期2. EventListener使用案例3. EventListener.Factory4. 调用失败的请…

【LeetCode】28. 找出字符串中第一个匹配项的下标 【字符串单模匹配:KMP算法】

题目链接 Python3 直觉解法 class Solution:def strStr(self, haystack: str, needle: str) -> int:pn, ph 0, 0n len(needle) h len(haystack)while ph < h:if haystack[ph] needle[pn]:if pn n-1: # 1234 123return ph - len(needle) 1else: pn 1ph 1else:…

RocketMQ的监控和管理工具有哪些❓

RocketMQ 提供了一些监控和管理工具&#xff0c;以便于用户对消息中间件的运行状态进行监控、管理和调优。以下是一些常用的 RocketMQ 监控和管理工具&#xff1a; 1. RocketMQ Console RocketMQ Console 是 RocketMQ 官方提供的监控和管理工具&#xff0c;提供了直观的图形界…

【LeetCode刷题】-- 163.缺失的区间

163.缺失的区间 class Solution {public List<List<Integer>> findMissingRanges(int[] nums, int lower, int upper) {List<List<Integer>> res new ArrayList<>();for(int num : nums){if(lower < num){res.add(Arrays.asList(lower,num -…

c语言函数与指针

//本文有待补充。 一、函数 1.函数的定义 如果程序的逻辑比较复杂、代码量比较大&#xff0c;或者重复性功能比较多&#xff0c;那么全部写在主函数里就会显得十分冗长和杂乱。为了使代码更加简洁、思路更加清晰&#xff0c;C语言提供了”函数“。函数是一个实现一定功能的语…

dockerfile创建镜像 lNMP+wordpress

dockerfile创建镜像 lNMPwordpress nginx dockernginx mysql dockermysql php dockerphp nginx vim nginx.conf vim Dockerfile docker network create --subnet172.17.0.0/16 --opt "com.docker.network.bridge.name""docker1" mynetwork docker buil…

Web是什么?它具体的功能是什么?它值不值得我们去学习?我们该如何去学习?

Web是指“World Wide Web”的缩写&#xff0c;是互联网上的一种信息系统&#xff0c;通过超文本链接方式将全球各地的文档链接在一起&#xff0c;形成一个巨大的信息资源库。Web的基本构成包括网页、超文本传输协议&#xff08;HTTP&#xff09;、网页浏览器等。 网页是Web的基…

失败的2x2 Mipi Raw10转RGB565

反转了&#xff0c;有思路改了&#xff0c;待我思考一番。 --------------------------------------------------------------------------------------------------------------------------------- 希望完成的目标&#xff1a; MIPI在解析以后是四个像素四个像素地产出数据…

最新Applestore建立其他地区账号简单快捷一看就会

1、首先打开创建appleid网站 2、点击创建你的Apple ID开始创建&#xff08;这里以美国为例&#xff09; 电话号码可以填大陆手机号即可 这两个选项建议不要勾选 3、更改付款方式 3.1点击付款与配送 3.2添加付款方式&#xff0c;这里是最重要的一步&#xff0c;传统方法已经无法…

真正的力量不是摧毁,而是在困境中保持微笑,坚持向前。

真正的力量不是摧毁&#xff0c;而是在困境中保持微笑&#xff0c;坚持向前。

Mac电脑投屏AirServer 2024怎么下载安装激活许可期限

对于那些想要将 iPhone、iPad 或其他 iOS 设备上的小屏幕镜像到计算机上的大屏幕的人来说&#xff0c;AirPlay 是一个很好的工具。 基于此&#xff0c;AirServer 非常需要将您的 Mac 或 PC 变成 AirPlay 设备。 但是如何使用计算机上的设置对 iPhone 等 iOS 设备进行屏幕镜像&a…

docker save多个镜像打包成一个tar.gz压缩文件

docker save多个镜像打包成一个tar.gz压缩文件 有时候我们需要将docker中的多个镜像批量的传输到另一台机器&#xff0c;如果通过docker save这种命令则需要制作多个tar文件&#xff0c;这样以来冗余的操作较多而且tar文件占据的空间较大&#xff0c;不利于传输。 可以通过以…

Netperf使用总结

什么是Netperf Netperf是由惠普公司开发的一种网络性能测量工具&#xff0c;主要针对基于TCP或UDP的传输。Netperf根据应用的不同&#xff0c;可以进行不同模式的网络性能测试&#xff0c;即批量数据传输&#xff08;bulk data transfer&#xff09;模式和请求/应答&#xff0…

Ubuntu 22安装PHP环境

参考博客为《练习 0&#xff08;2/2&#xff09;&#xff1a;Ubuntu 环境下安装PHP&#xff08;PHP-FPM&#xff09;》和《原生态Ubuntu部署LAMP环境 PHP8.1MySQLApache》 sudo apt-get install -y php7.4想要安装php7.4&#xff0c;发现安装的是php8.1。 完成如下图&#xf…

超声波测距HC-SR04模块的简单应用

文章目录 一、HC-SR04HC-SR04是什么&#xff1f;HC-SR04测距的原理 二、使用步骤1.硬件最远探测距离调节硬件连接 2.软件1.初始化配置代码如下&#xff08;示例&#xff09;&#xff1a;引脚初始化定时器初始化 2.引脚输入输出配置代码如下&#xff08;示例&#xff09;&#x…

【linux系统】用户功能与权限详细总结

前言 菜某的笔记总结&#xff0c;有错误还请指正。 linux用户的概念与root用户 这么理解&#xff1a;一台电脑有多个操作者&#xff0c;每个操作者只能无限制操作自己文件夹中的东西&#xff0c;其他地方的操作需要给与相应权限才能操作。 root用户&#xff1a;就是最高级的…