React 如何设计一个弹窗组件

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了React 如何设计一个弹窗组件相关的知识,希望对你有一定的参考价值。

参考技术A 鉴于以上需求,将整个弹窗拆分为一个UI组件(Dialog),一个处理函数部分(Staticize)

react实现Modal弹窗

一、Dialog.js文件

import React, useMemo, useEffect, useState from 'react'
import ReactDOM from 'react-dom'


/**
 *
 * 需要把元素渲染到组件之外,用 createPortal 把元素直接渲染到 document.body 下,为了防止函数组件每一次执行都触发 createPortal, 所以通过 useMemo 做性能优化。

 因为需要渐变的动画效果,所以需要两个变量 modelShow / modelShowAync 来控制显示/隐藏,modelShow 让元素显示/隐藏,modelShowAync 控制动画执行。

 当弹窗要显示的时候,要先设置 modelShow 让组件显示,然后用 setTimeout 调度让 modelShowAync 触发执行动画。
 当弹窗要隐藏的时候,需要先让动画执行,所以先控制 modelShowAync ,然后通过控制 modelShow 元素隐藏,和上述流程相反。
 用一个控制器 controlShow 来流畅执行更新任务。

 */


// 控制弹窗隐藏以及动画效果
const controlShow = (f1, f2, value, timer) => 
    f1(value)
    return setTimeout(() => 
        f2(value)
    , timer)



export const Dialog = (props) => 
    const width, visible, closeCb, onClose = props
    // 控制 modelShow动画效果
    const [modelShow, setModelShow] = useState(visible)
    const [modelShowAsync, setModelShowAsync] = useState(visible)

    const renderChildren = useMemo(() => 
        // 把元素渲染到组件之外的document.body 上
        return ReactDOM.createPortal(<div style=display: modelShow ? 'block' : 'none'>
            <div className="model_container" style=opacity: modelShowAsync ? 1 : 0>
                <div className="model_wrap">
                    <div style=width: width + 'px'> props.children </div>
                </div>
            </div>
            <div className="model_container mast" onClick=() => onClose && onClose()
                 style=opacity: modelShowAsync ? 0.6 : 0/>
        </div>, document.body)
    , [modelShow, modelShowAsync])

    useEffect(() => 
        let timer
        if (visible) 
            // 打开弹窗,
            timer = controlShow(setModelShow, setModelShowAsync, visible, 30)
         else 
            timer = controlShow(setModelShowAsync,setModelShow,visible,1000)
        
        return () => 
            timer && clearTimeout(timer)
        
    , [visible])
    return renderChildren


二、Modal.js

import Dialog from "./Dialog";
import React, useEffect, useState from 'react'
import ReactDOM from 'react-dom'
import './style.scss'

class Modal extends React.PureComponent 
    // 渲染底部按钮
    renderFooter = () => 
        const onOk, onCancel, cancelText, okText, footer = this.props
        //    触发onOk / onCancel回调
        if (footer && React.isValidElement(footer)) return footer
        return <div className="model_bottom">
            <div className="model_btn_box">
                <button className="searchbtn"
                        onClick=(e) => 
                            onOk && onOk(e)
                        >okText || '确定'
                </button>
                <button className="concellbtn"
                        onClick=(e) => 
                            onCancel && onCancel(e)
                        >cancelText || '取消'
                </button>
            </div>
        </div>
    

    // 渲染底部
    renderTop = () => 
        const title, onClose = this.props
        return <div className="model_top">
            <p>title</p>
            <span className="model_top_close" onClick=() => onClose && onClose()>X</span>
        </div>
    

    // 渲染弹窗内容
    renderContent = () => 
        const content, children = this.props
        return React.isValidElement(content) ? content : children ? children : null
    

    render() 
        const visible, width = 500, closeCb, onClose = this.props
        return <Dialog
            closeCb=closeCb
            onClose=onClose
            visible=visible
            width=width
        >
            this.renderTop()
            this.renderContent()
            this.renderFooter()
        </Dialog>
    



// 静态方法
let ModalContainer = null
const modelSymbol = Symbol('$$_model_Container_hidden')

// 静态属性show——控制
Modal.show = (config) => 
    //  如果modal已经存在,name就不需要第二次show
    if (ModalContainer) return
    const props = ...config, visible: true
    const container = ModalContainer = document.createElement('div')
    // 创建一个管理者,管理model状态
    const manager = container[modelSymbol] = 
        setShow: null,
        mounted: false,
        hidden() 
            const setShow = manager
            setShow && setShow(false)
        ,
        destroy() 
            //    卸载组件
            ReactDOM.unmountComponentAtNode(container)
            // 移除节点
            document.body.removeChild(container)
            // 置空元素
            ModalContainer = null
        
    

    const ModelApp = (props) => 
        const [show, setShow] = useState(false)
        manager.setShow = setShow
        const visible, ...trueProps = props
        useEffect(() => 
            // 加载完成,设置状态
            manager.mounted = true
            setShow(visible)
        , [])
        return <Modal ...trueProps closeCb=() => manager.mounted && manager.destroy() visible=show/>
    
    // 插入到body中
    document.appendChild(container)
    // 渲染React元素
    ReactDOM.render(<ModelApp/>, container)
    return manager


