React實(shí)現(xiàn)動(dòng)效彈窗組件
我們?cè)趯懸恍?UI 組件時(shí),若不考慮動(dòng)效,就很容易實(shí)現(xiàn),主要就是有無(wú)的切換(類似于 Vue 中的 v-if 屬性)或者可見性的切換(類似于 Vue 中的 v-show 屬性)。

1. 沒有動(dòng)效的彈窗
在 React 中,可以這樣來(lái)實(shí)現(xiàn):
interface ModalProps {
open: boolean;
onClose?: () => void;
children?: any;
}
const Modal = ({open. onClose, children}: ModalProps) => {
if (!open) {
return null;
}
return createPortal(<div>
<div classname="modal-content">{children}</div>
<div classname="modal-close-btn" onclick="{onClose}">x</div>
</div>, document.body);
};
使用方式:
const App = () => {
const [open, setOpen] = useState(false);
return (
<div classname="app">
<button onclick="{()" ==""> setOpen(true)}>show modal</button>
<modal open="{open}" onclose="{()" ==""> setOpen(false)}>
modal content
</modal>
</div>
);
};
我們?cè)谶@里就是使用open屬性來(lái)控制展示還是不展示,但完全沒有漸變的效果。
若我們想實(shí)現(xiàn) fade, zoom 等動(dòng)畫效果,還需要對(duì)此進(jìn)行改造。

2. 自己動(dòng)手實(shí)現(xiàn)有動(dòng)效的彈窗
很多同學(xué)在自己實(shí)現(xiàn)動(dòng)效時(shí),經(jīng)常是展示的時(shí)候有動(dòng)效,關(guān)閉的時(shí)候沒有動(dòng)效。都是動(dòng)效的時(shí)機(jī)沒有控制好。這里我們先自己來(lái)實(shí)現(xiàn)一下動(dòng)效的流轉(zhuǎn)。
剛開始我實(shí)現(xiàn)的時(shí)候,動(dòng)效只有開始狀態(tài)和結(jié)束狀態(tài),需要很多的變量和邏輯來(lái)控制這個(gè)動(dòng)效。
后來(lái)我參考了react-transition-group組件的實(shí)現(xiàn),他是將動(dòng)效拆分成了幾個(gè)部分,每個(gè)部分分別進(jìn)行控制。
- 展開動(dòng)效的順序:enter -> enter-active -> enter-done;
- 關(guān)閉動(dòng)效的順序:exit -> exit-active -> exit-done;
動(dòng)效過程在enter-active和exit-active的過程中。
我們?cè)偻ㄟ^一個(gè)變量 active 來(lái)控制是關(guān)閉動(dòng)效是否已執(zhí)行關(guān)閉,參數(shù) open 只控制是執(zhí)行展開動(dòng)效還是關(guān)閉動(dòng)效。
當(dāng) open 和 active 都為 false 時(shí),才銷毀彈窗。
const Modal = ({ open, children, onClose }) => {
const [active, setActive] = useState(false); // 彈窗的存在周期
if (!open && !active) {
return null;
}
return ReactDOM.createPortal(
<div classname="modal">
<div classname="modal-content">{children}</div>
<div classname="modal-close-btn" onclick="{onClose}">
x
</div>
</div>,
document.body,
);
};
這里我們接著添加動(dòng)效過程的變化:
const [aniClassName, setAniClassName] = useState(''); // 動(dòng)效的class
// transition執(zhí)行完畢的監(jiān)聽函數(shù)
const onTransitionEnd = () => {
// 當(dāng)open為rue時(shí),則結(jié)束狀態(tài)為'enter-done'
// 當(dāng)open未false時(shí),則結(jié)束狀態(tài)為'exit-done'
setAniClassName(open ? 'enter-done' : 'exit-done');
// 若open為false,則動(dòng)畫結(jié)束時(shí),彈窗的生命周期結(jié)束
if (!open) {
setActive(false);
}
};
useEffect(() => {
if (open) {
setActive(true);
setAniClassName('enter');
// setTimeout用來(lái)切換class,讓transition動(dòng)起來(lái)
setTimeout(() => {
setAniClassName('enter-active');
});
} else {
setAniClassName('exit');
setTimeout(() => {
setAniClassName('exit-active');
});
}
}, [open]);
Modal 組件完整的代碼如下:
const Modal = ({ open, children, onClose }) => {
const [active, setActive] = useState(false); // 彈窗的存在周期
const [aniClassName, setAniClassName] = useState(''); // 動(dòng)效的class
const onTransitionEnd = () => {
setAniClassName(open ? 'enter-done' : 'exit-done');
if (!open) {
setActive(false);
}
};
useEffect(() => {
if (open) {
setActive(true);
setAniClassName('enter');
setTimeout(() => {
setAniClassName('enter-active');
});
} else {
setAniClassName('exit');
setTimeout(() => {
setAniClassName('exit-active');
});
}
}, [open]);
if (!open && !active) {
return null;
}
return ReactDOM.createPortal(
<div classname="{'modal" '="" +="" aniclassname}="" ontransitionend="{onTransitionEnd}">
<div classname="modal-content">{children}</div>
<div classname="modal-close-btn" onclick="{onClose}">
x
</div>
</div>,
document.body,
);
};
動(dòng)效的流轉(zhuǎn)過程已經(jīng)實(shí)現(xiàn)了,樣式也要一起寫上。比如我們要實(shí)現(xiàn)漸隱漸現(xiàn)的 fade 效果:
.enter {
opacity: 0;
}
.enter-active {
transition: opacity 200ms ease-in-out;
opacity: 1;
}
.enter-done {
opacity: 1;
}
.exit {
opacity: 1;
}
.exit-active {
opacity: 0;
transition: opacity 200ms ease-in-out;
}
.exit-done {
opacity: 0;
}
如果是要實(shí)現(xiàn)放大縮小的 zoom 效果,修改這幾個(gè) class 就行。
一個(gè)帶有動(dòng)效的彈窗就已經(jīng)實(shí)現(xiàn)了。
使用方式:
const App = () => {
const [open, setOpen] = useState(false);
return (
<div classname="app">
<button onclick="{()" ==""> setOpen(true)}>show modal</button>
<modal open="{open}" onclose="{()" ==""> setOpen(false)}>
modal content
</modal>
</div>
);
};
點(diǎn)擊鏈接自己實(shí)現(xiàn)動(dòng)效的 React 彈窗 demo查看效果。
類似地,還有 Toast 之類的,也可以這樣實(shí)現(xiàn)。

