Skip to content

Instantly share code, notes, and snippets.

@kyle1elyk
Last active February 27, 2024 07:08
Show Gist options
  • Save kyle1elyk/7cc9e052e196c8e37b12006a5a545d1c to your computer and use it in GitHub Desktop.
Save kyle1elyk/7cc9e052e196c8e37b12006a5a545d1c to your computer and use it in GitHub Desktop.
Simple UTC Clock with copy paste
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Timestamps in Different Timezones</title>
<style>
body {
background-color: black;
color: white;
font-family: 'Courier New', monospace;
font-size: 60px;
line-height: 1.5;
text-align: center;
margin: 50px;
padding-top: 50px;
}
table {
border-collapse: separate;
border-spacing: 0;
}
.timestamp {
color: #BBB;
width: 100%;
background: #000;
border-radius: 20px;
z-index: -1;
padding-right: 10px;
padding-left: 10px;
}
.timestamp:hover { background: #222; }
.timestamp:hover * { color: #FFF; }
.timestamp-td { padding-left: 20px; }
.dial { z-index: 100; display: inline-block; height: 1em; width: 1em; padding: 0; overflow: hidden; margin-right: 20px; border: 2px #000 solid; border-radius: 100px; vertical-align: text-top; }
.dial * { width: 100%; height: 100%; margin: 0px; display: inline-block; position: relative; top: 0; left: 0; background-size: contain; }
.dial .over { z-index: 10; background-image: url(''); }
.dial .under { transform: rotate(90deg); top: -152%; z-index: 1; background-image: url(''); }
.date, .time { font-weight: bold; font-size: 85px; }
.date { color: #FFF; }
.t { color: #444; }
.time { color: #FFF; }
.second { font-size: 60px; }
.tz { font-size: .83em; vertical-align: text-bottom; background: #222; border-radius: 40px; padding-left: 20px; padding-right: 20px; border: 3px solid #000; }
.tz-td { text-align: center; padding-right: 20px; }
.timestamp[data-timezone="Etc/UTC"] .tz { background: #555; }
.timestamp[data-timezone="Etc/UTC"] .second { border-color: #8d8d8d !important; border-bottom: 2px solid; border-top: 2px solid; border-image: linear-gradient(to left, #8d8d8d, transparent) 1; }
.timestamp[data-timezone="Etc/UTC"] .offset { border-color: #8d8d8d !important; border-bottom: 2px solid; border-top: 2px solid; border-right: 15px solid; padding-right: 15px; border-radius: 0 10px 10px 0; }
/*
.glow .tz {
animation: glowAnimation 0.2s ease-in-out;
box-shadow: 0 0 0px 0px #FFF;
z-index: 1;
}
.glow td {
animation: glowAnimationBackground 0.2s ease-in-out;
outline-bottom: 1px #000 solid;
}
@keyframes glowAnimation {
0% {
box-shadow: 0 0 0px 0px #FFF;
border-color: #000;
}
3% {
box-shadow: 0 0 10px 5px #FFF;
border-color: #FFF;
}
100% {
box-shadow: 0 0 0px 0px rgba(255,255,255, 0);
border-color: #000;
}
}
@keyframes glowAnimationBackground {
0% {
background: #222;;
}
5% {
background: #444;
}
100% {
background: #222;
}
}
*/
.glow .tz {
box-shadow: 0 0 10px 5px #FFF;
border-color: #FFF;
animation: glowAnimation 0.01s ease-in-out;
}
.glow td {
background: #444;
animation: glowAnimationBackground 0.01s ease-in-out;
}
.unglow .tz {
animation: glowAnimation 0.5s ease-in-out;
animation-direction: reverse;
}
.unglow td {
animation: glowAnimationBackground 0.5s ease-in-out;
animation-direction: reverse;
}
@keyframes glowAnimation {
0% { box-shadow: 0 0 0px 0px #FFF; border-color: #000; }
100% { box-shadow: 0 0 10px 5px #FFF; border-color: #FFF; }
}
@keyframes glowAnimationBackground {
0% { background: #222; }
100% { background: #444; }
}
::selection {
background: rgba(200, 200, 200, 0.6);
color: #333;
}
nav {
z-index: -1;
color: white;
padding-bottom: 5px;
padding-top: 0;
text-align: left;
position: absolute;
margin: 0;
top: 0;
font-size: 18px;
border-bottom: 1px #666 solid;
}
/* Optional: Add styles to make the text stand out more */
nav h1 {
margin: 0;
display: inline-block;
font-size: 36px;
}
/* Firefox (uncomment to work in Firefox, although other properties will not work!) */
/** {
scrollbar-width: thin;
scrollbar-color: #FFFFFF #000000;
}*/
/* Chrome, Edge and Safari */
*::-webkit-scrollbar {
height: 8px;
width: 8px;
}
*::-webkit-scrollbar-track {
border-radius: 0px;
background-color: #000000;
}
*::-webkit-scrollbar-track:hover {
background-color: #000000;
}
*::-webkit-scrollbar-track:active {
background-color: #000000;
}
*::-webkit-scrollbar-thumb {
border-radius: 5px;
background-color: #FFFFFF;
border: 1px solid #000000;
}
*::-webkit-scrollbar-thumb:hover {
background-color: #FFFFFF;
}
*::-webkit-scrollbar-thumb:active {
background-color: #FFFFFF;
}
</style>
</head>
<body>
<nav>
<h1>ISO 8601 Timestamps</h1> (click to copy)
</nav>
<center><table id="timestamps"></table></center>
<script>
function createInitial(timezones) {
let timestampsDiv = document.getElementById("timestamps");
timezones.forEach(timezone => timestampsDiv.innerHTML += `<tr class="timestamp" data-timezone="${timezone}">` +
`<td class="tz-td"><span class="tz"><\/span><\/td>` +
`<td><div class="dial"><div class="over"><\/div><div class="under"><\/div><\/div><\/td>` +
`<td class="timestamp-td"><span class="date"><\/span>` +
`<span class="t">T<\/span>` +
`<span class="time"><\/span><span class="second"><\/span>` +
`<span class="offset"><\/span><\/td>` +
`<\/tr>`);
document.querySelectorAll(".timestamp").forEach(tsRow => {
tsRow.addEventListener("mousedown", (e) => {
tsRow.classList.add('glow');
});
tsRow.addEventListener("mouseup", (e) => {
tsRow.classList.remove('glow');
tsRow.classList.add('unglow');
setTimeout(() => {
tsRow.classList.remove('unglow');
}, 500);
});
tsRow.addEventListener("click", (e) => {
const text = tsRow.querySelector(".timestamp-td").innerText;
navigator.clipboard.writeText(text);
});
});
}
const timeRegex = /^(?<month>\d{1,2})\/(?<day>\d{1,2})\/(?<year>\d{4}), (?<hour>\d{2}):(?<minute>\d{2}):(?<second>\d{2}).(?<secondPart>\d) GMT(?<offset>[+-]\d{2}:\d{2})?/;
const pad = n => n.toString().padStart(2, "0");
const updateText = (element, text) => { if (element.innerText !== text) element.innerText = text };
let title;
function updateTimestamps() {
const now = new Date();
document.querySelectorAll(".timestamp").forEach(timestamp_p => {
const timezone = timestamp_p.dataset.timezone;
const raw_timestamp = now.toLocaleString('en-US', {
timeZone: timezone,
timeZoneName: "longOffset",
hourCycle: 'h23',
year: "numeric",
month: "numeric",
day: "numeric",
hour: "numeric",
minute: "numeric",
second: "numeric",
fractionalSecondDigits: 1,
});
const {year, month, day, hour, minute, second, secondPart, offset} = timeRegex.exec(raw_timestamp).groups;
const mappedOffset = offset == null ? "Z" : offset;
const date = `${year}-${pad(month)}-${pad(day)}`;
const short_time = `${hour}:${minute}`;
const fullSecond = `${second}.${secondPart}`;
updateText(timestamp_p.querySelector(".tz"), timezone);
updateText(timestamp_p.querySelector(".date"), date);
updateText(timestamp_p.querySelector(".t"), "T");
updateText(timestamp_p.querySelector(".time"), short_time);
updateText(timestamp_p.querySelector(".second"), `:${fullSecond}`);
updateText(timestamp_p.querySelector(".offset"), mappedOffset);
timestamp_p.querySelector(".under").style.transform = `rotate(${360 - ((parseInt(hour) * 60 + parseInt(minute)) / 4) }deg)`;
if (timezone === "Etc/UTC" && document.title !== `${date} ${short_time}`) {
document.title = `${date} ${short_time}Z`;
}
});
}
createInitial(["US/Pacific", "America/Phoenix", "US/Central", "US/Eastern", "Etc/UTC", "Asia/Kuwait", "Asia/Kolkata", "Asia/Singapore", "Japan"]);
setInterval(updateTimestamps, 100);
updateTimestamps();
</script>
</body>
</html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment