<div id="root">
</div>
@import url('//fonts.googleapis.com/css?family=Roboto');
.mosaicEffect {
  position: absolute;
  top: 0;
  left: 0;
  right: 0;
  bottom: 0;
}

// animation
@keyframes fadeOut {
  0% {
    opacity: 1;
  }
  100% {
    opacity: 0;
  }
}

$wide_duration: 150;
$wide_delay_brightness: 75;
$wide_delay_alpha: 50;

[data-intersecting="true"] {
  .group {
    animation: fadeOut forwards;
    animation-duration: #{$wide_duration * 1.5}ms;
  }
  
  // group 1
  [id*="mask_alpha"] .group:nth-child(1) {
    animation-delay: #{$wide_duration - $wide_delay_alpha}ms;
  }
  // group 2
  [id*="mask_brightness"] .group:nth-child(2) {
    animation-delay: #{$wide_delay_brightness * 1}ms;
  }
  [id*="mask_alpha"] .group:nth-child(2) {
    animation-delay: #{($wide_delay_brightness * 1) + $wide_duration - $wide_delay_alpha}ms;
  }
  // group 3
  [id*="mask_brightness"] .group:nth-child(3) {
    animation-delay: #{$wide_delay_brightness * 2}ms;
  }
  [id*="mask_alpha"] .group:nth-child(3) {
    animation-delay: #{($wide_delay_brightness * 2) + $wide_duration - $wide_delay_alpha}ms;
  }
  // group 4
  [id*="mask_brightness"] .group:nth-child(2) {
    animation-delay: #{$wide_delay_brightness * 3}ms;
  }
  [id*="mask_alpha"] .group:nth-child(2) {
    animation-delay: #{($wide_delay_brightness * 3) + $wide_duration - $wide_delay_alpha}ms;
  }
}

// ----------
html, * {
  font-family: 'Roboto';
  box-sizing: border-box;
}

body {
  padding: 1.5em 0 4em;
}

.title {
  text-align: center;
  margin: 0 auto 1em;
  font-size: 1.5em;
  font-weight: 700;
}

.figure {
  diplay: block;
  position: relative;
  aspect-ratio: 2;
  margin: auto;
  width: min(80vw, 600px);
  background-color: whitesmoke;
  
  ~ .figure {
    margin-top: 4em;
  }
}

.caption {
  position: absolute;
  top: 100%;
  margin: 0.5em 0;
}

.button {
  position: absolute;
  top: 100%;
  right: 0;
  white-space: nowrap;
  margin: 0.5em 0;
  
  ~ .button {
    right: 8em;
  }
}

.image {
  width: 100%;
  height: 100%;
  object-fit: cover;
  vertical-align: top;
}
View Compiled
import { FC, useEffect, useState, useRef, useCallback } from "https://cdn.skypack.dev/react@17.0.1";
import ReactDOM from "https://cdn.skypack.dev/react-dom@17.0.1";

const cellParams = [
    1,
    0,
    1,
    2,
    2,
    2,
    3,
    2,
    1,
    1,
    3,
    1,
    0,
    1,
    3,
    1,
    2,
    0,
    2,
    1,
    2,
    1,
    0,
    2,
    0,
    3,
    1,
    1,
    1,
    3,
    2,
    1,
    1,
    0,
    3,
    0,
    1,
    3,
    3,
    0,
    3,
    0,
    0,
    2,
    3,
    1,
    3,
    3,
    1,
    1,
    1,
    3,
    3,
    2,
    0,
    2,
    0,
    0,
    1,
    0,
    0,
    3,
    0,
    1,
    1,
    2,
    3,
    1,
    3,
    0,
    0,
    3,
    3,
    3,
    3,
    3,
    0,
    2,
    2,
    2,
    3,
    3,
    2,
    3,
    1,
    3,
    3,
    2,
    0,
    0,
    1,
    1,
    2,
    2,
    3,
    2,
    1,
    0,
    0,
    3,
    0,
    0,
    3,
    0,
    2,
    3,
    0,
    3,
    1,
    1,
    0,
    1,
    2,
    0,
    0,
    2,
    3,
    2,
    2,
    2,
    3,
    2,
    0,
    1,
    1,
    1,
    2,
    3,
    1,
    3,
    0,
    3,
    3,
    2,
    2,
    3,
    2,
    1,
    1,
    2,
    1,
    2,
    3,
    3,
    1,
    0,
    2,
    1,
    0,
    1,
    3,
    0,
    0,
    2,
    1,
    2,
    1,
    3,
    3,
    0,
    1,
    0,
    0,
    2,
    2,
    3,
    3,
    2,
    1,
    0,
    2,
    2,
    1,
    0,
    0,
    0,
    1,
    3,
    0,
    2,
    0,
    1,
    3,
    0,
    0,
    3,
    2,
    1,
    2,
    2,
    2,
    2,
    2,
    0,
    1,
    2,
    1,
    0,
    3,
    0
];