3. react-transition-group
我們?cè)趯?shí)現(xiàn)動(dòng)效的思路上借鑒了 react-transition-group 中的CSSTransition組件。CSSTransition已經(jīng)幫我封裝好了動(dòng)效展開和關(guān)閉的過程,我們?cè)趯?shí)現(xiàn)彈窗時(shí),可以直接使用該組件。
這里有一個(gè)重要的屬性:unmountOnExit,表示在動(dòng)效結(jié)束后,卸載該組件。
const Modal = ({ open, onClose }) => {
// http://reactcommunity.org/react-transition-group/css-transition/
// in屬性為true/false,true為展開動(dòng)效,false為關(guān)閉動(dòng)效
return createPortal(
<csstransition in="{open}" timeout="{200}" unmountonexit="">
<div classname="modal">
<div classname="modal-content">{children}</div>
<div classname="modal-close-btn" onclick="{onClose}">
x
</div>
</div>
</csstransition>,
document.body,
);
};
在使用 CSSTransition 組件后,Modal 的動(dòng)效就方便多了。

4. 總結(jié)
至此已把待動(dòng)效的 React Modal 組件實(shí)現(xiàn)出來(lái)了。雖然 React 中沒有類似 Vue 官方定義的<transition>標(biāo)簽,不過我們可以自己或者借助第三方組件來(lái)實(shí)現(xiàn)。
以上就是React實(shí)現(xiàn)動(dòng)效彈窗組件的詳細(xì)內(nèi)容,更多關(guān)于React彈窗組件的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
webpack4 + react 搭建多頁(yè)面應(yīng)用示例
這篇文章主要介紹了webpack4 + react 搭建多頁(yè)面應(yīng)用示例,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來(lái)看看吧2018-08-08
webpack構(gòu)建react多頁(yè)面應(yīng)用詳解
這篇文章主要介紹了webpack構(gòu)建react多頁(yè)面應(yīng)用詳解,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來(lái)看看吧2017-09-09
圖文示例講解useState與useReducer性能區(qū)別
這篇文章主要為大家介紹了useState與useReducer性能區(qū)別圖文示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-05-05
react中實(shí)現(xiàn)搜索結(jié)果中關(guān)鍵詞高亮顯示
這篇文章主要介紹了react中實(shí)現(xiàn)搜索結(jié)果中關(guān)鍵詞高亮顯示,使用react實(shí)現(xiàn)要比js簡(jiǎn)單很多,方法都是大同小異,具體實(shí)現(xiàn)代碼大家跟隨腳本之家小編一起看看吧2018-07-07
VSCode配置react開發(fā)環(huán)境的步驟
本篇文章主要介紹了VSCode配置react開發(fā)環(huán)境的步驟,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來(lái)看看吧2017-12-12
基于React.js實(shí)現(xiàn)原生js拖拽效果引發(fā)的思考
這篇文章主要為大家詳細(xì)介紹了基于React.js實(shí)現(xiàn)原生js拖拽效果,繼而引發(fā)的一系列思考,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-03-03

