diff --git a/clock.js b/clock.js
index efe7cae..1e1bec8 100644
--- a/clock.js
+++ b/clock.js
@@ -1,66 +1,15 @@
// 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 = 0;
- const COBIE_UNITS = {
- second: 1,
- xenocycle: 0x10,
- quantic: 0x100,
- chronon: 0x1000,
- eonstrip: 0x10000,
- megasequence: 0x100000,
- cosmocycle: 0x1000000,
- };
+ const {
+ COBIE_EPOCH,
+ COBIE_UNITS,
+ floorDiv,
+ getTAIOffsetAt,
+ toCobiets
+ } = window.Cobie;
- function floorDiv(a, b) {
- return Math.trunc(a / b);
- }
-
- 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 d = new Date(leapSeconds[i].date);
- if (date < d) 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;
- }
+ // CoBiE helpers pulled from cobie.js
function placeMarkers() {
const clock = document.getElementById('clock');
diff --git a/cobie.js b/cobie.js
index dc95493..37fceef 100644
--- a/cobie.js
+++ b/cobie.js
@@ -163,7 +163,7 @@ function formatCobieTimestamp(cobiets) {
return sign + rawDateHex + '.' + paddedTimeHex;
}
-module.exports = {
+const Cobie = {
COBIE_EPOCH,
COBIE_UNITS,
floorDiv,
@@ -171,5 +171,15 @@ module.exports = {
getTAIOffsetAt,
toCobiets,
fromCobiets,
- formatCobieTimestamp
+ 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/index.html b/index.html
index c02e6b9..28ea08a 100644
--- a/index.html
+++ b/index.html
@@ -149,6 +149,7 @@
+
diff --git a/script.js b/script.js
index ef10ef9..aaf47b4 100644
--- a/script.js
+++ b/script.js
@@ -1,26 +1,17 @@
// CoBiE Time System Implementation
// Using Unix TAI epoch (January 1, 1970, 00:00:00 TAI)
// Note: TAI differs from UTC by leap seconds (37 seconds as of 2025)
-const COBIE_EPOCH = 0; // Unix TAI epoch in seconds
-
-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
-};
+const {
+ COBIE_EPOCH,
+ COBIE_UNITS,
+ floorDiv,
+ parseCobiets,
+ getTAIOffsetAt,
+ toCobiets,
+ fromCobiets,
+ formatCobieTimestamp,
+ breakdownNonNeg
+} = window.Cobie;
const EONSTRIP_NAMES = [
'Solprime', 'Lunex', 'Terros', 'Aquarion',
@@ -70,77 +61,7 @@ function formatSafeDate(rawDate, cobSeconds, intlOptions) {
}
}
-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;
-
- // All date‐units in descending “size” order, exactly as in COBIE_UNITS
- const allDateKeys = [
- 'astralmillennia',
- 'infinitum',
- 'eternum',
- 'metaepoch',
- 'cosmic_aeon',
- 'epoch_of_cosmos',
- 'celestial_era',
- 'universal_eon',
- 'galactic_year',
- 'cosmocycle',
- 'megasequence',
- 'eonstrip'
- ];
-
- // Raw hex‐string on the left of “.”
- let rawDateHex = m[2];
- // Pad on the left with zeros if shorter than the number of allDateKeys
- if (rawDateHex.length < allDateKeys.length) {
- rawDateHex = rawDateHex.padStart(allDateKeys.length, '0');
- }
-
- // If user provided MORE hex‐digits than allDateKeys.length, we treat each extra
- // digit as an even higher‐order “super‐unit” whose weight is:
- // 16^( position_from_right_of_eonstrip + 1 ) * COBIE_UNITS['eonstrip']
- let dateKeys = [...allDateKeys];
- if (rawDateHex.length > allDateKeys.length) {
- const extraCount = rawDateHex.length - allDateKeys.length;
- for (let i = 0; i < extraCount; i++) {
- // Insert null placeholders at the front for “beyond astralmillennia”
- dateKeys.unshift(null);
- }
- }
-
- // Time portion (always exactly 4 hex digits → chronon, quantic, xenocycle, second)
- const timeHexRaw = m[3];
- const timeHex = timeHexRaw.padStart(4, '0');
- const timeKeys = ['chronon', 'quantic', 'xenocycle', 'second'];
-
- let total = 0;
- // 1) Process every date‐hex digit
- for (let i = 0; i < rawDateHex.length; i++) {
- const digit = parseInt(rawDateHex[i], 16);
- const key = dateKeys[i];
- if (key === null) {
- // “Super‐unit” weight = (16 ^ power) * (size of eonstrip),
- // where power = (#digits_left_of_eonstrip) − (index_of_this_digit) − 1
- const power = rawDateHex.length - 1 - i;
- total += digit * Math.pow(16, power) * COBIE_UNITS['eonstrip'];
- } else {
- total += digit * COBIE_UNITS[key];
- }
- }
-
- // 2) Add time‐portion
- timeHex.split('').forEach((h, i) => {
- total += parseInt(h, 16) * COBIE_UNITS[timeKeys[i]];
- });
-
- return sign * total;
-}
-
-function floorDiv(a, b) {
- return Math.trunc(a / b);
-}
+// parseCobiets, floorDiv and other CoBiE helpers are provided by cobie.js
function getCurrentTAIOffset() {
return getTAIOffsetAt(new Date());
@@ -212,135 +133,8 @@ function getHumanDiff(d1, d2) {
return { years, months, days, hours, minutes, seconds };
}
-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; // Most recent known offset
-}
-
-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);
-
- // All date‐units in descending “size” order
- const dateUnits = [
- 'astralmillennia',
- 'infinitum',
- 'eternum',
- 'metaepoch',
- 'cosmic_aeon',
- 'epoch_of_cosmos',
- 'celestial_era',
- 'universal_eon',
- 'galactic_year',
- 'cosmocycle',
- 'megasequence',
- 'eonstrip'
- ];
-
- // Build a raw hex‐string by concatenating each unit’s hex
- let rawDateHex = dateUnits
- .map(key => bd[key].toString(16))
- .join('');
-
- // Trim leading zeros but leave at least one “0” if everything is zero
- rawDateHex = rawDateHex.replace(/^0+/, '');
- if (rawDateHex === '') rawDateHex = '0';
-
- // Time portion: exactly 4 hex digits (chronon → second)
- 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;
-}
+// getTAIOffsetAt, toCobiets, fromCobiets, breakdownNonNeg and
+// formatCobieTimestamp come from cobie.js
function updateCurrentTime() {
let cobiets, baseDate;