我们在写一些 UI 组件时,若不考虑动效,就很容易实现,主要就是有无的切换(类似于 Vue 中的 v-if 属性)或者可见性的切换(类似于 Vue 中的 v-show 属性)。

成都创新互联公司主营崇州网站建设的网络公司,主营网站建设方案,app软件定制开发,崇州h5小程序开发搭建,崇州网站营销推广欢迎崇州等地区企业咨询
在 React 中,可以这样来实现:
- interface ModalProps {
 - open: boolean;
 - onClose?: () => void;
 - children?: any;
 - }
 - const Modal = ({open. onClose, children}: ModalProps) => {
 - if (!open) {
 - return null;
 - }
 - return createPortal(
 , document.body); {children} x- };
 
使用方式:
- const App = () => {
 - const [open, setOpen] = useState(false);
 - return (
 setOpen(false)}> - modal content
 - );
 - };
 
我们在这里就是使用 open 属性来控制展示还是不展示,但完全没有渐变的效果。
若我们想实现 fade, zoom 等动画效果,还需要对此进行改造。
很多同学在自己实现动效时,经常是展示的时候有动效,关闭的时候没有动效。都是动效的时机没有控制好。这里我们先自己来实现一下动效的流转。
刚开始我实现的时候,动效只有开始状态和结束状态,需要很多的变量和逻辑来控制这个动效。
后来我参考了 react-transition-group 组件的实现,他是将动效拆分成了几个部分,每个部分分别进行控制。
展开动效的顺序:enter -> enter-active -> enter-done;
关闭动效的顺序:exit -> exit-active -> exit-done;
动效过程在 enter-active 和 exit-active 的过程中。
我们再通过一个变量 active 来控制是关闭动效是否已执行关闭,参数 open 只控制是执行展开动效还是关闭动效。
当 open 和 active 都为 false 时,才销毁弹窗。
- const Modal = ({ open, children, onClose }) => {
 - const [active, setActive] = useState(false); // 弹窗的存在周期
 - if (!open && !active) {
 - return null;
 - }
 - return ReactDOM.createPortal(
  , {children}- x
 - document.body,
 - );
 - };
 
这里我们接着添加动效过程的变化:
- const [aniClassName, setAniClassName] = useState(''); // 动效的class
 - // transition执行完毕的监听函数
 - const onTransitionEnd = () => {
 - // 当open为rue时,则结束状态为'enter-done'
 - // 当open未false时,则结束状态为'exit-done'
 - setAniClassName(open ? 'enter-done' : 'exit-done');
 - // 若open为false,则动画结束时,弹窗的生命周期结束
 - if (!open) {
 - setActive(false);
 - }
 - };
 - useEffect(() => {
 - if (open) {
 - setActive(true);
 - setAniClassName('enter');
 - // setTimeout用来切换class,让transition动起来
 - 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(''); // 动效的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(
  , {children}- x
 - document.body,
 - );
 - };
 
动效的流转过程已经实现了,样式也要一起写上。比如我们要实现渐隐渐现的 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;
 - }
 
如果是要实现放大缩小的 zoom 效果,修改这几个 class 就行。
一个带有动效的弹窗就已经实现了。
使用方式:
- const App = () => {
 - const [open, setOpen] = useState(false);
 - return (
 setOpen(false)}> - modal content
 - );
 - };
 
类似地,还有 Toast 之类的,也可以这样实现。
我们在实现动效的思路上借鉴了 react-transition-group 中的 CSSTransition 组件。 CSSTransition 已经帮我封装好了动效展开和关闭的过程,我们在实现弹窗时,可以直接使用该组件。
这里有一个重要的属性: unmountOnExit ,表示在动效结束后,卸载该组件。
- const Modal = ({ open, onClose }) => {
 - // http://reactcommunity.org/react-transition-group/css-transition/
 - // in属性为true/false,true为展开动效,false为关闭动效
 - return createPortal(
  {children}- x
 - ,
 - document.body,
 - );
 - };
 
在使用 CSSTransition 组件后,Modal 的动效就方便多了。
至此已把待动效的 React Modal 组件实现出来了。虽然 React 中没有类似 Vue 官方定义的  标签,不过我们可以自己或者借助第三方组件来实现。
                文章名称:实现一个带有动效的React弹窗组件
                
                网站URL:http://www.csdahua.cn/qtweb/news29/43529.html
            
网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等
声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网