DEV Community

sakethk
sakethk

Posted on

Advantages of useTimeout and useInterval hooks over setTimeout and setInterval ?

Hello ๐Ÿ‘‹ ,

Hope all are doing good in this pandemic time.

In this article i am going to explain what are advantages of useTimeout and useInterval hooks over setTimeout and setInterval also i will show how to create those hooks.

Why do we need useInterval and useTimeout hooks if we have setTimeout and setInterval.

In Javascript once we initialise setTimeout or setInterval we cannot modify either callback function or delay time. For better understanding i will explain with an example.

Consider we are building an Alert app which will show the given message to user after N seconds once we click on Done button. Here message and seconds are user inputs. You can refer the ๐Ÿ‘‡๐Ÿป image how the UI will look like

Screenshot 2021-07-26 at 11.00.31 PM.png

For this requirement we can blindly code like this

const Alert = ({}) => {
  const [message, setMessage] = useState("");
  const [delay, setDelay] = useState(0);

  const messageInputHandler = (e) => {
    setMessage(e.target.value);
  };

  const dalayInputHandler = (e) => {
    setDelay(e.target.value);
  };

  const showMessageAfterDelay = () => {
    setTimeout(() => {
      alert(message);
    }, delay * 1000);
  };

  return (
    <div>
      Show <input onChange={messageInputHandler} type="text" /> after
      <input onChange={dalayInputHandler} type="number" /> seconds
      <button onClick={showMessageAfterDelay}>Done</button>
    </div>
  );
};
Enter fullscreen mode Exit fullscreen mode

So the above code will work without any issue but if we think in a users prospective we cannot give a guarantee that user will not change the message and delay.

How to handle message and delay dynamically ๐Ÿค”

we are using setTimeout so we need to clear it and we need to call setTimeout again with an updated values. Luckily we have useEffect we can handle using it now we need to modify our code it will look like ๐Ÿ‘‡๐Ÿป

const Alert = ({}) => {
  const [message, setMessage] = useState("");
  const [delay, setDelay] = useState(0);

  const messageInputHandler = (e) => {
    setMessage(e.target.value);
  };

  const dalayInputHandler = (e) => {
    setDelay(e.target.value);
  };

  const showMessageAfterDelay = () => {
    setTimeout(() => {
      alert(message);
    }, delay * 1000);
  };

  useEffect(() => {
    const idx = setTimeout(() => alert(message), delay);

    return () => clearTimeout(idx);
  }, [message, delay]);
  return (
    <div>
      Show <input onChange={messageInputHandler} type="text" /> after
      <input onChange={dalayInputHandler} type="number" />
      seconds
      <button onClick={showMessageAfterDelay}>Done</button>
    </div>
  );
};
Enter fullscreen mode Exit fullscreen mode

Here useEffect will be called automatically if message or delay values are updated so do we need Done button ๐Ÿค” really. Initially we used it to trigger the setTimeout now here useEffect is taking care of that

After refactoring the code will be like this ๐Ÿ‘‡๐Ÿป

const Alert = ({}) => {
  const [message, setMessage] = useState("");
  const [delay, setDelay] = useState(0);

  const messageInputHandler = (e) => {
    setMessage(e.target.value);
  };

  const dalayInputHandler = (e) => {
    setDelay(e.target.value);
  };

  useEffect(() => {
    const idx = setTimeout(() => alert(message), delay);

    return () => clearTimeout(idx);
  }, [message, delay]);
  return (
    <div>
      Show <input onChange={messageInputHandler} type="text" /> after
      <input onChange={dalayInputHandler} type="number" />
      seconds
    </div>
  );
};
Enter fullscreen mode Exit fullscreen mode

We are good with the above one. But after few days we got same scenario in other page but this time we need to change function dynamically instead of message. So how can we achieve this ๐Ÿค”.

Solution

We can build a hook called useTimeout and can pass callback function and delay as arguments once any argument got updated that hook itself should update callback function and delay.

Here is the code for useTimeout hook

const useTimeout = (fn, delay) => {
  const fnRef = useRef(null);

//When ever function got updated this๐Ÿ‘‡๐ŸปuseEffect will update fnRef
  useEffect(() => {
    fnRef.current = fn;
  }, [fn]);

//When ever delay got updated this๐Ÿ‘‡๐ŸปuseEffect will clear current one and create a new one with updated delay value
  useEffect(() => {
    const idx = setTimeout(fn, delay);

    return () => clearTimeout(idx);
  }, [delay]);

  return;
};
Enter fullscreen mode Exit fullscreen mode

So now the problem solved similarly we can do it for useInterval as well it will be like this ๐Ÿ‘‡๐Ÿป

const useInterval = (fn, delay) => {
  const fnRef = useRef(null);

//When ever function got updated this๐Ÿ‘‡๐ŸปuseEffect will update fnRef
  useEffect(() => {
    fnRef.current = fn;
  }, [fn]);

//When ever delay got updated this๐Ÿ‘‡๐ŸปuseEffect will clear current one and create a new one with updated delay value
  useEffect(() => {
    const idx = setInterval(fn, delay);

    return () => clearInterval(idx);
  }, [delay]);

  return;
};
Enter fullscreen mode Exit fullscreen mode

Hope you learned something. Please share and react something ๐Ÿคจ if you liked it

Thank you ๐Ÿ™๐Ÿป

Follow me on
Linkedin : https://www.linkedin.com/in/saketh-kowtha/
Twitter : https://twitter.com/sakethkowtha
Github : https://github.com/saketh-kowtha

Top comments (0)