const image_src = "https://images.unsplash.com/photo-1688550378756-866114814fee?crop=entropy&cs=srgb&fm=jpg&ixid=M3wzMjM4NDZ8MHwxfHJhbmRvbXx8fHx8fHx8fDE2ODg3MTc0OTN8&ixlib=rb-4.0.3&q=85";
const mosaic_image = "";


export const MosaicEffect: FC<PixelateEffectProps> = ({ addClass = [] }) => {
  
  const item = useRef<SVGSVGElement>(null);
  const [isIntersecting, setIsIntersecting] = useState<boolean>(false);
  
  const handleAdd = useCallback(()=> {
    setIsIntersecting(true);
  },[isIntersecting]);
  
  const handleRemove = useCallback(()=> {
    setIsIntersecting(false);
  },[isIntersecting]);
  
  return(
  <div
      className={["area", ...addClass].join(" ")}
    >
      <svg
        ref={item}
        className={"mosaicEffect"}
      data-intersecting={isIntersecting}
        xmlns="http://www.w3.org/2000/svg"
        viewBox="0 0 20 10"
        preserveAspectRatio="none"
      >
        <defs>
          <mask id="mask_alpha">
            {[0, 1, 2, 3].map(group => (
              <g key={group} className={"group"}>
                {cellParams.map((grid, index) => {
                  const x = index % 20;
                  const y = Math.floor(index / 20);
                  return grid === group ? (
                    <rect  className={"cell"} key={index} x={x} y={y} width="1" height="1" fill="white" stroke="white" strokeWidth="0.05" />
                  ) : null;
                })}
              </g>
            ))}
          </mask>
          <mask id="mask_brightness">
            {[0, 1, 2, 3].map(group => (
              <g key={group} className={"group"}>
                {cellParams.map((grid, index) => {
                  const x = index % 20;
                  const y = Math.floor(index / 20);
                  return grid === group ? (
                    <rect  className={"cell"} key={index} x={x} y={y} width="1" height="1" fill="white" stroke="white" strokeWidth="0.05" />
                  ) : null;
                })}
              </g>
            ))}
          </mask>
        </defs>
        <image href={mosaic_image} width="20" height="10" mask="url(#mask_alpha)" />
        <rect stroke="white" width="20" height="10" fill="white"
          mask="url(#mask_brightness)"
        />
      </svg>
      <button className={"button"} disabled={isIntersecting} onClick={handleAdd}>Mosaicize</button>
      <button className={"button"} disabled={!isIntersecting} onClick={handleRemove}>Reset</button>
    </div>
  );
};

ReactDOM.render(
  <React.StrictMode>
    <h1 className="title">Mosaic Effect</h1>
    <figure class="figure">
      <MosaicEffect />
      <img class="image" src={image_src} alt="" />
      <figcaption className="caption">Unprocessed image + Mosaic effect</figcaption>
    </figure>
    <figure class="figure">
      <img class="image" src={image_src} alt="" />
      <figcaption className="caption">Unprocessed image</figcaption>
    </figure>
    <figure class="figure">
      <img class="image" src={mosaic_image} alt="" />
      <figcaption className="caption">Mosaic image</figcaption>
    </figure>
  </React.StrictMode>,
  document.getElementById('root')
);
View Compiled

External CSS

This Pen doesn't use any external CSS resources.

External JavaScript

  1. https://cdnjs.cloudflare.com/ajax/libs/react/18.2.0/umd/react.production.min.js
  2. https://cdnjs.cloudflare.com/ajax/libs/react-dom/18.2.0/umd/react-dom.production.min.js