<div class="clock">
	<span class="clock__digit">0</span>
	<span class="clock__digit">0</span>
	<span class="clock__digit">:</span>
	<span class="clock__digit">0</span>
	<span class="clock__digit">0</span>
	<span class="clock__digit">:</span>
	<span class="clock__digit">0</span>
	<span class="clock__digit">0</span>
</div>
* {
	border: 0;
	box-sizing: border-box;
	margin: 0;
	padding: 0;
}
:root {
	--hue: 223;
	--bg: hsl(var(--hue),10%,90%);
	--fg: hsl(var(--hue),10%,10%);
	--primary: hsl(var(--hue),90%,55%);
	--shadow: hsl(var(--hue),90%,35%);
	font-size: calc(20px + (40 - 20) * (100vw - 320px) / (1280 - 320));
}
body {
	background: var(--bg);
	color: var(--fg);
	font: 1em/1.5 Spartan, sans-serif;
	height: 100vh;
	display: grid;
	place-items: center;
}
.clock {
	background-color: var(--primary);
	background-image: linear-gradient(-45deg,hsla(var(--hue),10%,10%,0),hsla(var(--hue),10%,10%,0.2));
	border-radius: 0.25rem;
	box-shadow: 0 0 0 0.1rem hsla(var(--hue),10%,10%,0.2) inset;
	color: hsl(var(--hue),10%,100%);
	display: flex;
	justify-content: center;
	font-size: 2em;
	line-height: 1;
	padding: 0.25rem 0.5rem;
}
.clock__digit {
	display: inline-block;
	font-weight: bold;
	text-align: center;
	text-shadow:
		0 0 0 var(--shadow),
		0 0 0 var(--shadow),
		1px 1px 0 var(--shadow),
		2px 2px 0 var(--shadow),
		3px 3px 0 var(--shadow),
		3px 3px 0 var(--shadow),
		4px 4px 0 var(--shadow);
	width: 1ch;
}
.clock__digit:not(:nth-child(3n)) {
	margin-top: 0.25rem;
}
.clock__digit--bounce {
	animation: bounce 0.5s ease-in;
}

/* Dark theme */
@media (prefers-color-scheme: dark) {
	:root {
		--bg: hsl(var(--hue),10%,10%);
		--fg: hsl(var(--hue),10%,90%);
	}
}

/* Animations */
@keyframes bounce {
	from, to {
		animation-timing-function: ease-in;
		text-shadow:
			0 0 0 var(--shadow),
			0 0 0 var(--shadow),
			1px 1px 0 var(--shadow),
			2px 2px 0 var(--shadow),
			3px 3px 0 var(--shadow),
			3px 3px 0 var(--shadow),
			4px 4px 0 var(--shadow);
		transform: translate(0,0);
	}
	33% {
		animation-timing-function: ease-out;
		text-shadow:
			0 0 0 var(--shadow),
			0 0 0 var(--shadow),
			0 0 0 var(--shadow),
			0 0 0 var(--shadow),
			0 0 0 var(--shadow),
			1px 1px 0 var(--shadow);
		transform: translate(4px,4px);
	}
	67% {
		animation-timing-function: ease-in;
		text-shadow:
			1px 1px 0 var(--shadow),
			2px 2px 0 var(--shadow),
			3px 3px 0 var(--shadow),
			4px 4px 0 var(--shadow),
			5px 5px 0 var(--shadow),
			6px 6px 0 var(--shadow);
		transform: translate(-2px,-2px);
	}
}
window.addEventListener("DOMContentLoaded",() => {
	const clock = new BouncyEmbossedClock(".clock");
});

class BouncyEmbossedClock {
	constructor(el) {
		this.el = document.querySelector(el);
		this.els = this.el ? this.el.querySelectorAll(".clock__digit") : [];
		this.digits = [];
		this.to = null;
		this.dto = [
			[null,null,null],
			[null,null,null],
			[null,null],
			[null,null,null],
			[null,null,null],
			[null,null],
			[null,null,null],
			[null,null,null],
		];
		this.staticUpdate();
		this.update();
	}
	getTime() {
		const time = new Date();
		const hms = [
			time.getHours(),
			time.getMinutes(),
			time.getSeconds()
		];

		return hms.map(u => u < 10 ? `0${u}` : `${u}`).join(":").split("");	
	}
	staticUpdate() {
		if (this.els) {
			this.digits = this.getTime();
			this.digits.forEach((d,i) => {
				this.els[i].textContent = d;
			});
		}
	}
	update() {
		if (this.els) {
			// get the time
			const display = this.getTime();
			const bounce = "clock__digit--bounce";
			const baseDelay = 350;
			const delayDec = 50;

			// display the digits
			display.forEach((d,i) => {
				if (+d > +this.digits[i] || +d === 0 && +this.digits[i] !== 0) {
					const colonElCL = display[i + 1] === ":" ? this.els[i + 1].classList : null;
					const el = this.els[i];
					const timeout = baseDelay - delayDec * i;

					this.dto[i].forEach(t => {
						clearTimeout(t);
					});

					// run the animation
					this.dto[i][0] = setTimeout(() => {
						el.classList.add(bounce);
					}, timeout);

					// show the next digit
					this.dto[i][1] = setTimeout(() => {
						el.textContent = d;
					}, timeout + 167);

					// kill the animation
					this.dto[i][2] = setTimeout(() => {
						el.classList.remove(bounce);
					}, timeout + 500);

					// colon animation (if applicable)
					if (colonElCL) {
						this.dto[i + 1].forEach(t => {
							clearTimeout(t);
						});

						this.dto[i + 1][0] = setTimeout(() => {
							 colonElCL.add(bounce);
						}, timeout - delayDec);

						this.dto[i + 1][1] = setTimeout(() => {
							 colonElCL.remove(bounce);
						}, (timeout - delayDec) + 500);
					}
				}

				this.digits[i] = d;
			});

			// loop
			clearTimeout(this.to);
			this.to = setTimeout(this.update.bind(this),1e3);
		}
	}
}

External CSS

  1. https://fonts.googleapis.com/css2?family=Spartan:wght@400;700&amp;display=swap

External JavaScript

This Pen doesn't use any external JavaScript resources.