欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  科技

React实现动效弹窗组件

程序员文章站 2022-04-25 08:05:47
我们在写一些 ui 组件时,若不考虑动效,就很容易实现,主要就是有无的切换(类似于 vue 中的 v-if 属性)或者可见性的切换(类似于 vue 中的 v-show 属性)。1. 没有动效的弹窗在...

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

React实现动效弹窗组件

1. 没有动效的弹窗

在 react 中,可以这样来实现:

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>
  );
};

我们在这里就是使用open属性来控制展示还是不展示,但完全没有渐变的效果。

若我们想实现 fade, zoom 等动画效果,还需要对此进行改造。

React实现动效弹窗组件

2. 自己动手实现有动效的弹窗

很多同学在自己实现动效时,经常是展示的时候有动效,关闭的时候没有动效。都是动效的时机没有控制好。这里我们先自己来实现一下动效的流转。

刚开始我实现的时候,动效只有开始状态和结束状态,需要很多的变量和逻辑来控制这个动效。

后来我参考了react-transition-group组件的实现,他是将动效拆分成了几个部分,每个部分分别进行控制。

  1. 展开动效的顺序:enter -> enter-active -> enter-done;
  2. 关闭动效的顺序:exit -> exit-active -> exit-done;

动效过程在enter-activeexit-active的过程中。

我们再通过一个变量 active 来控制是关闭动效是否已执行关闭,参数 open 只控制是执行展开动效还是关闭动效。

当 open 和 active 都为 false 时,才销毁弹窗。

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,
  );
};

这里我们接着添加动效过程的变化:

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(
    <div classname="{'modal" '="" +="" aniclassname}="" ontransitionend="{ontransitionend}">
      <div classname="modal-content">{children}</div>
      <div classname="modal-close-btn" onclick="{onclose}">
        x
      </div>
    </div>,
    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 (
    <div classname="app">
      <button onclick="{()" ==""> setopen(true)}>show modal</button>
      <modal open="{open}" onclose="{()" ==""> setopen(false)}>
        modal content
      </modal>
    </div>
  );
};

点击链接自己实现动效的 react 弹窗 demo查看效果。

类似地,还有 toast 之类的,也可以这样实现。

React实现动效弹窗组件

3. react-transition-group

我们在实现动效的思路上借鉴了 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(
    <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 的动效就方便多了。

React实现动效弹窗组件

4. 总结

至此已把待动效的 react modal 组件实现出来了。虽然 react 中没有类似 vue 官方定义的<transition>标签,不过我们可以自己或者借助第三方组件来实现。

以上就是react实现动效弹窗组件的详细内容,更多关于react弹窗组件的资料请关注其它相关文章!