Modal.hidden = () => 
    if(!ModalContainer) return
    // 如果存在ModalContainer 那么隐藏ModalContainer
    ModalContainer[modelSymbol] && ModalContainer[modelSymbol].hidden()


export default Modal

三、style.scss样式文件

$bg-linear-gradien-red-light : linear-gradient(135deg, #fc4838 0%, #f6346b  100%);
$bg-linear-gradien-red-dark : linear-gradient(135deg, #fc4838 0%, #f6346b  100%);

.constrol
  padding: 30px;
  width: 500px;
  border: 1px solid #ccc;
  height: 200px;


.feel
  padding: 24px;


.model_top
  height: 40px;
  border-radius: 5px  5px 0 0 ;
  position: relative;
  p
    height: 40px;
    font-weight: bold;
    line-height: 40px;
    padding-left: 14px;
  
  background-color: #eee;
  .model_top_close
    position: absolute;
    font-size: 16px;
    cursor: pointer;
    right: 24px;
    top: 8px;
  


.model_bottom
  height: 50px;
  padding-top: 10px;
  .model_btn_box
    display: inline-block;
    margin-left: 50%;
    transform: translateX(-50%);
  


.model_container
  .model_wrap
    position: absolute;
    border-radius:5px ;
    background: #fff;
    left:50%;
    top:50%;
    transform: translate(-50%,-50%);
  
  position: fixed;
  z-index: 10000;
  left:0;
  top:0;
  transition: opacity 0.3s;
  right: 0;
  bottom: 0;



.mast
  background-color: #000;
  z-index: 9999;


.searchbtn
  background: linear-gradient(135deg, #fc4838 0%, #f6346b  100%);
  color: #fff;
  min-width: 96px;
  height :36px;
  border :none;
  border-radius: 18px;
  font-size: 14px;
  font-weight: 500;
  cursor: pointer;
  margin-left: 20px !important;

.searchbtn:focus
  background: $bg-linear-gradien-red-dark;
  color: #fff;
  min-width: 96px;
  height: 36px;
  border: none;
  border-radius: 18px;
  font-size: 14px;
  font-weight: 500;
  cursor: pointer;
  margin-left: 20px !important;
  box-shadow: 0 2px 7px 0 #FAA79B;

.searchbtn:hover
  background :$bg-linear-gradien-red-light;
  color :#fff;
  min-width: 96px;
  height :36px;
  margin-left: 20px !important;
  border: none;
  border-radius: 18px;
  font-size :14px;
  font-weight: 500;
  cursor: pointer;
  box-shadow: 0 2px 7px 0 #FAA79B;

.searchbtn:disabled
  background: #c0c6c6;
  color :#fff;
  min-width: 96px;
  height :36px;
  font-size :14px;
  font-weight: 500;
  border: none;
  border-radius: 18px;
  cursor: not-allowed;


.concellbtn
  background :#fff;
  color :#303133;
  width: 96px;
  height: 36px;
  font-size: 14px;
  font-weight: 500;
  border :1px solid #E4E7ED;
  border-radius: 18px;
  cursor: pointer;
  // margin-right: 10px;
  margin-left: 10px;

.concellbtn:hover
  background :rgba(220, 223, 230, 0.1);
  color: #303133;
  width :96px;
  height: 36px;
  font-size: 14px;
  font-weight: 500;
  border :1px solid #E4E7ED;
  border-radius: 18px;
  cursor: pointer;
  // margin-right: 10px;
  margin-left: 10px;

.concellbtn:focus
  background :rgba(220, 223, 230, 0.24);
  color: #303133;
  width :96px;
  height: 36px;
  font-size: 14px;
  font-weight: 500;
  border: 1px solid #C0C4CC;
  border-radius: 18px;
  cursor: pointer;
  margin-right: 10px;
  margin-left: 10px;


四、调用例子

import React, useState, useMemo from 'react'
import Modal from './customPopup/Modal'

/* 挂载方式调用modal */
export default function App() 
    const [ visible , setVisible ] = useState(false)
    const [ nameShow , setNameShow ] = useState(false)
    const handleClick = () => 
        setVisible(!visible)
        setNameShow(!nameShow)
    
    /* 防止 Model 的 PureComponent 失去作用 */
    const [ handleClose ,handleOk, handleCancel ] = useMemo(()=>
        const Ok = () =>  console.log('点击确定按钮')
        const Close = () => setVisible(false)
        const Cancel = () => console.log('点击取消按钮')
        return [Close , Ok , Cancel]
    ,[])

    return <div>
        <Modal
            onCancel=handleCancel
            onClose=handleClose
            onOk=handleOk
            title='标题'
            visible=visible
            width=700
        >
            <div className="feel" >
              内容。。。。。。。
            </div>
        </Modal>
        <button onClick=() => 
            setVisible(!visible)
            setNameShow(false)
        
        > model show </button>
        <button onClick=handleClick > model show ( 显示作者 ) </button>
    </div>


实现效果

以上是关于React 如何设计一个弹窗组件的主要内容,如果未能解决你的问题,请参考以下文章

如何将设计数据传递给React组件?

我的react学习之行-03如何设计高质量的React组件

我的react学习之行-03如何设计高质量的React组件

如何设计 React 代码结构?

如何设计 React Slick carrousel 的箭头样式?

从零开始设计一个右键菜单组件