Creating a chameleon text effect
I recently created a new page on this website. From an early time in my front-end journey, I have been amazed by the websites that are featured on awwwards and I wanted to create something fancy like them.
Here's how I did it.
What we want to build
Let's take a look again at what we're trying to build here:
Hello there!Changing colors
We can animate the color changing of the text by using transition
:
transition-property: color;transition-duration: 5s;transition-timing-function: linear;/* or in one line */transition: color 5s linear;
And since I'm using styled-components
, we can put this in a re-usable CSS string:
import { css } from "styled-components";const WaveMixin = css` transition-property: color; transition-duration: 5s; transition-timing-function: linear;`;
But this doesn't do anything at the moment — to actually change the color, we will need to update the color of the <span>
that we are targeting after it has been rendered to the screen.
Let's start writing our little wrapper component for this:
// usage<ChameleonHighlight>Hello there!</ChameleonHighlight>;const ChameleonHighlight = ({ children }) => <Wave>{children}</Wave>;// import styled, { css } from "styled-components";const WaveMixin = css` transition-property: color; transition-duration: 5s; transition-timing-function: linear;`;const Wave = styled.span` ${WaveMixin}`;
And now we change the color once the component has mounted:
<ChameleonHighlight>Hello there!</ChameleonHighlight>;let root: HTMLElement;const getNewColor = () => { const h = random(1, 360); const s = random(80, 90); const l = random(50, 60); return `hsl(${h}, ${s}%, ${l}%)`;};const ChameleonHighlight = ({ children }) => { const changeColor = () => { const newColor = getNewColor(); if (root === undefined) root = document?.documentElement; root.style.setProperty("--color-chameleon", newColor); }; useEffect(() => { changeColor(); }, []); return <Wave>{children}</Wave>;};const WaveMixin = css` color: var(--color-chameleon); transition-property: color; transition-duration: 5s; transition-timing-function: linear;`;const Wave = styled.span` ${WaveMixin}`;
Huh, that didn't seem to work out 🤔 but that's because the transition already ran since the above live example was already "mounted" by the time you scrolled down to it.
A wild custom hook appeared!
To keep changing the color after a pre-defined interval, we'll be defining our very own useInterval
custom hook:
const useInterval = (callback, delay) => { const savedCallback = useRef(); useEffect(() => { savedCallback.current = callback; }, [callback]); useEffect(() => { const handler = (...args) => savedCallback.current(...args); if (delay !== null) { const intervalId = setInterval(handler, delay); return () => clearInterval(intervalId); } }, [delay]);};
Let's now use this hook to change the color of our text just as it completes its transition.
import { useInterval } from "@/utils/hooks";const ChameleonHighlight = ({ children }) => { const changeColor = () => { const newColor = getNewColor(); if (root === undefined) root = document?.documentElement; root.style.setProperty("--color-chameleon", newColor); }; useEffect(() => { changeColor(); }, []); useInterval(() => { changeColor(); }, 5000); return <Wave>{children}</Wave>;};
Voila! ✨
There we go. What's great is that you can also use this to highlight link, like so:
This is a link.Wrapping up
I'm really glad with how the effect turned out, and it was also easier to accomplish because I was using CSS-in-JS. It's great to see how powerful tools like styled-components can be! You can check the effect out in action and see it in all its glory 😄
Have a good one 👋
Getting page views