91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

React中怎么實現一個動效彈窗組件

發布時間:2021-06-21 18:09:53 來源:億速云 閱讀:300 作者:Leah 欄目:開發技術

今天就跟大家聊聊有關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>
  );
};

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

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中怎么實現一個動效彈窗組件

看完上述內容,你們對React中怎么實現一個動效彈窗組件有進一步的了解嗎?如果還想了解更多知識或者相關內容,請關注億速云行業資訊頻道,感謝大家的支持。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

饶阳县| 金乡县| 汉寿县| 大埔区| 宽甸| 潍坊市| 淮安市| 南投市| 广水市| 唐山市| 平原县| 册亨县| 井陉县| 桦甸市| 大悟县| 武冈市| 武隆县| 小金县| 两当县| 建平县| 林口县| 静安区| 涡阳县| 当涂县| 大石桥市| 姜堰市| 容城县| 民乐县| 宜昌市| 云阳县| 大连市| 铜梁县| 锡林浩特市| 郴州市| 甘泉县| 衢州市| 尖扎县| 河间市| 宁陵县| 凤庆县| 革吉县|