最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - How to handle ref.current.contains() on a children that use ReactDOM.createPortal()? - Stack Overflow

programmeradmin3浏览0评论

I create an app using React hooks. I have a helper function onClickOutsideHook(ref, callback) that trigger the callback when you click outside of the ponent that provide the ref using React.useRef:

export const onClickOutsideHook = (ref, callback) => {
  // Hook get from 

  React.useEffect(() => {
    const handleClickOutside = event => {
      if (ref?.current && !ref.current.contains(event.target)) {
        callback();
      }
    };
    // Bind the event listener
    document.addEventListener("mousedown", handleClickOutside);
    return () => {
      // Unbind the event listener on clean up
      document.removeEventListener("mousedown", handleClickOutside);
    };
  }, [callback, ref]);
};

I have a ponent Dropdown that use this helper so it close when you click outside of it. This ponent has a Modal ponent as children that use ReactDOM.createPortal. I use it to render the Modal in body so it can cover all the app screen. My Modal contain a button that alert a message when you clicked on it:

function Modal() {
  return ReactDOM.createPortal(
    <div
      style={{
        position: "absolute",
        top: 0,
        left: 0,
        height: "100%",
        width: "100%",
        background: "rgba(0,0,0,0.6)"
      }}
    >
      <button onClick={() => alert("Clicked on Modal")}>Click</button>
    </div>,
    document.body
  );
}

function Dropdown(props) {
  const [isModalOpen, setIsModalOpen] = React.useState(false);

  const dropdownRef = React.useRef(null);
  onClickOutsideHook(dropdownRef, props.onClose);

  return (
    <div ref={dropdownRef}>
      Click outside and I will close
      <button onClick={() => setIsModalOpen(true)}>open Modal</button>
      {isModalOpen ? <Modal /> : <></>}
    </div>
  );
}

The problem is when I click on the Modal button to trigger the alert, the Dropdown is closed before since I clicked outside of it (Modal is not rendered as a children of Dropdown but in body). So my alert is never triggered.

Is there a way to define Modal as a children of Dropdown using ref but still render it in body using ReactDOM.createPortal?

Just have a look to the CodeSandbox.

I create an app using React hooks. I have a helper function onClickOutsideHook(ref, callback) that trigger the callback when you click outside of the ponent that provide the ref using React.useRef:

export const onClickOutsideHook = (ref, callback) => {
  // Hook get from https://stackoverflow./a/42234988/8583669

  React.useEffect(() => {
    const handleClickOutside = event => {
      if (ref?.current && !ref.current.contains(event.target)) {
        callback();
      }
    };
    // Bind the event listener
    document.addEventListener("mousedown", handleClickOutside);
    return () => {
      // Unbind the event listener on clean up
      document.removeEventListener("mousedown", handleClickOutside);
    };
  }, [callback, ref]);
};

I have a ponent Dropdown that use this helper so it close when you click outside of it. This ponent has a Modal ponent as children that use ReactDOM.createPortal. I use it to render the Modal in body so it can cover all the app screen. My Modal contain a button that alert a message when you clicked on it:

function Modal() {
  return ReactDOM.createPortal(
    <div
      style={{
        position: "absolute",
        top: 0,
        left: 0,
        height: "100%",
        width: "100%",
        background: "rgba(0,0,0,0.6)"
      }}
    >
      <button onClick={() => alert("Clicked on Modal")}>Click</button>
    </div>,
    document.body
  );
}

function Dropdown(props) {
  const [isModalOpen, setIsModalOpen] = React.useState(false);

  const dropdownRef = React.useRef(null);
  onClickOutsideHook(dropdownRef, props.onClose);

  return (
    <div ref={dropdownRef}>
      Click outside and I will close
      <button onClick={() => setIsModalOpen(true)}>open Modal</button>
      {isModalOpen ? <Modal /> : <></>}
    </div>
  );
}

The problem is when I click on the Modal button to trigger the alert, the Dropdown is closed before since I clicked outside of it (Modal is not rendered as a children of Dropdown but in body). So my alert is never triggered.

Is there a way to define Modal as a children of Dropdown using ref but still render it in body using ReactDOM.createPortal?

Just have a look to the CodeSandbox.

Share Improve this question asked Jun 29, 2020 at 12:35 johannchopinjohannchopin 14.9k11 gold badges62 silver badges121 bronze badges
Add a ment  | 

2 Answers 2

Reset to default 4

As a workaround you can add an ID attribute to your modal and then check if the click was outside the modal

function Modal() {
  return ReactDOM.createPortal(
    <div id="modalId">
      <button onClick={() => alert("Clicked on Modal")}>Click</button>
    </div>,
    document.body
  );
}
...
React.useEffect(() => {
  const handleClickOutside = (event) => {
    if (
      ref?.current &&
      !ref.current.contains(event.target) &&
      document.getElementById("modalId") &&
      !document.getElementById("modalId").contains(event.target) // check if click was outside your modal
    ) {
      callback();
    }
  };
  // Bind the event listener
  document.addEventListener("mousedown", handleClickOutside);
  return () => {
    // Unbind the event listener on clean up
    document.removeEventListener("mousedown", handleClickOutside);
  };
}, [callback, ref]);
...

Like the Portals docs says:

Even though a portal can be anywhere in the DOM tree, it behaves like a normal React child in every other way ...

This includes event bubbling. An event fired from inside a portal will propagate to ancestors in the containing React tree, even if those elements are not ancestors in the DOM tree.

But here is not the case, the mousedown event listener was added on the document and not only the Dropdown ponent. Even so, the bubbling still happens.

This means that if you add a ref to the Modal, and then add an event listener on the mousedown event with the whole purpose to stop the propagation, the handleClickOutside function will never be called.

It may still seem like a workaround, I don't know if there is a proper way to check.

function Modal() {
  const modalRef = useRef();

  useEffect(() => {
    const stopPropagation = e => {
      e.stopPropagation();
    };

    const { current: modalDom } = modalRef;
    modalDom.addEventListener("mousedown", stopPropagation);

    return () => {
      modalDom.removeEventListener("mousedown", stopPropagation);
    };
  }, []);

  return ReactDOM.createPortal(
    <div
      ref={modalRef}
      style={{
        position: "absolute",
        top: 0,
        left: 0,
        height: "100%",
        width: "100%",
        background: "rgba(0,0,0,0.6)"
      }}
    >
      <button onClick={() => alert("Clicked on Modal")}>Click</button>
    </div>,
    document.body
  );
}

Watch the Modal ponent from the following CodeSandbox.

发布评论

评论列表(0)

  1. 暂无评论