mirror of
https://github.com/tiennm99/WPlace-AutoBOT.git
synced 2026-05-30 06:20:10 +00:00
8222 lines
312 KiB
JavaScript
8222 lines
312 KiB
JavaScript
// ==UserScript==
|
||
// @name WPlaceBot
|
||
// @namespace http://tampermonkey.net/
|
||
// @version 2025-08-08.3
|
||
// @description Bot
|
||
// @author Wbot
|
||
// @match https://wplace.live/*
|
||
// @grant none
|
||
// @icon
|
||
// ==/UserScript==
|
||
localStorage.removeItem("lp");
|
||
; (async () => {
|
||
// CONFIGURATION CONSTANTS
|
||
const CONFIG = {
|
||
COOLDOWN_DEFAULT: 31000,
|
||
TRANSPARENCY_THRESHOLD: 100,
|
||
WHITE_THRESHOLD: 250,
|
||
LOG_INTERVAL: 10,
|
||
PAINTING_SPEED: {
|
||
MIN: 1, // Minimum 1 pixel batch size
|
||
MAX: 1000, // Maximum 1000 pixels batch size
|
||
DEFAULT: 5, // Default 5 pixels batch size
|
||
},
|
||
BATCH_MODE: 'normal', // "normal" or "random" - default to normal
|
||
RANDOM_BATCH_RANGE: {
|
||
MIN: 3, // Random range minimum
|
||
MAX: 20, // Random range maximum
|
||
},
|
||
PAINTING_SPEED_ENABLED: true, // On by default
|
||
AUTO_CAPTCHA_ENABLED: true, // Turnstile generator enabled by default
|
||
TOKEN_SOURCE: 'generator', // "generator", "manual", or "hybrid" - default to generator
|
||
COOLDOWN_CHARGE_THRESHOLD: 1, // Default wait threshold
|
||
// Desktop Notifications (defaults)
|
||
NOTIFICATIONS: {
|
||
ENABLED: false,
|
||
ON_CHARGES_REACHED: true,
|
||
ONLY_WHEN_UNFOCUSED: true,
|
||
REPEAT_MINUTES: 5, // repeat reminder while threshold condition holds
|
||
},
|
||
OVERLAY: {
|
||
OPACITY_DEFAULT: 0.2,
|
||
BLUE_MARBLE_DEFAULT: false,
|
||
ditheringEnabled: false,
|
||
}, // --- START: Color data from colour-converter.js ---
|
||
// New color structure with proper ID mapping
|
||
COLOR_MAP: {
|
||
0: { id: 1, name: 'Black', rgb: { r: 0, g: 0, b: 0 } },
|
||
1: { id: 2, name: 'Dark Gray', rgb: { r: 60, g: 60, b: 60 } },
|
||
2: { id: 3, name: 'Gray', rgb: { r: 120, g: 120, b: 120 } },
|
||
3: { id: 4, name: 'Light Gray', rgb: { r: 210, g: 210, b: 210 } },
|
||
4: { id: 5, name: 'White', rgb: { r: 255, g: 255, b: 255 } },
|
||
5: { id: 6, name: 'Deep Red', rgb: { r: 96, g: 0, b: 24 } },
|
||
6: { id: 7, name: 'Red', rgb: { r: 237, g: 28, b: 36 } },
|
||
7: { id: 8, name: 'Orange', rgb: { r: 255, g: 127, b: 39 } },
|
||
8: { id: 9, name: 'Gold', rgb: { r: 246, g: 170, b: 9 } },
|
||
9: { id: 10, name: 'Yellow', rgb: { r: 249, g: 221, b: 59 } },
|
||
10: { id: 11, name: 'Light Yellow', rgb: { r: 255, g: 250, b: 188 } },
|
||
11: { id: 12, name: 'Dark Green', rgb: { r: 14, g: 185, b: 104 } },
|
||
12: { id: 13, name: 'Green', rgb: { r: 19, g: 230, b: 123 } },
|
||
13: { id: 14, name: 'Light Green', rgb: { r: 135, g: 255, b: 94 } },
|
||
14: { id: 15, name: 'Dark Teal', rgb: { r: 12, g: 129, b: 110 } },
|
||
15: { id: 16, name: 'Teal', rgb: { r: 16, g: 174, b: 166 } },
|
||
16: { id: 17, name: 'Light Teal', rgb: { r: 19, g: 225, b: 190 } },
|
||
17: { id: 20, name: 'Cyan', rgb: { r: 96, g: 247, b: 242 } },
|
||
18: { id: 44, name: 'Light Cyan', rgb: { r: 187, g: 250, b: 242 } },
|
||
19: { id: 18, name: 'Dark Blue', rgb: { r: 40, g: 80, b: 158 } },
|
||
20: { id: 19, name: 'Blue', rgb: { r: 64, g: 147, b: 228 } },
|
||
21: { id: 21, name: 'Indigo', rgb: { r: 107, g: 80, b: 246 } },
|
||
22: { id: 22, name: 'Light Indigo', rgb: { r: 153, g: 177, b: 251 } },
|
||
23: { id: 23, name: 'Dark Purple', rgb: { r: 120, g: 12, b: 153 } },
|
||
24: { id: 24, name: 'Purple', rgb: { r: 170, g: 56, b: 185 } },
|
||
25: { id: 25, name: 'Light Purple', rgb: { r: 224, g: 159, b: 249 } },
|
||
26: { id: 26, name: 'Dark Pink', rgb: { r: 203, g: 0, b: 122 } },
|
||
27: { id: 27, name: 'Pink', rgb: { r: 236, g: 31, b: 128 } },
|
||
28: { id: 28, name: 'Light Pink', rgb: { r: 243, g: 141, b: 169 } },
|
||
29: { id: 29, name: 'Dark Brown', rgb: { r: 104, g: 70, b: 52 } },
|
||
30: { id: 30, name: 'Brown', rgb: { r: 149, g: 104, b: 42 } },
|
||
31: { id: 31, name: 'Beige', rgb: { r: 248, g: 178, b: 119 } },
|
||
32: { id: 52, name: 'Light Beige', rgb: { r: 255, g: 197, b: 165 } },
|
||
33: { id: 32, name: 'Medium Gray', rgb: { r: 170, g: 170, b: 170 } },
|
||
34: { id: 33, name: 'Dark Red', rgb: { r: 165, g: 14, b: 30 } },
|
||
35: { id: 34, name: 'Light Red', rgb: { r: 250, g: 128, b: 114 } },
|
||
36: { id: 35, name: 'Dark Orange', rgb: { r: 228, g: 92, b: 26 } },
|
||
37: { id: 37, name: 'Dark Goldenrod', rgb: { r: 156, g: 132, b: 49 } },
|
||
38: { id: 38, name: 'Goldenrod', rgb: { r: 197, g: 173, b: 49 } },
|
||
39: { id: 39, name: 'Light Goldenrod', rgb: { r: 232, g: 212, b: 95 } },
|
||
40: { id: 40, name: 'Dark Olive', rgb: { r: 74, g: 107, b: 58 } },
|
||
41: { id: 41, name: 'Olive', rgb: { r: 90, g: 148, b: 74 } },
|
||
42: { id: 42, name: 'Light Olive', rgb: { r: 132, g: 197, b: 115 } },
|
||
43: { id: 43, name: 'Dark Cyan', rgb: { r: 15, g: 121, b: 159 } },
|
||
44: { id: 45, name: 'Light Blue', rgb: { r: 125, g: 199, b: 255 } },
|
||
45: { id: 46, name: 'Dark Indigo', rgb: { r: 77, g: 49, b: 184 } },
|
||
46: { id: 47, name: 'Dark Slate Blue', rgb: { r: 74, g: 66, b: 132 } },
|
||
47: { id: 48, name: 'Slate Blue', rgb: { r: 122, g: 113, b: 196 } },
|
||
48: { id: 49, name: 'Light Slate Blue', rgb: { r: 181, g: 174, b: 241 } },
|
||
49: { id: 53, name: 'Dark Peach', rgb: { r: 155, g: 82, b: 73 } },
|
||
50: { id: 54, name: 'Peach', rgb: { r: 209, g: 128, b: 120 } },
|
||
51: { id: 55, name: 'Light Peach', rgb: { r: 250, g: 182, b: 164 } },
|
||
52: { id: 50, name: 'Light Brown', rgb: { r: 219, g: 164, b: 99 } },
|
||
53: { id: 56, name: 'Dark Tan', rgb: { r: 123, g: 99, b: 82 } },
|
||
54: { id: 57, name: 'Tan', rgb: { r: 156, g: 132, b: 107 } },
|
||
55: { id: 36, name: 'Light Tan', rgb: { r: 214, g: 181, b: 148 } },
|
||
56: { id: 51, name: 'Dark Beige', rgb: { r: 209, g: 128, b: 81 } },
|
||
57: { id: 61, name: 'Dark Stone', rgb: { r: 109, g: 100, b: 63 } },
|
||
58: { id: 62, name: 'Stone', rgb: { r: 148, g: 140, b: 107 } },
|
||
59: { id: 63, name: 'Light Stone', rgb: { r: 205, g: 197, b: 158 } },
|
||
60: { id: 58, name: 'Dark Slate', rgb: { r: 51, g: 57, b: 65 } },
|
||
61: { id: 59, name: 'Slate', rgb: { r: 109, g: 117, b: 141 } },
|
||
62: { id: 60, name: 'Light Slate', rgb: { r: 179, g: 185, b: 209 } },
|
||
63: { id: 0, name: 'Transparent', rgb: null },
|
||
}, // --- END: Color data ---
|
||
// Optimized CSS Classes for reuse
|
||
CSS_CLASSES: {
|
||
BUTTON_PRIMARY: `
|
||
background: linear-gradient(135deg, #4facfe 0%, #00f2fe 100%);
|
||
color: white; border: none; border-radius: 8px; padding: 10px 16px;
|
||
cursor: pointer; font-weight: 500; transition: all 0.3s ease;
|
||
display: flex; align-items: center; gap: 8px;
|
||
`,
|
||
BUTTON_SECONDARY: `
|
||
background: rgba(255,255,255,0.1); color: white;
|
||
border: 1px solid rgba(255,255,255,0.2); border-radius: 8px;
|
||
padding: 8px 12px; cursor: pointer; transition: all 0.3s ease;
|
||
`,
|
||
MODERN_CARD: `
|
||
background: rgba(255,255,255,0.1); border-radius: 12px;
|
||
padding: 18px; border: 1px solid rgba(255,255,255,0.1);
|
||
backdrop-filter: blur(5px);
|
||
`,
|
||
GRADIENT_TEXT: `
|
||
background: linear-gradient(135deg, #4facfe 0%, #00f2fe 100%);
|
||
-webkit-background-clip: text; -webkit-text-fill-color: transparent;
|
||
background-clip: text; font-weight: bold;
|
||
`,
|
||
},
|
||
THEMES: {
|
||
'Classic Autobot': {
|
||
primary: '#000000',
|
||
secondary: '#111111',
|
||
accent: '#222222',
|
||
text: '#ffffff',
|
||
highlight: '#775ce3',
|
||
success: '#00ff00',
|
||
error: '#ff0000',
|
||
warning: '#ffaa00',
|
||
fontFamily: "'Segoe UI', Roboto, sans-serif",
|
||
borderRadius: '12px',
|
||
borderStyle: 'solid',
|
||
borderWidth: '1px',
|
||
boxShadow: '0 8px 32px rgba(0,0,0,0.6), 0 0 0 1px rgba(255,255,255,0.1)',
|
||
backdropFilter: 'blur(10px)',
|
||
animations: {
|
||
glow: false,
|
||
scanline: false,
|
||
'pixel-blink': false,
|
||
},
|
||
},
|
||
'Classic Light': {
|
||
primary: '#ffffff',
|
||
secondary: '#f8f9fa',
|
||
accent: '#e9ecef',
|
||
text: '#212529',
|
||
highlight: '#6f42c1',
|
||
success: '#28a745',
|
||
error: '#dc3545',
|
||
warning: '#ffc107',
|
||
fontFamily: "'Segoe UI', Roboto, sans-serif",
|
||
borderRadius: '12px',
|
||
borderStyle: 'solid',
|
||
borderWidth: '1px',
|
||
boxShadow: '0 8px 32px rgba(0,0,0,0.15), 0 0 0 1px rgba(0,0,0,0.08)',
|
||
backdropFilter: 'blur(10px)',
|
||
animations: {
|
||
glow: false,
|
||
scanline: false,
|
||
'pixel-blink': false,
|
||
},
|
||
},
|
||
'Neon Retro': {
|
||
primary: '#1a1a2e',
|
||
secondary: '#16213e',
|
||
accent: '#0f3460',
|
||
text: '#00ff41',
|
||
highlight: '#ff6b35',
|
||
success: '#39ff14',
|
||
error: '#ff073a',
|
||
warning: '#ffff00',
|
||
neon: '#00ffff',
|
||
purple: '#bf00ff',
|
||
pink: '#ff1493',
|
||
fontFamily: "'Press Start 2P', monospace",
|
||
borderRadius: '0',
|
||
borderStyle: 'solid',
|
||
borderWidth: '3px',
|
||
boxShadow: '0 0 20px rgba(0, 255, 65, 0.3), inset 0 0 20px rgba(0, 255, 65, 0.1)',
|
||
backdropFilter: 'none',
|
||
animations: {
|
||
glow: true,
|
||
scanline: true,
|
||
'pixel-blink': true,
|
||
},
|
||
},
|
||
'Acrylic': {
|
||
primary: '#00000080',
|
||
secondary: '#00000040',
|
||
accent: 'rgba(0,0,0,0.75)',
|
||
text: '#ffffff',
|
||
highlight: '#ffffff',
|
||
success: '#00e500',
|
||
error: '#e50000',
|
||
warning: '#e5e500',
|
||
fontFamily: "'Inter', 'Apple Color Emoji'",
|
||
borderRadius: '10px',
|
||
borderStyle: 'solid',
|
||
borderWidth: '0px',
|
||
boxShadow: 'none',
|
||
backdropFilter: 'blur(20px)',
|
||
animations: {
|
||
glow: false,
|
||
scanline: false,
|
||
'pixel-blink': false,
|
||
},
|
||
},
|
||
},
|
||
currentTheme: 'Classic Autobot',
|
||
PAINT_UNAVAILABLE: true,
|
||
COORDINATE_MODE: 'rows',
|
||
COORDINATE_DIRECTION: 'top-left',
|
||
COORDINATE_SNAKE: true,
|
||
COORDINATE_BLOCK_WIDTH: 6,
|
||
COORDINATE_BLOCK_HEIGHT: 2,
|
||
};
|
||
|
||
const getCurrentTheme = () => CONFIG.THEMES[CONFIG.currentTheme];
|
||
|
||
const switchTheme = (themeName) => {
|
||
if (CONFIG.THEMES[themeName]) {
|
||
CONFIG.currentTheme = themeName;
|
||
saveThemePreference();
|
||
|
||
// APPLY THEME VARS/CLASS (new)
|
||
applyTheme();
|
||
|
||
// Recreate UI (kept for now)
|
||
createUI();
|
||
}
|
||
};
|
||
|
||
// Add this helper (place it after getCurrentTheme/switchTheme definitions)
|
||
function applyTheme() {
|
||
const theme = getCurrentTheme();
|
||
console.group('%c🎨 Applying Theme in Auto-Image Script', 'color: #8b5cf6; font-weight: bold;');
|
||
console.log(`%c🎯 Target theme: ${CONFIG.currentTheme}`, 'color: #8b5cf6;');
|
||
|
||
// Toggle theme class on documentElement so CSS vars cascade to our UI
|
||
document.documentElement.classList.remove(
|
||
'wplace-theme-classic',
|
||
'wplace-theme-classic-light',
|
||
'wplace-theme-acrylic',
|
||
'wplace-theme-neon'
|
||
);
|
||
|
||
let themeClass = 'wplace-theme-classic'; // default
|
||
let themeFileName = 'classic'; // corresponding file name
|
||
|
||
if (CONFIG.currentTheme === 'Neon Retro') {
|
||
themeClass = 'wplace-theme-neon';
|
||
themeFileName = 'neon';
|
||
} else if (CONFIG.currentTheme === 'Classic Light') {
|
||
themeClass = 'wplace-theme-classic-light';
|
||
themeFileName = 'classic-light';
|
||
} else if (CONFIG.currentTheme === 'Acrylic') {
|
||
themeClass = 'wplace-theme-acrylic';
|
||
themeFileName = 'acrylic';
|
||
}
|
||
|
||
document.documentElement.classList.add(themeClass);
|
||
console.log(`%c✅ Applied theme class: ${themeClass}`, 'color: #8b5cf6;');
|
||
|
||
// Use extension's applyTheme helper if available (loads from local extension resources)
|
||
if (typeof window.applyTheme === 'function') {
|
||
console.log(`%c🔧 Using extension's applyTheme() helper for: ${themeFileName}`, 'color: #10b981; font-weight: bold;');
|
||
const success = window.applyTheme(themeFileName);
|
||
if (success) {
|
||
console.log(`%c✅ Theme CSS loaded from extension local resources`, 'color: #10b981; font-weight: bold;');
|
||
console.log(` 📍 Source: Extension local file (themes/${themeFileName}.css)`);
|
||
console.log(` 🚀 Performance: Instant load (no network request)`);
|
||
} else {
|
||
console.warn(`%c⚠️ Extension theme loading failed, fallback to CSS classes only`, 'color: #f59e0b;');
|
||
console.log(` 🎨 Using CSS class: ${themeClass}`);
|
||
console.log(` 📝 Note: Theme variables may be limited without CSS file`);
|
||
}
|
||
} else {
|
||
console.log(`%c📝 Extension applyTheme() not available, using CSS class only`, 'color: #f59e0b;');
|
||
console.log(` 🎨 Using CSS class: ${themeClass}`);
|
||
console.log(` ⚠️ Note: Full theme styling requires extension helper function`);
|
||
}
|
||
|
||
// Also set CSS variables explicitly in case you want runtime overrides
|
||
const root = document.documentElement;
|
||
const setVar = (k, v) => {
|
||
try {
|
||
root.style.setProperty(k, v);
|
||
} catch { }
|
||
};
|
||
|
||
setVar('--wplace-primary', theme.primary);
|
||
setVar('--wplace-secondary', theme.secondary);
|
||
setVar('--wplace-accent', theme.accent);
|
||
setVar('--wplace-text', theme.text);
|
||
setVar('--wplace-highlight', theme.highlight);
|
||
setVar('--wplace-success', theme.success);
|
||
setVar('--wplace-error', theme.error);
|
||
setVar('--wplace-warning', theme.warning);
|
||
|
||
// Typography + look
|
||
setVar('--wplace-font', theme.fontFamily || "'Segoe UI', Roboto, sans-serif");
|
||
setVar('--wplace-radius', '' + (theme.borderRadius || '12px'));
|
||
setVar('--wplace-border-style', '' + (theme.borderStyle || 'solid'));
|
||
setVar('--wplace-border-width', '' + (theme.borderWidth || '1px'));
|
||
setVar('--wplace-backdrop', '' + (theme.backdropFilter || 'blur(10px)'));
|
||
setVar('--wplace-border-color', 'rgba(255,255,255,0.1)');
|
||
|
||
console.log(`%c🎨 Theme application complete`, 'color: #8b5cf6; font-weight: bold;');
|
||
console.groupEnd();
|
||
}
|
||
|
||
const saveThemePreference = () => {
|
||
try {
|
||
localStorage.setItem('wplace-theme', CONFIG.currentTheme);
|
||
} catch (e) {
|
||
console.warn('Could not save theme preference:', e);
|
||
}
|
||
};
|
||
|
||
const loadThemePreference = () => {
|
||
try {
|
||
const saved = localStorage.getItem('wplace-theme');
|
||
if (saved && CONFIG.THEMES[saved]) {
|
||
CONFIG.currentTheme = saved;
|
||
}
|
||
} catch (e) {
|
||
console.warn('Could not load theme preference:', e);
|
||
}
|
||
};
|
||
|
||
// Simple translation cache
|
||
const translationCache = new Map();
|
||
|
||
// Dynamically loaded translations
|
||
let loadedTranslations = {};
|
||
|
||
// Available languages
|
||
const AVAILABLE_LANGUAGES = [
|
||
'en',
|
||
'ru',
|
||
'pt',
|
||
'vi',
|
||
'fr',
|
||
'id',
|
||
'tr',
|
||
'zh-CN',
|
||
'zh-TW',
|
||
'ja',
|
||
'ko',
|
||
'uk',
|
||
];
|
||
|
||
// Function to load translations from JSON file with retry mechanism
|
||
const loadTranslations = async (language, retryCount = 0) => {
|
||
if (loadedTranslations[language]) {
|
||
return loadedTranslations[language];
|
||
}
|
||
|
||
console.group(`%c🌍 Loading ${language.toUpperCase()} translations`, 'color: #06b6d4; font-weight: bold;');
|
||
|
||
// First try: Check if extension has loaded local resources
|
||
if (window.AUTOBOT_LANGUAGES && Object.keys(window.AUTOBOT_LANGUAGES).length > 0) {
|
||
console.log(`%c🔍 Checking extension local resources...`, 'color: #06b6d4;');
|
||
|
||
const langFile = language + '.json';
|
||
if (window.AUTOBOT_LANGUAGES[langFile]) {
|
||
const translations = window.AUTOBOT_LANGUAGES[langFile];
|
||
|
||
// Validate that translations is an object with keys
|
||
if (
|
||
typeof translations === 'object' &&
|
||
translations !== null &&
|
||
Object.keys(translations).length > 0
|
||
) {
|
||
loadedTranslations[language] = translations;
|
||
console.log(`%c✅ Loaded ${language} translations from EXTENSION LOCAL FILES`, 'color: #10b981; font-weight: bold;');
|
||
console.log(` 📍 Source: Extension local storage (chrome-extension://)`);
|
||
console.log(` 📏 Keys count: ${Object.keys(translations).length}`);
|
||
console.log(` 🚀 Performance: Instant load (no network request)`);
|
||
console.groupEnd();
|
||
return translations;
|
||
}
|
||
} else {
|
||
console.log(`%c📝 ${langFile} not found in extension resources`, 'color: #f59e0b;');
|
||
console.log(` 📋 Available in extension: ${Object.keys(window.AUTOBOT_LANGUAGES).join(', ')}`);
|
||
}
|
||
} else {
|
||
console.log(`%c📝 No extension local resources available`, 'color: #f59e0b;');
|
||
console.log(` 🔍 window.AUTOBOT_LANGUAGES: ${typeof window.AUTOBOT_LANGUAGES}`);
|
||
}
|
||
|
||
// Second try: Use helper function if available
|
||
if (typeof window.getLanguage === 'function') {
|
||
console.log(`%c🔧 Trying extension getLanguage() helper...`, 'color: #06b6d4;');
|
||
try {
|
||
const translations = window.getLanguage(language);
|
||
if (
|
||
typeof translations === 'object' &&
|
||
translations !== null &&
|
||
Object.keys(translations).length > 0
|
||
) {
|
||
loadedTranslations[language] = translations;
|
||
console.log(`%c✅ Loaded ${language} translations via extension helper`, 'color: #10b981; font-weight: bold;');
|
||
console.log(` 📍 Source: Extension getLanguage() function`);
|
||
console.log(` 📏 Keys count: ${Object.keys(translations).length}`);
|
||
console.groupEnd();
|
||
return translations;
|
||
}
|
||
} catch (error) {
|
||
console.warn(`%c⚠️ Extension helper failed:`, 'color: #f59e0b;', error);
|
||
}
|
||
}
|
||
|
||
// Fallback: Load from CDN (original behavior)
|
||
console.log(`%c🌐 Falling back to CDN loading...`, 'color: #8b5cf6;');
|
||
const url = `https://wplace-autobot.github.io/WPlace-AutoBOT/main/lang/${language}.json`;
|
||
const maxRetries = 3;
|
||
const baseDelay = 1000; // 1 second
|
||
|
||
try {
|
||
if (retryCount === 0) {
|
||
console.log(`🔄 Loading ${language} translations from CDN...`);
|
||
} else {
|
||
console.log(
|
||
`🔄 Retrying ${language} translations (attempt ${retryCount + 1}/${maxRetries + 1})...`
|
||
);
|
||
}
|
||
|
||
const response = await fetch(url);
|
||
if (response.ok) {
|
||
const translations = await response.json();
|
||
|
||
// Validate that translations is an object with keys
|
||
if (
|
||
typeof translations === 'object' &&
|
||
translations !== null &&
|
||
Object.keys(translations).length > 0
|
||
) {
|
||
loadedTranslations[language] = translations;
|
||
console.log(
|
||
`%c📚 Loaded ${language} translations from CDN (${Object.keys(translations).length} keys)`, 'color: #f59e0b; font-weight: bold;'
|
||
);
|
||
console.log(` 📍 Source: CDN (wplace-autobot.github.io)`);
|
||
console.log(` 🌐 URL: ${url}`);
|
||
console.log(` ⚠️ Performance: Network request required`);
|
||
console.groupEnd();
|
||
return translations;
|
||
} else {
|
||
console.warn(`❌ Invalid translation format for ${language}`);
|
||
throw new Error('Invalid translation format');
|
||
}
|
||
} else {
|
||
console.warn(
|
||
`❌ CDN returned HTTP ${response.status}: ${response.statusText} for ${language} translations`
|
||
);
|
||
throw new Error(`HTTP ${response.status}: ${response.statusText}`);
|
||
}
|
||
} catch (error) {
|
||
console.error(
|
||
`❌ Failed to load ${language} translations from CDN (attempt ${retryCount + 1}):`,
|
||
error
|
||
);
|
||
|
||
// Retry with exponential backoff
|
||
if (retryCount < maxRetries) {
|
||
const delay = baseDelay * Math.pow(2, retryCount);
|
||
console.log(`⏳ Retrying in ${delay}ms...`);
|
||
await new Promise((resolve) => setTimeout(resolve, delay));
|
||
console.groupEnd();
|
||
return loadTranslations(language, retryCount + 1);
|
||
}
|
||
}
|
||
|
||
console.error(`%c💥 All translation loading methods failed for ${language}`, 'color: #ef4444; font-weight: bold;');
|
||
console.groupEnd();
|
||
return null;
|
||
};
|
||
|
||
const loadLanguagePreference = async () => {
|
||
const savedLanguage = localStorage.getItem('wplace_language');
|
||
const browserLocale = navigator.language;
|
||
const browserLanguage = browserLocale.split('-')[0];
|
||
|
||
let selectedLanguage = 'en'; // Default fallback
|
||
|
||
try {
|
||
// Check if we have the saved language available
|
||
if (savedLanguage && AVAILABLE_LANGUAGES.includes(savedLanguage)) {
|
||
selectedLanguage = savedLanguage;
|
||
console.log(`🔄 Using saved language preference: ${selectedLanguage}`);
|
||
}
|
||
// Try full locale match (e.g. "zh-CN", "zh-TW" etc)
|
||
else if (AVAILABLE_LANGUAGES.includes(browserLocale)) {
|
||
selectedLanguage = browserLocale;
|
||
localStorage.setItem('wplace_language', browserLocale);
|
||
console.log(`🔄 Using browser locale: ${selectedLanguage}`);
|
||
}
|
||
// Try base language match (e.g. "en" for "en-US" or "en-GB" etc)
|
||
else if (AVAILABLE_LANGUAGES.includes(browserLanguage)) {
|
||
selectedLanguage = browserLanguage;
|
||
localStorage.setItem('wplace_language', browserLanguage);
|
||
console.log(`🔄 Using browser language: ${selectedLanguage}`);
|
||
}
|
||
// Use English as fallback
|
||
else {
|
||
console.log(`🔄 No matching language found, using English fallback`);
|
||
}
|
||
|
||
// Set the language in state first
|
||
state.language = selectedLanguage;
|
||
|
||
// Only load translations if not already loaded and not English (which should already be loaded)
|
||
if (selectedLanguage !== 'en' && !loadedTranslations[selectedLanguage]) {
|
||
const loaded = await loadTranslations(selectedLanguage);
|
||
if (!loaded) {
|
||
console.warn(
|
||
`⚠️ Failed to load ${selectedLanguage} translations, falling back to English`
|
||
);
|
||
state.language = 'en';
|
||
localStorage.setItem('wplace_language', 'en');
|
||
}
|
||
}
|
||
} catch (error) {
|
||
console.error(`❌ Error in loadLanguagePreference:`, error);
|
||
state.language = 'en'; // Always ensure we have a valid language
|
||
}
|
||
};
|
||
|
||
// Simple user notification function for critical issues
|
||
const showTranslationWarning = (message) => {
|
||
try {
|
||
// Create a simple temporary notification banner
|
||
const warning = document.createElement('div');
|
||
warning.style.cssText = `
|
||
position: fixed; top: 10px; right: 10px; z-index: 10001;
|
||
background: rgba(255, 193, 7, 0.95); color: #212529; padding: 12px 16px;
|
||
border-radius: 8px; font-size: 14px; font-weight: 500;
|
||
box-shadow: 0 4px 12px rgba(0,0,0,0.3); border: 1px solid rgba(255, 193, 7, 0.8);
|
||
max-width: 300px; word-wrap: break-word;
|
||
`;
|
||
warning.textContent = message;
|
||
document.body.appendChild(warning);
|
||
|
||
// Auto-remove after 8 seconds
|
||
setTimeout(() => {
|
||
if (warning.parentNode) {
|
||
warning.remove();
|
||
}
|
||
}, 8000);
|
||
} catch (e) {
|
||
// If DOM manipulation fails, just log
|
||
console.warn('Failed to show translation warning UI:', e);
|
||
}
|
||
};
|
||
|
||
// Initialize translations function
|
||
const initializeTranslations = async () => {
|
||
try {
|
||
console.log('🌐 Initializing translation system...');
|
||
|
||
// Always ensure English is loaded as fallback first
|
||
if (!loadedTranslations['en']) {
|
||
const englishLoaded = await loadTranslations('en');
|
||
if (!englishLoaded) {
|
||
console.warn('⚠️ Failed to load English translations from CDN, using fallback');
|
||
showTranslationWarning('⚠️ Translation loading failed, using basic fallbacks');
|
||
}
|
||
}
|
||
|
||
// Then load user's language preference
|
||
await loadLanguagePreference();
|
||
|
||
console.log(`✅ Translation system initialized. Active language: ${state.language}`);
|
||
} catch (error) {
|
||
console.error('❌ Translation initialization failed:', error);
|
||
// Ensure state has a valid language even if loading fails
|
||
if (!state.language) {
|
||
state.language = 'en';
|
||
}
|
||
console.warn('⚠️ Using fallback translations due to initialization failure');
|
||
showTranslationWarning('⚠️ Translation system error, using basic English');
|
||
}
|
||
};
|
||
|
||
// Emergency fallback TEXT (minimal)
|
||
const FALLBACK_TEXT = {
|
||
en: {
|
||
title: 'WPlace Auto-Image',
|
||
toggleOverlay: 'Toggle Overlay',
|
||
scanColors: 'Scan Colors',
|
||
uploadImage: 'Upload Image',
|
||
resizeImage: 'Resize Image',
|
||
selectPosition: 'Select Position',
|
||
startPainting: 'Start Painting',
|
||
stopPainting: 'Stop Painting',
|
||
progress: 'Progress',
|
||
pixels: 'Pixels',
|
||
charges: 'Charges',
|
||
batchSize: 'Batch Size',
|
||
cooldownSettings: 'Cooldown Settings',
|
||
waitCharges: 'Wait for Charges',
|
||
settings: 'Settings',
|
||
showStats: 'Show Statistics',
|
||
compactMode: 'Compact Mode',
|
||
minimize: 'Minimize',
|
||
tokenCapturedSuccess: 'Token captured successfully',
|
||
turnstileInstructions: 'Complete the verification',
|
||
hideTurnstileBtn: 'Hide',
|
||
notificationsNotSupported: 'Notifications not supported',
|
||
chargesReadyMessage: 'Charges are ready',
|
||
chargesReadyNotification: 'WPlace AutoBot',
|
||
initMessage: "Click 'Upload Image' to begin",
|
||
},
|
||
};
|
||
|
||
// Safe translation function with multiple fallback levels
|
||
const getText = (key, replacements = {}) => {
|
||
// Try current language first
|
||
let text = loadedTranslations[state.language]?.[key];
|
||
|
||
// Fallback to English translations
|
||
if (!text && state.language !== 'en') {
|
||
text = loadedTranslations['en']?.[key];
|
||
}
|
||
|
||
// Fallback to hardcoded English
|
||
if (!text) {
|
||
text = FALLBACK_TEXT['en']?.[key];
|
||
}
|
||
|
||
// Last resort - return the key itself
|
||
if (!text) {
|
||
console.warn(`⚠️ Missing translation for key: ${key}`);
|
||
return key;
|
||
}
|
||
|
||
// Handle string replacements like {count}, {time}, etc.
|
||
return Object.entries(replacements).reduce((result, [placeholder, value]) => {
|
||
return result.replace(new RegExp(`\\{${placeholder}\\}`, 'g'), value);
|
||
}, text);
|
||
};
|
||
|
||
// GLOBAL STATE
|
||
const state = {
|
||
running: false,
|
||
imageLoaded: false,
|
||
processing: false,
|
||
totalPixels: 0,
|
||
paintedPixels: 0,
|
||
availableColors: [],
|
||
activeColorPalette: [], // User-selected colors for conversion
|
||
paintWhitePixels: true, // Default to ON
|
||
fullChargeData: null,
|
||
fullChargeInterval: null,
|
||
paintTransparentPixels: false, // Default to OFF
|
||
displayCharges: 0,
|
||
preciseCurrentCharges: 0,
|
||
maxCharges: 1, // Default max charges
|
||
cooldown: CONFIG.COOLDOWN_DEFAULT,
|
||
imageData: null,
|
||
stopFlag: false,
|
||
colorsChecked: false,
|
||
startPosition: null,
|
||
selectingPosition: false,
|
||
region: null,
|
||
minimized: false,
|
||
lastPosition: { x: 0, y: 0 },
|
||
estimatedTime: 0,
|
||
language: 'en',
|
||
paintingSpeed: CONFIG.PAINTING_SPEED.DEFAULT, // pixels batch size
|
||
batchMode: CONFIG.BATCH_MODE, // "normal" or "random"
|
||
randomBatchMin: CONFIG.RANDOM_BATCH_RANGE.MIN, // Random range minimum
|
||
randomBatchMax: CONFIG.RANDOM_BATCH_RANGE.MAX, // Random range maximum
|
||
cooldownChargeThreshold: CONFIG.COOLDOWN_CHARGE_THRESHOLD,
|
||
chargesThresholdInterval: null,
|
||
tokenSource: CONFIG.TOKEN_SOURCE, // "generator" or "manual"
|
||
initialSetupComplete: false, // Track if initial startup setup is complete (only happens once)
|
||
overlayOpacity: CONFIG.OVERLAY.OPACITY_DEFAULT,
|
||
blueMarbleEnabled: CONFIG.OVERLAY.BLUE_MARBLE_DEFAULT,
|
||
ditheringEnabled: true,
|
||
// Advanced color matching settings
|
||
colorMatchingAlgorithm: 'lab',
|
||
enableChromaPenalty: true,
|
||
chromaPenaltyWeight: 0.15,
|
||
customTransparencyThreshold: CONFIG.TRANSPARENCY_THRESHOLD,
|
||
customWhiteThreshold: CONFIG.WHITE_THRESHOLD,
|
||
resizeSettings: null,
|
||
originalImage: null,
|
||
resizeIgnoreMask: null,
|
||
paintUnavailablePixels: CONFIG.PAINT_UNAVAILABLE,
|
||
// Coordinate generation settings
|
||
coordinateMode: CONFIG.COORDINATE_MODE,
|
||
coordinateDirection: CONFIG.COORDINATE_DIRECTION,
|
||
coordinateSnake: CONFIG.COORDINATE_SNAKE,
|
||
blockWidth: CONFIG.COORDINATE_BLOCK_WIDTH,
|
||
blockHeight: CONFIG.COORDINATE_BLOCK_HEIGHT,
|
||
notificationsEnabled: CONFIG.NOTIFICATIONS.ENABLED,
|
||
notifyOnChargesReached: CONFIG.NOTIFICATIONS.ON_CHARGES_REACHED,
|
||
notifyOnlyWhenUnfocused: CONFIG.NOTIFICATIONS.ONLY_WHEN_UNFOCUSED,
|
||
notificationIntervalMinutes: CONFIG.NOTIFICATIONS.REPEAT_MINUTES,
|
||
_lastChargesNotifyAt: 0,
|
||
_lastChargesBelow: true,
|
||
// Smart save tracking
|
||
_lastSavePixelCount: 0,
|
||
_lastSaveTime: 0,
|
||
_saveInProgress: false,
|
||
paintedMap: null,
|
||
};
|
||
|
||
let _updateResizePreview = () => { };
|
||
let _resizeDialogCleanup = null;
|
||
|
||
// --- OVERLAY UPDATE: Optimized OverlayManager class with performance improvements ---
|
||
class OverlayManager {
|
||
constructor() {
|
||
this.isEnabled = false;
|
||
this.startCoords = null; // { region: {x, y}, pixel: {x, y} }
|
||
this.imageBitmap = null;
|
||
this.chunkedTiles = new Map(); // Map<"tileX,tileY", ImageBitmap>
|
||
this.originalTiles = new Map(); // Map<"tileX,tileY", ImageBitmap> store latest original tile bitmaps
|
||
this.originalTilesData = new Map(); // Map<"tileX,tileY", {w,h,data:Uint8ClampedArray}> cache full ImageData for fast pixel reads
|
||
this.tileSize = 1000;
|
||
this.processPromise = null; // Track ongoing processing
|
||
this.lastProcessedHash = null; // Cache invalidation
|
||
this.workerPool = null; // Web worker pool for heavy processing
|
||
}
|
||
|
||
toggle() {
|
||
this.isEnabled = !this.isEnabled;
|
||
console.log(`Overlay ${this.isEnabled ? 'enabled' : 'disabled'}.`);
|
||
return this.isEnabled;
|
||
}
|
||
|
||
enable() {
|
||
this.isEnabled = true;
|
||
}
|
||
|
||
disable() {
|
||
this.isEnabled = false;
|
||
}
|
||
|
||
clear() {
|
||
this.disable();
|
||
this.imageBitmap = null;
|
||
this.chunkedTiles.clear();
|
||
this.originalTiles.clear();
|
||
this.originalTilesData.clear();
|
||
this.lastProcessedHash = null;
|
||
if (this.processPromise) {
|
||
this.processPromise = null;
|
||
}
|
||
}
|
||
|
||
async setImage(imageBitmap) {
|
||
this.imageBitmap = imageBitmap;
|
||
this.lastProcessedHash = null; // Invalidate cache
|
||
if (this.imageBitmap && this.startCoords) {
|
||
await this.processImageIntoChunks();
|
||
}
|
||
}
|
||
|
||
async setPosition(startPosition, region) {
|
||
if (!startPosition || !region) {
|
||
this.startCoords = null;
|
||
this.chunkedTiles.clear();
|
||
this.lastProcessedHash = null;
|
||
return;
|
||
}
|
||
this.startCoords = { region, pixel: startPosition };
|
||
this.lastProcessedHash = null; // Invalidate cache
|
||
if (this.imageBitmap) {
|
||
await this.processImageIntoChunks();
|
||
}
|
||
}
|
||
|
||
// Generate hash for cache invalidation
|
||
_generateProcessHash() {
|
||
if (!this.imageBitmap || !this.startCoords) return null;
|
||
const { width, height } = this.imageBitmap;
|
||
const { x: px, y: py } = this.startCoords.pixel;
|
||
const { x: rx, y: ry } = this.startCoords.region;
|
||
return `${width}x${height}_${px},${py}_${rx},${ry}_${state.blueMarbleEnabled}_${state.overlayOpacity}`;
|
||
}
|
||
|
||
// --- OVERLAY UPDATE: Optimized chunking with caching and batch processing ---
|
||
async processImageIntoChunks() {
|
||
if (!this.imageBitmap || !this.startCoords) return;
|
||
|
||
// Check if we're already processing to avoid duplicate work
|
||
if (this.processPromise) {
|
||
return this.processPromise;
|
||
}
|
||
|
||
// Check cache validity
|
||
const currentHash = this._generateProcessHash();
|
||
if (this.lastProcessedHash === currentHash && this.chunkedTiles.size > 0) {
|
||
console.log(`📦 Using cached overlay chunks (${this.chunkedTiles.size} tiles)`);
|
||
return;
|
||
}
|
||
|
||
// Start processing
|
||
this.processPromise = this._doProcessImageIntoChunks();
|
||
try {
|
||
await this.processPromise;
|
||
this.lastProcessedHash = currentHash;
|
||
} finally {
|
||
this.processPromise = null;
|
||
}
|
||
}
|
||
|
||
async _doProcessImageIntoChunks() {
|
||
const startTime = performance.now();
|
||
this.chunkedTiles.clear();
|
||
|
||
const { width: imageWidth, height: imageHeight } = this.imageBitmap;
|
||
const { x: startPixelX, y: startPixelY } = this.startCoords.pixel;
|
||
const { x: startRegionX, y: startRegionY } = this.startCoords.region;
|
||
|
||
const { startTileX, startTileY, endTileX, endTileY } = Utils.calculateTileRange(
|
||
startRegionX,
|
||
startRegionY,
|
||
startPixelX,
|
||
startPixelY,
|
||
imageWidth,
|
||
imageHeight,
|
||
this.tileSize
|
||
);
|
||
|
||
const totalTiles = (endTileX - startTileX + 1) * (endTileY - startTileY + 1);
|
||
console.log(`🔄 Processing ${totalTiles} overlay tiles...`);
|
||
|
||
// Process tiles in batches to avoid blocking the main thread
|
||
const batchSize = 4; // Process 4 tiles at a time
|
||
const tilesToProcess = [];
|
||
|
||
for (let ty = startTileY; ty <= endTileY; ty++) {
|
||
for (let tx = startTileX; tx <= endTileX; tx++) {
|
||
tilesToProcess.push({ tx, ty });
|
||
}
|
||
}
|
||
|
||
// Process tiles in batches with yielding
|
||
for (let i = 0; i < tilesToProcess.length; i += batchSize) {
|
||
const batch = tilesToProcess.slice(i, i + batchSize);
|
||
|
||
await Promise.all(
|
||
batch.map(async ({ tx, ty }) => {
|
||
const tileKey = `${tx},${ty}`;
|
||
const chunkBitmap = await this._processTile(
|
||
tx,
|
||
ty,
|
||
imageWidth,
|
||
imageHeight,
|
||
startPixelX,
|
||
startPixelY,
|
||
startRegionX,
|
||
startRegionY
|
||
);
|
||
if (chunkBitmap) {
|
||
this.chunkedTiles.set(tileKey, chunkBitmap);
|
||
}
|
||
})
|
||
);
|
||
|
||
// Yield control to prevent blocking
|
||
if (i + batchSize < tilesToProcess.length) {
|
||
await new Promise((resolve) => setTimeout(resolve, 0));
|
||
}
|
||
}
|
||
|
||
const processingTime = performance.now() - startTime;
|
||
console.log(
|
||
`✅ Overlay processed ${this.chunkedTiles.size} tiles in ${Math.round(processingTime)}ms`
|
||
);
|
||
}
|
||
|
||
async _processTile(
|
||
tx,
|
||
ty,
|
||
imageWidth,
|
||
imageHeight,
|
||
startPixelX,
|
||
startPixelY,
|
||
startRegionX,
|
||
startRegionY
|
||
) {
|
||
const tileKey = `${tx},${ty}`;
|
||
|
||
// Calculate the portion of the image that overlaps with this tile
|
||
const imgStartX = (tx - startRegionX) * this.tileSize - startPixelX;
|
||
const imgStartY = (ty - startRegionY) * this.tileSize - startPixelY;
|
||
|
||
// Crop coordinates within the source image
|
||
const sX = Math.max(0, imgStartX);
|
||
const sY = Math.max(0, imgStartY);
|
||
const sW = Math.min(imageWidth - sX, this.tileSize - (sX - imgStartX));
|
||
const sH = Math.min(imageHeight - sY, this.tileSize - (sY - imgStartY));
|
||
|
||
if (sW <= 0 || sH <= 0) return null;
|
||
|
||
// Destination coordinates on the new chunk canvas
|
||
const dX = Math.max(0, -imgStartX);
|
||
const dY = Math.max(0, -imgStartY);
|
||
|
||
const chunkCanvas = new OffscreenCanvas(this.tileSize, this.tileSize);
|
||
const chunkCtx = chunkCanvas.getContext('2d');
|
||
chunkCtx.imageSmoothingEnabled = false;
|
||
|
||
chunkCtx.drawImage(this.imageBitmap, sX, sY, sW, sH, dX, dY, sW, sH);
|
||
|
||
// --- OPTIMIZED: Blue marble effect with faster pixel manipulation ---
|
||
if (state.blueMarbleEnabled) {
|
||
const imageData = chunkCtx.getImageData(dX, dY, sW, sH);
|
||
const data = imageData.data;
|
||
|
||
// Faster pixel manipulation using typed arrays
|
||
for (let i = 0; i < data.length; i += 4) {
|
||
const pixelIndex = i / 4;
|
||
const pixelY = Math.floor(pixelIndex / sW);
|
||
const pixelX = pixelIndex % sW;
|
||
|
||
if ((pixelX + pixelY) % 2 === 0 && data[i + 3] > 0) {
|
||
data[i + 3] = 0; // Set alpha to 0
|
||
}
|
||
}
|
||
|
||
chunkCtx.putImageData(imageData, dX, dY);
|
||
}
|
||
|
||
return await chunkCanvas.transferToImageBitmap();
|
||
}
|
||
|
||
// --- OVERLAY UPDATE: Optimized compositing with caching ---
|
||
async processAndRespondToTileRequest(eventData) {
|
||
const { endpoint, blobID, blobData } = eventData;
|
||
|
||
let finalBlob = blobData;
|
||
|
||
if (this.isEnabled && this.chunkedTiles.size > 0) {
|
||
const tileMatch = endpoint.match(/(\d+)\/(\d+)\.png/);
|
||
if (tileMatch) {
|
||
const tileX = parseInt(tileMatch[1], 10);
|
||
const tileY = parseInt(tileMatch[2], 10);
|
||
const tileKey = `${tileX},${tileY}`;
|
||
|
||
const chunkBitmap = this.chunkedTiles.get(tileKey);
|
||
// Also store the original tile bitmap for later pixel color checks
|
||
try {
|
||
const originalBitmap = await createImageBitmap(blobData);
|
||
this.originalTiles.set(tileKey, originalBitmap);
|
||
// Cache full ImageData for fast pixel access (avoid repeated drawImage/getImageData)
|
||
try {
|
||
let canvas, ctx;
|
||
if (typeof OffscreenCanvas !== 'undefined') {
|
||
canvas = new OffscreenCanvas(originalBitmap.width, originalBitmap.height);
|
||
ctx = canvas.getContext('2d');
|
||
} else {
|
||
canvas = document.createElement('canvas');
|
||
canvas.width = originalBitmap.width;
|
||
canvas.height = originalBitmap.height;
|
||
ctx = canvas.getContext('2d');
|
||
}
|
||
ctx.imageSmoothingEnabled = false;
|
||
ctx.drawImage(originalBitmap, 0, 0);
|
||
const imgData = ctx.getImageData(0, 0, originalBitmap.width, originalBitmap.height);
|
||
// Store typed array copy to avoid retaining large canvas
|
||
this.originalTilesData.set(tileKey, {
|
||
w: originalBitmap.width,
|
||
h: originalBitmap.height,
|
||
data: new Uint8ClampedArray(imgData.data),
|
||
});
|
||
} catch (e) {
|
||
// If ImageData extraction fails, still keep the bitmap as fallback
|
||
console.warn('OverlayManager: could not cache ImageData for', tileKey, e);
|
||
}
|
||
} catch (e) {
|
||
console.warn('OverlayManager: could not create original bitmap for', tileKey, e);
|
||
}
|
||
if (chunkBitmap) {
|
||
try {
|
||
// Use faster compositing for better performance
|
||
finalBlob = await this._compositeTileOptimized(blobData, chunkBitmap);
|
||
} catch (e) {
|
||
console.error('Error compositing overlay:', e);
|
||
// Fallback to original tile on error
|
||
finalBlob = blobData;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
// Send the (possibly modified) blob back to the injected script
|
||
window.postMessage(
|
||
{
|
||
source: 'auto-image-overlay',
|
||
blobID: blobID,
|
||
blobData: finalBlob,
|
||
},
|
||
'*'
|
||
);
|
||
}
|
||
|
||
// Returns [r,g,b,a] for a pixel inside a region tile (tileX, tileY are region coords)
|
||
async getTilePixelColor(tileX, tileY, pixelX, pixelY) {
|
||
const tileKey = `${tileX},${tileY}`;
|
||
const alphaThresh = state.customTransparencyThreshold || CONFIG.TRANSPARENCY_THRESHOLD;
|
||
|
||
// 1. Prefer cached ImageData if available
|
||
const cached = this.originalTilesData.get(tileKey);
|
||
if (cached && cached.data && cached.w > 0 && cached.h > 0) {
|
||
const x = Math.max(0, Math.min(cached.w - 1, pixelX));
|
||
const y = Math.max(0, Math.min(cached.h - 1, pixelY));
|
||
const idx = (y * cached.w + x) * 4;
|
||
const d = cached.data;
|
||
const a = d[idx + 3];
|
||
|
||
if (!state.paintTransparentPixels && a < alphaThresh) {
|
||
// Treat as transparent / unavailable
|
||
// Lightweight debug: show when transparency causes skip (only if verbose enabled)
|
||
if (window._overlayDebug)
|
||
console.debug('OverlayManager: pixel transparent (cached), skipping', tileKey, x, y, a);
|
||
return null;
|
||
}
|
||
return [d[idx], d[idx + 1], d[idx + 2], a];
|
||
}
|
||
|
||
// 2. Fallback: use bitmap, with retry
|
||
const maxRetries = 3;
|
||
for (let attempt = 1; attempt <= maxRetries; attempt++) {
|
||
const bitmap = this.originalTiles.get(tileKey);
|
||
if (!bitmap) {
|
||
if (attempt === maxRetries) {
|
||
console.warn('OverlayManager: no bitmap for', tileKey, 'after', maxRetries, 'attempts');
|
||
} else {
|
||
await Utils.sleep(50 * attempt); // exponential delay
|
||
}
|
||
continue;
|
||
}
|
||
|
||
try {
|
||
let canvas, ctx;
|
||
if (typeof OffscreenCanvas !== 'undefined') {
|
||
canvas = new OffscreenCanvas(bitmap.width, bitmap.height);
|
||
ctx = canvas.getContext('2d');
|
||
} else {
|
||
canvas = document.createElement('canvas');
|
||
canvas.width = bitmap.width;
|
||
canvas.height = bitmap.height;
|
||
ctx = canvas.getContext('2d');
|
||
}
|
||
ctx.imageSmoothingEnabled = false;
|
||
ctx.drawImage(bitmap, 0, 0);
|
||
|
||
const x = Math.max(0, Math.min(bitmap.width - 1, pixelX));
|
||
const y = Math.max(0, Math.min(bitmap.height - 1, pixelY));
|
||
const data = ctx.getImageData(x, y, 1, 1).data;
|
||
const a = data[3];
|
||
|
||
if (!state.paintTransparentPixels && a < alphaThresh) {
|
||
if (window._overlayDebug)
|
||
console.debug('OverlayManager: pixel transparent (fallback)', tileKey, x, y, a);
|
||
return null;
|
||
}
|
||
|
||
return [data[0], data[1], data[2], a];
|
||
} catch (e) {
|
||
console.warn('OverlayManager: failed to read pixel (attempt', attempt, ')', tileKey, e);
|
||
if (attempt < maxRetries) {
|
||
await Utils.sleep(50 * attempt);
|
||
} else {
|
||
console.error(
|
||
'OverlayManager: failed to read pixel after',
|
||
maxRetries,
|
||
'attempts',
|
||
tileKey
|
||
);
|
||
}
|
||
}
|
||
}
|
||
|
||
// 3. If everything fails — you can return null or [0,0,0,0]
|
||
// Prefer null — to avoid misleading
|
||
return null;
|
||
}
|
||
|
||
async _compositeTileOptimized(originalBlob, overlayBitmap) {
|
||
const originalBitmap = await createImageBitmap(originalBlob);
|
||
const canvas = new OffscreenCanvas(originalBitmap.width, originalBitmap.height);
|
||
const ctx = canvas.getContext('2d');
|
||
|
||
// Disable antialiasing for pixel-perfect rendering
|
||
ctx.imageSmoothingEnabled = false;
|
||
|
||
// Draw original tile first
|
||
ctx.drawImage(originalBitmap, 0, 0);
|
||
|
||
// Set opacity and draw overlay with optimized blend mode
|
||
ctx.globalAlpha = state.overlayOpacity;
|
||
ctx.globalCompositeOperation = 'source-over';
|
||
ctx.drawImage(overlayBitmap, 0, 0);
|
||
|
||
// Use faster blob conversion with compression settings
|
||
return await canvas.convertToBlob({
|
||
type: 'image/png',
|
||
quality: 0.95, // Slight compression for faster processing
|
||
});
|
||
}
|
||
|
||
/**
|
||
* Wait until all required tiles are loaded and cached
|
||
* @param {number} startRegionX
|
||
* @param {number} startRegionY
|
||
* @param {number} pixelWidth
|
||
* @param {number} pixelHeight
|
||
* @param {number} startPixelX
|
||
* @param {number} startPixelY
|
||
* @param {number} timeoutMs
|
||
* @returns {Promise<boolean>} true if tiles are ready
|
||
*/
|
||
async waitForTiles(
|
||
startRegionX,
|
||
startRegionY,
|
||
pixelWidth,
|
||
pixelHeight,
|
||
startPixelX = 0,
|
||
startPixelY = 0,
|
||
timeoutMs = 10000
|
||
) {
|
||
const { startTileX, startTileY, endTileX, endTileY } = Utils.calculateTileRange(
|
||
startRegionX,
|
||
startRegionY,
|
||
startPixelX,
|
||
startPixelY,
|
||
pixelWidth,
|
||
pixelHeight,
|
||
this.tileSize
|
||
);
|
||
|
||
const requiredTiles = [];
|
||
for (let ty = startTileY; ty <= endTileY; ty++) {
|
||
for (let tx = startTileX; tx <= endTileX; tx++) {
|
||
requiredTiles.push(`${tx},${ty}`);
|
||
}
|
||
}
|
||
|
||
if (requiredTiles.length === 0) return true;
|
||
|
||
const startTime = Date.now();
|
||
|
||
while (Date.now() - startTime < timeoutMs) {
|
||
if (state.stopFlag) {
|
||
console.log('waitForTiles: stopped by user');
|
||
return false;
|
||
}
|
||
|
||
const missing = requiredTiles.filter((key) => !this.originalTiles.has(key));
|
||
if (missing.length === 0) {
|
||
console.log(`✅ All ${requiredTiles.length} required tiles are loaded`);
|
||
return true;
|
||
}
|
||
|
||
await Utils.sleep(100);
|
||
}
|
||
|
||
console.warn(`❌ Timeout waiting for tiles: ${requiredTiles.length} required,
|
||
${requiredTiles.filter((k) => this.originalTiles.has(k)).length} loaded`);
|
||
return false;
|
||
}
|
||
}
|
||
|
||
const overlayManager = new OverlayManager();
|
||
|
||
// Optimized Turnstile token handling with improved caching and retry logic
|
||
let turnstileToken = null;
|
||
let tokenExpiryTime = 0;
|
||
let tokenGenerationInProgress = false;
|
||
let _resolveToken = null;
|
||
let tokenPromise = new Promise((resolve) => {
|
||
_resolveToken = resolve;
|
||
});
|
||
let retryCount = 0;
|
||
const MAX_RETRIES = 10;
|
||
const MAX_BATCH_RETRIES = 10; // Maximum attempts for batch sending
|
||
const TOKEN_LIFETIME = 240000; // 4 minutes (tokens typically last 5 min, use 4 for safety)
|
||
|
||
function setTurnstileToken(token) {
|
||
if (_resolveToken) {
|
||
_resolveToken(token);
|
||
_resolveToken = null;
|
||
}
|
||
turnstileToken = token;
|
||
tokenExpiryTime = Date.now() + TOKEN_LIFETIME;
|
||
console.log('✅ Turnstile token set successfully');
|
||
}
|
||
|
||
function isTokenValid() {
|
||
return turnstileToken && Date.now() < tokenExpiryTime;
|
||
}
|
||
|
||
function invalidateToken() {
|
||
turnstileToken = null;
|
||
tokenExpiryTime = 0;
|
||
console.log('🗑️ Token invalidated, will force fresh generation');
|
||
}
|
||
|
||
async function ensureToken(forceRefresh = false) {
|
||
// Return cached token if still valid and not forcing refresh
|
||
if (isTokenValid() && !forceRefresh) {
|
||
return turnstileToken;
|
||
}
|
||
|
||
// Invalidate token if forcing refresh
|
||
if (forceRefresh) invalidateToken();
|
||
|
||
// Avoid multiple simultaneous token generations
|
||
if (tokenGenerationInProgress) {
|
||
console.log('🔄 Token generation already in progress, waiting...');
|
||
await Utils.sleep(2000);
|
||
return isTokenValid() ? turnstileToken : null;
|
||
}
|
||
|
||
tokenGenerationInProgress = true;
|
||
|
||
try {
|
||
console.log('🔄 Token expired or missing, generating new one...');
|
||
const token = await handleCaptchaWithRetry();
|
||
if (token && token.length > 20) {
|
||
setTurnstileToken(token);
|
||
console.log('✅ Token captured and cached successfully');
|
||
return token;
|
||
}
|
||
|
||
console.log('⚠️ Invisible Turnstile failed, forcing browser automation...');
|
||
const fallbackToken = await handleCaptchaFallback();
|
||
if (fallbackToken && fallbackToken.length > 20) {
|
||
setTurnstileToken(fallbackToken);
|
||
console.log('✅ Fallback token captured successfully');
|
||
return fallbackToken;
|
||
}
|
||
|
||
console.log('❌ All token generation methods failed');
|
||
return null;
|
||
} finally {
|
||
tokenGenerationInProgress = false;
|
||
}
|
||
}
|
||
|
||
async function handleCaptchaWithRetry() {
|
||
const startTime = performance.now();
|
||
|
||
try {
|
||
const { sitekey, token: preGeneratedToken } = await Utils.obtainSitekeyAndToken();
|
||
|
||
if (!sitekey) {
|
||
throw new Error('No valid sitekey found');
|
||
}
|
||
|
||
console.log('🔑 Using sitekey:', sitekey);
|
||
|
||
if (typeof window !== 'undefined' && window.navigator) {
|
||
console.log(
|
||
'🧭 UA:',
|
||
window.navigator.userAgent.substring(0, 50) + '...',
|
||
'Platform:',
|
||
window.navigator.platform
|
||
);
|
||
}
|
||
|
||
let token = null;
|
||
|
||
if (
|
||
preGeneratedToken &&
|
||
typeof preGeneratedToken === 'string' &&
|
||
preGeneratedToken.length > 20
|
||
) {
|
||
console.log('♻️ Reusing pre-generated Turnstile token');
|
||
token = preGeneratedToken;
|
||
} else {
|
||
if (isTokenValid()) {
|
||
console.log('♻️ Using existing cached token (from previous session)');
|
||
token = turnstileToken;
|
||
} else {
|
||
console.log('🔐 Generating new token with executeTurnstile...');
|
||
token = await Utils.executeTurnstile(sitekey, 'paint');
|
||
if (token) setTurnstileToken(token);
|
||
}
|
||
}
|
||
|
||
if (token && typeof token === 'string' && token.length > 20) {
|
||
const elapsed = Math.round(performance.now() - startTime);
|
||
console.log(`✅ Turnstile token generated successfully in ${elapsed}ms`);
|
||
return token;
|
||
} else {
|
||
throw new Error(`Invalid or empty token received - Length: ${token?.length || 0}`);
|
||
}
|
||
} catch (error) {
|
||
const elapsed = Math.round(performance.now() - startTime);
|
||
console.error(`❌ Turnstile token generation failed after ${elapsed}ms:`, error);
|
||
throw error;
|
||
}
|
||
}
|
||
const randStr = (len, chars = 'abcdefghijklmnopqrstuvwxyz0123456789') =>
|
||
[...Array(len)].map(() => chars[(crypto?.getRandomValues?.(new Uint32Array(1))[0] % chars.length) || Math.floor(Math.random() * chars.length)]).join('')
|
||
const fpStr32 = randStr(32);
|
||
async function handleCaptchaFallback() {
|
||
// Implementation for fallback token generation would go here
|
||
// This is a placeholder for browser automation fallback
|
||
console.log('🔄 Attempting fallback token generation...');
|
||
return null;
|
||
}
|
||
|
||
function inject(callback) {
|
||
const script = document.createElement('script');
|
||
script.textContent = `(${callback})();`;
|
||
document.documentElement?.appendChild(script);
|
||
script.remove();
|
||
}
|
||
|
||
inject(() => {
|
||
const fetchedBlobQueue = new Map();
|
||
|
||
window.addEventListener('message', (event) => {
|
||
const { source, blobID, blobData } = event.data;
|
||
if (source === 'auto-image-overlay' && blobID && blobData) {
|
||
const callback = fetchedBlobQueue.get(blobID);
|
||
if (typeof callback === 'function') {
|
||
callback(blobData);
|
||
}
|
||
fetchedBlobQueue.delete(blobID);
|
||
}
|
||
});
|
||
|
||
const originalFetch = window.fetch;
|
||
window.fetch = async function (...args) {
|
||
const response = await originalFetch.apply(this, args);
|
||
const url = args[0] instanceof Request ? args[0].url : args[0];
|
||
|
||
if (typeof url === 'string') {
|
||
if (url.includes('https://backend.wplace.live/s0/pixel/')) {
|
||
try {
|
||
const payload = JSON.parse(args[1].body);
|
||
if (payload.t) {
|
||
// 📊 Debug log
|
||
console.log(
|
||
`🔍✅ Turnstile Token Captured - Type: ${typeof payload.t}, Value: ${payload.t
|
||
? typeof payload.t === 'string'
|
||
? payload.t.length > 50
|
||
? payload.t.substring(0, 50) + '...'
|
||
: payload.t
|
||
: JSON.stringify(payload.t)
|
||
: 'null/undefined'
|
||
}, Length: ${payload.t?.length || 0}`
|
||
);
|
||
window.postMessage({ source: 'turnstile-capture', token: payload.t }, '*');
|
||
}
|
||
} catch (_) {
|
||
/* ignore */
|
||
}
|
||
}
|
||
|
||
const contentType = response.headers.get('content-type') || '';
|
||
if (contentType.includes('image/png') && url.includes('.png')) {
|
||
const cloned = response.clone();
|
||
return new Promise(async (resolve) => {
|
||
const blobUUID = crypto.randomUUID();
|
||
const originalBlob = await cloned.blob();
|
||
|
||
fetchedBlobQueue.set(blobUUID, (processedBlob) => {
|
||
resolve(
|
||
new Response(processedBlob, {
|
||
headers: cloned.headers,
|
||
status: cloned.status,
|
||
statusText: cloned.statusText,
|
||
})
|
||
);
|
||
});
|
||
|
||
window.postMessage(
|
||
{
|
||
source: 'auto-image-tile',
|
||
endpoint: url,
|
||
blobID: blobUUID,
|
||
blobData: originalBlob,
|
||
},
|
||
'*'
|
||
);
|
||
});
|
||
}
|
||
}
|
||
|
||
return response;
|
||
};
|
||
});
|
||
|
||
window.addEventListener('message', (event) => {
|
||
const { source, endpoint, blobID, blobData, token } = event.data;
|
||
|
||
if (source === 'auto-image-tile' && endpoint && blobID && blobData) {
|
||
overlayManager.processAndRespondToTileRequest(event.data);
|
||
}
|
||
|
||
if (source === 'turnstile-capture' && token) {
|
||
setTurnstileToken(token);
|
||
if (document.querySelector('#statusText')?.textContent.includes('CAPTCHA')) {
|
||
Utils.showAlert(Utils.t('tokenCapturedSuccess'), 'success');
|
||
updateUI('colorsFound', 'success', { count: state.availableColors.length });
|
||
}
|
||
}
|
||
});
|
||
|
||
async function detectLanguage() {
|
||
try {
|
||
const response = await fetch('https://backend.wplace.live/me', {
|
||
credentials: 'include',
|
||
});
|
||
const data = await response.json();
|
||
state.language = data.language === 'pt' ? 'pt' : 'en';
|
||
} catch {
|
||
state.language = navigator.language.startsWith('pt') ? 'pt' : 'en';
|
||
}
|
||
}
|
||
|
||
// UTILITY FUNCTIONS
|
||
const Utils = {
|
||
sleep: (ms) => new Promise((r) => setTimeout(r, ms)),
|
||
|
||
dynamicSleep: async function (tickAndGetRemainingMs) {
|
||
let remaining = Math.max(0, await tickAndGetRemainingMs());
|
||
while (remaining > 0) {
|
||
const interval = remaining > 5000 ? 2000 : remaining > 1000 ? 500 : 100;
|
||
await this.sleep(Math.min(interval, remaining));
|
||
remaining = Math.max(0, await tickAndGetRemainingMs());
|
||
}
|
||
},
|
||
|
||
waitForSelector: async (selector, interval = 200, timeout = 5000) => {
|
||
const start = Date.now();
|
||
while (Date.now() - start < timeout) {
|
||
const el = document.querySelector(selector);
|
||
if (el) return el;
|
||
await Utils.sleep(interval);
|
||
}
|
||
return null;
|
||
},
|
||
|
||
msToTimeText(ms) {
|
||
const totalSeconds = Math.ceil(ms / 1000);
|
||
const hours = Math.floor(totalSeconds / 3600);
|
||
const minutes = Math.floor((totalSeconds % 3600) / 60);
|
||
const seconds = totalSeconds % 60;
|
||
|
||
if (hours > 0) return `${hours}h ${minutes}m ${seconds}s`;
|
||
if (minutes > 0) return `${minutes}m ${seconds}s`;
|
||
return `${seconds}s`;
|
||
},
|
||
|
||
// Debounced scroll-to-adjust handler for sliders
|
||
createScrollToAdjust: (element, updateCallback, min, max, step = 1) => {
|
||
let debounceTimer = null;
|
||
|
||
const handleWheel = (e) => {
|
||
// Only trigger when hovering over the slider
|
||
if (e.target !== element) return;
|
||
|
||
e.preventDefault();
|
||
e.stopPropagation();
|
||
|
||
// Clear existing debounce timer
|
||
if (debounceTimer) {
|
||
clearTimeout(debounceTimer);
|
||
}
|
||
|
||
// Debounce the adjustment to make it precise
|
||
debounceTimer = setTimeout(() => {
|
||
const currentValue = parseInt(element.value) || 0;
|
||
const delta = e.deltaY > 0 ? -step : step;
|
||
const newValue = Math.max(min, Math.min(max, currentValue + delta));
|
||
|
||
if (newValue !== currentValue) {
|
||
element.value = newValue;
|
||
updateCallback(newValue);
|
||
}
|
||
}, 50); // 50ms debounce for precise control
|
||
};
|
||
|
||
element.addEventListener('wheel', handleWheel, { passive: false });
|
||
|
||
// Return cleanup function
|
||
return () => {
|
||
if (debounceTimer) clearTimeout(debounceTimer);
|
||
element.removeEventListener('wheel', handleWheel);
|
||
};
|
||
},
|
||
|
||
/**
|
||
* Calculate the range of tile coordinates (in region space) that cover a given image area.
|
||
* @param {number} startRegionX - Base region X
|
||
* @param {number} startRegionY - Base region Y
|
||
* @param {number} startPixelX - Starting pixel X within the region grid
|
||
* @param {number} startPixelY - Starting pixel Y within the region grid
|
||
* @param {number} width - Image width in pixels
|
||
* @param {number} height - Image height in pixels
|
||
* @param {number} tileSize - Size of a tile (default 1000)
|
||
* @returns {{ startTileX: number, startTileY: number, endTileX: number, endTileY: number }}
|
||
*/
|
||
calculateTileRange(
|
||
startRegionX,
|
||
startRegionY,
|
||
startPixelX,
|
||
startPixelY,
|
||
width,
|
||
height,
|
||
tileSize = 1000
|
||
) {
|
||
const endPixelX = startPixelX + width;
|
||
const endPixelY = startPixelY + height;
|
||
|
||
return {
|
||
startTileX: startRegionX + Math.floor(startPixelX / tileSize),
|
||
startTileY: startRegionY + Math.floor(startPixelY / tileSize),
|
||
endTileX: startRegionX + Math.floor((endPixelX - 1) / tileSize),
|
||
endTileY: startRegionY + Math.floor((endPixelY - 1) / tileSize),
|
||
};
|
||
}, // Turnstile Generator Integration - Optimized with widget reuse and proper cleanup
|
||
turnstileLoaded: false,
|
||
_turnstileContainer: null,
|
||
_turnstileOverlay: null,
|
||
_turnstileWidgetId: null,
|
||
_lastSitekey: null,
|
||
|
||
async loadTurnstile() {
|
||
// If Turnstile is already present, just resolve.
|
||
if (window.turnstile) {
|
||
this.turnstileLoaded = true;
|
||
return Promise.resolve();
|
||
}
|
||
|
||
return new Promise((resolve, reject) => {
|
||
// Avoid adding the script twice
|
||
if (
|
||
document.querySelector(
|
||
'script[src^="https://challenges.cloudflare.com/turnstile/v0/api.js"]'
|
||
)
|
||
) {
|
||
const checkReady = () => {
|
||
if (window.turnstile) {
|
||
this.turnstileLoaded = true;
|
||
resolve();
|
||
} else {
|
||
setTimeout(checkReady, 100);
|
||
}
|
||
};
|
||
return checkReady();
|
||
}
|
||
|
||
const script = document.createElement('script');
|
||
script.src = 'https://challenges.cloudflare.com/turnstile/v0/api.js?render=explicit';
|
||
script.async = true;
|
||
script.defer = true;
|
||
script.onload = () => {
|
||
this.turnstileLoaded = true;
|
||
console.log('✅ Turnstile script loaded successfully');
|
||
resolve();
|
||
};
|
||
script.onerror = () => {
|
||
console.error('❌ Failed to load Turnstile script');
|
||
reject(new Error('Failed to load Turnstile'));
|
||
};
|
||
document.head.appendChild(script);
|
||
});
|
||
},
|
||
|
||
// Create or reuse the turnstile container - completely hidden for token generation
|
||
ensureTurnstileContainer() {
|
||
if (!this._turnstileContainer || !document.body.contains(this._turnstileContainer)) {
|
||
// Clean up old container if it exists
|
||
if (this._turnstileContainer) {
|
||
this._turnstileContainer.remove();
|
||
}
|
||
|
||
this._turnstileContainer = document.createElement('div');
|
||
this._turnstileContainer.className = 'wplace-turnstile-hidden';
|
||
this._turnstileContainer.setAttribute('aria-hidden', 'true');
|
||
this._turnstileContainer.id = 'turnstile-widget-container';
|
||
document.body.appendChild(this._turnstileContainer);
|
||
}
|
||
return this._turnstileContainer;
|
||
},
|
||
|
||
// Interactive overlay container for visible widgets when needed
|
||
ensureTurnstileOverlayContainer() {
|
||
if (this._turnstileOverlay && document.body.contains(this._turnstileOverlay)) {
|
||
return this._turnstileOverlay;
|
||
}
|
||
|
||
const overlay = document.createElement('div');
|
||
overlay.id = 'turnstile-overlay-container';
|
||
overlay.className = 'wplace-turnstile-overlay wplace-overlay-hidden';
|
||
|
||
const title = document.createElement('div');
|
||
title.textContent = Utils.t('turnstileInstructions');
|
||
title.className = 'wplace-turnstile-title';
|
||
|
||
const host = document.createElement('div');
|
||
host.id = 'turnstile-overlay-host';
|
||
host.className = 'wplace-turnstile-host';
|
||
|
||
const hideBtn = document.createElement('button');
|
||
hideBtn.textContent = Utils.t('hideTurnstileBtn');
|
||
hideBtn.className = 'wplace-turnstile-hide-btn';
|
||
hideBtn.addEventListener('click', () => overlay.remove());
|
||
|
||
overlay.appendChild(title);
|
||
overlay.appendChild(host);
|
||
overlay.appendChild(hideBtn);
|
||
document.body.appendChild(overlay);
|
||
|
||
this._turnstileOverlay = overlay;
|
||
return overlay;
|
||
},
|
||
|
||
async executeTurnstile(sitekey, action = 'paint') {
|
||
await this.loadTurnstile();
|
||
|
||
// Try reusing existing widget first if sitekey matches
|
||
if (this._turnstileWidgetId && this._lastSitekey === sitekey && window.turnstile?.execute) {
|
||
try {
|
||
console.log('🔄 Reusing existing Turnstile widget...');
|
||
const token = await Promise.race([
|
||
window.turnstile.execute(this._turnstileWidgetId, { action }),
|
||
new Promise((_, reject) =>
|
||
setTimeout(() => reject(new Error('Execute timeout')), 15000)
|
||
),
|
||
]);
|
||
if (token && token.length > 20) {
|
||
console.log('✅ Token generated via widget reuse');
|
||
return token;
|
||
}
|
||
} catch (error) {
|
||
console.log('� Widget reuse failed, will create a fresh widget:', error.message);
|
||
}
|
||
}
|
||
|
||
// Try invisible widget first
|
||
const invisibleToken = await this.createTurnstileWidget(sitekey, action);
|
||
if (invisibleToken && invisibleToken.length > 20) {
|
||
return invisibleToken;
|
||
}
|
||
|
||
console.log('� Falling back to interactive Turnstile (visible).');
|
||
return await this.createTurnstileWidgetInteractive(sitekey, action);
|
||
},
|
||
|
||
async createTurnstileWidget(sitekey, action) {
|
||
return new Promise((resolve) => {
|
||
try {
|
||
// Force cleanup of any existing widget
|
||
if (this._turnstileWidgetId && window.turnstile?.remove) {
|
||
try {
|
||
window.turnstile.remove(this._turnstileWidgetId);
|
||
console.log('🧹 Cleaned up existing Turnstile widget');
|
||
} catch (e) {
|
||
console.warn('⚠️ Widget cleanup warning:', e.message);
|
||
}
|
||
}
|
||
|
||
const container = this.ensureTurnstileContainer();
|
||
container.innerHTML = '';
|
||
|
||
// Verify Turnstile is available
|
||
if (!window.turnstile?.render) {
|
||
console.error('❌ Turnstile not available for rendering');
|
||
resolve(null);
|
||
return;
|
||
}
|
||
|
||
console.log('🔧 Creating invisible Turnstile widget...');
|
||
const widgetId = window.turnstile.render(container, {
|
||
sitekey,
|
||
action,
|
||
size: 'invisible',
|
||
retry: 'auto',
|
||
'retry-interval': 8000,
|
||
callback: (token) => {
|
||
console.log('✅ Invisible Turnstile callback');
|
||
resolve(token);
|
||
},
|
||
'error-callback': () => resolve(null),
|
||
'timeout-callback': () => resolve(null),
|
||
});
|
||
|
||
this._turnstileWidgetId = widgetId;
|
||
this._lastSitekey = sitekey;
|
||
|
||
if (!widgetId) {
|
||
return resolve(null);
|
||
}
|
||
|
||
// Execute the widget and race with timeout
|
||
Promise.race([
|
||
window.turnstile.execute(widgetId, { action }),
|
||
new Promise((_, reject) =>
|
||
setTimeout(() => reject(new Error('Invisible execute timeout')), 12000)
|
||
),
|
||
])
|
||
.then(resolve)
|
||
.catch(() => resolve(null));
|
||
} catch (e) {
|
||
console.error('❌ Invisible Turnstile creation failed:', e);
|
||
resolve(null);
|
||
}
|
||
});
|
||
},
|
||
|
||
async createTurnstileWidgetInteractive(sitekey, action) {
|
||
// Create a visible widget that users can interact with if needed
|
||
console.log('🔄 Creating interactive Turnstile widget (visible)');
|
||
|
||
return new Promise((resolve) => {
|
||
try {
|
||
// Force cleanup of any existing widget
|
||
if (this._turnstileWidgetId && window.turnstile?.remove) {
|
||
try {
|
||
window.turnstile.remove(this._turnstileWidgetId);
|
||
} catch (e) {
|
||
console.warn('⚠️ Widget cleanup warning:', e.message);
|
||
}
|
||
}
|
||
|
||
const overlay = this.ensureTurnstileOverlayContainer();
|
||
overlay.classList.remove('wplace-overlay-hidden');
|
||
overlay.style.display = 'block';
|
||
|
||
const host = overlay.querySelector('#turnstile-overlay-host');
|
||
host.innerHTML = '';
|
||
|
||
// Set a timeout for interactive mode
|
||
const timeout = setTimeout(() => {
|
||
console.warn('⏰ Interactive Turnstile widget timeout');
|
||
overlay.classList.add('wplace-overlay-hidden');
|
||
overlay.style.display = 'none';
|
||
resolve(null);
|
||
}, 60000); // 60 seconds for user interaction
|
||
|
||
const widgetId = window.turnstile.render(host, {
|
||
sitekey,
|
||
action,
|
||
size: 'normal',
|
||
theme: 'light',
|
||
callback: (token) => {
|
||
clearTimeout(timeout);
|
||
overlay.classList.add('wplace-overlay-hidden');
|
||
overlay.style.display = 'none';
|
||
console.log('✅ Interactive Turnstile completed successfully');
|
||
|
||
if (typeof token === 'string' && token.length > 20) {
|
||
resolve(token);
|
||
} else {
|
||
console.warn('❌ Invalid token from interactive widget');
|
||
resolve(null);
|
||
}
|
||
},
|
||
'error-callback': (error) => {
|
||
clearTimeout(timeout);
|
||
overlay.classList.add('wplace-overlay-hidden');
|
||
overlay.style.display = 'none';
|
||
console.warn('❌ Interactive Turnstile error:', error);
|
||
resolve(null);
|
||
},
|
||
});
|
||
|
||
this._turnstileWidgetId = widgetId;
|
||
this._lastSitekey = sitekey;
|
||
|
||
if (!widgetId) {
|
||
clearTimeout(timeout);
|
||
overlay.classList.add('wplace-overlay-hidden');
|
||
overlay.style.display = 'none';
|
||
console.warn('❌ Failed to create interactive Turnstile widget');
|
||
resolve(null);
|
||
} else {
|
||
console.log('✅ Interactive Turnstile widget created, waiting for user interaction...');
|
||
}
|
||
} catch (e) {
|
||
console.error('❌ Interactive Turnstile creation failed:', e);
|
||
resolve(null);
|
||
}
|
||
});
|
||
},
|
||
|
||
// Cleanup method for when the script is disabled/reloaded
|
||
cleanupTurnstile() {
|
||
if (this._turnstileWidgetId && window.turnstile?.remove) {
|
||
try {
|
||
window.turnstile.remove(this._turnstileWidgetId);
|
||
} catch (e) {
|
||
console.warn('Failed to cleanup Turnstile widget:', e);
|
||
}
|
||
}
|
||
|
||
if (this._turnstileContainer && document.body.contains(this._turnstileContainer)) {
|
||
this._turnstileContainer.remove();
|
||
}
|
||
|
||
if (this._turnstileOverlay && document.body.contains(this._turnstileOverlay)) {
|
||
this._turnstileOverlay.remove();
|
||
}
|
||
|
||
this._turnstileWidgetId = null;
|
||
this._turnstileContainer = null;
|
||
this._turnstileOverlay = null;
|
||
this._lastSitekey = null;
|
||
},
|
||
|
||
async obtainSitekeyAndToken(fallback = '0x4AAAAAABpqJe8FO0N84q0F') {
|
||
// Cache sitekey to avoid repeated DOM queries
|
||
if (this._cachedSitekey) {
|
||
console.log('🔍 Using cached sitekey:', this._cachedSitekey);
|
||
|
||
return isTokenValid()
|
||
? {
|
||
sitekey: this._cachedSitekey,
|
||
token: turnstileToken,
|
||
}
|
||
: { sitekey: this._cachedSitekey, token: null };
|
||
}
|
||
|
||
// List of potential sitekeys to try
|
||
const potentialSitekeys = [
|
||
'0x4AAAAAABpqJe8FO0N84q0F', // WPlace common sitekey
|
||
'0x4AAAAAAAJ7xjKAp6Mt_7zw', // Alternative WPlace sitekey
|
||
'0x4AAAAAADm5QWx6Ov2LNF2g', // Another common sitekey
|
||
];
|
||
const trySitekey = async (sitekey, source) => {
|
||
if (!sitekey || sitekey.length < 10) return null;
|
||
|
||
console.log(`🔍 Testing sitekey from ${source}:`, sitekey);
|
||
const token = await this.executeTurnstile(sitekey);
|
||
|
||
if (token && token.length >= 20) {
|
||
console.log(`✅ Valid token generated from ${source} sitekey`);
|
||
setTurnstileToken(token);
|
||
this._cachedSitekey = sitekey;
|
||
return { sitekey, token };
|
||
} else {
|
||
console.log(`❌ Failed to get token from ${source} sitekey`);
|
||
return null;
|
||
}
|
||
};
|
||
|
||
try {
|
||
// 1️⃣ data-sitekey attribute
|
||
const sitekeySel = document.querySelector('[data-sitekey]');
|
||
if (sitekeySel) {
|
||
const sitekey = sitekeySel.getAttribute('data-sitekey');
|
||
const result = await trySitekey(sitekey, 'data attribute');
|
||
if (result) {
|
||
return result;
|
||
}
|
||
}
|
||
|
||
// 2️⃣ Turnstile element
|
||
const turnstileEl = document.querySelector('.cf-turnstile');
|
||
if (turnstileEl?.dataset?.sitekey) {
|
||
const sitekey = turnstileEl.dataset.sitekey;
|
||
const result = await trySitekey(sitekey, 'turnstile element');
|
||
if (result) {
|
||
return result;
|
||
}
|
||
}
|
||
|
||
// 3️⃣ Meta tags
|
||
const metaTags = document.querySelectorAll(
|
||
'meta[name*="turnstile"], meta[property*="turnstile"]'
|
||
);
|
||
for (const meta of metaTags) {
|
||
const content = meta.getAttribute('content');
|
||
const result = await trySitekey(content, 'meta tag');
|
||
if (result) {
|
||
return result;
|
||
}
|
||
}
|
||
|
||
// 4️⃣ Global variable
|
||
if (window.__TURNSTILE_SITEKEY) {
|
||
const result = await trySitekey(window.__TURNSTILE_SITEKEY, 'global variable');
|
||
if (result) {
|
||
return result;
|
||
}
|
||
}
|
||
|
||
// 5️⃣ Script tags
|
||
const scripts = document.querySelectorAll('script');
|
||
for (const script of scripts) {
|
||
const content = script.textContent || script.innerHTML;
|
||
const match = content.match(
|
||
/(?:sitekey|data-sitekey)['"\s\[\]:\=\(]*['"]?([0-9a-zA-Z_-]{20,})['"]?/i
|
||
);
|
||
if (match && match[1]) {
|
||
const extracted = match[1].replace(/['"]/g, '');
|
||
const result = await trySitekey(extracted, 'script content');
|
||
if (result) {
|
||
return result;
|
||
}
|
||
}
|
||
}
|
||
|
||
// 6️⃣ Known potential sitekeys
|
||
console.log('🔍 Testing known potential sitekeys...');
|
||
for (const testSitekey of potentialSitekeys) {
|
||
const result = await trySitekey(testSitekey, 'known list');
|
||
if (result) {
|
||
return result;
|
||
}
|
||
}
|
||
} catch (error) {
|
||
console.warn('⚠️ Error during sitekey detection:', error);
|
||
}
|
||
|
||
// 7️⃣ Fallback
|
||
console.log('🔧 Trying fallback sitekey:', fallback);
|
||
const fallbackResult = await trySitekey(fallback, 'fallback');
|
||
if (fallbackResult) {
|
||
return fallbackResult;
|
||
}
|
||
|
||
console.error('❌ No working sitekey or token found.');
|
||
return { sitekey: null, token: null };
|
||
},
|
||
|
||
createElement: (tag, props = {}, children = []) => {
|
||
const element = document.createElement(tag);
|
||
|
||
Object.entries(props).forEach(([key, value]) => {
|
||
if (key === 'style' && typeof value === 'object') {
|
||
Object.assign(element.style, value);
|
||
} else if (key === 'className') {
|
||
element.className = value;
|
||
} else if (key === 'innerHTML') {
|
||
element.innerHTML = value;
|
||
} else {
|
||
element.setAttribute(key, value);
|
||
}
|
||
});
|
||
|
||
if (typeof children === 'string') {
|
||
element.textContent = children;
|
||
} else if (Array.isArray(children)) {
|
||
children.forEach((child) => {
|
||
if (typeof child === 'string') {
|
||
element.appendChild(document.createTextNode(child));
|
||
} else {
|
||
element.appendChild(child);
|
||
}
|
||
});
|
||
}
|
||
|
||
return element;
|
||
},
|
||
|
||
createButton: (id, text, icon, onClick, style = CONFIG.CSS_CLASSES.BUTTON_PRIMARY) => {
|
||
const button = Utils.createElement('button', {
|
||
id: id,
|
||
style: style,
|
||
innerHTML: `${icon ? `<i class="${icon}"></i>` : ''}<span>${text}</span>`,
|
||
});
|
||
if (onClick) button.addEventListener('click', onClick);
|
||
return button;
|
||
},
|
||
|
||
// Synchronous translation function for UI rendering
|
||
t: (key, params = {}) => {
|
||
// Try to get from cache first
|
||
const cacheKey = `${state.language}_${key}`;
|
||
if (translationCache.has(cacheKey)) {
|
||
let text = translationCache.get(cacheKey);
|
||
Object.keys(params).forEach((param) => {
|
||
text = text.replace(`{${param}}`, params[param]);
|
||
});
|
||
return text;
|
||
}
|
||
|
||
// Try dynamically loaded translations (already loaded)
|
||
if (loadedTranslations[state.language]?.[key]) {
|
||
let text = loadedTranslations[state.language][key];
|
||
// Cache for future use
|
||
translationCache.set(cacheKey, text);
|
||
Object.keys(params).forEach((param) => {
|
||
text = text.replace(`{${param}}`, params[param]);
|
||
});
|
||
return text;
|
||
}
|
||
|
||
// Fallback to English if current language failed
|
||
if (state.language !== 'en' && loadedTranslations['en']?.[key]) {
|
||
let text = loadedTranslations['en'][key];
|
||
Object.keys(params).forEach((param) => {
|
||
text = text.replace(`{${param}}`, params[param]);
|
||
});
|
||
return text;
|
||
}
|
||
|
||
// Final fallback to emergency fallback or key
|
||
let text = FALLBACK_TEXT[state.language]?.[key] || FALLBACK_TEXT.en?.[key] || key;
|
||
Object.keys(params).forEach((param) => {
|
||
text = text.replace(new RegExp(`\\{${param}\\}`, 'g'), params[param]);
|
||
});
|
||
|
||
// Log missing translations for debugging
|
||
if (text === key && key !== 'undefined') {
|
||
console.warn(`⚠️ Missing translation for key: ${key} (language: ${state.language})`);
|
||
}
|
||
|
||
return text;
|
||
},
|
||
|
||
showAlert: (message, type = 'info') => {
|
||
const alertDiv = document.createElement('div');
|
||
alertDiv.className = `wplace-alert-base wplace-alert-${type}`;
|
||
|
||
alertDiv.textContent = message;
|
||
document.body.appendChild(alertDiv);
|
||
|
||
setTimeout(() => {
|
||
alertDiv.style.animation = 'slide-down 0.3s ease-out reverse';
|
||
setTimeout(() => {
|
||
document.body.removeChild(alertDiv);
|
||
}, 300);
|
||
}, 4000);
|
||
},
|
||
|
||
colorDistance: (a, b) =>
|
||
Math.sqrt(Math.pow(a[0] - b[0], 2) + Math.pow(a[1] - b[1], 2) + Math.pow(a[2] - b[2], 2)),
|
||
_labCache: new Map(), // key: (r<<16)|(g<<8)|b value: [L,a,b]
|
||
_rgbToLab: (r, g, b) => {
|
||
// sRGB -> linear
|
||
const srgbToLinear = (v) => {
|
||
v /= 255;
|
||
return v <= 0.04045 ? v / 12.92 : Math.pow((v + 0.055) / 1.055, 2.4);
|
||
};
|
||
const rl = srgbToLinear(r);
|
||
const gl = srgbToLinear(g);
|
||
const bl = srgbToLinear(b);
|
||
let X = rl * 0.4124 + gl * 0.3576 + bl * 0.1805;
|
||
let Y = rl * 0.2126 + gl * 0.7152 + bl * 0.0722;
|
||
let Z = rl * 0.0193 + gl * 0.1192 + bl * 0.9505;
|
||
X /= 0.95047;
|
||
Y /= 1.0;
|
||
Z /= 1.08883;
|
||
const f = (t) => (t > 0.008856 ? Math.cbrt(t) : 7.787 * t + 16 / 116);
|
||
const fX = f(X),
|
||
fY = f(Y),
|
||
fZ = f(Z);
|
||
const L = 116 * fY - 16;
|
||
const a = 500 * (fX - fY);
|
||
const b2 = 200 * (fY - fZ);
|
||
return [L, a, b2];
|
||
},
|
||
_lab: (r, g, b) => {
|
||
const key = (r << 16) | (g << 8) | b;
|
||
let v = Utils._labCache.get(key);
|
||
if (!v) {
|
||
v = Utils._rgbToLab(r, g, b);
|
||
Utils._labCache.set(key, v);
|
||
}
|
||
return v;
|
||
},
|
||
findClosestPaletteColor: (r, g, b, palette) => {
|
||
// Use provided palette or derive from COLOR_MAP
|
||
if (!palette || palette.length === 0) {
|
||
palette = Object.values(CONFIG.COLOR_MAP)
|
||
.filter((c) => c.rgb)
|
||
.map((c) => [c.rgb.r, c.rgb.g, c.rgb.b]);
|
||
}
|
||
if (state.colorMatchingAlgorithm === 'legacy') {
|
||
let menorDist = Infinity;
|
||
let cor = [0, 0, 0];
|
||
for (let i = 0; i < palette.length; i++) {
|
||
const [pr, pg, pb] = palette[i];
|
||
const rmean = (pr + r) / 2;
|
||
const rdiff = pr - r;
|
||
const gdiff = pg - g;
|
||
const bdiff = pb - b;
|
||
const dist = Math.sqrt(
|
||
(((512 + rmean) * rdiff * rdiff) >> 8) +
|
||
4 * gdiff * gdiff +
|
||
(((767 - rmean) * bdiff * bdiff) >> 8)
|
||
);
|
||
if (dist < menorDist) {
|
||
menorDist = dist;
|
||
cor = [pr, pg, pb];
|
||
}
|
||
}
|
||
return cor;
|
||
}
|
||
// LAB algorithm
|
||
const [Lt, at, bt] = Utils._lab(r, g, b);
|
||
const targetChroma = Math.sqrt(at * at + bt * bt);
|
||
let best = null;
|
||
let bestDist = Infinity;
|
||
for (let i = 0; i < palette.length; i++) {
|
||
const [pr, pg, pb] = palette[i];
|
||
const [Lp, ap, bp] = Utils._lab(pr, pg, pb);
|
||
const dL = Lt - Lp;
|
||
const da = at - ap;
|
||
const db = bt - bp;
|
||
let dist = dL * dL + da * da + db * db;
|
||
if (state.enableChromaPenalty && targetChroma > 20) {
|
||
const candChroma = Math.sqrt(ap * ap + bp * bp);
|
||
if (candChroma < targetChroma) {
|
||
const chromaDiff = targetChroma - candChroma;
|
||
dist += chromaDiff * chromaDiff * state.chromaPenaltyWeight;
|
||
}
|
||
}
|
||
if (dist < bestDist) {
|
||
bestDist = dist;
|
||
best = palette[i];
|
||
if (bestDist === 0) break;
|
||
}
|
||
}
|
||
return best || [0, 0, 0];
|
||
},
|
||
|
||
isWhitePixel: (r, g, b) => {
|
||
const wt = state.customWhiteThreshold || CONFIG.WHITE_THRESHOLD;
|
||
return r >= wt && g >= wt && b >= wt;
|
||
},
|
||
|
||
resolveColor(targetRgb, availableColors, exactMatch = false) {
|
||
if (!availableColors || availableColors.length === 0) {
|
||
return {
|
||
id: null,
|
||
rgb: targetRgb,
|
||
};
|
||
}
|
||
|
||
const cacheKey = `${targetRgb[0]},${targetRgb[1]},${targetRgb[2]}|${state.colorMatchingAlgorithm}|${state.enableChromaPenalty ? 'c' : 'nc'
|
||
}|${state.chromaPenaltyWeight}|${exactMatch ? 'exact' : 'closest'}`;
|
||
|
||
if (colorCache.has(cacheKey)) return colorCache.get(cacheKey);
|
||
|
||
// Check for an exact color match in availableColors.
|
||
// If found, return the matched color with its ID.
|
||
// If not found, return the target color with null ID.
|
||
// Cache the result for future lookups.
|
||
if (exactMatch) {
|
||
const match = availableColors.find(
|
||
(c) => c.rgb[0] === targetRgb[0] && c.rgb[1] === targetRgb[1] && c.rgb[2] === targetRgb[2]
|
||
);
|
||
const result = match ? { id: match.id, rgb: [...match.rgb] } : { id: null, rgb: targetRgb };
|
||
colorCache.set(cacheKey, result);
|
||
return result;
|
||
}
|
||
|
||
// check for white using threshold
|
||
const whiteThreshold = state.customWhiteThreshold || CONFIG.WHITE_THRESHOLD;
|
||
if (
|
||
targetRgb[0] >= whiteThreshold &&
|
||
targetRgb[1] >= whiteThreshold &&
|
||
targetRgb[2] >= whiteThreshold
|
||
) {
|
||
const whiteEntry = availableColors.find(
|
||
(c) =>
|
||
c.rgb[0] >= whiteThreshold && c.rgb[1] >= whiteThreshold && c.rgb[2] >= whiteThreshold
|
||
);
|
||
if (whiteEntry) {
|
||
const result = { id: whiteEntry.id, rgb: [...whiteEntry.rgb] };
|
||
colorCache.set(cacheKey, result);
|
||
return result;
|
||
}
|
||
}
|
||
|
||
// find nearest color
|
||
let bestId = availableColors[0].id;
|
||
let bestRgb = [...availableColors[0].rgb];
|
||
let bestScore = Infinity;
|
||
|
||
if (state.colorMatchingAlgorithm === 'legacy') {
|
||
for (let i = 0; i < availableColors.length; i++) {
|
||
const c = availableColors[i];
|
||
const [r, g, b] = c.rgb;
|
||
const rmean = (r + targetRgb[0]) / 2;
|
||
const rdiff = r - targetRgb[0];
|
||
const gdiff = g - targetRgb[1];
|
||
const bdiff = b - targetRgb[2];
|
||
const dist = Math.sqrt(
|
||
(((512 + rmean) * rdiff * rdiff) >> 8) +
|
||
4 * gdiff * gdiff +
|
||
(((767 - rmean) * bdiff * bdiff) >> 8)
|
||
);
|
||
if (dist < bestScore) {
|
||
bestScore = dist;
|
||
bestId = c.id;
|
||
bestRgb = [...c.rgb];
|
||
if (dist === 0) break;
|
||
}
|
||
}
|
||
} else {
|
||
const [Lt, at, bt] = Utils._lab(targetRgb[0], targetRgb[1], targetRgb[2]);
|
||
const targetChroma = Math.sqrt(at * at + bt * bt);
|
||
const penaltyWeight = state.enableChromaPenalty ? state.chromaPenaltyWeight || 0.15 : 0;
|
||
|
||
for (let i = 0; i < availableColors.length; i++) {
|
||
const c = availableColors[i];
|
||
const [r, g, b] = c.rgb;
|
||
const [L2, a2, b2] = Utils._lab(r, g, b);
|
||
const dL = Lt - L2,
|
||
da = at - a2,
|
||
db = bt - b2;
|
||
let dist = dL * dL + da * da + db * db;
|
||
|
||
if (penaltyWeight > 0 && targetChroma > 20) {
|
||
const candChroma = Math.sqrt(a2 * a2 + b2 * b2);
|
||
if (candChroma < targetChroma) {
|
||
const cd = targetChroma - candChroma;
|
||
dist += cd * cd * penaltyWeight;
|
||
}
|
||
}
|
||
|
||
if (dist < bestScore) {
|
||
bestScore = dist;
|
||
bestId = c.id;
|
||
bestRgb = [...c.rgb];
|
||
if (dist === 0) break;
|
||
}
|
||
}
|
||
}
|
||
|
||
const result = { id: bestId, rgb: bestRgb };
|
||
colorCache.set(cacheKey, result);
|
||
|
||
// limit the size of the cache
|
||
if (colorCache.size > 15000) {
|
||
const firstKey = colorCache.keys().next().value;
|
||
colorCache.delete(firstKey);
|
||
}
|
||
|
||
return result;
|
||
},
|
||
|
||
createImageUploader: () =>
|
||
new Promise((resolve) => {
|
||
const input = document.createElement('input');
|
||
input.type = 'file';
|
||
input.accept = 'image/png,image/jpeg';
|
||
input.onchange = () => {
|
||
const fr = new FileReader();
|
||
fr.onload = () => resolve(fr.result);
|
||
fr.readAsDataURL(input.files[0]);
|
||
};
|
||
input.click();
|
||
}),
|
||
|
||
createFileDownloader: (data, filename) => {
|
||
const blob = new Blob([data], { type: 'application/json' });
|
||
const url = URL.createObjectURL(blob);
|
||
const a = document.createElement('a');
|
||
a.href = url;
|
||
a.download = filename;
|
||
document.body.appendChild(a);
|
||
a.click();
|
||
document.body.removeChild(a);
|
||
URL.revokeObjectURL(url);
|
||
},
|
||
|
||
createFileUploader: () =>
|
||
new Promise((resolve, reject) => {
|
||
const input = document.createElement('input');
|
||
input.type = 'file';
|
||
input.accept = '.json';
|
||
input.onchange = (e) => {
|
||
const file = e.target.files[0];
|
||
if (file) {
|
||
const reader = new FileReader();
|
||
reader.onload = () => {
|
||
try {
|
||
const data = JSON.parse(reader.result);
|
||
resolve(data);
|
||
} catch (error) {
|
||
reject(new Error('Invalid JSON file'));
|
||
}
|
||
};
|
||
reader.onerror = () => reject(new Error('File reading error'));
|
||
reader.readAsText(file);
|
||
} else {
|
||
reject(new Error('No file selected'));
|
||
}
|
||
};
|
||
input.click();
|
||
}),
|
||
|
||
extractAvailableColors: () => {
|
||
const colorElements = document.querySelectorAll('.tooltip button[id^="color-"]');
|
||
if (colorElements.length === 0) {
|
||
console.log('❌ No color elements found on page');
|
||
return null;
|
||
}
|
||
// Separate available and unavailable colors
|
||
const availableColors = [];
|
||
const unavailableColors = [];
|
||
|
||
Array.from(colorElements).forEach((el) => {
|
||
const id = Number.parseInt(el.id.replace('color-', ''));
|
||
if (id === 0) return; // Skip transparent color
|
||
|
||
const rgbStr = el.style.backgroundColor.match(/\d+/g);
|
||
if (!rgbStr || rgbStr.length < 3) {
|
||
console.warn(`Skipping color element ${el.id} — cannot parse RGB`);
|
||
return;
|
||
}
|
||
const rgb = rgbStr.map(Number);
|
||
|
||
// Find color name from COLOR_MAP
|
||
const colorInfo = Object.values(CONFIG.COLOR_MAP).find((color) => color.id === id);
|
||
const name = colorInfo ? colorInfo.name : `Unknown Color ${id}`;
|
||
|
||
const colorData = { id, name, rgb };
|
||
|
||
// Check if color is available (no SVG overlay means available)
|
||
if (!el.querySelector('svg')) {
|
||
availableColors.push(colorData);
|
||
} else {
|
||
unavailableColors.push(colorData);
|
||
}
|
||
});
|
||
|
||
// Console log detailed color information
|
||
console.log('=== CAPTURED COLORS STATUS ===');
|
||
console.log(`Total available colors: ${availableColors.length}`);
|
||
console.log(`Total unavailable colors: ${unavailableColors.length}`);
|
||
console.log(`Total colors scanned: ${availableColors.length + unavailableColors.length}`);
|
||
|
||
if (availableColors.length > 0) {
|
||
console.log('\n--- AVAILABLE COLORS ---');
|
||
availableColors.forEach((color, index) => {
|
||
console.log(
|
||
`${index + 1
|
||
}. ID: ${color.id}, Name: "${color.name}", RGB: (${color.rgb[0]}, ${color.rgb[1]}, ${color.rgb[2]})`
|
||
);
|
||
});
|
||
}
|
||
|
||
if (unavailableColors.length > 0) {
|
||
console.log('\n--- UNAVAILABLE COLORS ---');
|
||
unavailableColors.forEach((color, index) => {
|
||
console.log(
|
||
`${index + 1
|
||
}. ID: ${color.id}, Name: "${color.name}", RGB: (${color.rgb[0]}, ${color.rgb[1]}, ${color.rgb[2]}) [LOCKED]`
|
||
);
|
||
});
|
||
}
|
||
|
||
console.log('=== END COLOR STATUS ===');
|
||
|
||
return availableColors;
|
||
},
|
||
|
||
formatTime: (ms) => {
|
||
const seconds = Math.floor((ms / 1000) % 60);
|
||
const minutes = Math.floor((ms / (1000 * 60)) % 60);
|
||
const hours = Math.floor((ms / (1000 * 60 * 60)) % 24);
|
||
const days = Math.floor(ms / (1000 * 60 * 60 * 24));
|
||
|
||
let result = '';
|
||
if (days > 0) result += `${days}d `;
|
||
if (hours > 0 || days > 0) result += `${hours}h `;
|
||
if (minutes > 0 || hours > 0 || days > 0) result += `${minutes}m `;
|
||
result += `${seconds}s`;
|
||
|
||
return result;
|
||
},
|
||
|
||
calculateEstimatedTime: (remainingPixels, charges, cooldown) => {
|
||
if (remainingPixels <= 0) return 0;
|
||
|
||
const paintingSpeedDelay = state.paintingSpeed > 0 ? 1000 / state.paintingSpeed : 1000;
|
||
const timeFromSpeed = remainingPixels * paintingSpeedDelay;
|
||
|
||
const cyclesNeeded = Math.ceil(remainingPixels / Math.max(charges, 1));
|
||
const timeFromCharges = cyclesNeeded * cooldown;
|
||
|
||
return timeFromSpeed + timeFromCharges; // combine instead of taking max
|
||
},
|
||
|
||
// --- Painted pixel tracking helpers ---
|
||
initializePaintedMap: (width, height) => {
|
||
if (!state.paintedMap || state.paintedMap.length !== height) {
|
||
state.paintedMap = Array(height)
|
||
.fill()
|
||
.map(() => Array(width).fill(false));
|
||
console.log(`📋 Initialized painted map: ${width}x${height}`);
|
||
}
|
||
},
|
||
|
||
markPixelPainted: (x, y, regionX = 0, regionY = 0) => {
|
||
const actualX = x + regionX;
|
||
const actualY = y + regionY;
|
||
|
||
if (
|
||
state.paintedMap &&
|
||
state.paintedMap[actualY] &&
|
||
actualX >= 0 &&
|
||
actualX < state.paintedMap[actualY].length
|
||
) {
|
||
state.paintedMap[actualY][actualX] = true;
|
||
}
|
||
},
|
||
|
||
isPixelPainted: (x, y, regionX = 0, regionY = 0) => {
|
||
const actualX = x + regionX;
|
||
const actualY = y + regionY;
|
||
|
||
if (
|
||
state.paintedMap &&
|
||
state.paintedMap[actualY] &&
|
||
actualX >= 0 &&
|
||
actualX < state.paintedMap[actualY].length
|
||
) {
|
||
return state.paintedMap[actualY][actualX];
|
||
}
|
||
return false;
|
||
},
|
||
|
||
// Smart save - only save if significant changes
|
||
shouldAutoSave: () => {
|
||
const now = Date.now();
|
||
const pixelsSinceLastSave = state.paintedPixels - state._lastSavePixelCount;
|
||
const timeSinceLastSave = now - state._lastSaveTime;
|
||
|
||
// Save conditions:
|
||
// 1. Every 25 pixels (reduced from 50 for more frequent saves)
|
||
// 2. At least 30 seconds since last save (prevent spam)
|
||
// 3. Not already saving
|
||
return !state._saveInProgress && pixelsSinceLastSave >= 25 && timeSinceLastSave >= 30000;
|
||
},
|
||
|
||
performSmartSave: () => {
|
||
if (!Utils.shouldAutoSave()) return false;
|
||
|
||
state._saveInProgress = true;
|
||
const success = Utils.saveProgress();
|
||
|
||
if (success) {
|
||
state._lastSavePixelCount = state.paintedPixels;
|
||
state._lastSaveTime = Date.now();
|
||
console.log(`💾 Auto-saved at ${state.paintedPixels} pixels`);
|
||
}
|
||
|
||
state._saveInProgress = false;
|
||
return success;
|
||
},
|
||
|
||
// --- Data management helpers ---
|
||
|
||
// Base64 compression helpers for efficient storage
|
||
packPaintedMapToBase64: (paintedMap, width, height) => {
|
||
if (!paintedMap || !width || !height) return null;
|
||
const totalBits = width * height;
|
||
const byteLen = Math.ceil(totalBits / 8);
|
||
const bytes = new Uint8Array(byteLen);
|
||
let bitIndex = 0;
|
||
for (let y = 0; y < height; y++) {
|
||
const row = paintedMap[y];
|
||
for (let x = 0; x < width; x++) {
|
||
const bit = row && row[x] ? 1 : 0;
|
||
const b = bitIndex >> 3; // byte index
|
||
const o = bitIndex & 7; // bit offset
|
||
if (bit) bytes[b] |= 1 << o;
|
||
bitIndex++;
|
||
}
|
||
}
|
||
let binary = '';
|
||
const chunk = 0x8000;
|
||
for (let i = 0; i < bytes.length; i += chunk) {
|
||
binary += String.fromCharCode.apply(
|
||
null,
|
||
bytes.subarray(i, Math.min(i + chunk, bytes.length))
|
||
);
|
||
}
|
||
return btoa(binary);
|
||
},
|
||
|
||
unpackPaintedMapFromBase64: (base64, width, height) => {
|
||
if (!base64 || !width || !height) return null;
|
||
const binary = atob(base64);
|
||
const bytes = new Uint8Array(binary.length);
|
||
for (let i = 0; i < binary.length; i++) bytes[i] = binary.charCodeAt(i);
|
||
const map = Array(height)
|
||
.fill()
|
||
.map(() => Array(width).fill(false));
|
||
let bitIndex = 0;
|
||
for (let y = 0; y < height; y++) {
|
||
for (let x = 0; x < width; x++) {
|
||
const b = bitIndex >> 3;
|
||
const o = bitIndex & 7;
|
||
map[y][x] = ((bytes[b] >> o) & 1) === 1;
|
||
bitIndex++;
|
||
}
|
||
}
|
||
return map;
|
||
},
|
||
|
||
// Migration helpers for backward compatibility
|
||
migrateProgressToV2: (saved) => {
|
||
if (!saved) return saved;
|
||
const isV1 =
|
||
!saved.version ||
|
||
saved.version === '1' ||
|
||
saved.version === '1.0' ||
|
||
saved.version === '1.1';
|
||
if (!isV1) return saved;
|
||
|
||
try {
|
||
const migrated = { ...saved };
|
||
const width = migrated.imageData?.width;
|
||
const height = migrated.imageData?.height;
|
||
if (migrated.paintedMap && width && height) {
|
||
const data = Utils.packPaintedMapToBase64(migrated.paintedMap, width, height);
|
||
migrated.paintedMapPacked = { width, height, data };
|
||
}
|
||
delete migrated.paintedMap;
|
||
migrated.version = '2';
|
||
return migrated;
|
||
} catch (e) {
|
||
console.warn('Migration to v2 failed, using original data:', e);
|
||
return saved;
|
||
}
|
||
},
|
||
|
||
migrateProgressToV21: (saved) => {
|
||
if (!saved) return saved;
|
||
if (saved.version === '2.1') return saved;
|
||
const isV2 = saved.version === '2' || saved.version === '2.0';
|
||
const isV1 =
|
||
!saved.version ||
|
||
saved.version === '1' ||
|
||
saved.version === '1.0' ||
|
||
saved.version === '1.1';
|
||
if (!isV2 && !isV1) return saved; // save this for future
|
||
try {
|
||
const migrated = { ...saved };
|
||
// First migrate to v2 if needed
|
||
if (isV1) {
|
||
const width = migrated.imageData?.width;
|
||
const height = migrated.imageData?.height;
|
||
if (migrated.paintedMap && width && height) {
|
||
const data = Utils.packPaintedMapToBase64(migrated.paintedMap, width, height);
|
||
migrated.paintedMapPacked = { width, height, data };
|
||
}
|
||
delete migrated.paintedMap;
|
||
}
|
||
migrated.version = '2.1';
|
||
return migrated;
|
||
} catch (e) {
|
||
console.warn('Migration to v2.1 failed, using original data:', e);
|
||
return saved;
|
||
}
|
||
},
|
||
|
||
migrateProgressToV22: (data) => {
|
||
try {
|
||
const migrated = { ...data };
|
||
migrated.version = '2.2';
|
||
|
||
// Add new fields with default values
|
||
if (!migrated.state.coordinateMode) {
|
||
migrated.state.coordinateMode = CONFIG.COORDINATE_MODE;
|
||
}
|
||
if (!migrated.state.coordinateDirection) {
|
||
migrated.state.coordinateDirection = CONFIG.COORDINATE_DIRECTION;
|
||
}
|
||
if (!migrated.state.coordinateSnake) {
|
||
migrated.state.coordinateSnake = CONFIG.COORDINATE_SNAKE;
|
||
}
|
||
if (!migrated.state.blockWidth) {
|
||
migrated.state.blockWidth = CONFIG.COORDINATE_BLOCK_WIDTH;
|
||
}
|
||
if (!migrated.state.blockHeight) {
|
||
migrated.state.blockHeight = CONFIG.COORDINATE_BLOCK_HEIGHT;
|
||
}
|
||
|
||
return migrated;
|
||
} catch (e) {
|
||
console.warn('Migration to v2.2 failed, using original data:', e);
|
||
return data;
|
||
}
|
||
},
|
||
|
||
buildPaintedMapPacked() {
|
||
if (state.paintedMap && state.imageData) {
|
||
const data = Utils.packPaintedMapToBase64(
|
||
state.paintedMap,
|
||
state.imageData.width,
|
||
state.imageData.height
|
||
);
|
||
if (data) {
|
||
return {
|
||
width: state.imageData.width,
|
||
height: state.imageData.height,
|
||
data: data,
|
||
};
|
||
}
|
||
}
|
||
return null;
|
||
},
|
||
|
||
buildProgressData() {
|
||
return {
|
||
timestamp: Date.now(),
|
||
version: '2.2',
|
||
state: {
|
||
totalPixels: state.totalPixels,
|
||
paintedPixels: state.paintedPixels,
|
||
lastPosition: state.lastPosition,
|
||
startPosition: state.startPosition,
|
||
region: state.region,
|
||
imageLoaded: state.imageLoaded,
|
||
colorsChecked: state.colorsChecked,
|
||
coordinateMode: state.coordinateMode,
|
||
coordinateDirection: state.coordinateDirection,
|
||
coordinateSnake: state.coordinateSnake,
|
||
blockWidth: state.blockWidth,
|
||
blockHeight: state.blockHeight,
|
||
availableColors: state.availableColors,
|
||
},
|
||
imageData: state.imageData
|
||
? {
|
||
width: state.imageData.width,
|
||
height: state.imageData.height,
|
||
pixels: Array.from(state.imageData.pixels),
|
||
totalPixels: state.imageData.totalPixels,
|
||
}
|
||
: null,
|
||
paintedMapPacked: Utils.buildPaintedMapPacked(),
|
||
};
|
||
},
|
||
|
||
migrateProgress(saved) {
|
||
if (!saved) return null;
|
||
|
||
let data = saved;
|
||
const ver = data.version;
|
||
|
||
// If version is missing or ≤ 1.x → first migrate to v2
|
||
if (!ver || ver === '1' || ver === '1.0' || ver === '1.1') {
|
||
data = Utils.migrateProgressToV2(data);
|
||
}
|
||
|
||
// If still older than v2.1 → migrate to 2.1
|
||
if (data.version === '2' || data.version === '2.0') {
|
||
data = Utils.migrateProgressToV21(data);
|
||
}
|
||
|
||
// If still older than v2.2 → migrate to 2.2
|
||
if (data.version === '2.1') {
|
||
data = Utils.migrateProgressToV22(data);
|
||
}
|
||
|
||
// Now data is guaranteed to be the latest version
|
||
return data;
|
||
},
|
||
|
||
saveProgress: () => {
|
||
try {
|
||
const progressData = Utils.buildProgressData(state);
|
||
|
||
localStorage.setItem('wplace-bot-progress', JSON.stringify(progressData));
|
||
return true;
|
||
} catch (error) {
|
||
console.error('Error saving progress:', error);
|
||
return false;
|
||
}
|
||
},
|
||
|
||
loadProgress: () => {
|
||
try {
|
||
const saved = localStorage.getItem('wplace-bot-progress');
|
||
if (!saved) return null;
|
||
let data = JSON.parse(saved);
|
||
const migrated = Utils.migrateProgress(data);
|
||
|
||
if (migrated && migrated !== data) {
|
||
try {
|
||
localStorage.setItem('wplace-bot-progress', JSON.stringify(migrated));
|
||
} catch { }
|
||
}
|
||
return migrated;
|
||
} catch (error) {
|
||
console.error('Error loading progress:', error);
|
||
return null;
|
||
}
|
||
},
|
||
|
||
clearProgress: () => {
|
||
try {
|
||
localStorage.removeItem('wplace-bot-progress');
|
||
// Also clear painted map from memory
|
||
state.paintedMap = null;
|
||
state._lastSavePixelCount = 0;
|
||
state._lastSaveTime = 0;
|
||
// Reset coordinate generation settings to their default values
|
||
state.coordinateMode = CONFIG.COORDINATE_MODE;
|
||
state.coordinateDirection = CONFIG.COORDINATE_DIRECTION;
|
||
state.coordinateSnake = CONFIG.COORDINATE_SNAKE;
|
||
state.blockWidth = CONFIG.COORDINATE_BLOCK_WIDTH;
|
||
state.blockHeight = CONFIG.COORDINATE_BLOCK_HEIGHT;
|
||
console.log('📋 Progress and painted map cleared');
|
||
return true;
|
||
} catch (error) {
|
||
console.error('Error clearing progress:', error);
|
||
return false;
|
||
}
|
||
},
|
||
|
||
restoreProgress: (savedData) => {
|
||
try {
|
||
Object.assign(state, savedData.state);
|
||
|
||
// Restore coordinate generation settings
|
||
if (savedData.state.coordinateMode) {
|
||
state.coordinateMode = savedData.state.coordinateMode;
|
||
}
|
||
if (savedData.state.coordinateDirection) {
|
||
state.coordinateDirection = savedData.state.coordinateDirection;
|
||
}
|
||
if (savedData.state.coordinateSnake !== undefined) {
|
||
state.coordinateSnake = savedData.state.coordinateSnake;
|
||
}
|
||
if (savedData.state.blockWidth) {
|
||
state.blockWidth = savedData.state.blockWidth;
|
||
}
|
||
if (savedData.state.blockHeight) {
|
||
state.blockHeight = savedData.state.blockHeight;
|
||
}
|
||
|
||
if (savedData.imageData) {
|
||
state.imageData = {
|
||
...savedData.imageData,
|
||
pixels: new Uint8ClampedArray(savedData.imageData.pixels),
|
||
};
|
||
|
||
try {
|
||
const canvas = document.createElement('canvas');
|
||
canvas.width = state.imageData.width;
|
||
canvas.height = state.imageData.height;
|
||
const ctx = canvas.getContext('2d');
|
||
const imageData = new ImageData(
|
||
state.imageData.pixels,
|
||
state.imageData.width,
|
||
state.imageData.height
|
||
);
|
||
ctx.putImageData(imageData, 0, 0);
|
||
const proc = new ImageProcessor('');
|
||
proc.img = canvas;
|
||
proc.canvas = canvas;
|
||
proc.ctx = ctx;
|
||
state.imageData.processor = proc;
|
||
} catch (e) {
|
||
console.warn('Could not rebuild processor from saved image data:', e);
|
||
}
|
||
}
|
||
|
||
// Prefer packed form if available; fallback to legacy paintedMap array for backward compatibility
|
||
if (savedData.paintedMapPacked && savedData.paintedMapPacked.data) {
|
||
const { width, height, data } = savedData.paintedMapPacked;
|
||
state.paintedMap = Utils.unpackPaintedMapFromBase64(data, width, height);
|
||
} else if (savedData.paintedMap) {
|
||
state.paintedMap = savedData.paintedMap.map((row) => Array.from(row));
|
||
}
|
||
|
||
return true;
|
||
} catch (error) {
|
||
console.error('Error restoring progress:', error);
|
||
return false;
|
||
}
|
||
},
|
||
|
||
saveProgressToFile: () => {
|
||
try {
|
||
const progressData = Utils.buildProgressData();
|
||
const filename = `wplace-bot-progress-${new Date()
|
||
.toISOString()
|
||
.slice(0, 19)
|
||
.replace(/:/g, '-')}.json`;
|
||
Utils.createFileDownloader(JSON.stringify(progressData, null, 2), filename);
|
||
return true;
|
||
} catch (error) {
|
||
console.error('Error saving to file:', error);
|
||
return false;
|
||
}
|
||
},
|
||
|
||
loadProgressFromFile: async () => {
|
||
try {
|
||
const data = await Utils.createFileUploader();
|
||
if (!data || !data.state) {
|
||
throw new Error('Invalid file format');
|
||
}
|
||
const migrated = Utils.migrateProgress(data);
|
||
|
||
const success = Utils.restoreProgress(migrated);
|
||
return success;
|
||
} catch (error) {
|
||
console.error('Error loading from file:', error);
|
||
throw error;
|
||
}
|
||
},
|
||
|
||
// Helper function to restore overlay from loaded data
|
||
restoreOverlayFromData: async () => {
|
||
if (!state.imageLoaded || !state.imageData || !state.startPosition || !state.region) {
|
||
return false;
|
||
}
|
||
|
||
try {
|
||
// Recreate ImageBitmap from loaded pixel data
|
||
const imageData = new ImageData(
|
||
state.imageData.pixels,
|
||
state.imageData.width,
|
||
state.imageData.height
|
||
);
|
||
|
||
const canvas = new OffscreenCanvas(state.imageData.width, state.imageData.height);
|
||
const ctx = canvas.getContext('2d');
|
||
ctx.putImageData(imageData, 0, 0);
|
||
const imageBitmap = await canvas.transferToImageBitmap();
|
||
|
||
// Set up overlay with restored data
|
||
await overlayManager.setImage(imageBitmap);
|
||
await overlayManager.setPosition(state.startPosition, state.region);
|
||
overlayManager.enable();
|
||
|
||
// Update overlay button state
|
||
const toggleOverlayBtn = document.getElementById('toggleOverlayBtn');
|
||
if (toggleOverlayBtn) {
|
||
toggleOverlayBtn.disabled = false;
|
||
toggleOverlayBtn.classList.add('active');
|
||
}
|
||
|
||
console.log('Overlay restored from data');
|
||
return true;
|
||
} catch (error) {
|
||
console.error('Failed to restore overlay from data:', error);
|
||
return false;
|
||
}
|
||
},
|
||
|
||
updateCoordinateUI({ mode, directionControls, snakeControls, blockControls }) {
|
||
const isLinear = mode === 'rows' || mode === 'columns';
|
||
const isBlock = mode === 'blocks' || mode === 'shuffle-blocks';
|
||
|
||
if (directionControls) directionControls.style.display = isLinear ? 'block' : 'none';
|
||
if (snakeControls) snakeControls.style.display = isLinear ? 'block' : 'none';
|
||
if (blockControls) blockControls.style.display = isBlock ? 'block' : 'none';
|
||
},
|
||
};
|
||
|
||
// IMAGE PROCESSOR CLASS
|
||
class ImageProcessor {
|
||
constructor(imageSrc) {
|
||
this.imageSrc = imageSrc;
|
||
this.img = null;
|
||
this.canvas = null;
|
||
this.ctx = null;
|
||
}
|
||
|
||
async load() {
|
||
return new Promise((resolve, reject) => {
|
||
this.img = new Image();
|
||
this.img.crossOrigin = 'anonymous';
|
||
this.img.onload = () => {
|
||
this.canvas = document.createElement('canvas');
|
||
this.ctx = this.canvas.getContext('2d');
|
||
this.canvas.width = this.img.width;
|
||
this.canvas.height = this.img.height;
|
||
this.ctx.drawImage(this.img, 0, 0);
|
||
resolve();
|
||
};
|
||
this.img.onerror = reject;
|
||
this.img.src = this.imageSrc;
|
||
});
|
||
}
|
||
|
||
getDimensions() {
|
||
return {
|
||
width: this.canvas.width,
|
||
height: this.canvas.height,
|
||
};
|
||
}
|
||
|
||
getPixelData() {
|
||
return this.ctx.getImageData(0, 0, this.canvas.width, this.canvas.height).data;
|
||
}
|
||
|
||
resize(newWidth, newHeight) {
|
||
const tempCanvas = document.createElement('canvas');
|
||
const tempCtx = tempCanvas.getContext('2d');
|
||
|
||
tempCanvas.width = newWidth;
|
||
tempCanvas.height = newHeight;
|
||
|
||
tempCtx.imageSmoothingEnabled = false;
|
||
tempCtx.drawImage(this.canvas, 0, 0, newWidth, newHeight);
|
||
|
||
this.canvas.width = newWidth;
|
||
this.canvas.height = newHeight;
|
||
this.ctx.imageSmoothingEnabled = false;
|
||
this.ctx.drawImage(tempCanvas, 0, 0);
|
||
|
||
return this.ctx.getImageData(0, 0, newWidth, newHeight).data;
|
||
}
|
||
|
||
generatePreview(width, height) {
|
||
const previewCanvas = document.createElement('canvas');
|
||
const previewCtx = previewCanvas.getContext('2d');
|
||
|
||
previewCanvas.width = width;
|
||
previewCanvas.height = height;
|
||
|
||
previewCtx.imageSmoothingEnabled = false;
|
||
previewCtx.drawImage(this.img, 0, 0, width, height);
|
||
|
||
return previewCanvas.toDataURL();
|
||
}
|
||
}
|
||
|
||
// WPLACE API SERVICE
|
||
const WPlaceService = {
|
||
async paintPixelInRegion(regionX, regionY, pixelX, pixelY, color) {
|
||
try {
|
||
await ensureToken();
|
||
if (!turnstileToken) return 'token_error';
|
||
const payload = {
|
||
coords: [pixelX, pixelY],
|
||
colors: [color],
|
||
t: turnstileToken,
|
||
fp: fpStr32,
|
||
};
|
||
var token = await createWasmToken(regionX, regionY, payload);
|
||
const res = await fetch(`https://backend.wplace.live/s0/pixel/${regionX}/${regionY}`, {
|
||
method: 'POST',
|
||
headers: { 'Content-Type': 'text/plain;charset=UTF-8', "x-pawtect-token": token },
|
||
credentials: 'include',
|
||
body: JSON.stringify(payload),
|
||
});
|
||
if (res.status === 403) {
|
||
console.error('❌ 403 Forbidden. Turnstile token might be invalid or expired.');
|
||
turnstileToken = null;
|
||
tokenPromise = new Promise((resolve) => {
|
||
_resolveToken = resolve;
|
||
});
|
||
return 'token_error';
|
||
}
|
||
const data = await res.json();
|
||
return data?.painted === 1;
|
||
} catch (e) {
|
||
console.error('Paint request failed:', e);
|
||
return false;
|
||
}
|
||
},
|
||
|
||
async getCharges() {
|
||
const defaultResult = {
|
||
charges: 0,
|
||
max: 1,
|
||
cooldown: CONFIG.COOLDOWN_DEFAULT,
|
||
};
|
||
|
||
try {
|
||
const res = await fetch('https://backend.wplace.live/me', {
|
||
credentials: 'include',
|
||
});
|
||
|
||
if (!res.ok) {
|
||
console.error(`Failed to get charges: HTTP ${res.status}`);
|
||
return defaultResult;
|
||
}
|
||
|
||
const data = await res.json();
|
||
|
||
return {
|
||
charges: data.charges?.count ?? 0,
|
||
max: data.charges?.max ?? 1,
|
||
cooldown: data.charges?.cooldownMs ?? CONFIG.COOLDOWN_DEFAULT,
|
||
};
|
||
} catch (e) {
|
||
console.error('Failed to get charges:', e);
|
||
return defaultResult;
|
||
}
|
||
},
|
||
};
|
||
|
||
// Desktop Notification Manager
|
||
const NotificationManager = {
|
||
pollTimer: null,
|
||
pollIntervalMs: 60_000,
|
||
icon() {
|
||
const link = document.querySelector("link[rel~='icon']");
|
||
return link?.href || location.origin + '/favicon.ico';
|
||
},
|
||
async requestPermission() {
|
||
if (!('Notification' in window)) {
|
||
Utils.showAlert(Utils.t('notificationsNotSupported'), 'warning');
|
||
return 'denied';
|
||
}
|
||
if (Notification.permission === 'granted') return 'granted';
|
||
try {
|
||
const perm = await Notification.requestPermission();
|
||
return perm;
|
||
} catch {
|
||
return Notification.permission;
|
||
}
|
||
},
|
||
canNotify() {
|
||
return (
|
||
state.notificationsEnabled &&
|
||
typeof Notification !== 'undefined' &&
|
||
Notification.permission === 'granted'
|
||
);
|
||
},
|
||
notify(title, body, tag = 'wplace-charges', force = false) {
|
||
if (!this.canNotify()) return false;
|
||
if (!force && state.notifyOnlyWhenUnfocused && document.hasFocus()) return false;
|
||
try {
|
||
new Notification(title, {
|
||
body,
|
||
tag,
|
||
renotify: true,
|
||
icon: this.icon(),
|
||
badge: this.icon(),
|
||
silent: false,
|
||
});
|
||
return true;
|
||
} catch {
|
||
// Graceful fallback
|
||
Utils.showAlert(body, 'info');
|
||
return false;
|
||
}
|
||
},
|
||
resetEdgeTracking() {
|
||
state._lastChargesBelow = state.displayCharges < state.cooldownChargeThreshold;
|
||
state._lastChargesNotifyAt = 0;
|
||
},
|
||
maybeNotifyChargesReached(force = false) {
|
||
if (!state.notificationsEnabled || !state.notifyOnChargesReached) return;
|
||
const reached = state.displayCharges >= state.cooldownChargeThreshold;
|
||
const now = Date.now();
|
||
const repeatMs = Math.max(1, Number(state.notificationIntervalMinutes || 5)) * 60_000;
|
||
if (reached) {
|
||
const shouldEdge = state._lastChargesBelow || force;
|
||
const shouldRepeat = now - (state._lastChargesNotifyAt || 0) >= repeatMs;
|
||
if (shouldEdge || shouldRepeat) {
|
||
const msg = Utils.t('chargesReadyMessage', {
|
||
current: state.displayCharges,
|
||
max: state.maxCharges,
|
||
threshold: state.cooldownChargeThreshold,
|
||
});
|
||
this.notify(Utils.t('chargesReadyNotification'), msg, 'wplace-notify-charges');
|
||
state._lastChargesNotifyAt = now;
|
||
}
|
||
state._lastChargesBelow = false;
|
||
} else {
|
||
state._lastChargesBelow = true;
|
||
}
|
||
},
|
||
startPolling() {
|
||
this.stopPolling();
|
||
if (!state.notificationsEnabled || !state.notifyOnChargesReached) return;
|
||
// lightweight background polling
|
||
this.pollTimer = setInterval(async () => {
|
||
try {
|
||
const { charges, cooldown, max } = await WPlaceService.getCharges();
|
||
state.displayCharges = Math.floor(charges);
|
||
state.cooldown = cooldown;
|
||
state.maxCharges = Math.max(1, Math.floor(max));
|
||
this.maybeNotifyChargesReached();
|
||
} catch {
|
||
/* ignore */
|
||
}
|
||
}, this.pollIntervalMs);
|
||
},
|
||
stopPolling() {
|
||
if (this.pollTimer) {
|
||
clearInterval(this.pollTimer);
|
||
this.pollTimer = null;
|
||
}
|
||
},
|
||
syncFromState() {
|
||
this.resetEdgeTracking();
|
||
if (state.notificationsEnabled && state.notifyOnChargesReached) this.startPolling();
|
||
else this.stopPolling();
|
||
},
|
||
};
|
||
|
||
// COLOR MATCHING FUNCTION - Optimized with caching
|
||
const colorCache = new Map();
|
||
|
||
// UI UPDATE FUNCTIONS (declared early to avoid reference errors)
|
||
let updateUI = () => { };
|
||
let updateStats = (isManualRefresh) => { };
|
||
let updateDataButtons = () => { };
|
||
|
||
function updateActiveColorPalette() {
|
||
state.activeColorPalette = [];
|
||
const activeSwatches = document.querySelectorAll('.wplace-color-swatch.active');
|
||
if (activeSwatches) {
|
||
activeSwatches.forEach((swatch) => {
|
||
const rgbStr = swatch.getAttribute('data-rgb');
|
||
if (rgbStr) {
|
||
const rgb = rgbStr.split(',').map(Number);
|
||
state.activeColorPalette.push(rgb);
|
||
}
|
||
});
|
||
}
|
||
if (document.querySelector('.resize-container')?.style.display === 'block') {
|
||
_updateResizePreview();
|
||
}
|
||
}
|
||
|
||
function toggleAllColors(select, showingUnavailable = false) {
|
||
const swatches = document.querySelectorAll('.wplace-color-swatch');
|
||
if (swatches) {
|
||
swatches.forEach((swatch) => {
|
||
// Only toggle colors that are available or if we're showing unavailable colors
|
||
const isUnavailable = swatch.classList.contains('unavailable');
|
||
if (!isUnavailable || showingUnavailable) {
|
||
// Don't try to select unavailable colors
|
||
if (!isUnavailable) {
|
||
swatch.classList.toggle('active', select);
|
||
}
|
||
}
|
||
});
|
||
}
|
||
updateActiveColorPalette();
|
||
}
|
||
|
||
function unselectAllPaidColors() {
|
||
const swatches = document.querySelectorAll('.wplace-color-swatch');
|
||
if (swatches) {
|
||
swatches.forEach((swatch) => {
|
||
const colorId = parseInt(swatch.getAttribute('data-color-id'), 10);
|
||
if (!isNaN(colorId) && colorId >= 32) {
|
||
swatch.classList.toggle('active', false);
|
||
}
|
||
});
|
||
}
|
||
updateActiveColorPalette();
|
||
}
|
||
|
||
function initializeColorPalette(container) {
|
||
const colorsContainer = container.querySelector('#colors-container');
|
||
const showAllToggle = container.querySelector('#showAllColorsToggle');
|
||
if (!colorsContainer) return;
|
||
|
||
// Use already captured colors from state (captured during upload)
|
||
// Don't re-fetch colors here, use what was captured when user clicked upload
|
||
if (!state.availableColors || state.availableColors.length === 0) {
|
||
// If no colors have been captured yet, show message
|
||
colorsContainer.innerHTML = `<div class="wplace-colors-placeholder">${Utils.t(
|
||
'uploadImageFirst'
|
||
)}</div>`;
|
||
return;
|
||
}
|
||
|
||
function populateColors(showUnavailable = false) {
|
||
colorsContainer.innerHTML = '';
|
||
let availableCount = 0;
|
||
let totalCount = 0;
|
||
|
||
// Convert COLOR_MAP to array and filter out transparent
|
||
const allColors = Object.values(CONFIG.COLOR_MAP).filter((color) => color.rgb !== null);
|
||
|
||
allColors.forEach((colorData) => {
|
||
const { id, name, rgb } = colorData;
|
||
const rgbKey = `${rgb.r},${rgb.g},${rgb.b}`;
|
||
totalCount++;
|
||
|
||
// Check if this color is available in the captured colors
|
||
const isAvailable = state.availableColors.some(
|
||
(c) => c.rgb[0] === rgb.r && c.rgb[1] === rgb.g && c.rgb[2] === rgb.b
|
||
);
|
||
|
||
// If not showing all colors and this color is not available, skip it
|
||
if (!showUnavailable && !isAvailable) {
|
||
return;
|
||
}
|
||
|
||
if (isAvailable) availableCount++;
|
||
|
||
const colorItem = Utils.createElement('div', {
|
||
className: 'wplace-color-item',
|
||
});
|
||
const swatch = Utils.createElement('button', {
|
||
className: `wplace-color-swatch ${!isAvailable ? 'unavailable' : ''}`,
|
||
title: `${name} (ID: ${id})${!isAvailable ? ' (Unavailable)' : ''}`,
|
||
'data-rgb': rgbKey,
|
||
'data-color-id': id,
|
||
});
|
||
swatch.style.backgroundColor = `rgb(${rgb.r}, ${rgb.g}, ${rgb.b})`;
|
||
|
||
// Make unavailable colors visually distinct
|
||
if (!isAvailable) {
|
||
swatch.style.opacity = '0.4';
|
||
swatch.style.filter = 'grayscale(50%)';
|
||
swatch.disabled = true;
|
||
} else {
|
||
// Select available colors by default
|
||
swatch.classList.add('active');
|
||
}
|
||
|
||
const nameLabel = Utils.createElement(
|
||
'span',
|
||
{
|
||
className: 'wplace-color-item-name',
|
||
style: !isAvailable ? 'color: #888; font-style: italic;' : '',
|
||
},
|
||
name + (!isAvailable ? ' (N/A)' : '')
|
||
);
|
||
|
||
// Only add click listener for available colors
|
||
if (isAvailable) {
|
||
swatch.addEventListener('click', () => {
|
||
swatch.classList.toggle('active');
|
||
updateActiveColorPalette();
|
||
});
|
||
}
|
||
|
||
colorItem.appendChild(swatch);
|
||
colorItem.appendChild(nameLabel);
|
||
colorsContainer.appendChild(colorItem);
|
||
});
|
||
|
||
updateActiveColorPalette();
|
||
}
|
||
|
||
// Initialize with only available colors
|
||
populateColors(false);
|
||
|
||
// Add toggle functionality
|
||
if (showAllToggle) {
|
||
showAllToggle.addEventListener('change', (e) => {
|
||
populateColors(e.target.checked);
|
||
});
|
||
}
|
||
|
||
container
|
||
.querySelector('#selectAllBtn')
|
||
?.addEventListener('click', () => toggleAllColors(true, showAllToggle?.checked));
|
||
container
|
||
.querySelector('#unselectAllBtn')
|
||
?.addEventListener('click', () => toggleAllColors(false, showAllToggle?.checked));
|
||
container
|
||
.querySelector('#unselectPaidBtn')
|
||
?.addEventListener('click', () => unselectAllPaidColors());
|
||
}
|
||
|
||
async function handleCaptcha() {
|
||
const startTime = performance.now();
|
||
|
||
// Check user's token source preference
|
||
if (state.tokenSource === 'manual') {
|
||
console.log('🎯 Manual token source selected - using pixel placement automation');
|
||
return await handleCaptchaFallback();
|
||
}
|
||
|
||
// Generator mode (pure) or Hybrid mode - try generator first
|
||
try {
|
||
// Use optimized token generation with automatic sitekey detection
|
||
const { sitekey, token: preGeneratedToken } = await Utils.obtainSitekeyAndToken();
|
||
|
||
if (!sitekey) {
|
||
throw new Error('No valid sitekey found');
|
||
}
|
||
|
||
console.log('🔑 Generating Turnstile token for sitekey:', sitekey);
|
||
console.log(
|
||
'🧭 UA:',
|
||
navigator.userAgent.substring(0, 50) + '...',
|
||
'Platform:',
|
||
navigator.platform
|
||
);
|
||
|
||
// Add additional checks before token generation
|
||
if (!window.turnstile) {
|
||
await Utils.loadTurnstile();
|
||
}
|
||
|
||
let token = null;
|
||
|
||
// ✅ Reuse pre-generated token if available and valid
|
||
if (
|
||
preGeneratedToken &&
|
||
typeof preGeneratedToken === 'string' &&
|
||
preGeneratedToken.length > 20
|
||
) {
|
||
console.log('♻️ Reusing pre-generated token from sitekey detection phase');
|
||
token = preGeneratedToken;
|
||
}
|
||
// ✅ Or use globally cached token if still valid
|
||
else if (isTokenValid()) {
|
||
console.log('♻️ Using existing cached token (from previous operation)');
|
||
token = turnstileToken;
|
||
}
|
||
// ✅ Otherwise generate a new one
|
||
else {
|
||
console.log('🔐 No valid pre-generated or cached token, creating new one...');
|
||
token = await Utils.executeTurnstile(sitekey, 'paint');
|
||
if (token) {
|
||
setTurnstileToken(token);
|
||
}
|
||
}
|
||
|
||
// 📊 Debug log
|
||
console.log(
|
||
`🔍 Token received - Type: ${typeof token}, Value: ${token
|
||
? typeof token === 'string'
|
||
? token.length > 50
|
||
? token.substring(0, 50) + '...'
|
||
: token
|
||
: JSON.stringify(token)
|
||
: 'null/undefined'
|
||
}, Length: ${token?.length || 0}`
|
||
);
|
||
|
||
// ✅ Final validation
|
||
if (typeof token === 'string' && token.length > 20) {
|
||
const duration = Math.round(performance.now() - startTime);
|
||
console.log(`✅ Turnstile token generated successfully in ${duration}ms`);
|
||
return token;
|
||
} else {
|
||
throw new Error(
|
||
`Invalid or empty token received - Type: ${typeof token}, Value: ${JSON.stringify(
|
||
token
|
||
)}, Length: ${token?.length || 0}`
|
||
);
|
||
}
|
||
} catch (error) {
|
||
const duration = Math.round(performance.now() - startTime);
|
||
console.error(`❌ Turnstile token generation failed after ${duration}ms:`, error);
|
||
|
||
// Fallback to manual pixel placement for hybrid mode
|
||
if (state.tokenSource === 'hybrid') {
|
||
console.log(
|
||
'🔄 Hybrid mode: Generator failed, automatically switching to manual pixel placement...'
|
||
);
|
||
const fbToken = await handleCaptchaFallback();
|
||
return fbToken;
|
||
} else {
|
||
// Pure generator mode - don't fallback, just fail
|
||
throw error;
|
||
}
|
||
}
|
||
}
|
||
|
||
// Keep original method as fallback
|
||
async function handleCaptchaFallback() {
|
||
return new Promise(async (resolve, reject) => {
|
||
try {
|
||
// Ensure we have a fresh promise to await for a new token capture
|
||
if (!_resolveToken) {
|
||
tokenPromise = new Promise((res) => {
|
||
_resolveToken = res;
|
||
});
|
||
}
|
||
const timeoutPromise = Utils.sleep(20000).then(() =>
|
||
reject(new Error('Auto-CAPTCHA timed out.'))
|
||
);
|
||
|
||
const solvePromise = (async () => {
|
||
const mainPaintBtn = await Utils.waitForSelector(
|
||
'button.btn.btn-primary.btn-lg, button.btn-primary.sm\\:btn-xl',
|
||
200,
|
||
10000
|
||
);
|
||
if (!mainPaintBtn) throw new Error('Could not find the main paint button.');
|
||
mainPaintBtn.click();
|
||
await Utils.sleep(500);
|
||
|
||
const transBtn = await Utils.waitForSelector('button#color-0', 200, 5000);
|
||
if (!transBtn) throw new Error('Could not find the transparent color button.');
|
||
transBtn.click();
|
||
await Utils.sleep(500);
|
||
|
||
const canvas = await Utils.waitForSelector('canvas', 200, 5000);
|
||
if (!canvas) throw new Error('Could not find the canvas element.');
|
||
|
||
canvas.setAttribute('tabindex', '0');
|
||
canvas.focus();
|
||
const rect = canvas.getBoundingClientRect();
|
||
const centerX = Math.round(rect.left + rect.width / 2);
|
||
const centerY = Math.round(rect.top + rect.height / 2);
|
||
|
||
canvas.dispatchEvent(
|
||
new MouseEvent('mousemove', {
|
||
clientX: centerX,
|
||
clientY: centerY,
|
||
bubbles: true,
|
||
})
|
||
);
|
||
canvas.dispatchEvent(
|
||
new KeyboardEvent('keydown', {
|
||
key: ' ',
|
||
code: 'Space',
|
||
bubbles: true,
|
||
})
|
||
);
|
||
await Utils.sleep(50);
|
||
canvas.dispatchEvent(
|
||
new KeyboardEvent('keyup', {
|
||
key: ' ',
|
||
code: 'Space',
|
||
bubbles: true,
|
||
})
|
||
);
|
||
await Utils.sleep(500);
|
||
|
||
// 800ms delay before sending confirmation
|
||
await Utils.sleep(800);
|
||
|
||
// Keep confirming until token is captured
|
||
const confirmLoop = async () => {
|
||
while (!turnstileToken) {
|
||
let confirmBtn = await Utils.waitForSelector(
|
||
'button.btn.btn-primary.btn-lg, button.btn.btn-primary.sm\\:btn-xl'
|
||
);
|
||
if (!confirmBtn) {
|
||
const allPrimary = Array.from(document.querySelectorAll('button.btn-primary'));
|
||
confirmBtn = allPrimary.length ? allPrimary[allPrimary.length - 1] : null;
|
||
}
|
||
if (confirmBtn) {
|
||
confirmBtn.click();
|
||
}
|
||
await Utils.sleep(500); // 500ms delay between confirmation attempts
|
||
}
|
||
};
|
||
|
||
// Start confirmation loop and wait for token
|
||
confirmLoop();
|
||
const token = await tokenPromise;
|
||
await Utils.sleep(300); // small delay after token is captured
|
||
resolve(token);
|
||
})();
|
||
|
||
await Promise.race([solvePromise, timeoutPromise]);
|
||
} catch (error) {
|
||
console.error('Auto-CAPTCHA process failed:', error);
|
||
reject(error);
|
||
}
|
||
});
|
||
}
|
||
|
||
async function createUI() {
|
||
await detectLanguage();
|
||
|
||
const existingContainer = document.getElementById('wplace-image-bot-container');
|
||
const existingStats = document.getElementById('wplace-stats-container');
|
||
const existingSettings = document.getElementById('wplace-settings-container');
|
||
const existingResizeContainer = document.querySelector('.resize-container');
|
||
const existingResizeOverlay = document.querySelector('.resize-overlay');
|
||
|
||
if (existingContainer) existingContainer.remove();
|
||
if (existingStats) existingStats.remove();
|
||
if (existingSettings) existingSettings.remove();
|
||
if (existingResizeContainer) existingResizeContainer.remove();
|
||
if (existingResizeOverlay) existingResizeOverlay.remove();
|
||
|
||
loadThemePreference();
|
||
await initializeTranslations();
|
||
|
||
const theme = getCurrentTheme();
|
||
applyTheme(); // <- new: set CSS vars and theme class before building UI
|
||
|
||
function appendLinkOnce(href, attributes = {}) {
|
||
// Check if a link with the same href already exists in the document head
|
||
const exists = Array.from(document.head.querySelectorAll('link')).some(
|
||
(link) => link.href === href
|
||
);
|
||
if (exists) return;
|
||
|
||
// Create a new link element
|
||
const link = document.createElement('link');
|
||
link.rel = 'stylesheet';
|
||
link.href = href;
|
||
|
||
// Add any additional attributes (e.g., data-* attributes)
|
||
for (const [key, value] of Object.entries(attributes)) {
|
||
link.setAttribute(key, value);
|
||
}
|
||
|
||
// Append the link element to the document head
|
||
document.head.appendChild(link);
|
||
}
|
||
|
||
appendLinkOnce('https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css');
|
||
|
||
if (theme.fontFamily.includes('Press Start 2P')) {
|
||
appendLinkOnce('https://fonts.googleapis.com/css2?family=Press+Start+2P&display=swap');
|
||
}
|
||
|
||
// Load auto-image-styles.css - prioritize extension local resources
|
||
console.group('%c🎨 Loading Auto-Image Styles', 'color: #8b5cf6; font-weight: bold;');
|
||
|
||
let stylesLoaded = false;
|
||
|
||
// First try: Check if extension has loaded local CSS resources
|
||
if (window.AUTOBOT_THEMES && window.AUTOBOT_THEMES['auto-image-styles.css']) {
|
||
console.log('%c🔍 Found auto-image-styles.css in extension local resources!', 'color: #10b981; font-weight: bold;');
|
||
|
||
// Check if it's already injected by the extension
|
||
const existingStyle = document.getElementById('autobot-auto-image-styles');
|
||
if (existingStyle) {
|
||
console.log('%c✅ Styles already injected by extension - using local version', 'color: #10b981; font-weight: bold;');
|
||
console.log(' 📍 Source: Extension local file (already in DOM)');
|
||
console.log(' 🆔 Element ID: autobot-auto-image-styles');
|
||
stylesLoaded = true;
|
||
} else {
|
||
// Inject it ourselves
|
||
const styleElement = document.createElement('style');
|
||
styleElement.id = 'autobot-auto-image-styles-script';
|
||
styleElement.setAttribute('data-wplace-theme', 'true');
|
||
styleElement.textContent = window.AUTOBOT_THEMES['auto-image-styles.css'];
|
||
document.head.appendChild(styleElement);
|
||
|
||
console.log('%c✅ Injected auto-image-styles.css from extension local resources', 'color: #10b981; font-weight: bold;');
|
||
console.log(' 📍 Source: Extension local file');
|
||
console.log(' 📏 Size: ' + window.AUTOBOT_THEMES['auto-image-styles.css'].length + ' characters');
|
||
console.log(' 🆔 Element ID: autobot-auto-image-styles-script');
|
||
stylesLoaded = true;
|
||
}
|
||
} else {
|
||
console.log('%c📝 auto-image-styles.css not found in extension resources', 'color: #f59e0b;');
|
||
console.log(' 🔍 window.AUTOBOT_THEMES:', typeof window.AUTOBOT_THEMES);
|
||
if (window.AUTOBOT_THEMES) {
|
||
console.log(' 📋 Available themes:', Object.keys(window.AUTOBOT_THEMES));
|
||
}
|
||
}
|
||
|
||
// Fallback: Load from CDN if local resources not available
|
||
if (!stylesLoaded) {
|
||
console.log('%c🌐 Falling back to CDN loading...', 'color: #8b5cf6;');
|
||
appendLinkOnce(
|
||
'https://wplace-autobot.github.io/WPlace-AutoBOT/main/auto-image-styles.css',
|
||
{ 'data-wplace-theme': 'true' }
|
||
);
|
||
console.log('%c📚 Loaded auto-image-styles.css from CDN (fallback)', 'color: #f59e0b; font-weight: bold;');
|
||
console.log(' 📍 Source: CDN (wplace-autobot.github.io)');
|
||
console.log(' 🌐 URL: https://wplace-autobot.github.io/WPlace-AutoBOT/main/auto-image-styles.css');
|
||
console.log(' ⚠️ Performance: Network request required');
|
||
}
|
||
|
||
console.groupEnd();
|
||
|
||
// Ensure default theme is loaded from extension resources
|
||
if (window.applyTheme && typeof window.applyTheme === 'function') {
|
||
console.group('%c🎨 Loading Default Theme from Extension', 'color: #8b5cf6; font-weight: bold;');
|
||
|
||
// Determine the current theme file name
|
||
let defaultTheme = 'classic'; // fallback
|
||
if (CONFIG.currentTheme === 'Neon Retro') {
|
||
defaultTheme = 'neon';
|
||
} else if (CONFIG.currentTheme === 'Classic Light') {
|
||
defaultTheme = 'classic-light';
|
||
} else if (CONFIG.currentTheme === 'Acrylic') {
|
||
defaultTheme = 'acrylic';
|
||
}
|
||
|
||
console.log(`%c🎯 Loading theme: ${defaultTheme} (${CONFIG.currentTheme})`, 'color: #8b5cf6;');
|
||
const success = window.applyTheme(defaultTheme);
|
||
|
||
if (success) {
|
||
console.log(`%c✅ Default theme loaded from extension local resources`, 'color: #10b981; font-weight: bold;');
|
||
console.log(` 📍 Source: Extension local file (themes/${defaultTheme}.css)`);
|
||
console.log(` 🚀 Performance: Instant load (no CDN request)`);
|
||
console.log(` 🎨 Theme: ${CONFIG.currentTheme}`);
|
||
} else {
|
||
console.warn(`%c⚠️ Failed to load theme ${defaultTheme} from extension`, 'color: #f59e0b;');
|
||
console.log(` 📝 Note: Theme CSS classes will still work`);
|
||
}
|
||
|
||
console.groupEnd();
|
||
} else {
|
||
console.warn(`%c⚠️ Extension applyTheme() function not available`, 'color: #f59e0b; font-weight: bold;');
|
||
console.log(` 📝 Themes will be limited to CSS classes only`);
|
||
}
|
||
|
||
const container = document.createElement('div');
|
||
container.id = 'wplace-image-bot-container';
|
||
container.innerHTML = `
|
||
<div class="wplace-header">
|
||
<div class="wplace-header-title">
|
||
<i class="fas fa-image"></i>
|
||
<span>${Utils.t('title')}</span>
|
||
</div>
|
||
<div class="wplace-header-controls">
|
||
<button id="settingsBtn" class="wplace-header-btn" title="${Utils.t('settings')}">
|
||
<i class="fas fa-cog"></i>
|
||
</button>
|
||
<button id="statsBtn" class="wplace-header-btn" title="${Utils.t('showStats')}">
|
||
<i class="fas fa-chart-bar"></i>
|
||
</button>
|
||
<button id="compactBtn" class="wplace-header-btn" title="${Utils.t('compactMode')}">
|
||
<i class="fas fa-compress"></i>
|
||
</button>
|
||
<button id="minimizeBtn" class="wplace-header-btn" title="${Utils.t('minimize')}">
|
||
<i class="fas fa-minus"></i>
|
||
</button>
|
||
</div>
|
||
</div>
|
||
<div class="wplace-content">
|
||
<!-- Status Section - Always visible -->
|
||
<div class="wplace-status-section">
|
||
<div id="statusText" class="wplace-status status-default">
|
||
${Utils.t('initMessage')}
|
||
</div>
|
||
<div class="wplace-progress">
|
||
<div id="progressBar" class="wplace-progress-bar" style="width: 0%"></div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Image Section -->
|
||
<div class="wplace-section">
|
||
<div class="wplace-section-title">🖼️ Image Management</div>
|
||
<div class="wplace-controls">
|
||
<div class="wplace-row">
|
||
<button id="uploadBtn" class="wplace-btn wplace-btn-upload" disabled title="${Utils.t(
|
||
'waitingSetupComplete'
|
||
)}">
|
||
<i class="fas fa-upload"></i>
|
||
<span>${Utils.t('uploadImage')}</span>
|
||
</button>
|
||
<button id="resizeBtn" class="wplace-btn wplace-btn-primary" disabled>
|
||
<i class="fas fa-expand"></i>
|
||
<span>${Utils.t('resizeImage')}</span>
|
||
</button>
|
||
</div>
|
||
<div class="wplace-row single">
|
||
<button id="selectPosBtn" class="wplace-btn wplace-btn-select" disabled>
|
||
<i class="fas fa-crosshairs"></i>
|
||
<span>${Utils.t('selectPosition')}</span>
|
||
</button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Control Section -->
|
||
<div class="wplace-section">
|
||
<div class="wplace-section-title">🎮 Painting Control</div>
|
||
<div class="wplace-controls">
|
||
<div class="wplace-row">
|
||
<button id="startBtn" class="wplace-btn wplace-btn-start" disabled>
|
||
<i class="fas fa-play"></i>
|
||
<span>${Utils.t('startPainting')}</span>
|
||
</button>
|
||
<button id="stopBtn" class="wplace-btn wplace-btn-stop" disabled>
|
||
<i class="fas fa-stop"></i>
|
||
<span>${Utils.t('stopPainting')}</span>
|
||
</button>
|
||
</div>
|
||
<div class="wplace-row single">
|
||
<button id="toggleOverlayBtn" class="wplace-btn wplace-btn-overlay" disabled>
|
||
<i class="fas fa-eye"></i>
|
||
<span>${Utils.t('toggleOverlay')}</span>
|
||
</button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Cooldown Section -->
|
||
<div class="wplace-section">
|
||
<div class="wplace-section-title">⏱️ ${Utils.t('cooldownSettings')}</div>
|
||
<div class="wplace-cooldown-control">
|
||
<label id="cooldownLabel">${Utils.t('waitCharges')}:</label>
|
||
<div class="wplace-dual-control-compact">
|
||
<div class="wplace-slider-container-compact">
|
||
<input type="range" id="cooldownSlider" class="wplace-slider" min="1" max="1" value="${state.cooldownChargeThreshold}">
|
||
</div>
|
||
<div class="wplace-input-group-compact">
|
||
<button id="cooldownDecrease" class="wplace-input-btn-compact" type="button">-</button>
|
||
<input type="number" id="cooldownInput" class="wplace-number-input-compact" min="1" max="999" value="${state.cooldownChargeThreshold}">
|
||
<button id="cooldownIncrease" class="wplace-input-btn-compact" type="button">+</button>
|
||
<span id="cooldownValue" class="wplace-input-label-compact">${Utils.t('charges')}</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Data Section -->
|
||
<div class="wplace-section">
|
||
<div class="wplace-section-title">💾 Data Management</div>
|
||
<div class="wplace-controls">
|
||
<div class="wplace-row">
|
||
<button id="saveBtn" class="wplace-btn wplace-btn-primary" disabled>
|
||
<i class="fas fa-save"></i>
|
||
<span>${Utils.t('saveData')}</span>
|
||
</button>
|
||
<button id="loadBtn" class="wplace-btn wplace-btn-primary" disabled title="${Utils.t(
|
||
'waitingTokenGenerator'
|
||
)}">
|
||
<i class="fas fa-folder-open"></i>
|
||
<span>${Utils.t('loadData')}</span>
|
||
</button>
|
||
</div>
|
||
<div class="wplace-row">
|
||
<button id="saveToFileBtn" class="wplace-btn wplace-btn-file" disabled>
|
||
<i class="fas fa-download"></i>
|
||
<span>${Utils.t('saveToFile')}</span>
|
||
</button>
|
||
<button id="loadFromFileBtn" class="wplace-btn wplace-btn-file" disabled title="${Utils.t(
|
||
'waitingTokenGenerator'
|
||
)}">
|
||
<i class="fas fa-upload"></i>
|
||
<span>${Utils.t('loadFromFile')}</span>
|
||
</button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
`;
|
||
|
||
// Stats Window - Separate UI
|
||
const statsContainer = document.createElement('div');
|
||
statsContainer.id = 'wplace-stats-container';
|
||
statsContainer.style.display = 'none';
|
||
statsContainer.innerHTML = `
|
||
<div class="wplace-header">
|
||
<div class="wplace-header-title">
|
||
<i class="fas fa-chart-bar"></i>
|
||
<span>${Utils.t('paintingStats')}</span>
|
||
</div>
|
||
<div class="wplace-header-controls">
|
||
<button id="refreshChargesBtn" class="wplace-header-btn" title="${Utils.t(
|
||
'refreshCharges'
|
||
)}">
|
||
<i class="fas fa-sync"></i>
|
||
</button>
|
||
<button id="closeStatsBtn" class="wplace-header-btn" title="${Utils.t('closeStats')}">
|
||
<i class="fas fa-times"></i>
|
||
</button>
|
||
</div>
|
||
</div>
|
||
<div class="wplace-content">
|
||
<div class="wplace-stats">
|
||
<div id="statsArea">
|
||
<div class="wplace-stat-item">
|
||
<div class="wplace-stat-label"><i class="fas fa-info-circle"></i> ${Utils.t(
|
||
'initMessage'
|
||
)}</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
`;
|
||
|
||
// Modern Settings Container with Theme Support
|
||
// Use the theme variable already declared at the top of createUI function
|
||
const settingsContainer = document.createElement('div');
|
||
settingsContainer.id = 'wplace-settings-container';
|
||
|
||
// Apply theme-based styling
|
||
const themeBackground = theme.primary
|
||
? `linear-gradient(135deg, ${theme.primary} 0%, ${theme.secondary || theme.primary} 100%)`
|
||
: `linear-gradient(135deg, #667eea 0%, #764ba2 100%)`;
|
||
|
||
settingsContainer.className = 'wplace-settings-container-base';
|
||
// Apply theme-specific background
|
||
settingsContainer.style.background = themeBackground;
|
||
settingsContainer.style.cssText += `
|
||
min-width: 420px;
|
||
max-width: 480px;
|
||
z-index: 99999;
|
||
color: ${theme.text || 'white'};
|
||
font-family: ${theme.fontFamily || "'Segoe UI', Tahoma, Geneva, Verdana, sans-serif"};
|
||
box-shadow: ${theme.boxShadow || '0 20px 40px rgba(0,0,0,0.3), 0 0 0 1px rgba(255,255,255,0.1)'
|
||
};
|
||
backdrop-filter: ${theme.backdropFilter || 'blur(10px)'};
|
||
overflow: hidden;
|
||
animation: settings-slide-in 0.4s ease-out;
|
||
${theme.animations?.glow
|
||
? `
|
||
box-shadow: ${theme.boxShadow || '0 20px 40px rgba(0,0,0,0.3)'},
|
||
0 0 30px ${theme.highlight || theme.neon || '#00ffff'};
|
||
`
|
||
: ''
|
||
}
|
||
`;
|
||
|
||
// noinspection CssInvalidFunction
|
||
settingsContainer.innerHTML = `
|
||
<div class="wplace-settings-header">
|
||
<div class="wplace-settings-title-wrapper">
|
||
<h3 class="wplace-settings-title">
|
||
<i class="fas fa-cog wplace-settings-icon"></i>
|
||
${Utils.t('settings')}
|
||
</h3>
|
||
<button id="closeSettingsBtn" class="wplace-settings-close-btn">✕</button>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="wplace-settings-content">
|
||
|
||
<!-- Token Source Selection -->
|
||
<div class="wplace-settings-section">
|
||
<label class="wplace-settings-section-label">
|
||
<i class="fas fa-key wplace-icon-key"></i>
|
||
Token Source
|
||
</label>
|
||
<div class="wplace-settings-section-wrapper">
|
||
<select id="tokenSourceSelect" class="wplace-settings-select">
|
||
<option value="generator" ${state.tokenSource === 'generator' ? 'selected' : ''
|
||
} class="wplace-settings-option">🤖 Automatic Token Generator (Recommended)</option>
|
||
<option value="hybrid" ${state.tokenSource === 'hybrid' ? 'selected' : ''
|
||
} class="wplace-settings-option">🔄 Generator + Auto Fallback</option>
|
||
<option value="manual" ${state.tokenSource === 'manual' ? 'selected' : ''
|
||
} class="wplace-settings-option">🎯 Manual Pixel Placement</option>
|
||
</select>
|
||
<p class="wplace-settings-description">
|
||
Generator mode creates tokens automatically. Hybrid mode falls back to manual when generator fails. Manual mode only uses pixel placement.
|
||
</p>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Automation Section -->
|
||
<div class="wplace-settings-section">
|
||
<label class="wplace-settings-section-label">
|
||
<i class="fas fa-robot wplace-icon-robot"></i>
|
||
${Utils.t('automation')}
|
||
</label>
|
||
<!-- Token generator is always enabled - settings moved to Token Source above -->
|
||
</div>
|
||
|
||
<!-- Overlay Settings Section -->
|
||
<div class="wplace-settings-section">
|
||
<label class="wplace-settings-section-label" style="color: ${theme.text || 'white'};">
|
||
<i class="fas fa-eye wplace-icon-eye" style="color: ${theme.highlight || '#48dbfb'
|
||
};"></i>
|
||
Overlay Settings
|
||
</label>
|
||
<div class="wplace-settings-section-wrapper wplace-overlay-wrapper" style="
|
||
background: ${theme.accent ? `${theme.accent}20` : 'rgba(255,255,255,0.1)'};
|
||
border-radius: ${theme.borderRadius || '12px'};
|
||
padding: 18px;
|
||
border: 1px solid ${theme.accent || 'rgba(255,255,255,0.1)'};
|
||
${theme.animations?.glow
|
||
? `
|
||
box-shadow: 0 0 15px ${theme.accent || 'rgba(255,255,255,0.1)'}33;
|
||
`
|
||
: ''
|
||
}
|
||
">
|
||
<!-- Opacity Slider -->
|
||
<div class="wplace-overlay-opacity-control">
|
||
<div class="wplace-overlay-opacity-header">
|
||
<span class="wplace-overlay-opacity-label" style="color: ${theme.text || 'white'
|
||
};">Overlay Opacity</span>
|
||
<div id="overlayOpacityValue" class="wplace-overlay-opacity-value" style="
|
||
background: ${theme.secondary || 'rgba(0,0,0,0.2)'};
|
||
color: ${theme.text || 'white'};
|
||
padding: 4px 8px;
|
||
border-radius: ${theme.borderRadius === '0' ? '0' : '6px'};
|
||
font-size: 12px;
|
||
border: 1px solid ${theme.accent || 'transparent'};
|
||
">${Math.round(state.overlayOpacity * 100)}%</div>
|
||
</div>
|
||
<input type="range" id="overlayOpacitySlider" min="0.1" max="1" step="0.05" value="${state.overlayOpacity}" class="wplace-overlay-opacity-slider" style="
|
||
background: linear-gradient(to right, ${theme.highlight || '#48dbfb'
|
||
} 0%, ${theme.purple || theme.neon || '#d3a4ff'} 100%);
|
||
border-radius: ${theme.borderRadius === '0' ? '0' : '4px'};
|
||
">
|
||
</div>
|
||
<!-- Blue Marble Toggle -->
|
||
<label for="enableBlueMarbleToggle" class="wplace-settings-toggle">
|
||
<div>
|
||
<span class="wplace-settings-toggle-title" style="color: ${theme.text || 'white'
|
||
};">Blue Marble Effect</span>
|
||
<p class="wplace-settings-toggle-description" style="color: ${theme.text ? `${theme.text}BB` : 'rgba(255,255,255,0.7)'
|
||
};">Renders a dithered "shredded" overlay.</p>
|
||
</div>
|
||
<input type="checkbox" id="enableBlueMarbleToggle" ${state.blueMarbleEnabled ? 'checked' : ''
|
||
} class="wplace-settings-checkbox" style="
|
||
accent-color: ${theme.highlight || '#48dbfb'};
|
||
"/>
|
||
</label>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Paint Options Section -->
|
||
<div class="wplace-settings-section">
|
||
<label class="wplace-settings-section-label">
|
||
<i class="fas fa-paint-brush wplace-icon-paint"></i>
|
||
${Utils.t('paintOptions')}
|
||
</label>
|
||
<!-- Pixel Filter Toggles -->
|
||
<div id="pixelFilterControls" class="wplace-settings-section-wrapper wplace-pixel-filter-controls">
|
||
<!-- Paint White Pixels -->
|
||
<label class="wplace-settings-toggle">
|
||
<div>
|
||
<span class="wplace-settings-toggle-title" style="color: ${theme.text || 'white'};">
|
||
${Utils.t('paintWhitePixels')}
|
||
</span>
|
||
<p class="wplace-settings-toggle-description" style="color: ${theme.text ? `${theme.text}BB` : 'rgba(255,255,255,0.7)'
|
||
};">
|
||
${Utils.t('paintWhitePixelsDescription')}
|
||
</p>
|
||
</div>
|
||
<input type="checkbox" id="settingsPaintWhiteToggle" ${state.paintWhitePixels ? 'checked' : ''}
|
||
class="wplace-settings-checkbox"
|
||
style="accent-color: ${theme.highlight || '#48dbfb'};"/>
|
||
</label>
|
||
|
||
<!-- Paint Transparent Pixels -->
|
||
<label class="wplace-settings-toggle">
|
||
<div>
|
||
<span class="wplace-settings-toggle-title" style="color: ${theme.text || 'white'};">
|
||
${Utils.t('paintTransparentPixels')}
|
||
</span>
|
||
<p class="wplace-settings-toggle-description" style="color: ${theme.text ? `${theme.text}BB` : 'rgba(255,255,255,0.7)'
|
||
};">
|
||
${Utils.t('paintTransparentPixelsDescription')}
|
||
</p>
|
||
</div>
|
||
<input type="checkbox" id="settingsPaintTransparentToggle" ${state.paintTransparentPixels ? 'checked' : ''}
|
||
class="wplace-settings-checkbox"
|
||
style="accent-color: ${theme.highlight || '#48dbfb'};"/>
|
||
</label>
|
||
<label class="wplace-settings-toggle">
|
||
<div>
|
||
<span class="wplace-settings-toggle-title" style="color: ${theme.text || 'white'
|
||
};">${Utils.t('paintUnavailablePixels')}</span>
|
||
<p class="wplace-settings-toggle-description" style="color: ${theme.text ? `${theme.text}BB` : 'rgba(255,255,255,0.7)'
|
||
};">${Utils.t('paintUnavailablePixelsDescription')}</p>
|
||
</div>
|
||
<input type="checkbox" id="paintUnavailablePixelsToggle" ${state.paintUnavailablePixels ? 'checked' : ''
|
||
} class="wplace-settings-checkbox" style="
|
||
accent-color: ${theme.highlight || '#48dbfb'};
|
||
"/>
|
||
</label>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Speed Control Section -->
|
||
<div class="wplace-settings-section">
|
||
<label class="wplace-settings-section-label">
|
||
<i class="fas fa-tachometer-alt wplace-icon-speed"></i>
|
||
${Utils.t('paintingSpeed')}
|
||
</label>
|
||
|
||
<!-- Batch Mode Selection -->
|
||
<div class="wplace-mode-selection">
|
||
<label class="wplace-mode-label">
|
||
<i class="fas fa-dice wplace-icon-dice"></i>
|
||
Batch Mode
|
||
</label>
|
||
<select id="batchModeSelect" class="wplace-settings-select">
|
||
<option value="normal" class="wplace-settings-option">📦 Normal (Fixed Size)</option>
|
||
<option value="random" class="wplace-settings-option">🎲 Random (Range)</option>
|
||
</select>
|
||
</div>
|
||
|
||
<!-- Normal Mode: Fixed Size Controls -->
|
||
<div id="normalBatchControls" class="wplace-batch-controls wplace-normal-batch-controls">
|
||
<div class="wplace-batch-size-header">
|
||
<span class="wplace-batch-size-label">${Utils.t('batchSize')}</span>
|
||
</div>
|
||
<div class="wplace-dual-control-compact">
|
||
<div class="wplace-speed-slider-container-compact">
|
||
<input type="range" id="speedSlider" min="${CONFIG.PAINTING_SPEED.MIN}" max="${CONFIG.PAINTING_SPEED.MAX}" value="${CONFIG.PAINTING_SPEED.DEFAULT}" class="wplace-speed-slider">
|
||
</div>
|
||
<div class="wplace-speed-input-container-compact">
|
||
<div class="wplace-input-group-compact">
|
||
<button id="speedDecrease" class="wplace-input-btn-compact" type="button">-</button>
|
||
<input type="number" id="speedInput" class="wplace-number-input-compact" min="${CONFIG.PAINTING_SPEED.MIN}" max="${CONFIG.PAINTING_SPEED.MAX}" value="${CONFIG.PAINTING_SPEED.DEFAULT}">
|
||
<button id="speedIncrease" class="wplace-input-btn-compact" type="button">+</button>
|
||
<span id="speedValue" class="wplace-input-label-compact">pixels</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="wplace-speed-labels">
|
||
<span class="wplace-speed-min"><i class="fas fa-turtle"></i> ${CONFIG.PAINTING_SPEED.MIN}</span>
|
||
<span class="wplace-speed-max"><i class="fas fa-rabbit"></i> ${CONFIG.PAINTING_SPEED.MAX}</span>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Random Mode: Range Controls -->
|
||
<div id="randomBatchControls" class="wplace-batch-controls wplace-random-batch-controls">
|
||
<div class="wplace-random-batch-grid">
|
||
<div>
|
||
<label class="wplace-random-batch-label">
|
||
<i class="fas fa-arrow-down wplace-icon-min"></i>
|
||
Minimum Batch Size
|
||
</label>
|
||
<input type="number" id="randomBatchMin" min="1" max="1000" value="${CONFIG.RANDOM_BATCH_RANGE.MIN}" class="wplace-settings-number-input">
|
||
</div>
|
||
<div>
|
||
<label class="wplace-random-batch-label">
|
||
<i class="fas fa-arrow-up wplace-icon-max"></i>
|
||
Maximum Batch Size
|
||
</label>
|
||
<input type="number" id="randomBatchMax" min="1" max="1000" value="${CONFIG.RANDOM_BATCH_RANGE.MAX}" class="wplace-settings-number-input">
|
||
</div>
|
||
</div>
|
||
<p class="wplace-random-batch-description">
|
||
🎲 Random batch size between min and max values
|
||
</p>
|
||
</div>
|
||
|
||
<!-- Speed Control Toggle -->
|
||
<label class="wplace-speed-control-toggle">
|
||
<input type="checkbox" id="enableSpeedToggle" ${CONFIG.PAINTING_SPEED_ENABLED ? 'checked' : ''
|
||
} class="wplace-speed-checkbox"/>
|
||
<span>${Utils.t('enablePaintingSpeedLimit')}</span>
|
||
</label>
|
||
</div>
|
||
|
||
<!-- Coordinate Generation Section -->
|
||
<div class="wplace-settings-section">
|
||
<label class="wplace-settings-section-label">
|
||
<i class="fas fa-route wplace-icon-route"></i>
|
||
Coordinate Generation
|
||
</label>
|
||
|
||
<!-- Mode Selection -->
|
||
<div class="wplace-mode-selection">
|
||
<label class="wplace-mode-label">
|
||
<i class="fas fa-th wplace-icon-table"></i>
|
||
Generation Mode
|
||
</label>
|
||
<select id="coordinateModeSelect" class="wplace-settings-select">
|
||
<option value="rows" class="wplace-settings-option">📏 Rows (Horizontal Lines)</option>
|
||
<option value="columns" class="wplace-settings-option">📐 Columns (Vertical Lines)</option>
|
||
<option value="circle-out" class="wplace-settings-option">⭕ Circle Out (Center → Edges)</option>
|
||
<option value="circle-in" class="wplace-settings-option">⭕ Circle In (Edges → Center)</option>
|
||
<option value="blocks" class="wplace-settings-option">🟫 Blocks (Ordered)</option>
|
||
<option value="shuffle-blocks" class="wplace-settings-option">🎲 Shuffle Blocks (Random)</option>
|
||
</select>
|
||
</div>
|
||
|
||
<!-- Direction Selection (only for rows/columns) -->
|
||
<div id="directionControls" class="wplace-mode-selection">
|
||
<label class="wplace-mode-label">
|
||
<i class="fas fa-compass wplace-icon-compass"></i>
|
||
Starting Direction
|
||
</label>
|
||
<select id="coordinateDirectionSelect" class="wplace-settings-select">
|
||
<option value="top-left" class="wplace-settings-option">↖️ Top-Left</option>
|
||
<option value="top-right" class="wplace-settings-option">↗️ Top-Right</option>
|
||
<option value="bottom-left" class="wplace-settings-option">↙️ Bottom-Left</option>
|
||
<option value="bottom-right" class="wplace-settings-option">↘️ Bottom-Right</option>
|
||
</select>
|
||
</div>
|
||
|
||
<!-- Snake Pattern Toggle (only for rows/columns) -->
|
||
<div id="snakeControls" class="wplace-snake-pattern-controls wplace-settings-section-wrapper">
|
||
<label class="wplace-settings-toggle">
|
||
<div>
|
||
<span class="wplace-settings-toggle-title" style="color: ${theme.text || 'white'
|
||
};">Snake Pattern</span>
|
||
<p class="wplace-settings-toggle-description" style="color: ${theme.text ? `${theme.text}BB` : 'rgba(255,255,255,0.7)'
|
||
};">Alternate direction for each row/column (zigzag pattern)</p>
|
||
</div>
|
||
<input type="checkbox" id="coordinateSnakeToggle" ${state.coordinateSnake ? 'checked' : ''
|
||
} class="wplace-settings-checkbox" style="
|
||
accent-color: ${theme.highlight || '#48dbfb'};
|
||
"/>
|
||
</label>
|
||
</div>
|
||
|
||
<!-- Block Size Controls (only for blocks/shuffle-blocks) -->
|
||
<div id="blockControls" class="wplace-block-size-controls wplace-settings-section-wrapper wplace-shuffle-block-size-controls">
|
||
<div class="wplace-block-size-grid">
|
||
<div>
|
||
<label class="wplace-block-size-label">
|
||
<i class="fas fa-arrows-alt-h wplace-icon-width"></i>
|
||
Block Width
|
||
</label>
|
||
<input type="number" id="blockWidthInput" min="1" max="50" value="6" class="wplace-settings-number-input">
|
||
</div>
|
||
<div>
|
||
<label style="display: block; color: rgba(255,255,255,0.8); font-size: 12px; margin-bottom: 8px;">
|
||
<i class="fas fa-arrows-alt-v wplace-icon-height"></i>
|
||
Block Height
|
||
</label>
|
||
<input type="number" id="blockHeightInput" min="1" max="50" value="2" class="wplace-settings-number-input">
|
||
</div>
|
||
</div>
|
||
<p class="wplace-block-size-description">
|
||
🧱 Block dimensions for block-based generation modes
|
||
</p>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Notifications Section -->
|
||
<div class="wplace-settings-section">
|
||
<label class="wplace-settings-section-label">
|
||
<i class="fas fa-bell wplace-icon-bell"></i>
|
||
Desktop Notifications
|
||
</label>
|
||
<div class="wplace-settings-section-wrapper wplace-notifications-wrapper">
|
||
<label class="wplace-notification-toggle">
|
||
<span>${Utils.t('enableNotifications')}</span>
|
||
<input type="checkbox" id="notifEnabledToggle" ${state.notificationsEnabled ? 'checked' : ''
|
||
} class="wplace-notification-checkbox" />
|
||
</label>
|
||
<label class="wplace-notification-toggle">
|
||
<span>${Utils.t('notifyOnChargesThreshold')}</span>
|
||
<input type="checkbox" id="notifOnChargesToggle" ${state.notifyOnChargesReached ? 'checked' : ''
|
||
} class="wplace-notification-checkbox" />
|
||
</label>
|
||
<label class="wplace-notification-toggle">
|
||
<span>${Utils.t('onlyWhenNotFocused')}</span>
|
||
<input type="checkbox" id="notifOnlyUnfocusedToggle" ${state.notifyOnlyWhenUnfocused ? 'checked' : ''
|
||
} class="wplace-notification-checkbox" />
|
||
</label>
|
||
<div class="wplace-notification-interval">
|
||
<span>${Utils.t('repeatEvery')}</span>
|
||
<input type="number" id="notifIntervalInput" min="1" max="60" value="${state.notificationIntervalMinutes}" class="wplace-notification-interval-input" />
|
||
<span>${Utils.t('minutesPl')}</span>
|
||
</div>
|
||
<div class="wplace-notification-buttons">
|
||
<button id="notifRequestPermBtn" class="wplace-btn wplace-btn-secondary wplace-notification-perm-btn"><i class="fas fa-unlock"></i><span>${Utils.t(
|
||
'grantPermission'
|
||
)}</span></button>
|
||
<button id="notifTestBtn" class="wplace-btn wplace-notification-test-btn"><i class="fas fa-bell"></i><span>${Utils.t(
|
||
'test'
|
||
)}</span></button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Theme Selection Section -->
|
||
<div class="wplace-settings-section">
|
||
<label class="wplace-settings-section-label">
|
||
<i class="fas fa-palette wplace-icon-palette"></i>
|
||
${Utils.t('themeSettings')}
|
||
</label>
|
||
<div class="wplace-settings-section-wrapper">
|
||
<select id="themeSelect" class="wplace-settings-select">
|
||
${Object.keys(CONFIG.THEMES)
|
||
.map(
|
||
(themeName) =>
|
||
`<option value="${themeName}" ${CONFIG.currentTheme === themeName ? 'selected' : ''
|
||
} class="wplace-settings-option">${themeName}</option>`
|
||
)
|
||
.join('')}
|
||
</select>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Language Selection Section -->
|
||
<div class="wplace-settings-section">
|
||
<label class="wplace-settings-section-label">
|
||
<i class="fas fa-globe wplace-icon-globe"></i>
|
||
${Utils.t('language')}
|
||
</label>
|
||
<div class="wplace-settings-section-wrapper">
|
||
<select id="languageSelect" class="wplace-settings-select">
|
||
<option value="vi" ${state.language === 'vi' ? 'selected' : ''} class="wplace-settings-option">🇻🇳 Tiếng Việt</option>
|
||
<option value="id" ${state.language === 'id' ? 'selected' : ''} class="wplace-settings-option">🇮🇩 Bahasa Indonesia</option>
|
||
<option value="ru" ${state.language === 'ru' ? 'selected' : ''} class="wplace-settings-option">🇷🇺 Русский</option>
|
||
<option value="uk" ${state.language === 'uk' ? 'selected' : ''} class="wplace-settings-option">🇺🇦 Українська</option>
|
||
<option value="en" ${state.language === 'en' ? 'selected' : ''} class="wplace-settings-option">🇺🇸 English</option>
|
||
<option value="pt" ${state.language === 'pt' ? 'selected' : ''} class="wplace-settings-option">🇧🇷 Português</option>
|
||
<option value="fr" ${state.language === 'fr' ? 'selected' : ''} class="wplace-settings-option">🇫🇷 Français</option>
|
||
<option value="tr" ${state.language === 'tr' ? 'selected' : ''} class="wplace-settings-option">🇹🇷 Türkçe</option>
|
||
<option value="zh-CN" ${state.language === 'zh-CN' ? 'selected' : ''} class="wplace-settings-option">🇨🇳 简体中文</option>
|
||
<option value="zh-TW" ${state.language === 'zh-TW' ? 'selected' : ''} class="wplace-settings-option">🇹🇼 繁體中文</option>
|
||
<option value="ja" ${state.language === 'ja' ? 'selected' : ''} class="wplace-settings-option">🇯🇵 日本語</option>
|
||
<option value="ko" ${state.language === 'ko' ? 'selected' : ''} class="wplace-settings-option">🇰🇷 한국어</option>
|
||
</select>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="wplace-settings-footer">
|
||
<button id="applySettingsBtn" class="wplace-settings-apply-btn">
|
||
<i class="fas fa-check"></i> ${Utils.t('applySettings')}
|
||
</button>
|
||
</div>
|
||
|
||
<style>
|
||
@keyframes spin {
|
||
from { transform: rotate(0deg); }
|
||
to { transform: rotate(360deg); }
|
||
}
|
||
|
||
@keyframes settings-slide-in {
|
||
from {
|
||
opacity: 0;
|
||
transform: translate(-50%, -50%) scale(0.9);
|
||
}
|
||
to {
|
||
opacity: 1;
|
||
transform: translate(-50%, -50%) scale(1);
|
||
}
|
||
}
|
||
|
||
@keyframes settings-fade-out {
|
||
from {
|
||
opacity: 1;
|
||
transform: translate(-50%, -50%) scale(1);
|
||
}
|
||
to {
|
||
opacity: 0;
|
||
transform: translate(-50%, -50%) scale(0.9);
|
||
}
|
||
}
|
||
|
||
#speedSlider::-webkit-slider-thumb, #cooldownSlider::-webkit-slider-thumb, #overlayOpacitySlider::-webkit-slider-thumb {
|
||
-webkit-appearance: none;
|
||
width: 18px;
|
||
height: 18px;
|
||
border-radius: 50%;
|
||
background: white;
|
||
box-shadow: 0 3px 6px rgba(0,0,0,0.3), 0 0 0 2px #4facfe;
|
||
cursor: pointer;
|
||
transition: all 0.2s ease;
|
||
}
|
||
|
||
#speedSlider::-webkit-slider-thumb:hover, #cooldownSlider::-webkit-slider-thumb:hover, #overlayOpacitySlider::-webkit-slider-thumb:hover {
|
||
transform: scale(1.2);
|
||
box-shadow: 0 4px 8px rgba(0,0,0,0.4), 0 0 0 3px #4facfe;
|
||
}
|
||
|
||
#speedSlider::-moz-range-thumb, #cooldownSlider::-moz-range-thumb, #overlayOpacitySlider::-moz-range-thumb {
|
||
width: 18px;
|
||
height: 18px;
|
||
border-radius: 50%;
|
||
background: white;
|
||
box-shadow: 0 3px 6px rgba(0,0,0,0.3), 0 0 0 2px #4facfe;
|
||
cursor: pointer;
|
||
border: none;
|
||
transition: all 0.2s ease;
|
||
}
|
||
|
||
#themeSelect:hover, #languageSelect:hover {
|
||
border-color: rgba(255,255,255,0.4);
|
||
background: rgba(255,255,255,0.2);
|
||
transform: translateY(-1px);
|
||
box-shadow: 0 5px 15px rgba(0,0,0,0.15);
|
||
}
|
||
|
||
#themeSelect:focus, #languageSelect:focus {
|
||
border-color: #4facfe;
|
||
box-shadow: 0 0 0 3px rgba(79, 172, 254, 0.3);
|
||
}
|
||
|
||
#themeSelect option, #languageSelect option {
|
||
background: #2d3748;
|
||
color: white;
|
||
padding: 10px;
|
||
border-radius: 6px;
|
||
}
|
||
|
||
#themeSelect option:hover, #languageSelect option:hover {
|
||
background: #4a5568;
|
||
}
|
||
|
||
.wplace-dragging {
|
||
opacity: 0.9;
|
||
box-shadow: 0 30px 60px rgba(0,0,0,0.4), 0 0 0 1px rgba(255,255,255,0.2);
|
||
transition: none;
|
||
}
|
||
|
||
.wplace-settings-header:hover {
|
||
background: rgba(255,255,255,0.15) !important;
|
||
}
|
||
|
||
.wplace-settings-header:active {
|
||
background: rgba(255,255,255,0.2) !important;
|
||
}
|
||
|
||
/* Custom Scrollbar for Content Area */
|
||
.wplace-content::-webkit-scrollbar {
|
||
width: 6px;
|
||
}
|
||
|
||
.wplace-content::-webkit-scrollbar-track {
|
||
background: rgba(255,255,255,0.1);
|
||
border-radius: 3px;
|
||
}
|
||
|
||
.wplace-content::-webkit-scrollbar-thumb {
|
||
background: rgba(255,255,255,0.3);
|
||
border-radius: 3px;
|
||
}
|
||
|
||
.wplace-content::-webkit-scrollbar-thumb:hover {
|
||
background: rgba(255,255,255,0.5);
|
||
}
|
||
</style>
|
||
`;
|
||
|
||
const resizeContainer = document.createElement('div');
|
||
resizeContainer.className = 'resize-container';
|
||
resizeContainer.innerHTML = `
|
||
<h3 class="resize-dialog-title" style="color: ${theme.text}">${Utils.t('resizeImage')}</h3>
|
||
<div class="resize-controls">
|
||
<label class="resize-control-label">
|
||
Width: <span id="widthValue">0</span>px
|
||
<input type="range" id="widthSlider" class="resize-slider" min="10" max="500" value="100">
|
||
</label>
|
||
<label class="resize-control-label">
|
||
Height: <span id="heightValue">0</span>px
|
||
<input type="range" id="heightSlider" class="resize-slider" min="10" max="500" value="100">
|
||
</label>
|
||
<label class="resize-checkbox-label">
|
||
<input type="checkbox" id="keepAspect" checked>
|
||
${Utils.t('keepAspectRatio')}
|
||
</label>
|
||
<label class="resize-checkbox-label">
|
||
<input type="checkbox" id="paintWhiteToggle" checked>
|
||
${Utils.t('paintWhitePixels')}
|
||
</label>
|
||
<label class="resize-checkbox-label">
|
||
<input type="checkbox" id="paintTransparentToggle" checked>
|
||
${Utils.t('paintTransparentPixels')}
|
||
</label>
|
||
<div class="resize-zoom-controls">
|
||
<button id="zoomOutBtn" class="wplace-btn resize-zoom-btn" title="${Utils.t(
|
||
'zoomOut'
|
||
)}"><i class="fas fa-search-minus"></i></button>
|
||
<input type="range" id="zoomSlider" class="resize-slider resize-zoom-slider" min="0.1" max="20" value="1" step="0.05">
|
||
<button id="zoomInBtn" class="wplace-btn resize-zoom-btn" title="${Utils.t(
|
||
'zoomIn'
|
||
)}"><i class="fas fa-search-plus"></i></button>
|
||
<button id="zoomFitBtn" class="wplace-btn resize-zoom-btn" title="${Utils.t(
|
||
'fitToView'
|
||
)}">${Utils.t('fit')}</button>
|
||
<button id="zoomActualBtn" class="wplace-btn resize-zoom-btn" title="${Utils.t(
|
||
'actualSize'
|
||
)}">${Utils.t('hundred')}</button>
|
||
<button id="panModeBtn" class="wplace-btn resize-zoom-btn" title="${Utils.t('panMode')}">
|
||
<i class="fas fa-hand-paper"></i>
|
||
</button>
|
||
<span id="zoomValue" class="resize-zoom-value">100%</span>
|
||
<div id="cameraHelp" class="resize-camera-help">
|
||
Drag to pan • Pinch to zoom • Double‑tap to zoom
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="resize-preview-wrapper">
|
||
<div id="resizePanStage" class="resize-pan-stage">
|
||
<div id="resizeCanvasStack" class="resize-canvas-stack resize-canvas-positioned">
|
||
<canvas id="resizeCanvas" class="resize-base-canvas"></canvas>
|
||
<canvas id="maskCanvas" class="resize-mask-canvas"></canvas>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="resize-tools">
|
||
<div class="resize-tools-container">
|
||
<div class="resize-brush-controls">
|
||
<div class="resize-brush-control">
|
||
<label class="resize-tool-label">Brush</label>
|
||
<div class="resize-tool-input-group">
|
||
<input id="maskBrushSize" type="range" min="1" max="192" step="1" value="1" class="resize-tool-slider">
|
||
<span id="maskBrushSizeValue" class="resize-tool-value">1</span>
|
||
</div>
|
||
</div>
|
||
<div class="resize-brush-control">
|
||
<label class="resize-tool-label">Row/col size</label>
|
||
<div class="resize-tool-input-group">
|
||
<input id="rowColSize" type="range" min="1" max="192" step="1" value="1" class="resize-tool-slider">
|
||
<span id="rowColSizeValue" class="resize-tool-value">1</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="resize-mode-controls">
|
||
<label class="resize-tool-label">Mode</label>
|
||
<div class="mask-mode-group resize-mode-group">
|
||
<button id="maskModeIgnore" class="wplace-btn resize-mode-btn">Ignore</button>
|
||
<button id="maskModeUnignore" class="wplace-btn resize-mode-btn">Unignore</button>
|
||
<button id="maskModeToggle" class="wplace-btn wplace-btn-primary resize-mode-btn">Toggle</button>
|
||
</div>
|
||
</div>
|
||
<button id="clearIgnoredBtn" class="wplace-btn resize-clear-btn" title="Clear all ignored pixels">Clear</button>
|
||
<button id="invertMaskBtn" class="wplace-btn resize-invert-btn" title="Invert mask">Invert</button>
|
||
<span class="resize-shortcut-help">Shift = Row • Alt = Column</span>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="wplace-section resize-color-palette-section" id="color-palette-section">
|
||
<div class="wplace-section-title">
|
||
<i class="fas fa-palette"></i> Color Palette
|
||
</div>
|
||
<div class="wplace-controls">
|
||
<div class="wplace-row single">
|
||
<label class="resize-color-toggle-label">
|
||
<input type="checkbox" id="showAllColorsToggle" class="resize-color-checkbox">
|
||
<span>${Utils.t('showAllColorsIncluding')}</span>
|
||
</label>
|
||
</div>
|
||
<div class="wplace-row" style="display: flex;">
|
||
<button id="selectAllBtn" class="wplace-btn" style="flex: 1;">Select All</button>
|
||
<button id="unselectAllBtn" class="wplace-btn" style="flex: 1;">Unselect All</button>
|
||
<button id="unselectPaidBtn" class="wplace-btn">Unselect Paid</button>
|
||
</div>
|
||
<div id="colors-container" class="wplace-color-grid"></div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="wplace-section resize-advanced-color-section" id="advanced-color-section">
|
||
<div class="wplace-section-title">
|
||
<i class="fas fa-flask"></i> Advanced Color Matching
|
||
</div>
|
||
<div class="resize-advanced-controls">
|
||
<label class="resize-advanced-label">
|
||
<span class="resize-advanced-label-text">Algorithm</span>
|
||
<select id="colorAlgorithmSelect" class="resize-advanced-select">
|
||
<option value="lab" ${state.colorMatchingAlgorithm === 'lab' ? 'selected' : ''
|
||
}>Perceptual (Lab)</option>
|
||
<option value="legacy" ${state.colorMatchingAlgorithm === 'legacy' ? 'selected' : ''
|
||
}>Legacy (RGB)</option>
|
||
</select>
|
||
</label>
|
||
<label class="resize-advanced-toggle">
|
||
<div class="resize-advanced-toggle-content">
|
||
<span class="resize-advanced-label-text">Chroma Penalty</span>
|
||
<div class="resize-advanced-description">Preserve vivid colors (Lab only)</div>
|
||
</div>
|
||
<input type="checkbox" id="enableChromaPenaltyToggle" ${state.enableChromaPenalty ? 'checked' : ''
|
||
} class="resize-advanced-checkbox" />
|
||
</label>
|
||
<div class="resize-chroma-weight-control">
|
||
<div class="resize-chroma-weight-header">
|
||
<span>${Utils.t('chromaWeight')}</span>
|
||
<span id="chromaWeightValue" class="resize-chroma-weight-value">${state.chromaPenaltyWeight}</span>
|
||
</div>
|
||
<input type="range" id="chromaPenaltyWeightSlider" min="0" max="0.5" step="0.01" value="${state.chromaPenaltyWeight}" class="resize-chroma-weight-slider" />
|
||
</div>
|
||
<label class="resize-advanced-toggle">
|
||
<div class="resize-advanced-toggle-content">
|
||
<span class="resize-advanced-label-text">Enable Dithering</span>
|
||
<div class="resize-advanced-description">Floyd–Steinberg error diffusion in preview and applied output</div>
|
||
</div>
|
||
<input type="checkbox" id="enableDitheringToggle" ${state.ditheringEnabled ? 'checked' : ''
|
||
} class="resize-advanced-checkbox" />
|
||
</label>
|
||
<div class="resize-threshold-controls">
|
||
<label class="resize-threshold-label">
|
||
<span class="resize-advanced-label-text">Transparency</span>
|
||
<input type="number" id="transparencyThresholdInput" min="0" max="255" value="${state.customTransparencyThreshold}" class="resize-threshold-input" />
|
||
</label>
|
||
<label class="resize-threshold-label">
|
||
<span class="resize-advanced-label-text">White Thresh</span>
|
||
<input type="number" id="whiteThresholdInput" min="200" max="255" value="${state.customWhiteThreshold}" class="resize-threshold-input" />
|
||
</label>
|
||
</div>
|
||
<button id="resetAdvancedColorBtn" class="wplace-btn resize-reset-advanced-btn">Reset Advanced</button>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="resize-buttons">
|
||
<button id="downloadPreviewBtn" class="wplace-btn wplace-btn-primary">
|
||
<i class="fas fa-download"></i>
|
||
<span>${Utils.t('downloadPreview')}</span>
|
||
</button>
|
||
<button id="confirmResize" class="wplace-btn wplace-btn-start">
|
||
<i class="fas fa-check"></i>
|
||
<span>${Utils.t('apply')}</span>
|
||
</button>
|
||
<button id="cancelResize" class="wplace-btn wplace-btn-stop">
|
||
<i class="fas fa-times"></i>
|
||
<span>${Utils.t('cancel')}</span>
|
||
</button>
|
||
</div>
|
||
`;
|
||
|
||
const resizeOverlay = document.createElement('div');
|
||
resizeOverlay.className = 'resize-overlay';
|
||
|
||
document.body.appendChild(container);
|
||
document.body.appendChild(resizeOverlay);
|
||
document.body.appendChild(resizeContainer);
|
||
document.body.appendChild(statsContainer);
|
||
document.body.appendChild(settingsContainer);
|
||
|
||
// Show the main container after all elements are appended
|
||
container.style.display = 'block';
|
||
|
||
const uploadBtn = container.querySelector('#uploadBtn');
|
||
const resizeBtn = container.querySelector('#resizeBtn');
|
||
const selectPosBtn = container.querySelector('#selectPosBtn');
|
||
const startBtn = container.querySelector('#startBtn');
|
||
const stopBtn = container.querySelector('#stopBtn');
|
||
const saveBtn = container.querySelector('#saveBtn');
|
||
const loadBtn = container.querySelector('#loadBtn');
|
||
const saveToFileBtn = container.querySelector('#saveToFileBtn');
|
||
const loadFromFileBtn = container.querySelector('#loadFromFileBtn');
|
||
|
||
container.querySelectorAll('.wplace-section-title').forEach((title) => {
|
||
// Add a right-side arrow if it doesn't exist
|
||
if (!title.querySelector('i.arrow')) {
|
||
const arrow = document.createElement('i');
|
||
arrow.className = 'fas fa-chevron-down arrow'; // FontAwesome down arrow
|
||
title.appendChild(arrow);
|
||
}
|
||
|
||
// Click event to toggle collapse/expand of the section
|
||
title.addEventListener('click', () => {
|
||
const section = title.parentElement;
|
||
section.classList.toggle('collapsed');
|
||
});
|
||
});
|
||
|
||
// Disable load/upload buttons until initial setup is complete (startup only)
|
||
if (loadBtn) {
|
||
loadBtn.disabled = !state.initialSetupComplete;
|
||
loadBtn.title = state.initialSetupComplete
|
||
? ''
|
||
: '🔄 Waiting for initial setup to complete...';
|
||
}
|
||
if (loadFromFileBtn) {
|
||
loadFromFileBtn.disabled = !state.initialSetupComplete;
|
||
loadFromFileBtn.title = state.initialSetupComplete
|
||
? ''
|
||
: '🔄 Waiting for initial setup to complete...';
|
||
}
|
||
if (uploadBtn) {
|
||
uploadBtn.disabled = !state.initialSetupComplete;
|
||
uploadBtn.title = state.initialSetupComplete
|
||
? ''
|
||
: '🔄 Waiting for initial setup to complete...';
|
||
}
|
||
|
||
const minimizeBtn = container.querySelector('#minimizeBtn');
|
||
const compactBtn = container.querySelector('#compactBtn');
|
||
const statsBtn = container.querySelector('#statsBtn');
|
||
const toggleOverlayBtn = container.querySelector('#toggleOverlayBtn');
|
||
const statusText = container.querySelector('#statusText');
|
||
const progressBar = container.querySelector('#progressBar');
|
||
const statsArea = statsContainer.querySelector('#statsArea');
|
||
const content = container.querySelector('.wplace-content');
|
||
const closeStatsBtn = statsContainer.querySelector('#closeStatsBtn');
|
||
const refreshChargesBtn = statsContainer.querySelector('#refreshChargesBtn');
|
||
const cooldownSlider = container.querySelector('#cooldownSlider');
|
||
const cooldownInput = container.querySelector('#cooldownInput');
|
||
const cooldownDecrease = container.querySelector('#cooldownDecrease');
|
||
const cooldownIncrease = container.querySelector('#cooldownIncrease');
|
||
const cooldownValue = container.querySelector('#cooldownValue');
|
||
|
||
if (!uploadBtn || !selectPosBtn || !startBtn || !stopBtn) {
|
||
console.error('Some UI elements not found:', {
|
||
uploadBtn: !!uploadBtn,
|
||
selectPosBtn: !!selectPosBtn,
|
||
startBtn: !!startBtn,
|
||
stopBtn: !!stopBtn,
|
||
});
|
||
}
|
||
|
||
if (!statsContainer || !statsArea || !closeStatsBtn) {
|
||
// Note: base CSS now aligns with this layout: main panel at left:20px (width 280), stats at left:330px.
|
||
}
|
||
|
||
const header = container.querySelector('.wplace-header');
|
||
|
||
makeDraggable(container);
|
||
|
||
function makeDraggable(element) {
|
||
let pos1 = 0,
|
||
pos2 = 0,
|
||
pos3 = 0,
|
||
pos4 = 0;
|
||
let isDragging = false;
|
||
const header =
|
||
element.querySelector('.wplace-header') || element.querySelector('.wplace-settings-header');
|
||
|
||
if (!header) {
|
||
console.warn('No draggable header found for element:', element);
|
||
return;
|
||
}
|
||
|
||
header.onmousedown = dragMouseDown;
|
||
|
||
function dragMouseDown(e) {
|
||
if (e.target.closest('.wplace-header-btn') || e.target.closest('button')) return;
|
||
|
||
e.preventDefault();
|
||
isDragging = true;
|
||
|
||
const rect = element.getBoundingClientRect();
|
||
|
||
element.style.transform = 'none';
|
||
element.style.top = rect.top + 'px';
|
||
element.style.left = rect.left + 'px';
|
||
|
||
pos3 = e.clientX;
|
||
pos4 = e.clientY;
|
||
element.classList.add('wplace-dragging');
|
||
document.onmouseup = closeDragElement;
|
||
document.onmousemove = elementDrag;
|
||
|
||
document.body.style.userSelect = 'none';
|
||
}
|
||
|
||
function elementDrag(e) {
|
||
if (!isDragging) return;
|
||
|
||
e.preventDefault();
|
||
pos1 = pos3 - e.clientX;
|
||
pos2 = pos4 - e.clientY;
|
||
pos3 = e.clientX;
|
||
pos4 = e.clientY;
|
||
|
||
let newTop = element.offsetTop - pos2;
|
||
let newLeft = element.offsetLeft - pos1;
|
||
|
||
const rect = element.getBoundingClientRect();
|
||
const maxTop = window.innerHeight - rect.height;
|
||
const maxLeft = window.innerWidth - rect.width;
|
||
|
||
newTop = Math.max(0, Math.min(newTop, maxTop));
|
||
newLeft = Math.max(0, Math.min(newLeft, maxLeft));
|
||
|
||
element.style.top = newTop + 'px';
|
||
element.style.left = newLeft + 'px';
|
||
}
|
||
|
||
function closeDragElement() {
|
||
isDragging = false;
|
||
element.classList.remove('wplace-dragging');
|
||
document.onmouseup = null;
|
||
document.onmousemove = null;
|
||
document.body.style.userSelect = '';
|
||
}
|
||
}
|
||
|
||
makeDraggable(statsContainer);
|
||
makeDraggable(container);
|
||
|
||
if (statsBtn && closeStatsBtn) {
|
||
statsBtn.addEventListener('click', () => {
|
||
const isVisible = statsContainer.style.display !== 'none';
|
||
if (isVisible) {
|
||
statsContainer.style.display = 'none';
|
||
statsBtn.innerHTML = '<i class="fas fa-chart-bar"></i>';
|
||
statsBtn.title = Utils.t('showStats');
|
||
} else {
|
||
statsContainer.style.display = 'block';
|
||
statsBtn.innerHTML = '<i class="fas fa-chart-line"></i>';
|
||
statsBtn.title = Utils.t('hideStats');
|
||
}
|
||
});
|
||
|
||
closeStatsBtn.addEventListener('click', () => {
|
||
statsContainer.style.display = 'none';
|
||
statsBtn.innerHTML = '<i class="fas fa-chart-bar"></i>';
|
||
statsBtn.title = Utils.t('showStats');
|
||
});
|
||
|
||
if (refreshChargesBtn) {
|
||
refreshChargesBtn.addEventListener('click', async () => {
|
||
refreshChargesBtn.innerHTML = '<i class="fas fa-spinner fa-spin"></i>';
|
||
refreshChargesBtn.disabled = true;
|
||
|
||
try {
|
||
await updateStats(true);
|
||
} catch (error) {
|
||
console.error('Error refreshing charges:', error);
|
||
} finally {
|
||
refreshChargesBtn.innerHTML = '<i class="fas fa-sync"></i>';
|
||
refreshChargesBtn.disabled = false;
|
||
}
|
||
});
|
||
}
|
||
}
|
||
if (statsContainer && statsBtn) {
|
||
// Stats container starts hidden - user clicks button to show
|
||
statsBtn.innerHTML = '<i class="fas fa-chart-bar"></i>';
|
||
statsBtn.title = Utils.t('showStats');
|
||
}
|
||
|
||
const settingsBtn = container.querySelector('#settingsBtn');
|
||
const closeSettingsBtn = settingsContainer.querySelector('#closeSettingsBtn');
|
||
const applySettingsBtn = settingsContainer.querySelector('#applySettingsBtn');
|
||
|
||
if (settingsBtn && closeSettingsBtn && applySettingsBtn) {
|
||
settingsBtn.addEventListener('click', () => {
|
||
const isVisible = settingsContainer.classList.contains('show');
|
||
if (isVisible) {
|
||
settingsContainer.style.animation = 'settings-fade-out 0.3s ease-out forwards';
|
||
settingsContainer.classList.remove('show');
|
||
setTimeout(() => {
|
||
settingsContainer.style.animation = '';
|
||
}, 300);
|
||
} else {
|
||
settingsContainer.style.top = '50%';
|
||
settingsContainer.style.left = '50%';
|
||
settingsContainer.style.transform = 'translate(-50%, -50%)';
|
||
settingsContainer.classList.add('show');
|
||
settingsContainer.style.animation = 'settings-slide-in 0.4s ease-out';
|
||
}
|
||
});
|
||
|
||
closeSettingsBtn.addEventListener('click', () => {
|
||
settingsContainer.style.animation = 'settings-fade-out 0.3s ease-out forwards';
|
||
settingsContainer.classList.remove('show');
|
||
setTimeout(() => {
|
||
settingsContainer.style.animation = '';
|
||
settingsContainer.style.top = '50%';
|
||
settingsContainer.style.left = '50%';
|
||
settingsContainer.style.transform = 'translate(-50%, -50%)';
|
||
}, 300);
|
||
});
|
||
|
||
applySettingsBtn.addEventListener('click', () => {
|
||
// Sync advanced settings before save
|
||
const colorAlgorithmSelect = document.getElementById('colorAlgorithmSelect');
|
||
if (colorAlgorithmSelect) state.colorMatchingAlgorithm = colorAlgorithmSelect.value;
|
||
const enableChromaPenaltyToggle = document.getElementById('enableChromaPenaltyToggle');
|
||
if (enableChromaPenaltyToggle)
|
||
state.enableChromaPenalty = enableChromaPenaltyToggle.checked;
|
||
const chromaPenaltyWeightSlider = document.getElementById('chromaPenaltyWeightSlider');
|
||
if (chromaPenaltyWeightSlider)
|
||
state.chromaPenaltyWeight = parseFloat(chromaPenaltyWeightSlider.value) || 0.15;
|
||
const transparencyThresholdInput = document.getElementById('transparencyThresholdInput');
|
||
if (transparencyThresholdInput) {
|
||
const v = parseInt(transparencyThresholdInput.value, 10);
|
||
if (!isNaN(v) && v >= 0 && v <= 255) state.customTransparencyThreshold = v;
|
||
}
|
||
const whiteThresholdInput = document.getElementById('whiteThresholdInput');
|
||
if (whiteThresholdInput) {
|
||
const v = parseInt(whiteThresholdInput.value, 10);
|
||
if (!isNaN(v) && v >= 200 && v <= 255) state.customWhiteThreshold = v;
|
||
}
|
||
// Update functional thresholds
|
||
CONFIG.TRANSPARENCY_THRESHOLD = state.customTransparencyThreshold;
|
||
CONFIG.WHITE_THRESHOLD = state.customWhiteThreshold;
|
||
// Notifications
|
||
const notifEnabledToggle = document.getElementById('notifEnabledToggle');
|
||
const notifOnChargesToggle = document.getElementById('notifOnChargesToggle');
|
||
const notifOnlyUnfocusedToggle = document.getElementById('notifOnlyUnfocusedToggle');
|
||
const notifIntervalInput = document.getElementById('notifIntervalInput');
|
||
if (notifEnabledToggle) state.notificationsEnabled = !!notifEnabledToggle.checked;
|
||
if (notifOnChargesToggle) state.notifyOnChargesReached = !!notifOnChargesToggle.checked;
|
||
if (notifOnlyUnfocusedToggle)
|
||
state.notifyOnlyWhenUnfocused = !!notifOnlyUnfocusedToggle.checked;
|
||
if (notifIntervalInput) {
|
||
const v = parseInt(notifIntervalInput.value, 10);
|
||
if (!isNaN(v) && v >= 1 && v <= 60) state.notificationIntervalMinutes = v;
|
||
}
|
||
saveBotSettings();
|
||
Utils.showAlert(Utils.t('settingsSaved'), 'success');
|
||
closeSettingsBtn.click();
|
||
NotificationManager.syncFromState();
|
||
});
|
||
|
||
makeDraggable(settingsContainer);
|
||
|
||
const tokenSourceSelect = settingsContainer.querySelector('#tokenSourceSelect');
|
||
if (tokenSourceSelect) {
|
||
tokenSourceSelect.addEventListener('change', (e) => {
|
||
state.tokenSource = e.target.value;
|
||
saveBotSettings();
|
||
console.log(`🔑 Token source changed to: ${state.tokenSource}`);
|
||
const sourceNames = {
|
||
generator: 'Automatic Generator',
|
||
hybrid: 'Generator + Auto Fallback',
|
||
manual: 'Manual Pixel Placement',
|
||
};
|
||
Utils.showAlert(
|
||
Utils.t('tokenSourceSet', { source: sourceNames[state.tokenSource] }),
|
||
'success'
|
||
);
|
||
});
|
||
}
|
||
|
||
// Batch mode controls
|
||
const batchModeSelect = settingsContainer.querySelector('#batchModeSelect');
|
||
const normalBatchControls = settingsContainer.querySelector('#normalBatchControls');
|
||
const randomBatchControls = settingsContainer.querySelector('#randomBatchControls');
|
||
const randomBatchMin = settingsContainer.querySelector('#randomBatchMin');
|
||
const randomBatchMax = settingsContainer.querySelector('#randomBatchMax');
|
||
|
||
if (batchModeSelect) {
|
||
batchModeSelect.addEventListener('change', (e) => {
|
||
state.batchMode = e.target.value;
|
||
|
||
// Switch between normal and random controls
|
||
if (normalBatchControls && randomBatchControls) {
|
||
if (e.target.value === 'random') {
|
||
normalBatchControls.style.display = 'none';
|
||
randomBatchControls.style.display = 'block';
|
||
} else {
|
||
normalBatchControls.style.display = 'block';
|
||
randomBatchControls.style.display = 'none';
|
||
}
|
||
}
|
||
|
||
saveBotSettings();
|
||
console.log(`📦 Batch mode changed to: ${state.batchMode}`);
|
||
Utils.showAlert(
|
||
Utils.t('batchModeSet', {
|
||
mode:
|
||
state.batchMode === 'random' ? Utils.t('randomRange') : Utils.t('normalFixedSize'),
|
||
}),
|
||
'success'
|
||
);
|
||
});
|
||
}
|
||
|
||
if (randomBatchMin) {
|
||
randomBatchMin.addEventListener('input', (e) => {
|
||
const min = parseInt(e.target.value);
|
||
if (min >= 1 && min <= 1000) {
|
||
state.randomBatchMin = min;
|
||
// Ensure min doesn't exceed max
|
||
if (randomBatchMax && min > state.randomBatchMax) {
|
||
state.randomBatchMax = min;
|
||
randomBatchMax.value = min;
|
||
}
|
||
saveBotSettings();
|
||
}
|
||
});
|
||
}
|
||
|
||
if (randomBatchMax) {
|
||
randomBatchMax.addEventListener('input', (e) => {
|
||
const max = parseInt(e.target.value);
|
||
if (max >= 1 && max <= 1000) {
|
||
state.randomBatchMax = max;
|
||
// Ensure max doesn't go below min
|
||
if (randomBatchMin && max < state.randomBatchMin) {
|
||
state.randomBatchMin = max;
|
||
randomBatchMin.value = max;
|
||
}
|
||
saveBotSettings();
|
||
}
|
||
});
|
||
}
|
||
|
||
const languageSelect = settingsContainer.querySelector('#languageSelect');
|
||
if (languageSelect) {
|
||
languageSelect.addEventListener('change', async (e) => {
|
||
const newLanguage = e.target.value;
|
||
state.language = newLanguage;
|
||
localStorage.setItem('wplace_language', newLanguage);
|
||
|
||
// Load the new language translations
|
||
await loadTranslations(newLanguage);
|
||
|
||
setTimeout(() => {
|
||
settingsContainer.style.display = 'none';
|
||
createUI();
|
||
}, 100);
|
||
});
|
||
}
|
||
|
||
const themeSelect = settingsContainer.querySelector('#themeSelect');
|
||
if (themeSelect) {
|
||
themeSelect.addEventListener('change', (e) => {
|
||
const newTheme = e.target.value;
|
||
switchTheme(newTheme);
|
||
});
|
||
}
|
||
|
||
const overlayOpacitySlider = settingsContainer.querySelector('#overlayOpacitySlider');
|
||
const overlayOpacityValue = settingsContainer.querySelector('#overlayOpacityValue');
|
||
const enableBlueMarbleToggle = settingsContainer.querySelector('#enableBlueMarbleToggle');
|
||
const settingsPaintWhiteToggle = settingsContainer.querySelector('#settingsPaintWhiteToggle');
|
||
const settingsPaintTransparentToggle = settingsContainer.querySelector(
|
||
'#settingsPaintTransparentToggle'
|
||
);
|
||
|
||
if (overlayOpacitySlider && overlayOpacityValue) {
|
||
const updateOpacity = (newValue) => {
|
||
const opacity = parseFloat(newValue);
|
||
state.overlayOpacity = opacity;
|
||
overlayOpacitySlider.value = opacity;
|
||
overlayOpacityValue.textContent = `${Math.round(opacity * 100)}%`;
|
||
};
|
||
|
||
overlayOpacitySlider.addEventListener('input', (e) => {
|
||
updateOpacity(e.target.value);
|
||
});
|
||
|
||
// Add scroll-to-adjust for overlay opacity slider
|
||
Utils.createScrollToAdjust(overlayOpacitySlider, updateOpacity, 0, 1, 0.05);
|
||
}
|
||
|
||
if (settingsPaintWhiteToggle) {
|
||
settingsPaintWhiteToggle.checked = state.paintWhitePixels;
|
||
settingsPaintWhiteToggle.addEventListener('change', (e) => {
|
||
state.paintWhitePixels = e.target.checked;
|
||
saveBotSettings();
|
||
console.log(`🎨 Paint white pixels: ${state.paintWhitePixels ? 'ON' : 'OFF'}`);
|
||
const statusText = state.paintWhitePixels
|
||
? 'White pixels in the template will be painted'
|
||
: 'White pixels will be skipped';
|
||
Utils.showAlert(statusText, 'success');
|
||
});
|
||
}
|
||
|
||
if (settingsPaintTransparentToggle) {
|
||
settingsPaintTransparentToggle.checked = state.paintTransparentPixels;
|
||
settingsPaintTransparentToggle.addEventListener('change', (e) => {
|
||
state.paintTransparentPixels = e.target.checked;
|
||
saveBotSettings();
|
||
console.log(
|
||
`🎨 Paint transparent pixels: ${state.paintTransparentPixels ? 'ON' : 'OFF'}`
|
||
);
|
||
const statusText = state.paintTransparentPixels
|
||
? 'Transparent pixels in the template will be painted with the closest available color'
|
||
: 'Transparent pixels will be skipped';
|
||
Utils.showAlert(statusText, 'success');
|
||
});
|
||
}
|
||
|
||
// Speed controls - both slider and input
|
||
const speedSlider = settingsContainer.querySelector('#speedSlider');
|
||
const speedInput = settingsContainer.querySelector('#speedInput');
|
||
const speedDecrease = settingsContainer.querySelector('#speedDecrease');
|
||
const speedIncrease = settingsContainer.querySelector('#speedIncrease');
|
||
const speedValue = settingsContainer.querySelector('#speedValue');
|
||
|
||
if (speedSlider && speedInput && speedValue && speedDecrease && speedIncrease) {
|
||
const updateSpeed = (newValue) => {
|
||
const speed = Math.max(CONFIG.PAINTING_SPEED.MIN, Math.min(CONFIG.PAINTING_SPEED.MAX, parseInt(newValue)));
|
||
state.paintingSpeed = speed;
|
||
|
||
// Update both controls (value shows in input, label shows unit only)
|
||
speedSlider.value = speed;
|
||
speedInput.value = speed;
|
||
speedValue.textContent = `pixels`;
|
||
|
||
saveBotSettings();
|
||
};
|
||
|
||
// Slider event listener
|
||
speedSlider.addEventListener('input', (e) => {
|
||
updateSpeed(e.target.value);
|
||
});
|
||
|
||
// Number input event listener
|
||
speedInput.addEventListener('input', (e) => {
|
||
updateSpeed(e.target.value);
|
||
});
|
||
|
||
// Decrease button
|
||
speedDecrease.addEventListener('click', () => {
|
||
updateSpeed(parseInt(speedInput.value) - 1);
|
||
});
|
||
|
||
// Increase button
|
||
speedIncrease.addEventListener('click', () => {
|
||
updateSpeed(parseInt(speedInput.value) + 1);
|
||
});
|
||
|
||
// Add scroll-to-adjust for speed slider
|
||
Utils.createScrollToAdjust(speedSlider, updateSpeed, CONFIG.PAINTING_SPEED.MIN, CONFIG.PAINTING_SPEED.MAX, 1);
|
||
}
|
||
|
||
if (enableBlueMarbleToggle) {
|
||
enableBlueMarbleToggle.addEventListener('click', async () => {
|
||
state.blueMarbleEnabled = enableBlueMarbleToggle.checked;
|
||
if (state.imageLoaded && overlayManager.imageBitmap) {
|
||
Utils.showAlert(Utils.t('reprocessingOverlay'), 'info');
|
||
await overlayManager.processImageIntoChunks();
|
||
Utils.showAlert(Utils.t('overlayUpdated'), 'success');
|
||
}
|
||
});
|
||
}
|
||
|
||
// (Advanced color listeners moved outside to work with resize dialog)
|
||
// (Advanced color listeners moved outside to work with resize dialog)
|
||
// Notifications listeners
|
||
const notifPermBtn = settingsContainer.querySelector('#notifRequestPermBtn');
|
||
const notifTestBtn = settingsContainer.querySelector('#notifTestBtn');
|
||
if (notifPermBtn) {
|
||
notifPermBtn.addEventListener('click', async () => {
|
||
const perm = await NotificationManager.requestPermission();
|
||
if (perm === 'granted') Utils.showAlert(Utils.t('notificationsEnabled'), 'success');
|
||
else Utils.showAlert(Utils.t('notificationsPermissionDenied'), 'warning');
|
||
});
|
||
}
|
||
if (notifTestBtn) {
|
||
notifTestBtn.addEventListener('click', () => {
|
||
NotificationManager.notify(
|
||
Utils.t('testNotificationTitle'),
|
||
Utils.t('testNotificationMessage'),
|
||
'wplace-notify-test',
|
||
true
|
||
);
|
||
});
|
||
}
|
||
}
|
||
|
||
const widthSlider = resizeContainer.querySelector('#widthSlider');
|
||
const heightSlider = resizeContainer.querySelector('#heightSlider');
|
||
const widthValue = resizeContainer.querySelector('#widthValue');
|
||
const heightValue = resizeContainer.querySelector('#heightValue');
|
||
const keepAspect = resizeContainer.querySelector('#keepAspect');
|
||
const paintWhiteToggle = resizeContainer.querySelector('#paintWhiteToggle');
|
||
const paintTransparentToggle = resizeContainer.querySelector('#paintTransparentToggle');
|
||
const zoomSlider = resizeContainer.querySelector('#zoomSlider');
|
||
const zoomValue = resizeContainer.querySelector('#zoomValue');
|
||
const zoomInBtn = resizeContainer.querySelector('#zoomInBtn');
|
||
const zoomOutBtn = resizeContainer.querySelector('#zoomOutBtn');
|
||
const zoomFitBtn = resizeContainer.querySelector('#zoomFitBtn');
|
||
const zoomActualBtn = resizeContainer.querySelector('#zoomActualBtn');
|
||
const panModeBtn = resizeContainer.querySelector('#panModeBtn');
|
||
const panStage = resizeContainer.querySelector('#resizePanStage');
|
||
const canvasStack = resizeContainer.querySelector('#resizeCanvasStack');
|
||
const baseCanvas = resizeContainer.querySelector('#resizeCanvas');
|
||
const maskCanvas = resizeContainer.querySelector('#maskCanvas');
|
||
const baseCtx = baseCanvas.getContext('2d');
|
||
const maskCtx = maskCanvas.getContext('2d');
|
||
const confirmResize = resizeContainer.querySelector('#confirmResize');
|
||
const cancelResize = resizeContainer.querySelector('#cancelResize');
|
||
const downloadPreviewBtn = resizeContainer.querySelector('#downloadPreviewBtn');
|
||
const clearIgnoredBtn = resizeContainer.querySelector('#clearIgnoredBtn');
|
||
|
||
// Coordinate generation controls with smart visibility
|
||
const coordinateModeSelect = settingsContainer.querySelector('#coordinateModeSelect');
|
||
const coordinateDirectionSelect = settingsContainer.querySelector('#coordinateDirectionSelect');
|
||
const coordinateSnakeToggle = settingsContainer.querySelector('#coordinateSnakeToggle');
|
||
const directionControls = settingsContainer.querySelector('#directionControls');
|
||
const snakeControls = settingsContainer.querySelector('#snakeControls');
|
||
const blockControls = settingsContainer.querySelector('#blockControls');
|
||
const blockWidthInput = settingsContainer.querySelector('#blockWidthInput');
|
||
const blockHeightInput = settingsContainer.querySelector('#blockHeightInput');
|
||
const paintUnavailablePixelsToggle = settingsContainer.querySelector(
|
||
'#paintUnavailablePixelsToggle'
|
||
);
|
||
|
||
if (paintUnavailablePixelsToggle) {
|
||
paintUnavailablePixelsToggle.checked = state.paintUnavailablePixels;
|
||
paintUnavailablePixelsToggle.addEventListener('change', (e) => {
|
||
state.paintUnavailablePixels = e.target.checked;
|
||
saveBotSettings();
|
||
console.log(`🎨 Paint unavailable colors: ${state.paintUnavailablePixels ? 'ON' : 'OFF'}`);
|
||
const statusText = state.paintUnavailablePixels
|
||
? 'Unavailable template colors will be painted with the closest available color'
|
||
: 'Unavailable template colors will be skipped';
|
||
Utils.showAlert(statusText, 'success');
|
||
});
|
||
}
|
||
if (coordinateModeSelect) {
|
||
coordinateModeSelect.value = state.coordinateMode;
|
||
coordinateModeSelect.addEventListener('change', (e) => {
|
||
state.coordinateMode = e.target.value;
|
||
Utils.updateCoordinateUI({
|
||
mode: state.coordinateMode,
|
||
directionControls,
|
||
snakeControls,
|
||
blockControls,
|
||
});
|
||
saveBotSettings();
|
||
console.log(`🔄 Coordinate mode changed to: ${state.coordinateMode}`);
|
||
Utils.showAlert(`Coordinate mode set to: ${state.coordinateMode}`, 'success');
|
||
});
|
||
}
|
||
|
||
if (coordinateDirectionSelect) {
|
||
coordinateDirectionSelect.value = state.coordinateDirection;
|
||
coordinateDirectionSelect.addEventListener('change', (e) => {
|
||
state.coordinateDirection = e.target.value;
|
||
saveBotSettings();
|
||
console.log(`🧭 Coordinate direction changed to: ${state.coordinateDirection}`);
|
||
Utils.showAlert(`Coordinate direction set to: ${state.coordinateDirection}`, 'success');
|
||
});
|
||
}
|
||
|
||
if (coordinateSnakeToggle) {
|
||
coordinateSnakeToggle.checked = state.coordinateSnake;
|
||
coordinateSnakeToggle.addEventListener('change', (e) => {
|
||
state.coordinateSnake = e.target.checked;
|
||
saveBotSettings();
|
||
console.log(`🐍 Snake pattern ${state.coordinateSnake ? 'enabled' : 'disabled'}`);
|
||
Utils.showAlert(
|
||
`Snake pattern ${state.coordinateSnake ? 'enabled' : 'disabled'}`,
|
||
'success'
|
||
);
|
||
});
|
||
}
|
||
|
||
if (blockWidthInput) {
|
||
blockWidthInput.value = state.blockWidth;
|
||
blockWidthInput.addEventListener('input', (e) => {
|
||
const width = parseInt(e.target.value);
|
||
if (width >= 1 && width <= 50) {
|
||
state.blockWidth = width;
|
||
saveBotSettings();
|
||
}
|
||
});
|
||
}
|
||
|
||
if (blockHeightInput) {
|
||
blockHeightInput.value = state.blockHeight;
|
||
blockHeightInput.addEventListener('change', (e) => {
|
||
const height = parseInt(e.target.value);
|
||
if (height >= 1 && height <= 50) {
|
||
state.blockHeight = height;
|
||
saveBotSettings();
|
||
}
|
||
});
|
||
}
|
||
|
||
if (compactBtn) {
|
||
compactBtn.addEventListener('click', () => {
|
||
container.classList.toggle('wplace-compact');
|
||
const isCompact = container.classList.contains('wplace-compact');
|
||
|
||
if (isCompact) {
|
||
compactBtn.innerHTML = '<i class="fas fa-expand"></i>';
|
||
compactBtn.title = Utils.t('expandMode');
|
||
} else {
|
||
compactBtn.innerHTML = '<i class="fas fa-compress"></i>';
|
||
compactBtn.title = Utils.t('compactMode');
|
||
}
|
||
});
|
||
}
|
||
|
||
if (minimizeBtn) {
|
||
minimizeBtn.addEventListener('click', () => {
|
||
state.minimized = !state.minimized;
|
||
if (state.minimized) {
|
||
container.classList.add('wplace-minimized');
|
||
content.classList.add('wplace-hidden');
|
||
minimizeBtn.innerHTML = '<i class="fas fa-expand"></i>';
|
||
minimizeBtn.title = Utils.t('restore');
|
||
} else {
|
||
container.classList.remove('wplace-minimized');
|
||
content.classList.remove('wplace-hidden');
|
||
minimizeBtn.innerHTML = '<i class="fas fa-minus"></i>';
|
||
minimizeBtn.title = Utils.t('minimize');
|
||
}
|
||
saveBotSettings();
|
||
});
|
||
}
|
||
|
||
if (toggleOverlayBtn) {
|
||
toggleOverlayBtn.addEventListener('click', () => {
|
||
const isEnabled = overlayManager.toggle();
|
||
toggleOverlayBtn.classList.toggle('active', isEnabled);
|
||
toggleOverlayBtn.setAttribute('aria-pressed', isEnabled ? 'true' : 'false');
|
||
Utils.showAlert(isEnabled ? Utils.t('overlayEnabled') : Utils.t('overlayDisabled'), 'info');
|
||
});
|
||
}
|
||
|
||
if (state.minimized) {
|
||
container.classList.add('wplace-minimized');
|
||
content.classList.add('wplace-hidden');
|
||
if (minimizeBtn) {
|
||
minimizeBtn.innerHTML = '<i class="fas fa-expand"></i>';
|
||
minimizeBtn.title = Utils.t('restore');
|
||
}
|
||
} else {
|
||
container.classList.remove('wplace-minimized');
|
||
content.classList.remove('wplace-hidden');
|
||
if (minimizeBtn) {
|
||
minimizeBtn.innerHTML = '<i class="fas fa-minus"></i>';
|
||
minimizeBtn.title = Utils.t('minimize');
|
||
}
|
||
}
|
||
|
||
if (saveBtn) {
|
||
saveBtn.addEventListener('click', () => {
|
||
if (!state.imageLoaded) {
|
||
Utils.showAlert(Utils.t('missingRequirements'), 'error');
|
||
return;
|
||
}
|
||
|
||
const success = Utils.saveProgress();
|
||
if (success) {
|
||
updateUI('autoSaved', 'success');
|
||
Utils.showAlert(Utils.t('autoSaved'), 'success');
|
||
} else {
|
||
Utils.showAlert(Utils.t('errorSavingProgress'), 'error');
|
||
}
|
||
});
|
||
}
|
||
|
||
if (loadBtn) {
|
||
loadBtn.addEventListener('click', () => {
|
||
// Check if initial setup is complete
|
||
if (!state.initialSetupComplete) {
|
||
Utils.showAlert(Utils.t('pleaseWaitInitialSetup'), 'warning');
|
||
return;
|
||
}
|
||
|
||
const savedData = Utils.loadProgress();
|
||
if (!savedData) {
|
||
updateUI('noSavedData', 'warning');
|
||
Utils.showAlert(Utils.t('noSavedData'), 'warning');
|
||
return;
|
||
}
|
||
|
||
const confirmLoad = confirm(
|
||
`${Utils.t('savedDataFound')}\n\n` +
|
||
`Saved: ${new Date(savedData.timestamp).toLocaleString()}\n` +
|
||
`Progress: ${savedData.state.paintedPixels}/${savedData.state.totalPixels} pixels`
|
||
);
|
||
|
||
if (confirmLoad) {
|
||
const success = Utils.restoreProgress(savedData);
|
||
if (success) {
|
||
updateUI('dataLoaded', 'success');
|
||
Utils.showAlert(Utils.t('dataLoaded'), 'success');
|
||
updateDataButtons();
|
||
|
||
updateStats();
|
||
|
||
// Restore overlay if image data was loaded from localStorage
|
||
Utils.restoreOverlayFromData().catch((error) => {
|
||
console.error('Failed to restore overlay from localStorage:', error);
|
||
});
|
||
|
||
if (!state.colorsChecked) {
|
||
uploadBtn.disabled = false;
|
||
} else {
|
||
uploadBtn.disabled = false;
|
||
selectPosBtn.disabled = false;
|
||
}
|
||
|
||
if (state.imageLoaded && state.startPosition && state.region && state.colorsChecked) {
|
||
startBtn.disabled = false;
|
||
}
|
||
} else {
|
||
Utils.showAlert(Utils.t('errorLoadingProgress'), 'error');
|
||
}
|
||
}
|
||
});
|
||
}
|
||
|
||
if (saveToFileBtn) {
|
||
saveToFileBtn.addEventListener('click', () => {
|
||
const success = Utils.saveProgressToFile();
|
||
if (success) {
|
||
updateUI('fileSaved', 'success');
|
||
Utils.showAlert(Utils.t('fileSaved'), 'success');
|
||
} else {
|
||
Utils.showAlert(Utils.t('fileError'), 'error');
|
||
}
|
||
});
|
||
}
|
||
|
||
if (loadFromFileBtn) {
|
||
loadFromFileBtn.addEventListener('click', async () => {
|
||
// Check if initial setup is complete
|
||
if (!state.initialSetupComplete) {
|
||
Utils.showAlert(Utils.t('pleaseWaitFileSetup'), 'warning');
|
||
return;
|
||
}
|
||
|
||
try {
|
||
const success = await Utils.loadProgressFromFile();
|
||
if (success) {
|
||
updateUI('fileLoaded', 'success');
|
||
Utils.showAlert(Utils.t('fileLoaded'), 'success');
|
||
updateDataButtons();
|
||
|
||
await updateStats();
|
||
|
||
// Restore overlay if image data was loaded from file
|
||
await Utils.restoreOverlayFromData().catch((error) => {
|
||
console.error('Failed to restore overlay from file:', error);
|
||
});
|
||
|
||
if (state.colorsChecked) {
|
||
uploadBtn.disabled = false;
|
||
selectPosBtn.disabled = false;
|
||
resizeBtn.disabled = false;
|
||
} else {
|
||
uploadBtn.disabled = false;
|
||
}
|
||
|
||
if (state.imageLoaded && state.startPosition && state.region && state.colorsChecked) {
|
||
startBtn.disabled = false;
|
||
}
|
||
}
|
||
} catch (error) {
|
||
if (error.message === 'Invalid JSON file') {
|
||
Utils.showAlert(Utils.t('invalidFileFormat'), 'error');
|
||
} else {
|
||
Utils.showAlert(Utils.t('fileError'), 'error');
|
||
}
|
||
}
|
||
});
|
||
}
|
||
|
||
updateUI = (messageKey, type = 'default', params = {}, silent = false) => {
|
||
const message = Utils.t(messageKey, params);
|
||
statusText.textContent = message;
|
||
statusText.className = `wplace-status status-${type}`;
|
||
|
||
if (!silent) {
|
||
// Trigger animation only when silent = false
|
||
statusText.style.animation = 'none';
|
||
void statusText.offsetWidth; // trick to restart the animation
|
||
statusText.style.animation = 'slide-in 0.3s ease-out';
|
||
}
|
||
};
|
||
|
||
function updateChargeStatsDisplay(intervalMs) {
|
||
const currentChargesEl = document.getElementById('wplace-stat-charges-value');
|
||
const fullChargeEl = document.getElementById('wplace-stat-fullcharge-value');
|
||
if (!fullChargeEl && !currentChargesEl) return;
|
||
if (!state.fullChargeData) {
|
||
fullChargeEl.textContent = '--:--:--';
|
||
return;
|
||
}
|
||
|
||
const { current, max, cooldownMs, startTime, spentSinceShot } = state.fullChargeData;
|
||
const elapsed = Date.now() - startTime;
|
||
|
||
// total charges including elapsed time and spent during painting since snapshot
|
||
const chargesGained = elapsed / cooldownMs;
|
||
const rawCharges = current + chargesGained - spentSinceShot;
|
||
const cappedCharges = Math.min(rawCharges, max);
|
||
|
||
// rounding with 0.95 threshold
|
||
let displayCharges;
|
||
const fraction = cappedCharges - Math.floor(cappedCharges);
|
||
if (fraction >= 0.95) {
|
||
displayCharges = Math.ceil(cappedCharges);
|
||
} else {
|
||
displayCharges = Math.floor(cappedCharges);
|
||
}
|
||
|
||
state.displayCharges = Math.max(0, displayCharges);
|
||
state.preciseCurrentCharges = cappedCharges;
|
||
|
||
const remainingMs = getMsToTargetCharges(cappedCharges, max, state.cooldown, intervalMs);
|
||
const timeText = Utils.msToTimeText(remainingMs);
|
||
|
||
if (currentChargesEl) {
|
||
currentChargesEl.innerHTML = `${state.displayCharges} / ${state.maxCharges}`;
|
||
}
|
||
|
||
if (
|
||
state.displayCharges < state.cooldownChargeThreshold &&
|
||
!state.stopFlag &&
|
||
state.running
|
||
) {
|
||
updateChargesThresholdUI(intervalMs);
|
||
}
|
||
|
||
if (fullChargeEl) {
|
||
if (state.displayCharges >= max) {
|
||
fullChargeEl.innerHTML = `<span style="color:#10b981;">FULL</span>`;
|
||
} else {
|
||
fullChargeEl.innerHTML = `
|
||
<span style="color:#f59e0b;">${timeText}</span>
|
||
`;
|
||
}
|
||
}
|
||
}
|
||
|
||
updateStats = async (isManualRefresh = false) => {
|
||
const isForcedRefresh = isManualRefresh;
|
||
const isFirstCheck = !state.fullChargeData?.startTime;
|
||
|
||
const minUpdateInterval = 60_000;
|
||
const maxUpdateInterval = 90_000;
|
||
const randomUpdateThreshold =
|
||
minUpdateInterval + Math.random() * (maxUpdateInterval - minUpdateInterval);
|
||
const timeSinceLastUpdate = Date.now() - (state.fullChargeData?.startTime || 0);
|
||
const isTimeToUpdate = timeSinceLastUpdate >= randomUpdateThreshold;
|
||
|
||
const shouldCallApi = isForcedRefresh || isFirstCheck || isTimeToUpdate;
|
||
|
||
if (shouldCallApi) {
|
||
const { charges, max, cooldown } = await WPlaceService.getCharges();
|
||
state.displayCharges = Math.floor(charges);
|
||
state.preciseCurrentCharges = charges;
|
||
state.cooldown = cooldown;
|
||
state.maxCharges = Math.floor(max) > 1 ? Math.floor(max) : state.maxCharges;
|
||
|
||
state.fullChargeData = {
|
||
current: charges,
|
||
max: max,
|
||
cooldownMs: cooldown,
|
||
startTime: Date.now(),
|
||
spentSinceShot: 0,
|
||
};
|
||
// Evaluate notifications every time we refresh server-side charges
|
||
NotificationManager.maybeNotifyChargesReached();
|
||
}
|
||
|
||
if (state.fullChargeInterval) {
|
||
clearInterval(state.fullChargeInterval);
|
||
state.fullChargeInterval = null;
|
||
}
|
||
const intervalMs = 1000;
|
||
state.fullChargeInterval = setInterval(
|
||
() => updateChargeStatsDisplay(intervalMs),
|
||
intervalMs
|
||
);
|
||
|
||
if (cooldownSlider && cooldownSlider.max !== state.maxCharges) {
|
||
cooldownSlider.max = state.maxCharges;
|
||
}
|
||
if (cooldownInput && cooldownInput.max !== state.maxCharges) {
|
||
cooldownInput.max = state.maxCharges;
|
||
}
|
||
|
||
let imageStatsHTML = '';
|
||
if (state.imageLoaded) {
|
||
const progress =
|
||
state.totalPixels > 0 ? Math.round((state.paintedPixels / state.totalPixels) * 100) : 0;
|
||
const remainingPixels = state.totalPixels - state.paintedPixels;
|
||
state.estimatedTime = Utils.calculateEstimatedTime(
|
||
remainingPixels,
|
||
state.displayCharges,
|
||
state.cooldown
|
||
);
|
||
progressBar.style.width = `${progress}%`;
|
||
|
||
imageStatsHTML = `
|
||
<div class="wplace-stat-item">
|
||
<div class="wplace-stat-label"><i class="fas fa-image"></i> ${Utils.t('progress')}</div>
|
||
<div class="wplace-stat-value">${progress}%</div>
|
||
</div>
|
||
<div class="wplace-stat-item">
|
||
<div class="wplace-stat-label"><i class="fas fa-paint-brush"></i> ${Utils.t(
|
||
'pixels'
|
||
)}</div>
|
||
<div class="wplace-stat-value">${state.paintedPixels}/${state.totalPixels}</div>
|
||
</div>
|
||
<div class="wplace-stat-item">
|
||
<div class="wplace-stat-label"><i class="fas fa-clock"></i> ${Utils.t(
|
||
'estimatedTime'
|
||
)}</div>
|
||
<div class="wplace-stat-value">${Utils.formatTime(state.estimatedTime)}</div>
|
||
</div>
|
||
`;
|
||
}
|
||
|
||
let colorSwatchesHTML = '';
|
||
state.availableColors = state.availableColors.filter(
|
||
(c) => c.name !== 'Unknown CoIor NaN' && c.id !== null
|
||
);
|
||
|
||
const availableColors = Utils.extractAvailableColors();
|
||
const newCount = Array.isArray(availableColors) ? availableColors.length : 0;
|
||
|
||
if (newCount === 0 && isManualRefresh) {
|
||
Utils.showAlert(Utils.t('noColorsFound'), 'warning');
|
||
} else if (newCount > 0 && state.availableColors.length < newCount) {
|
||
const oldCount = state.availableColors.length;
|
||
|
||
Utils.showAlert(
|
||
Utils.t('colorsUpdated', {
|
||
oldCount,
|
||
newCount: newCount,
|
||
diffCount: newCount - oldCount,
|
||
}),
|
||
'success'
|
||
);
|
||
state.availableColors = availableColors;
|
||
}
|
||
if (state.colorsChecked) {
|
||
colorSwatchesHTML = state.availableColors
|
||
.map((color) => {
|
||
const rgbString = `rgb(${color.rgb.join(',')})`;
|
||
return `<div class="wplace-stat-color-swatch" style="background-color: ${rgbString};" title="${Utils.t(
|
||
'colorTooltip',
|
||
{ id: color.id, rgb: color.rgb.join(', ') }
|
||
)}"></div>`;
|
||
})
|
||
.join('');
|
||
}
|
||
|
||
statsArea.innerHTML = `
|
||
${imageStatsHTML}
|
||
<div class="wplace-stat-item">
|
||
<div class="wplace-stat-label">
|
||
<i class="fas fa-bolt"></i> ${Utils.t('charges')}
|
||
</div>
|
||
<div class="wplace-stat-value" id="wplace-stat-charges-value">
|
||
${state.displayCharges} / ${state.maxCharges}
|
||
</div>
|
||
</div>
|
||
<div class="wplace-stat-item">
|
||
<div class="wplace-stat-label">
|
||
<i class="fas fa-battery-half"></i> ${Utils.t('fullChargeIn')}
|
||
</div>
|
||
<div class="wplace-stat-value" id="wplace-stat-fullcharge-value">--:--:--</div>
|
||
</div>
|
||
${state.colorsChecked
|
||
? `
|
||
<div class="wplace-colors-section">
|
||
<div class="wplace-stat-label"><i class="fas fa-palette"></i> ${Utils.t(
|
||
'availableColors',
|
||
{ count: state.availableColors.length }
|
||
)}</div>
|
||
<div class="wplace-stat-colors-grid">
|
||
${colorSwatchesHTML}
|
||
</div>
|
||
</div>
|
||
`
|
||
: ''
|
||
}
|
||
`;
|
||
|
||
// should be after statsArea.innerHTML = '...'. todo make full stats ui update partial
|
||
updateChargeStatsDisplay(intervalMs);
|
||
};
|
||
|
||
updateDataButtons = () => {
|
||
const hasImageData = state.imageLoaded && state.imageData;
|
||
saveBtn.disabled = !hasImageData;
|
||
saveToFileBtn.disabled = !hasImageData;
|
||
};
|
||
|
||
updateDataButtons();
|
||
|
||
function showResizeDialog(processor) {
|
||
let baseProcessor = processor;
|
||
let width, height;
|
||
if (state.originalImage?.dataUrl) {
|
||
baseProcessor = new ImageProcessor(state.originalImage.dataUrl);
|
||
width = state.originalImage.width;
|
||
height = state.originalImage.height;
|
||
} else {
|
||
const dims = processor.getDimensions();
|
||
width = dims.width;
|
||
height = dims.height;
|
||
}
|
||
const aspectRatio = width / height;
|
||
|
||
const rs = state.resizeSettings;
|
||
widthSlider.max = width * 2;
|
||
heightSlider.max = height * 2;
|
||
let initialW = width;
|
||
let initialH = height;
|
||
if (
|
||
rs &&
|
||
Number.isFinite(rs.width) &&
|
||
Number.isFinite(rs.height) &&
|
||
rs.width > 0 &&
|
||
rs.height > 0
|
||
) {
|
||
initialW = rs.width;
|
||
initialH = rs.height;
|
||
}
|
||
// Clamp to slider ranges
|
||
initialW = Math.max(
|
||
parseInt(widthSlider.min, 10) || 10,
|
||
Math.min(initialW, parseInt(widthSlider.max, 10))
|
||
);
|
||
initialH = Math.max(
|
||
parseInt(heightSlider.min, 10) || 10,
|
||
Math.min(initialH, parseInt(heightSlider.max, 10))
|
||
);
|
||
widthSlider.value = initialW;
|
||
heightSlider.value = initialH;
|
||
widthValue.textContent = initialW;
|
||
heightValue.textContent = initialH;
|
||
zoomSlider.value = 1;
|
||
if (zoomValue) zoomValue.textContent = '100%';
|
||
paintWhiteToggle.checked = state.paintWhitePixels;
|
||
paintTransparentToggle.checked = state.paintTransparentPixels;
|
||
|
||
let _previewTimer = null;
|
||
let _previewJobId = 0;
|
||
let _isDraggingSize = false;
|
||
let _zoomLevel = 1;
|
||
let _ditherWorkBuf = null;
|
||
let _ditherEligibleBuf = null;
|
||
const ensureDitherBuffers = (n) => {
|
||
if (!_ditherWorkBuf || _ditherWorkBuf.length !== n * 3)
|
||
_ditherWorkBuf = new Float32Array(n * 3);
|
||
if (!_ditherEligibleBuf || _ditherEligibleBuf.length !== n)
|
||
_ditherEligibleBuf = new Uint8Array(n);
|
||
return { work: _ditherWorkBuf, eligible: _ditherEligibleBuf };
|
||
};
|
||
let _maskImageData = null;
|
||
let _maskData = null;
|
||
let _dirty = null;
|
||
const _resetDirty = () => {
|
||
_dirty = { minX: Infinity, minY: Infinity, maxX: -1, maxY: -1 };
|
||
};
|
||
const _markDirty = (x, y) => {
|
||
if (!_dirty) _resetDirty();
|
||
if (x < _dirty.minX) _dirty.minX = x;
|
||
if (y < _dirty.minY) _dirty.minY = y;
|
||
if (x > _dirty.maxX) _dirty.maxX = x;
|
||
if (y > _dirty.maxY) _dirty.maxY = y;
|
||
};
|
||
const _flushDirty = () => {
|
||
if (!_dirty || _dirty.maxX < _dirty.minX || _dirty.maxY < _dirty.minY) return;
|
||
const x = Math.max(0, _dirty.minX);
|
||
const y = Math.max(0, _dirty.minY);
|
||
const w = Math.min(maskCanvas.width - x, _dirty.maxX - x + 1);
|
||
const h = Math.min(maskCanvas.height - y, _dirty.maxY - y + 1);
|
||
if (w > 0 && h > 0) maskCtx.putImageData(_maskImageData, 0, 0, x, y, w, h);
|
||
_resetDirty();
|
||
};
|
||
const _ensureMaskOverlayBuffers = (w, h, rebuildFromMask = false) => {
|
||
if (!_maskImageData || _maskImageData.width !== w || _maskImageData.height !== h) {
|
||
_maskImageData = maskCtx.createImageData(w, h);
|
||
_maskData = _maskImageData.data;
|
||
rebuildFromMask = true;
|
||
}
|
||
if (rebuildFromMask) {
|
||
const m = state.resizeIgnoreMask;
|
||
const md = _maskData;
|
||
md.fill(0);
|
||
if (m) {
|
||
for (let i = 0; i < m.length; i++)
|
||
if (m[i]) {
|
||
const p = i * 4;
|
||
md[p] = 255;
|
||
md[p + 1] = 0;
|
||
md[p + 2] = 0;
|
||
md[p + 3] = 150;
|
||
}
|
||
}
|
||
maskCtx.putImageData(_maskImageData, 0, 0);
|
||
_resetDirty();
|
||
}
|
||
};
|
||
const ensureMaskSize = (w, h) => {
|
||
if (!state.resizeIgnoreMask || state.resizeIgnoreMask.length !== w * h) {
|
||
state.resizeIgnoreMask = new Uint8Array(w * h);
|
||
}
|
||
baseCanvas.width = w;
|
||
baseCanvas.height = h;
|
||
maskCanvas.width = w;
|
||
maskCanvas.height = h;
|
||
maskCtx.clearRect(0, 0, maskCanvas.width, maskCanvas.height);
|
||
// Ensure overlay buffers exist and rebuild from mask when dimensions change
|
||
_ensureMaskOverlayBuffers(w, h, true);
|
||
};
|
||
_updateResizePreview = async () => {
|
||
const jobId = ++_previewJobId;
|
||
const newWidth = parseInt(widthSlider.value, 10);
|
||
const newHeight = parseInt(heightSlider.value, 10);
|
||
_zoomLevel = parseFloat(zoomSlider.value);
|
||
|
||
widthValue.textContent = newWidth;
|
||
heightValue.textContent = newHeight;
|
||
|
||
ensureMaskSize(newWidth, newHeight);
|
||
canvasStack.style.width = newWidth + 'px';
|
||
canvasStack.style.height = newHeight + 'px';
|
||
baseCtx.imageSmoothingEnabled = false;
|
||
if (!state.availableColors || state.availableColors.length === 0) {
|
||
if (baseProcessor !== processor && (!baseProcessor.img || !baseProcessor.canvas)) {
|
||
await baseProcessor.load();
|
||
}
|
||
baseCtx.clearRect(0, 0, newWidth, newHeight);
|
||
baseCtx.drawImage(baseProcessor.img, 0, 0, newWidth, newHeight);
|
||
// Draw existing mask overlay buffer
|
||
maskCtx.clearRect(0, 0, maskCanvas.width, maskCanvas.height);
|
||
if (_maskImageData) maskCtx.putImageData(_maskImageData, 0, 0);
|
||
updateZoomLayout();
|
||
return;
|
||
}
|
||
if (baseProcessor !== processor && (!baseProcessor.img || !baseProcessor.canvas)) {
|
||
await baseProcessor.load();
|
||
}
|
||
baseCtx.clearRect(0, 0, newWidth, newHeight);
|
||
baseCtx.drawImage(baseProcessor.img, 0, 0, newWidth, newHeight);
|
||
const imgData = baseCtx.getImageData(0, 0, newWidth, newHeight);
|
||
const data = imgData.data;
|
||
|
||
const tThresh = state.customTransparencyThreshold || CONFIG.TRANSPARENCY_THRESHOLD;
|
||
|
||
const applyFSDither = () => {
|
||
const w = newWidth,
|
||
h = newHeight;
|
||
const n = w * h;
|
||
const { work, eligible } = ensureDitherBuffers(n);
|
||
for (let y = 0; y < h; y++) {
|
||
for (let x = 0; x < w; x++) {
|
||
const idx = y * w + x;
|
||
const i4 = idx * 4;
|
||
const r = data[i4],
|
||
g = data[i4 + 1],
|
||
b = data[i4 + 2],
|
||
a = data[i4 + 3];
|
||
const isEligible =
|
||
(state.paintTransparentPixels || a >= tThresh) &&
|
||
(state.paintWhitePixels || !Utils.isWhitePixel(r, g, b));
|
||
eligible[idx] = isEligible ? 1 : 0;
|
||
work[idx * 3] = r;
|
||
work[idx * 3 + 1] = g;
|
||
work[idx * 3 + 2] = b;
|
||
if (!isEligible) {
|
||
data[i4 + 3] = 0; // transparent in preview overlay
|
||
}
|
||
}
|
||
}
|
||
|
||
const diffuse = (nx, ny, er, eg, eb, factor) => {
|
||
if (nx < 0 || nx >= w || ny < 0 || ny >= h) return;
|
||
const nidx = ny * w + nx;
|
||
if (!eligible[nidx]) return;
|
||
const base = nidx * 3;
|
||
work[base] = Math.min(255, Math.max(0, work[base] + er * factor));
|
||
work[base + 1] = Math.min(255, Math.max(0, work[base + 1] + eg * factor));
|
||
work[base + 2] = Math.min(255, Math.max(0, work[base + 2] + eb * factor));
|
||
};
|
||
|
||
for (let y = 0; y < h; y++) {
|
||
for (let x = 0; x < w; x++) {
|
||
const idx = y * w + x;
|
||
if (!eligible[idx]) continue;
|
||
const base = idx * 3;
|
||
const r0 = work[base],
|
||
g0 = work[base + 1],
|
||
b0 = work[base + 2];
|
||
const [nr, ng, nb] = Utils.findClosestPaletteColor(
|
||
r0,
|
||
g0,
|
||
b0,
|
||
state.activeColorPalette
|
||
);
|
||
const i4 = idx * 4;
|
||
data[i4] = nr;
|
||
data[i4 + 1] = ng;
|
||
data[i4 + 2] = nb;
|
||
data[i4 + 3] = 255;
|
||
|
||
const er = r0 - nr;
|
||
const eg = g0 - ng;
|
||
const eb = b0 - nb;
|
||
|
||
diffuse(x + 1, y, er, eg, eb, 7 / 16);
|
||
diffuse(x - 1, y + 1, er, eg, eb, 3 / 16);
|
||
diffuse(x, y + 1, er, eg, eb, 5 / 16);
|
||
diffuse(x + 1, y + 1, er, eg, eb, 1 / 16);
|
||
}
|
||
}
|
||
};
|
||
|
||
// Skip expensive dithering while user is dragging sliders
|
||
if (state.ditheringEnabled && !_isDraggingSize) {
|
||
applyFSDither();
|
||
} else {
|
||
for (let i = 0; i < data.length; i += 4) {
|
||
const r = data[i],
|
||
g = data[i + 1],
|
||
b = data[i + 2],
|
||
a = data[i + 3];
|
||
if (
|
||
(!state.paintTransparentPixels && a < tThresh) ||
|
||
(!state.paintWhitePixels && Utils.isWhitePixel(r, g, b))
|
||
) {
|
||
data[i + 3] = 0;
|
||
continue;
|
||
}
|
||
const [nr, ng, nb] = Utils.findClosestPaletteColor(r, g, b, state.activeColorPalette);
|
||
data[i] = nr;
|
||
data[i + 1] = ng;
|
||
data[i + 2] = nb;
|
||
data[i + 3] = 255;
|
||
}
|
||
}
|
||
|
||
if (jobId !== _previewJobId) return;
|
||
baseCtx.putImageData(imgData, 0, 0);
|
||
maskCtx.clearRect(0, 0, maskCanvas.width, maskCanvas.height);
|
||
if (_maskImageData) maskCtx.putImageData(_maskImageData, 0, 0);
|
||
updateZoomLayout();
|
||
};
|
||
|
||
const onWidthInput = () => {
|
||
if (keepAspect.checked) {
|
||
heightSlider.value = Math.round(parseInt(widthSlider.value, 10) / aspectRatio);
|
||
}
|
||
_updateResizePreview();
|
||
const curW = parseInt(widthSlider.value, 10);
|
||
const curH = parseInt(heightSlider.value, 10);
|
||
state.resizeSettings = {
|
||
baseWidth: width,
|
||
baseHeight: height,
|
||
width: curW,
|
||
height: curH,
|
||
};
|
||
saveBotSettings();
|
||
// Auto-fit after size changes
|
||
const fit = typeof computeFitZoom === 'function' ? computeFitZoom() : 1;
|
||
if (!isNaN(fit) && isFinite(fit)) applyZoom(fit);
|
||
};
|
||
|
||
const onHeightInput = () => {
|
||
if (keepAspect.checked) {
|
||
widthSlider.value = Math.round(parseInt(heightSlider.value, 10) * aspectRatio);
|
||
}
|
||
_updateResizePreview();
|
||
const curW = parseInt(widthSlider.value, 10);
|
||
const curH = parseInt(heightSlider.value, 10);
|
||
state.resizeSettings = {
|
||
baseWidth: width,
|
||
baseHeight: height,
|
||
width: curW,
|
||
height: curH,
|
||
};
|
||
saveBotSettings();
|
||
// Auto-fit after size changes
|
||
const fit = typeof computeFitZoom === 'function' ? computeFitZoom() : 1;
|
||
if (!isNaN(fit) && isFinite(fit)) applyZoom(fit);
|
||
};
|
||
|
||
paintWhiteToggle.onchange = (e) => {
|
||
state.paintWhitePixels = e.target.checked;
|
||
_updateResizePreview();
|
||
saveBotSettings();
|
||
};
|
||
|
||
paintTransparentToggle.onchange = (e) => {
|
||
state.paintTransparentPixels = e.target.checked;
|
||
_updateResizePreview();
|
||
saveBotSettings();
|
||
};
|
||
|
||
let panX = 0,
|
||
panY = 0;
|
||
const clampPan = () => {
|
||
const wrapRect = panStage?.getBoundingClientRect() || {
|
||
width: 0,
|
||
height: 0,
|
||
};
|
||
const w = (baseCanvas.width || 1) * _zoomLevel;
|
||
const h = (baseCanvas.height || 1) * _zoomLevel;
|
||
if (w <= wrapRect.width) {
|
||
panX = Math.floor((wrapRect.width - w) / 2);
|
||
} else {
|
||
const minX = wrapRect.width - w;
|
||
panX = Math.min(0, Math.max(minX, panX));
|
||
}
|
||
if (h <= wrapRect.height) {
|
||
panY = Math.floor((wrapRect.height - h) / 2);
|
||
} else {
|
||
const minY = wrapRect.height - h;
|
||
panY = Math.min(0, Math.max(minY, panY));
|
||
}
|
||
};
|
||
let _panRaf = 0;
|
||
const applyPan = () => {
|
||
if (_panRaf) return;
|
||
_panRaf = requestAnimationFrame(() => {
|
||
clampPan();
|
||
canvasStack.style.transform = `translate3d(${Math.round(
|
||
panX
|
||
)}px, ${Math.round(panY)}px, 0) scale(${_zoomLevel})`;
|
||
_panRaf = 0;
|
||
});
|
||
};
|
||
|
||
const updateZoomLayout = () => {
|
||
const w = baseCanvas.width || 1,
|
||
h = baseCanvas.height || 1;
|
||
baseCanvas.style.width = w + 'px';
|
||
baseCanvas.style.height = h + 'px';
|
||
maskCanvas.style.width = w + 'px';
|
||
maskCanvas.style.height = h + 'px';
|
||
canvasStack.style.width = w + 'px';
|
||
canvasStack.style.height = h + 'px';
|
||
applyPan();
|
||
};
|
||
const applyZoom = (z) => {
|
||
_zoomLevel = Math.max(0.05, Math.min(20, z || 1));
|
||
zoomSlider.value = _zoomLevel;
|
||
updateZoomLayout();
|
||
if (zoomValue) zoomValue.textContent = `${Math.round(_zoomLevel * 100)}%`;
|
||
};
|
||
zoomSlider.addEventListener('input', () => {
|
||
applyZoom(parseFloat(zoomSlider.value));
|
||
});
|
||
if (zoomInBtn)
|
||
zoomInBtn.addEventListener('click', () => applyZoom(parseFloat(zoomSlider.value) + 0.1));
|
||
if (zoomOutBtn)
|
||
zoomOutBtn.addEventListener('click', () => applyZoom(parseFloat(zoomSlider.value) - 0.1));
|
||
const computeFitZoom = () => {
|
||
const wrapRect = panStage?.getBoundingClientRect();
|
||
if (!wrapRect) return 1;
|
||
const w = baseCanvas.width || 1;
|
||
const h = baseCanvas.height || 1;
|
||
const margin = 10;
|
||
const scaleX = (wrapRect.width - margin) / w;
|
||
const scaleY = (wrapRect.height - margin) / h;
|
||
return Math.max(0.05, Math.min(20, Math.min(scaleX, scaleY)));
|
||
};
|
||
if (zoomFitBtn)
|
||
zoomFitBtn.addEventListener('click', () => {
|
||
applyZoom(computeFitZoom());
|
||
centerInView();
|
||
});
|
||
if (zoomActualBtn)
|
||
zoomActualBtn.addEventListener('click', () => {
|
||
applyZoom(1);
|
||
centerInView();
|
||
});
|
||
|
||
const centerInView = () => {
|
||
if (!panStage) return;
|
||
const rect = panStage.getBoundingClientRect();
|
||
const w = (baseCanvas.width || 1) * _zoomLevel;
|
||
const h = (baseCanvas.height || 1) * _zoomLevel;
|
||
panX = Math.floor((rect.width - w) / 2);
|
||
panY = Math.floor((rect.height - h) / 2);
|
||
applyPan();
|
||
};
|
||
|
||
let isPanning = false;
|
||
let startX = 0,
|
||
startY = 0,
|
||
startPanX = 0,
|
||
startPanY = 0;
|
||
let allowPan = false; // Space key
|
||
let panMode = false; // Explicit pan mode toggle for touch/one-button mice
|
||
const isPanMouseButton = (e) => e.button === 1 || e.button === 2;
|
||
const setCursor = (val) => {
|
||
if (panStage) panStage.style.cursor = val;
|
||
};
|
||
const isPanActive = (e) => panMode || allowPan || isPanMouseButton(e);
|
||
const updatePanModeBtn = () => {
|
||
if (!panModeBtn) return;
|
||
panModeBtn.classList.toggle('active', panMode);
|
||
panModeBtn.setAttribute('aria-pressed', panMode ? 'true' : 'false');
|
||
};
|
||
if (panModeBtn) {
|
||
updatePanModeBtn();
|
||
panModeBtn.addEventListener('click', () => {
|
||
panMode = !panMode;
|
||
updatePanModeBtn();
|
||
setCursor(panMode ? 'grab' : '');
|
||
});
|
||
}
|
||
if (panStage) {
|
||
panStage.addEventListener('contextmenu', (e) => {
|
||
if (allowPan) e.preventDefault();
|
||
});
|
||
window.addEventListener('keydown', (e) => {
|
||
if (e.code === 'Space') {
|
||
allowPan = true;
|
||
setCursor('grab');
|
||
}
|
||
});
|
||
window.addEventListener('keyup', (e) => {
|
||
if (e.code === 'Space') {
|
||
allowPan = false;
|
||
if (!isPanning) setCursor('');
|
||
}
|
||
});
|
||
panStage.addEventListener('mousedown', (e) => {
|
||
if (!isPanActive(e)) return;
|
||
e.preventDefault();
|
||
isPanning = true;
|
||
startX = e.clientX;
|
||
startY = e.clientY;
|
||
startPanX = panX;
|
||
startPanY = panY;
|
||
setCursor('grabbing');
|
||
});
|
||
window.addEventListener('mousemove', (e) => {
|
||
if (!isPanning) return;
|
||
const dx = e.clientX - startX;
|
||
const dy = e.clientY - startY;
|
||
panX = startPanX + dx;
|
||
panY = startPanY + dy;
|
||
applyPan();
|
||
});
|
||
window.addEventListener('mouseup', () => {
|
||
if (isPanning) {
|
||
isPanning = false;
|
||
setCursor(allowPan ? 'grab' : '');
|
||
}
|
||
});
|
||
panStage.addEventListener(
|
||
'wheel',
|
||
(e) => {
|
||
if (!e.ctrlKey && !e.metaKey) return;
|
||
e.preventDefault();
|
||
const rect = panStage.getBoundingClientRect();
|
||
const cx = e.clientX - rect.left - panX;
|
||
const cy = e.clientY - rect.top - panY;
|
||
const before = _zoomLevel;
|
||
const step = Math.max(0.05, Math.min(0.5, Math.abs(e.deltaY) > 20 ? 0.2 : 0.1));
|
||
const next = Math.max(0.05, Math.min(20, before + (e.deltaY > 0 ? -step : step)));
|
||
if (next === before) return;
|
||
const scale = next / before;
|
||
panX = panX - cx * (scale - 1);
|
||
panY = panY - cy * (scale - 1);
|
||
applyZoom(next);
|
||
},
|
||
{ passive: false }
|
||
);
|
||
let lastTouchDist = null;
|
||
let touchStartTime = 0;
|
||
let doubleTapTimer = null;
|
||
panStage.addEventListener(
|
||
'touchstart',
|
||
(e) => {
|
||
if (e.touches.length === 1) {
|
||
const t = e.touches[0];
|
||
isPanning = true;
|
||
startX = t.clientX;
|
||
startY = t.clientY;
|
||
startPanX = panX;
|
||
startPanY = panY;
|
||
setCursor('grabbing');
|
||
const now = Date.now();
|
||
if (now - touchStartTime < 300) {
|
||
// double tap -> toggle 100%/fit
|
||
const z = Math.abs(_zoomLevel - 1) < 0.01 ? computeFitZoom() : 1;
|
||
applyZoom(z);
|
||
centerInView();
|
||
if (doubleTapTimer) clearTimeout(doubleTapTimer);
|
||
} else {
|
||
touchStartTime = now;
|
||
doubleTapTimer = setTimeout(() => {
|
||
doubleTapTimer = null;
|
||
}, 320);
|
||
}
|
||
} else if (e.touches.length === 2) {
|
||
// Pinch start
|
||
const [a, b] = e.touches;
|
||
lastTouchDist = Math.hypot(b.clientX - a.clientX, b.clientY - a.clientY);
|
||
}
|
||
},
|
||
{ passive: true }
|
||
);
|
||
panStage.addEventListener(
|
||
'touchmove',
|
||
(e) => {
|
||
if (e.touches.length === 1 && isPanning) {
|
||
const t = e.touches[0];
|
||
const dx = t.clientX - startX;
|
||
const dy = t.clientY - startY;
|
||
panX = startPanX + dx;
|
||
panY = startPanY + dy;
|
||
applyPan();
|
||
} else if (e.touches.length === 2 && lastTouchDist != null) {
|
||
e.preventDefault();
|
||
const [a, b] = e.touches;
|
||
const dist = Math.hypot(b.clientX - a.clientX, b.clientY - a.clientY);
|
||
const rect = panStage.getBoundingClientRect();
|
||
const centerX = (a.clientX + b.clientX) / 2 - rect.left - panX;
|
||
const centerY = (a.clientY + b.clientY) / 2 - rect.top - panY;
|
||
const before = _zoomLevel;
|
||
const scale = dist / (lastTouchDist || dist);
|
||
const next = Math.max(0.05, Math.min(20, before * scale));
|
||
if (next !== before) {
|
||
panX = panX - centerX * (next / before - 1);
|
||
panY = panY - centerY * (next / before - 1);
|
||
applyZoom(next);
|
||
}
|
||
lastTouchDist = dist;
|
||
}
|
||
},
|
||
{ passive: false }
|
||
);
|
||
panStage.addEventListener('touchend', () => {
|
||
isPanning = false;
|
||
lastTouchDist = null;
|
||
setCursor(panMode || allowPan ? 'grab' : '');
|
||
});
|
||
}
|
||
const schedulePreview = () => {
|
||
if (_previewTimer) clearTimeout(_previewTimer);
|
||
const run = () => {
|
||
_previewTimer = null;
|
||
_updateResizePreview();
|
||
};
|
||
if (window.requestIdleCallback) {
|
||
_previewTimer = setTimeout(() => requestIdleCallback(run, { timeout: 150 }), 50);
|
||
} else {
|
||
_previewTimer = setTimeout(() => requestAnimationFrame(run), 50);
|
||
}
|
||
};
|
||
// Track dragging to reduce work and skip dithering during drag
|
||
const markDragStart = () => {
|
||
_isDraggingSize = true;
|
||
};
|
||
const markDragEnd = () => {
|
||
_isDraggingSize = false;
|
||
schedulePreview();
|
||
};
|
||
widthSlider.addEventListener('pointerdown', markDragStart);
|
||
heightSlider.addEventListener('pointerdown', markDragStart);
|
||
widthSlider.addEventListener('pointerup', markDragEnd);
|
||
heightSlider.addEventListener('pointerup', markDragEnd);
|
||
widthSlider.addEventListener('input', () => {
|
||
onWidthInput();
|
||
schedulePreview();
|
||
});
|
||
heightSlider.addEventListener('input', () => {
|
||
onHeightInput();
|
||
schedulePreview();
|
||
});
|
||
|
||
// Mask painting UX: brush size, modes, row/column fills, and precise coords
|
||
let draggingMask = false;
|
||
let lastPaintX = -1,
|
||
lastPaintY = -1;
|
||
let brushSize = 1;
|
||
let rowColSize = 1;
|
||
let maskMode = 'ignore'; // 'ignore' | 'unignore' | 'toggle'
|
||
const brushEl = resizeContainer.querySelector('#maskBrushSize');
|
||
const brushValEl = resizeContainer.querySelector('#maskBrushSizeValue');
|
||
const btnIgnore = resizeContainer.querySelector('#maskModeIgnore');
|
||
const btnUnignore = resizeContainer.querySelector('#maskModeUnignore');
|
||
const btnToggle = resizeContainer.querySelector('#maskModeToggle');
|
||
const clearIgnoredBtnEl = resizeContainer.querySelector('#clearIgnoredBtn');
|
||
const invertMaskBtn = resizeContainer.querySelector('#invertMaskBtn');
|
||
const rowColSizeEl = resizeContainer.querySelector('#rowColSize');
|
||
const rowColSizeValEl = resizeContainer.querySelector('#rowColSizeValue');
|
||
|
||
const updateModeButtons = () => {
|
||
const map = [
|
||
[btnIgnore, 'ignore'],
|
||
[btnUnignore, 'unignore'],
|
||
[btnToggle, 'toggle'],
|
||
];
|
||
for (const [el, m] of map) {
|
||
if (!el) continue;
|
||
const active = maskMode === m;
|
||
el.classList.toggle('active', active);
|
||
el.setAttribute('aria-pressed', active ? 'true' : 'false');
|
||
}
|
||
};
|
||
const setMode = (mode) => {
|
||
maskMode = mode;
|
||
updateModeButtons();
|
||
};
|
||
if (brushEl && brushValEl) {
|
||
brushEl.addEventListener('input', () => {
|
||
brushSize = parseInt(brushEl.value, 10) || 1;
|
||
brushValEl.textContent = brushSize;
|
||
});
|
||
brushValEl.textContent = brushEl.value;
|
||
brushSize = parseInt(brushEl.value, 10) || 1;
|
||
}
|
||
if (rowColSizeEl && rowColSizeValEl) {
|
||
rowColSizeEl.addEventListener('input', () => {
|
||
rowColSize = parseInt(rowColSizeEl.value, 10) || 1;
|
||
rowColSizeValEl.textContent = rowColSize;
|
||
});
|
||
rowColSizeValEl.textContent = rowColSizeEl.value;
|
||
rowColSize = parseInt(rowColSizeEl.value, 10) || 1;
|
||
}
|
||
if (btnIgnore) btnIgnore.addEventListener('click', () => setMode('ignore'));
|
||
if (btnUnignore) btnUnignore.addEventListener('click', () => setMode('unignore'));
|
||
if (btnToggle) btnToggle.addEventListener('click', () => setMode('toggle'));
|
||
// Initialize button state (default to toggle mode)
|
||
updateModeButtons();
|
||
|
||
const mapClientToPixel = (clientX, clientY) => {
|
||
// Compute without rounding until final step to avoid drift at higher zoom
|
||
const rect = baseCanvas.getBoundingClientRect();
|
||
const scaleX = rect.width / baseCanvas.width;
|
||
const scaleY = rect.height / baseCanvas.height;
|
||
const dx = (clientX - rect.left) / scaleX;
|
||
const dy = (clientY - rect.top) / scaleY;
|
||
const x = Math.floor(dx);
|
||
const y = Math.floor(dy);
|
||
return { x, y };
|
||
};
|
||
|
||
const ensureMask = (w, h) => {
|
||
if (!state.resizeIgnoreMask || state.resizeIgnoreMask.length !== w * h) {
|
||
state.resizeIgnoreMask = new Uint8Array(w * h);
|
||
}
|
||
};
|
||
|
||
const paintCircle = (cx, cy, radius, value) => {
|
||
const w = baseCanvas.width,
|
||
h = baseCanvas.height;
|
||
ensureMask(w, h);
|
||
const r2 = radius * radius;
|
||
for (let yy = cy - radius; yy <= cy + radius; yy++) {
|
||
if (yy < 0 || yy >= h) continue;
|
||
for (let xx = cx - radius; xx <= cx + radius; xx++) {
|
||
if (xx < 0 || xx >= w) continue;
|
||
const dx = xx - cx,
|
||
dy = yy - cy;
|
||
if (dx * dx + dy * dy <= r2) {
|
||
const idx = yy * w + xx;
|
||
let val = state.resizeIgnoreMask[idx];
|
||
if (maskMode === 'toggle') {
|
||
val = val ? 0 : 1;
|
||
} else if (maskMode === 'ignore') {
|
||
val = 1;
|
||
} else {
|
||
val = 0;
|
||
}
|
||
state.resizeIgnoreMask[idx] = val;
|
||
if (_maskData) {
|
||
const p = idx * 4;
|
||
if (val) {
|
||
_maskData[p] = 255;
|
||
_maskData[p + 1] = 0;
|
||
_maskData[p + 2] = 0;
|
||
_maskData[p + 3] = 150;
|
||
} else {
|
||
_maskData[p] = 0;
|
||
_maskData[p + 1] = 0;
|
||
_maskData[p + 2] = 0;
|
||
_maskData[p + 3] = 0;
|
||
}
|
||
_markDirty(xx, yy);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
};
|
||
|
||
const paintRow = (y, value) => {
|
||
const w = baseCanvas.width,
|
||
h = baseCanvas.height;
|
||
ensureMask(w, h);
|
||
if (y < 0 || y >= h) return;
|
||
|
||
// Paint multiple rows based on rowColSize
|
||
const halfSize = Math.floor(rowColSize / 2);
|
||
const startY = Math.max(0, y - halfSize);
|
||
const endY = Math.min(h - 1, y + halfSize);
|
||
|
||
for (let rowY = startY; rowY <= endY; rowY++) {
|
||
for (let x = 0; x < w; x++) {
|
||
const idx = rowY * w + x;
|
||
let val = state.resizeIgnoreMask[idx];
|
||
if (maskMode === 'toggle') {
|
||
val = val ? 0 : 1;
|
||
} else if (maskMode === 'ignore') {
|
||
val = 1;
|
||
} else {
|
||
val = 0;
|
||
}
|
||
state.resizeIgnoreMask[idx] = val;
|
||
if (_maskData) {
|
||
const p = idx * 4;
|
||
if (val) {
|
||
_maskData[p] = 255;
|
||
_maskData[p + 1] = 0;
|
||
_maskData[p + 2] = 0;
|
||
_maskData[p + 3] = 150;
|
||
} else {
|
||
_maskData[p] = 0;
|
||
_maskData[p + 1] = 0;
|
||
_maskData[p + 2] = 0;
|
||
_maskData[p + 3] = 0;
|
||
}
|
||
}
|
||
}
|
||
if (_maskData) {
|
||
_markDirty(0, rowY);
|
||
_markDirty(w - 1, rowY);
|
||
}
|
||
}
|
||
};
|
||
|
||
const paintColumn = (x, value) => {
|
||
const w = baseCanvas.width,
|
||
h = baseCanvas.height;
|
||
ensureMask(w, h);
|
||
if (x < 0 || x >= w) return;
|
||
|
||
// Paint multiple columns based on rowColSize
|
||
const halfSize = Math.floor(rowColSize / 2);
|
||
const startX = Math.max(0, x - halfSize);
|
||
const endX = Math.min(w - 1, x + halfSize);
|
||
|
||
for (let colX = startX; colX <= endX; colX++) {
|
||
for (let y = 0; y < h; y++) {
|
||
const idx = y * w + colX;
|
||
let val = state.resizeIgnoreMask[idx];
|
||
if (maskMode === 'toggle') {
|
||
val = val ? 0 : 1;
|
||
} else if (maskMode === 'ignore') {
|
||
val = 1;
|
||
} else {
|
||
val = 0;
|
||
}
|
||
state.resizeIgnoreMask[idx] = val;
|
||
if (_maskData) {
|
||
const p = idx * 4;
|
||
if (val) {
|
||
_maskData[p] = 255;
|
||
_maskData[p + 1] = 0;
|
||
_maskData[p + 2] = 0;
|
||
_maskData[p + 3] = 150;
|
||
} else {
|
||
_maskData[p] = 0;
|
||
_maskData[p + 1] = 0;
|
||
_maskData[p + 2] = 0;
|
||
_maskData[p + 3] = 0;
|
||
}
|
||
}
|
||
}
|
||
if (_maskData) {
|
||
_markDirty(colX, 0);
|
||
_markDirty(colX, h - 1);
|
||
}
|
||
}
|
||
};
|
||
|
||
const redrawMaskOverlay = () => {
|
||
// Only flush the dirty region; full rebuild happens on size change
|
||
_flushDirty();
|
||
};
|
||
|
||
const handlePaint = (e) => {
|
||
// Suppress painting while panning
|
||
if ((e.buttons & 4) === 4 || (e.buttons & 2) === 2 || allowPan) return;
|
||
const { x, y } = mapClientToPixel(e.clientX, e.clientY);
|
||
const w = baseCanvas.width,
|
||
h = baseCanvas.height;
|
||
if (x < 0 || y < 0 || x >= w || y >= h) return;
|
||
const radius = Math.max(1, Math.floor(brushSize / 2));
|
||
if (e.shiftKey) {
|
||
paintRow(y);
|
||
} else if (e.altKey) {
|
||
paintColumn(x);
|
||
} else {
|
||
paintCircle(x, y, radius);
|
||
}
|
||
lastPaintX = x;
|
||
lastPaintY = y;
|
||
redrawMaskOverlay();
|
||
};
|
||
|
||
maskCanvas.addEventListener('mousedown', (e) => {
|
||
if (e.button === 1 || e.button === 2 || allowPan) return; // let pan handler manage
|
||
draggingMask = true;
|
||
handlePaint(e);
|
||
});
|
||
// Avoid hijacking touch gestures for panning/zooming
|
||
maskCanvas.addEventListener(
|
||
'touchstart',
|
||
(e) => {
|
||
/* let panStage handle */
|
||
},
|
||
{ passive: true }
|
||
);
|
||
maskCanvas.addEventListener(
|
||
'touchmove',
|
||
(e) => {
|
||
/* let panStage handle */
|
||
},
|
||
{ passive: true }
|
||
);
|
||
maskCanvas.addEventListener(
|
||
'touchend',
|
||
(e) => {
|
||
/* let panStage handle */
|
||
},
|
||
{ passive: true }
|
||
);
|
||
window.addEventListener('mousemove', (e) => {
|
||
if (draggingMask) handlePaint(e);
|
||
});
|
||
window.addEventListener('mouseup', () => {
|
||
if (draggingMask) {
|
||
draggingMask = false;
|
||
saveBotSettings();
|
||
}
|
||
});
|
||
|
||
if (clearIgnoredBtnEl)
|
||
clearIgnoredBtnEl.addEventListener('click', () => {
|
||
const w = baseCanvas.width,
|
||
h = baseCanvas.height;
|
||
if (state.resizeIgnoreMask) state.resizeIgnoreMask.fill(0);
|
||
_ensureMaskOverlayBuffers(w, h, true);
|
||
_updateResizePreview();
|
||
saveBotSettings();
|
||
});
|
||
|
||
if (invertMaskBtn)
|
||
invertMaskBtn.addEventListener('click', () => {
|
||
if (!state.resizeIgnoreMask) return;
|
||
for (let i = 0; i < state.resizeIgnoreMask.length; i++)
|
||
state.resizeIgnoreMask[i] = state.resizeIgnoreMask[i] ? 0 : 1;
|
||
const w = baseCanvas.width,
|
||
h = baseCanvas.height;
|
||
_ensureMaskOverlayBuffers(w, h, true);
|
||
_updateResizePreview();
|
||
saveBotSettings();
|
||
});
|
||
|
||
confirmResize.onclick = async () => {
|
||
const newWidth = parseInt(widthSlider.value, 10);
|
||
const newHeight = parseInt(heightSlider.value, 10);
|
||
|
||
// Generate the final paletted image data
|
||
const tempCanvas = document.createElement('canvas');
|
||
const tempCtx = tempCanvas.getContext('2d');
|
||
tempCanvas.width = newWidth;
|
||
tempCanvas.height = newHeight;
|
||
tempCtx.imageSmoothingEnabled = false;
|
||
if (baseProcessor !== processor && (!baseProcessor.img || !baseProcessor.canvas)) {
|
||
await baseProcessor.load();
|
||
}
|
||
tempCtx.drawImage(baseProcessor.img, 0, 0, newWidth, newHeight);
|
||
const imgData = tempCtx.getImageData(0, 0, newWidth, newHeight);
|
||
const data = imgData.data;
|
||
const tThresh2 = state.customTransparencyThreshold || CONFIG.TRANSPARENCY_THRESHOLD;
|
||
let totalValidPixels = 0;
|
||
const mask =
|
||
state.resizeIgnoreMask && state.resizeIgnoreMask.length === newWidth * newHeight
|
||
? state.resizeIgnoreMask
|
||
: null;
|
||
|
||
const applyFSDitherFinal = async () => {
|
||
const w = newWidth,
|
||
h = newHeight;
|
||
const n = w * h;
|
||
const { work, eligible } = ensureDitherBuffers(n);
|
||
for (let y = 0; y < h; y++) {
|
||
for (let x = 0; x < w; x++) {
|
||
const idx = y * w + x;
|
||
const i4 = idx * 4;
|
||
const r = data[i4],
|
||
g = data[i4 + 1],
|
||
b = data[i4 + 2],
|
||
a = data[i4 + 3];
|
||
const masked = mask && mask[idx];
|
||
const isEligible =
|
||
!masked &&
|
||
(state.paintTransparentPixels || a >= tThresh2) &&
|
||
(state.paintWhitePixels || !Utils.isWhitePixel(r, g, b));
|
||
eligible[idx] = isEligible ? 1 : 0;
|
||
work[idx * 3] = r;
|
||
work[idx * 3 + 1] = g;
|
||
work[idx * 3 + 2] = b;
|
||
if (!isEligible) {
|
||
data[i4 + 3] = 0;
|
||
}
|
||
}
|
||
// Yield to keep UI responsive
|
||
if ((y & 15) === 0) await Promise.resolve();
|
||
}
|
||
|
||
const diffuse = (nx, ny, er, eg, eb, factor) => {
|
||
if (nx < 0 || nx >= w || ny < 0 || ny >= h) return;
|
||
const nidx = ny * w + nx;
|
||
if (!eligible[nidx]) return;
|
||
const base = nidx * 3;
|
||
work[base] = Math.min(255, Math.max(0, work[base] + er * factor));
|
||
work[base + 1] = Math.min(255, Math.max(0, work[base + 1] + eg * factor));
|
||
work[base + 2] = Math.min(255, Math.max(0, work[base + 2] + eb * factor));
|
||
};
|
||
|
||
for (let y = 0; y < h; y++) {
|
||
for (let x = 0; x < w; x++) {
|
||
const idx = y * w + x;
|
||
if (!eligible[idx]) continue;
|
||
const base = idx * 3;
|
||
const r0 = work[base],
|
||
g0 = work[base + 1],
|
||
b0 = work[base + 2];
|
||
const [nr, ng, nb] = Utils.findClosestPaletteColor(
|
||
r0,
|
||
g0,
|
||
b0,
|
||
state.activeColorPalette
|
||
);
|
||
const i4 = idx * 4;
|
||
data[i4] = nr;
|
||
data[i4 + 1] = ng;
|
||
data[i4 + 2] = nb;
|
||
data[i4 + 3] = 255;
|
||
totalValidPixels++;
|
||
|
||
const er = r0 - nr;
|
||
const eg = g0 - ng;
|
||
const eb = b0 - nb;
|
||
|
||
diffuse(x + 1, y, er, eg, eb, 7 / 16);
|
||
diffuse(x - 1, y + 1, er, eg, eb, 3 / 16);
|
||
diffuse(x, y + 1, er, eg, eb, 5 / 16);
|
||
diffuse(x + 1, y + 1, er, eg, eb, 1 / 16);
|
||
}
|
||
// Yield every row to reduce jank
|
||
await Promise.resolve();
|
||
}
|
||
};
|
||
|
||
if (state.ditheringEnabled) {
|
||
await applyFSDitherFinal();
|
||
} else {
|
||
for (let i = 0; i < data.length; i += 4) {
|
||
const r = data[i],
|
||
g = data[i + 1],
|
||
b = data[i + 2],
|
||
a = data[i + 3];
|
||
const masked = mask && mask[i >> 2];
|
||
const isTransparent = (!state.paintTransparentPixels && a < tThresh2) || masked;
|
||
const isWhiteAndSkipped = !state.paintWhitePixels && Utils.isWhitePixel(r, g, b);
|
||
if (isTransparent || isWhiteAndSkipped) {
|
||
data[i + 3] = 0; // overlay transparency
|
||
continue;
|
||
}
|
||
totalValidPixels++;
|
||
const [nr, ng, nb] = Utils.findClosestPaletteColor(r, g, b, state.activeColorPalette);
|
||
data[i] = nr;
|
||
data[i + 1] = ng;
|
||
data[i + 2] = nb;
|
||
data[i + 3] = 255;
|
||
}
|
||
}
|
||
tempCtx.putImageData(imgData, 0, 0);
|
||
|
||
// Save the final pixel data for painting
|
||
// Persist the paletted (and possibly dithered) pixels so painting uses the same output seen in overlay
|
||
const palettedPixels = new Uint8ClampedArray(imgData.data);
|
||
state.imageData.pixels = palettedPixels;
|
||
state.imageData.width = newWidth;
|
||
state.imageData.height = newHeight;
|
||
state.imageData.totalPixels = totalValidPixels;
|
||
state.totalPixels = totalValidPixels;
|
||
state.paintedPixels = 0;
|
||
|
||
state.resizeSettings = {
|
||
baseWidth: width,
|
||
baseHeight: height,
|
||
width: newWidth,
|
||
height: newHeight,
|
||
};
|
||
saveBotSettings();
|
||
|
||
const finalImageBitmap = await createImageBitmap(tempCanvas);
|
||
await overlayManager.setImage(finalImageBitmap);
|
||
overlayManager.enable();
|
||
toggleOverlayBtn.classList.add('active');
|
||
toggleOverlayBtn.setAttribute('aria-pressed', 'true');
|
||
|
||
// Keep state.imageData.processor as the original-based source; painting uses paletted pixels already stored
|
||
|
||
updateStats();
|
||
updateUI('resizeSuccess', 'success', {
|
||
width: newWidth,
|
||
height: newHeight,
|
||
});
|
||
closeResizeDialog();
|
||
};
|
||
|
||
downloadPreviewBtn.onclick = () => {
|
||
try {
|
||
const w = baseCanvas.width,
|
||
h = baseCanvas.height;
|
||
const out = document.createElement('canvas');
|
||
out.width = w;
|
||
out.height = h;
|
||
const octx = out.getContext('2d');
|
||
octx.imageSmoothingEnabled = false;
|
||
octx.drawImage(baseCanvas, 0, 0);
|
||
octx.drawImage(maskCanvas, 0, 0);
|
||
const link = document.createElement('a');
|
||
link.download = 'wplace-preview.png';
|
||
link.href = out.toDataURL();
|
||
link.click();
|
||
} catch (e) {
|
||
console.warn('Failed to download preview:', e);
|
||
}
|
||
};
|
||
|
||
cancelResize.onclick = closeResizeDialog;
|
||
|
||
resizeOverlay.style.display = 'block';
|
||
resizeContainer.style.display = 'block';
|
||
|
||
// Reinitialize color palette with current available colors
|
||
initializeColorPalette(resizeContainer);
|
||
|
||
_updateResizePreview();
|
||
_resizeDialogCleanup = () => {
|
||
try {
|
||
zoomSlider.replaceWith(zoomSlider.cloneNode(true));
|
||
} catch { }
|
||
try {
|
||
if (zoomInBtn) zoomInBtn.replaceWith(zoomInBtn.cloneNode(true));
|
||
} catch { }
|
||
try {
|
||
if (zoomOutBtn) zoomOutBtn.replaceWith(zoomOutBtn.cloneNode(true));
|
||
} catch { }
|
||
};
|
||
setTimeout(() => {
|
||
if (typeof computeFitZoom === 'function') {
|
||
const z = computeFitZoom();
|
||
if (!isNaN(z) && isFinite(z)) {
|
||
applyZoom(z);
|
||
centerInView();
|
||
}
|
||
} else {
|
||
centerInView();
|
||
}
|
||
}, 0);
|
||
}
|
||
|
||
function closeResizeDialog() {
|
||
try {
|
||
if (typeof _resizeDialogCleanup === 'function') {
|
||
_resizeDialogCleanup();
|
||
}
|
||
} catch { }
|
||
resizeOverlay.style.display = 'none';
|
||
resizeContainer.style.display = 'none';
|
||
_updateResizePreview = () => { };
|
||
try {
|
||
if (typeof cancelAnimationFrame === 'function' && _panRaf) {
|
||
cancelAnimationFrame(_panRaf);
|
||
}
|
||
} catch { }
|
||
try {
|
||
if (_previewTimer) {
|
||
clearTimeout(_previewTimer);
|
||
_previewTimer = null;
|
||
}
|
||
} catch { }
|
||
_maskImageData = null;
|
||
_maskData = null;
|
||
_dirty = null;
|
||
_ditherWorkBuf = null;
|
||
_ditherEligibleBuf = null;
|
||
_resizeDialogCleanup = null;
|
||
}
|
||
|
||
if (uploadBtn) {
|
||
uploadBtn.addEventListener('click', async () => {
|
||
const availableColors = Utils.extractAvailableColors();
|
||
if (availableColors === null || availableColors.length < 10) {
|
||
updateUI('noColorsFound', 'error');
|
||
Utils.showAlert(Utils.t('noColorsFound'), 'error');
|
||
return;
|
||
}
|
||
|
||
if (!state.colorsChecked) {
|
||
state.availableColors = availableColors;
|
||
state.colorsChecked = true;
|
||
updateUI('colorsFound', 'success', { count: availableColors.length });
|
||
updateStats();
|
||
selectPosBtn.disabled = false;
|
||
// Only enable resize button if image is also loaded
|
||
if (state.imageLoaded) {
|
||
resizeBtn.disabled = false;
|
||
}
|
||
}
|
||
|
||
try {
|
||
updateUI('loadingImage', 'default');
|
||
const imageSrc = await Utils.createImageUploader();
|
||
if (!imageSrc) {
|
||
updateUI('colorsFound', 'success', {
|
||
count: state.availableColors.length,
|
||
});
|
||
return;
|
||
}
|
||
|
||
const processor = new ImageProcessor(imageSrc);
|
||
await processor.load();
|
||
|
||
const { width, height } = processor.getDimensions();
|
||
const pixels = processor.getPixelData();
|
||
|
||
let totalValidPixels = 0;
|
||
for (let i = 0; i < pixels.length; i += 4) {
|
||
const isTransparent =
|
||
!state.paintTransparentPixels &&
|
||
pixels[i + 3] < (state.customTransparencyThreshold || CONFIG.TRANSPARENCY_THRESHOLD);
|
||
const isWhiteAndSkipped =
|
||
!state.paintWhitePixels &&
|
||
Utils.isWhitePixel(pixels[i], pixels[i + 1], pixels[i + 2]);
|
||
if (!isTransparent && !isWhiteAndSkipped) {
|
||
totalValidPixels++;
|
||
}
|
||
}
|
||
|
||
state.imageData = {
|
||
width,
|
||
height,
|
||
pixels,
|
||
totalPixels: totalValidPixels,
|
||
processor,
|
||
};
|
||
|
||
state.totalPixels = totalValidPixels;
|
||
state.paintedPixels = 0;
|
||
state.imageLoaded = true;
|
||
state.lastPosition = { x: 0, y: 0 };
|
||
|
||
// Initialize painted map for tracking
|
||
Utils.initializePaintedMap(width, height);
|
||
|
||
// New image: clear previous resize settings
|
||
state.resizeSettings = null;
|
||
// Also clear any previous ignore mask
|
||
state.resizeIgnoreMask = null;
|
||
// Save original image for this browser (dataUrl + dims)
|
||
state.originalImage = { dataUrl: imageSrc, width, height };
|
||
saveBotSettings();
|
||
|
||
// Use the original image for the overlay initially
|
||
const imageBitmap = await createImageBitmap(processor.img);
|
||
await overlayManager.setImage(imageBitmap);
|
||
overlayManager.enable();
|
||
toggleOverlayBtn.disabled = false;
|
||
toggleOverlayBtn.classList.add('active');
|
||
toggleOverlayBtn.setAttribute('aria-pressed', 'true');
|
||
|
||
// Only enable resize button if colors have also been captured
|
||
if (state.colorsChecked) {
|
||
resizeBtn.disabled = false;
|
||
}
|
||
saveBtn.disabled = false;
|
||
|
||
if (state.startPosition) {
|
||
startBtn.disabled = false;
|
||
}
|
||
|
||
updateStats();
|
||
updateDataButtons();
|
||
updateUI('imageLoaded', 'success', { count: totalValidPixels });
|
||
} catch {
|
||
updateUI('imageError', 'error');
|
||
}
|
||
});
|
||
}
|
||
|
||
if (resizeBtn) {
|
||
resizeBtn.addEventListener('click', () => {
|
||
if (state.imageLoaded && state.imageData.processor && state.colorsChecked) {
|
||
showResizeDialog(state.imageData.processor);
|
||
} else if (!state.colorsChecked) {
|
||
Utils.showAlert(Utils.t('uploadImageFirstColors'), 'warning');
|
||
}
|
||
});
|
||
}
|
||
|
||
if (selectPosBtn) {
|
||
selectPosBtn.addEventListener('click', async () => {
|
||
if (state.selectingPosition) return;
|
||
|
||
state.selectingPosition = true;
|
||
state.startPosition = null;
|
||
state.region = null;
|
||
startBtn.disabled = true;
|
||
|
||
Utils.showAlert(Utils.t('selectPositionAlert'), 'info');
|
||
updateUI('waitingPosition', 'default');
|
||
|
||
const tempFetch = async (url, options) => {
|
||
if (
|
||
typeof url === 'string' &&
|
||
url.includes('https://backend.wplace.live/s0/pixel/') &&
|
||
options?.method?.toUpperCase() === 'POST'
|
||
) {
|
||
try {
|
||
const response = await originalFetch(url, options);
|
||
const clonedResponse = response.clone();
|
||
const data = await clonedResponse.json();
|
||
|
||
if (data?.painted === 1) {
|
||
const regionMatch = url.match(/\/pixel\/(\d+)\/(\d+)/);
|
||
if (regionMatch && regionMatch.length >= 3) {
|
||
state.region = {
|
||
x: Number.parseInt(regionMatch[1]),
|
||
y: Number.parseInt(regionMatch[2]),
|
||
};
|
||
}
|
||
|
||
const payload = JSON.parse(options.body);
|
||
if (payload?.coords && Array.isArray(payload.coords)) {
|
||
state.startPosition = {
|
||
x: payload.coords[0],
|
||
y: payload.coords[1],
|
||
};
|
||
state.lastPosition = { x: 0, y: 0 };
|
||
|
||
await overlayManager.setPosition(state.startPosition, state.region);
|
||
|
||
if (state.imageLoaded) {
|
||
startBtn.disabled = false;
|
||
}
|
||
|
||
window.fetch = originalFetch;
|
||
state.selectingPosition = false;
|
||
updateUI('positionSet', 'success');
|
||
}
|
||
}
|
||
|
||
return response;
|
||
} catch {
|
||
return originalFetch(url, options);
|
||
}
|
||
}
|
||
return originalFetch(url, options);
|
||
};
|
||
|
||
const originalFetch = window.fetch;
|
||
window.fetch = tempFetch;
|
||
|
||
setTimeout(() => {
|
||
if (state.selectingPosition) {
|
||
window.fetch = originalFetch;
|
||
state.selectingPosition = false;
|
||
updateUI('positionTimeout', 'error');
|
||
Utils.showAlert(Utils.t('positionTimeout'), 'error');
|
||
}
|
||
}, 120000);
|
||
});
|
||
}
|
||
|
||
async function startPainting() {
|
||
if (!state.imageLoaded || !state.startPosition || !state.region) {
|
||
updateUI('missingRequirements', 'error');
|
||
return;
|
||
}
|
||
await ensureToken();
|
||
if (!turnstileToken) return;
|
||
|
||
state.running = true;
|
||
state.stopFlag = false;
|
||
startBtn.disabled = true;
|
||
stopBtn.disabled = false;
|
||
uploadBtn.disabled = true;
|
||
selectPosBtn.disabled = true;
|
||
resizeBtn.disabled = true;
|
||
saveBtn.disabled = true;
|
||
toggleOverlayBtn.disabled = true;
|
||
|
||
updateUI('startPaintingMsg', 'success');
|
||
|
||
try {
|
||
await processImage();
|
||
} catch (e) {
|
||
console.error('Unexpected error:', e);
|
||
updateUI('paintingError', 'error');
|
||
} finally {
|
||
state.running = false;
|
||
stopBtn.disabled = true;
|
||
saveBtn.disabled = false;
|
||
|
||
if (state.stopFlag) {
|
||
startBtn.disabled = false;
|
||
} else {
|
||
startBtn.disabled = true;
|
||
uploadBtn.disabled = false;
|
||
selectPosBtn.disabled = false;
|
||
resizeBtn.disabled = false;
|
||
}
|
||
toggleOverlayBtn.disabled = false;
|
||
}
|
||
}
|
||
|
||
if (startBtn) {
|
||
startBtn.addEventListener('click', startPainting);
|
||
}
|
||
|
||
if (stopBtn) {
|
||
stopBtn.addEventListener('click', () => {
|
||
state.stopFlag = true;
|
||
state.running = false;
|
||
stopBtn.disabled = true;
|
||
updateUI('paintingStoppedByUser', 'warning');
|
||
|
||
if (state.imageLoaded && state.paintedPixels > 0) {
|
||
Utils.saveProgress();
|
||
Utils.showAlert(Utils.t('autoSaved'), 'success');
|
||
}
|
||
});
|
||
}
|
||
|
||
const checkSavedProgress = () => {
|
||
const savedData = Utils.loadProgress();
|
||
if (savedData && savedData.state.paintedPixels > 0) {
|
||
const savedDate = new Date(savedData.timestamp).toLocaleString();
|
||
const progress = Math.round(
|
||
(savedData.state.paintedPixels / savedData.state.totalPixels) * 100
|
||
);
|
||
|
||
Utils.showAlert(
|
||
`${Utils.t('savedDataFound')}\n\n` +
|
||
`Saved: ${savedDate}\n` +
|
||
`Progress: ${savedData.state.paintedPixels}/${savedData.state.totalPixels} pixels (${progress}%)\n` +
|
||
`${Utils.t('clickLoadToContinue')}`,
|
||
'info'
|
||
);
|
||
}
|
||
};
|
||
|
||
setTimeout(checkSavedProgress, 1000);
|
||
|
||
if (cooldownSlider && cooldownInput && cooldownValue && cooldownDecrease && cooldownIncrease) {
|
||
const updateCooldown = (newValue) => {
|
||
const threshold = Math.max(1, Math.min(state.maxCharges || 999, parseInt(newValue)));
|
||
state.cooldownChargeThreshold = threshold;
|
||
|
||
// Update both controls (value shows in input, label shows unit only)
|
||
cooldownSlider.value = threshold;
|
||
cooldownInput.value = threshold;
|
||
cooldownValue.textContent = `${Utils.t('charges')}`;
|
||
|
||
saveBotSettings();
|
||
NotificationManager.resetEdgeTracking(); // prevent spurious notify after threshold change
|
||
};
|
||
|
||
// Slider event listener
|
||
cooldownSlider.addEventListener('input', (e) => {
|
||
updateCooldown(e.target.value);
|
||
});
|
||
|
||
// Number input event listener
|
||
cooldownInput.addEventListener('input', (e) => {
|
||
updateCooldown(e.target.value);
|
||
});
|
||
|
||
// Decrease button
|
||
cooldownDecrease.addEventListener('click', () => {
|
||
updateCooldown(parseInt(cooldownInput.value) - 1);
|
||
});
|
||
|
||
// Increase button
|
||
cooldownIncrease.addEventListener('click', () => {
|
||
updateCooldown(parseInt(cooldownInput.value) + 1);
|
||
});
|
||
|
||
// Add scroll-to-adjust for cooldown slider
|
||
Utils.createScrollToAdjust(cooldownSlider, updateCooldown, 1, state.maxCharges, 1);
|
||
}
|
||
|
||
loadBotSettings();
|
||
// Ensure notification poller reflects current settings
|
||
NotificationManager.syncFromState();
|
||
}
|
||
|
||
function getMsToTargetCharges(current, target, cooldown, intervalMs = 0) {
|
||
const remainingCharges = target - current;
|
||
return Math.max(0, remainingCharges * cooldown - intervalMs);
|
||
}
|
||
|
||
function updateChargesThresholdUI(intervalMs) {
|
||
if (state.stopFlag) return;
|
||
|
||
const threshold = state.cooldownChargeThreshold;
|
||
const remainingMs = getMsToTargetCharges(
|
||
state.preciseCurrentCharges,
|
||
threshold,
|
||
state.cooldown,
|
||
intervalMs
|
||
);
|
||
const timeText = Utils.msToTimeText(remainingMs);
|
||
|
||
updateUI(
|
||
'noChargesThreshold',
|
||
'warning',
|
||
{
|
||
threshold,
|
||
current: state.displayCharges,
|
||
time: timeText,
|
||
},
|
||
true
|
||
);
|
||
}
|
||
|
||
function generateCoordinates(width, height, mode, direction, snake, blockWidth, blockHeight) {
|
||
const coords = [];
|
||
console.log(
|
||
'Generating coordinates with \n mode:',
|
||
mode,
|
||
'\n direction:',
|
||
direction,
|
||
'\n snake:',
|
||
snake,
|
||
'\n blockWidth:',
|
||
blockWidth,
|
||
'\n blockHeight:',
|
||
blockHeight
|
||
);
|
||
// --------- Standard 4 corners traversal ----------
|
||
let xStart, xEnd, xStep;
|
||
let yStart, yEnd, yStep;
|
||
switch (direction) {
|
||
case 'top-left':
|
||
xStart = 0;
|
||
xEnd = width;
|
||
xStep = 1;
|
||
yStart = 0;
|
||
yEnd = height;
|
||
yStep = 1;
|
||
break;
|
||
case 'top-right':
|
||
xStart = width - 1;
|
||
xEnd = -1;
|
||
xStep = -1;
|
||
yStart = 0;
|
||
yEnd = height;
|
||
yStep = 1;
|
||
break;
|
||
case 'bottom-left':
|
||
xStart = 0;
|
||
xEnd = width;
|
||
xStep = 1;
|
||
yStart = height - 1;
|
||
yEnd = -1;
|
||
yStep = -1;
|
||
break;
|
||
case 'bottom-right':
|
||
xStart = width - 1;
|
||
xEnd = -1;
|
||
xStep = -1;
|
||
yStart = height - 1;
|
||
yEnd = -1;
|
||
yStep = -1;
|
||
break;
|
||
default:
|
||
throw new Error(`Unknown direction: ${direction}`);
|
||
}
|
||
|
||
// --------- Traversal modes ----------
|
||
if (mode === 'rows') {
|
||
for (let y = yStart; y !== yEnd; y += yStep) {
|
||
if (snake && (y - yStart) % 2 !== 0) {
|
||
for (let x = xEnd - xStep; x !== xStart - xStep; x -= xStep) {
|
||
coords.push([x, y]);
|
||
}
|
||
} else {
|
||
for (let x = xStart; x !== xEnd; x += xStep) {
|
||
coords.push([x, y]);
|
||
}
|
||
}
|
||
}
|
||
} else if (mode === 'columns') {
|
||
for (let x = xStart; x !== xEnd; x += xStep) {
|
||
if (snake && (x - xStart) % 2 !== 0) {
|
||
for (let y = yEnd - yStep; y !== yStart - yStep; y -= yStep) {
|
||
coords.push([x, y]);
|
||
}
|
||
} else {
|
||
for (let y = yStart; y !== yEnd; y += yStep) {
|
||
coords.push([x, y]);
|
||
}
|
||
}
|
||
}
|
||
} else if (mode === 'circle-out') {
|
||
const cx = Math.floor(width / 2);
|
||
const cy = Math.floor(height / 2);
|
||
const maxRadius = Math.ceil(Math.sqrt(cx * cx + cy * cy));
|
||
|
||
for (let r = 0; r <= maxRadius; r++) {
|
||
for (let y = cy - r; y <= cy + r; y++) {
|
||
for (let x = cx - r; x <= cx + r; x++) {
|
||
if (x >= 0 && x < width && y >= 0 && y < height) {
|
||
const dist = Math.max(Math.abs(x - cx), Math.abs(y - cy));
|
||
if (dist === r) coords.push([x, y]);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
} else if (mode === 'circle-in') {
|
||
const cx = Math.floor(width / 2);
|
||
const cy = Math.floor(height / 2);
|
||
const maxRadius = Math.ceil(Math.sqrt(cx * cx + cy * cy));
|
||
|
||
for (let r = maxRadius; r >= 0; r--) {
|
||
for (let y = cy - r; y <= cy + r; y++) {
|
||
for (let x = cx - r; x <= cx + r; x++) {
|
||
if (x >= 0 && x < width && y >= 0 && y < height) {
|
||
const dist = Math.max(Math.abs(x - cx), Math.abs(y - cy));
|
||
if (dist === r) coords.push([x, y]);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
} else if (mode === 'blocks' || mode === 'shuffle-blocks') {
|
||
const blocks = [];
|
||
for (let by = 0; by < height; by += blockHeight) {
|
||
for (let bx = 0; bx < width; bx += blockWidth) {
|
||
const block = [];
|
||
for (let y = by; y < Math.min(by + blockHeight, height); y++) {
|
||
for (let x = bx; x < Math.min(bx + blockWidth, width); x++) {
|
||
block.push([x, y]);
|
||
}
|
||
}
|
||
blocks.push(block);
|
||
}
|
||
}
|
||
|
||
if (mode === 'shuffle-blocks') {
|
||
// Simple Fisher-Yates shuffle
|
||
for (let i = blocks.length - 1; i > 0; i--) {
|
||
const j = Math.floor(Math.random() * (i + 1));
|
||
[blocks[i], blocks[j]] = [blocks[j], blocks[i]];
|
||
}
|
||
}
|
||
|
||
// Concatenate all blocks
|
||
for (const block of blocks) {
|
||
coords.push(...block);
|
||
}
|
||
} else {
|
||
throw new Error(`Unknown mode: ${mode}`);
|
||
}
|
||
|
||
return coords;
|
||
}
|
||
|
||
async function flushPixelBatch(pixelBatch) {
|
||
if (!pixelBatch || pixelBatch.pixels.length === 0) return true;
|
||
|
||
const batchSize = pixelBatch.pixels.length;
|
||
console.log(
|
||
`📦 Sending batch with ${batchSize} pixels (region: ${pixelBatch.regionX},${pixelBatch.regionY})`
|
||
);
|
||
const success = await sendBatchWithRetry(
|
||
pixelBatch.pixels,
|
||
pixelBatch.regionX,
|
||
pixelBatch.regionY
|
||
);
|
||
if (success) {
|
||
pixelBatch.pixels.forEach((p) => {
|
||
state.paintedPixels++;
|
||
Utils.markPixelPainted(p.x, p.y, pixelBatch.regionX, pixelBatch.regionY);
|
||
});
|
||
state.fullChargeData = {
|
||
...state.fullChargeData,
|
||
spentSinceShot: state.fullChargeData.spentSinceShot + batchSize,
|
||
};
|
||
updateStats();
|
||
updateUI('paintingProgress', 'default', {
|
||
painted: state.paintedPixels,
|
||
total: state.totalPixels,
|
||
});
|
||
Utils.performSmartSave();
|
||
|
||
if (CONFIG.PAINTING_SPEED_ENABLED && state.paintingSpeed > 0 && batchSize > 0) {
|
||
const delayPerPixel = 1000 / state.paintingSpeed;
|
||
const totalDelay = Math.max(100, delayPerPixel * batchSize);
|
||
await Utils.sleep(totalDelay);
|
||
}
|
||
} else {
|
||
console.error(`❌ Batch failed permanently after retries. Stopping painting.`);
|
||
state.stopFlag = true;
|
||
updateUI('paintingBatchFailed', 'error');
|
||
}
|
||
|
||
pixelBatch.pixels = [];
|
||
return success;
|
||
}
|
||
|
||
async function processImage() {
|
||
const { width, height, pixels } = state.imageData;
|
||
const { x: startX, y: startY } = state.startPosition;
|
||
const { x: regionX, y: regionY } = state.region;
|
||
|
||
// todo force load tiles
|
||
const tilesReady = await overlayManager.waitForTiles(
|
||
regionX,
|
||
regionY,
|
||
width,
|
||
height,
|
||
startX,
|
||
startY,
|
||
10000 // timeout 10s
|
||
);
|
||
|
||
if (!tilesReady) {
|
||
updateUI('overlayTilesNotLoaded', 'error');
|
||
state.stopFlag = true;
|
||
return;
|
||
}
|
||
|
||
let pixelBatch = null;
|
||
let skippedPixels = {
|
||
transparent: 0,
|
||
white: 0,
|
||
alreadyPainted: 0,
|
||
colorUnavailable: 0,
|
||
};
|
||
|
||
const transparencyThreshold =
|
||
state.customTransparencyThreshold || CONFIG.TRANSPARENCY_THRESHOLD;
|
||
|
||
function checkPixelEligibility(x, y) {
|
||
const idx = (y * width + x) * 4;
|
||
const r = pixels[idx],
|
||
g = pixels[idx + 1],
|
||
b = pixels[idx + 2],
|
||
a = pixels[idx + 3];
|
||
|
||
if (!state.paintTransparentPixels && a < transparencyThreshold)
|
||
return {
|
||
eligible: false,
|
||
reason: 'transparent',
|
||
};
|
||
if (!state.paintWhitePixels && Utils.isWhitePixel(r, g, b))
|
||
return {
|
||
eligible: false,
|
||
reason: 'white',
|
||
};
|
||
|
||
let targetRgb = Utils.isWhitePixel(r, g, b)
|
||
? [255, 255, 255]
|
||
: Utils.findClosestPaletteColor(r, g, b, state.activeColorPalette);
|
||
|
||
// Template color ID, normalized/mapped to the nearest available color in our palette.
|
||
// Example: template requires "Slate", but we only have "Dark Gray" available
|
||
// → mappedTargetColorId = ID of Dark Gray.
|
||
//
|
||
// If `state.paintUnavailablePixels` is enabled, the painting would stop earlier
|
||
// because "Slate" was not found (null returned).
|
||
//
|
||
// Else, the template "Slate" is mapped to the closest available color (e.g., "Dark Gray"),
|
||
// and we proceed with painting using that mapped color.
|
||
//
|
||
// In this case, if the canvas pixel is already Slate (mapped to available Dark Gray),
|
||
// we skip painting, since template and canvas both resolve to the same available color (Dark Gray).
|
||
const mappedTargetColorId = Utils.resolveColor(
|
||
targetRgb,
|
||
state.availableColors,
|
||
!state.paintUnavailablePixels
|
||
);
|
||
|
||
// Technically, checking only `!mappedTargetColorId.id` would be enough,
|
||
// but combined with `state.paintUnavailablePixels` it makes the logic explicit:
|
||
// we only skip when the template color cannot be mapped AND strict mode is on.
|
||
if (!state.paintUnavailablePixels && !mappedTargetColorId.id) {
|
||
return {
|
||
eligible: false,
|
||
reason: 'colorUnavailable',
|
||
r,
|
||
g,
|
||
b,
|
||
a,
|
||
mappedColorId: mappedTargetColorId.id,
|
||
};
|
||
}
|
||
return { eligible: true, r, g, b, a, mappedColorId: mappedTargetColorId.id };
|
||
}
|
||
|
||
function skipPixel(reason, id, rgb, x, y) {
|
||
if (reason !== 'transparent') {
|
||
console.log(`Skipped pixel for ${reason} (id: ${id}, (${rgb.join(', ')})) at (${x}, ${y})`);
|
||
}
|
||
skippedPixels[reason]++;
|
||
}
|
||
|
||
try {
|
||
const coords = generateCoordinates(
|
||
width,
|
||
height,
|
||
state.coordinateMode,
|
||
state.coordinateDirection,
|
||
state.coordinateSnake,
|
||
state.blockWidth,
|
||
state.blockHeight
|
||
);
|
||
|
||
outerLoop: for (const [x, y] of coords) {
|
||
if (state.stopFlag) {
|
||
if (pixelBatch && pixelBatch.pixels.length > 0) {
|
||
console.log(
|
||
`🎯 Sending last batch before stop with ${pixelBatch.pixels.length} pixels`
|
||
);
|
||
await flushPixelBatch(pixelBatch);
|
||
}
|
||
state.lastPosition = { x, y };
|
||
updateUI('paintingPaused', 'warning', { x, y });
|
||
// noinspection UnnecessaryLabelOnBreakStatementJS
|
||
break outerLoop;
|
||
}
|
||
|
||
const targetPixelInfo = checkPixelEligibility(x, y);
|
||
let absX = startX + x;
|
||
let absY = startY + y;
|
||
|
||
let adderX = Math.floor(absX / 1000);
|
||
let adderY = Math.floor(absY / 1000);
|
||
let pixelX = absX % 1000;
|
||
let pixelY = absY % 1000;
|
||
|
||
// Template color ID, normalized/mapped to the nearest available color in our palette.
|
||
// Example: template requires "Slate", but we only have "Dark Gray" available
|
||
// → mappedTargetColorId = ID of Dark Gray.
|
||
//
|
||
// If `state.paintUnavailablePixels` is enabled, the painting would stop earlier
|
||
// because "Slate" was not found (null returned).
|
||
//
|
||
// Else, the template "Slate" is mapped to the closest available color (e.g., "Dark Gray"),
|
||
// and we proceed with painting using that mapped color.
|
||
//
|
||
// In this case, if the canvas pixel is already Slate (mapped to available Dark Gray),
|
||
// we skip painting, since template and canvas both resolve to the same available color (Dark Gray).
|
||
const targetMappedColorId = targetPixelInfo.mappedColorId;
|
||
|
||
if (!targetPixelInfo.eligible) {
|
||
skipPixel(
|
||
targetPixelInfo.reason,
|
||
targetMappedColorId,
|
||
[targetPixelInfo.r, targetPixelInfo.g, targetPixelInfo.b],
|
||
pixelX,
|
||
pixelY
|
||
);
|
||
continue;
|
||
}
|
||
|
||
// console.log(`[DEBUG] Pixel at (${pixelX}, ${pixelY}) eligible: RGB=${targetPixelInfo.r}, ${targetPixelInfo.g}, ${targetPixelInfo.b},
|
||
// alpha=${targetPixelInfo.a}, mappedColorId=${targetMappedColorId}`);
|
||
|
||
if (
|
||
!pixelBatch ||
|
||
pixelBatch.regionX !== regionX + adderX ||
|
||
pixelBatch.regionY !== regionY + adderY
|
||
) {
|
||
if (pixelBatch && pixelBatch.pixels.length > 0) {
|
||
console.log(
|
||
`🌍 Sending region-change batch with ${pixelBatch.pixels.length} pixels (switching to region ${regionX + adderX
|
||
},${regionY + adderY})`
|
||
);
|
||
const success = await flushPixelBatch(pixelBatch);
|
||
|
||
if (success) {
|
||
if (
|
||
CONFIG.PAINTING_SPEED_ENABLED &&
|
||
state.paintingSpeed > 0 &&
|
||
pixelBatch.pixels.length > 0
|
||
) {
|
||
const batchDelayFactor = Math.max(1, 100 / state.paintingSpeed);
|
||
const totalDelay = Math.max(100, batchDelayFactor * pixelBatch.pixels.length);
|
||
await Utils.sleep(totalDelay);
|
||
}
|
||
updateStats();
|
||
} else {
|
||
console.error(`❌ Batch failed permanently after retries. Stopping painting.`);
|
||
state.stopFlag = true;
|
||
updateUI('paintingBatchFailed', 'error');
|
||
// noinspection UnnecessaryLabelOnBreakStatementJS
|
||
break outerLoop;
|
||
}
|
||
}
|
||
|
||
pixelBatch = {
|
||
regionX: regionX + adderX,
|
||
regionY: regionY + adderY,
|
||
pixels: [],
|
||
};
|
||
}
|
||
|
||
try {
|
||
const tileKeyParts = [pixelBatch.regionX, pixelBatch.regionY];
|
||
|
||
const tilePixelRGBA = await overlayManager.getTilePixelColor(
|
||
tileKeyParts[0],
|
||
tileKeyParts[1],
|
||
pixelX,
|
||
pixelY
|
||
);
|
||
|
||
if (tilePixelRGBA && Array.isArray(tilePixelRGBA)) {
|
||
// Resolve the actual canvas pixel color to the closest available color.
|
||
// (The raw canvas RGB [er, eg, eb] is mapped into state.availableColors)
|
||
// so that comparison is consistent with targetMappedColorId.
|
||
const mappedCanvasColor = Utils.resolveColor(
|
||
tilePixelRGBA.slice(0, 3),
|
||
state.availableColors
|
||
);
|
||
const isMatch = mappedCanvasColor.id === targetMappedColorId;
|
||
if (isMatch) {
|
||
skipPixel(
|
||
'alreadyPainted',
|
||
targetMappedColorId,
|
||
[targetPixelInfo.r, targetPixelInfo.g, targetPixelInfo.b],
|
||
pixelX,
|
||
pixelY
|
||
);
|
||
continue;
|
||
}
|
||
console.debug(
|
||
`[COMPARE] Pixel at 📍 (${pixelX}, ${pixelY}) in region (${regionX + adderX
|
||
}, ${regionY + adderY})\n` +
|
||
` ├── Current color: rgb(${tilePixelRGBA.slice(0, 3).join(', ')}) (id: ${mappedCanvasColor.id})\n` +
|
||
` ├── Target color: rgb(${targetPixelInfo.r}, ${targetPixelInfo.g}, ${targetPixelInfo.b}) (id: ${targetMappedColorId})\n` +
|
||
` └── Status: ${isMatch ? '✅ Already painted → SKIP' : '🔴 Needs paint → PAINT'
|
||
}\n`
|
||
);
|
||
}
|
||
} catch (e) {
|
||
console.error(`[DEBUG] Error checking existing pixel at (${pixelX}, ${pixelY}):`, e);
|
||
updateUI('paintingPixelCheckFailed', 'error', { x: pixelX, y: pixelY });
|
||
state.stopFlag = true;
|
||
// noinspection UnnecessaryLabelOnBreakStatementJS
|
||
break outerLoop;
|
||
}
|
||
|
||
pixelBatch.pixels.push({
|
||
x: pixelX,
|
||
y: pixelY,
|
||
color: targetMappedColorId,
|
||
localX: x,
|
||
localY: y,
|
||
});
|
||
|
||
const maxBatchSize = calculateBatchSize();
|
||
if (pixelBatch.pixels.length >= maxBatchSize) {
|
||
const modeText =
|
||
state.batchMode === 'random'
|
||
? `random (${state.randomBatchMin}-${state.randomBatchMax})`
|
||
: 'normal';
|
||
console.log(
|
||
`📦 Sending batch with ${pixelBatch.pixels.length} pixels (mode: ${modeText}, target: ${maxBatchSize})`
|
||
);
|
||
const success = await flushPixelBatch(pixelBatch);
|
||
if (!success) {
|
||
console.error(`❌ Batch failed permanently after retries. Stopping painting.`);
|
||
state.stopFlag = true;
|
||
updateUI('paintingBatchFailed', 'error');
|
||
// noinspection UnnecessaryLabelOnBreakStatementJS
|
||
break outerLoop;
|
||
}
|
||
|
||
pixelBatch.pixels = [];
|
||
}
|
||
|
||
if (state.displayCharges < state.cooldownChargeThreshold && !state.stopFlag) {
|
||
await Utils.dynamicSleep(() => {
|
||
if (state.displayCharges >= state.cooldownChargeThreshold) {
|
||
NotificationManager.maybeNotifyChargesReached(true);
|
||
return 0;
|
||
}
|
||
if (state.stopFlag) return 0;
|
||
return getMsToTargetCharges(
|
||
state.preciseCurrentCharges,
|
||
state.cooldownChargeThreshold,
|
||
state.cooldown
|
||
);
|
||
});
|
||
}
|
||
|
||
if (state.stopFlag) {
|
||
// noinspection UnnecessaryLabelOnBreakStatementJS
|
||
break outerLoop;
|
||
}
|
||
}
|
||
|
||
if (pixelBatch && pixelBatch.pixels.length > 0 && !state.stopFlag) {
|
||
console.log(`🏁 Sending final batch with ${pixelBatch.pixels.length} pixels`);
|
||
const success = await flushPixelBatch(pixelBatch);
|
||
if (!success) {
|
||
console.warn(
|
||
`⚠️ Final batch failed with ${pixelBatch.pixels.length} pixels after all retries.`
|
||
);
|
||
}
|
||
}
|
||
} finally {
|
||
if (window._chargesInterval) clearInterval(window._chargesInterval);
|
||
window._chargesInterval = null;
|
||
}
|
||
|
||
if (state.stopFlag) {
|
||
// Save progress when stopped to preserve painted map
|
||
Utils.saveProgress();
|
||
} else {
|
||
updateUI('paintingComplete', 'success', { count: state.paintedPixels });
|
||
state.lastPosition = { x: 0, y: 0 };
|
||
// Keep painted map until user starts new project
|
||
// state.paintedMap = null // Commented out to preserve data
|
||
Utils.saveProgress(); // Save final complete state
|
||
overlayManager.clear();
|
||
const toggleOverlayBtn = document.getElementById('toggleOverlayBtn');
|
||
if (toggleOverlayBtn) {
|
||
toggleOverlayBtn.classList.remove('active');
|
||
toggleOverlayBtn.disabled = true;
|
||
}
|
||
}
|
||
|
||
// Log skip statistics
|
||
console.log(`📊 Pixel Statistics:`);
|
||
console.log(` Painted: ${state.paintedPixels}`);
|
||
console.log(` Skipped - Transparent: ${skippedPixels.transparent}`);
|
||
console.log(` Skipped - White (disabled): ${skippedPixels.white}`);
|
||
console.log(` Skipped - Already painted: ${skippedPixels.alreadyPainted}`);
|
||
console.log(` Skipped - Color Unavailable: ${skippedPixels.colorUnavailable}`);
|
||
console.log(
|
||
` Total processed: ${state.paintedPixels +
|
||
skippedPixels.transparent +
|
||
skippedPixels.white +
|
||
skippedPixels.alreadyPainted +
|
||
skippedPixels.colorUnavailable
|
||
}`
|
||
);
|
||
|
||
updateStats();
|
||
}
|
||
|
||
// Helper function to calculate batch size based on mode
|
||
function calculateBatchSize() {
|
||
let targetBatchSize;
|
||
|
||
if (state.batchMode === 'random') {
|
||
// Generate random batch size within the specified range
|
||
const min = Math.max(1, state.randomBatchMin);
|
||
const max = Math.max(min, state.randomBatchMax);
|
||
targetBatchSize = Math.floor(Math.random() * (max - min + 1)) + min;
|
||
console.log(`🎲 Random batch size generated: ${targetBatchSize} (range: ${min}-${max})`);
|
||
} else {
|
||
// Normal mode - use the fixed paintingSpeed value
|
||
targetBatchSize = state.paintingSpeed;
|
||
}
|
||
|
||
// Always limit by available charges
|
||
const maxAllowed = state.displayCharges;
|
||
const finalBatchSize = Math.min(targetBatchSize, maxAllowed);
|
||
|
||
return finalBatchSize;
|
||
}
|
||
|
||
// Helper function to retry batch until success with exponential backoff
|
||
async function sendBatchWithRetry(pixels, regionX, regionY, maxRetries = MAX_BATCH_RETRIES) {
|
||
let attempt = 0;
|
||
while (attempt < maxRetries && !state.stopFlag) {
|
||
attempt++;
|
||
console.log(
|
||
`🔄 Attempting to send batch (attempt ${attempt}/${maxRetries}) for region ${regionX},${regionY} with ${pixels.length} pixels`
|
||
);
|
||
|
||
const result = await sendPixelBatch(pixels, regionX, regionY);
|
||
|
||
if (result === true) {
|
||
console.log(`✅ Batch succeeded on attempt ${attempt}`);
|
||
return true;
|
||
} else if (result === 'token_error') {
|
||
console.log(`🔑 Token error on attempt ${attempt}, regenerating...`);
|
||
updateUI('captchaSolving', 'warning');
|
||
try {
|
||
await handleCaptcha();
|
||
// Don't count token regeneration as a failed attempt
|
||
attempt--;
|
||
continue;
|
||
} catch (e) {
|
||
console.error(`❌ Token regeneration failed on attempt ${attempt}:`, e);
|
||
updateUI('captchaFailed', 'error');
|
||
// Wait longer before retrying after token failure
|
||
await Utils.sleep(5000);
|
||
}
|
||
} else {
|
||
console.warn(`⚠️ Batch failed on attempt ${attempt}, retrying...`);
|
||
// Exponential backoff with jitter
|
||
const baseDelay = Math.min(1000 * Math.pow(2, attempt - 1), 30000); // Max 30s
|
||
const jitter = Math.random() * 1000; // Add up to 1s random delay
|
||
await Utils.sleep(baseDelay + jitter);
|
||
}
|
||
}
|
||
|
||
if (attempt >= maxRetries) {
|
||
console.error(
|
||
`❌ Batch failed after ${maxRetries} attempts (MAX_BATCH_RETRIES=${MAX_BATCH_RETRIES}). This will stop painting to prevent infinite loops.`
|
||
);
|
||
updateUI('paintingError', 'error');
|
||
return false;
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
async function sendPixelBatch(pixelBatch, regionX, regionY) {
|
||
let token = turnstileToken;
|
||
|
||
// Generate new token if we don't have one
|
||
if (!token) {
|
||
try {
|
||
console.log('🔑 Generating Turnstile token for pixel batch...');
|
||
token = await handleCaptcha();
|
||
turnstileToken = token; // Store for potential reuse
|
||
} catch (error) {
|
||
console.error('❌ Failed to generate Turnstile token:', error);
|
||
tokenPromise = new Promise((resolve) => {
|
||
_resolveToken = resolve;
|
||
});
|
||
return 'token_error';
|
||
}
|
||
}
|
||
|
||
const coords = new Array(pixelBatch.length * 2);
|
||
const colors = new Array(pixelBatch.length);
|
||
for (let i = 0; i < pixelBatch.length; i++) {
|
||
const pixel = pixelBatch[i];
|
||
coords[i * 2] = pixel.x;
|
||
coords[i * 2 + 1] = pixel.y;
|
||
colors[i] = pixel.color;
|
||
}
|
||
|
||
try {
|
||
const payload = { coords, colors, t: token, fp: fpStr32 };
|
||
var wasmtoken = await createWasmToken(regionX, regionY, payload);
|
||
const res = await fetch(`https://backend.wplace.live/s0/pixel/${regionX}/${regionY}`, {
|
||
method: 'POST',
|
||
headers: { 'Content-Type': 'text/plain;charset=UTF-8', "x-pawtect-token": wasmtoken },
|
||
credentials: 'include',
|
||
body: JSON.stringify(payload),
|
||
});
|
||
|
||
if (res.status === 403) {
|
||
let data = null;
|
||
try {
|
||
data = await res.json();
|
||
} catch (_) { }
|
||
console.error('❌ 403 Forbidden. Turnstile token might be invalid or expired.');
|
||
|
||
// Try to generate a new token and retry once
|
||
try {
|
||
console.log('🔄 Regenerating Turnstile token after 403...');
|
||
token = await handleCaptcha();
|
||
turnstileToken = token;
|
||
|
||
// Retry the request with new token
|
||
const retryPayload = { coords, colors, t: token, fp: fpStr32 };
|
||
var wasmtoken = await createWasmToken(regionX, regionY, retryPayload);
|
||
const retryRes = await fetch(
|
||
`https://backend.wplace.live/s0/pixel/${regionX}/${regionY}`,
|
||
{
|
||
method: 'POST',
|
||
headers: { 'Content-Type': 'text/plain;charset=UTF-8', "x-pawtect-token": wasmtoken },
|
||
credentials: 'include',
|
||
body: JSON.stringify(retryPayload),
|
||
}
|
||
);
|
||
|
||
if (retryRes.status === 403) {
|
||
turnstileToken = null;
|
||
tokenPromise = new Promise((resolve) => {
|
||
_resolveToken = resolve;
|
||
});
|
||
return 'token_error';
|
||
}
|
||
|
||
const retryData = await retryRes.json();
|
||
return retryData?.painted === pixelBatch.length;
|
||
} catch (retryError) {
|
||
console.error('❌ Token regeneration failed:', retryError);
|
||
turnstileToken = null;
|
||
tokenPromise = new Promise((resolve) => {
|
||
_resolveToken = resolve;
|
||
});
|
||
return 'token_error';
|
||
}
|
||
}
|
||
|
||
const data = await res.json();
|
||
return data?.painted === pixelBatch.length;
|
||
} catch (e) {
|
||
console.error('Batch paint request failed:', e);
|
||
return false;
|
||
}
|
||
}
|
||
|
||
function saveBotSettings() {
|
||
try {
|
||
const settings = {
|
||
paintingSpeed: state.paintingSpeed,
|
||
paintingSpeedEnabled: document.getElementById('enableSpeedToggle')?.checked,
|
||
batchMode: state.batchMode, // "normal" or "random"
|
||
randomBatchMin: state.randomBatchMin,
|
||
randomBatchMax: state.randomBatchMax,
|
||
cooldownChargeThreshold: state.cooldownChargeThreshold,
|
||
tokenSource: state.tokenSource, // "generator", "hybrid", or "manual"
|
||
minimized: state.minimized,
|
||
overlayOpacity: state.overlayOpacity,
|
||
blueMarbleEnabled: document.getElementById('enableBlueMarbleToggle')?.checked,
|
||
ditheringEnabled: state.ditheringEnabled,
|
||
colorMatchingAlgorithm: state.colorMatchingAlgorithm,
|
||
enableChromaPenalty: state.enableChromaPenalty,
|
||
chromaPenaltyWeight: state.chromaPenaltyWeight,
|
||
customTransparencyThreshold: state.customTransparencyThreshold,
|
||
customWhiteThreshold: state.customWhiteThreshold,
|
||
paintWhitePixels: state.paintWhitePixels,
|
||
paintTransparentPixels: state.paintTransparentPixels,
|
||
resizeSettings: state.resizeSettings,
|
||
paintUnavailablePixels: state.paintUnavailablePixels,
|
||
coordinateMode: state.coordinateMode,
|
||
coordinateDirection: state.coordinateDirection,
|
||
coordinateSnake: state.coordinateSnake,
|
||
blockWidth: state.blockWidth,
|
||
blockHeight: state.blockHeight, // Save ignore mask (as base64) with its dimensions
|
||
resizeIgnoreMask:
|
||
state.resizeIgnoreMask &&
|
||
state.resizeSettings &&
|
||
state.resizeSettings.width * state.resizeSettings.height === state.resizeIgnoreMask.length
|
||
? {
|
||
w: state.resizeSettings.width,
|
||
h: state.resizeSettings.height,
|
||
data: btoa(String.fromCharCode(...state.resizeIgnoreMask)),
|
||
}
|
||
: null, // Notifications
|
||
notificationsEnabled: state.notificationsEnabled,
|
||
notifyOnChargesReached: state.notifyOnChargesReached,
|
||
notifyOnlyWhenUnfocused: state.notifyOnlyWhenUnfocused,
|
||
notificationIntervalMinutes: state.notificationIntervalMinutes,
|
||
originalImage: state.originalImage,
|
||
};
|
||
CONFIG.PAINTING_SPEED_ENABLED = settings.paintingSpeedEnabled;
|
||
// AUTO_CAPTCHA_ENABLED is always true - no need to save/load
|
||
|
||
localStorage.setItem('wplace-bot-settings', JSON.stringify(settings));
|
||
} catch (e) {
|
||
console.warn('Could not save bot settings:', e);
|
||
}
|
||
}
|
||
|
||
function loadBotSettings() {
|
||
try {
|
||
const saved = localStorage.getItem('wplace-bot-settings');
|
||
if (!saved) return;
|
||
const settings = JSON.parse(saved);
|
||
|
||
state.paintingSpeed = settings.paintingSpeed || CONFIG.PAINTING_SPEED.DEFAULT;
|
||
state.batchMode = settings.batchMode || CONFIG.BATCH_MODE; // Default to "normal"
|
||
state.randomBatchMin = settings.randomBatchMin || CONFIG.RANDOM_BATCH_RANGE.MIN;
|
||
state.randomBatchMax = settings.randomBatchMax || CONFIG.RANDOM_BATCH_RANGE.MAX;
|
||
state.cooldownChargeThreshold =
|
||
settings.cooldownChargeThreshold || CONFIG.COOLDOWN_CHARGE_THRESHOLD;
|
||
state.tokenSource = settings.tokenSource || CONFIG.TOKEN_SOURCE; // Default to "generator"
|
||
state.minimized = settings.minimized ?? false;
|
||
CONFIG.PAINTING_SPEED_ENABLED = settings.paintingSpeedEnabled ?? false;
|
||
CONFIG.AUTO_CAPTCHA_ENABLED = settings.autoCaptchaEnabled ?? false;
|
||
state.overlayOpacity = settings.overlayOpacity ?? CONFIG.OVERLAY.OPACITY_DEFAULT;
|
||
state.blueMarbleEnabled = settings.blueMarbleEnabled ?? CONFIG.OVERLAY.BLUE_MARBLE_DEFAULT;
|
||
state.ditheringEnabled = settings.ditheringEnabled ?? false;
|
||
state.colorMatchingAlgorithm = settings.colorMatchingAlgorithm || 'lab';
|
||
state.enableChromaPenalty = settings.enableChromaPenalty ?? true;
|
||
state.chromaPenaltyWeight = settings.chromaPenaltyWeight ?? 0.15;
|
||
state.customTransparencyThreshold =
|
||
settings.customTransparencyThreshold ?? CONFIG.TRANSPARENCY_THRESHOLD;
|
||
state.customWhiteThreshold = settings.customWhiteThreshold ?? CONFIG.WHITE_THRESHOLD;
|
||
state.paintWhitePixels = settings.paintWhitePixels ?? true;
|
||
state.paintTransparentPixels = settings.paintTransparentPixels ?? false;
|
||
state.resizeSettings = settings.resizeSettings ?? null;
|
||
state.originalImage = settings.originalImage ?? null;
|
||
state.paintUnavailablePixels = settings.paintUnavailablePixels ?? CONFIG.PAINT_UNAVAILABLE;
|
||
state.coordinateMode = settings.coordinateMode ?? CONFIG.COORDINATE_MODE;
|
||
state.coordinateDirection = settings.coordinateDirection ?? CONFIG.COORDINATE_DIRECTION;
|
||
state.coordinateSnake = settings.coordinateSnake ?? CONFIG.COORDINATE_SNAKE;
|
||
state.blockWidth = settings.blockWidth ?? CONFIG.COORDINATE_BLOCK_WIDTH;
|
||
state.blockHeight = settings.blockHeight ?? CONFIG.COORDINATE_BLOCK_HEIGHT;
|
||
// Notifications
|
||
state.notificationsEnabled = settings.notificationsEnabled ?? CONFIG.NOTIFICATIONS.ENABLED;
|
||
state.notifyOnChargesReached =
|
||
settings.notifyOnChargesReached ?? CONFIG.NOTIFICATIONS.ON_CHARGES_REACHED;
|
||
state.notifyOnlyWhenUnfocused =
|
||
settings.notifyOnlyWhenUnfocused ?? CONFIG.NOTIFICATIONS.ONLY_WHEN_UNFOCUSED;
|
||
state.notificationIntervalMinutes =
|
||
settings.notificationIntervalMinutes ?? CONFIG.NOTIFICATIONS.REPEAT_MINUTES;
|
||
// Restore ignore mask if dims match current resizeSettings
|
||
if (
|
||
settings.resizeIgnoreMask &&
|
||
settings.resizeIgnoreMask.data &&
|
||
state.resizeSettings &&
|
||
settings.resizeIgnoreMask.w === state.resizeSettings.width &&
|
||
settings.resizeIgnoreMask.h === state.resizeSettings.height
|
||
) {
|
||
try {
|
||
const bin = atob(settings.resizeIgnoreMask.data);
|
||
const arr = new Uint8Array(bin.length);
|
||
for (let i = 0; i < bin.length; i++) arr[i] = bin.charCodeAt(i);
|
||
state.resizeIgnoreMask = arr;
|
||
} catch {
|
||
state.resizeIgnoreMask = null;
|
||
}
|
||
} else {
|
||
state.resizeIgnoreMask = null;
|
||
}
|
||
// Initialize coordinate generation UI
|
||
const coordinateModeSelect = document.getElementById('coordinateModeSelect');
|
||
if (coordinateModeSelect) coordinateModeSelect.value = state.coordinateMode;
|
||
|
||
const coordinateDirectionSelect = document.getElementById('coordinateDirectionSelect');
|
||
if (coordinateDirectionSelect) coordinateDirectionSelect.value = state.coordinateDirection;
|
||
|
||
const coordinateSnakeToggle = document.getElementById('coordinateSnakeToggle');
|
||
if (coordinateSnakeToggle) coordinateSnakeToggle.checked = state.coordinateSnake;
|
||
|
||
const settingsContainer = document.getElementById('wplace-settings-container');
|
||
const directionControls = settingsContainer.querySelector('#directionControls');
|
||
const snakeControls = settingsContainer.querySelector('#snakeControls');
|
||
const blockControls = settingsContainer.querySelector('#blockControls');
|
||
Utils.updateCoordinateUI({
|
||
mode: state.coordinateMode,
|
||
directionControls,
|
||
snakeControls,
|
||
blockControls,
|
||
});
|
||
|
||
const paintUnavailablePixelsToggle = document.getElementById('paintUnavailablePixelsToggle');
|
||
if (paintUnavailablePixelsToggle) {
|
||
paintUnavailablePixelsToggle.checked = state.paintUnavailablePixels;
|
||
}
|
||
|
||
const settingsPaintWhiteToggle = settingsContainer.querySelector('#settingsPaintWhiteToggle');
|
||
if (settingsPaintWhiteToggle) {
|
||
settingsPaintWhiteToggle.checked = state.paintWhitePixels;
|
||
}
|
||
|
||
const settingsPaintTransparentToggle = settingsContainer.querySelector(
|
||
'#settingsPaintTransparentToggle'
|
||
);
|
||
if (settingsPaintTransparentToggle) {
|
||
settingsPaintTransparentToggle.checked = state.paintTransparentPixels;
|
||
}
|
||
|
||
const speedSlider = document.getElementById('speedSlider');
|
||
const speedInput = document.getElementById('speedInput');
|
||
const speedValue = document.getElementById('speedValue');
|
||
if (speedSlider) speedSlider.value = state.paintingSpeed;
|
||
if (speedInput) speedInput.value = state.paintingSpeed;
|
||
if (speedValue) speedValue.textContent = `pixels`;
|
||
|
||
const enableSpeedToggle = document.getElementById('enableSpeedToggle');
|
||
if (enableSpeedToggle) enableSpeedToggle.checked = CONFIG.PAINTING_SPEED_ENABLED;
|
||
|
||
// Batch mode UI initialization
|
||
const batchModeSelect = document.getElementById('batchModeSelect');
|
||
if (batchModeSelect) batchModeSelect.value = state.batchMode;
|
||
|
||
const normalBatchControls = document.getElementById('normalBatchControls');
|
||
const randomBatchControls = document.getElementById('randomBatchControls');
|
||
|
||
// Show/hide appropriate controls based on batch mode
|
||
if (normalBatchControls && randomBatchControls) {
|
||
if (state.batchMode === 'random') {
|
||
normalBatchControls.style.display = 'none';
|
||
randomBatchControls.style.display = 'block';
|
||
} else {
|
||
normalBatchControls.style.display = 'block';
|
||
randomBatchControls.style.display = 'none';
|
||
}
|
||
}
|
||
|
||
const randomBatchMin = document.getElementById('randomBatchMin');
|
||
if (randomBatchMin) randomBatchMin.value = state.randomBatchMin;
|
||
|
||
const randomBatchMax = document.getElementById('randomBatchMax');
|
||
if (randomBatchMax) randomBatchMax.value = state.randomBatchMax;
|
||
|
||
// AUTO_CAPTCHA_ENABLED is always true - no toggle to set
|
||
|
||
const cooldownSlider = document.getElementById('cooldownSlider');
|
||
const cooldownInput = document.getElementById('cooldownInput');
|
||
const cooldownValue = document.getElementById('cooldownValue');
|
||
if (cooldownSlider) cooldownSlider.value = state.cooldownChargeThreshold;
|
||
if (cooldownInput) cooldownInput.value = state.cooldownChargeThreshold;
|
||
if (cooldownValue) cooldownValue.textContent = `${Utils.t('charges')}`;
|
||
|
||
const overlayOpacitySlider = document.getElementById('overlayOpacitySlider');
|
||
if (overlayOpacitySlider) overlayOpacitySlider.value = state.overlayOpacity;
|
||
const overlayOpacityValue = document.getElementById('overlayOpacityValue');
|
||
if (overlayOpacityValue)
|
||
overlayOpacityValue.textContent = `${Math.round(state.overlayOpacity * 100)}%`;
|
||
const enableBlueMarbleToggle = document.getElementById('enableBlueMarbleToggle');
|
||
if (enableBlueMarbleToggle) enableBlueMarbleToggle.checked = state.blueMarbleEnabled;
|
||
|
||
const tokenSourceSelect = document.getElementById('tokenSourceSelect');
|
||
if (tokenSourceSelect) tokenSourceSelect.value = state.tokenSource;
|
||
|
||
const colorAlgorithmSelect = document.getElementById('colorAlgorithmSelect');
|
||
if (colorAlgorithmSelect) colorAlgorithmSelect.value = state.colorMatchingAlgorithm;
|
||
const enableChromaPenaltyToggle = document.getElementById('enableChromaPenaltyToggle');
|
||
if (enableChromaPenaltyToggle) enableChromaPenaltyToggle.checked = state.enableChromaPenalty;
|
||
const chromaPenaltyWeightSlider = document.getElementById('chromaPenaltyWeightSlider');
|
||
if (chromaPenaltyWeightSlider) chromaPenaltyWeightSlider.value = state.chromaPenaltyWeight;
|
||
const chromaWeightValue = document.getElementById('chromaWeightValue');
|
||
if (chromaWeightValue) chromaWeightValue.textContent = state.chromaPenaltyWeight;
|
||
const transparencyThresholdInput = document.getElementById('transparencyThresholdInput');
|
||
if (transparencyThresholdInput)
|
||
transparencyThresholdInput.value = state.customTransparencyThreshold;
|
||
const whiteThresholdInput = document.getElementById('whiteThresholdInput');
|
||
if (whiteThresholdInput) whiteThresholdInput.value = state.customWhiteThreshold;
|
||
// Notifications UI
|
||
const notifEnabledToggle = document.getElementById('notifEnabledToggle');
|
||
if (notifEnabledToggle) notifEnabledToggle.checked = state.notificationsEnabled;
|
||
const notifOnChargesToggle = document.getElementById('notifOnChargesToggle');
|
||
if (notifOnChargesToggle) notifOnChargesToggle.checked = state.notifyOnChargesReached;
|
||
const notifOnlyUnfocusedToggle = document.getElementById('notifOnlyUnfocusedToggle');
|
||
if (notifOnlyUnfocusedToggle)
|
||
notifOnlyUnfocusedToggle.checked = state.notifyOnlyWhenUnfocused;
|
||
const notifIntervalInput = document.getElementById('notifIntervalInput');
|
||
if (notifIntervalInput) notifIntervalInput.value = state.notificationIntervalMinutes;
|
||
NotificationManager.resetEdgeTracking();
|
||
} catch (e) {
|
||
console.warn('Could not load bot settings:', e);
|
||
}
|
||
}
|
||
|
||
// Initialize Turnstile generator integration
|
||
console.log('🚀 WPlace Auto-Image with Turnstile Token Generator loaded');
|
||
console.log('🔑 Turnstile token generator: ALWAYS ENABLED (Background mode)');
|
||
console.log('🎯 Manual pixel captcha solving: Available as fallback/alternative');
|
||
console.log('📱 Turnstile widgets: DISABLED - pure background token generation only!');
|
||
|
||
// Function to enable file operations after initial startup setup is complete
|
||
function enableFileOperations() {
|
||
state.initialSetupComplete = true;
|
||
|
||
const loadBtn = document.querySelector('#loadBtn');
|
||
const loadFromFileBtn = document.querySelector('#loadFromFileBtn');
|
||
const uploadBtn = document.querySelector('#uploadBtn');
|
||
|
||
if (loadBtn) {
|
||
loadBtn.disabled = false;
|
||
loadBtn.title = '';
|
||
// Add a subtle animation to indicate the button is now available
|
||
loadBtn.style.animation = 'pulse 0.6s ease-in-out';
|
||
setTimeout(() => {
|
||
if (loadBtn) loadBtn.style.animation = '';
|
||
}, 600);
|
||
console.log('✅ Load Progress button enabled after initial setup');
|
||
}
|
||
|
||
if (loadFromFileBtn) {
|
||
loadFromFileBtn.disabled = false;
|
||
loadFromFileBtn.title = '';
|
||
// Add a subtle animation to indicate the button is now available
|
||
loadFromFileBtn.style.animation = 'pulse 0.6s ease-in-out';
|
||
setTimeout(() => {
|
||
if (loadFromFileBtn) loadFromFileBtn.style.animation = '';
|
||
}, 600);
|
||
console.log('✅ Load from File button enabled after initial setup');
|
||
}
|
||
|
||
if (uploadBtn) {
|
||
uploadBtn.disabled = false;
|
||
uploadBtn.title = '';
|
||
// Add a subtle animation to indicate the button is now available
|
||
uploadBtn.style.animation = 'pulse 0.6s ease-in-out';
|
||
setTimeout(() => {
|
||
if (uploadBtn) uploadBtn.style.animation = '';
|
||
}, 600);
|
||
console.log('✅ Upload Image button enabled after initial setup');
|
||
}
|
||
|
||
// Show a notification that file operations are now available
|
||
Utils.showAlert(Utils.t('fileOperationsAvailable'), 'success');
|
||
}
|
||
|
||
// Optimized token initialization with better timing and error handling
|
||
async function initializeTokenGenerator() {
|
||
// Skip if already have valid token
|
||
if (isTokenValid()) {
|
||
console.log('✅ Valid token already available, skipping initialization');
|
||
updateUI('tokenReady', 'success');
|
||
enableFileOperations(); // Enable file operations since initial setup is complete
|
||
return;
|
||
}
|
||
|
||
try {
|
||
console.log('🔧 Initializing Turnstile token generator...');
|
||
updateUI('initializingToken', 'default');
|
||
|
||
console.log('Attempting to load Turnstile script...');
|
||
await Utils.loadTurnstile();
|
||
console.log('Turnstile script loaded. Attempting to generate token...');
|
||
|
||
const token = await handleCaptchaWithRetry();
|
||
if (token) {
|
||
setTurnstileToken(token);
|
||
console.log('✅ Startup token generated successfully');
|
||
updateUI('tokenReady', 'success');
|
||
Utils.showAlert(Utils.t('tokenGeneratorReady'), 'success');
|
||
enableFileOperations(); // Enable file operations since initial setup is complete
|
||
} else {
|
||
console.warn(
|
||
'⚠️ Startup token generation failed (no token received), will retry when needed'
|
||
);
|
||
updateUI('tokenRetryLater', 'warning');
|
||
// Still enable file operations even if initial token generation fails
|
||
// Users can load progress and use manual/hybrid modes
|
||
enableFileOperations();
|
||
}
|
||
} catch (error) {
|
||
console.error('❌ Critical error during Turnstile initialization:', error); // More specific error
|
||
updateUI('tokenRetryLater', 'warning');
|
||
// Still enable file operations even if initial setup fails
|
||
// Users can load progress and use manual/hybrid modes
|
||
enableFileOperations();
|
||
// Don't show error alert for initialization failures, just log them
|
||
}
|
||
}
|
||
|
||
// Load theme preference immediately on startup before creating UI
|
||
loadThemePreference();
|
||
applyTheme();
|
||
|
||
var pawtect_chunk = null;
|
||
|
||
//find module if pawtect_chunk is null
|
||
pawtect_chunk ??= await findTokenModule("pawtect_wasm_bg.wasm");
|
||
|
||
async function createWasmToken(regionX, regionY, payload) {
|
||
try {
|
||
// Load the Pawtect module and WASM
|
||
const mod = await import(new URL('/_app/immutable/chunks/'+pawtect_chunk, location.origin).href);
|
||
let wasm;
|
||
try {
|
||
wasm = await mod._();
|
||
console.log('✅ WASM initialized successfully');
|
||
} catch (wasmError) {
|
||
console.error('❌ WASM initialization failed:', wasmError);
|
||
return null;
|
||
}
|
||
try {
|
||
try {
|
||
const me = await fetch(`https://backend.wplace.live/me`, { credentials: 'include' }).then(r => r.ok ? r.json() : null);
|
||
if (me?.id) {
|
||
mod.i(me.id);
|
||
console.log('✅ user ID set:', me.id);
|
||
}
|
||
} catch { }
|
||
} catch (userIdError) {
|
||
console.log('⚠️ Error setting user ID:', userIdError.message);
|
||
}
|
||
try {
|
||
const testUrl = `https://backend.wplace.live/s0/pixel/${regionX}/${regionY}`;
|
||
if (mod.r) {
|
||
mod.r(testUrl);
|
||
console.log('✅ Request URL set:', testUrl);
|
||
} else {
|
||
console.log('⚠️ request_url function (mod.r) not available');
|
||
}
|
||
} catch (urlError) {
|
||
console.log('⚠️ Error setting request URL:', urlError.message);
|
||
}
|
||
|
||
// Create test payload
|
||
|
||
console.log('📝 payload:', payload);
|
||
|
||
// Encode payload
|
||
const enc = new TextEncoder();
|
||
const dec = new TextDecoder();
|
||
const bodyStr = JSON.stringify(payload);
|
||
const bytes = enc.encode(bodyStr);
|
||
console.log('📏 Payload size:', bytes.length, 'bytes');
|
||
console.log('📄 Payload string:', bodyStr);
|
||
|
||
// Allocate WASM memory with validation
|
||
let inPtr;
|
||
try {
|
||
if (!wasm.__wbindgen_malloc) {
|
||
console.error('❌ __wbindgen_malloc function not found');
|
||
return null;
|
||
}
|
||
|
||
inPtr = wasm.__wbindgen_malloc(bytes.length, 1);
|
||
console.log('✅ WASM memory allocated, pointer:', inPtr);
|
||
|
||
// Copy data to WASM memory
|
||
const wasmBuffer = new Uint8Array(wasm.memory.buffer, inPtr, bytes.length);
|
||
wasmBuffer.set(bytes);
|
||
console.log('✅ Data copied to WASM memory');
|
||
} catch (memError) {
|
||
console.error('❌ Memory allocation error:', memError);
|
||
return null;
|
||
}
|
||
|
||
// Call the WASM function
|
||
console.log('🚀 Calling get_pawtected_endpoint_payload...');
|
||
let outPtr, outLen, token;
|
||
try {
|
||
const result = wasm.get_pawtected_endpoint_payload(inPtr, bytes.length);
|
||
console.log('✅ Function called, result type:', typeof result, result);
|
||
|
||
if (Array.isArray(result) && result.length === 2) {
|
||
[outPtr, outLen] = result;
|
||
console.log('✅ Got output pointer:', outPtr, 'length:', outLen);
|
||
|
||
// Decode the result
|
||
const outputBuffer = new Uint8Array(wasm.memory.buffer, outPtr, outLen);
|
||
token = dec.decode(outputBuffer);
|
||
console.log('✅ Token decoded successfully');
|
||
} else {
|
||
console.error('❌ Unexpected function result format:', result);
|
||
return null;
|
||
}
|
||
} catch (funcError) {
|
||
console.error('❌ Function call error:', funcError);
|
||
console.error('Stack trace:', funcError.stack);
|
||
return null;
|
||
}
|
||
|
||
// Cleanup memory
|
||
try {
|
||
if (wasm.__wbindgen_free && outPtr && outLen) {
|
||
wasm.__wbindgen_free(outPtr, outLen, 1);
|
||
console.log('✅ Output memory freed');
|
||
}
|
||
if (wasm.__wbindgen_free && inPtr) {
|
||
wasm.__wbindgen_free(inPtr, bytes.length, 1);
|
||
console.log('✅ Input memory freed');
|
||
}
|
||
} catch (cleanupError) {
|
||
console.log('⚠️ Cleanup warning:', cleanupError.message);
|
||
}
|
||
|
||
// Display results
|
||
console.log('');
|
||
console.log('🎉 SUCCESS!');
|
||
console.log('🔑 Full token:');
|
||
console.log(token);
|
||
return token;
|
||
} catch (error) {
|
||
console.error('❌ Failed to generate fp parameter:', error);
|
||
return null;
|
||
}
|
||
}
|
||
|
||
async function findTokenModule(str) {
|
||
console.log('🔎 Searching for wasm Module...');
|
||
const links = Array.from(document.querySelectorAll('link[rel="modulepreload"][href$=".js"]'));
|
||
|
||
for (const link of links) {
|
||
try {
|
||
const url = new URL(link.getAttribute("href"), location.origin).href;
|
||
const code = await fetch(url).then(r => r.text());
|
||
if (code.includes(str)) {
|
||
console.log('✅ Found wasm Module...');
|
||
return url.split('/').pop();
|
||
}
|
||
} catch { }
|
||
}
|
||
console.error(`❌ Could not find Pawtect chunk: `, error);
|
||
}
|
||
|
||
createUI().then(() => {
|
||
// Generate token automatically after UI is ready
|
||
setTimeout(initializeTokenGenerator, 1000);
|
||
|
||
// Attach advanced color matching listeners (resize dialog)
|
||
const advancedInit = () => {
|
||
const chromaSlider = document.getElementById('chromaPenaltyWeightSlider');
|
||
const chromaValue = document.getElementById('chromaWeightValue');
|
||
const resetBtn = document.getElementById('resetAdvancedColorBtn');
|
||
const algoSelect = document.getElementById('colorAlgorithmSelect');
|
||
const chromaToggle = document.getElementById('enableChromaPenaltyToggle');
|
||
const transInput = document.getElementById('transparencyThresholdInput');
|
||
const whiteInput = document.getElementById('whiteThresholdInput');
|
||
const ditherToggle = document.getElementById('enableDitheringToggle');
|
||
if (algoSelect)
|
||
algoSelect.addEventListener('change', (e) => {
|
||
state.colorMatchingAlgorithm = e.target.value;
|
||
saveBotSettings();
|
||
_updateResizePreview();
|
||
});
|
||
if (chromaToggle)
|
||
chromaToggle.addEventListener('change', (e) => {
|
||
state.enableChromaPenalty = e.target.checked;
|
||
saveBotSettings();
|
||
_updateResizePreview();
|
||
});
|
||
if (chromaSlider && chromaValue)
|
||
chromaSlider.addEventListener('input', (e) => {
|
||
state.chromaPenaltyWeight = parseFloat(e.target.value) || 0.15;
|
||
chromaValue.textContent = state.chromaPenaltyWeight.toFixed(2);
|
||
saveBotSettings();
|
||
_updateResizePreview();
|
||
});
|
||
if (transInput)
|
||
transInput.addEventListener('change', (e) => {
|
||
const v = parseInt(e.target.value, 10);
|
||
if (!isNaN(v) && v >= 0 && v <= 255) {
|
||
state.customTransparencyThreshold = v;
|
||
CONFIG.TRANSPARENCY_THRESHOLD = v;
|
||
saveBotSettings();
|
||
_updateResizePreview();
|
||
}
|
||
});
|
||
if (whiteInput)
|
||
whiteInput.addEventListener('change', (e) => {
|
||
const v = parseInt(e.target.value, 10);
|
||
if (!isNaN(v) && v >= 200 && v <= 255) {
|
||
state.customWhiteThreshold = v;
|
||
CONFIG.WHITE_THRESHOLD = v;
|
||
saveBotSettings();
|
||
_updateResizePreview();
|
||
}
|
||
});
|
||
if (ditherToggle)
|
||
ditherToggle.addEventListener('change', (e) => {
|
||
state.ditheringEnabled = e.target.checked;
|
||
saveBotSettings();
|
||
_updateResizePreview();
|
||
});
|
||
if (resetBtn)
|
||
resetBtn.addEventListener('click', () => {
|
||
state.colorMatchingAlgorithm = 'lab';
|
||
state.enableChromaPenalty = true;
|
||
state.chromaPenaltyWeight = 0.15;
|
||
state.customTransparencyThreshold = CONFIG.TRANSPARENCY_THRESHOLD = 100;
|
||
state.customWhiteThreshold = CONFIG.WHITE_THRESHOLD = 250;
|
||
saveBotSettings();
|
||
const a = document.getElementById('colorAlgorithmSelect');
|
||
if (a) a.value = 'lab';
|
||
const ct = document.getElementById('enableChromaPenaltyToggle');
|
||
if (ct) ct.checked = true;
|
||
if (chromaSlider) chromaSlider.value = 0.15;
|
||
if (chromaValue) chromaValue.textContent = '0.15';
|
||
if (transInput) transInput.value = 100;
|
||
if (whiteInput) whiteInput.value = 250;
|
||
_updateResizePreview();
|
||
Utils.showAlert(Utils.t('advancedColorSettingsReset'), 'success');
|
||
});
|
||
};
|
||
// Delay to ensure resize UI built
|
||
setTimeout(advancedInit, 500);
|
||
|
||
// Add cleanup on page unload
|
||
window.addEventListener('beforeunload', () => {
|
||
Utils.cleanupTurnstile();
|
||
});
|
||
});
|
||
})();
|
||
|