4919 lines
193 KiB
JavaScript
4919 lines
193 KiB
JavaScript
// Game state management
|
|
class TaskChallengeGame {
|
|
constructor() {
|
|
// Initialize data management system first
|
|
this.dataManager = new DataManager();
|
|
|
|
// Initialize desktop features early
|
|
this.initDesktopFeatures();
|
|
|
|
this.gameState = {
|
|
isRunning: false,
|
|
isPaused: false,
|
|
currentTask: null,
|
|
isConsequenceTask: false,
|
|
startTime: null,
|
|
pausedTime: 0,
|
|
totalPausedTime: 0,
|
|
completedCount: 0,
|
|
skippedCount: 0,
|
|
consequenceCount: 0,
|
|
score: 0,
|
|
lastSkippedTask: null, // Track the last skipped task for mercy cost calculation
|
|
usedMainTasks: [],
|
|
usedConsequenceTasks: [],
|
|
usedTaskImages: [], // Track which task images have been shown
|
|
usedConsequenceImages: [], // Track which consequence images have been shown
|
|
gameMode: 'complete-all', // Game mode: 'complete-all', 'timed', 'score-target'
|
|
timeLimit: 300, // Time limit in seconds (5 minutes default)
|
|
scoreTarget: 1000, // Score target (default 1000 points)
|
|
currentStreak: 0, // Track consecutive completed regular tasks
|
|
totalStreakBonuses: 0, // Track total streak bonus points earned
|
|
lastStreakMilestone: 0 // Track the last streak milestone reached
|
|
};
|
|
|
|
this.timerInterval = null;
|
|
this.imageDiscoveryComplete = false;
|
|
this.imageManagementListenersAttached = false;
|
|
this.audioDiscoveryComplete = false;
|
|
this.audioManagementListenersAttached = false;
|
|
this.musicManager = new MusicManager(this.dataManager);
|
|
|
|
// Initialize Flash Message System
|
|
this.flashMessageManager = new FlashMessageManager(this.dataManager);
|
|
|
|
// Initialize Popup Image System (Punishment for skips)
|
|
this.popupImageManager = new PopupImageManager(this.dataManager);
|
|
|
|
this.initializeEventListeners();
|
|
this.setupKeyboardShortcuts();
|
|
this.setupWindowResizeHandling();
|
|
this.initializeCustomTasks();
|
|
this.discoverImages().then(() => {
|
|
this.showScreen('start-screen');
|
|
});
|
|
this.discoverAudio();
|
|
|
|
// Check for auto-resume after initialization
|
|
this.checkAutoResume();
|
|
}
|
|
|
|
async initDesktopFeatures() {
|
|
// Initialize desktop file manager
|
|
if (typeof DesktopFileManager !== 'undefined') {
|
|
this.fileManager = new DesktopFileManager(this.dataManager);
|
|
window.desktopFileManager = this.fileManager;
|
|
|
|
// Auto-scan directories on startup
|
|
setTimeout(async () => {
|
|
if (this.fileManager && this.fileManager.isElectron) {
|
|
console.log('🔍 Auto-scanning directories on startup...');
|
|
await this.fileManager.scanAllDirectories();
|
|
}
|
|
}, 1000); // Wait 1 second for initialization to complete
|
|
}
|
|
|
|
// Check if we're in Electron and update UI accordingly
|
|
setTimeout(() => {
|
|
const isElectron = window.electronAPI !== undefined;
|
|
|
|
if (isElectron) {
|
|
document.body.classList.add('desktop-mode');
|
|
// Show desktop-specific features
|
|
document.querySelectorAll('.desktop-only').forEach(el => el.style.display = '');
|
|
document.querySelectorAll('.desktop-feature').forEach(el => el.style.display = '');
|
|
document.querySelectorAll('.web-feature').forEach(el => el.style.display = 'none');
|
|
console.log('🖥️ Desktop mode activated');
|
|
} else {
|
|
document.body.classList.add('web-mode');
|
|
// Hide desktop-only features
|
|
document.querySelectorAll('.desktop-only').forEach(el => el.style.display = 'none');
|
|
document.querySelectorAll('.desktop-feature').forEach(el => el.style.display = 'none');
|
|
document.querySelectorAll('.web-feature').forEach(el => el.style.display = '');
|
|
console.log('🌐 Web mode activated');
|
|
}
|
|
}, 100);
|
|
}
|
|
|
|
initializeCustomTasks() {
|
|
// Load custom tasks from localStorage or use defaults
|
|
const savedMainTasks = localStorage.getItem('customMainTasks');
|
|
const savedConsequenceTasks = localStorage.getItem('customConsequenceTasks');
|
|
|
|
if (savedMainTasks) {
|
|
gameData.mainTasks = JSON.parse(savedMainTasks);
|
|
}
|
|
|
|
if (savedConsequenceTasks) {
|
|
gameData.consequenceTasks = JSON.parse(savedConsequenceTasks);
|
|
}
|
|
|
|
console.log(`Loaded ${gameData.mainTasks.length} main tasks and ${gameData.consequenceTasks.length} consequence tasks`);
|
|
}
|
|
|
|
async discoverImages() {
|
|
try {
|
|
console.log('Loading embedded image manifest...');
|
|
|
|
// Get the current embedded manifest (empty by default)
|
|
let manifest = this.getEmbeddedManifest();
|
|
|
|
// Only scan if the user has explicitly requested it via the scan button
|
|
// No automatic scanning on startup
|
|
console.log('Skipping automatic scan - user must manually scan or upload images');
|
|
|
|
// Build full paths and verify images exist
|
|
gameData.discoveredTaskImages = await this.verifyImagesFromManifest(manifest.tasks, 'images/tasks/');
|
|
gameData.discoveredConsequenceImages = await this.verifyImagesFromManifest(manifest.consequences, 'images/consequences/');
|
|
|
|
console.log(`Task images found: ${gameData.discoveredTaskImages.length}`);
|
|
console.log(`Consequence images found: ${gameData.discoveredConsequenceImages.length}`);
|
|
|
|
} catch (error) {
|
|
console.log('Image manifest loading failed:', error);
|
|
// Don't fallback to pattern detection - keep empty
|
|
gameData.discoveredTaskImages = [];
|
|
gameData.discoveredConsequenceImages = [];
|
|
}
|
|
this.imageDiscoveryComplete = true;
|
|
console.log('Image discovery completed');
|
|
}
|
|
|
|
getEmbeddedManifest() {
|
|
// Empty manifest - users must upload or scan for their own images
|
|
return {
|
|
"tasks": [],
|
|
"consequences": []
|
|
};
|
|
}
|
|
|
|
async updateManifestWithNewImages(manifest) {
|
|
console.log('Scanning for new images...');
|
|
|
|
// Get cached manifest or current one
|
|
let cachedManifest = this.dataManager.get('cachedManifest');
|
|
if (!cachedManifest) {
|
|
cachedManifest = {...manifest}; // Copy original
|
|
}
|
|
|
|
// Scan for new images using pattern detection
|
|
const newTaskImages = await this.scanDirectoryForNewImages('images/tasks/', cachedManifest.tasks);
|
|
const newConsequenceImages = await this.scanDirectoryForNewImages('images/consequences/', cachedManifest.consequences);
|
|
|
|
// Add any new images found
|
|
if (newTaskImages.length > 0) {
|
|
cachedManifest.tasks = [...new Set([...cachedManifest.tasks, ...newTaskImages])]; // Remove duplicates
|
|
console.log(`Found ${newTaskImages.length} new task images:`, newTaskImages);
|
|
}
|
|
|
|
if (newConsequenceImages.length > 0) {
|
|
cachedManifest.consequences = [...new Set([...cachedManifest.consequences, ...newConsequenceImages])]; // Remove duplicates
|
|
console.log(`Found ${newConsequenceImages.length} new consequence images:`, newConsequenceImages);
|
|
}
|
|
|
|
// Save updated manifest to localStorage
|
|
this.dataManager.set('cachedManifest', cachedManifest);
|
|
|
|
return cachedManifest;
|
|
}
|
|
|
|
async scanDirectoryForNewImages(directory, knownImages) {
|
|
const newImages = [];
|
|
|
|
// Comprehensive pattern scan for new images
|
|
const patterns = [
|
|
// Numbers 1-100
|
|
...Array.from({length: 100}, (_, i) => (i + 1).toString()),
|
|
// Image patterns
|
|
...Array.from({length: 50}, (_, i) => `image${i + 1}`),
|
|
...Array.from({length: 50}, (_, i) => `img${i + 1}`),
|
|
...Array.from({length: 50}, (_, i) => `photo${i + 1}`),
|
|
...Array.from({length: 50}, (_, i) => `pic${i + 1}`),
|
|
// Date-based patterns (common camera formats)
|
|
...Array.from({length: 50}, (_, i) => `IMG_${20210101 + i}`),
|
|
...Array.from({length: 50}, (_, i) => `DSC${String(i + 1).padStart(4, '0')}`),
|
|
// Letters a-z
|
|
...Array.from({length: 26}, (_, i) => String.fromCharCode(97 + i)),
|
|
// Random common names
|
|
'new', 'test', 'sample', 'demo', 'example', 'temp', 'screenshot', 'capture',
|
|
// UUID-like patterns
|
|
...Array.from({length: 20}, (_, i) => `img_${Date.now() + i}`)
|
|
];
|
|
|
|
console.log(`Scanning ${directory} for new images (${patterns.length} patterns)...`);
|
|
|
|
// Test each pattern with each supported format
|
|
let checkedCount = 0;
|
|
const maxChecks = 500; // Reasonable limit to prevent hanging
|
|
|
|
for (const pattern of patterns) {
|
|
if (checkedCount >= maxChecks) {
|
|
console.log(`Reached maximum scan limit (${maxChecks} checks)`);
|
|
break;
|
|
}
|
|
|
|
for (const format of gameData.supportedImageFormats) {
|
|
checkedCount++;
|
|
const filename = `${pattern}${format}`;
|
|
|
|
// Skip if we already know about this image
|
|
if (knownImages.includes(filename)) {
|
|
continue;
|
|
}
|
|
|
|
const imagePath = `${directory}${filename}`;
|
|
const exists = await this.checkImageExists(imagePath);
|
|
if (exists) {
|
|
newImages.push(filename);
|
|
console.log('✓ Found NEW image:', filename);
|
|
}
|
|
}
|
|
}
|
|
|
|
console.log(`Scanned ${checkedCount} possibilities, found ${newImages.length} new images`);
|
|
return newImages;
|
|
}
|
|
|
|
async verifyImagesFromManifest(imageList, directory) {
|
|
const validImages = [];
|
|
|
|
for (const imageName of imageList) {
|
|
const imagePath = `${directory}${imageName}`;
|
|
const exists = await this.checkImageExists(imagePath);
|
|
if (exists) {
|
|
validImages.push(imagePath);
|
|
console.log('✓ Verified image:', imagePath);
|
|
} else {
|
|
console.log('✗ Missing image:', imagePath);
|
|
}
|
|
}
|
|
|
|
return validImages;
|
|
}
|
|
|
|
async fallbackImageDiscovery() {
|
|
// Fallback to simplified pattern detection if manifest fails
|
|
console.log('Using fallback pattern detection...');
|
|
|
|
const commonPatterns = ['1', '2', '3', '4', '5', '6', '7', '8', '9', '10'];
|
|
|
|
gameData.discoveredTaskImages = await this.findImagesWithPatterns('images/tasks/', commonPatterns);
|
|
gameData.discoveredConsequenceImages = await this.findImagesWithPatterns('images/consequences/', commonPatterns);
|
|
|
|
// If still no images found, show helpful message but don't use placeholders
|
|
if (gameData.discoveredTaskImages.length === 0 && gameData.discoveredConsequenceImages.length === 0) {
|
|
console.log('No images found. Users will need to upload or scan for images.');
|
|
gameData.discoveredTaskImages = [];
|
|
gameData.discoveredConsequenceImages = [];
|
|
}
|
|
}
|
|
|
|
async findImagesWithPatterns(directory, patterns) {
|
|
const foundImages = [];
|
|
|
|
for (const pattern of patterns) {
|
|
for (const format of gameData.supportedImageFormats) {
|
|
const imagePath = `${directory}${pattern}${format}`;
|
|
const exists = await this.checkImageExists(imagePath);
|
|
if (exists) {
|
|
foundImages.push(imagePath);
|
|
console.log('✓ Found image:', imagePath);
|
|
}
|
|
}
|
|
}
|
|
|
|
return foundImages;
|
|
}
|
|
|
|
setupPlaceholderImages() {
|
|
gameData.discoveredTaskImages = [this.createPlaceholderImage('Task Image')];
|
|
gameData.discoveredConsequenceImages = [this.createPlaceholderImage('Consequence Image')];
|
|
}
|
|
|
|
checkImageExists(imagePath) {
|
|
return new Promise((resolve) => {
|
|
const img = new Image();
|
|
img.onload = () => {
|
|
console.log(`✓ Found image: ${imagePath}`);
|
|
resolve(true);
|
|
};
|
|
img.onerror = () => {
|
|
console.log(`✗ Missing image: ${imagePath}`);
|
|
resolve(false);
|
|
};
|
|
img.src = imagePath;
|
|
|
|
// Timeout after 1 second (faster discovery)
|
|
setTimeout(() => {
|
|
console.log(`⏰ Timeout for image: ${imagePath}`);
|
|
resolve(false);
|
|
}, 1000);
|
|
});
|
|
}
|
|
|
|
createPlaceholderImage(label = 'Task Image') {
|
|
const encodedLabel = encodeURIComponent(label);
|
|
return ``;
|
|
}
|
|
|
|
// Audio Discovery Functions
|
|
async discoverAudio() {
|
|
try {
|
|
console.log('Discovering audio files...');
|
|
|
|
// Initialize audio discovery - scan directories if desktop mode
|
|
if (this.fileManager) {
|
|
await this.fileManager.scanDirectoryForAudio('background');
|
|
await this.fileManager.scanDirectoryForAudio('ambient');
|
|
await this.fileManager.scanDirectoryForAudio('effects');
|
|
console.log('Desktop audio discovery completed');
|
|
} else {
|
|
console.log('Web mode - audio discovery skipped');
|
|
}
|
|
|
|
} catch (error) {
|
|
console.log('Audio discovery failed:', error);
|
|
}
|
|
|
|
this.audioDiscoveryComplete = true;
|
|
console.log('Audio discovery completed');
|
|
}
|
|
|
|
initializeEventListeners() {
|
|
// Screen navigation
|
|
document.getElementById('start-btn').addEventListener('click', () => this.startGame());
|
|
document.getElementById('resume-btn').addEventListener('click', () => this.resumeGame());
|
|
document.getElementById('quit-btn').addEventListener('click', () => this.quitGame());
|
|
document.getElementById('play-again-btn').addEventListener('click', () => this.resetGame());
|
|
|
|
// Game mode selection
|
|
this.initializeGameModeListeners();
|
|
|
|
// Game actions
|
|
document.getElementById('complete-btn').addEventListener('click', () => this.completeTask());
|
|
document.getElementById('skip-btn').addEventListener('click', () => this.skipTask());
|
|
document.getElementById('mercy-skip-btn').addEventListener('click', () => this.mercySkip());
|
|
document.getElementById('pause-btn').addEventListener('click', () => this.pauseGame());
|
|
|
|
// Theme selector
|
|
document.getElementById('theme-dropdown').addEventListener('change', (e) => this.changeTheme(e.target.value));
|
|
|
|
// Options menu toggle
|
|
document.getElementById('options-menu-btn').addEventListener('click', () => this.toggleOptionsMenu());
|
|
|
|
// Music controls
|
|
document.getElementById('music-toggle').addEventListener('click', () => this.toggleMusic());
|
|
document.getElementById('music-toggle-compact').addEventListener('click', (e) => {
|
|
e.stopPropagation(); // Prevent event bubbling
|
|
// The hover panel will show automatically, just indicate it's interactive
|
|
});
|
|
document.getElementById('loop-btn').addEventListener('click', () => this.toggleLoop());
|
|
document.getElementById('shuffle-btn').addEventListener('click', () => this.toggleShuffle());
|
|
document.getElementById('track-selector').addEventListener('change', (e) => this.changeTrack(parseInt(e.target.value)));
|
|
document.getElementById('volume-slider').addEventListener('input', (e) => this.changeVolume(parseInt(e.target.value)));
|
|
|
|
// Task management
|
|
document.getElementById('manage-tasks-btn').addEventListener('click', () => this.showTaskManagement());
|
|
document.getElementById('back-to-start-btn').addEventListener('click', () => this.showScreen('start-screen'));
|
|
document.getElementById('add-task-btn').addEventListener('click', () => this.addNewTask());
|
|
document.getElementById('reset-tasks-btn').addEventListener('click', () => this.resetToDefaultTasks());
|
|
document.getElementById('main-tasks-tab').addEventListener('click', () => this.showTaskTab('main'));
|
|
document.getElementById('consequence-tasks-tab').addEventListener('click', () => this.showTaskTab('consequence'));
|
|
document.getElementById('new-task-type').addEventListener('change', () => this.toggleDifficultyDropdown());
|
|
|
|
// Data management
|
|
document.getElementById('export-btn').addEventListener('click', () => this.exportData());
|
|
document.getElementById('import-btn').addEventListener('click', () => this.importData());
|
|
document.getElementById('import-file').addEventListener('change', (e) => this.handleFileImport(e));
|
|
document.getElementById('stats-btn').addEventListener('click', () => this.showStats());
|
|
document.getElementById('help-btn').addEventListener('click', () => this.showHelp());
|
|
document.getElementById('close-stats').addEventListener('click', () => this.hideStats());
|
|
document.getElementById('close-help').addEventListener('click', () => this.hideHelp());
|
|
document.getElementById('reset-stats-btn').addEventListener('click', () => this.resetStats());
|
|
document.getElementById('export-stats-btn').addEventListener('click', () => this.exportStatsOnly());
|
|
|
|
// Image management - only the main button, others will be attached when screen is shown
|
|
document.getElementById('manage-images-btn').addEventListener('click', () => this.showImageManagement());
|
|
|
|
// Audio management - only the main button, others will be attached when screen is shown
|
|
document.getElementById('manage-audio-btn').addEventListener('click', () => this.showAudioManagement());
|
|
|
|
// Annoyance management - main button and basic controls
|
|
document.getElementById('manage-annoyance-btn').addEventListener('click', () => this.showAnnoyanceManagement());
|
|
|
|
// Load saved theme
|
|
this.loadSavedTheme();
|
|
}
|
|
|
|
initializeGameModeListeners() {
|
|
const gameModeRadios = document.querySelectorAll('input[name="gameMode"]');
|
|
gameModeRadios.forEach(radio => {
|
|
radio.addEventListener('change', () => this.handleGameModeChange());
|
|
});
|
|
|
|
// Add listeners for dropdown changes
|
|
document.getElementById('time-limit-select').addEventListener('change', () => {
|
|
this.handleTimeLimitChange();
|
|
});
|
|
|
|
document.getElementById('score-target-select').addEventListener('change', () => {
|
|
this.handleScoreTargetChange();
|
|
});
|
|
|
|
// Add listeners for custom input changes
|
|
document.getElementById('custom-time-input').addEventListener('input', () => {
|
|
this.handleCustomTimeChange();
|
|
});
|
|
|
|
document.getElementById('custom-score-input').addEventListener('input', () => {
|
|
this.handleCustomScoreChange();
|
|
});
|
|
|
|
// Initialize with default mode
|
|
this.handleGameModeChange();
|
|
}
|
|
|
|
handleGameModeChange() {
|
|
const selectedMode = document.querySelector('input[name="gameMode"]:checked').value;
|
|
this.gameState.gameMode = selectedMode;
|
|
|
|
// Show/hide configuration options based on selected mode
|
|
document.getElementById('timed-config').style.display =
|
|
selectedMode === 'timed' ? 'block' : 'none';
|
|
document.getElementById('score-target-config').style.display =
|
|
selectedMode === 'score-target' ? 'block' : 'none';
|
|
|
|
// Update game state with selected values
|
|
if (selectedMode === 'timed') {
|
|
this.handleTimeLimitChange();
|
|
} else if (selectedMode === 'score-target') {
|
|
this.handleScoreTargetChange();
|
|
}
|
|
|
|
console.log(`Game mode changed to: ${selectedMode}`, this.gameState);
|
|
}
|
|
|
|
handleTimeLimitChange() {
|
|
const timeLimitSelect = document.getElementById('time-limit-select');
|
|
const customTimeInput = document.getElementById('custom-time-input');
|
|
const selectedValue = timeLimitSelect.value;
|
|
|
|
if (selectedValue === 'custom') {
|
|
customTimeInput.style.display = 'block';
|
|
this.handleCustomTimeChange();
|
|
} else {
|
|
customTimeInput.style.display = 'none';
|
|
this.gameState.timeLimit = parseInt(selectedValue);
|
|
}
|
|
}
|
|
|
|
handleScoreTargetChange() {
|
|
const scoreTargetSelect = document.getElementById('score-target-select');
|
|
const customScoreInput = document.getElementById('custom-score-input');
|
|
const selectedValue = scoreTargetSelect.value;
|
|
|
|
if (selectedValue === 'custom') {
|
|
customScoreInput.style.display = 'block';
|
|
this.handleCustomScoreChange();
|
|
} else {
|
|
customScoreInput.style.display = 'none';
|
|
this.gameState.scoreTarget = parseInt(selectedValue);
|
|
}
|
|
}
|
|
|
|
handleCustomTimeChange() {
|
|
const customTimeInput = document.getElementById('custom-time-input');
|
|
const minutes = parseInt(customTimeInput.value) || 15;
|
|
this.gameState.timeLimit = minutes * 60; // Convert minutes to seconds
|
|
}
|
|
|
|
handleCustomScoreChange() {
|
|
const customScoreInput = document.getElementById('custom-score-input');
|
|
const score = parseInt(customScoreInput.value) || 300;
|
|
this.gameState.scoreTarget = score;
|
|
}
|
|
|
|
setupKeyboardShortcuts() {
|
|
document.addEventListener('keydown', (e) => {
|
|
// Don't trigger shortcuts when typing in inputs or textareas
|
|
if (e.target.tagName === 'INPUT' || e.target.tagName === 'TEXTAREA') {
|
|
return;
|
|
}
|
|
|
|
switch(e.key.toLowerCase()) {
|
|
case ' ': // Spacebar - pause/unpause game
|
|
e.preventDefault();
|
|
if (this.gameState.isRunning) {
|
|
this.pauseGame();
|
|
} else if (this.gameState.isPaused) {
|
|
this.resumeGame();
|
|
}
|
|
break;
|
|
|
|
case 'p': // P key - pause/unpause game
|
|
e.preventDefault();
|
|
if (this.gameState.isRunning) {
|
|
this.pauseGame();
|
|
} else if (this.gameState.isPaused) {
|
|
this.resumeGame();
|
|
}
|
|
break;
|
|
|
|
case 'm': // M key - toggle music
|
|
e.preventDefault();
|
|
this.toggleMusic();
|
|
break;
|
|
|
|
case 'escape': // Escape key - close modals or return to start
|
|
e.preventDefault();
|
|
// Close help modal if open
|
|
const helpModal = document.getElementById('help-modal');
|
|
if (helpModal && helpModal.style.display !== 'none') {
|
|
this.hideHelp();
|
|
return;
|
|
}
|
|
|
|
// Close stats modal if open
|
|
const statsModal = document.getElementById('stats-modal');
|
|
if (statsModal && !statsModal.hidden) {
|
|
this.hideStats();
|
|
return;
|
|
}
|
|
|
|
// If in task management, go back to start screen
|
|
if (document.getElementById('task-management-screen').style.display !== 'none') {
|
|
this.showScreen('start-screen');
|
|
return;
|
|
}
|
|
|
|
// If in game, pause it
|
|
if (this.gameState.isRunning) {
|
|
this.pauseGame();
|
|
}
|
|
break;
|
|
|
|
case 'h': // H key - toggle help menu
|
|
e.preventDefault();
|
|
this.toggleHelp();
|
|
break;
|
|
|
|
case 'enter': // Enter key - complete task
|
|
e.preventDefault();
|
|
if (this.gameState.isRunning && !this.gameState.isPaused) {
|
|
this.completeTask();
|
|
}
|
|
break;
|
|
|
|
case 'control': // Ctrl key - regular skip only
|
|
e.preventDefault();
|
|
if (this.gameState.isRunning && !this.gameState.isPaused) {
|
|
// Check for Shift+Ctrl for mercy skip
|
|
if (e.shiftKey) {
|
|
const mercyBtn = document.getElementById('mercy-skip-btn');
|
|
if (mercyBtn && mercyBtn.style.display !== 'none' && !mercyBtn.disabled) {
|
|
this.mercySkip();
|
|
}
|
|
} else {
|
|
this.skipTask();
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
});
|
|
|
|
// Show keyboard shortcuts hint
|
|
console.log('Keyboard shortcuts enabled: Enter (complete), Ctrl (skip), Shift+Ctrl (mercy skip), Space/P (pause), M (music), H (help), Escape (close/back)');
|
|
}
|
|
|
|
setupWindowResizeHandling() {
|
|
let resizeTimeout;
|
|
|
|
// Handle window resize events for dynamic image scaling
|
|
window.addEventListener('resize', () => {
|
|
// Debounce resize events to avoid excessive processing
|
|
clearTimeout(resizeTimeout);
|
|
resizeTimeout = setTimeout(() => {
|
|
this.handleWindowResize();
|
|
}, 150); // Wait 150ms after resize stops
|
|
});
|
|
|
|
// Initial setup
|
|
this.handleWindowResize();
|
|
}
|
|
|
|
handleWindowResize() {
|
|
const taskImage = document.getElementById('task-image');
|
|
if (!taskImage) return;
|
|
|
|
// Get current window dimensions
|
|
const windowWidth = window.innerWidth;
|
|
const windowHeight = window.innerHeight;
|
|
|
|
// Add dynamic class based on window size for additional styling hooks
|
|
const gameContainer = document.querySelector('.game-container');
|
|
if (gameContainer) {
|
|
gameContainer.classList.remove('window-small', 'window-medium', 'window-large', 'window-xl');
|
|
|
|
if (windowWidth >= 1600) {
|
|
gameContainer.classList.add('window-xl');
|
|
} else if (windowWidth >= 1200) {
|
|
gameContainer.classList.add('window-large');
|
|
} else if (windowWidth >= 900) {
|
|
gameContainer.classList.add('window-medium');
|
|
} else {
|
|
gameContainer.classList.add('window-small');
|
|
}
|
|
}
|
|
|
|
// Force layout recalculation for better image sizing
|
|
if (taskImage.src && this.gameState.isRunning) {
|
|
// Trigger a reflow to ensure proper image sizing
|
|
taskImage.style.opacity = '0.99';
|
|
requestAnimationFrame(() => {
|
|
taskImage.style.opacity = '';
|
|
});
|
|
}
|
|
|
|
console.log(`Window resized to ${windowWidth}x${windowHeight}, image scaling updated`);
|
|
}
|
|
|
|
checkAutoResume() {
|
|
try {
|
|
// Check if there's a saved game state to resume
|
|
const savedGameState = this.dataManager.get('autoSaveGameState');
|
|
|
|
if (savedGameState && savedGameState.isRunning === false && savedGameState.isPaused === true) {
|
|
// Show auto-resume prompt
|
|
setTimeout(() => {
|
|
if (confirm('You have a paused game session. Would you like to resume where you left off?')) {
|
|
this.loadAutoSavedGame(savedGameState);
|
|
} else {
|
|
// Clear the auto-save if user declines
|
|
localStorage.removeItem('autoSaveGameState');
|
|
}
|
|
}, 500); // Small delay to ensure UI is ready
|
|
}
|
|
} catch (error) {
|
|
console.warn('Auto-resume check failed:', this.formatErrorMessage('auto-resume', error));
|
|
// Clear potentially corrupted auto-save data
|
|
localStorage.removeItem('autoSaveGameState');
|
|
}
|
|
}
|
|
|
|
autoSaveGameState() {
|
|
// Save current game state for auto-resume
|
|
if (this.gameState.isRunning || this.gameState.isPaused) {
|
|
const saveState = {
|
|
...this.gameState,
|
|
currentTask: this.gameState.currentTask,
|
|
savedAt: new Date().toISOString()
|
|
};
|
|
this.dataManager.set('autoSaveGameState', saveState);
|
|
}
|
|
}
|
|
|
|
loadAutoSavedGame(savedState) {
|
|
// Restore the saved game state
|
|
this.gameState = {
|
|
...this.gameState,
|
|
...savedState,
|
|
isRunning: false, // Will be set to true when resumed
|
|
isPaused: true
|
|
};
|
|
|
|
// Update UI to show the paused game
|
|
this.showScreen('game-screen');
|
|
this.updateDisplay();
|
|
|
|
// Show the resume notification
|
|
this.showNotification('Game session restored! Click Resume to continue.', 'success');
|
|
}
|
|
|
|
// Loading indicator methods
|
|
showButtonLoading(buttonId) {
|
|
const button = document.getElementById(buttonId);
|
|
if (button) {
|
|
button.disabled = true;
|
|
const textSpan = button.querySelector('.btn-text');
|
|
const loadingSpan = button.querySelector('.btn-loading');
|
|
if (textSpan && loadingSpan) {
|
|
textSpan.style.display = 'none';
|
|
loadingSpan.style.display = 'inline';
|
|
}
|
|
}
|
|
}
|
|
|
|
hideButtonLoading(buttonId) {
|
|
const button = document.getElementById(buttonId);
|
|
if (button) {
|
|
button.disabled = false;
|
|
const textSpan = button.querySelector('.btn-text');
|
|
const loadingSpan = button.querySelector('.btn-loading');
|
|
if (textSpan && loadingSpan) {
|
|
textSpan.style.display = 'inline';
|
|
loadingSpan.style.display = 'none';
|
|
}
|
|
}
|
|
}
|
|
|
|
toggleDifficultyDropdown() {
|
|
const taskType = document.getElementById('new-task-type').value;
|
|
const difficultyGroup = document.getElementById('difficulty-input-group');
|
|
|
|
if (taskType === 'main') {
|
|
difficultyGroup.style.display = 'block';
|
|
} else {
|
|
difficultyGroup.style.display = 'none';
|
|
}
|
|
}
|
|
|
|
// Task Management Methods
|
|
showTaskManagement() {
|
|
this.showScreen('task-management-screen');
|
|
this.populateTaskLists();
|
|
this.toggleDifficultyDropdown(); // Set initial visibility
|
|
}
|
|
|
|
showTaskTab(tabType) {
|
|
// Update tab buttons
|
|
document.querySelectorAll('.tab-btn').forEach(btn => btn.classList.remove('active'));
|
|
document.getElementById(`${tabType}-tasks-tab`).classList.add('active');
|
|
|
|
// Update task lists
|
|
document.querySelectorAll('.task-list').forEach(list => list.classList.remove('active'));
|
|
document.getElementById(`${tabType}-tasks-list`).classList.add('active');
|
|
}
|
|
|
|
populateTaskLists() {
|
|
this.populateTaskList('main', gameData.mainTasks);
|
|
this.populateTaskList('consequence', gameData.consequenceTasks);
|
|
}
|
|
|
|
populateTaskList(type, tasks) {
|
|
const listElement = document.getElementById(`${type}-tasks-list`);
|
|
|
|
if (tasks.length === 0) {
|
|
listElement.innerHTML = '<div class="empty-list">No tasks added yet. Add some tasks to get started!</div>';
|
|
return;
|
|
}
|
|
|
|
listElement.innerHTML = tasks.map((task, index) => {
|
|
let difficultyDisplay = '';
|
|
if (type === 'main' && task.difficulty) {
|
|
const emoji = this.getDifficultyEmoji(task.difficulty);
|
|
const points = this.getPointsForDifficulty(task.difficulty);
|
|
difficultyDisplay = `<div class="task-difficulty-display">${emoji} ${task.difficulty} (${points} ${points === 1 ? 'point' : 'points'})</div>`;
|
|
}
|
|
|
|
return `
|
|
<div class="task-item" data-type="${type}" data-index="${index}">
|
|
<div class="task-text-display">${task.text}</div>
|
|
${difficultyDisplay}
|
|
<div class="task-actions">
|
|
<button class="btn btn-info btn-small" onclick="game.editTask('${type}', ${index})">Edit</button>
|
|
<button class="btn btn-danger btn-small" onclick="game.deleteTask('${type}', ${index})">Delete</button>
|
|
</div>
|
|
</div>
|
|
`;
|
|
}).join('');
|
|
}
|
|
|
|
addNewTask() {
|
|
const taskText = document.getElementById('new-task-text').value.trim();
|
|
const taskType = document.getElementById('new-task-type').value;
|
|
const taskDifficulty = document.getElementById('new-task-difficulty').value;
|
|
|
|
if (!taskText) {
|
|
alert('Please enter a task description!');
|
|
return;
|
|
}
|
|
|
|
// Create new task object
|
|
const newTask = {
|
|
id: Date.now(), // Use timestamp as unique ID
|
|
text: taskText
|
|
};
|
|
|
|
// Add difficulty for main tasks only
|
|
if (taskType === 'main') {
|
|
newTask.difficulty = taskDifficulty;
|
|
}
|
|
|
|
// Add to appropriate array
|
|
if (taskType === 'main') {
|
|
gameData.mainTasks.push(newTask);
|
|
localStorage.setItem('customMainTasks', JSON.stringify(gameData.mainTasks));
|
|
} else {
|
|
gameData.consequenceTasks.push(newTask);
|
|
localStorage.setItem('customConsequenceTasks', JSON.stringify(gameData.consequenceTasks));
|
|
}
|
|
|
|
// Clear input and refresh list
|
|
document.getElementById('new-task-text').value = '';
|
|
document.getElementById('new-task-difficulty').value = 'Medium';
|
|
this.populateTaskLists();
|
|
|
|
// Switch to the appropriate tab
|
|
this.showTaskTab(taskType === 'main' ? 'main' : 'consequence');
|
|
|
|
console.log(`Added new ${taskType} task: ${taskText}`);
|
|
}
|
|
|
|
editTask(type, index) {
|
|
const tasks = type === 'main' ? gameData.mainTasks : gameData.consequenceTasks;
|
|
const task = tasks[index];
|
|
|
|
const newText = prompt(`Edit ${type} task:`, task.text);
|
|
if (newText !== null && newText.trim() !== '') {
|
|
tasks[index].text = newText.trim();
|
|
|
|
// Save to localStorage
|
|
const storageKey = type === 'main' ? 'customMainTasks' : 'customConsequenceTasks';
|
|
localStorage.setItem(storageKey, JSON.stringify(tasks));
|
|
|
|
// Refresh display
|
|
this.populateTaskLists();
|
|
console.log(`Edited ${type} task: ${newText}`);
|
|
}
|
|
}
|
|
|
|
deleteTask(type, index) {
|
|
const tasks = type === 'main' ? gameData.mainTasks : gameData.consequenceTasks;
|
|
const task = tasks[index];
|
|
|
|
if (confirm(`Are you sure you want to delete this ${type} task?\n\n"${task.text}"`)) {
|
|
tasks.splice(index, 1);
|
|
|
|
// Save to localStorage
|
|
const storageKey = type === 'main' ? 'customMainTasks' : 'customConsequenceTasks';
|
|
localStorage.setItem(storageKey, JSON.stringify(tasks));
|
|
|
|
// Refresh display
|
|
this.populateTaskLists();
|
|
console.log(`Deleted ${type} task: ${task.text}`);
|
|
}
|
|
}
|
|
|
|
resetToDefaultTasks() {
|
|
if (confirm('This will reset all tasks to the original defaults. Are you sure?')) {
|
|
// Clear localStorage
|
|
localStorage.removeItem('customMainTasks');
|
|
localStorage.removeItem('customConsequenceTasks');
|
|
|
|
// Reload the page to get fresh default tasks
|
|
location.reload();
|
|
}
|
|
}
|
|
|
|
changeTheme(themeName) {
|
|
// Remove all existing theme classes
|
|
const themeClasses = ['theme-ocean', 'theme-sunset', 'theme-forest', 'theme-midnight', 'theme-pastel', 'theme-neon', 'theme-autumn', 'theme-monochrome'];
|
|
document.body.classList.remove(...themeClasses);
|
|
|
|
// Add new theme class (ocean is default, no class needed)
|
|
if (themeName !== 'ocean') {
|
|
document.body.classList.add(`theme-${themeName}`);
|
|
}
|
|
|
|
// Save theme preference
|
|
localStorage.setItem('gameTheme', themeName);
|
|
console.log(`Theme changed to: ${themeName}`);
|
|
}
|
|
|
|
loadSavedTheme() {
|
|
const savedTheme = localStorage.getItem('gameTheme') || 'ocean';
|
|
document.getElementById('theme-dropdown').value = savedTheme;
|
|
this.changeTheme(savedTheme);
|
|
}
|
|
|
|
showScreen(screenId) {
|
|
console.log(`showScreen(${screenId}) called`);
|
|
// Hide all screens
|
|
document.querySelectorAll('.screen').forEach(screen => {
|
|
screen.classList.remove('active');
|
|
console.log(`Removed active class from screen:`, screen.id);
|
|
});
|
|
|
|
// Show target screen
|
|
const targetScreen = document.getElementById(screenId);
|
|
if (targetScreen) {
|
|
targetScreen.classList.add('active');
|
|
console.log(`Added active class to screen: ${screenId}`);
|
|
} else {
|
|
console.error(`Screen element not found: ${screenId}`);
|
|
}
|
|
|
|
// Update start screen status when showing start screen
|
|
if (screenId === 'start-screen') {
|
|
this.updateStartScreenStatus();
|
|
}
|
|
}
|
|
|
|
updateStartScreenStatus() {
|
|
// Status message removed for cleaner home screen interface
|
|
const statusDiv = document.getElementById('start-screen-status');
|
|
if (statusDiv) {
|
|
statusDiv.style.display = 'none';
|
|
}
|
|
}
|
|
|
|
toggleOptionsMenu() {
|
|
const optionsMenu = document.getElementById('options-menu');
|
|
const button = document.getElementById('options-menu-btn');
|
|
|
|
if (optionsMenu.style.display === 'none' || optionsMenu.style.display === '') {
|
|
optionsMenu.style.display = 'block';
|
|
button.textContent = '⚙️ Close Options';
|
|
} else {
|
|
optionsMenu.style.display = 'none';
|
|
button.textContent = '⚙️ Options';
|
|
}
|
|
}
|
|
|
|
toggleMusic() {
|
|
this.musicManager.toggle();
|
|
}
|
|
|
|
toggleLoop() {
|
|
this.musicManager.toggleLoopMode();
|
|
}
|
|
|
|
toggleShuffle() {
|
|
this.musicManager.toggleShuffleMode();
|
|
}
|
|
|
|
changeTrack(trackIndex) {
|
|
this.musicManager.changeTrack(trackIndex);
|
|
}
|
|
|
|
changeVolume(volume) {
|
|
this.musicManager.setVolume(volume);
|
|
}
|
|
|
|
// Data Management Methods
|
|
exportData() {
|
|
try {
|
|
this.showButtonLoading('export-btn');
|
|
|
|
// Small delay to show loading indicator
|
|
setTimeout(() => {
|
|
try {
|
|
this.dataManager.exportData(true);
|
|
this.showNotification('💾 Save file exported successfully!', 'success');
|
|
} catch (error) {
|
|
this.showNotification('❌ ' + this.formatErrorMessage('export', error), 'error');
|
|
} finally {
|
|
this.hideButtonLoading('export-btn');
|
|
}
|
|
}, 300);
|
|
} catch (error) {
|
|
this.showNotification('❌ ' + this.formatErrorMessage('export', error), 'error');
|
|
this.hideButtonLoading('export-btn');
|
|
}
|
|
}
|
|
|
|
importData() {
|
|
// Trigger the file input
|
|
document.getElementById('import-file').click();
|
|
}
|
|
|
|
async handleFileImport(event) {
|
|
const file = event.target.files[0];
|
|
if (!file) return;
|
|
|
|
this.showButtonLoading('import-btn');
|
|
|
|
try {
|
|
await this.dataManager.importData(file);
|
|
this.showNotification('📁 Save file imported successfully!', 'success');
|
|
|
|
// Refresh UI to show new data
|
|
this.musicManager.volume = this.dataManager.getSetting('music.volume') || 30;
|
|
this.musicManager.currentTrackIndex = this.dataManager.getSetting('music.currentTrack') || 0;
|
|
this.musicManager.loopMode = this.dataManager.getSetting('music.loopMode') || 0;
|
|
this.musicManager.shuffleMode = this.dataManager.getSetting('music.shuffle') || false;
|
|
this.musicManager.initializeVolumeUI();
|
|
|
|
// Update theme
|
|
const theme = this.dataManager.getSetting('theme') || 'ocean';
|
|
this.changeTheme(theme);
|
|
|
|
} catch (error) {
|
|
this.showNotification('❌ ' + this.formatErrorMessage('import', error), 'error');
|
|
} finally {
|
|
this.hideButtonLoading('import-btn');
|
|
}
|
|
|
|
// Clear the file input
|
|
event.target.value = '';
|
|
}
|
|
|
|
showStats() {
|
|
const stats = this.dataManager.getStats();
|
|
|
|
// Update stat displays
|
|
document.getElementById('stat-games').textContent = stats.totalGamesPlayed;
|
|
document.getElementById('stat-completed').textContent = stats.totalTasksCompleted;
|
|
document.getElementById('stat-score').textContent = stats.bestScore;
|
|
document.getElementById('stat-streak').textContent = stats.currentStreak;
|
|
document.getElementById('stat-rate').textContent = stats.completionPercentage + '%';
|
|
document.getElementById('stat-hours').textContent = stats.hoursPlayed;
|
|
|
|
document.getElementById('stats-modal').style.display = 'block';
|
|
}
|
|
|
|
hideStats() {
|
|
document.getElementById('stats-modal').style.display = 'none';
|
|
}
|
|
|
|
showHelp() {
|
|
document.getElementById('help-modal').style.display = 'block';
|
|
}
|
|
|
|
hideHelp() {
|
|
document.getElementById('help-modal').style.display = 'none';
|
|
}
|
|
|
|
toggleHelp() {
|
|
const helpModal = document.getElementById('help-modal');
|
|
if (helpModal.style.display === 'none' || helpModal.style.display === '') {
|
|
this.showHelp();
|
|
} else {
|
|
this.hideHelp();
|
|
}
|
|
}
|
|
|
|
// Image Management Methods
|
|
showImageManagement() {
|
|
// Reset listener flag to allow fresh attachment
|
|
this.imageManagementListenersAttached = false;
|
|
|
|
this.showScreen('image-management-screen');
|
|
this.setupImageManagementEventListeners();
|
|
|
|
// Wait for image discovery to complete before loading gallery
|
|
if (!this.imageDiscoveryComplete) {
|
|
const gallery = document.getElementById('image-gallery');
|
|
gallery.innerHTML = '<div class="loading">Discovering images...</div>';
|
|
|
|
// Wait and try again
|
|
setTimeout(() => {
|
|
if (this.imageDiscoveryComplete) {
|
|
this.loadImageGallery();
|
|
} else {
|
|
gallery.innerHTML = '<div class="loading">Still discovering images... Please wait</div>';
|
|
setTimeout(() => this.loadImageGallery(), 1000);
|
|
}
|
|
}, 500);
|
|
} else {
|
|
this.loadImageGallery();
|
|
}
|
|
}
|
|
|
|
switchImageTab(tabType) {
|
|
// Update tab buttons
|
|
const taskTab = document.getElementById('task-images-tab');
|
|
const consequenceTab = document.getElementById('consequence-images-tab');
|
|
const taskGallery = document.getElementById('task-images-gallery');
|
|
const consequenceGallery = document.getElementById('consequence-images-gallery');
|
|
|
|
if (tabType === 'task') {
|
|
taskTab.classList.add('active');
|
|
consequenceTab.classList.remove('active');
|
|
taskGallery.classList.add('active');
|
|
consequenceGallery.classList.remove('active');
|
|
} else {
|
|
taskTab.classList.remove('active');
|
|
consequenceTab.classList.add('active');
|
|
taskGallery.classList.remove('active');
|
|
consequenceGallery.classList.add('active');
|
|
}
|
|
|
|
// Update gallery controls to work with current tab
|
|
this.updateImageGalleryControls(tabType);
|
|
}
|
|
|
|
updateImageGalleryControls(activeTab) {
|
|
// Update the select/deselect/delete buttons to work with the active tab
|
|
const selectAllBtn = document.getElementById('select-all-images-btn');
|
|
const deselectAllBtn = document.getElementById('deselect-all-images-btn');
|
|
const deleteBtn = document.getElementById('delete-selected-btn');
|
|
|
|
if (selectAllBtn) {
|
|
selectAllBtn.onclick = () => this.selectAllImages(activeTab);
|
|
}
|
|
|
|
if (deselectAllBtn) {
|
|
deselectAllBtn.onclick = () => this.deselectAllImages(activeTab);
|
|
}
|
|
|
|
if (deleteBtn) {
|
|
deleteBtn.onclick = () => this.deleteSelectedImages(activeTab);
|
|
}
|
|
}
|
|
|
|
setupImageManagementEventListeners() {
|
|
// Check if we already have listeners attached to prevent duplicates
|
|
if (this.imageManagementListenersAttached) {
|
|
return;
|
|
}
|
|
|
|
// Back button
|
|
const backBtn = document.getElementById('back-to-start-from-images-btn');
|
|
if (backBtn) {
|
|
backBtn.onclick = () => this.showScreen('start-screen');
|
|
}
|
|
|
|
// Desktop import buttons
|
|
const importTaskBtn = document.getElementById('import-task-images-btn');
|
|
if (importTaskBtn) {
|
|
importTaskBtn.onclick = async () => {
|
|
if (this.fileManager) {
|
|
await this.fileManager.selectAndImportImages('task');
|
|
this.loadImageGallery(); // Refresh the gallery to show new images
|
|
} else {
|
|
this.showNotification('Desktop file manager not available', 'warning');
|
|
}
|
|
};
|
|
}
|
|
|
|
const importConsequenceBtn = document.getElementById('import-consequence-images-btn');
|
|
if (importConsequenceBtn) {
|
|
importConsequenceBtn.onclick = async () => {
|
|
if (this.fileManager) {
|
|
await this.fileManager.selectAndImportImages('consequence');
|
|
this.loadImageGallery(); // Refresh the gallery to show new images
|
|
} else {
|
|
this.showNotification('Desktop file manager not available', 'warning');
|
|
}
|
|
};
|
|
}
|
|
|
|
// Storage info button
|
|
const storageInfoBtn = document.getElementById('storage-info-btn');
|
|
if (storageInfoBtn) {
|
|
storageInfoBtn.onclick = () => this.showStorageInfo();
|
|
}
|
|
|
|
// Tab buttons
|
|
const taskImagesTab = document.getElementById('task-images-tab');
|
|
if (taskImagesTab) {
|
|
taskImagesTab.onclick = () => this.switchImageTab('task');
|
|
}
|
|
|
|
const consequenceImagesTab = document.getElementById('consequence-images-tab');
|
|
if (consequenceImagesTab) {
|
|
consequenceImagesTab.onclick = () => this.switchImageTab('consequence');
|
|
}
|
|
|
|
// Upload input (fallback for web mode)
|
|
const uploadInput = document.getElementById('image-upload-input');
|
|
if (uploadInput) {
|
|
uploadInput.onchange = (e) => this.handleImageUpload(e);
|
|
}
|
|
|
|
// Web upload button (fallback)
|
|
const uploadBtn = document.getElementById('upload-images-btn');
|
|
if (uploadBtn) {
|
|
uploadBtn.onclick = () => this.uploadImages();
|
|
}
|
|
|
|
// Mark listeners as attached
|
|
this.imageManagementListenersAttached = true;
|
|
}
|
|
|
|
loadImageGallery() {
|
|
// Load both task and consequence image galleries
|
|
this.cleanupInvalidImages(); // Clean up invalid images first
|
|
this.loadTaskImages();
|
|
this.loadConsequenceImages();
|
|
this.updateImageCounts();
|
|
|
|
// Initialize with task tab active and update controls
|
|
this.updateImageGalleryControls('task');
|
|
}
|
|
|
|
loadTaskImages() {
|
|
const gallery = document.getElementById('task-images-gallery');
|
|
gallery.innerHTML = '<div class="loading">Loading task images...</div>';
|
|
|
|
// Get task images
|
|
const taskImages = gameData.discoveredTaskImages || [];
|
|
const customImages = this.dataManager.get('customImages') || { task: [], consequence: [] };
|
|
const disabledImages = this.dataManager.get('disabledImages') || [];
|
|
|
|
// Get custom task images (handle both old array format and new object format)
|
|
let customTaskImages = [];
|
|
if (Array.isArray(customImages)) {
|
|
// Old format - treat all as task images for backward compatibility
|
|
customTaskImages = customImages;
|
|
} else {
|
|
// New format - get task images specifically
|
|
customTaskImages = customImages.task || [];
|
|
}
|
|
|
|
const allTaskImages = [...taskImages, ...customTaskImages];
|
|
|
|
if (allTaskImages.length === 0) {
|
|
gallery.innerHTML = '<div class="no-images">No task images found. Upload or scan for task images!</div>';
|
|
return;
|
|
}
|
|
|
|
gallery.innerHTML = '';
|
|
this.renderImageItems(gallery, allTaskImages, disabledImages, customTaskImages);
|
|
}
|
|
|
|
loadConsequenceImages() {
|
|
const gallery = document.getElementById('consequence-images-gallery');
|
|
gallery.innerHTML = '<div class="loading">Loading consequence images...</div>';
|
|
|
|
// Get consequence images
|
|
const consequenceImages = gameData.discoveredConsequenceImages || [];
|
|
const customImages = this.dataManager.get('customImages') || { task: [], consequence: [] };
|
|
const disabledImages = this.dataManager.get('disabledImages') || [];
|
|
|
|
// Get custom consequence images (new format only)
|
|
let customConsequenceImages = [];
|
|
if (!Array.isArray(customImages)) {
|
|
customConsequenceImages = customImages.consequence || [];
|
|
}
|
|
|
|
const allConsequenceImages = [...consequenceImages, ...customConsequenceImages];
|
|
|
|
if (allConsequenceImages.length === 0) {
|
|
gallery.innerHTML = '<div class="no-images">No consequence images found. Upload or scan for consequence images!</div>';
|
|
return;
|
|
}
|
|
|
|
gallery.innerHTML = '';
|
|
this.renderImageItems(gallery, allConsequenceImages, disabledImages, customConsequenceImages);
|
|
}
|
|
|
|
renderImageItems(gallery, images, disabledImages, customImages) {
|
|
images.forEach((imageData, index) => {
|
|
const imageItem = document.createElement('div');
|
|
imageItem.className = 'image-item';
|
|
|
|
// Handle both old base64 format and new cached metadata format
|
|
let imagePath, displayName, imageSrc;
|
|
|
|
if (typeof imageData === 'string') {
|
|
// Old base64 format or regular file path
|
|
imagePath = imageData;
|
|
imageSrc = imageData;
|
|
displayName = this.getImageDisplayName(imageData);
|
|
} else {
|
|
// New cached metadata format
|
|
imagePath = imageData.cachedPath || imageData.originalName;
|
|
|
|
// Use dataUrl if available, otherwise show placeholder
|
|
if (imageData.dataUrl) {
|
|
imageSrc = imageData.dataUrl;
|
|
} else {
|
|
// Legacy cached image without dataUrl - show placeholder
|
|
imageSrc = this.createPlaceholderImage('Missing Data');
|
|
console.warn('Image missing dataUrl:', imageData.originalName);
|
|
}
|
|
|
|
displayName = imageData.originalName || 'Cached Image';
|
|
}
|
|
|
|
imageItem.dataset.imagePath = imagePath;
|
|
|
|
const isDisabled = disabledImages.includes(imagePath);
|
|
|
|
imageItem.innerHTML = `
|
|
<img src="${imageSrc}" alt="Image" class="image-preview" title="${displayName}" onerror="this.src=''">
|
|
<div class="image-info">
|
|
<div class="image-controls">
|
|
<label>
|
|
<input type="checkbox" class="image-checkbox" ${isDisabled ? '' : 'checked'}>
|
|
Enable
|
|
</label>
|
|
<span class="image-status ${isDisabled ? 'image-disabled' : 'image-enabled'}">
|
|
${isDisabled ? 'Disabled' : 'Enabled'}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
`;
|
|
|
|
// Add event listener for enable/disable
|
|
const checkbox = imageItem.querySelector('.image-checkbox');
|
|
checkbox.addEventListener('change', (e) => this.toggleImageEnabled(imagePath, e.target.checked));
|
|
|
|
// Add event listener for individual image selection (click to select/deselect)
|
|
imageItem.addEventListener('click', (e) => {
|
|
// Don't toggle selection if clicking on the checkbox
|
|
if (e.target.type !== 'checkbox') {
|
|
this.toggleImageSelection(imageItem);
|
|
}
|
|
});
|
|
|
|
gallery.appendChild(imageItem);
|
|
});
|
|
}
|
|
|
|
updateImageCounts() {
|
|
const imageCount = document.querySelector('.image-count');
|
|
const taskImages = gameData.discoveredTaskImages || [];
|
|
const consequenceImages = gameData.discoveredConsequenceImages || [];
|
|
const customImages = this.dataManager.get('customImages') || { task: [], consequence: [] };
|
|
const disabledImages = this.dataManager.get('disabledImages') || [];
|
|
|
|
// Handle both old array format and new object format
|
|
let customTaskCount = 0;
|
|
let customConsequenceCount = 0;
|
|
let totalCustomImages = 0;
|
|
|
|
if (Array.isArray(customImages)) {
|
|
// Old format - treat all as task images
|
|
customTaskCount = customImages.length;
|
|
totalCustomImages = customImages.length;
|
|
} else {
|
|
// New format
|
|
customTaskCount = (customImages.task || []).length;
|
|
customConsequenceCount = (customImages.consequence || []).length;
|
|
totalCustomImages = customTaskCount + customConsequenceCount;
|
|
}
|
|
|
|
const totalTaskImages = taskImages.length + customTaskCount;
|
|
const totalConsequenceImages = consequenceImages.length + customConsequenceCount;
|
|
const totalImages = totalTaskImages + totalConsequenceImages;
|
|
const enabledImages = totalImages - disabledImages.length;
|
|
|
|
if (imageCount) {
|
|
imageCount.textContent = `${totalImages} total images (${totalTaskImages} task, ${totalConsequenceImages} consequence, ${enabledImages} enabled)`;
|
|
}
|
|
}
|
|
|
|
getAllImages() {
|
|
// Return the current images from the game's discovery system
|
|
const taskImages = gameData.discoveredTaskImages || [];
|
|
const consequenceImages = gameData.discoveredConsequenceImages || [];
|
|
|
|
console.log('getAllImages - taskImages:', taskImages);
|
|
console.log('getAllImages - consequenceImages:', consequenceImages);
|
|
console.log('getAllImages - imageDiscoveryComplete:', this.imageDiscoveryComplete);
|
|
|
|
return [...taskImages, ...consequenceImages];
|
|
}
|
|
|
|
getImageDisplayName(imagePath) {
|
|
// Extract filename from path
|
|
const parts = imagePath.split('/');
|
|
return parts[parts.length - 1];
|
|
}
|
|
|
|
toggleImageEnabled(imagePath, enabled) {
|
|
let disabledImages = this.dataManager.get('disabledImages') || [];
|
|
|
|
if (enabled) {
|
|
disabledImages = disabledImages.filter(img => img !== imagePath);
|
|
} else {
|
|
if (!disabledImages.includes(imagePath)) {
|
|
disabledImages.push(imagePath);
|
|
}
|
|
}
|
|
|
|
this.dataManager.set('disabledImages', disabledImages);
|
|
|
|
// Update the status display
|
|
const imageItem = document.querySelector(`[data-image-path="${imagePath}"]`);
|
|
if (imageItem) {
|
|
const statusElement = imageItem.querySelector('.image-status');
|
|
statusElement.className = `image-status ${enabled ? 'image-enabled' : 'image-disabled'}`;
|
|
statusElement.textContent = enabled ? 'Enabled' : 'Disabled';
|
|
}
|
|
|
|
// Update count
|
|
this.updateImageCount();
|
|
|
|
this.showNotification(`Image ${enabled ? 'enabled' : 'disabled'} successfully!`, 'success');
|
|
}
|
|
|
|
toggleImageSelection(imageItem) {
|
|
imageItem.classList.toggle('selected');
|
|
}
|
|
|
|
uploadImages() {
|
|
document.getElementById('image-upload-input').click();
|
|
}
|
|
|
|
async handleImageUpload(event) {
|
|
const files = Array.from(event.target.files);
|
|
if (files.length === 0) return;
|
|
|
|
// Determine which tab is active to know where to save images
|
|
const activeTab = this.getActiveImageTab();
|
|
|
|
this.showNotification(`Processing ${files.length} images for ${activeTab} category...`, 'info');
|
|
let successCount = 0;
|
|
let failedCount = 0;
|
|
const maxImages = 50; // Reasonable limit to prevent storage overflow
|
|
|
|
// Check current image count
|
|
const customImages = this.dataManager.get('customImages') || { task: [], consequence: [] };
|
|
const currentTaskCount = Array.isArray(customImages) ? customImages.length : (customImages.task || []).length;
|
|
const currentConsequenceCount = Array.isArray(customImages) ? 0 : (customImages.consequence || []).length;
|
|
const totalCurrentImages = currentTaskCount + currentConsequenceCount;
|
|
|
|
if (totalCurrentImages >= maxImages) {
|
|
this.showNotification(`Image limit reached (${maxImages} max). Please delete some images first.`, 'error');
|
|
return;
|
|
}
|
|
|
|
for (const file of files) {
|
|
if (successCount + totalCurrentImages >= maxImages) {
|
|
this.showNotification(`Stopped at image limit (${maxImages} max). ${successCount} uploaded, ${files.length - successCount} skipped.`, 'warning');
|
|
break;
|
|
}
|
|
|
|
try {
|
|
const imageData = await this.processUploadedImage(file, activeTab);
|
|
if (imageData) {
|
|
// Try to save to localStorage with error handling
|
|
if (await this.saveImageWithQuotaCheck(imageData, activeTab)) {
|
|
successCount++;
|
|
} else {
|
|
failedCount++;
|
|
console.warn('Storage quota exceeded for image:', file.name);
|
|
break; // Stop processing if quota exceeded
|
|
}
|
|
}
|
|
} catch (error) {
|
|
console.warn('Failed to process image:', file.name, error);
|
|
failedCount++;
|
|
}
|
|
}
|
|
|
|
if (successCount > 0) {
|
|
this.showNotification(`${successCount} ${activeTab} image(s) uploaded successfully!${failedCount > 0 ? ` (${failedCount} failed due to storage limits)` : ''}`, successCount > failedCount ? 'success' : 'warning');
|
|
this.loadImageGallery(); // Refresh the gallery
|
|
this.cleanupInvalidImages(); // Clean up any images without dataUrl
|
|
} else {
|
|
this.showNotification(`Upload failed. ${failedCount > 0 ? 'Storage quota exceeded. Try deleting some images first.' : 'Please try again.'}`, 'error');
|
|
}
|
|
|
|
// Clear the input
|
|
event.target.value = '';
|
|
}
|
|
|
|
async saveImageWithQuotaCheck(imageData, activeTab) {
|
|
try {
|
|
let customImages = this.dataManager.get('customImages') || { task: [], consequence: [] };
|
|
|
|
// Ensure the structure exists
|
|
if (!customImages.task) customImages.task = [];
|
|
if (!customImages.consequence) customImages.consequence = [];
|
|
|
|
// Add to the appropriate category
|
|
customImages[activeTab].push(imageData);
|
|
|
|
// Try to save with quota checking
|
|
this.dataManager.set('customImages', customImages);
|
|
return true;
|
|
} catch (error) {
|
|
if (error.name === 'QuotaExceededError') {
|
|
console.error('Storage quota exceeded:', error);
|
|
return false;
|
|
} else {
|
|
console.error('Error saving image:', error);
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
getActiveImageTab() {
|
|
const taskTab = document.getElementById('task-images-tab');
|
|
return taskTab && taskTab.classList.contains('active') ? 'task' : 'consequence';
|
|
}
|
|
|
|
cleanupInvalidImages() {
|
|
const customImages = this.dataManager.get('customImages') || { task: [], consequence: [] };
|
|
let hasChanges = false;
|
|
|
|
if (Array.isArray(customImages)) {
|
|
// Old format - clean up and convert to new format
|
|
const validImages = customImages.filter(img => {
|
|
if (typeof img === 'string') {
|
|
return true; // Keep old base64 strings
|
|
} else {
|
|
return !!img.dataUrl; // Only keep metadata objects with dataUrl
|
|
}
|
|
});
|
|
|
|
if (validImages.length !== customImages.length) {
|
|
console.log(`Cleaned up ${customImages.length - validImages.length} invalid cached images`);
|
|
// Convert to new format - put all in task category for backward compatibility
|
|
this.dataManager.set('customImages', { task: validImages, consequence: [] });
|
|
hasChanges = true;
|
|
} else {
|
|
// Convert to new format without cleaning
|
|
this.dataManager.set('customImages', { task: customImages, consequence: [] });
|
|
hasChanges = true;
|
|
}
|
|
} else {
|
|
// New format - clean up both categories
|
|
const cleanCategory = (category) => {
|
|
const images = customImages[category] || [];
|
|
return images.filter(img => {
|
|
if (typeof img === 'string') {
|
|
return true; // Keep old base64 strings
|
|
} else {
|
|
return !!img.dataUrl; // Only keep metadata objects with dataUrl
|
|
}
|
|
});
|
|
};
|
|
|
|
const validTaskImages = cleanCategory('task');
|
|
const validConsequenceImages = cleanCategory('consequence');
|
|
|
|
const originalTaskCount = (customImages.task || []).length;
|
|
const originalConsequenceCount = (customImages.consequence || []).length;
|
|
|
|
if (validTaskImages.length !== originalTaskCount || validConsequenceImages.length !== originalConsequenceCount) {
|
|
const cleanedCount = (originalTaskCount + originalConsequenceCount) - (validTaskImages.length + validConsequenceImages.length);
|
|
console.log(`Cleaned up ${cleanedCount} invalid cached images`);
|
|
this.dataManager.set('customImages', { task: validTaskImages, consequence: validConsequenceImages });
|
|
hasChanges = true;
|
|
}
|
|
}
|
|
|
|
return hasChanges;
|
|
}
|
|
|
|
async scanForNewImages() {
|
|
this.showNotification('Scanning directories for new images...', 'info');
|
|
|
|
try {
|
|
// Get the current embedded manifest
|
|
let manifest = this.getEmbeddedManifest();
|
|
|
|
// Force a fresh scan by clearing any cached manifest
|
|
this.dataManager.set('cachedManifest', null);
|
|
|
|
// Perform comprehensive scan for new images
|
|
console.log('Performing user-requested scan for new images...');
|
|
manifest = await this.updateManifestWithNewImages(manifest);
|
|
|
|
// Update discovered images with new manifest
|
|
gameData.discoveredTaskImages = await this.verifyImagesFromManifest(manifest.tasks, 'images/tasks/');
|
|
gameData.discoveredConsequenceImages = await this.verifyImagesFromManifest(manifest.consequences, 'images/consequences/');
|
|
|
|
// Check how many new images were found
|
|
const totalImages = gameData.discoveredTaskImages.length + gameData.discoveredConsequenceImages.length;
|
|
|
|
this.showNotification(`Scan complete! Found ${totalImages} total images.`, 'success');
|
|
this.loadImageGallery(); // Refresh the gallery to show any new images
|
|
|
|
} catch (error) {
|
|
console.error('Directory scan failed:', error);
|
|
this.showNotification('Directory scan failed. Please try again.', 'error');
|
|
}
|
|
}
|
|
|
|
clearImageCache() {
|
|
// Clear the cached manifest to force a fresh scan next time
|
|
this.dataManager.set('cachedManifest', null);
|
|
this.showNotification('Image cache cleared. Next scan will be fresh.', 'success');
|
|
}
|
|
|
|
showStorageInfo() {
|
|
try {
|
|
const isDesktop = window.electronAPI !== undefined;
|
|
const customImages = this.dataManager.get('customImages') || { task: [], consequence: [] };
|
|
|
|
let totalCustomImages = 0;
|
|
let taskCount = 0;
|
|
let consequenceCount = 0;
|
|
|
|
if (Array.isArray(customImages)) {
|
|
// Old format
|
|
totalCustomImages = customImages.length;
|
|
taskCount = customImages.length;
|
|
consequenceCount = 0;
|
|
} else {
|
|
// New format - count both categories
|
|
taskCount = (customImages.task || []).length;
|
|
consequenceCount = (customImages.consequence || []).length;
|
|
totalCustomImages = taskCount + consequenceCount;
|
|
}
|
|
|
|
let message;
|
|
|
|
if (isDesktop) {
|
|
message = `
|
|
📊 Desktop Application Storage:
|
|
|
|
🖼️ Image Library:
|
|
├─ Task Images: ${taskCount}
|
|
└─ Consequence Images: ${consequenceCount}
|
|
Total Images: ${totalCustomImages}
|
|
|
|
💾 Storage Type: Native File System
|
|
📁 Storage Location: images/ folder in app directory
|
|
💡 Storage Capacity: Unlimited (depends on available disk space)
|
|
|
|
✅ Desktop Benefits:
|
|
• No browser storage limitations
|
|
• Full-resolution image support
|
|
• Native file system performance
|
|
• Automatic directory organization
|
|
• Cross-platform file compatibility
|
|
|
|
🎯 Image Management:
|
|
• Import images using native file dialogs
|
|
• Images automatically organized by category
|
|
• No compression or size restrictions
|
|
• Direct file system access for best performance
|
|
|
|
${totalCustomImages === 0 ? '📷 No custom images imported yet. Use the Import buttons to add your images!' : ''}
|
|
`.trim();
|
|
} else {
|
|
// Original web version info
|
|
const storageData = localStorage.getItem(this.dataManager.storageKey);
|
|
const storageSize = storageData ? storageData.length : 0;
|
|
const storageMB = (storageSize / (1024 * 1024)).toFixed(2);
|
|
const maxMB = 6;
|
|
const usagePercent = ((storageSize / (maxMB * 1024 * 1024)) * 100).toFixed(1);
|
|
|
|
message = `
|
|
📊 Browser Storage Information:
|
|
|
|
LocalStorage Used: ${storageMB} MB / ~${maxMB} MB (${usagePercent}%)
|
|
Total Custom Images: ${totalCustomImages}
|
|
|
|
${usagePercent > 85 ? '⚠️ Storage getting full - consider deleting some images' :
|
|
usagePercent > 70 ? '⚡ Storage usage is moderate' :
|
|
'✅ Storage usage is healthy'}
|
|
|
|
💡 Browser Limitations:
|
|
- Limited to ~6MB total storage
|
|
- Image compression required
|
|
- 50 image limit to prevent storage issues
|
|
|
|
💡 Consider upgrading to the desktop version for unlimited storage!
|
|
`.trim();
|
|
}
|
|
|
|
alert(message);
|
|
} catch (error) {
|
|
this.showNotification('Failed to get storage info', 'error');
|
|
}
|
|
}
|
|
|
|
async processUploadedImage(file, category = 'task') {
|
|
return new Promise(async (resolve, reject) => {
|
|
try {
|
|
// Validate file type
|
|
if (!file.type.startsWith('image/')) {
|
|
reject(new Error('Not an image file'));
|
|
return;
|
|
}
|
|
|
|
// Check file size before processing - 20MB limit
|
|
if (file.size > 20 * 1024 * 1024) {
|
|
reject(new Error('Image too large. Please use images smaller than 20MB.'));
|
|
return;
|
|
}
|
|
|
|
console.log(`Processing image: ${file.name}, size: ${(file.size / (1024 * 1024)).toFixed(2)}MB`);
|
|
|
|
// Generate unique filename for cached image
|
|
const timestamp = Date.now();
|
|
const fileExtension = file.name.split('.').pop().toLowerCase();
|
|
const cachedFileName = `cached_${timestamp}.${fileExtension}`;
|
|
const cachedImagePath = `images/cached/${cachedFileName}`;
|
|
|
|
// For web compatibility, we'll still compress but save as blob URL reference
|
|
const canvas = document.createElement('canvas');
|
|
const ctx = canvas.getContext('2d');
|
|
const img = new Image();
|
|
|
|
img.onload = async () => {
|
|
try {
|
|
// Calculate new dimensions (higher quality - 1600x1200)
|
|
let { width, height } = img;
|
|
const maxWidth = 1600;
|
|
const maxHeight = 1200;
|
|
|
|
console.log(`Original dimensions: ${width}x${height}`);
|
|
|
|
if (width > maxWidth || height > maxHeight) {
|
|
const ratio = Math.min(maxWidth / width, maxHeight / height);
|
|
width = Math.floor(width * ratio);
|
|
height = Math.floor(height * ratio);
|
|
console.log(`Resized dimensions: ${width}x${height}`);
|
|
}
|
|
|
|
// Set canvas size and draw high-quality image
|
|
canvas.width = width;
|
|
canvas.height = height;
|
|
ctx.drawImage(img, 0, 0, width, height);
|
|
|
|
// Convert to blob with high quality (0.95 for minimal compression)
|
|
canvas.toBlob(async (blob) => {
|
|
try {
|
|
// Save the blob as file reference
|
|
const imageData = await this.saveBlobToCache(blob, cachedFileName, file.name);
|
|
const compressedSize = (blob.size / (1024 * 1024)).toFixed(2);
|
|
console.log(`Cached image size: ${compressedSize}MB at ${cachedImagePath}`);
|
|
|
|
resolve(imageData);
|
|
} catch (error) {
|
|
console.error('Failed to save to cache:', error);
|
|
reject(error);
|
|
}
|
|
}, 'image/jpeg', 0.95);
|
|
|
|
} catch (error) {
|
|
console.error('Image processing error:', error);
|
|
reject(error);
|
|
}
|
|
};
|
|
|
|
img.onerror = () => reject(new Error('Failed to load image'));
|
|
|
|
// Create object URL for the image
|
|
const reader = new FileReader();
|
|
reader.onload = (e) => {
|
|
img.src = e.target.result;
|
|
};
|
|
reader.onerror = reject;
|
|
reader.readAsDataURL(file);
|
|
|
|
} catch (error) {
|
|
reject(error);
|
|
}
|
|
});
|
|
}
|
|
|
|
async saveBlobToCache(blob, cachedFileName, originalName) {
|
|
// Convert blob to base64 for reliable storage and display
|
|
return new Promise((resolve, reject) => {
|
|
const reader = new FileReader();
|
|
reader.onload = () => {
|
|
const imageMetadata = {
|
|
cachedFileName: cachedFileName,
|
|
originalName: originalName,
|
|
cachedPath: `images/cached/${cachedFileName}`,
|
|
dataUrl: reader.result, // Store the base64 data URL
|
|
size: blob.size,
|
|
type: blob.type,
|
|
timestamp: Date.now()
|
|
};
|
|
resolve(imageMetadata);
|
|
};
|
|
reader.onerror = reject;
|
|
reader.readAsDataURL(blob);
|
|
});
|
|
}
|
|
|
|
selectAllImages(tabType = 'task') {
|
|
const galleryId = tabType === 'task' ? 'task-images-gallery' : 'consequence-images-gallery';
|
|
const gallery = document.getElementById(galleryId);
|
|
const imageItems = gallery.querySelectorAll('.image-item');
|
|
imageItems.forEach(imageItem => {
|
|
imageItem.classList.add('selected');
|
|
});
|
|
}
|
|
|
|
deselectAllImages(tabType = 'task') {
|
|
const galleryId = tabType === 'task' ? 'task-images-gallery' : 'consequence-images-gallery';
|
|
const gallery = document.getElementById(galleryId);
|
|
const imageItems = gallery.querySelectorAll('.image-item');
|
|
imageItems.forEach(imageItem => {
|
|
imageItem.classList.remove('selected');
|
|
});
|
|
}
|
|
|
|
async deleteSelectedImages(tabType = 'task') {
|
|
const galleryId = tabType === 'task' ? 'task-images-gallery' : 'consequence-images-gallery';
|
|
const gallery = document.getElementById(galleryId);
|
|
const selectedItems = gallery.querySelectorAll('.image-item.selected');
|
|
|
|
if (selectedItems.length === 0) {
|
|
this.showNotification('No images selected for deletion.', 'warning');
|
|
return;
|
|
}
|
|
|
|
if (!confirm(`Are you sure you want to delete ${selectedItems.length} selected image(s)?`)) {
|
|
return;
|
|
}
|
|
|
|
let customImages = this.dataManager.get('customImages') || { task: [], consequence: [] };
|
|
let deletedCount = 0;
|
|
let fileDeletedCount = 0;
|
|
|
|
// Handle both old array format and new object format
|
|
if (Array.isArray(customImages)) {
|
|
// Old format - convert to new format first
|
|
customImages = { task: customImages, consequence: [] };
|
|
}
|
|
|
|
// If we're in desktop mode, also delete the actual files
|
|
const isDesktop = this.fileManager && this.fileManager.isElectron;
|
|
|
|
for (const item of selectedItems) {
|
|
const imagePath = item.dataset.imagePath;
|
|
|
|
// Determine which category this image belongs to
|
|
let imageCategory = 'task'; // default
|
|
if (imagePath.includes('consequences') || imagePath.includes('consequence')) {
|
|
imageCategory = 'consequence';
|
|
} else if (imagePath.includes('tasks') || imagePath.includes('task')) {
|
|
imageCategory = 'task';
|
|
}
|
|
|
|
// Delete the actual file if in desktop mode
|
|
if (isDesktop) {
|
|
try {
|
|
const fileDeleted = await this.fileManager.deleteImage(imagePath, imageCategory);
|
|
if (fileDeleted) {
|
|
fileDeletedCount++;
|
|
deletedCount++; // File manager handles storage cleanup
|
|
}
|
|
} catch (error) {
|
|
console.error('Error deleting file:', error);
|
|
}
|
|
} else {
|
|
// Web mode - only remove from storage
|
|
['task', 'consequence'].forEach(category => {
|
|
const initialLength = customImages[category].length;
|
|
customImages[category] = customImages[category].filter(img => {
|
|
if (typeof img === 'string') {
|
|
return img !== imagePath; // Old format: direct string comparison
|
|
} else {
|
|
// New format: compare against cachedPath or originalName
|
|
const imgPath = img.cachedPath || img.originalName;
|
|
return imgPath !== imagePath;
|
|
}
|
|
});
|
|
|
|
if (customImages[category].length < initialLength) {
|
|
deletedCount++;
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
// Update storage if not in desktop mode (desktop mode handles this in deleteImage)
|
|
if (!isDesktop && deletedCount > 0) {
|
|
this.dataManager.set('customImages', customImages);
|
|
}
|
|
|
|
// Clean up disabled image entries for deleted images
|
|
if (deletedCount > 0) {
|
|
let disabledImages = this.dataManager.get('disabledImages') || [];
|
|
const originalDisabledLength = disabledImages.length;
|
|
|
|
// Remove disabled entries for deleted images
|
|
selectedItems.forEach(item => {
|
|
const imagePath = item.dataset.imagePath;
|
|
disabledImages = disabledImages.filter(img => img !== imagePath);
|
|
});
|
|
|
|
if (disabledImages.length < originalDisabledLength) {
|
|
this.dataManager.set('disabledImages', disabledImages);
|
|
}
|
|
|
|
const message = isDesktop
|
|
? `${fileDeletedCount} image file(s) and references deleted successfully!`
|
|
: `${deletedCount} custom image(s) deleted successfully!`;
|
|
this.showNotification(message, 'success');
|
|
this.loadImageGallery(); // Refresh the gallery
|
|
} else {
|
|
this.showNotification('No custom images were selected. Default images cannot be deleted.', 'warning');
|
|
}
|
|
}
|
|
|
|
updateImageCount() {
|
|
const images = this.getAllImages();
|
|
const customImages = this.dataManager.get('customImages') || { task: [], consequence: [] };
|
|
const disabledImages = this.dataManager.get('disabledImages') || [];
|
|
|
|
// Handle both old and new customImages format
|
|
let customImagesArray = [];
|
|
if (Array.isArray(customImages)) {
|
|
// Old format - flat array
|
|
customImagesArray = customImages;
|
|
} else {
|
|
// New format - object with task and consequence arrays
|
|
customImagesArray = [...(customImages.task || []), ...(customImages.consequence || [])];
|
|
}
|
|
|
|
const allImages = [...images, ...customImagesArray];
|
|
|
|
const imageCount = document.querySelector('.image-count');
|
|
if (imageCount) {
|
|
imageCount.textContent = `${allImages.length} images (${allImages.length - disabledImages.length} enabled)`;
|
|
}
|
|
}
|
|
|
|
resetStats() {
|
|
if (this.dataManager.resetStats()) {
|
|
this.showNotification('📊 Statistics reset successfully!', 'success');
|
|
this.hideStats();
|
|
}
|
|
}
|
|
|
|
exportStatsOnly() {
|
|
try {
|
|
this.showButtonLoading('export-stats-btn');
|
|
|
|
// Small delay to show loading indicator
|
|
setTimeout(() => {
|
|
try {
|
|
this.dataManager.exportData(false);
|
|
this.showNotification('📊 Statistics exported successfully!', 'success');
|
|
} catch (error) {
|
|
this.showNotification('❌ ' + this.formatErrorMessage('stats-export', error), 'error');
|
|
} finally {
|
|
this.hideButtonLoading('export-stats-btn');
|
|
}
|
|
}, 300);
|
|
} catch (error) {
|
|
this.showNotification('❌ ' + this.formatErrorMessage('stats-export', error), 'error');
|
|
this.hideButtonLoading('export-stats-btn');
|
|
}
|
|
}
|
|
|
|
// Better error message formatting
|
|
formatErrorMessage(operation, error) {
|
|
const baseMessages = {
|
|
'export': 'Failed to export save data',
|
|
'import': 'Failed to import save data',
|
|
'stats-export': 'Failed to export statistics',
|
|
'auto-resume': 'Failed to restore previous session'
|
|
};
|
|
|
|
const specificErrors = {
|
|
'QuotaExceededError': 'Not enough storage space available',
|
|
'SecurityError': 'File access blocked by browser security',
|
|
'TypeError': 'Invalid file format detected',
|
|
'SyntaxError': 'Corrupted save file data'
|
|
};
|
|
|
|
const baseMessage = baseMessages[operation] || 'Operation failed';
|
|
const specificMessage = specificErrors[error.name] || error.message;
|
|
|
|
return `${baseMessage}: ${specificMessage}`;
|
|
}
|
|
|
|
// Audio Management Functions
|
|
showAudioManagement() {
|
|
// Reset listener flag to allow fresh attachment
|
|
this.audioManagementListenersAttached = false;
|
|
|
|
this.showScreen('audio-management-screen');
|
|
this.setupAudioManagementEventListeners();
|
|
|
|
// Wait for audio discovery to complete before loading gallery
|
|
if (!this.audioDiscoveryComplete) {
|
|
const galleries = document.querySelectorAll('.audio-gallery');
|
|
galleries.forEach(gallery => {
|
|
gallery.innerHTML = '<div class="loading">Discovering audio files...</div>';
|
|
});
|
|
|
|
// Wait and try again
|
|
setTimeout(() => {
|
|
if (this.audioDiscoveryComplete) {
|
|
this.loadAudioGallery();
|
|
} else {
|
|
galleries.forEach(gallery => {
|
|
gallery.innerHTML = '<div class="loading">Still discovering audio... Please wait</div>';
|
|
});
|
|
setTimeout(() => this.loadAudioGallery(), 1000);
|
|
}
|
|
}, 500);
|
|
} else {
|
|
this.loadAudioGallery();
|
|
}
|
|
}
|
|
|
|
switchAudioTab(tabType) {
|
|
// Update tab buttons
|
|
const backgroundTab = document.getElementById('background-audio-tab');
|
|
const ambientTab = document.getElementById('ambient-audio-tab');
|
|
const effectsTab = document.getElementById('effects-audio-tab');
|
|
const backgroundGallery = document.getElementById('background-audio-gallery');
|
|
const ambientGallery = document.getElementById('ambient-audio-gallery');
|
|
const effectsGallery = document.getElementById('effects-audio-gallery');
|
|
|
|
// Remove active class from all tabs and galleries
|
|
[backgroundTab, ambientTab, effectsTab].forEach(tab => tab && tab.classList.remove('active'));
|
|
[backgroundGallery, ambientGallery, effectsGallery].forEach(gallery => gallery && gallery.classList.remove('active'));
|
|
|
|
// Add active class to selected tab and gallery
|
|
if (tabType === 'background') {
|
|
backgroundTab && backgroundTab.classList.add('active');
|
|
backgroundGallery && backgroundGallery.classList.add('active');
|
|
} else if (tabType === 'ambient') {
|
|
ambientTab && ambientTab.classList.add('active');
|
|
ambientGallery && ambientGallery.classList.add('active');
|
|
} else if (tabType === 'effects') {
|
|
effectsTab && effectsTab.classList.add('active');
|
|
effectsGallery && effectsGallery.classList.add('active');
|
|
}
|
|
|
|
// Update gallery controls to work with current tab
|
|
this.updateAudioGalleryControls(tabType);
|
|
}
|
|
|
|
setupAudioManagementEventListeners() {
|
|
// Check if we already have listeners attached to prevent duplicates
|
|
if (this.audioManagementListenersAttached) {
|
|
return;
|
|
}
|
|
|
|
// Back button
|
|
const backBtn = document.getElementById('back-to-start-from-audio-btn');
|
|
if (backBtn) {
|
|
backBtn.onclick = () => this.showScreen('start-screen');
|
|
}
|
|
|
|
// Desktop import buttons
|
|
const importBackgroundBtn = document.getElementById('import-background-audio-btn');
|
|
if (importBackgroundBtn) {
|
|
importBackgroundBtn.onclick = async () => {
|
|
if (this.fileManager) {
|
|
await this.fileManager.selectAndImportAudio('background');
|
|
this.loadAudioGallery(); // Refresh the gallery to show new audio
|
|
} else {
|
|
this.showNotification('Desktop file manager not available', 'warning');
|
|
}
|
|
};
|
|
}
|
|
|
|
const importAmbientBtn = document.getElementById('import-ambient-audio-btn');
|
|
if (importAmbientBtn) {
|
|
importAmbientBtn.onclick = async () => {
|
|
if (this.fileManager) {
|
|
await this.fileManager.selectAndImportAudio('ambient');
|
|
this.loadAudioGallery(); // Refresh the gallery to show new audio
|
|
} else {
|
|
this.showNotification('Desktop file manager not available', 'warning');
|
|
}
|
|
};
|
|
}
|
|
|
|
const importEffectsBtn = document.getElementById('import-effects-audio-btn');
|
|
if (importEffectsBtn) {
|
|
importEffectsBtn.onclick = async () => {
|
|
if (this.fileManager) {
|
|
await this.fileManager.selectAndImportAudio('effects');
|
|
this.loadAudioGallery(); // Refresh the gallery to show new audio
|
|
} else {
|
|
this.showNotification('Desktop file manager not available', 'warning');
|
|
}
|
|
};
|
|
}
|
|
|
|
// Audio storage info button
|
|
const audioStorageInfoBtn = document.getElementById('audio-storage-info-btn');
|
|
if (audioStorageInfoBtn) {
|
|
audioStorageInfoBtn.onclick = () => this.showAudioStorageInfo();
|
|
}
|
|
|
|
// Tab buttons
|
|
const backgroundAudioTab = document.getElementById('background-audio-tab');
|
|
if (backgroundAudioTab) {
|
|
backgroundAudioTab.onclick = () => this.switchAudioTab('background');
|
|
}
|
|
|
|
const ambientAudioTab = document.getElementById('ambient-audio-tab');
|
|
if (ambientAudioTab) {
|
|
ambientAudioTab.onclick = () => this.switchAudioTab('ambient');
|
|
}
|
|
|
|
const effectsAudioTab = document.getElementById('effects-audio-tab');
|
|
if (effectsAudioTab) {
|
|
effectsAudioTab.onclick = () => this.switchAudioTab('effects');
|
|
}
|
|
|
|
// Gallery control buttons - assign onclick directly to avoid expensive DOM operations
|
|
const selectAllAudioBtn = document.getElementById('select-all-audio-btn');
|
|
if (selectAllAudioBtn) {
|
|
selectAllAudioBtn.onclick = () => this.selectAllAudio();
|
|
}
|
|
|
|
const deselectAllAudioBtn = document.getElementById('deselect-all-audio-btn');
|
|
if (deselectAllAudioBtn) {
|
|
deselectAllAudioBtn.onclick = () => this.deselectAllAudio();
|
|
}
|
|
|
|
const deleteSelectedAudioBtn = document.getElementById('delete-selected-audio-btn');
|
|
if (deleteSelectedAudioBtn) {
|
|
deleteSelectedAudioBtn.onclick = () => this.deleteSelectedAudio();
|
|
}
|
|
|
|
const previewSelectedAudioBtn = document.getElementById('preview-selected-audio-btn');
|
|
if (previewSelectedAudioBtn) {
|
|
previewSelectedAudioBtn.onclick = () => this.previewSelectedAudio();
|
|
}
|
|
|
|
// Close preview button
|
|
const closePreviewBtn = document.getElementById('close-preview-btn');
|
|
if (closePreviewBtn) {
|
|
closePreviewBtn.onclick = () => this.closeAudioPreview();
|
|
}
|
|
|
|
// Mark listeners as attached
|
|
this.audioManagementListenersAttached = true;
|
|
}
|
|
|
|
loadAudioGallery() {
|
|
const backgroundGallery = document.getElementById('background-audio-gallery');
|
|
const ambientGallery = document.getElementById('ambient-audio-gallery');
|
|
const effectsGallery = document.getElementById('effects-audio-gallery');
|
|
|
|
if (!backgroundGallery || !ambientGallery || !effectsGallery) {
|
|
console.error('Audio gallery elements not found');
|
|
return;
|
|
}
|
|
|
|
// Get custom audio from storage
|
|
const customAudio = this.dataManager.get('customAudio') || { background: [], ambient: [], effects: [] };
|
|
|
|
// Load each category
|
|
this.loadAudioCategory('background', backgroundGallery, customAudio.background);
|
|
this.loadAudioCategory('ambient', ambientGallery, customAudio.ambient);
|
|
this.loadAudioCategory('effects', effectsGallery, customAudio.effects);
|
|
|
|
// Update audio count
|
|
this.updateAudioCount();
|
|
|
|
// Setup initial gallery controls for the active tab
|
|
this.updateAudioGalleryControls('background');
|
|
|
|
// Refresh the music manager to include new custom tracks
|
|
if (this.musicManager) {
|
|
this.musicManager.refreshCustomTracks();
|
|
}
|
|
}
|
|
|
|
loadAudioCategory(category, gallery, audioFiles) {
|
|
if (!audioFiles || audioFiles.length === 0) {
|
|
gallery.innerHTML = `<div class="no-audio">No ${category} audio files found. Use the import button to add some!</div>`;
|
|
return;
|
|
}
|
|
|
|
const audioItems = audioFiles.map(audio => {
|
|
return `
|
|
<div class="audio-item" data-category="${category}" data-filename="${audio.name}" onclick="game.toggleAudioSelection(this)">
|
|
<div class="audio-icon" data-category="${category}"></div>
|
|
<div class="audio-title">${audio.title}</div>
|
|
<div class="audio-filename">${audio.name}</div>
|
|
<div class="audio-controls">
|
|
<button class="audio-preview-btn" onclick="event.stopPropagation(); game.previewAudio('${audio.path}', '${audio.title}')">🎧 Preview</button>
|
|
<label class="audio-status" onclick="event.stopPropagation()">
|
|
<input type="checkbox" class="audio-checkbox" ${audio.enabled !== false ? 'checked' : ''}
|
|
onchange="game.toggleAudioEnabled('${category}', '${audio.name}', this.checked)">
|
|
<span class="${audio.enabled !== false ? 'audio-enabled' : 'audio-disabled'}">
|
|
${audio.enabled !== false ? 'Enabled' : 'Disabled'}
|
|
</span>
|
|
</label>
|
|
</div>
|
|
</div>
|
|
`;
|
|
}).join('');
|
|
|
|
gallery.innerHTML = audioItems;
|
|
}
|
|
|
|
updateAudioGalleryControls(activeCategory = 'background') {
|
|
// This will be called when the active tab changes to update controls
|
|
// for the current category
|
|
console.log(`Audio gallery controls updated for ${activeCategory} category`);
|
|
}
|
|
|
|
updateAudioCount() {
|
|
const customAudio = this.dataManager.get('customAudio') || { background: [], ambient: [], effects: [] };
|
|
const backgroundCount = customAudio.background ? customAudio.background.length : 0;
|
|
const ambientCount = customAudio.ambient ? customAudio.ambient.length : 0;
|
|
const effectsCount = customAudio.effects ? customAudio.effects.length : 0;
|
|
const total = backgroundCount + ambientCount + effectsCount;
|
|
|
|
const audioCountElement = document.querySelector('.audio-count');
|
|
if (audioCountElement) {
|
|
audioCountElement.textContent = `${total} total audio files (${backgroundCount} background, ${ambientCount} ambient, ${effectsCount} effects)`;
|
|
}
|
|
}
|
|
|
|
selectAllAudio() {
|
|
const activeGallery = document.querySelector('.audio-gallery.active');
|
|
if (activeGallery) {
|
|
const audioItems = activeGallery.querySelectorAll('.audio-item');
|
|
audioItems.forEach(item => item.classList.add('selected'));
|
|
}
|
|
}
|
|
|
|
deselectAllAudio() {
|
|
const activeGallery = document.querySelector('.audio-gallery.active');
|
|
if (activeGallery) {
|
|
const audioItems = activeGallery.querySelectorAll('.audio-item');
|
|
audioItems.forEach(item => item.classList.remove('selected'));
|
|
}
|
|
}
|
|
|
|
async deleteSelectedAudio() {
|
|
const activeGallery = document.querySelector('.audio-gallery.active');
|
|
if (!activeGallery) return;
|
|
|
|
const selectedItems = activeGallery.querySelectorAll('.audio-item.selected');
|
|
if (selectedItems.length === 0) {
|
|
this.showNotification('No audio files selected', 'warning');
|
|
return;
|
|
}
|
|
|
|
if (!confirm(`Are you sure you want to delete ${selectedItems.length} selected audio file(s)?`)) {
|
|
return;
|
|
}
|
|
|
|
let deletedCount = 0;
|
|
const isDesktop = window.electronAPI !== undefined;
|
|
|
|
// Use Promise.all to handle async operations properly
|
|
const deletionPromises = Array.from(selectedItems).map(async (item) => {
|
|
const category = item.dataset.category;
|
|
const filename = item.dataset.filename;
|
|
|
|
if (isDesktop && this.fileManager) {
|
|
// Desktop mode - delete actual file
|
|
// Need to find the full audio file path from storage
|
|
const customAudio = this.dataManager.get('customAudio') || { background: [], ambient: [], effects: [] };
|
|
console.log(`Looking for audio file in category ${category}:`, customAudio[category]);
|
|
console.log(`Searching for filename: ${filename}`);
|
|
|
|
const audioFile = customAudio[category].find(audio =>
|
|
(typeof audio === 'string' && audio.includes(filename)) ||
|
|
(typeof audio === 'object' && (audio.name === filename))
|
|
);
|
|
|
|
console.log(`Found audio file:`, audioFile);
|
|
|
|
if (audioFile) {
|
|
const audioPath = typeof audioFile === 'string' ? audioFile : audioFile.path;
|
|
console.log(`Attempting to delete audio file: ${audioPath}`);
|
|
const success = await this.fileManager.deleteAudio(audioPath, category);
|
|
if (success) {
|
|
console.log(`Successfully deleted: ${filename}`);
|
|
return true;
|
|
} else {
|
|
console.error(`Failed to delete audio file: ${filename}`);
|
|
return false;
|
|
}
|
|
} else {
|
|
console.error(`Audio file not found in storage: ${filename}`);
|
|
return false;
|
|
}
|
|
} else {
|
|
// Web mode - remove from storage only
|
|
this.removeAudioFromStorage(category, filename);
|
|
return true;
|
|
}
|
|
});
|
|
|
|
// Wait for all deletions to complete
|
|
try {
|
|
const results = await Promise.all(deletionPromises);
|
|
deletedCount = results.filter(result => result === true).length;
|
|
|
|
if (deletedCount > 0) {
|
|
const modeText = isDesktop ? 'file(s) deleted from disk' : 'reference(s) removed from storage';
|
|
this.showNotification(`${deletedCount} audio ${modeText}`, 'success');
|
|
this.loadAudioGallery(); // Refresh the gallery
|
|
} else {
|
|
this.showNotification('No audio files were deleted', 'warning');
|
|
}
|
|
} catch (error) {
|
|
console.error('Error during audio deletion:', error);
|
|
this.showNotification('Error occurred during deletion', 'error');
|
|
}
|
|
}
|
|
|
|
removeAudioFromStorage(category, filename) {
|
|
const customAudio = this.dataManager.get('customAudio') || { background: [], ambient: [], effects: [] };
|
|
if (customAudio[category]) {
|
|
customAudio[category] = customAudio[category].filter(audio => {
|
|
if (typeof audio === 'string') {
|
|
return !audio.includes(filename);
|
|
} else if (typeof audio === 'object') {
|
|
return audio.name !== filename;
|
|
}
|
|
return true;
|
|
});
|
|
this.dataManager.set('customAudio', customAudio);
|
|
}
|
|
}
|
|
|
|
previewSelectedAudio() {
|
|
const activeGallery = document.querySelector('.audio-gallery.active');
|
|
if (!activeGallery) return;
|
|
|
|
const selectedItems = activeGallery.querySelectorAll('.audio-item.selected');
|
|
if (selectedItems.length === 0) {
|
|
this.showNotification('No audio file selected for preview', 'warning');
|
|
return;
|
|
}
|
|
|
|
if (selectedItems.length > 1) {
|
|
this.showNotification('Please select only one audio file for preview', 'warning');
|
|
return;
|
|
}
|
|
|
|
const selectedItem = selectedItems[0];
|
|
const category = selectedItem.dataset.category;
|
|
const filename = selectedItem.dataset.filename;
|
|
|
|
// Find the audio file data
|
|
const customAudio = this.dataManager.get('customAudio') || { background: [], ambient: [], effects: [] };
|
|
const audioFile = customAudio[category].find(audio => audio.filename === filename);
|
|
|
|
if (audioFile) {
|
|
this.previewAudio(audioFile.path, audioFile.title);
|
|
}
|
|
}
|
|
|
|
previewAudio(audioPath, title) {
|
|
const previewSection = document.getElementById('audio-preview-section');
|
|
const audioPlayer = document.getElementById('audio-preview-player');
|
|
const previewFileName = document.getElementById('preview-file-name');
|
|
|
|
if (previewSection && audioPlayer && previewFileName) {
|
|
previewSection.style.display = 'block';
|
|
audioPlayer.src = audioPath;
|
|
previewFileName.textContent = title || 'Unknown';
|
|
|
|
// Scroll to preview section
|
|
previewSection.scrollIntoView({ behavior: 'smooth', block: 'nearest' });
|
|
}
|
|
}
|
|
|
|
closeAudioPreview() {
|
|
const previewSection = document.getElementById('audio-preview-section');
|
|
const audioPlayer = document.getElementById('audio-preview-player');
|
|
|
|
if (previewSection && audioPlayer) {
|
|
previewSection.style.display = 'none';
|
|
audioPlayer.pause();
|
|
audioPlayer.src = '';
|
|
}
|
|
}
|
|
|
|
toggleAudioEnabled(category, filename, enabled) {
|
|
const customAudio = this.dataManager.get('customAudio') || { background: [], ambient: [], effects: [] };
|
|
|
|
if (customAudio[category]) {
|
|
const audioFile = customAudio[category].find(audio => audio.name === filename);
|
|
if (audioFile) {
|
|
audioFile.enabled = enabled;
|
|
this.dataManager.set('customAudio', customAudio);
|
|
|
|
// Update the visual status
|
|
this.loadAudioGallery();
|
|
this.showNotification(`Audio ${enabled ? 'enabled' : 'disabled'}`, 'success');
|
|
}
|
|
}
|
|
}
|
|
|
|
toggleAudioSelection(audioItem) {
|
|
audioItem.classList.toggle('selected');
|
|
}
|
|
|
|
showAudioStorageInfo() {
|
|
try {
|
|
const isDesktop = window.electronAPI !== undefined;
|
|
const customAudio = this.dataManager.get('customAudio') || { background: [], ambient: [], effects: [] };
|
|
|
|
const backgroundCount = customAudio.background ? customAudio.background.length : 0;
|
|
const ambientCount = customAudio.ambient ? customAudio.ambient.length : 0;
|
|
const effectsCount = customAudio.effects ? customAudio.effects.length : 0;
|
|
const totalCustomAudio = backgroundCount + ambientCount + effectsCount;
|
|
|
|
let message = `📊 Audio Storage Information\n\n`;
|
|
|
|
if (isDesktop) {
|
|
message += `🖥️ Desktop Mode - Unlimited Storage\n`;
|
|
message += `Files stored in native file system\n\n`;
|
|
message += `📁 Audio Categories:\n`;
|
|
message += `🎵 Background Music: ${backgroundCount} files\n`;
|
|
message += `🌿 Ambient Sounds: ${ambientCount} files\n`;
|
|
message += `🔊 Sound Effects: ${effectsCount} files\n`;
|
|
message += `📊 Total Custom Audio: ${totalCustomAudio} files\n\n`;
|
|
message += `💾 Storage: Uses native file system\n`;
|
|
message += `📂 Location: audio/ folder in app directory\n`;
|
|
message += `🔄 Auto-scanned on startup`;
|
|
} else {
|
|
message += `🌐 Web Mode - Browser Storage\n`;
|
|
message += `Limited by browser storage quotas\n\n`;
|
|
message += `📁 Audio References:\n`;
|
|
message += `🎵 Background Music: ${backgroundCount} files\n`;
|
|
message += `🌿 Ambient Sounds: ${ambientCount} files\n`;
|
|
message += `🔊 Sound Effects: ${effectsCount} files\n`;
|
|
message += `📊 Total Custom Audio: ${totalCustomAudio} files\n\n`;
|
|
message += `💾 Storage: Browser localStorage\n`;
|
|
message += `⚠️ Subject to browser storage limits`;
|
|
}
|
|
|
|
alert(message);
|
|
} catch (error) {
|
|
console.error('Error showing audio storage info:', error);
|
|
alert('Error retrieving audio storage information.');
|
|
}
|
|
}
|
|
|
|
showNotification(message, type = 'info') {
|
|
// Create notification element if it doesn't exist
|
|
let notification = document.getElementById('notification');
|
|
if (!notification) {
|
|
notification = document.createElement('div');
|
|
notification.id = 'notification';
|
|
notification.className = 'notification';
|
|
document.body.appendChild(notification);
|
|
}
|
|
|
|
notification.textContent = message;
|
|
notification.className = `notification ${type} show`;
|
|
|
|
// Hide after 3 seconds
|
|
setTimeout(() => {
|
|
notification.classList.remove('show');
|
|
}, 3000);
|
|
}
|
|
|
|
startGame() {
|
|
console.log('startGame() called');
|
|
if (!this.imageDiscoveryComplete) {
|
|
console.log('Image discovery not complete, retrying in 500ms...');
|
|
// Wait a bit longer for image discovery
|
|
setTimeout(() => this.startGame(), 500);
|
|
return;
|
|
}
|
|
|
|
// Check if we have any images available
|
|
const totalImages = gameData.discoveredTaskImages.length + gameData.discoveredConsequenceImages.length;
|
|
const customImages = this.dataManager.get('customImages') || { task: [], consequence: [] };
|
|
|
|
// Handle both old array format and new object format for backward compatibility
|
|
let customImageCount = 0;
|
|
if (Array.isArray(customImages)) {
|
|
customImageCount = customImages.length;
|
|
} else {
|
|
customImageCount = (customImages.task || []).length + (customImages.consequence || []).length;
|
|
}
|
|
|
|
if (totalImages === 0 && customImageCount === 0) {
|
|
// No images available - guide user to add images
|
|
this.showNotification('No images found! Please upload images or scan directories first.', 'error', 5000);
|
|
this.showImageManagement(); // Use the proper method that sets up event listeners
|
|
return;
|
|
}
|
|
|
|
console.log('Starting game initialization...');
|
|
this.gameState.isRunning = true;
|
|
this.gameState.isPaused = false;
|
|
this.gameState.startTime = Date.now();
|
|
this.gameState.totalPausedTime = 0;
|
|
this.gameState.score = 0;
|
|
this.gameState.lastSkippedTask = null;
|
|
|
|
console.log('Recording game start in data manager...');
|
|
// Record game start in data manager
|
|
this.dataManager.recordGameStart();
|
|
|
|
console.log('Starting timer...');
|
|
this.startTimer();
|
|
console.log('Loading next task...');
|
|
this.loadNextTask();
|
|
console.log('Showing game screen...');
|
|
this.showScreen('game-screen');
|
|
console.log('Starting flash message system...');
|
|
this.flashMessageManager.start();
|
|
this.flashMessageManager.triggerEventMessage('gameStart');
|
|
console.log('Updating stats...');
|
|
this.updateStats();
|
|
console.log('startGame() completed successfully');
|
|
}
|
|
|
|
loadNextTask() {
|
|
if (this.gameState.isConsequenceTask) {
|
|
// Load a consequence task
|
|
this.loadConsequenceTask();
|
|
} else {
|
|
// Load a main task
|
|
this.loadMainTask();
|
|
}
|
|
|
|
this.displayCurrentTask();
|
|
}
|
|
|
|
loadMainTask() {
|
|
const availableTasks = gameData.mainTasks.filter(
|
|
task => !this.gameState.usedMainTasks.includes(task.id)
|
|
);
|
|
|
|
if (availableTasks.length === 0) {
|
|
// All main tasks completed
|
|
if (this.gameState.gameMode === 'complete-all') {
|
|
// Only end game in complete-all mode
|
|
this.endGame('complete-all');
|
|
return;
|
|
} else {
|
|
// In timed and score-target modes, reset used tasks and continue
|
|
console.log(`All tasks completed in ${this.gameState.gameMode} mode, cycling through tasks again`);
|
|
this.gameState.usedMainTasks = [];
|
|
// Recursively call loadMainTask to select from reset pool
|
|
this.loadMainTask();
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Select random task and random image from task pool
|
|
const randomIndex = Math.floor(Math.random() * availableTasks.length);
|
|
this.gameState.currentTask = {
|
|
...availableTasks[randomIndex],
|
|
image: this.getRandomImage(false) // false = main task image
|
|
};
|
|
this.gameState.isConsequenceTask = false;
|
|
}
|
|
|
|
loadConsequenceTask() {
|
|
const availableTasks = gameData.consequenceTasks.filter(
|
|
task => !this.gameState.usedConsequenceTasks.includes(task.id)
|
|
);
|
|
|
|
if (availableTasks.length === 0) {
|
|
// Reset consequence tasks if all used
|
|
this.gameState.usedConsequenceTasks = [];
|
|
}
|
|
|
|
const tasksToChooseFrom = availableTasks.length > 0 ? availableTasks : gameData.consequenceTasks;
|
|
const randomIndex = Math.floor(Math.random() * tasksToChooseFrom.length);
|
|
this.gameState.currentTask = {
|
|
...tasksToChooseFrom[randomIndex],
|
|
image: this.getRandomImage(true) // true = consequence task image
|
|
};
|
|
this.gameState.isConsequenceTask = true;
|
|
}
|
|
|
|
getRandomImage(isConsequence = false) {
|
|
let imagePool;
|
|
let imageType;
|
|
let usedImagesArray;
|
|
|
|
if (isConsequence) {
|
|
imagePool = gameData.discoveredConsequenceImages;
|
|
imageType = 'consequence';
|
|
usedImagesArray = this.gameState.usedConsequenceImages;
|
|
} else {
|
|
imagePool = gameData.discoveredTaskImages;
|
|
imageType = 'task';
|
|
usedImagesArray = this.gameState.usedTaskImages;
|
|
}
|
|
|
|
// Add custom images to the pool
|
|
const customImages = this.dataManager.get('customImages') || { task: [], consequence: [] };
|
|
|
|
// Handle both old array format and new object format
|
|
let customImagePool = [];
|
|
if (Array.isArray(customImages)) {
|
|
// Old format - treat all as task images
|
|
customImagePool = isConsequence ? [] : customImages;
|
|
} else {
|
|
// New format - get appropriate category
|
|
customImagePool = customImages[imageType] || [];
|
|
}
|
|
|
|
imagePool = [...imagePool, ...customImagePool];
|
|
|
|
// Filter out disabled images - need to handle both formats
|
|
const disabledImages = this.dataManager.get('disabledImages') || [];
|
|
imagePool = imagePool.filter(img => {
|
|
// Handle both string paths and metadata objects
|
|
const imagePath = typeof img === 'string' ? img : (img.cachedPath || img.originalName);
|
|
return !disabledImages.includes(imagePath);
|
|
});
|
|
|
|
if (imagePool.length === 0) {
|
|
console.log(`No enabled ${imageType} images found, using placeholder`);
|
|
return this.createPlaceholderImage(isConsequence ? 'Consequence Image' : 'Task Image');
|
|
}
|
|
|
|
// Get image identifiers for tracking
|
|
const getImageId = (img) => {
|
|
return typeof img === 'string' ? img : (img.cachedPath || img.originalName);
|
|
};
|
|
|
|
// Filter out images that have already been used
|
|
let availableImages = imagePool.filter(img => {
|
|
const imageId = getImageId(img);
|
|
return !usedImagesArray.includes(imageId);
|
|
});
|
|
|
|
// If all images have been used, reset the used array and use all images again
|
|
if (availableImages.length === 0) {
|
|
console.log(`All ${imageType} images have been shown, resetting for repeat cycle`);
|
|
usedImagesArray.length = 0; // Clear the used images array
|
|
availableImages = imagePool; // Use all available images again
|
|
}
|
|
|
|
// Select a random image from available images
|
|
const randomIndex = Math.floor(Math.random() * availableImages.length);
|
|
const selectedImage = availableImages[randomIndex];
|
|
const selectedImageId = getImageId(selectedImage);
|
|
|
|
// Mark this image as used
|
|
usedImagesArray.push(selectedImageId);
|
|
|
|
// Convert to displayable format
|
|
const displayImage = this.getImageSrc(selectedImage);
|
|
console.log(`Selected ${imageType} image: ${typeof selectedImage === 'string' ? selectedImage : selectedImage.originalName} (${usedImagesArray.length}/${imagePool.length} used)`);
|
|
return displayImage;
|
|
}
|
|
|
|
getImageSrc(imageData) {
|
|
// Handle both old base64/path format and new cached metadata format
|
|
if (typeof imageData === 'string') {
|
|
return imageData; // Old format: base64 or file path
|
|
} else {
|
|
return imageData.dataUrl || imageData.cachedPath; // New format: use data URL or cached path
|
|
}
|
|
}
|
|
|
|
displayCurrentTask() {
|
|
const taskText = document.getElementById('task-text');
|
|
const taskImage = document.getElementById('task-image');
|
|
const taskTypeIndicator = document.getElementById('task-type-indicator');
|
|
const taskDifficulty = document.getElementById('task-difficulty');
|
|
const taskPoints = document.getElementById('task-points');
|
|
const mercySkipBtn = document.getElementById('mercy-skip-btn');
|
|
const skipBtn = document.getElementById('skip-btn');
|
|
|
|
taskText.textContent = this.gameState.currentTask.text;
|
|
|
|
// Set image with error handling
|
|
taskImage.src = this.gameState.currentTask.image;
|
|
taskImage.onerror = () => {
|
|
console.log('Image failed to load:', this.gameState.currentTask.image);
|
|
taskImage.src = this.createPlaceholderImage();
|
|
};
|
|
|
|
// Update task type indicator and button visibility
|
|
if (this.gameState.isConsequenceTask) {
|
|
taskTypeIndicator.textContent = 'CONSEQUENCE TASK';
|
|
taskTypeIndicator.classList.add('consequence');
|
|
// Hide difficulty/points for consequence tasks
|
|
taskDifficulty.style.display = 'none';
|
|
taskPoints.style.display = 'none';
|
|
// Hide regular skip button for consequence tasks
|
|
skipBtn.style.display = 'none';
|
|
|
|
// Show mercy skip button for consequence tasks
|
|
const originalTask = this.findOriginalSkippedTask();
|
|
if (originalTask) {
|
|
const mercyCost = this.getPointsForDifficulty(originalTask.difficulty || 'Medium') * 2;
|
|
const canAfford = this.gameState.score >= mercyCost;
|
|
|
|
if (canAfford) {
|
|
mercySkipBtn.style.display = 'block';
|
|
document.getElementById('mercy-skip-cost').textContent = `-${mercyCost} pts`;
|
|
mercySkipBtn.disabled = false;
|
|
} else {
|
|
mercySkipBtn.style.display = 'block';
|
|
document.getElementById('mercy-skip-cost').textContent = `-${mercyCost} pts (Not enough!)`;
|
|
mercySkipBtn.disabled = true;
|
|
}
|
|
} else {
|
|
mercySkipBtn.style.display = 'none';
|
|
}
|
|
} else {
|
|
taskTypeIndicator.textContent = 'MAIN TASK';
|
|
taskTypeIndicator.classList.remove('consequence');
|
|
|
|
// Show regular skip button for main tasks
|
|
skipBtn.style.display = 'block';
|
|
// Hide mercy skip button for main tasks
|
|
mercySkipBtn.style.display = 'none';
|
|
|
|
// Show difficulty and points for main tasks
|
|
const difficulty = this.gameState.currentTask.difficulty || 'Medium';
|
|
const points = this.getPointsForDifficulty(difficulty);
|
|
const difficultyEmoji = this.getDifficultyEmoji(difficulty);
|
|
|
|
taskDifficulty.textContent = `${difficultyEmoji} ${difficulty}`;
|
|
taskPoints.textContent = `${points} ${points === 1 ? 'point' : 'points'}`;
|
|
taskDifficulty.style.display = 'block';
|
|
taskPoints.style.display = 'inline-block';
|
|
}
|
|
}
|
|
|
|
findOriginalSkippedTask() {
|
|
return this.gameState.lastSkippedTask;
|
|
}
|
|
|
|
mercySkip() {
|
|
if (!this.gameState.isRunning || this.gameState.isPaused) return;
|
|
if (!this.gameState.isConsequenceTask) return;
|
|
|
|
const originalTask = this.findOriginalSkippedTask();
|
|
if (!originalTask) return;
|
|
|
|
const mercyCost = this.getPointsForDifficulty(originalTask.difficulty || 'Medium') * 2;
|
|
|
|
if (this.gameState.score < mercyCost) {
|
|
alert(`Not enough points! You need ${mercyCost} points but only have ${this.gameState.score}.`);
|
|
return;
|
|
}
|
|
|
|
// Confirm the mercy skip
|
|
const confirmed = confirm(
|
|
`Use Mercy Skip?\n\n` +
|
|
`This will cost you ${mercyCost} points from your score.\n` +
|
|
`Your score will go from ${this.gameState.score} to ${this.gameState.score - mercyCost}.\n\n` +
|
|
`Are you sure you want to skip this consequence task?`
|
|
);
|
|
|
|
if (!confirmed) return;
|
|
|
|
// Deduct points and skip the consequence task
|
|
this.gameState.score -= mercyCost;
|
|
|
|
// Clear the last skipped task and load next main task
|
|
this.gameState.lastSkippedTask = null;
|
|
this.gameState.isConsequenceTask = false;
|
|
this.loadNextTask();
|
|
this.updateStats();
|
|
|
|
// Show feedback
|
|
alert(`Mercy Skip used! ${mercyCost} points deducted from your score.`);
|
|
}
|
|
|
|
getPointsForDifficulty(difficulty) {
|
|
switch(difficulty) {
|
|
case 'Easy': return 1;
|
|
case 'Medium': return 3;
|
|
case 'Hard': return 5;
|
|
default: return 3;
|
|
}
|
|
}
|
|
|
|
getDifficultyEmoji(difficulty) {
|
|
switch(difficulty) {
|
|
case 'Easy': return '🟢';
|
|
case 'Medium': return '🟡';
|
|
case 'Hard': return '🔴';
|
|
default: return '🟡';
|
|
}
|
|
}
|
|
|
|
checkStreakBonus() {
|
|
// Award streak bonus every 10 completed tasks
|
|
const streakMilestone = Math.floor(this.gameState.currentStreak / 10);
|
|
|
|
if (streakMilestone > this.gameState.lastStreakMilestone) {
|
|
this.gameState.lastStreakMilestone = streakMilestone;
|
|
return 10; // 10 bonus points per streak milestone
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
showStreakBonusNotification(streak, bonusPoints) {
|
|
// Create streak bonus notification
|
|
const notification = document.createElement('div');
|
|
notification.className = 'streak-bonus-notification';
|
|
notification.innerHTML = `
|
|
<div class="streak-bonus-content">
|
|
<div class="streak-icon">🔥</div>
|
|
<div class="streak-text">
|
|
<div class="streak-title">${streak} Task Streak!</div>
|
|
<div class="streak-bonus">+${bonusPoints} Bonus Points</div>
|
|
</div>
|
|
</div>
|
|
`;
|
|
|
|
document.body.appendChild(notification);
|
|
|
|
// Animate in
|
|
setTimeout(() => notification.classList.add('show'), 10);
|
|
|
|
// Remove after 3 seconds
|
|
setTimeout(() => {
|
|
notification.classList.remove('show');
|
|
setTimeout(() => document.body.removeChild(notification), 300);
|
|
}, 3000);
|
|
}
|
|
|
|
completeTask() {
|
|
if (!this.gameState.isRunning || this.gameState.isPaused) return;
|
|
|
|
// Mark task as used and award points
|
|
if (this.gameState.isConsequenceTask) {
|
|
this.gameState.usedConsequenceTasks.push(this.gameState.currentTask.id);
|
|
this.gameState.consequenceCount++;
|
|
// Clear the last skipped task when consequence is completed
|
|
this.gameState.lastSkippedTask = null;
|
|
// Consequence tasks don't award points or affect streak
|
|
} else {
|
|
this.gameState.usedMainTasks.push(this.gameState.currentTask.id);
|
|
this.gameState.completedCount++;
|
|
|
|
// Increment streak for regular tasks
|
|
this.gameState.currentStreak++;
|
|
|
|
// Award points for completing main tasks
|
|
const difficulty = this.gameState.currentTask.difficulty || 'Medium';
|
|
const basePoints = this.getPointsForDifficulty(difficulty);
|
|
this.gameState.score += basePoints;
|
|
|
|
// Check for streak bonus (every 10 consecutive completed tasks)
|
|
const streakBonus = this.checkStreakBonus();
|
|
if (streakBonus > 0) {
|
|
this.gameState.score += streakBonus;
|
|
this.gameState.totalStreakBonuses += streakBonus;
|
|
this.showStreakBonusNotification(this.gameState.currentStreak, streakBonus);
|
|
// Trigger special streak message
|
|
this.flashMessageManager.triggerEventMessage('streak');
|
|
} else {
|
|
// Trigger regular completion message
|
|
this.flashMessageManager.triggerEventMessage('taskComplete');
|
|
}
|
|
|
|
// Check for score target win condition
|
|
if (this.gameState.gameMode === 'score-target' && this.gameState.score >= this.gameState.scoreTarget) {
|
|
this.updateStats(); // Update stats before ending game
|
|
this.endGame('score-target');
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Load next main task (consequence tasks don't chain)
|
|
this.gameState.isConsequenceTask = false;
|
|
this.loadNextTask();
|
|
|
|
this.updateStats();
|
|
}
|
|
|
|
skipTask() {
|
|
if (!this.gameState.isRunning || this.gameState.isPaused) return;
|
|
|
|
if (this.gameState.isConsequenceTask) {
|
|
// Can't skip consequence tasks - they must be completed
|
|
alert("Consequence tasks cannot be skipped!");
|
|
return;
|
|
}
|
|
|
|
// Store the skipped task for mercy cost calculation
|
|
this.gameState.lastSkippedTask = this.gameState.currentTask;
|
|
|
|
// Mark main task as used and increment skip count
|
|
this.gameState.usedMainTasks.push(this.gameState.currentTask.id);
|
|
this.gameState.skippedCount++;
|
|
|
|
// Reset streak when task is skipped
|
|
this.gameState.currentStreak = 0;
|
|
this.gameState.lastStreakMilestone = 0;
|
|
|
|
// Trigger skip message
|
|
this.flashMessageManager.triggerEventMessage('taskSkip');
|
|
|
|
// Trigger punishment popups for skipping
|
|
this.popupImageManager.triggerPunishmentPopups();
|
|
|
|
// Load a consequence task
|
|
this.gameState.isConsequenceTask = true;
|
|
this.loadNextTask();
|
|
|
|
this.updateStats();
|
|
}
|
|
|
|
pauseGame() {
|
|
if (!this.gameState.isRunning) return;
|
|
|
|
this.gameState.isPaused = true;
|
|
this.gameState.pausedTime = Date.now();
|
|
this.stopTimer();
|
|
this.musicManager.pause();
|
|
this.flashMessageManager.pause();
|
|
this.showScreen('paused-screen');
|
|
|
|
// Auto-save the paused game state
|
|
this.autoSaveGameState();
|
|
}
|
|
|
|
resumeGame() {
|
|
if (!this.gameState.isRunning || !this.gameState.isPaused) return;
|
|
|
|
this.gameState.totalPausedTime += Date.now() - this.gameState.pausedTime;
|
|
this.gameState.isPaused = false;
|
|
this.startTimer();
|
|
this.musicManager.resume();
|
|
this.flashMessageManager.resume();
|
|
this.showScreen('game-screen');
|
|
}
|
|
|
|
quitGame() {
|
|
this.endGame();
|
|
}
|
|
|
|
endGame(reason = 'complete-all') {
|
|
this.gameState.isRunning = false;
|
|
this.stopTimer();
|
|
this.flashMessageManager.stop();
|
|
this.showFinalStats(reason);
|
|
this.showScreen('game-over-screen');
|
|
}
|
|
|
|
resetGame() {
|
|
this.gameState = {
|
|
isRunning: false,
|
|
isPaused: false,
|
|
currentTask: null,
|
|
isConsequenceTask: false,
|
|
startTime: null,
|
|
pausedTime: 0,
|
|
totalPausedTime: 0,
|
|
completedCount: 0,
|
|
skippedCount: 0,
|
|
consequenceCount: 0,
|
|
score: 0,
|
|
lastSkippedTask: null,
|
|
usedMainTasks: [],
|
|
usedConsequenceTasks: [],
|
|
usedTaskImages: [],
|
|
usedConsequenceImages: [],
|
|
gameMode: 'complete-all',
|
|
timeLimit: 300,
|
|
scoreTarget: 1000,
|
|
currentStreak: 0,
|
|
totalStreakBonuses: 0,
|
|
lastStreakMilestone: 0
|
|
};
|
|
|
|
this.stopTimer();
|
|
this.showScreen('start-screen');
|
|
}
|
|
|
|
startTimer() {
|
|
this.timerInterval = setInterval(() => {
|
|
this.updateTimer();
|
|
}, 1000);
|
|
}
|
|
|
|
stopTimer() {
|
|
if (this.timerInterval) {
|
|
clearInterval(this.timerInterval);
|
|
this.timerInterval = null;
|
|
}
|
|
}
|
|
|
|
updateTimer() {
|
|
if (!this.gameState.isRunning || this.gameState.isPaused) return;
|
|
|
|
const currentTime = Date.now();
|
|
const elapsed = currentTime - this.gameState.startTime - this.gameState.totalPausedTime;
|
|
|
|
let formattedTime;
|
|
|
|
if (this.gameState.gameMode === 'timed') {
|
|
// Countdown timer for timed mode
|
|
const remainingMs = (this.gameState.timeLimit * 1000) - elapsed;
|
|
if (remainingMs <= 0) {
|
|
// Time's up!
|
|
formattedTime = '00:00';
|
|
document.getElementById('timer').textContent = formattedTime;
|
|
this.endGame('time');
|
|
return;
|
|
}
|
|
formattedTime = this.formatTime(remainingMs);
|
|
|
|
// Change color when time is running low (less than 30 seconds)
|
|
const timerElement = document.getElementById('timer');
|
|
if (remainingMs <= 30000) {
|
|
timerElement.style.color = '#ff4757';
|
|
timerElement.style.fontWeight = 'bold';
|
|
} else if (remainingMs <= 60000) {
|
|
timerElement.style.color = '#ffa502';
|
|
timerElement.style.fontWeight = 'bold';
|
|
} else {
|
|
timerElement.style.color = '';
|
|
timerElement.style.fontWeight = '';
|
|
}
|
|
} else {
|
|
// Normal elapsed timer for other modes
|
|
formattedTime = this.formatTime(elapsed);
|
|
}
|
|
|
|
document.getElementById('timer').textContent = formattedTime;
|
|
|
|
// Update timer status
|
|
const timerStatus = document.getElementById('timer-status');
|
|
if (this.gameState.isPaused) {
|
|
timerStatus.textContent = '(PAUSED)';
|
|
} else if (this.gameState.gameMode === 'timed') {
|
|
timerStatus.textContent = '(TIME LEFT)';
|
|
} else {
|
|
timerStatus.textContent = '';
|
|
}
|
|
}
|
|
|
|
formatTime(milliseconds) {
|
|
const totalSeconds = Math.floor(milliseconds / 1000);
|
|
const minutes = Math.floor(totalSeconds / 60);
|
|
const seconds = totalSeconds % 60;
|
|
|
|
return `${minutes.toString().padStart(2, '0')}:${seconds.toString().padStart(2, '0')}`;
|
|
}
|
|
|
|
updateStats() {
|
|
document.getElementById('completed-count').textContent = this.gameState.completedCount;
|
|
document.getElementById('skipped-count').textContent = this.gameState.skippedCount;
|
|
document.getElementById('consequence-count').textContent = this.gameState.consequenceCount;
|
|
document.getElementById('score').textContent = this.gameState.score;
|
|
|
|
// Update streak display
|
|
const streakElement = document.getElementById('current-streak');
|
|
if (streakElement) {
|
|
streakElement.textContent = this.gameState.currentStreak;
|
|
|
|
// Add visual indicator for streak milestones
|
|
const streakContainer = streakElement.parentElement;
|
|
if (this.gameState.currentStreak >= 10) {
|
|
streakContainer.classList.add('streak-milestone');
|
|
} else {
|
|
streakContainer.classList.remove('streak-milestone');
|
|
}
|
|
}
|
|
}
|
|
|
|
showFinalStats(reason = 'complete-all') {
|
|
const currentTime = Date.now();
|
|
const totalTime = currentTime - this.gameState.startTime - this.gameState.totalPausedTime;
|
|
const formattedTime = this.formatTime(totalTime);
|
|
|
|
// Update the game over title based on end reason
|
|
const gameOverTitle = document.querySelector('#game-over-screen h2');
|
|
const gameOverMessage = document.querySelector('#game-over-screen p');
|
|
|
|
switch (reason) {
|
|
case 'time':
|
|
gameOverTitle.textContent = '⏰ Time\'s Up!';
|
|
gameOverMessage.textContent = 'You ran out of time! See how many tasks you completed.';
|
|
break;
|
|
case 'score-target':
|
|
gameOverTitle.textContent = '🏆 Target Reached!';
|
|
gameOverMessage.textContent = `Congratulations! You reached the target score of ${this.gameState.scoreTarget} points!`;
|
|
break;
|
|
case 'complete-all':
|
|
default:
|
|
gameOverTitle.textContent = '🎉 All Tasks Complete!';
|
|
gameOverMessage.textContent = 'Congratulations! You\'ve completed all available tasks!';
|
|
break;
|
|
}
|
|
|
|
document.getElementById('final-score').textContent = this.gameState.score;
|
|
document.getElementById('final-time').textContent = formattedTime;
|
|
document.getElementById('final-completed').textContent = this.gameState.completedCount;
|
|
document.getElementById('final-skipped').textContent = this.gameState.skippedCount;
|
|
document.getElementById('final-consequences').textContent = this.gameState.consequenceCount;
|
|
|
|
// Update streak bonus stats
|
|
const bestStreakElement = document.getElementById('final-best-streak');
|
|
const streakBonusesElement = document.getElementById('final-streak-bonuses');
|
|
if (bestStreakElement) {
|
|
bestStreakElement.textContent = this.gameState.currentStreak;
|
|
}
|
|
if (streakBonusesElement) {
|
|
streakBonusesElement.textContent = this.gameState.totalStreakBonuses;
|
|
}
|
|
|
|
// Add game mode info to stats
|
|
let gameModeText = '';
|
|
switch (this.gameState.gameMode) {
|
|
case 'timed':
|
|
gameModeText = `Timed Challenge (${this.gameState.timeLimit / 60} minutes)`;
|
|
break;
|
|
case 'score-target':
|
|
gameModeText = `Score Target (${this.gameState.scoreTarget} points)`;
|
|
break;
|
|
case 'complete-all':
|
|
default:
|
|
gameModeText = 'Complete All Tasks';
|
|
break;
|
|
}
|
|
|
|
// Show game mode info if element exists
|
|
const gameModeElement = document.getElementById('final-game-mode');
|
|
if (gameModeElement) {
|
|
gameModeElement.textContent = gameModeText;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Unified Data Management System
|
|
class DataManager {
|
|
constructor() {
|
|
this.version = "1.0";
|
|
this.storageKey = "webGame-data";
|
|
this.data = this.loadData();
|
|
this.autoSaveInterval = null;
|
|
this.startAutoSave();
|
|
}
|
|
|
|
getDefaultData() {
|
|
return {
|
|
version: this.version,
|
|
timestamp: Date.now(),
|
|
settings: {
|
|
theme: "default",
|
|
difficulty: "medium",
|
|
music: {
|
|
volume: 30,
|
|
loopMode: 0,
|
|
shuffle: false,
|
|
currentTrack: 0
|
|
}
|
|
},
|
|
gameplay: {
|
|
totalGamesPlayed: 0,
|
|
totalTasksCompleted: 0,
|
|
totalTasksSkipped: 0,
|
|
bestScore: 0,
|
|
currentStreak: 0,
|
|
longestStreak: 0,
|
|
totalPlayTime: 0,
|
|
mercyPointsEarned: 0,
|
|
mercyPointsSpent: 0
|
|
},
|
|
customContent: {
|
|
tasks: {
|
|
main: [],
|
|
consequence: []
|
|
},
|
|
customImages: [],
|
|
disabledImages: []
|
|
},
|
|
statistics: {
|
|
taskCompletionRate: 0,
|
|
averageGameDuration: 0,
|
|
difficultyStats: {
|
|
easy: { played: 0, completed: 0 },
|
|
medium: { played: 0, completed: 0 },
|
|
hard: { played: 0, completed: 0 }
|
|
},
|
|
themeUsage: {},
|
|
lastPlayed: null
|
|
}
|
|
};
|
|
}
|
|
|
|
loadData() {
|
|
try {
|
|
// Try new unified format first
|
|
const unified = localStorage.getItem(this.storageKey);
|
|
if (unified) {
|
|
const data = JSON.parse(unified);
|
|
return this.migrateData(data);
|
|
}
|
|
|
|
// Migrate from old scattered localStorage
|
|
return this.migrateFromOldFormat();
|
|
} catch (error) {
|
|
console.warn('Data loading failed, using defaults:', error);
|
|
return this.getDefaultData();
|
|
}
|
|
}
|
|
|
|
migrateFromOldFormat() {
|
|
const data = this.getDefaultData();
|
|
|
|
// Migrate existing scattered data
|
|
try {
|
|
// Theme
|
|
const theme = localStorage.getItem('selectedTheme');
|
|
if (theme) data.settings.theme = theme;
|
|
|
|
// Music settings
|
|
const volume = localStorage.getItem('gameMusic-volume');
|
|
if (volume) data.settings.music.volume = parseInt(volume);
|
|
|
|
const track = localStorage.getItem('gameMusic-track');
|
|
if (track) data.settings.music.currentTrack = parseInt(track);
|
|
|
|
const loopMode = localStorage.getItem('gameMusic-loopMode');
|
|
if (loopMode) data.settings.music.loopMode = parseInt(loopMode);
|
|
|
|
const shuffle = localStorage.getItem('gameMusic-shuffleMode');
|
|
if (shuffle) data.settings.music.shuffle = shuffle === 'true';
|
|
|
|
// Custom tasks
|
|
const mainTasks = localStorage.getItem('customMainTasks');
|
|
if (mainTasks) data.customContent.tasks.main = JSON.parse(mainTasks);
|
|
|
|
const consequenceTasks = localStorage.getItem('customConsequenceTasks');
|
|
if (consequenceTasks) data.customContent.tasks.consequence = JSON.parse(consequenceTasks);
|
|
|
|
// Difficulty
|
|
const difficulty = localStorage.getItem('selectedDifficulty');
|
|
if (difficulty) data.settings.difficulty = difficulty;
|
|
|
|
console.log('Migrated data from old format');
|
|
this.saveData(); // Save in new format
|
|
this.cleanupOldStorage(); // Remove old keys
|
|
|
|
} catch (error) {
|
|
console.warn('Migration failed, using defaults:', error);
|
|
}
|
|
|
|
return data;
|
|
}
|
|
|
|
migrateData(data) {
|
|
// Handle version upgrades here
|
|
if (!data.version || data.version < this.version) {
|
|
console.log(`Upgrading data from ${data.version || 'unknown'} to ${this.version}`);
|
|
|
|
// Add any missing properties from default
|
|
const defaultData = this.getDefaultData();
|
|
data = this.deepMerge(defaultData, data);
|
|
data.version = this.version;
|
|
}
|
|
|
|
return data;
|
|
}
|
|
|
|
deepMerge(target, source) {
|
|
const result = { ...target };
|
|
|
|
for (const key in source) {
|
|
if (source[key] && typeof source[key] === 'object' && !Array.isArray(source[key])) {
|
|
result[key] = this.deepMerge(target[key] || {}, source[key]);
|
|
} else {
|
|
result[key] = source[key];
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
cleanupOldStorage() {
|
|
// Remove old scattered localStorage keys
|
|
const oldKeys = [
|
|
'selectedTheme', 'gameMusic-volume', 'gameMusic-track',
|
|
'gameMusic-loopMode', 'gameMusic-shuffleMode',
|
|
'customMainTasks', 'customConsequenceTasks', 'selectedDifficulty'
|
|
];
|
|
|
|
oldKeys.forEach(key => localStorage.removeItem(key));
|
|
console.log('Cleaned up old storage keys');
|
|
}
|
|
|
|
saveData() {
|
|
try {
|
|
if (!this.data) {
|
|
this.data = this.getDefaultData();
|
|
}
|
|
this.data.timestamp = Date.now();
|
|
localStorage.setItem(this.storageKey, JSON.stringify(this.data));
|
|
} catch (error) {
|
|
console.error('Failed to save data:', error);
|
|
throw error; // Re-throw so calling code can handle it
|
|
}
|
|
}
|
|
|
|
// Settings Management
|
|
getSetting(path) {
|
|
return this.getNestedValue(this.data.settings, path);
|
|
}
|
|
|
|
setSetting(path, value) {
|
|
this.setNestedValue(this.data.settings, path, value);
|
|
this.saveData();
|
|
}
|
|
|
|
// Generic data access
|
|
get(key) {
|
|
// Handle special cases first
|
|
if (key === 'autoSaveGameState') {
|
|
return JSON.parse(localStorage.getItem('autoSaveGameState') || 'null');
|
|
}
|
|
|
|
// For custom images and disabled images, store them in customContent
|
|
if (key === 'customImages') {
|
|
return this.data.customContent.customImages || [];
|
|
}
|
|
|
|
if (key === 'disabledImages') {
|
|
return this.data.customContent.disabledImages || [];
|
|
}
|
|
|
|
// Generic access to data properties
|
|
return this.data[key];
|
|
}
|
|
|
|
set(key, value) {
|
|
// Handle special cases first
|
|
if (key === 'autoSaveGameState') {
|
|
localStorage.setItem('autoSaveGameState', JSON.stringify(value));
|
|
return;
|
|
}
|
|
|
|
// For custom images and disabled images, store them in customContent
|
|
if (key === 'customImages') {
|
|
if (!this.data.customContent.customImages) {
|
|
this.data.customContent.customImages = [];
|
|
}
|
|
this.data.customContent.customImages = value;
|
|
this.saveData();
|
|
return;
|
|
}
|
|
|
|
if (key === 'disabledImages') {
|
|
if (!this.data.customContent.disabledImages) {
|
|
this.data.customContent.disabledImages = [];
|
|
}
|
|
this.data.customContent.disabledImages = value;
|
|
this.saveData();
|
|
return;
|
|
}
|
|
|
|
// Generic setter
|
|
this.data[key] = value;
|
|
this.saveData();
|
|
}
|
|
|
|
// Gameplay Statistics
|
|
recordGameStart() {
|
|
this.data.gameplay.totalGamesPlayed++;
|
|
this.data.statistics.lastPlayed = Date.now();
|
|
this.saveData();
|
|
}
|
|
|
|
recordTaskComplete(difficulty, points, isConsequence = false) {
|
|
this.data.gameplay.totalTasksCompleted++;
|
|
|
|
if (!isConsequence) {
|
|
this.data.gameplay.currentStreak++;
|
|
this.data.gameplay.longestStreak = Math.max(
|
|
this.data.gameplay.longestStreak,
|
|
this.data.gameplay.currentStreak
|
|
);
|
|
|
|
// Track difficulty stats
|
|
if (this.data.statistics.difficultyStats[difficulty]) {
|
|
this.data.statistics.difficultyStats[difficulty].completed++;
|
|
}
|
|
}
|
|
|
|
this.updateStatistics();
|
|
this.saveData();
|
|
}
|
|
|
|
recordTaskSkip(mercyPoints = 0) {
|
|
this.data.gameplay.totalTasksSkipped++;
|
|
this.data.gameplay.currentStreak = 0;
|
|
|
|
if (mercyPoints > 0) {
|
|
this.data.gameplay.mercyPointsSpent += mercyPoints;
|
|
}
|
|
|
|
this.updateStatistics();
|
|
this.saveData();
|
|
}
|
|
|
|
recordScore(score) {
|
|
this.data.gameplay.bestScore = Math.max(this.data.gameplay.bestScore, score);
|
|
this.saveData();
|
|
}
|
|
|
|
recordPlayTime(duration) {
|
|
this.data.gameplay.totalPlayTime += duration;
|
|
this.updateStatistics();
|
|
this.saveData();
|
|
}
|
|
|
|
updateStatistics() {
|
|
const { totalTasksCompleted, totalTasksSkipped } = this.data.gameplay;
|
|
const total = totalTasksCompleted + totalTasksSkipped;
|
|
|
|
this.data.statistics.taskCompletionRate = total > 0 ? totalTasksCompleted / total : 0;
|
|
|
|
if (this.data.gameplay.totalGamesPlayed > 0) {
|
|
this.data.statistics.averageGameDuration =
|
|
this.data.gameplay.totalPlayTime / this.data.gameplay.totalGamesPlayed;
|
|
}
|
|
}
|
|
|
|
// Custom Content Management
|
|
getCustomTasks(type = 'main') {
|
|
return this.data.customContent.tasks[type] || [];
|
|
}
|
|
|
|
addCustomTask(task, type = 'main') {
|
|
if (!this.data.customContent.tasks[type]) {
|
|
this.data.customContent.tasks[type] = [];
|
|
}
|
|
this.data.customContent.tasks[type].push(task);
|
|
this.saveData();
|
|
}
|
|
|
|
removeCustomTask(index, type = 'main') {
|
|
if (this.data.customContent.tasks[type]) {
|
|
this.data.customContent.tasks[type].splice(index, 1);
|
|
this.saveData();
|
|
}
|
|
}
|
|
|
|
resetCustomTasks(type = 'main') {
|
|
this.data.customContent.tasks[type] = [];
|
|
this.saveData();
|
|
}
|
|
|
|
// Export/Import System
|
|
exportData(includeStats = true) {
|
|
const exportData = {
|
|
version: this.version,
|
|
exportDate: new Date().toISOString(),
|
|
settings: this.data.settings,
|
|
customContent: this.data.customContent
|
|
};
|
|
|
|
if (includeStats) {
|
|
exportData.gameplay = this.data.gameplay;
|
|
exportData.statistics = this.data.statistics;
|
|
}
|
|
|
|
const dataStr = JSON.stringify(exportData, null, 2);
|
|
const blob = new Blob([dataStr], { type: 'application/json' });
|
|
const url = URL.createObjectURL(blob);
|
|
|
|
const a = document.createElement('a');
|
|
a.href = url;
|
|
a.download = `webgame-save-${new Date().toISOString().split('T')[0]}.json`;
|
|
document.body.appendChild(a);
|
|
a.click();
|
|
document.body.removeChild(a);
|
|
URL.revokeObjectURL(url);
|
|
|
|
return exportData;
|
|
}
|
|
|
|
importData(file) {
|
|
return new Promise((resolve, reject) => {
|
|
const reader = new FileReader();
|
|
|
|
reader.onload = (e) => {
|
|
try {
|
|
const importedData = JSON.parse(e.target.result);
|
|
|
|
// Validate import
|
|
if (!this.validateImportData(importedData)) {
|
|
throw new Error('Invalid save file format');
|
|
}
|
|
|
|
// Merge imported data
|
|
this.data = this.deepMerge(this.getDefaultData(), importedData);
|
|
this.data.version = this.version;
|
|
this.data.timestamp = Date.now();
|
|
|
|
this.saveData();
|
|
resolve(importedData);
|
|
|
|
} catch (error) {
|
|
reject(error);
|
|
}
|
|
};
|
|
|
|
reader.onerror = () => reject(new Error('Failed to read file'));
|
|
reader.readAsText(file);
|
|
});
|
|
}
|
|
|
|
validateImportData(data) {
|
|
// Basic validation
|
|
return data &&
|
|
typeof data === 'object' &&
|
|
(data.settings || data.customContent || data.gameplay);
|
|
}
|
|
|
|
// Utility functions
|
|
getNestedValue(obj, path) {
|
|
return path.split('.').reduce((current, key) => current && current[key], obj);
|
|
}
|
|
|
|
setNestedValue(obj, path, value) {
|
|
const keys = path.split('.');
|
|
const lastKey = keys.pop();
|
|
const target = keys.reduce((current, key) => {
|
|
if (!current[key]) current[key] = {};
|
|
return current[key];
|
|
}, obj);
|
|
target[lastKey] = value;
|
|
}
|
|
|
|
// Auto-save functionality
|
|
startAutoSave() {
|
|
this.autoSaveInterval = setInterval(() => {
|
|
this.saveData();
|
|
}, 30000); // Save every 30 seconds
|
|
}
|
|
|
|
stopAutoSave() {
|
|
if (this.autoSaveInterval) {
|
|
clearInterval(this.autoSaveInterval);
|
|
this.autoSaveInterval = null;
|
|
}
|
|
}
|
|
|
|
// Statistics getters for UI
|
|
getStats() {
|
|
return {
|
|
...this.data.gameplay,
|
|
...this.data.statistics,
|
|
hoursPlayed: Math.round(this.data.gameplay.totalPlayTime / 3600000 * 10) / 10,
|
|
completionPercentage: Math.round(this.data.statistics.taskCompletionRate * 100)
|
|
};
|
|
}
|
|
|
|
// Reset functionality
|
|
resetAllData() {
|
|
if (confirm('Are you sure you want to reset all data? This cannot be undone!')) {
|
|
this.data = this.getDefaultData();
|
|
this.saveData();
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
resetStats() {
|
|
if (confirm('Reset all statistics? Your settings and custom tasks will be kept.')) {
|
|
this.data.gameplay = this.getDefaultData().gameplay;
|
|
this.data.statistics = this.getDefaultData().statistics;
|
|
this.saveData();
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Simple Music Management System with Playlist Controls
|
|
class MusicManager {
|
|
constructor(dataManager) {
|
|
this.dataManager = dataManager;
|
|
this.currentAudio = null;
|
|
this.isPlaying = false;
|
|
this.volume = this.dataManager.getSetting('music.volume') || 30;
|
|
this.loopMode = this.dataManager.getSetting('music.loopMode') || 0;
|
|
this.shuffleMode = this.dataManager.getSetting('music.shuffle') || false;
|
|
this.currentTrackIndex = this.dataManager.getSetting('music.currentTrack') || 0;
|
|
this.playHistory = [];
|
|
|
|
// Initialize empty tracks array - custom tracks will be loaded next
|
|
this.tracks = [];
|
|
|
|
// Load and add custom background music
|
|
this.loadCustomTracks();
|
|
|
|
this.updateUI();
|
|
this.initializeVolumeUI();
|
|
}
|
|
|
|
saveSettings() {
|
|
this.dataManager.setSetting('music.volume', this.volume);
|
|
this.dataManager.setSetting('music.currentTrack', this.currentTrackIndex);
|
|
this.dataManager.setSetting('music.loopMode', this.loopMode);
|
|
this.dataManager.setSetting('music.shuffle', this.shuffleMode);
|
|
}
|
|
|
|
loadCustomTracks() {
|
|
// Get custom background music from storage
|
|
const customAudio = this.dataManager.get('customAudio') || { background: [], ambient: [], effects: [] };
|
|
const backgroundMusic = customAudio.background || [];
|
|
|
|
// Add enabled custom background music to tracks
|
|
backgroundMusic.forEach(audio => {
|
|
if (audio.enabled !== false) { // Include if not explicitly disabled
|
|
this.tracks.push({
|
|
name: audio.title || audio.name,
|
|
file: audio.path,
|
|
isBuiltIn: false,
|
|
isCustom: true
|
|
});
|
|
}
|
|
});
|
|
|
|
console.log(`MusicManager: Loaded ${backgroundMusic.length} custom background tracks`);
|
|
}
|
|
|
|
refreshCustomTracks() {
|
|
// Clear all tracks since we only have custom tracks now
|
|
this.tracks = [];
|
|
|
|
// Reload custom tracks
|
|
this.loadCustomTracks();
|
|
|
|
// Update the UI to reflect new tracks
|
|
this.updateTrackSelector();
|
|
|
|
// If current track was custom and no longer exists, reset to first track
|
|
if (this.currentTrackIndex >= this.tracks.length) {
|
|
this.currentTrackIndex = 0;
|
|
this.saveSettings();
|
|
}
|
|
|
|
console.log(`MusicManager: Refreshed tracks, now have ${this.tracks.length} total tracks`);
|
|
}
|
|
|
|
initializeVolumeUI() {
|
|
const volumeSlider = document.getElementById('volume-slider');
|
|
const volumePercent = document.getElementById('volume-percent');
|
|
const trackSelector = document.getElementById('track-selector');
|
|
|
|
volumeSlider.value = this.volume;
|
|
volumePercent.textContent = `${this.volume}%`;
|
|
|
|
// Build track selector with all tracks
|
|
this.updateTrackSelector();
|
|
|
|
this.updateLoopButton();
|
|
this.updateShuffleButton();
|
|
}
|
|
|
|
// Loop mode cycling: 0 -> 1 -> 2 -> 0
|
|
toggleLoopMode() {
|
|
this.loopMode = (this.loopMode + 1) % 3;
|
|
this.saveSettings();
|
|
this.updateLoopButton();
|
|
this.updateAudioLoop();
|
|
}
|
|
|
|
toggleShuffleMode() {
|
|
this.shuffleMode = !this.shuffleMode;
|
|
this.saveSettings();
|
|
this.updateShuffleButton();
|
|
if (this.shuffleMode) {
|
|
this.playHistory = [this.currentTrackIndex]; // Start tracking
|
|
}
|
|
}
|
|
|
|
updateLoopButton() {
|
|
const loopBtn = document.getElementById('loop-btn');
|
|
switch(this.loopMode) {
|
|
case 0:
|
|
loopBtn.classList.remove('active');
|
|
loopBtn.innerHTML = '🔁';
|
|
loopBtn.title = 'Loop: Off';
|
|
break;
|
|
case 1:
|
|
loopBtn.classList.add('active');
|
|
loopBtn.innerHTML = '🔁';
|
|
loopBtn.title = 'Loop: Playlist';
|
|
break;
|
|
case 2:
|
|
loopBtn.classList.add('active');
|
|
loopBtn.innerHTML = '🔂';
|
|
loopBtn.title = 'Loop: Current Track';
|
|
break;
|
|
}
|
|
}
|
|
|
|
updateShuffleButton() {
|
|
const shuffleBtn = document.getElementById('shuffle-btn');
|
|
if (this.shuffleMode) {
|
|
shuffleBtn.classList.add('active');
|
|
shuffleBtn.title = 'Shuffle: On';
|
|
} else {
|
|
shuffleBtn.classList.remove('active');
|
|
shuffleBtn.title = 'Shuffle: Off';
|
|
}
|
|
}
|
|
|
|
updateAudioLoop() {
|
|
if (this.currentAudio) {
|
|
this.currentAudio.loop = (this.loopMode === 2);
|
|
}
|
|
}
|
|
|
|
getNextTrack() {
|
|
if (this.shuffleMode) {
|
|
return this.getShuffledNextTrack();
|
|
} else {
|
|
return (this.currentTrackIndex + 1) % this.tracks.length;
|
|
}
|
|
}
|
|
|
|
getShuffledNextTrack() {
|
|
if (this.tracks.length <= 1) return 0;
|
|
|
|
let availableTracks = [];
|
|
for (let i = 0; i < this.tracks.length; i++) {
|
|
// Avoid last 2 played tracks if possible
|
|
if (this.playHistory.length < 2 || !this.playHistory.slice(-2).includes(i)) {
|
|
availableTracks.push(i);
|
|
}
|
|
}
|
|
|
|
// If no tracks available (shouldn't happen with 4+ tracks), use all except current
|
|
if (availableTracks.length === 0) {
|
|
availableTracks = this.tracks.map((_, i) => i).filter(i => i !== this.currentTrackIndex);
|
|
}
|
|
|
|
return availableTracks[Math.floor(Math.random() * availableTracks.length)];
|
|
}
|
|
|
|
// Called when track ends naturally
|
|
onTrackEnded() {
|
|
if (this.loopMode === 2) {
|
|
// Loop current track - handled by audio.loop = true
|
|
return;
|
|
} else if (this.loopMode === 1) {
|
|
// Loop playlist - go to next track
|
|
this.playNextTrack();
|
|
} else {
|
|
// No loop - stop playing
|
|
this.isPlaying = false;
|
|
this.updateUI();
|
|
}
|
|
}
|
|
|
|
playNextTrack() {
|
|
const nextIndex = this.getNextTrack();
|
|
this.changeTrack(nextIndex);
|
|
|
|
// Update play history for shuffle
|
|
if (this.shuffleMode) {
|
|
this.playHistory.push(nextIndex);
|
|
// Keep only last 3 tracks in history
|
|
if (this.playHistory.length > 3) {
|
|
this.playHistory.shift();
|
|
}
|
|
}
|
|
}
|
|
|
|
toggle() {
|
|
if (this.isPlaying) {
|
|
this.stop();
|
|
} else {
|
|
this.play();
|
|
}
|
|
}
|
|
|
|
play() {
|
|
try {
|
|
if (!this.currentAudio) {
|
|
this.loadCurrentTrack();
|
|
}
|
|
|
|
this.currentAudio.play();
|
|
this.isPlaying = true;
|
|
this.updateUI();
|
|
} catch (error) {
|
|
console.log('Audio playback failed:', error);
|
|
// Fallback to silent mode
|
|
this.isPlaying = true;
|
|
this.updateUI();
|
|
}
|
|
}
|
|
|
|
stop() {
|
|
if (this.currentAudio) {
|
|
this.currentAudio.pause();
|
|
this.currentAudio.currentTime = 0;
|
|
}
|
|
this.isPlaying = false;
|
|
this.updateUI();
|
|
}
|
|
|
|
pause() {
|
|
if (this.currentAudio && this.isPlaying) {
|
|
this.currentAudio.pause();
|
|
}
|
|
}
|
|
|
|
resume() {
|
|
if (this.currentAudio && this.isPlaying) {
|
|
try {
|
|
this.currentAudio.play();
|
|
} catch (error) {
|
|
console.log('Audio resume failed:', error);
|
|
}
|
|
}
|
|
}
|
|
|
|
changeTrack(trackIndex) {
|
|
const wasPlaying = this.isPlaying;
|
|
|
|
if (this.currentAudio) {
|
|
this.currentAudio.pause();
|
|
this.currentAudio = null;
|
|
}
|
|
|
|
this.currentTrackIndex = trackIndex;
|
|
this.saveSettings();
|
|
|
|
if (wasPlaying) {
|
|
this.play();
|
|
} else {
|
|
this.updateUI();
|
|
}
|
|
}
|
|
|
|
setVolume(volume) {
|
|
this.volume = volume;
|
|
this.saveSettings();
|
|
|
|
if (this.currentAudio) {
|
|
this.currentAudio.volume = volume / 100;
|
|
}
|
|
|
|
const volumePercent = document.getElementById('volume-percent');
|
|
volumePercent.textContent = `${volume}%`;
|
|
|
|
// Update volume icon based on level
|
|
const volumeIcon = document.querySelector('.volume-icon');
|
|
if (volume === 0) {
|
|
volumeIcon.textContent = '🔇';
|
|
} else if (volume < 30) {
|
|
volumeIcon.textContent = '🔉';
|
|
} else {
|
|
volumeIcon.textContent = '🔊';
|
|
}
|
|
}
|
|
|
|
loadCurrentTrack() {
|
|
const track = this.tracks[this.currentTrackIndex];
|
|
|
|
// Create audio element with fallback
|
|
this.currentAudio = new Audio();
|
|
this.currentAudio.volume = this.volume / 100;
|
|
this.updateAudioLoop();
|
|
|
|
// Set source with error handling
|
|
this.currentAudio.src = track.file;
|
|
|
|
// Handle loading errors gracefully
|
|
this.currentAudio.onerror = () => {
|
|
console.log(`Could not load audio: ${track.file}`);
|
|
// Continue without audio
|
|
};
|
|
|
|
// Handle successful loading
|
|
this.currentAudio.onloadeddata = () => {
|
|
console.log(`Loaded audio: ${track.name}`);
|
|
};
|
|
|
|
// Handle track ending
|
|
this.currentAudio.onended = () => {
|
|
if (this.isPlaying) {
|
|
this.onTrackEnded();
|
|
}
|
|
};
|
|
}
|
|
|
|
updateUI() {
|
|
const toggleBtn = document.getElementById('music-toggle');
|
|
const statusSpan = document.getElementById('music-status');
|
|
const trackSelector = document.getElementById('track-selector');
|
|
const currentTrack = this.tracks[this.currentTrackIndex];
|
|
|
|
// Update track selector
|
|
trackSelector.value = this.currentTrackIndex;
|
|
|
|
if (this.isPlaying) {
|
|
toggleBtn.classList.add('playing');
|
|
toggleBtn.innerHTML = '⏸️';
|
|
|
|
// Show current mode in status
|
|
let modeText = '';
|
|
if (this.shuffleMode) modeText += ' 🔀';
|
|
if (this.loopMode === 1) modeText += ' 🔁';
|
|
else if (this.loopMode === 2) modeText += ' 🔂';
|
|
|
|
statusSpan.textContent = `♪ ${currentTrack.name}${modeText}`;
|
|
} else {
|
|
toggleBtn.classList.remove('playing');
|
|
toggleBtn.innerHTML = '🎵';
|
|
statusSpan.textContent = 'Music: Off';
|
|
}
|
|
}
|
|
|
|
updateTrackSelector() {
|
|
const trackSelector = document.getElementById('track-selector');
|
|
if (!trackSelector) return;
|
|
|
|
// Clear existing options
|
|
trackSelector.innerHTML = '';
|
|
|
|
// Add all tracks (built-in and custom)
|
|
this.tracks.forEach((track, index) => {
|
|
const option = document.createElement('option');
|
|
option.value = index;
|
|
option.textContent = track.isCustom ? `${track.name} (Custom)` : track.name;
|
|
trackSelector.appendChild(option);
|
|
});
|
|
|
|
// Set current selection
|
|
trackSelector.value = this.currentTrackIndex;
|
|
}
|
|
}
|
|
|
|
// Annoyance Management Methods - Phase 2: Advanced Message Management
|
|
TaskChallengeGame.prototype.showAnnoyanceManagement = function() {
|
|
this.showScreen('annoyance-management-screen');
|
|
this.setupAnnoyanceManagementEventListeners();
|
|
this.loadAnnoyanceSettings();
|
|
this.showAnnoyanceTab('messages'); // Default to messages tab
|
|
};
|
|
|
|
TaskChallengeGame.prototype.setupAnnoyanceManagementEventListeners = function() {
|
|
// Back button
|
|
const backBtn = document.getElementById('back-to-start-from-annoyance-btn');
|
|
if (backBtn) {
|
|
backBtn.onclick = () => this.showScreen('start-screen');
|
|
}
|
|
|
|
// Save settings button
|
|
const saveBtn = document.getElementById('save-annoyance-settings');
|
|
if (saveBtn) {
|
|
saveBtn.onclick = () => this.saveAllAnnoyanceSettings();
|
|
}
|
|
|
|
// Tab navigation
|
|
document.getElementById('messages-tab').onclick = () => this.showAnnoyanceTab('messages');
|
|
document.getElementById('appearance-tab').onclick = () => this.showAnnoyanceTab('appearance');
|
|
document.getElementById('behavior-tab').onclick = () => this.showAnnoyanceTab('behavior');
|
|
document.getElementById('popup-images-tab').onclick = () => this.showAnnoyanceTab('popup-images');
|
|
document.getElementById('import-export-tab').onclick = () => this.showAnnoyanceTab('import-export');
|
|
|
|
this.setupMessagesTabListeners();
|
|
this.setupAppearanceTabListeners();
|
|
this.setupBehaviorTabListeners();
|
|
this.setupPopupImagesTabListeners();
|
|
this.setupImportExportTabListeners();
|
|
};
|
|
|
|
TaskChallengeGame.prototype.showAnnoyanceTab = function(tabName) {
|
|
// Update tab buttons
|
|
document.querySelectorAll('.annoyance-tab').forEach(tab => tab.classList.remove('active'));
|
|
document.getElementById(`${tabName}-tab`).classList.add('active');
|
|
|
|
// Update tab content
|
|
document.querySelectorAll('.annoyance-tab-content').forEach(content => content.classList.remove('active'));
|
|
document.getElementById(`${tabName}-tab-content`).classList.add('active');
|
|
|
|
// Load tab-specific content
|
|
switch(tabName) {
|
|
case 'messages':
|
|
this.loadMessagesTab();
|
|
break;
|
|
case 'appearance':
|
|
this.loadAppearanceTab();
|
|
break;
|
|
case 'behavior':
|
|
this.loadBehaviorTab();
|
|
break;
|
|
case 'popup-images':
|
|
this.loadPopupImagesSettings();
|
|
break;
|
|
case 'import-export':
|
|
this.loadImportExportTab();
|
|
break;
|
|
}
|
|
};
|
|
|
|
// Messages Tab Management
|
|
TaskChallengeGame.prototype.setupMessagesTabListeners = function() {
|
|
const enabledCheckbox = document.getElementById('flash-messages-enabled');
|
|
const addBtn = document.getElementById('add-new-message-btn');
|
|
const closeEditorBtn = document.getElementById('close-editor-btn');
|
|
const saveMessageBtn = document.getElementById('save-message-btn');
|
|
const previewCurrentBtn = document.getElementById('preview-current-message-btn');
|
|
const cancelEditBtn = document.getElementById('cancel-edit-btn');
|
|
const categoryFilter = document.getElementById('category-filter');
|
|
const showDisabledCheckbox = document.getElementById('show-disabled-messages');
|
|
const messageTextarea = document.getElementById('message-text');
|
|
|
|
if (enabledCheckbox) {
|
|
enabledCheckbox.onchange = (e) => this.updateFlashMessageSetting('enabled', e.target.checked);
|
|
}
|
|
|
|
if (addBtn) {
|
|
addBtn.onclick = () => this.showMessageEditor();
|
|
}
|
|
|
|
if (closeEditorBtn) {
|
|
closeEditorBtn.onclick = () => this.hideMessageEditor();
|
|
}
|
|
|
|
if (saveMessageBtn) {
|
|
saveMessageBtn.onclick = () => this.saveCurrentMessage();
|
|
}
|
|
|
|
if (previewCurrentBtn) {
|
|
previewCurrentBtn.onclick = () => this.previewCurrentMessage();
|
|
}
|
|
|
|
if (cancelEditBtn) {
|
|
cancelEditBtn.onclick = () => this.hideMessageEditor();
|
|
}
|
|
|
|
if (categoryFilter) {
|
|
categoryFilter.onchange = () => this.refreshMessageList();
|
|
}
|
|
|
|
if (showDisabledCheckbox) {
|
|
showDisabledCheckbox.onchange = () => this.refreshMessageList();
|
|
}
|
|
|
|
if (messageTextarea) {
|
|
messageTextarea.oninput = () => this.updateCharacterCount();
|
|
}
|
|
};
|
|
|
|
TaskChallengeGame.prototype.loadMessagesTab = function() {
|
|
const config = this.flashMessageManager.getConfig();
|
|
const enabledCheckbox = document.getElementById('flash-messages-enabled');
|
|
if (enabledCheckbox) enabledCheckbox.checked = config.enabled;
|
|
|
|
this.refreshMessageList();
|
|
this.hideMessageEditor(); // Ensure editor is closed by default
|
|
};
|
|
|
|
TaskChallengeGame.prototype.refreshMessageList = function() {
|
|
const categoryFilter = document.getElementById('category-filter');
|
|
const showDisabled = document.getElementById('show-disabled-messages').checked;
|
|
const selectedCategory = categoryFilter.value;
|
|
|
|
let messages = this.flashMessageManager.getMessagesByCategory(selectedCategory);
|
|
|
|
if (!showDisabled) {
|
|
messages = messages.filter(msg => msg.enabled !== false);
|
|
}
|
|
|
|
this.renderMessageList(messages);
|
|
this.updateMessageStats();
|
|
};
|
|
|
|
TaskChallengeGame.prototype.renderMessageList = function(messages) {
|
|
const listElement = document.getElementById('message-list');
|
|
|
|
if (messages.length === 0) {
|
|
listElement.innerHTML = '<div style="padding: 20px; text-align: center; color: #666;">No messages match your criteria.</div>';
|
|
return;
|
|
}
|
|
|
|
listElement.innerHTML = messages.map(msg => `
|
|
<div class="message-item ${msg.enabled === false ? 'disabled' : ''}" data-message-id="${msg.id}">
|
|
<div class="message-content">
|
|
<div class="message-text">${msg.text}</div>
|
|
<div class="message-meta">
|
|
<span class="message-category ${msg.category}">${this.getCategoryEmoji(msg.category)} ${msg.category || 'Custom'}</span>
|
|
<span>Priority: ${msg.priority || 'Normal'}</span>
|
|
${msg.isCustom ? '<span>Custom</span>' : '<span>Default</span>'}
|
|
</div>
|
|
</div>
|
|
<div class="message-actions">
|
|
<div class="message-toggle ${msg.enabled !== false ? 'enabled' : ''}"
|
|
onclick="game.toggleMessage(${msg.id})"></div>
|
|
<button class="btn btn-info btn-small" onclick="game.editMessage(${msg.id})">Edit</button>
|
|
<button class="btn btn-success btn-small" onclick="game.previewMessage(${msg.id})">Preview</button>
|
|
${msg.isCustom ? `<button class="btn btn-danger btn-small" onclick="game.deleteMessage(${msg.id})">Delete</button>` : ''}
|
|
</div>
|
|
</div>
|
|
`).join('');
|
|
};
|
|
|
|
TaskChallengeGame.prototype.updateMessageStats = function() {
|
|
const stats = this.flashMessageManager.getMessageStats();
|
|
const statsElement = document.getElementById('message-stats');
|
|
if (statsElement) {
|
|
statsElement.textContent = `${stats.total} messages (${stats.enabled} enabled, ${stats.disabled} disabled)`;
|
|
}
|
|
};
|
|
|
|
TaskChallengeGame.prototype.showMessageEditor = function(messageData = null) {
|
|
const editor = document.getElementById('message-editor');
|
|
const title = document.getElementById('editor-title');
|
|
const textarea = document.getElementById('message-text');
|
|
const category = document.getElementById('message-category');
|
|
const priority = document.getElementById('message-priority');
|
|
|
|
if (messageData) {
|
|
title.textContent = 'Edit Message';
|
|
textarea.value = messageData.text || '';
|
|
category.value = messageData.category || 'custom';
|
|
priority.value = messageData.priority || 'normal';
|
|
editor.dataset.editingId = messageData.id;
|
|
} else {
|
|
title.textContent = 'Add New Message';
|
|
textarea.value = '';
|
|
category.value = 'custom';
|
|
priority.value = 'normal';
|
|
delete editor.dataset.editingId;
|
|
}
|
|
|
|
editor.style.display = 'block';
|
|
textarea.focus();
|
|
this.updateCharacterCount();
|
|
};
|
|
|
|
TaskChallengeGame.prototype.hideMessageEditor = function() {
|
|
const editor = document.getElementById('message-editor');
|
|
editor.style.display = 'none';
|
|
};
|
|
|
|
TaskChallengeGame.prototype.saveCurrentMessage = function() {
|
|
const textarea = document.getElementById('message-text');
|
|
const category = document.getElementById('message-category');
|
|
const priority = document.getElementById('message-priority');
|
|
const editor = document.getElementById('message-editor');
|
|
|
|
const messageText = textarea.value.trim();
|
|
if (!messageText) {
|
|
this.showNotification('Please enter a message text!', 'error');
|
|
return;
|
|
}
|
|
|
|
const messageData = {
|
|
text: messageText,
|
|
category: category.value,
|
|
priority: priority.value
|
|
};
|
|
|
|
if (editor.dataset.editingId) {
|
|
// Edit existing message
|
|
const updated = this.flashMessageManager.editMessage(parseInt(editor.dataset.editingId), messageData);
|
|
if (updated) {
|
|
this.showNotification('✅ Message updated successfully!', 'success');
|
|
} else {
|
|
this.showNotification('❌ Failed to update message', 'error');
|
|
}
|
|
} else {
|
|
// Add new message
|
|
const newMessage = this.flashMessageManager.addMessage(messageData);
|
|
this.showNotification('✅ Message added successfully!', 'success');
|
|
}
|
|
|
|
this.hideMessageEditor();
|
|
this.refreshMessageList();
|
|
};
|
|
|
|
TaskChallengeGame.prototype.previewCurrentMessage = function() {
|
|
const textarea = document.getElementById('message-text');
|
|
const messageText = textarea.value.trim();
|
|
|
|
if (!messageText) {
|
|
this.showNotification('Please enter a message to preview!', 'warning');
|
|
return;
|
|
}
|
|
|
|
this.flashMessageManager.previewMessage({ text: messageText });
|
|
};
|
|
|
|
TaskChallengeGame.prototype.updateCharacterCount = function() {
|
|
const textarea = document.getElementById('message-text');
|
|
const counter = document.getElementById('char-count');
|
|
const currentLength = textarea.value.length;
|
|
const maxLength = 200;
|
|
|
|
counter.textContent = currentLength;
|
|
|
|
const counterElement = counter.parentElement;
|
|
counterElement.className = 'char-counter';
|
|
|
|
if (currentLength > maxLength * 0.9) {
|
|
counterElement.className += ' warning';
|
|
}
|
|
if (currentLength > maxLength) {
|
|
counterElement.className += ' error';
|
|
}
|
|
};
|
|
|
|
// Message action methods
|
|
TaskChallengeGame.prototype.toggleMessage = function(messageId) {
|
|
const enabled = this.flashMessageManager.toggleMessageEnabled(messageId);
|
|
this.refreshMessageList();
|
|
this.showNotification(`Message ${enabled ? 'enabled' : 'disabled'}`, 'success');
|
|
};
|
|
|
|
TaskChallengeGame.prototype.editMessage = function(messageId) {
|
|
const messages = this.flashMessageManager.getAllMessages();
|
|
const message = messages.find(msg => msg.id === messageId);
|
|
if (message) {
|
|
this.showMessageEditor(message);
|
|
}
|
|
};
|
|
|
|
TaskChallengeGame.prototype.previewMessage = function(messageId) {
|
|
const messages = this.flashMessageManager.getAllMessages();
|
|
const message = messages.find(msg => msg.id === messageId);
|
|
if (message) {
|
|
this.flashMessageManager.previewMessage(message);
|
|
}
|
|
};
|
|
|
|
TaskChallengeGame.prototype.deleteMessage = function(messageId) {
|
|
const messages = this.flashMessageManager.getAllMessages();
|
|
const message = messages.find(msg => msg.id === messageId);
|
|
|
|
if (message && confirm(`Delete this message?\n\n"${message.text}"`)) {
|
|
const deleted = this.flashMessageManager.deleteMessage(messageId);
|
|
if (deleted) {
|
|
this.showNotification('✅ Message deleted', 'success');
|
|
this.refreshMessageList();
|
|
}
|
|
}
|
|
};
|
|
|
|
TaskChallengeGame.prototype.getCategoryEmoji = function(category) {
|
|
const emojis = {
|
|
motivational: '💪',
|
|
encouraging: '🌟',
|
|
achievement: '🏆',
|
|
persistence: '🔥',
|
|
custom: '✨'
|
|
};
|
|
return emojis[category] || '✨';
|
|
};
|
|
|
|
// Appearance Tab Management
|
|
TaskChallengeGame.prototype.setupAppearanceTabListeners = function() {
|
|
const controls = {
|
|
position: document.getElementById('message-position'),
|
|
animation: document.getElementById('animation-style'),
|
|
fontSize: document.getElementById('font-size'),
|
|
opacity: document.getElementById('message-opacity'),
|
|
textColor: document.getElementById('text-color'),
|
|
backgroundColor: document.getElementById('background-color'),
|
|
resetBtn: document.getElementById('reset-appearance-btn'),
|
|
previewBtn: document.getElementById('preview-appearance-btn')
|
|
};
|
|
|
|
if (controls.position) {
|
|
controls.position.onchange = (e) => this.updateFlashMessageSetting('position', e.target.value);
|
|
}
|
|
|
|
if (controls.animation) {
|
|
controls.animation.onchange = (e) => this.updateFlashMessageSetting('animation', e.target.value);
|
|
}
|
|
|
|
if (controls.fontSize) {
|
|
controls.fontSize.oninput = (e) => {
|
|
const value = parseInt(e.target.value);
|
|
document.getElementById('font-size-display').textContent = `${value}px`;
|
|
this.updateFlashMessageSetting('fontSize', `${value}px`);
|
|
};
|
|
}
|
|
|
|
if (controls.opacity) {
|
|
controls.opacity.oninput = (e) => {
|
|
const value = parseInt(e.target.value);
|
|
document.getElementById('opacity-display').textContent = `${value}%`;
|
|
const bgColor = this.hexToRgba(controls.backgroundColor.value, value / 100);
|
|
this.updateFlashMessageSetting('backgroundColor', bgColor);
|
|
};
|
|
}
|
|
|
|
if (controls.textColor) {
|
|
controls.textColor.onchange = (e) => this.updateFlashMessageSetting('color', e.target.value);
|
|
}
|
|
|
|
if (controls.backgroundColor) {
|
|
controls.backgroundColor.onchange = (e) => {
|
|
const opacity = parseInt(controls.opacity.value) / 100;
|
|
const bgColor = this.hexToRgba(e.target.value, opacity);
|
|
this.updateFlashMessageSetting('backgroundColor', bgColor);
|
|
};
|
|
}
|
|
|
|
if (controls.resetBtn) {
|
|
controls.resetBtn.onclick = () => this.resetAppearanceToDefaults();
|
|
}
|
|
|
|
if (controls.previewBtn) {
|
|
controls.previewBtn.onclick = () => this.previewAppearanceStyle();
|
|
}
|
|
};
|
|
|
|
TaskChallengeGame.prototype.loadAppearanceTab = function() {
|
|
const config = this.flashMessageManager.getConfig();
|
|
|
|
const controls = {
|
|
position: document.getElementById('message-position'),
|
|
animation: document.getElementById('animation-style'),
|
|
fontSize: document.getElementById('font-size'),
|
|
opacity: document.getElementById('message-opacity'),
|
|
textColor: document.getElementById('text-color'),
|
|
backgroundColor: document.getElementById('background-color')
|
|
};
|
|
|
|
if (controls.position) controls.position.value = config.position;
|
|
if (controls.animation) controls.animation.value = config.animation;
|
|
if (controls.fontSize) {
|
|
const fontSize = parseInt(config.fontSize) || 24;
|
|
controls.fontSize.value = fontSize;
|
|
document.getElementById('font-size-display').textContent = `${fontSize}px`;
|
|
}
|
|
|
|
// Extract opacity from backgroundColor if it's rgba
|
|
let opacity = 90;
|
|
let bgHex = '#007bff';
|
|
if (config.backgroundColor.includes('rgba')) {
|
|
const rgbaMatch = config.backgroundColor.match(/rgba\((\d+),\s*(\d+),\s*(\d+),\s*([01]?\.?\d*)\)/);
|
|
if (rgbaMatch) {
|
|
opacity = Math.round(parseFloat(rgbaMatch[4]) * 100);
|
|
bgHex = this.rgbToHex(parseInt(rgbaMatch[1]), parseInt(rgbaMatch[2]), parseInt(rgbaMatch[3]));
|
|
}
|
|
}
|
|
|
|
if (controls.opacity) {
|
|
controls.opacity.value = opacity;
|
|
document.getElementById('opacity-display').textContent = `${opacity}%`;
|
|
}
|
|
if (controls.textColor) controls.textColor.value = config.color || '#ffffff';
|
|
if (controls.backgroundColor) controls.backgroundColor.value = bgHex;
|
|
};
|
|
|
|
TaskChallengeGame.prototype.resetAppearanceToDefaults = function() {
|
|
const defaults = gameData.defaultFlashConfig;
|
|
this.flashMessageManager.updateConfig({
|
|
position: defaults.position,
|
|
animation: defaults.animation,
|
|
fontSize: defaults.fontSize,
|
|
color: defaults.color,
|
|
backgroundColor: defaults.backgroundColor
|
|
});
|
|
this.loadAppearanceTab();
|
|
this.showNotification('✅ Appearance reset to defaults', 'success');
|
|
};
|
|
|
|
TaskChallengeGame.prototype.previewAppearanceStyle = function() {
|
|
const messages = this.flashMessageManager.getMessages();
|
|
if (messages.length > 0) {
|
|
const sampleMessage = messages[0];
|
|
this.flashMessageManager.previewMessage(sampleMessage);
|
|
} else {
|
|
this.flashMessageManager.previewMessage({ text: "This is a preview of your style settings!" });
|
|
}
|
|
};
|
|
|
|
// Behavior Tab Management
|
|
TaskChallengeGame.prototype.setupBehaviorTabListeners = function() {
|
|
const controls = {
|
|
duration: document.getElementById('display-duration'),
|
|
interval: document.getElementById('interval-delay'),
|
|
variation: document.getElementById('time-variation'),
|
|
eventBased: document.getElementById('event-based-messages'),
|
|
pauseOnHover: document.getElementById('pause-on-hover'),
|
|
testBtn: document.getElementById('test-behavior-btn')
|
|
};
|
|
|
|
if (controls.duration) {
|
|
controls.duration.oninput = (e) => {
|
|
const value = parseInt(e.target.value);
|
|
document.getElementById('duration-display').textContent = `${(value / 1000).toFixed(1)}s`;
|
|
this.updateFlashMessageSetting('displayDuration', value);
|
|
};
|
|
}
|
|
|
|
if (controls.interval) {
|
|
controls.interval.oninput = (e) => {
|
|
const value = parseInt(e.target.value);
|
|
document.getElementById('interval-display').textContent = `${Math.round(value / 1000)}s`;
|
|
this.updateFlashMessageSetting('intervalDelay', value);
|
|
};
|
|
}
|
|
|
|
if (controls.variation) {
|
|
controls.variation.oninput = (e) => {
|
|
const value = parseInt(e.target.value);
|
|
document.getElementById('variation-display').textContent = `±${Math.round(value / 1000)}s`;
|
|
this.updateFlashMessageSetting('timeVariation', value);
|
|
};
|
|
}
|
|
|
|
if (controls.eventBased) {
|
|
controls.eventBased.onchange = (e) => this.updateFlashMessageSetting('eventBasedMessages', e.target.checked);
|
|
}
|
|
|
|
if (controls.pauseOnHover) {
|
|
controls.pauseOnHover.onchange = (e) => this.updateFlashMessageSetting('pauseOnHover', e.target.checked);
|
|
}
|
|
|
|
if (controls.testBtn) {
|
|
controls.testBtn.onclick = () => this.testCurrentBehaviorSettings();
|
|
}
|
|
};
|
|
|
|
TaskChallengeGame.prototype.loadBehaviorTab = function() {
|
|
const config = this.flashMessageManager.getConfig();
|
|
|
|
const durationSlider = document.getElementById('display-duration');
|
|
const intervalSlider = document.getElementById('interval-delay');
|
|
const variationSlider = document.getElementById('time-variation');
|
|
const eventBasedCheckbox = document.getElementById('event-based-messages');
|
|
const pauseOnHoverCheckbox = document.getElementById('pause-on-hover');
|
|
|
|
if (durationSlider) {
|
|
durationSlider.value = config.displayDuration;
|
|
document.getElementById('duration-display').textContent = `${(config.displayDuration / 1000).toFixed(1)}s`;
|
|
}
|
|
if (intervalSlider) {
|
|
intervalSlider.value = config.intervalDelay;
|
|
document.getElementById('interval-display').textContent = `${Math.round(config.intervalDelay / 1000)}s`;
|
|
}
|
|
if (variationSlider) {
|
|
const variation = config.timeVariation || 5000;
|
|
variationSlider.value = variation;
|
|
document.getElementById('variation-display').textContent = `±${Math.round(variation / 1000)}s`;
|
|
}
|
|
if (eventBasedCheckbox) {
|
|
eventBasedCheckbox.checked = config.eventBasedMessages !== false;
|
|
}
|
|
if (pauseOnHoverCheckbox) {
|
|
pauseOnHoverCheckbox.checked = config.pauseOnHover || false;
|
|
}
|
|
};
|
|
|
|
TaskChallengeGame.prototype.testCurrentBehaviorSettings = function() {
|
|
this.showNotification('🧪 Testing behavior settings with 3 quick messages...', 'info');
|
|
|
|
let count = 0;
|
|
const showTestMessage = () => {
|
|
if (count >= 3) return;
|
|
|
|
const messages = this.flashMessageManager.getMessages();
|
|
if (messages.length > 0) {
|
|
const message = messages[Math.floor(Math.random() * messages.length)];
|
|
this.flashMessageManager.previewMessage(message);
|
|
}
|
|
|
|
count++;
|
|
if (count < 3) {
|
|
setTimeout(showTestMessage, 2000);
|
|
} else {
|
|
setTimeout(() => {
|
|
this.showNotification('✅ Behavior test complete!', 'success');
|
|
}, this.flashMessageManager.getConfig().displayDuration + 500);
|
|
}
|
|
};
|
|
|
|
setTimeout(showTestMessage, 500);
|
|
};
|
|
|
|
// Popup Images Tab Management
|
|
TaskChallengeGame.prototype.setupPopupImagesTabListeners = function() {
|
|
// Enable/disable toggle
|
|
const enabledCheckbox = document.getElementById('popup-images-enabled');
|
|
if (enabledCheckbox) {
|
|
enabledCheckbox.onchange = () => {
|
|
const config = this.popupImageManager.getConfig();
|
|
config.enabled = enabledCheckbox.checked;
|
|
this.popupImageManager.updateConfig(config);
|
|
this.updatePopupImagesInfo();
|
|
};
|
|
}
|
|
|
|
// Image count mode
|
|
const countModeSelect = document.getElementById('popup-count-mode');
|
|
if (countModeSelect) {
|
|
countModeSelect.onchange = () => {
|
|
this.updatePopupCountControls(countModeSelect.value);
|
|
const config = this.popupImageManager.getConfig();
|
|
config.imageCountMode = countModeSelect.value;
|
|
this.popupImageManager.updateConfig(config);
|
|
};
|
|
}
|
|
|
|
// Fixed count slider
|
|
const countSlider = document.getElementById('popup-image-count');
|
|
const countValue = document.getElementById('popup-image-count-value');
|
|
if (countSlider && countValue) {
|
|
countSlider.oninput = () => {
|
|
countValue.textContent = countSlider.value;
|
|
const config = this.popupImageManager.getConfig();
|
|
config.imageCount = parseInt(countSlider.value);
|
|
this.popupImageManager.updateConfig(config);
|
|
};
|
|
}
|
|
|
|
// Range count inputs
|
|
const minCountInput = document.getElementById('popup-min-count');
|
|
const maxCountInput = document.getElementById('popup-max-count');
|
|
if (minCountInput) {
|
|
minCountInput.onchange = () => {
|
|
const config = this.popupImageManager.getConfig();
|
|
config.minCount = parseInt(minCountInput.value);
|
|
this.popupImageManager.updateConfig(config);
|
|
};
|
|
}
|
|
if (maxCountInput) {
|
|
maxCountInput.onchange = () => {
|
|
const config = this.popupImageManager.getConfig();
|
|
config.maxCount = parseInt(maxCountInput.value);
|
|
this.popupImageManager.updateConfig(config);
|
|
};
|
|
}
|
|
|
|
// Duration mode
|
|
const durationModeSelect = document.getElementById('popup-duration-mode');
|
|
if (durationModeSelect) {
|
|
durationModeSelect.onchange = () => {
|
|
this.updatePopupDurationControls(durationModeSelect.value);
|
|
const config = this.popupImageManager.getConfig();
|
|
config.durationMode = durationModeSelect.value;
|
|
this.popupImageManager.updateConfig(config);
|
|
};
|
|
}
|
|
|
|
// Fixed duration slider
|
|
const durationSlider = document.getElementById('popup-display-duration');
|
|
const durationValue = document.getElementById('popup-display-duration-value');
|
|
if (durationSlider && durationValue) {
|
|
durationSlider.oninput = () => {
|
|
durationValue.textContent = durationSlider.value + 's';
|
|
const config = this.popupImageManager.getConfig();
|
|
config.displayDuration = parseInt(durationSlider.value) * 1000;
|
|
this.popupImageManager.updateConfig(config);
|
|
};
|
|
}
|
|
|
|
// Range duration inputs
|
|
const minDurationInput = document.getElementById('popup-min-duration');
|
|
const maxDurationInput = document.getElementById('popup-max-duration');
|
|
if (minDurationInput) {
|
|
minDurationInput.onchange = () => {
|
|
const config = this.popupImageManager.getConfig();
|
|
config.minDuration = parseInt(minDurationInput.value) * 1000;
|
|
this.popupImageManager.updateConfig(config);
|
|
};
|
|
}
|
|
if (maxDurationInput) {
|
|
maxDurationInput.onchange = () => {
|
|
const config = this.popupImageManager.getConfig();
|
|
config.maxDuration = parseInt(maxDurationInput.value) * 1000;
|
|
this.popupImageManager.updateConfig(config);
|
|
};
|
|
}
|
|
|
|
// Positioning
|
|
const positioningSelect = document.getElementById('popup-positioning');
|
|
if (positioningSelect) {
|
|
positioningSelect.onchange = () => {
|
|
const config = this.popupImageManager.getConfig();
|
|
config.positioning = positioningSelect.value;
|
|
this.popupImageManager.updateConfig(config);
|
|
};
|
|
}
|
|
|
|
// Visual effect checkboxes
|
|
const setupCheckbox = (id, configKey) => {
|
|
const checkbox = document.getElementById(id);
|
|
if (checkbox) {
|
|
checkbox.onchange = () => {
|
|
const config = this.popupImageManager.getConfig();
|
|
config[configKey] = checkbox.checked;
|
|
this.popupImageManager.updateConfig(config);
|
|
};
|
|
}
|
|
};
|
|
|
|
setupCheckbox('popup-allow-overlap', 'allowOverlap');
|
|
setupCheckbox('popup-fade-animation', 'fadeAnimation');
|
|
setupCheckbox('popup-blur-background', 'blurBackground');
|
|
setupCheckbox('popup-show-timer', 'showTimer');
|
|
setupCheckbox('popup-prevent-close', 'preventClose');
|
|
|
|
// Test buttons
|
|
const testSingleBtn = document.getElementById('test-popup-single');
|
|
if (testSingleBtn) {
|
|
testSingleBtn.onclick = () => {
|
|
this.popupImageManager.previewPunishmentPopups(1);
|
|
setTimeout(() => this.updatePopupImagesInfo(), 100);
|
|
};
|
|
}
|
|
|
|
const testMultipleBtn = document.getElementById('test-popup-multiple');
|
|
if (testMultipleBtn) {
|
|
testMultipleBtn.onclick = () => {
|
|
this.popupImageManager.triggerPunishmentPopups();
|
|
setTimeout(() => this.updatePopupImagesInfo(), 100);
|
|
};
|
|
}
|
|
|
|
const clearAllBtn = document.getElementById('clear-all-popups');
|
|
if (clearAllBtn) {
|
|
clearAllBtn.onclick = () => {
|
|
this.popupImageManager.clearAllPopups();
|
|
setTimeout(() => this.updatePopupImagesInfo(), 100);
|
|
};
|
|
}
|
|
|
|
// Size control listeners
|
|
const setupSizeSlider = (elementId, configKey, suffix = '') => {
|
|
const slider = document.getElementById(elementId);
|
|
const valueDisplay = document.getElementById(`${elementId}-value`);
|
|
if (slider && valueDisplay) {
|
|
slider.oninput = () => {
|
|
const value = parseInt(slider.value);
|
|
valueDisplay.textContent = value + suffix;
|
|
const config = this.popupImageManager.getConfig();
|
|
config[configKey] = configKey.includes('viewport') ? value / 100 : value;
|
|
this.popupImageManager.updateConfig(config);
|
|
};
|
|
}
|
|
};
|
|
|
|
const setupSizeInput = (elementId, configKey) => {
|
|
const input = document.getElementById(elementId);
|
|
if (input) {
|
|
input.onchange = () => {
|
|
const value = parseInt(input.value);
|
|
if (!isNaN(value)) {
|
|
const config = this.popupImageManager.getConfig();
|
|
config[configKey] = value;
|
|
this.popupImageManager.updateConfig(config);
|
|
}
|
|
};
|
|
}
|
|
};
|
|
|
|
setupSizeSlider('popup-viewport-width', 'viewportWidthRatio', '%');
|
|
setupSizeSlider('popup-viewport-height', 'viewportHeightRatio', '%');
|
|
setupSizeInput('popup-min-width', 'minWidth');
|
|
setupSizeInput('popup-max-width', 'maxWidth');
|
|
setupSizeInput('popup-min-height', 'minHeight');
|
|
setupSizeInput('popup-max-height', 'maxHeight');
|
|
};
|
|
|
|
TaskChallengeGame.prototype.updatePopupCountControls = function(mode) {
|
|
const fixedDiv = document.getElementById('popup-fixed-count');
|
|
const rangeDiv = document.getElementById('popup-range-count');
|
|
|
|
if (fixedDiv) fixedDiv.style.display = mode === 'fixed' ? 'block' : 'none';
|
|
if (rangeDiv) rangeDiv.style.display = mode === 'range' ? 'block' : 'none';
|
|
};
|
|
|
|
TaskChallengeGame.prototype.updatePopupDurationControls = function(mode) {
|
|
const fixedDiv = document.getElementById('popup-fixed-duration');
|
|
const rangeDiv = document.getElementById('popup-range-duration');
|
|
|
|
if (fixedDiv) fixedDiv.style.display = mode === 'fixed' ? 'block' : 'none';
|
|
if (rangeDiv) rangeDiv.style.display = mode === 'range' ? 'block' : 'none';
|
|
};
|
|
|
|
TaskChallengeGame.prototype.loadPopupImagesSettings = function() {
|
|
const config = this.popupImageManager.getConfig();
|
|
|
|
// Enable/disable
|
|
const enabledCheckbox = document.getElementById('popup-images-enabled');
|
|
if (enabledCheckbox) enabledCheckbox.checked = config.enabled;
|
|
|
|
// Count settings
|
|
const countModeSelect = document.getElementById('popup-count-mode');
|
|
if (countModeSelect) countModeSelect.value = config.imageCountMode;
|
|
|
|
const countSlider = document.getElementById('popup-image-count');
|
|
const countValue = document.getElementById('popup-image-count-value');
|
|
if (countSlider) countSlider.value = config.imageCount;
|
|
if (countValue) countValue.textContent = config.imageCount;
|
|
|
|
const minCountInput = document.getElementById('popup-min-count');
|
|
const maxCountInput = document.getElementById('popup-max-count');
|
|
if (minCountInput) minCountInput.value = config.minCount;
|
|
if (maxCountInput) maxCountInput.value = config.maxCount;
|
|
|
|
// Duration settings
|
|
const durationModeSelect = document.getElementById('popup-duration-mode');
|
|
if (durationModeSelect) durationModeSelect.value = config.durationMode;
|
|
|
|
const durationSlider = document.getElementById('popup-display-duration');
|
|
const durationValue = document.getElementById('popup-display-duration-value');
|
|
if (durationSlider) durationSlider.value = config.displayDuration / 1000;
|
|
if (durationValue) durationValue.textContent = (config.displayDuration / 1000) + 's';
|
|
|
|
const minDurationInput = document.getElementById('popup-min-duration');
|
|
const maxDurationInput = document.getElementById('popup-max-duration');
|
|
if (minDurationInput) minDurationInput.value = config.minDuration / 1000;
|
|
if (maxDurationInput) maxDurationInput.value = config.maxDuration / 1000;
|
|
|
|
// Positioning
|
|
const positioningSelect = document.getElementById('popup-positioning');
|
|
if (positioningSelect) positioningSelect.value = config.positioning;
|
|
|
|
// Visual effects
|
|
const checkboxes = {
|
|
'popup-allow-overlap': config.allowOverlap,
|
|
'popup-fade-animation': config.fadeAnimation,
|
|
'popup-blur-background': config.blurBackground,
|
|
'popup-show-timer': config.showTimer,
|
|
'popup-prevent-close': config.preventClose
|
|
};
|
|
|
|
Object.entries(checkboxes).forEach(([id, value]) => {
|
|
const checkbox = document.getElementById(id);
|
|
if (checkbox) checkbox.checked = value;
|
|
});
|
|
|
|
// Size settings
|
|
const viewportWidthSlider = document.getElementById('popup-viewport-width');
|
|
const viewportWidthValue = document.getElementById('popup-viewport-width-value');
|
|
if (viewportWidthSlider) viewportWidthSlider.value = (config.viewportWidthRatio || 0.35) * 100;
|
|
if (viewportWidthValue) viewportWidthValue.textContent = Math.round((config.viewportWidthRatio || 0.35) * 100) + '%';
|
|
|
|
const viewportHeightSlider = document.getElementById('popup-viewport-height');
|
|
const viewportHeightValue = document.getElementById('popup-viewport-height-value');
|
|
if (viewportHeightSlider) viewportHeightSlider.value = (config.viewportHeightRatio || 0.4) * 100;
|
|
if (viewportHeightValue) viewportHeightValue.textContent = Math.round((config.viewportHeightRatio || 0.4) * 100) + '%';
|
|
|
|
const sizeInputs = {
|
|
'popup-min-width': config.minWidth || 200,
|
|
'popup-max-width': config.maxWidth || 500,
|
|
'popup-min-height': config.minHeight || 150,
|
|
'popup-max-height': config.maxHeight || 400
|
|
};
|
|
|
|
Object.entries(sizeInputs).forEach(([id, value]) => {
|
|
const input = document.getElementById(id);
|
|
if (input) input.value = value;
|
|
});
|
|
|
|
// Update control visibility
|
|
this.updatePopupCountControls(config.imageCountMode);
|
|
this.updatePopupDurationControls(config.durationMode);
|
|
|
|
// Update info display
|
|
this.updatePopupImagesInfo();
|
|
};
|
|
|
|
TaskChallengeGame.prototype.updatePopupImagesInfo = function() {
|
|
const availableCountEl = document.getElementById('available-images-count');
|
|
const activeCountEl = document.getElementById('active-popups-count');
|
|
|
|
if (availableCountEl) {
|
|
const availableImages = this.popupImageManager.getAvailableImages();
|
|
availableCountEl.textContent = availableImages.length;
|
|
}
|
|
|
|
if (activeCountEl) {
|
|
const activeCount = this.popupImageManager.getActiveCount();
|
|
activeCountEl.textContent = activeCount;
|
|
}
|
|
};
|
|
|
|
// Import/Export Tab Management
|
|
TaskChallengeGame.prototype.setupImportExportTabListeners = function() {
|
|
const exportAllBtn = document.getElementById('export-all-messages-btn');
|
|
const exportEnabledBtn = document.getElementById('export-enabled-messages-btn');
|
|
const exportCustomBtn = document.getElementById('export-custom-messages-btn');
|
|
const importBtn = document.getElementById('import-messages-btn');
|
|
const importFile = document.getElementById('import-messages-file');
|
|
const resetDefaultsBtn = document.getElementById('reset-to-defaults-btn');
|
|
const clearAllBtn = document.getElementById('clear-all-messages-btn');
|
|
|
|
if (exportAllBtn) {
|
|
exportAllBtn.onclick = () => this.exportMessages('all');
|
|
}
|
|
if (exportEnabledBtn) {
|
|
exportEnabledBtn.onclick = () => this.exportMessages('enabled');
|
|
}
|
|
if (exportCustomBtn) {
|
|
exportCustomBtn.onclick = () => this.exportMessages('custom');
|
|
}
|
|
if (importBtn) {
|
|
importBtn.onclick = () => importFile.click();
|
|
}
|
|
if (importFile) {
|
|
importFile.onchange = (e) => this.handleMessageImport(e);
|
|
}
|
|
if (resetDefaultsBtn) {
|
|
resetDefaultsBtn.onclick = () => this.resetMessagesToDefaults();
|
|
}
|
|
if (clearAllBtn) {
|
|
clearAllBtn.onclick = () => this.clearAllMessages();
|
|
}
|
|
};
|
|
|
|
TaskChallengeGame.prototype.loadImportExportTab = function() {
|
|
// No specific loading needed for this tab
|
|
};
|
|
|
|
TaskChallengeGame.prototype.exportMessages = function(type) {
|
|
let includeDisabled = true;
|
|
let customOnly = false;
|
|
let filename = 'flash_messages_all.json';
|
|
|
|
switch(type) {
|
|
case 'enabled':
|
|
includeDisabled = false;
|
|
filename = 'flash_messages_enabled.json';
|
|
break;
|
|
case 'custom':
|
|
customOnly = true;
|
|
filename = 'flash_messages_custom.json';
|
|
break;
|
|
}
|
|
|
|
const exportData = this.flashMessageManager.exportMessages(includeDisabled, customOnly);
|
|
this.downloadFile(exportData, filename, 'application/json');
|
|
this.showNotification(`✅ Messages exported to ${filename}`, 'success');
|
|
};
|
|
|
|
TaskChallengeGame.prototype.handleMessageImport = function(event) {
|
|
const file = event.target.files[0];
|
|
if (!file) return;
|
|
|
|
const reader = new FileReader();
|
|
reader.onload = (e) => {
|
|
try {
|
|
const importMode = document.querySelector('input[name="importMode"]:checked').value;
|
|
const result = this.flashMessageManager.importMessages(e.target.result, importMode);
|
|
|
|
if (result.success) {
|
|
this.showNotification(`✅ Successfully imported ${result.imported} messages (${result.total} total)`, 'success');
|
|
if (this.getCurrentAnnoyanceTab() === 'messages') {
|
|
this.refreshMessageList();
|
|
}
|
|
} else {
|
|
this.showNotification(`❌ Import failed: ${result.error}`, 'error');
|
|
}
|
|
} catch (error) {
|
|
this.showNotification('❌ Invalid file format', 'error');
|
|
}
|
|
};
|
|
reader.readAsText(file);
|
|
|
|
// Clear the file input
|
|
event.target.value = '';
|
|
};
|
|
|
|
TaskChallengeGame.prototype.resetMessagesToDefaults = function() {
|
|
if (confirm('Reset to default messages? This will remove all custom messages and cannot be undone.')) {
|
|
const result = this.flashMessageManager.resetToDefaults();
|
|
this.showNotification(`✅ Reset to ${result.messages} default messages`, 'success');
|
|
if (this.getCurrentAnnoyanceTab() === 'messages') {
|
|
this.refreshMessageList();
|
|
}
|
|
}
|
|
};
|
|
|
|
TaskChallengeGame.prototype.clearAllMessages = function() {
|
|
if (confirm('Clear ALL messages? This will remove every message and cannot be undone. You will need to import messages to use the flash message system.')) {
|
|
this.flashMessageManager.updateMessages([]);
|
|
this.showNotification('⚠️ All messages cleared', 'warning');
|
|
if (this.getCurrentAnnoyanceTab() === 'messages') {
|
|
this.refreshMessageList();
|
|
}
|
|
}
|
|
};
|
|
|
|
// Utility Methods
|
|
TaskChallengeGame.prototype.updateFlashMessageSetting = function(setting, value) {
|
|
const currentConfig = this.flashMessageManager.getConfig();
|
|
const newConfig = { ...currentConfig, [setting]: value };
|
|
this.flashMessageManager.updateConfig(newConfig);
|
|
console.log(`Updated flash message setting: ${setting} = ${value}`);
|
|
};
|
|
|
|
TaskChallengeGame.prototype.saveAllAnnoyanceSettings = function() {
|
|
this.showNotification('✅ All annoyance settings saved!', 'success');
|
|
};
|
|
|
|
TaskChallengeGame.prototype.loadAnnoyanceSettings = function() {
|
|
// This method is called when the annoyance screen first loads
|
|
// Individual tabs will load their specific settings when shown
|
|
};
|
|
|
|
TaskChallengeGame.prototype.getCurrentAnnoyanceTab = function() {
|
|
const activeTab = document.querySelector('.annoyance-tab.active');
|
|
return activeTab ? activeTab.id.replace('-tab', '') : 'messages';
|
|
};
|
|
|
|
TaskChallengeGame.prototype.downloadFile = function(content, filename, mimeType) {
|
|
const blob = new Blob([content], { type: mimeType });
|
|
const url = URL.createObjectURL(blob);
|
|
const link = document.createElement('a');
|
|
link.href = url;
|
|
link.download = filename;
|
|
document.body.appendChild(link);
|
|
link.click();
|
|
document.body.removeChild(link);
|
|
URL.revokeObjectURL(url);
|
|
};
|
|
|
|
TaskChallengeGame.prototype.hexToRgba = function(hex, alpha) {
|
|
const r = parseInt(hex.slice(1, 3), 16);
|
|
const g = parseInt(hex.slice(3, 5), 16);
|
|
const b = parseInt(hex.slice(5, 7), 16);
|
|
return `rgba(${r}, ${g}, ${b}, ${alpha})`;
|
|
};
|
|
|
|
TaskChallengeGame.prototype.rgbToHex = function(r, g, b) {
|
|
return "#" + ((1 << 24) + (r << 16) + (g << 8) + b).toString(16).slice(1);
|
|
};
|
|
|
|
// Initialize game when page loads
|
|
document.addEventListener('DOMContentLoaded', () => {
|
|
window.game = new TaskChallengeGame();
|
|
}); |