From a1f987ff597610d3025a7d69e4420d6d7d12a360 Mon Sep 17 00:00:00 2001 From: Kiyomichi Kosaka Date: Sun, 15 Jun 2025 14:56:16 +0200 Subject: [PATCH] Add macOS dashboard widget --- README.md | 11 + macos-widget/Info.plist | 18 ++ macos-widget/clock.js | 204 +++++++++++++++ macos-widget/cobie.js | 185 +++++++++++++ macos-widget/index.html | 23 ++ macos-widget/logo.svg | 22 ++ macos-widget/style.css | 561 ++++++++++++++++++++++++++++++++++++++++ 7 files changed, 1024 insertions(+) create mode 100644 macos-widget/Info.plist create mode 100644 macos-widget/clock.js create mode 100644 macos-widget/cobie.js create mode 100644 macos-widget/index.html create mode 100644 macos-widget/logo.svg create mode 100644 macos-widget/style.css diff --git a/README.md b/README.md index fbc9546..73cf8fe 100644 --- a/README.md +++ b/README.md @@ -63,6 +63,17 @@ An interactive web app that visualizes the **CosmoChron Binary Epoch (CoBiE)** t └── assets/ # (Optional) images or external CSS/JS ``` +## macOS Dashboard Widget + +The repository includes a minimal Dashboard widget under +`macos-widget/`. To install it on macOS: + +1. Compress the `macos-widget` folder into an archive. +2. Rename the archive to `CoBiEClock.wdgt` and double-click it. + +The widget runs offline and shows the analog CoBiE clock on your +desktop. + ## Contributing 1. Fork the repository. diff --git a/macos-widget/Info.plist b/macos-widget/Info.plist new file mode 100644 index 0000000..3d3359f --- /dev/null +++ b/macos-widget/Info.plist @@ -0,0 +1,18 @@ + + + + + CFBundleDisplayName + CoBiE Clock + CFBundleIdentifier + com.example.cobieclock + CFBundleVersion + 1.0 + CFBundlePackageType + BNDL + CFBundleSignature + ???? + MainHTML + index.html + + diff --git a/macos-widget/clock.js b/macos-widget/clock.js new file mode 100644 index 0000000..0ad372e --- /dev/null +++ b/macos-widget/clock.js @@ -0,0 +1,204 @@ +// Minimal CoBiE analog clock logic wrapped in its own scope to +// avoid clashes with variables from other scripts on the page. +(function () { + const { + COBIE_EPOCH, + COBIE_UNITS, + floorDiv, + getTAIOffsetAt, + toCobiets + } = window.Cobie; + + function getMarkerOffset(width) { + const points = [ + { width: 1024, value: 2 }, + { width: 450, value: 1.3 }, + { width: 200, value: 0.8 } + ]; + + // Sort points by width descending for easier handling + points.sort((a, b) => b.width - a.width); + + for (let i = 0; i < points.length - 1; i++) { + const p1 = points[i]; + const p2 = points[i + 1]; + if (width <= p1.width && width >= p2.width) { + // Linear interpolation + const t = (width - p2.width) / (p1.width - p2.width); + return p2.value + t * (p1.value - p2.value); + } + } + + // Extrapolation for width > max known + if (width > points[0].width) { + const p1 = points[0]; + const p2 = points[1]; + const slope = (p1.value - p2.value) / (p1.width - p2.width); + return p1.value + slope * (width - p1.width); + } + + // Extrapolation for width < min known + const p1 = points[points.length - 2]; + const p2 = points[points.length - 1]; + const slope = (p2.value - p1.value) / (p2.width - p1.width); + return p2.value + slope * (width - p2.width); + } + + function placeMarkers() { + const clock = document.getElementById('clock'); + let markers = clock.querySelectorAll('.marker'); + let ticks = clock.querySelectorAll('.tick'); + + // Create markers if they don't exist yet + if (markers.length === 0) { + for (let i = 0; i < 16; i++) { + const m = document.createElement('div'); + m.className = 'marker'; + m.textContent = i.toString(16).toUpperCase(); + clock.appendChild(m); + } + markers = clock.querySelectorAll('.marker'); + } + + // Create tick marks once + if (ticks.length === 0) { + for (let i = 0; i < 256; i++) { + const t = document.createElement('div'); + t.classList.add('tick'); + if (i % 16 === 0) t.classList.add('big'); + else if (i % 8 === 0) t.classList.add('mid'); + // insert before markers so digits sit on top + clock.insertBefore(t, clock.firstChild); + } + ticks = clock.querySelectorAll('.tick'); + } + + // Unified radius based on the actual clock size + const baseRadius = clock.offsetWidth / 2; + + // Tick lengths relative to the clock radius + const lenBig = baseRadius * 0.12; + const lenMid = baseRadius * 0.08; + const lenSmall = baseRadius * 0.05; + + const outerR = baseRadius - 2; // just inside the border + + // Distance from center for the marker digits so they sit just inside big ticks + const markerSize = markers[0] ? markers[0].offsetWidth : 0; + const markerRadius = outerR - lenBig - markerSize * getMarkerOffset(clock.offsetWidth); + + markers.forEach((m, i) => { + const angle = (i / 16) * 2 * Math.PI; + m.style.left = '50%'; + m.style.top = '50%'; + m.style.transform = + `translate(-50%, -50%) rotate(${angle}rad) translate(0, -${markerRadius}px) rotate(${-angle}rad)`; + }); + + ticks.forEach((t, i) => { + let len = lenSmall; + if (t.classList.contains('big')) len = lenBig; + else if (t.classList.contains('mid')) len = lenMid; + const innerR = outerR - len; + const angle = ((i + 1) / 256) * 2 * Math.PI; + t.style.height = `${len}px`; + t.style.left = '50%'; + t.style.top = '50%'; + t.style.transform = `translate(-50%, 0) rotate(${angle}rad) translate(0, -${innerR}px)`; + if (clock.offsetWidth < 200 && !t.classList.contains('big') && !t.classList.contains('mid')) { + t.style.display = 'none'; + } else { + t.style.display = ''; + } + }); + } + + const lastAngles = { + handXeno: 0, + handQuantic: 0, + handChronon: 0, + handEonstrip: 0, + handMegasequence: 0 + }; + + function rotateHand(id, angle) { + const el = document.getElementById(id); + if (!el) return; + const prev = lastAngles[id]; + + if (angle < prev) { + // When wrapping around (e.g. 15 → 0), animate to one full turn + // and then snap back to the new angle to avoid a jump. + const target = angle + 360; + const handle = () => { + el.removeEventListener('transitionend', handle); + // Snap back without animation + el.style.transition = 'none'; + el.style.transform = `translateX(-50%) translateZ(0) rotate(${angle}deg)`; + void el.offsetWidth; + el.style.transition = ''; + }; + el.addEventListener('transitionend', handle, { once: true }); + el.style.transform = `translateX(-50%) translateZ(0) rotate(${target}deg)`; + } else { + el.style.transform = `translateX(-50%) translateZ(0) rotate(${angle}deg)`; + } + + lastAngles[id] = angle; + } + + function renderClock(cob) { + // Use fractional progress within each unit so angles stay small + const xf = (cob % COBIE_UNITS.quantic) / COBIE_UNITS.quantic; + const qf = (cob % COBIE_UNITS.chronon) / COBIE_UNITS.chronon; + const cf = (cob % COBIE_UNITS.eonstrip) / COBIE_UNITS.eonstrip; + const ef = (cob % COBIE_UNITS.megasequence) / COBIE_UNITS.megasequence; + const mf = (cob % COBIE_UNITS.cosmocycle) / COBIE_UNITS.cosmocycle; + rotateHand('handXeno', xf * 360); + rotateHand('handQuantic', qf * 360); + rotateHand('handChronon', cf * 360); + rotateHand('handEonstrip', ef * 360); + rotateHand('handMegasequence', mf * 360); + } + + function updateClock() { + renderClock(toCobiets(new Date())); + } + + let intervalId = null; + + function startClock() { + clearInterval(intervalId); + updateClock(); + intervalId = setInterval(updateClock, 1000); + } + + function showTime(cob) { + clearInterval(intervalId); + renderClock(cob); + } + + function initClock() { + placeMarkers(); + startClock(); + const clk = document.getElementById('clock'); + if (clk) { + clk.addEventListener('click', () => { + document.body.classList.toggle('fullscreen-clock'); + // Re-position markers after toggling fullscreen + requestAnimationFrame(placeMarkers); + }); + } + window.addEventListener('resize', placeMarkers); + window.CobieClock = { + start: startClock, + showTime + }; + } + + if (document.readyState === 'loading') { + document.addEventListener('DOMContentLoaded', initClock); + } else { + initClock(); + } +})(); diff --git a/macos-widget/cobie.js b/macos-widget/cobie.js new file mode 100644 index 0000000..37fceef --- /dev/null +++ b/macos-widget/cobie.js @@ -0,0 +1,185 @@ +const COBIE_EPOCH = 0; + +const COBIE_UNITS = { + second: 1, + xenocycle: 0x10, + quantic: 0x100, + chronon: 0x1000, + eonstrip: 0x10000, + megasequence: 0x100000, + cosmocycle: 0x1000000, + galactic_year: 0x10000000, + universal_eon: 0x100000000, + celestial_era: 0x1000000000, + epoch_of_cosmos: 0x10000000000, + cosmic_aeon: 0x100000000000, + metaepoch: 0x1000000000000, + eternum: 0x10000000000000, + infinitum: 0x100000000000000, + astralmillennia: 0x1000000000000000 +}; + +function floorDiv(a, b) { + return Math.trunc(a / b); +} + +function parseCobiets(str) { + const m = /^([+-]?)([0-9A-Fa-f]+)\.([0-9A-Fa-f]{1,})$/.exec(str.trim()); + if (!m) return null; + const sign = m[1] === '-' ? -1 : 1; + + const allDateKeys = [ + 'astralmillennia','infinitum','eternum','metaepoch','cosmic_aeon', + 'epoch_of_cosmos','celestial_era','universal_eon','galactic_year', + 'cosmocycle','megasequence','eonstrip' + ]; + + let rawDateHex = m[2]; + if (rawDateHex.length < allDateKeys.length) { + rawDateHex = rawDateHex.padStart(allDateKeys.length, '0'); + } + + let dateKeys = [...allDateKeys]; + if (rawDateHex.length > allDateKeys.length) { + const extraCount = rawDateHex.length - allDateKeys.length; + for (let i = 0; i < extraCount; i++) { + dateKeys.unshift(null); + } + } + + const timeHexRaw = m[3]; + const timeHex = timeHexRaw.padStart(4, '0'); + const timeKeys = ['chronon', 'quantic', 'xenocycle', 'second']; + + let total = 0; + for (let i = 0; i < rawDateHex.length; i++) { + const digit = parseInt(rawDateHex[i], 16); + const key = dateKeys[i]; + if (key === null) { + const power = rawDateHex.length - 1 - i; + total += digit * Math.pow(16, power) * COBIE_UNITS['eonstrip']; + } else { + total += digit * COBIE_UNITS[key]; + } + } + + timeHex.split('').forEach((h, i) => { + total += parseInt(h, 16) * COBIE_UNITS[timeKeys[i]]; + }); + + return sign * total; +} + +function getTAIOffsetAt(date) { + const taiEpoch = new Date('1958-01-01T00:00:00Z'); + if (date < taiEpoch) { return 0; } + const leapSeconds = [ + { date: '1972-01-01T00:00:00Z', offset: 10 }, + { date: '1972-07-01T00:00:00Z', offset: 11 }, + { date: '1973-01-01T00:00:00Z', offset: 12 }, + { date: '1974-01-01T00:00:00Z', offset: 13 }, + { date: '1975-01-01T00:00:00Z', offset: 14 }, + { date: '1976-01-01T00:00:00Z', offset: 15 }, + { date: '1977-01-01T00:00:00Z', offset: 16 }, + { date: '1978-01-01T00:00:00Z', offset: 17 }, + { date: '1979-01-01T00:00:00Z', offset: 18 }, + { date: '1980-01-01T00:00:00Z', offset: 19 }, + { date: '1981-07-01T00:00:00Z', offset: 20 }, + { date: '1982-07-01T00:00:00Z', offset: 21 }, + { date: '1983-07-01T00:00:00Z', offset: 22 }, + { date: '1985-07-01T00:00:00Z', offset: 23 }, + { date: '1988-01-01T00:00:00Z', offset: 24 }, + { date: '1990-01-01T00:00:00Z', offset: 25 }, + { date: '1991-01-01T00:00:00Z', offset: 26 }, + { date: '1992-07-01T00:00:00Z', offset: 27 }, + { date: '1993-07-01T00:00:00Z', offset: 28 }, + { date: '1994-07-01T00:00:00Z', offset: 29 }, + { date: '1996-01-01T00:00:00Z', offset: 30 }, + { date: '1997-07-01T00:00:00Z', offset: 31 }, + { date: '1999-01-01T00:00:00Z', offset: 32 }, + { date: '2006-01-01T00:00:00Z', offset: 33 }, + { date: '2009-01-01T00:00:00Z', offset: 34 }, + { date: '2012-07-01T00:00:00Z', offset: 35 }, + { date: '2015-07-01T00:00:00Z', offset: 36 }, + { date: '2017-01-01T00:00:00Z', offset: 37 } + ]; + + for (let i = 0; i < leapSeconds.length; i++) { + const leapDate = new Date(leapSeconds[i].date); + if (date < leapDate) { + return i === 0 ? 10 : leapSeconds[i - 1].offset; + } + } + return 37; +} + +function toCobiets(date) { + const utcSec = floorDiv(date.getTime(), 1000); + const taiSec = utcSec + getTAIOffsetAt(date); + return taiSec - COBIE_EPOCH; +} + +function fromCobiets(cobiets) { + const taiSeconds = cobiets + COBIE_EPOCH; + const taiMs = taiSeconds * 1000; + let utcMs = taiMs; + for (let i = 0; i < 3; i++) { + const off = getTAIOffsetAt(new Date(utcMs)); + utcMs = taiMs - off * 1000; + } + return new Date(utcMs); +} + +const UNIT_KEYS = [ + 'astralmillennia','infinitum','eternum','metaepoch','cosmic_aeon','epoch_of_cosmos','celestial_era','universal_eon','galactic_year','cosmocycle','megasequence','eonstrip','chronon','quantic','xenocycle','second' +]; + +function breakdownNonNeg(cob) { + let rem = cob, bd = {}; + for (let key of UNIT_KEYS) { + bd[key] = floorDiv(rem, COBIE_UNITS[key]); + rem %= COBIE_UNITS[key]; + } + return bd; +} + +function formatCobieTimestamp(cobiets) { + const sign = cobiets < 0 ? '-' : '+'; + const absCob = Math.abs(cobiets); + const bd = breakdownNonNeg(absCob); + + const dateUnits = [ + 'astralmillennia','infinitum','eternum','metaepoch','cosmic_aeon','epoch_of_cosmos','celestial_era','universal_eon','galactic_year','cosmocycle','megasequence','eonstrip' + ]; + + let rawDateHex = dateUnits.map(key => bd[key].toString(16)).join(''); + rawDateHex = rawDateHex.replace(/^0+/, ''); + if (rawDateHex === '') rawDateHex = '0'; + + const timeHex = [bd.chronon, bd.quantic, bd.xenocycle, bd.second] + .map(n => n.toString(16)).join(''); + + const paddedTimeHex = timeHex.padStart(4, '0'); + return sign + rawDateHex + '.' + paddedTimeHex; +} + +const Cobie = { + COBIE_EPOCH, + COBIE_UNITS, + floorDiv, + parseCobiets, + getTAIOffsetAt, + toCobiets, + fromCobiets, + formatCobieTimestamp, + breakdownNonNeg +}; + +if (typeof module !== 'undefined' && module.exports) { + module.exports = Cobie; +} + +// Expose globally when loaded in a browser environment +if (typeof window !== 'undefined') { + window.Cobie = Cobie; +} diff --git a/macos-widget/index.html b/macos-widget/index.html new file mode 100644 index 0000000..4f78146 --- /dev/null +++ b/macos-widget/index.html @@ -0,0 +1,23 @@ + + + + + CoBiE Clock + + + +
+
+
+
+
+
+
+
+
CoBiE Time
+
+
+ + + + diff --git a/macos-widget/logo.svg b/macos-widget/logo.svg new file mode 100644 index 0000000..ed5d652 --- /dev/null +++ b/macos-widget/logo.svg @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/macos-widget/style.css b/macos-widget/style.css new file mode 100644 index 0000000..33fd056 --- /dev/null +++ b/macos-widget/style.css @@ -0,0 +1,561 @@ + * { + margin: 0; + padding: 0; + box-sizing: border-box; + } + + body { + font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif; + background: linear-gradient(135deg, #0a0e27 0%, #1a1f3a 100%); + color: #e0e0e0; + min-height: 100vh; + overflow-x: hidden; + } + + .container { + max-width: 1200px; + margin: 0 auto; + padding: 20px; + } + + .header { + text-align: center; + margin-bottom: 30px; + padding: 20px; + background: rgba(255, 255, 255, 0.05); + border-radius: 15px; + backdrop-filter: blur(10px); + border: 1px solid rgba(255, 255, 255, 0.1); + } + + .tooltip { + position: absolute; + top: 100%; + left: 50%; + transform: translateX(-50%); + background: rgba(0, 0, 0, 0.8); + color: #fff; + padding: 8px 12px; + border-radius: 4px; + white-space: nowrap; + pointer-events: none; + opacity: 0; + transition: opacity 0.2s; + z-index: 9999; + } + + h1 { + font-size: 2.5em; + background: linear-gradient(45deg, #00ffff, #ff00ff); + background-clip: text; + -webkit-background-clip: text; + -webkit-text-fill-color: transparent; + margin-bottom: 10px; + animation: glow 3s ease-in-out infinite; + } + + @keyframes glow { + 0%, 100% { opacity: 0.8; } + 50% { opacity: 1; } + } + + .current-time { + background: rgba(0, 255, 255, 0.1); + border: 2px solid rgba(0, 255, 255, 0.3); + border-radius: 10px; + padding: 20px; + margin-bottom: 20px; + text-align: center; + position: relative; + overflow: hidden; + flex: 1; + display: flex; + flex-direction: column; + justify-content: center; + height: var(--clock-size); + } + + .current-time.manual::before { + animation-play-state: paused; + } + + .current-time::before { + content: ''; + position: absolute; + top: -50%; + left: -50%; + width: 200%; + height: 200%; + background: linear-gradient(45deg, transparent, rgba(0, 255, 255, 0.1), transparent); + transform: rotate(45deg); + animation: sweep 3s linear infinite; + } + + @keyframes sweep { + 0% { transform: translateX(-100%) translateY(-100%) rotate(45deg); } + 100% { transform: translateX(100%) translateY(100%) rotate(45deg); } + } + + .cobie-time { + font-size: 2.5em; + font-family: 'Courier New', monospace; + letter-spacing: 2px; + margin: 10px 0; + text-shadow: 0 0 20px rgba(0, 255, 255, 0.5); + } + + .regular-time { + font-size: 1.2em; + color: #aaa; + } + + .timezone-selector { + margin: 20px 0; + text-align: center; + } + + select { + padding: 10px 20px; + font-size: 16px; + background: rgba(255, 255, 255, 0.1); + border: 1px solid rgba(255, 255, 255, 0.3); + border-radius: 5px; + color: #fff; + cursor: pointer; + transition: all 0.3s ease; + } + + select:hover { + background: rgba(255, 255, 255, 0.2); + transform: translateY(-2px); + } + + .calendar-controls { + display: flex; + justify-content: center; + align-items: center; + gap: 20px; + margin: 20px 0; + } + + button { + padding: 10px 20px; + font-size: 16px; + background: linear-gradient(45deg, #00ffff, #0080ff); + border: none; + border-radius: 5px; + color: #fff; + cursor: pointer; + transition: all 0.3s ease; + position: relative; + overflow: hidden; + } + + button::before { + content: ''; + position: absolute; + top: 50%; + left: 50%; + width: 0; + height: 0; + background: rgba(255, 255, 255, 0.3); + border-radius: 50%; + transform: translate(-50%, -50%); + transition: width 0.6s, height 0.6s; + } + + button:hover::before { + width: 300px; + height: 300px; + } + + button:hover { + transform: translateY(-2px); + box-shadow: 0 5px 15px rgba(0, 255, 255, 0.3); + } + + .calendar-view { + background: rgba(255, 255, 255, 0.05); + border-radius: 15px; + padding: 20px; + backdrop-filter: blur(10px); + border: 1px solid rgba(255, 255, 255, 0.1); + overflow: hidden; + } + + .calendar-header { + text-align: center; + font-size: 1.5em; + margin-bottom: 20px; + color: #00ffff; + } + + .eonstrip-grid { + display: grid; + grid-template-columns: repeat(4, 1fr); + gap: 15px; + margin-bottom: 20px; + transform: translateX(0); + transition: transform 0.3s ease; + will-change: transform; + } + + .eonstrip-card { + background: rgba(255, 255, 255, 0.08); + border: 1px solid rgba(255, 255, 255, 0.2); + border-radius: 10px; + padding: 15px; + text-align: center; + transition: all 0.3s ease; + cursor: pointer; + position: relative; + overflow: visible; + white-space: normal; + } + + .eonstrip-card::before { + content: ''; + position: absolute; + top: -50%; + left: -50%; + width: 200%; + height: 200%; + background: radial-gradient(circle, rgba(255, 255, 255, 0.1) 0%, transparent 70%); + opacity: 0; + transition: opacity 0.3s ease; + } + + .eonstrip-card:hover::before { + opacity: 1; + } + + .eonstrip-card:hover { + transform: translateY(-5px); + box-shadow: 0 10px 20px rgba(0, 255, 255, 0.2); + border-color: rgba(0, 255, 255, 0.5); + z-index: 1000; + } + + .eonstrip-card:hover .tooltip { + opacity: 1; + } + + .eonstrip-card.current { + background: rgba(0, 255, 255, 0.2); + border-color: rgba(0, 255, 255, 0.5); + } + + .eonstrip-name { + font-size: 1em; + font-weight: bold; + color: #00ffff; + margin-bottom: 5px; + } + + .eonstrip-hex { + font-size: 0.85em; /* was default monospace size */ + font-family: 'Courier New', monospace; + color: #ffaaff; + margin-bottom: 10px; + } + + .eonstrip-dates { + font-size: 0.7em; + color: #aaa; + line-height: 1.4; + } + + .event-tag { + display: inline-block; + margin-top: 4px; + padding: 2px 6px; + font-size: 0.75em; + font-weight: 600; + color: #fff; + background: linear-gradient(135deg, rgba(0,255,255,0.25), rgba(255,0,255,0.25)); + border: 1px solid rgba(255,255,255,0.2); + border-radius: 4px; + text-shadow: 0 0 6px rgba(0,255,255,0.7); + } + + .time-details { + background: rgba(255, 255, 255, 0.05); + border-radius: 10px; + padding: 20px; + margin-top: 20px; + } + + .time-unit { + display: flex; + justify-content: space-between; + padding: 10px 0; + border-bottom: 1px solid rgba(255, 255, 255, 0.1); + } + + .time-unit:last-child { + border-bottom: none; + } + + .unit-name { + color: #00ffff; + font-weight: bold; + } + + .unit-value { + font-family: 'Courier New', monospace; + color: #ffaaff; + } + + .toggle-btn { + background: linear-gradient(45deg, #00ffff, #0080ff); + border: none; + border-radius: 8px; + color: #fff; + font-size: 0.9em; + padding: 8px 16px; + cursor: pointer; + position: relative; + overflow: hidden; + text-transform: uppercase; + letter-spacing: 1px; + box-shadow: 0 0 10px rgba(0,255,255,0.4); + transition: all 0.3s ease; + } + + .toggle-btn .arrow-icon { + display: inline-block; + margin-right: 6px; + transition: transform 0.3s ease; + } + + .toggle-btn:hover { + transform: translateY(-2px); + box-shadow: 0 0 20px rgba(0,255,255,0.6); + } + + .extended-section { + display: none; /* hidden by default */ + margin-top: 10px; + animation: fadeIn 0.4s ease; + } + + /* Simple fade‐in for when extended units show */ + @keyframes fadeIn { + from { opacity: 0; } + to { opacity: 1; } + } + + @media only screen + and (max-width: 812px) /* iPhone portrait widths go up to ~812px */ + and (orientation: portrait) { + + /* scale down your main text by 30% */ + html { + font-size: 70%; + } + + /* if you prefer targeting only the big “cobie-time” element: */ + .cobie-time { + font-size: 1.75em; /* was 2.5em, which is 70% of 2.5em */ + } + } + + +.eonstrip-grid { + grid-template-columns: repeat(auto-fit, minmax(120px, 1fr)); + transform: translateX(0); + transition: transform 0.3s ease; + will-change: transform; +} + +/* Layout combining current time and analog clock */ +.time-display { + --clock-size: 40vmin; + display: flex; + flex-wrap: wrap; + justify-content: flex-start; + align-items: stretch; + gap: 20px; +} + + +.analog-clock-container { + flex: 0 0 auto; + width: var(--clock-size); + margin-left: auto; + display: flex; + justify-content: center; + align-items: center; +} + +#clock { + position: relative; + width: var(--clock-size); + height: var(--clock-size); + border: 2px solid rgba(255, 255, 255, 0.2); + border-radius: 50%; + background: radial-gradient(circle at center, #0a0e27 0%, #1a1f3a 100%); + box-shadow: 0 0 25px rgba(0, 255, 255, 0.2), inset 0 0 40px rgba(255, 0, 255, 0.2); +} + +.clock-center { + position: absolute; + top: 50%; + left: 50%; + width: calc(var(--clock-size) * 0.13); + height: calc(var(--clock-size) * 0.13); + transform: translate(-50%, -50%) translateZ(0); + background: url('logo.svg') center/contain no-repeat; + background-color: transparent; + border-radius: 50%; + /* box-shadow: 0 0 8px rgba(0, 255, 255, 0.8); */ + z-index: 2; + pointer-events: none; +} + +@media screen and (min-width: 1200px) { + .clock-center { + width: calc(var(--clock-size) * 0.085); + height: calc(var(--clock-size) * 0.085); + } +} + +.clock-label { + position: absolute; + bottom: 30%; + left: 50%; + transform: translateX(-50%); + font-family: 'Great Vibes', cursive; + font-size: calc(var(--clock-size) * 0.06); + color: #ffaaff; + text-shadow: 0 0 6px rgba(255, 0, 255, 0.6); + pointer-events: none; + z-index: 0; +} + +body.fullscreen-clock .clock-label { + font-size: calc(var(--clock-size) * 0.08); +} + +.marker { + position: absolute; + width: 2em; + height: 2em; + text-align: center; + line-height: 2em; + /* Use a futuristic font for the clock markers */ + font-family: 'Orbitron', 'Trebuchet MS', 'Lucida Sans', Arial, sans-serif; + font-size: 1.2em; + font-weight: 600; + color: #00ffff; + background: none; + border: none; + border-radius: 0; + text-shadow: 0 0 6px rgba(0, 255, 255, 0.9), 0 0 12px rgba(0, 255, 255, 0.7); + box-shadow: none; + transform-origin: center; + font-variant-numeric: tabular-nums; + user-select: none; + pointer-events: none; + will-change: transform; + z-index: 1; +} + +.tick { + position: absolute; + width: 2px; + background: #00ffff; + transform-origin: center top; + z-index: 0; +} + +.tick.mid { + background: #66ffff; +} + +.tick.big { + background: #ffffff; +} + +.hand { + position: absolute; + bottom: 50%; + left: 50%; + transform-origin: bottom center; + transform: translateX(-50%); + transition: transform 0.5s ease-in-out; + border-radius: 2px; + z-index: 1; +} + + +.hand.xeno { + width: 2px; + height: 44%; + background: linear-gradient(to top, #66ccff, #0044ff); + box-shadow: 0 0 8px #66ccff; +} + +.hand.quantic { + width: 3px; + height: 40%; + background: linear-gradient(to top, #ff66ff, #9900ff); + box-shadow: 0 0 8px #ff66ff; +} + +.hand.chronon { + width: 4px; + height: 34%; + background: linear-gradient(to top, #ff4444, #880000); + box-shadow: 0 0 8px #ff4444; +} + +.hand.eonstrip { + width: 5px; + height: 30%; + background: linear-gradient(to top, #33ff99, #006633); + box-shadow: 0 0 8px #33ff99; +} + + +.hand.megasequence { + width: 6px; + height: 26%; + background: linear-gradient(to top, #ffbb33, #aa5500); + box-shadow: 0 0 8px #ffbb33; +} + +@media only screen and (max-height: 430px) and (orientation: landscape) { + .time-display { + --clock-size: 70vmin; + } +} + +body.fullscreen-clock .header, +body.fullscreen-clock .current-time, +body.fullscreen-clock .timezone-selector, +body.fullscreen-clock .calendar-controls, +body.fullscreen-clock .calendar-view, +body.fullscreen-clock .time-details, +body.fullscreen-clock .explanations { + display: none; +} + +body.fullscreen-clock .time-display { + position: fixed; + inset: 0; + justify-content: center; + align-items: center; + z-index: 1000; +} + +body.fullscreen-clock .analog-clock-container { + --clock-size: 80vmin; + width: var(--clock-size); + margin-left: 0; +} + +body.fullscreen-clock #clock { + width: var(--clock-size); + height: var(--clock-size); +}