38950-vm/assets/js/main.js
Flatlogic Bot 9b97fdeae8 alpha_base
2026-03-03 17:10:40 +00:00

660 lines
25 KiB
JavaScript

/**
* INTERACTIVE TIMER APP
* Core Logic - Handles:
* - Landing Clock
* - Timer Modes (Time-watch, Countdown, Stopwatch, Lap, Relay, Custom)
* - Display Formats
* - Lap, Relay Split & Custom Activity Logic
* - Sound Alerts & Session Management
*/
const app = {
// STATE
currentView: 'landing',
currentMode: null,
isRunning: false,
isPaused: false,
isPreStarting: false,
startTime: 0,
elapsedTime: 0,
pausedTime: 0,
animationId: null,
history: [], // Universal storage for laps, splits, activities
countdownStartValue: 0,
// Relay State
currentParticipant: 1,
// Custom State
customActivities: [
{ name: 'Activity 1', duration: 60000 },
{ name: 'Activity 2', duration: 120000 }
],
currentActivityIndex: 0,
// Audio Helpers
audioCtx: null,
lastPlayedSecond: -1,
// CONFIG
modes: {
'time-watch': { title: 'Time-watch', allowPause: true, hasLaps: false, canToggleCountdown: true },
'countdown': { title: 'Countdown Timer', allowPause: true, hasLaps: false, forceCountdown: true },
'stopwatch': { title: 'Stopwatch', allowPause: false, hasLaps: false, canToggleCountdown: false },
'lap': { title: 'Lap Timer', allowPause: false, hasLaps: true, canToggleCountdown: false },
'relay': { title: 'Relay Timer', allowPause: true, hasRelay: true, canToggleCountdown: false },
'custom': { title: 'Custom Timer', allowPause: true, isCustom: true, canToggleCountdown: true }
},
// ELEMENTS
el: {
landingClock: document.getElementById('landing-clock'),
mainTimer: document.getElementById('main-timer'),
subTimer: document.getElementById('sub-timer'),
viewLanding: document.getElementById('view-landing'),
viewTimer: document.getElementById('view-timer'),
timerTitle: document.getElementById('timer-title'),
activeActivityName: document.getElementById('active-activity-name'),
sessionTitle: document.getElementById('session-title'),
btnStart: document.getElementById('btn-start'),
btnPause: document.getElementById('btn-pause'),
btnStop: document.getElementById('btn-stop'),
btnReset: document.getElementById('btn-reset'),
btnLap: document.getElementById('btn-lap'),
btnNext: document.getElementById('btn-next'),
formatSelect: document.getElementById('format-select'),
listContainer: document.getElementById('list-container'),
listTitle: document.getElementById('list-title'),
listHead: document.getElementById('list-head'),
listBody: document.getElementById('list-body'),
countdownInputs: document.getElementById('countdown-inputs'),
inputH: document.getElementById('input-h'),
inputM: document.getElementById('input-m'),
inputS: document.getElementById('input-s'),
relayConfig: document.getElementById('relay-config'),
participantCount: document.getElementById('participant-count'),
customBuilder: document.getElementById('custom-builder'),
activityList: document.getElementById('activity-list'),
btnAddActivity: document.getElementById('btn-add-activity'),
// Settings / Options
optCountdownContainer: document.getElementById('opt-countdown-container'),
settingIsCountdown: document.getElementById('setting-is-countdown'),
alertPreStart: document.getElementById('alert-pre-start'),
alertPreStartSec: document.getElementById('alert-pre-start-seconds'),
alertPreEnd: document.getElementById('alert-pre-end'),
alertPreEndSec: document.getElementById('alert-pre-end-seconds'),
alertCompletion: document.getElementById('alert-completion'),
brandLink: document.getElementById('brand-link')
},
init() {
console.log('Timer App Initializing...');
this.updateLandingClock();
this.bindEvents();
this.renderCustomBuilder();
this.resetTimer();
},
bindEvents() {
this.el.btnStart.addEventListener('click', () => this.handleStartClick());
this.el.btnPause.addEventListener('click', () => this.pauseTimer());
this.el.btnStop.addEventListener('click', () => this.stopTimer());
this.el.btnReset.addEventListener('click', () => this.resetTimer());
this.el.btnLap.addEventListener('click', () => this.recordLap());
this.el.btnNext.addEventListener('click', () => this.recordRelaySplit());
this.el.btnAddActivity.addEventListener('click', () => this.addActivity());
this.el.brandLink.addEventListener('click', (e) => {
e.preventDefault();
this.switchView('landing');
});
this.el.formatSelect.addEventListener('change', () => this.updateDisplay());
this.el.settingIsCountdown.addEventListener('change', () => this.resetTimer());
},
updateLandingClock() {
const now = new Date();
const h = String(now.getHours()).padStart(2, '0');
const m = String(now.getMinutes()).padStart(2, '0');
const s = String(now.getSeconds()).padStart(2, '0');
if (this.el.landingClock) {
this.el.landingClock.textContent = `${h}:${m}:${s}`;
}
setTimeout(() => this.updateLandingClock(), 1000);
},
switchView(viewName) {
this.currentView = viewName;
document.querySelectorAll('.view-section').forEach(s => s.classList.remove('active'));
if (viewName === 'landing') {
this.el.viewLanding.classList.add('active');
this.stopTimer();
} else {
this.el.viewTimer.classList.add('active');
}
},
setMode(modeKey) {
this.currentMode = modeKey;
const mode = this.modes[modeKey];
this.el.timerTitle.textContent = mode.title;
this.switchView('timer');
this.resetTimer();
// UI Adjustments
this.el.btnPause.classList.toggle('d-none', !mode.allowPause);
this.el.btnLap.classList.toggle('d-none', !mode.hasLaps);
this.el.btnNext.classList.toggle('d-none', !mode.hasRelay);
this.el.listContainer.classList.toggle('d-none', !mode.hasLaps && !mode.hasRelay && !mode.isCustom);
this.el.countdownInputs.classList.toggle('d-none', modeKey !== 'countdown');
this.el.relayConfig.classList.toggle('d-none', !mode.hasRelay);
this.el.customBuilder.classList.toggle('d-none', !mode.isCustom);
this.el.activeActivityName.classList.add('d-none');
// Options Box adjustments
this.el.optCountdownContainer.classList.toggle('d-none', !mode.canToggleCountdown && !mode.forceCountdown);
if (mode.forceCountdown) {
this.el.settingIsCountdown.checked = true;
this.el.settingIsCountdown.disabled = true;
} else if (modeKey === 'custom') {
this.el.settingIsCountdown.checked = false; // Default to count up as requested
this.el.settingIsCountdown.disabled = false;
} else {
this.el.settingIsCountdown.checked = false;
this.el.settingIsCountdown.disabled = false;
}
// Update List Header
if (mode.hasLaps) {
this.el.listTitle.textContent = 'Laps';
this.el.listHead.innerHTML = '<tr><th style="width: 80px;">Lap</th><th>Split Time</th><th>Delta</th><th class="text-end">Notes</th></tr>';
} else if (mode.hasRelay) {
this.el.listTitle.textContent = 'Relay Splits';
this.el.listHead.innerHTML = '<tr><th style="width: 120px;">Participant</th><th>Total Time</th><th>Split</th><th class="text-end">Status</th></tr>';
} else if (mode.isCustom) {
this.el.listTitle.textContent = 'Activities Completed';
this.el.listHead.innerHTML = '<tr><th>Activity</th><th>Duration</th><th>Total Elapsed</th><th class="text-end">Status</th></tr>';
}
this.updateDisplay();
},
handleStartClick() {
if (!this.el.sessionTitle.value.trim()) {
alert('Please enter a Session Name or Title before starting.');
this.el.sessionTitle.focus();
return;
}
if (this.isRunning && !this.isPaused) return;
// Sound Pre-start check
if (!this.isPaused && this.el.alertPreStart.checked) {
const preStartSec = parseInt(this.el.alertPreStartSec.value) || 0;
if (preStartSec > 0) {
this.runPreStartCountdown(preStartSec);
return;
}
}
this.startTimer();
},
runPreStartCountdown(seconds) {
this.isPreStarting = true;
this.el.btnStart.disabled = true;
this.el.btnReset.disabled = true;
this.el.sessionTitle.disabled = true;
let remaining = seconds;
this.el.mainTimer.textContent = `STARTING IN ${remaining}...`;
this.beep(440, 0.1);
const interval = setInterval(() => {
remaining--;
if (remaining > 0) {
this.el.mainTimer.textContent = `STARTING IN ${remaining}...`;
this.beep(440, 0.1);
} else {
clearInterval(interval);
this.isPreStarting = false;
this.beep(880, 0.3);
this.startTimer();
}
}, 1000);
},
startTimer() {
const mode = this.modes[this.currentMode];
const now = performance.now();
const isCountdownMode = this.el.settingIsCountdown.checked;
if (this.currentMode === 'countdown' && !this.isPaused) {
const h = parseInt(this.el.inputH.value) || 0;
const m = parseInt(this.el.inputM.value) || 0;
const s = parseInt(this.el.inputS.value) || 0;
const totalMs = ((h * 3600) + (m * 60) + s) * 1000;
if (totalMs <= 0) {
alert('Please enter a duration.');
return;
}
this.countdownStartValue = totalMs;
this.elapsedTime = totalMs;
}
if (mode.isCustom && !this.isPaused) {
this.currentActivityIndex = 0;
this.history = [];
this.renderHistory();
this.el.activeActivityName.classList.remove('d-none');
this.el.activeActivityName.textContent = this.customActivities[0].name;
this.countdownStartValue = this.customActivities[0].duration;
this.elapsedTime = isCountdownMode ? this.countdownStartValue : 0;
}
if (this.isPaused) {
const offset = (isCountdownMode) ? (this.countdownStartValue - this.elapsedTime) : this.elapsedTime;
this.startTime = now - offset;
} else {
this.startTime = now;
}
this.isRunning = true;
this.isPaused = false;
this.lastPlayedSecond = -1;
this.el.btnStart.disabled = true;
this.el.btnPause.disabled = false;
this.el.btnStop.disabled = false;
this.el.btnReset.disabled = true;
this.el.sessionTitle.disabled = true;
this.el.mainTimer.classList.remove('timer-finish');
// Lock inputs
if (this.el.inputH) this.el.inputH.disabled = true;
if (this.el.inputM) this.el.inputM.disabled = true;
if (this.el.inputS) this.el.inputS.disabled = true;
if (this.el.participantCount) this.el.participantCount.disabled = true;
this.tick();
},
pauseTimer() {
if (!this.isRunning || this.isPaused) return;
this.isPaused = true;
this.pausedTime = this.elapsedTime;
cancelAnimationFrame(this.animationId);
this.el.btnStart.disabled = false;
this.el.btnPause.disabled = true;
this.el.btnReset.disabled = false;
},
stopTimer() {
if (!this.isRunning && !this.isPaused) return;
this.isRunning = false;
this.isPaused = false;
cancelAnimationFrame(this.animationId);
this.el.btnStart.disabled = false;
this.el.btnPause.disabled = true;
this.el.btnStop.disabled = true;
this.el.btnReset.disabled = false;
if (this.currentMode === 'lap') {
this.recordLap();
} else if (this.currentMode === 'relay') {
this.recordRelaySplit(true);
}
},
resetTimer() {
this.stopTimer();
this.elapsedTime = 0;
this.pausedTime = 0;
this.countdownStartValue = 0;
this.history = [];
this.currentParticipant = 1;
this.currentActivityIndex = 0;
this.lastPlayedSecond = -1;
this.el.listBody.innerHTML = '';
this.el.mainTimer.classList.remove('timer-finish');
this.el.btnReset.disabled = true;
this.el.btnStart.disabled = false;
this.el.btnPause.disabled = true;
this.el.btnStop.disabled = true;
this.el.activeActivityName.classList.add('d-none');
this.el.sessionTitle.disabled = false;
if (this.el.inputH) this.el.inputH.disabled = false;
if (this.el.inputM) this.el.inputM.disabled = false;
if (this.el.inputS) this.el.inputS.disabled = false;
if (this.el.participantCount) this.el.participantCount.disabled = false;
this.updateDisplay();
},
tick() {
if (!this.isRunning || this.isPaused) return;
const mode = this.modes[this.currentMode];
const now = performance.now();
const isCountdownMode = this.el.settingIsCountdown.checked;
if (isCountdownMode) {
const delta = now - this.startTime;
this.elapsedTime = Math.max(0, this.countdownStartValue - delta);
this.checkAlerts(this.elapsedTime);
if (this.elapsedTime === 0) {
if (mode.isCustom) {
this.nextActivity();
} else {
this.updateDisplay();
this.stopTimer();
this.playCompletionAlert();
return;
}
}
} else if (mode.isCustom) {
// Count up custom mode
const delta = now - this.startTime;
this.elapsedTime = delta;
if (this.elapsedTime >= this.countdownStartValue) {
this.nextActivity();
}
} else {
// Standard count up
this.elapsedTime = now - this.startTime;
}
this.updateDisplay();
this.animationId = requestAnimationFrame(() => this.tick());
},
nextActivity() {
const isCountdownMode = this.el.settingIsCountdown.checked;
this.recordCustomActivity();
this.currentActivityIndex++;
if (this.currentActivityIndex < this.customActivities.length) {
const nextAct = this.customActivities[this.currentActivityIndex];
this.el.activeActivityName.textContent = nextAct.name;
this.countdownStartValue = nextAct.duration;
this.elapsedTime = isCountdownMode ? this.countdownStartValue : 0;
this.startTime = performance.now();
this.lastPlayedSecond = -1;
this.beep(660, 0.2);
} else {
this.updateDisplay();
this.stopTimer();
this.playCompletionAlert();
}
},
checkAlerts(ms) {
if (!this.el.alertPreEnd.checked) return;
const secondsLeft = Math.ceil(ms / 1000);
const warningSec = parseInt(this.el.alertPreEndSec.value) || 0;
if (secondsLeft <= warningSec && secondsLeft > 0 && secondsLeft !== this.lastPlayedSecond) {
this.lastPlayedSecond = secondsLeft;
this.beep(523, 0.1);
}
},
updateDisplay() {
const ms = this.elapsedTime;
const format = this.el.formatSelect.value;
this.el.mainTimer.textContent = this.formatTime(ms, format);
if (format !== 'hh:mm:ss.ms') {
this.el.subTimer.textContent = this.formatTime(ms, 'hh:mm:ss.ms');
this.el.subTimer.classList.remove('d-none');
} else {
this.el.subTimer.classList.add('d-none');
}
},
formatTime(ms, format) {
let absMs = Math.abs(ms);
let seconds = Math.floor(absMs / 1000);
let minutes = Math.floor(seconds / 60);
let hours = Math.floor(minutes / 60);
let milliseconds = Math.floor(absMs % 1000);
seconds %= 60;
minutes %= 60;
const pad = (n, l = 2) => String(n).padStart(l, '0');
switch (format) {
case 'hh:mm:ss.ms':
return `${pad(hours)}:${pad(minutes)}:${pad(seconds)}.${pad(milliseconds, 3)}`;
case 'hh:mm:ss':
return `${pad(hours)}:${pad(minutes)}:${pad(seconds)}`;
case 'hours':
return (ms / 3600000).toFixed(4) + ' h';
case 'minutes':
return (ms / 60000).toFixed(2) + ' m';
case 'seconds':
return (ms / 1000).toFixed(0) + ' s';
case 'seconds.ms':
return (ms / 1000).toFixed(3) + ' s';
default:
return '00:00:00.000';
}
},
recordLap() {
const now = this.elapsedTime;
const lastTime = this.history.length > 0 ? this.history[this.history.length - 1].time : 0;
const delta = now - lastTime;
this.history.push({
id: this.history.length + 1,
time: now,
delta: delta
});
this.renderHistory();
},
recordRelaySplit(isFinal = false) {
const now = this.elapsedTime;
const lastTime = this.history.length > 0 ? this.history[this.history.length - 1].time : 0;
const delta = now - lastTime;
const maxParticipants = parseInt(this.el.participantCount.value) || 1;
if (this.currentParticipant > maxParticipants && !isFinal) return;
this.history.push({
id: `P${this.currentParticipant}`,
time: now,
delta: delta,
status: isFinal ? 'DONE' : 'SPLIT'
});
if (!isFinal) {
this.currentParticipant++;
if (this.currentParticipant > maxParticipants) {
this.stopTimer();
}
}
this.renderHistory();
},
recordCustomActivity() {
const act = this.customActivities[this.currentActivityIndex];
const totalElapsedSoFar = this.history.reduce((acc, h) => acc + h.delta, 0) + act.duration;
this.history.push({
id: act.name,
delta: act.duration,
time: totalElapsedSoFar,
status: 'COMPLETED'
});
this.renderHistory();
},
renderHistory() {
if (this.currentMode === 'lap') {
let minDelta = Infinity;
let bestId = -1;
this.history.forEach(h => {
if (h.delta < minDelta) {
minDelta = h.delta;
bestId = h.id;
}
});
this.el.listBody.innerHTML = this.history.slice().reverse().map(h => `
<tr class="${h.id === bestId ? 'table-success' : ''}">
<td class="font-tabular fw-bold">#${h.id}</td>
<td class="font-tabular">${this.formatTime(h.time, 'hh:mm:ss.ms')}</td>
<td class="font-tabular text-muted">+${this.formatTime(h.delta, 'seconds.ms')}</td>
<td class="text-end">
${h.id === bestId ? '<span class="badge-best">BEST</span>' : ''}
</td>
</tr>
`).join('');
} else if (this.currentMode === 'relay') {
this.el.listBody.innerHTML = this.history.slice().reverse().map(h => `
<tr>
<td class="fw-bold">${h.id}</td>
<td class="font-tabular">${this.formatTime(h.time, 'hh:mm:ss.ms')}</td>
<td class="font-tabular text-muted">${this.formatTime(h.delta, 'seconds.ms')}</td>
<td class="text-end">
<span class="badge ${h.status === 'DONE' ? 'bg-danger' : 'bg-primary'} font-tabular" style="font-size: 0.6rem;">${h.status}</span>
</td>
</tr>
`).join('');
} else if (this.currentMode === 'custom') {
this.el.listBody.innerHTML = this.history.slice().reverse().map(h => `
<tr>
<td class="fw-bold">${h.id}</td>
<td class="font-tabular">${this.formatTime(h.delta, 'hh:mm:ss')}</td>
<td class="font-tabular text-muted">${this.formatTime(h.time, 'hh:mm:ss.ms')}</td>
<td class="text-end">
<span class="badge bg-success font-tabular" style="font-size: 0.6rem;">${h.status}</span>
</td>
</tr>
`).join('');
}
},
// Custom Activity Builder
renderCustomBuilder() {
this.el.activityList.innerHTML = this.customActivities.map((act, i) => `
<div class="activity-row p-3 border rounded bg-white">
<div class="row g-2 align-items-center">
<div class="col-md-5">
<label class="fs-tiny text-uppercase text-muted d-block mb-1">Activity Name</label>
<input type="text" class="form-control form-control-precise" value="${act.name}" onchange="app.updateActName(${i}, this.value)">
</div>
<div class="col-md-7">
<div class="d-flex justify-content-between align-items-end mb-1">
<label class="fs-tiny text-uppercase text-muted m-0">Duration: <span class="fw-bold text-primary font-tabular">${this.formatTime(act.duration, 'hh:mm:ss')}</span></label>
<button class="btn btn-link btn-sm p-0 text-danger text-decoration-none fs-tiny" onclick="app.removeActivity(${i})">REMOVE</button>
</div>
<div class="input-group input-group-sm">
<input type="number" id="act-input-${i}" class="form-control form-control-precise" placeholder="Enter number..." style="width: 80px;">
<button class="btn btn-outline-secondary" onclick="app.applyUnit(${i}, 'H')">H</button>
<button class="btn btn-outline-secondary" onclick="app.applyUnit(${i}, 'M')">M</button>
<button class="btn btn-outline-secondary" onclick="app.applyUnit(${i}, 'S')">S</button>
</div>
</div>
</div>
</div>
`).join('');
},
addActivity() {
this.customActivities.push({ name: `Activity ${this.customActivities.length + 1}`, duration: 60000 });
this.renderCustomBuilder();
},
removeActivity(index) {
if (this.customActivities.length <= 1) return;
this.customActivities.splice(index, 1);
this.renderCustomBuilder();
},
updateActName(index, val) {
this.customActivities[index].name = val;
},
applyUnit(index, unit) {
const input = document.getElementById(`act-input-${index}`);
const val = parseFloat(input.value) || 0;
let ms = 0;
switch(unit) {
case 'H': ms = val * 3600000; break;
case 'M': ms = val * 60000; break;
case 'S': ms = val * 1000; break;
}
this.customActivities[index].duration = ms;
this.renderCustomBuilder();
},
playCompletionAlert() {
this.el.mainTimer.classList.add('timer-finish');
if (this.el.alertCompletion.checked) {
this.beep(880, 0.5);
setTimeout(() => this.beep(1100, 0.5), 200);
}
},
beep(freq = 440, duration = 0.1) {
try {
if (!this.audioCtx) {
this.audioCtx = new (window.AudioContext || window.webkitAudioContext)();
}
const osc = this.audioCtx.createOscillator();
const gain = this.audioCtx.createGain();
osc.type = 'sine';
osc.frequency.setValueAtTime(freq, this.audioCtx.currentTime);
gain.gain.setValueAtTime(0.1, this.audioCtx.currentTime);
gain.gain.exponentialRampToValueAtTime(0.01, this.audioCtx.currentTime + duration);
osc.connect(gain);
gain.connect(this.audioCtx.destination);
osc.start();
osc.stop(this.audioCtx.currentTime + duration);
} catch (e) {
console.warn('Audio error:', e);
}
}
};
// Start the app
document.addEventListener('DOMContentLoaded', () => app.init());