playlists.js 54 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411
  1. // Constants for log message types
  2. const LOG_TYPE = {
  3. SUCCESS: 'success',
  4. WARNING: 'warning',
  5. ERROR: 'error',
  6. INFO: 'info',
  7. DEBUG: 'debug'
  8. };
  9. // Global variables
  10. let allPlaylists = [];
  11. let currentPlaylist = null;
  12. let availablePatterns = [];
  13. let filteredPatterns = [];
  14. let selectedPatterns = new Set();
  15. let previewCache = new Map();
  16. let intersectionObserver = null;
  17. let searchTimeout = null;
  18. // Mobile navigation state
  19. let isMobileView = false;
  20. // Global variables for batching lazy loading
  21. let pendingPatterns = new Map(); // pattern -> element mapping
  22. let batchTimeout = null;
  23. const BATCH_SIZE = 40; // Increased batch size for better performance
  24. const BATCH_DELAY = 150; // Wait 150ms to collect more patterns before batching
  25. // Shared caching for patterns list (persistent across sessions)
  26. const PATTERNS_CACHE_KEY = 'dune_weaver_patterns_cache';
  27. // IndexedDB cache for preview images with size management
  28. const PREVIEW_CACHE_DB_NAME = 'dune_weaver_previews';
  29. const PREVIEW_CACHE_DB_VERSION = 1;
  30. const PREVIEW_CACHE_STORE_NAME = 'previews';
  31. const MAX_CACHE_SIZE_MB = 200;
  32. const MAX_CACHE_SIZE_BYTES = MAX_CACHE_SIZE_MB * 1024 * 1024;
  33. let previewCacheDB = null;
  34. // --- Playback Settings Persistence ---
  35. const PLAYBACK_SETTINGS_KEY = 'dune_weaver_playback_settings';
  36. function savePlaybackSettings() {
  37. const runMode = document.querySelector('input[name="run_playlist"]:checked')?.value || 'single';
  38. const shuffle = document.getElementById('shuffleCheckbox')?.checked || false;
  39. const pauseTime = document.getElementById('pauseTimeInput')?.value || '5';
  40. const clearPattern = document.getElementById('clearPatternSelect')?.value || 'none';
  41. const settings = { runMode, shuffle, pauseTime, clearPattern };
  42. try {
  43. localStorage.setItem(PLAYBACK_SETTINGS_KEY, JSON.stringify(settings));
  44. } catch (e) {}
  45. }
  46. function restorePlaybackSettings() {
  47. try {
  48. const settings = JSON.parse(localStorage.getItem(PLAYBACK_SETTINGS_KEY));
  49. if (!settings) return;
  50. // Run mode
  51. if (settings.runMode) {
  52. const radio = document.querySelector(`input[name="run_playlist"][value="${settings.runMode}"]`);
  53. if (radio) radio.checked = true;
  54. }
  55. // Shuffle
  56. if (typeof settings.shuffle === 'boolean') {
  57. const shuffleBox = document.getElementById('shuffleCheckbox');
  58. if (shuffleBox) shuffleBox.checked = settings.shuffle;
  59. }
  60. // Pause time
  61. if (settings.pauseTime) {
  62. const pauseInput = document.getElementById('pauseTimeInput');
  63. if (pauseInput) pauseInput.value = settings.pauseTime;
  64. }
  65. // Clear pattern
  66. if (settings.clearPattern) {
  67. const clearSel = document.getElementById('clearPatternSelect');
  68. if (clearSel) clearSel.value = settings.clearPattern;
  69. }
  70. } catch (e) {}
  71. }
  72. // Attach listeners to save settings on change
  73. function setupPlaybackSettingsPersistence() {
  74. document.querySelectorAll('input[name="run_playlist"]').forEach(radio => {
  75. radio.addEventListener('change', savePlaybackSettings);
  76. });
  77. const shuffleBox = document.getElementById('shuffleCheckbox');
  78. if (shuffleBox) shuffleBox.addEventListener('change', savePlaybackSettings);
  79. const pauseInput = document.getElementById('pauseTimeInput');
  80. if (pauseInput) pauseInput.addEventListener('input', savePlaybackSettings);
  81. const clearSel = document.getElementById('clearPatternSelect');
  82. if (clearSel) clearSel.addEventListener('change', savePlaybackSettings);
  83. }
  84. // --- End Playback Settings Persistence ---
  85. // --- Playlist Selection Persistence ---
  86. const LAST_PLAYLIST_KEY = 'dune_weaver_last_playlist';
  87. function saveLastSelectedPlaylist(playlistName) {
  88. try {
  89. localStorage.setItem(LAST_PLAYLIST_KEY, playlistName);
  90. } catch (e) {}
  91. }
  92. function getLastSelectedPlaylist() {
  93. try {
  94. return localStorage.getItem(LAST_PLAYLIST_KEY);
  95. } catch (e) { return null; }
  96. }
  97. // --- End Playlist Selection Persistence ---
  98. // Initialize IndexedDB for preview caching
  99. async function initPreviewCacheDB() {
  100. if (previewCacheDB) return previewCacheDB;
  101. return new Promise((resolve, reject) => {
  102. const request = indexedDB.open(PREVIEW_CACHE_DB_NAME, PREVIEW_CACHE_DB_VERSION);
  103. request.onerror = () => {
  104. logMessage('Failed to open preview cache database', LOG_TYPE.ERROR);
  105. reject(request.error);
  106. };
  107. request.onsuccess = () => {
  108. previewCacheDB = request.result;
  109. logMessage('Preview cache database opened successfully', LOG_TYPE.DEBUG);
  110. resolve(previewCacheDB);
  111. };
  112. request.onupgradeneeded = (event) => {
  113. const db = event.target.result;
  114. // Create object store for preview cache
  115. const store = db.createObjectStore(PREVIEW_CACHE_STORE_NAME, { keyPath: 'pattern' });
  116. store.createIndex('lastAccessed', 'lastAccessed', { unique: false });
  117. store.createIndex('size', 'size', { unique: false });
  118. logMessage('Preview cache database schema created', LOG_TYPE.DEBUG);
  119. };
  120. });
  121. }
  122. // Get preview from IndexedDB cache
  123. async function getPreviewFromCache(pattern) {
  124. try {
  125. if (!previewCacheDB) await initPreviewCacheDB();
  126. const transaction = previewCacheDB.transaction([PREVIEW_CACHE_STORE_NAME], 'readwrite');
  127. const store = transaction.objectStore(PREVIEW_CACHE_STORE_NAME);
  128. return new Promise((resolve, reject) => {
  129. const request = store.get(pattern);
  130. request.onsuccess = () => {
  131. const result = request.result;
  132. if (result) {
  133. // Update last accessed time
  134. result.lastAccessed = Date.now();
  135. store.put(result);
  136. resolve(result.data);
  137. } else {
  138. resolve(null);
  139. }
  140. };
  141. request.onerror = () => reject(request.error);
  142. });
  143. } catch (error) {
  144. logMessage(`Error getting preview from cache: ${error.message}`, LOG_TYPE.WARNING);
  145. return null;
  146. }
  147. }
  148. // Save preview to IndexedDB cache with size management
  149. // Clear a specific pattern from IndexedDB cache
  150. async function clearPatternFromIndexedDB(pattern) {
  151. try {
  152. if (!previewCacheDB) await initPreviewCacheDB();
  153. const transaction = previewCacheDB.transaction([PREVIEW_CACHE_STORE_NAME], 'readwrite');
  154. const store = transaction.objectStore(PREVIEW_CACHE_STORE_NAME);
  155. await new Promise((resolve, reject) => {
  156. const deleteRequest = store.delete(pattern);
  157. deleteRequest.onsuccess = () => {
  158. logMessage(`Cleared ${pattern} from IndexedDB cache`, LOG_TYPE.DEBUG);
  159. resolve();
  160. };
  161. deleteRequest.onerror = () => {
  162. logMessage(`Failed to clear ${pattern} from IndexedDB: ${deleteRequest.error}`, LOG_TYPE.WARNING);
  163. reject(deleteRequest.error);
  164. };
  165. });
  166. } catch (error) {
  167. logMessage(`Error clearing pattern from IndexedDB: ${error.message}`, LOG_TYPE.WARNING);
  168. }
  169. }
  170. async function savePreviewToCache(pattern, previewData) {
  171. try {
  172. if (!previewCacheDB) await initPreviewCacheDB();
  173. // Validate preview data before attempting to fetch
  174. if (!previewData || !previewData.image_data) {
  175. logMessage(`Invalid preview data for ${pattern}, skipping cache save`, LOG_TYPE.WARNING);
  176. return;
  177. }
  178. // Convert preview URL to blob for size calculation
  179. const response = await fetch(previewData.image_data);
  180. const blob = await response.blob();
  181. const size = blob.size;
  182. // Check if we need to free up space
  183. await managePreviewCacheSize(size);
  184. const cacheEntry = {
  185. pattern: pattern,
  186. data: previewData,
  187. size: size,
  188. lastAccessed: Date.now(),
  189. created: Date.now()
  190. };
  191. const transaction = previewCacheDB.transaction([PREVIEW_CACHE_STORE_NAME], 'readwrite');
  192. const store = transaction.objectStore(PREVIEW_CACHE_STORE_NAME);
  193. return new Promise((resolve, reject) => {
  194. const request = store.put(cacheEntry);
  195. request.onsuccess = () => {
  196. logMessage(`Preview cached for ${pattern} (${(size / 1024).toFixed(1)}KB)`, LOG_TYPE.DEBUG);
  197. resolve();
  198. };
  199. request.onerror = () => reject(request.error);
  200. });
  201. } catch (error) {
  202. logMessage(`Error saving preview to cache: ${error.message}`, LOG_TYPE.WARNING);
  203. }
  204. }
  205. // Manage cache size by removing least recently used items
  206. async function managePreviewCacheSize(newItemSize) {
  207. try {
  208. const currentSize = await getPreviewCacheSize();
  209. if (currentSize + newItemSize <= MAX_CACHE_SIZE_BYTES) {
  210. return; // No cleanup needed
  211. }
  212. logMessage(`Cache size would exceed limit (${((currentSize + newItemSize) / 1024 / 1024).toFixed(1)}MB), cleaning up...`, LOG_TYPE.DEBUG);
  213. const transaction = previewCacheDB.transaction([PREVIEW_CACHE_STORE_NAME], 'readwrite');
  214. const store = transaction.objectStore(PREVIEW_CACHE_STORE_NAME);
  215. const index = store.index('lastAccessed');
  216. // Get all entries sorted by last accessed (oldest first)
  217. const entries = await new Promise((resolve, reject) => {
  218. const request = index.getAll();
  219. request.onsuccess = () => resolve(request.result);
  220. request.onerror = () => reject(request.error);
  221. });
  222. // Sort by last accessed time (oldest first)
  223. entries.sort((a, b) => a.lastAccessed - b.lastAccessed);
  224. let freedSpace = 0;
  225. const targetSpace = newItemSize + (MAX_CACHE_SIZE_BYTES * 0.1); // Free 10% extra buffer
  226. for (const entry of entries) {
  227. if (freedSpace >= targetSpace) break;
  228. await new Promise((resolve, reject) => {
  229. const deleteRequest = store.delete(entry.pattern);
  230. deleteRequest.onsuccess = () => {
  231. freedSpace += entry.size;
  232. logMessage(`Evicted cached preview for ${entry.pattern} (${(entry.size / 1024).toFixed(1)}KB)`, LOG_TYPE.DEBUG);
  233. resolve();
  234. };
  235. deleteRequest.onerror = () => reject(deleteRequest.error);
  236. });
  237. }
  238. logMessage(`Freed ${(freedSpace / 1024 / 1024).toFixed(1)}MB from preview cache`, LOG_TYPE.DEBUG);
  239. } catch (error) {
  240. logMessage(`Error managing cache size: ${error.message}`, LOG_TYPE.WARNING);
  241. }
  242. }
  243. // Get current cache size
  244. async function getPreviewCacheSize() {
  245. try {
  246. if (!previewCacheDB) return 0;
  247. const transaction = previewCacheDB.transaction([PREVIEW_CACHE_STORE_NAME], 'readonly');
  248. const store = transaction.objectStore(PREVIEW_CACHE_STORE_NAME);
  249. return new Promise((resolve, reject) => {
  250. const request = store.getAll();
  251. request.onsuccess = () => {
  252. const totalSize = request.result.reduce((sum, entry) => sum + (entry.size || 0), 0);
  253. resolve(totalSize);
  254. };
  255. request.onerror = () => reject(request.error);
  256. });
  257. } catch (error) {
  258. logMessage(`Error getting cache size: ${error.message}`, LOG_TYPE.WARNING);
  259. return 0;
  260. }
  261. }
  262. // Clear preview cache
  263. async function clearPreviewCache() {
  264. try {
  265. if (!previewCacheDB) return;
  266. const transaction = previewCacheDB.transaction([PREVIEW_CACHE_STORE_NAME], 'readwrite');
  267. const store = transaction.objectStore(PREVIEW_CACHE_STORE_NAME);
  268. return new Promise((resolve, reject) => {
  269. const request = store.clear();
  270. request.onsuccess = () => {
  271. logMessage('Preview cache cleared', LOG_TYPE.DEBUG);
  272. resolve();
  273. };
  274. request.onerror = () => reject(request.error);
  275. });
  276. } catch (error) {
  277. logMessage(`Error clearing preview cache: ${error.message}`, LOG_TYPE.WARNING);
  278. }
  279. }
  280. // Get cache statistics
  281. async function getPreviewCacheStats() {
  282. try {
  283. const size = await getPreviewCacheSize();
  284. const transaction = previewCacheDB.transaction([PREVIEW_CACHE_STORE_NAME], 'readonly');
  285. const store = transaction.objectStore(PREVIEW_CACHE_STORE_NAME);
  286. const count = await new Promise((resolve, reject) => {
  287. const request = store.count();
  288. request.onsuccess = () => resolve(request.result);
  289. request.onerror = () => reject(request.error);
  290. });
  291. return {
  292. count,
  293. size,
  294. sizeMB: size / 1024 / 1024,
  295. maxSizeMB: MAX_CACHE_SIZE_MB,
  296. utilizationPercent: (size / MAX_CACHE_SIZE_BYTES) * 100
  297. };
  298. } catch (error) {
  299. logMessage(`Error getting cache stats: ${error.message}`, LOG_TYPE.WARNING);
  300. return { count: 0, size: 0, sizeMB: 0, maxSizeMB: MAX_CACHE_SIZE_MB, utilizationPercent: 0 };
  301. }
  302. }
  303. // Initialize Intersection Observer for lazy loading
  304. function initializeIntersectionObserver() {
  305. intersectionObserver = new IntersectionObserver((entries) => {
  306. // Get all visible elements
  307. const visibleElements = entries.filter(entry => entry.isIntersecting);
  308. if (visibleElements.length === 0) return;
  309. // Collect all visible patterns
  310. const visiblePatterns = new Map();
  311. visibleElements.forEach(entry => {
  312. const patternElement = entry.target;
  313. const pattern = patternElement.dataset.pattern;
  314. if (pattern && !previewCache.has(pattern)) {
  315. visiblePatterns.set(pattern, patternElement);
  316. intersectionObserver.unobserve(patternElement);
  317. }
  318. });
  319. // If we have visible patterns that need loading, add them to the batch
  320. if (visiblePatterns.size > 0) {
  321. // Add to pending batch
  322. for (const [pattern, element] of visiblePatterns) {
  323. pendingPatterns.set(pattern, element);
  324. }
  325. // Clear existing timeout and set new one
  326. if (batchTimeout) {
  327. clearTimeout(batchTimeout);
  328. }
  329. batchTimeout = setTimeout(() => {
  330. processPendingBatch();
  331. }, BATCH_DELAY);
  332. }
  333. }, {
  334. rootMargin: '0px 0px 600px 0px', // Large bottom margin to trigger early as element approaches from bottom
  335. threshold: 0.1
  336. });
  337. }
  338. // Function to get visible patterns that are still loading
  339. function getVisibleLoadingPatterns() {
  340. const visibleLoadingPatterns = new Map();
  341. // Get all pattern elements that are currently visible
  342. const patternElements = document.querySelectorAll('[data-pattern]');
  343. patternElements.forEach(element => {
  344. const pattern = element.dataset.pattern;
  345. if (pattern && !previewCache.has(pattern)) {
  346. // Check if element is visible (intersecting with viewport)
  347. const rect = element.getBoundingClientRect();
  348. const isVisible = rect.top < window.innerHeight && rect.bottom > 0;
  349. if (isVisible) {
  350. visibleLoadingPatterns.set(pattern, element);
  351. }
  352. }
  353. });
  354. return visibleLoadingPatterns;
  355. }
  356. // Modified processPendingBatch to keep polling for loading previews
  357. async function processPendingBatch() {
  358. if (pendingPatterns.size === 0) return;
  359. // Create a copy of current pending patterns and clear the original
  360. const currentBatch = new Map(pendingPatterns);
  361. pendingPatterns.clear();
  362. batchTimeout = null;
  363. const patternsToLoad = Array.from(currentBatch.keys());
  364. try {
  365. logMessage(`Loading ${patternsToLoad.length} pattern previews`, LOG_TYPE.DEBUG);
  366. const response = await fetch('/preview_thr_batch', {
  367. method: 'POST',
  368. headers: { 'Content-Type': 'application/json' },
  369. body: JSON.stringify({ file_names: patternsToLoad })
  370. });
  371. if (response.ok) {
  372. const results = await response.json();
  373. // Process all results
  374. for (const [pattern, data] of Object.entries(results)) {
  375. const element = currentBatch.get(pattern);
  376. const previewContainer = element?.querySelector('.pattern-preview');
  377. if (data && !data.error && data.image_data) {
  378. // Cache both in memory and IndexedDB
  379. previewCache.set(pattern, data);
  380. await savePreviewToCache(pattern, data);
  381. if (previewContainer) {
  382. previewContainer.innerHTML = ''; // Remove loading indicator
  383. previewContainer.innerHTML = `<img src="${data.image_data}" alt="Pattern Preview" class="w-full h-full object-cover rounded-full" />`;
  384. }
  385. } else {
  386. previewCache.set(pattern, { error: true });
  387. }
  388. }
  389. } else {
  390. throw new Error(`HTTP error! status: ${response.status}`);
  391. }
  392. } catch (error) {
  393. logMessage(`Error loading pattern preview batch: ${error.message}`, LOG_TYPE.ERROR);
  394. // Mark as error in cache
  395. for (const pattern of patternsToLoad) {
  396. previewCache.set(pattern, { error: true });
  397. }
  398. }
  399. // After processing, check for any visible loading previews and request them
  400. const stillLoading = getVisibleLoadingPatterns();
  401. if (stillLoading.size > 0) {
  402. // Add to pendingPatterns and immediately process
  403. for (const [pattern, element] of stillLoading) {
  404. pendingPatterns.set(pattern, element);
  405. }
  406. await processPendingBatch();
  407. }
  408. }
  409. // Function to show status message
  410. function showStatusMessage(message, type = 'success') {
  411. const statusContainer = document.getElementById('status-message-container');
  412. const statusMessage = document.getElementById('status-message');
  413. if (!statusContainer || !statusMessage) return;
  414. // Set message and color based on type
  415. statusMessage.textContent = message;
  416. statusMessage.className = `text-base font-semibold opacity-0 transform -translate-y-2 transition-all duration-300 ease-in-out px-4 py-2 rounded-lg shadow-lg ${
  417. type === 'success' ? 'bg-green-50 text-green-700 border border-green-200' :
  418. type === 'error' ? 'bg-red-50 text-red-700 border border-red-200' :
  419. type === 'warning' ? 'bg-yellow-50 text-yellow-700 border border-yellow-200' :
  420. 'bg-blue-50 text-blue-700 border border-blue-200'
  421. }`;
  422. // Show message with animation
  423. requestAnimationFrame(() => {
  424. statusMessage.classList.remove('opacity-0', '-translate-y-2');
  425. statusMessage.classList.add('opacity-100', 'translate-y-0');
  426. });
  427. // Hide message after 5 seconds
  428. setTimeout(() => {
  429. statusMessage.classList.remove('opacity-100', 'translate-y-0');
  430. statusMessage.classList.add('opacity-0', '-translate-y-2');
  431. }, 5000);
  432. }
  433. // Function to log messages
  434. function logMessage(message, type = LOG_TYPE.DEBUG) {
  435. console.log(`[${type}] ${message}`);
  436. }
  437. // Load all playlists
  438. async function loadPlaylists() {
  439. try {
  440. const response = await fetch('/list_all_playlists');
  441. if (response.ok) {
  442. allPlaylists = await response.json();
  443. displayPlaylists();
  444. // Auto-select last selected
  445. const last = getLastSelectedPlaylist();
  446. if (last && allPlaylists.includes(last)) {
  447. setTimeout(() => {
  448. const nav = document.getElementById('playlistsNav');
  449. const el = Array.from(nav.querySelectorAll('a')).find(a => a.textContent.trim() === last);
  450. if (el) el.click();
  451. }, 0);
  452. }
  453. } else {
  454. throw new Error('Failed to load playlists');
  455. }
  456. } catch (error) {
  457. logMessage(`Error loading playlists: ${error.message}`, LOG_TYPE.ERROR);
  458. showStatusMessage('Failed to load playlists', 'error');
  459. }
  460. }
  461. // Display playlists in sidebar
  462. function displayPlaylists() {
  463. const playlistsNav = document.getElementById('playlistsNav');
  464. playlistsNav.innerHTML = '';
  465. if (allPlaylists.length === 0) {
  466. playlistsNav.innerHTML = `
  467. <div class="flex items-center justify-center py-8 text-gray-500 dark:text-gray-400">
  468. <span class="text-sm">No playlists found</span>
  469. </div>
  470. `;
  471. return;
  472. }
  473. allPlaylists.forEach(playlist => {
  474. const playlistItem = document.createElement('a');
  475. playlistItem.className = 'flex items-center gap-3 px-3 py-2.5 rounded-lg text-gray-700 dark:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-700 hover:text-gray-900 dark:hover:text-gray-100 transition-colors duration-150 cursor-pointer';
  476. playlistItem.innerHTML = `
  477. <span class="material-icons text-lg text-gray-500 dark:text-gray-400">queue_music</span>
  478. <span class="text-sm font-medium flex-1 truncate">${playlist}</span>
  479. <span class="material-icons text-lg text-gray-400 dark:text-gray-500">chevron_right</span>
  480. `;
  481. playlistItem.addEventListener('click', () => selectPlaylist(playlist, playlistItem));
  482. playlistsNav.appendChild(playlistItem);
  483. });
  484. }
  485. // Select a playlist
  486. async function selectPlaylist(playlistName, element) {
  487. // Remove active state from all playlist items
  488. document.querySelectorAll('#playlistsNav a').forEach(item => {
  489. item.classList.remove('text-gray-900', 'dark:text-gray-100', 'bg-gray-100', 'dark:bg-gray-700', 'font-semibold');
  490. item.classList.add('text-gray-700', 'dark:text-gray-300', 'font-medium');
  491. });
  492. // Add active state to selected item
  493. element.classList.remove('text-gray-700', 'dark:text-gray-300', 'font-medium');
  494. element.classList.add('text-gray-900', 'dark:text-gray-100', 'bg-gray-100', 'dark:bg-gray-700', 'font-semibold');
  495. // Update current playlist
  496. currentPlaylist = playlistName;
  497. // Update header with playlist name and delete button
  498. const header = document.getElementById('currentPlaylistTitle');
  499. header.innerHTML = `
  500. <h1 class="text-gray-900 dark:text-gray-100 text-2xl font-semibold leading-tight truncate">${playlistName}</h1>
  501. <button id="deletePlaylistBtn" class="p-1 rounded-lg hover:bg-red-100 dark:hover:bg-red-900/20 text-gray-500 dark:text-gray-500 hover:text-red-500 dark:hover:text-red-400 transition-all duration-150 flex-shrink-0" title="Delete playlist">
  502. <span class="material-icons text-lg">delete</span>
  503. </button>
  504. `;
  505. // Add delete button event listener
  506. document.getElementById('deletePlaylistBtn').addEventListener('click', () => deletePlaylist(playlistName));
  507. // Enable buttons
  508. document.getElementById('addPatternsBtn').disabled = false;
  509. document.getElementById('runPlaylistBtn').disabled = false;
  510. // Save last selected
  511. saveLastSelectedPlaylist(playlistName);
  512. // Show playlist details on mobile
  513. showPlaylistDetails();
  514. // Load playlist patterns
  515. await loadPlaylistPatterns(playlistName);
  516. }
  517. // Load patterns for selected playlist
  518. async function loadPlaylistPatterns(playlistName) {
  519. try {
  520. const response = await fetch(`/get_playlist?name=${encodeURIComponent(playlistName)}`);
  521. if (response.ok) {
  522. const playlistData = await response.json();
  523. displayPlaylistPatterns(playlistData.files || []);
  524. // Show playback settings
  525. document.getElementById('playbackSettings').classList.remove('hidden');
  526. } else {
  527. throw new Error('Failed to load playlist patterns');
  528. }
  529. } catch (error) {
  530. logMessage(`Error loading playlist patterns: ${error.message}`, LOG_TYPE.ERROR);
  531. showStatusMessage('Failed to load playlist patterns', 'error');
  532. }
  533. }
  534. // Display patterns in the current playlist
  535. async function displayPlaylistPatterns(patterns) {
  536. const patternsGrid = document.getElementById('patternsGrid');
  537. if (patterns.length === 0) {
  538. patternsGrid.innerHTML = `
  539. <div class="flex items-center justify-center col-span-full py-12 text-gray-500 dark:text-gray-400">
  540. <span class="text-sm">No patterns in this playlist</span>
  541. </div>
  542. `;
  543. return;
  544. }
  545. // No more pre-loading - all patterns will use lazy loading
  546. patternsGrid.innerHTML = '';
  547. patterns.forEach(pattern => {
  548. const patternCard = createPatternCard(pattern, true);
  549. patternsGrid.appendChild(patternCard);
  550. // Set up lazy loading for ALL patterns
  551. patternCard.dataset.pattern = pattern;
  552. intersectionObserver.observe(patternCard);
  553. });
  554. }
  555. // Create a pattern card
  556. function createPatternCard(pattern, showRemove = false) {
  557. const card = document.createElement('div');
  558. card.className = 'flex flex-col gap-3 group cursor-pointer relative';
  559. const previewContainer = document.createElement('div');
  560. previewContainer.className = 'w-full aspect-square bg-cover rounded-full shadow-sm group-hover:shadow-md transition-shadow duration-150 border border-gray-200 dark:border-gray-700 pattern-preview relative';
  561. // Only set preview image if already available in memory cache
  562. const previewData = previewCache.get(pattern);
  563. if (previewData && !previewData.error && previewData.image_data) {
  564. previewContainer.innerHTML = `<img src="${previewData.image_data}" alt="Pattern Preview" class="w-full h-full object-cover rounded-full" />`;
  565. }
  566. const patternName = document.createElement('p');
  567. patternName.className = 'text-sm text-gray-800 dark:text-gray-300 group-hover:text-gray-900 dark:group-hover:text-gray-100 font-medium truncate text-center';
  568. patternName.textContent = pattern.replace('.thr', '').split('/').pop();
  569. card.appendChild(previewContainer);
  570. card.appendChild(patternName);
  571. if (showRemove) {
  572. const removeBtn = document.createElement('button');
  573. removeBtn.className = 'absolute top-2 right-2 size-6 rounded-full bg-red-500 hover:bg-red-600 text-white opacity-0 group-hover:opacity-100 transition-opacity duration-150 flex items-center justify-center text-xs';
  574. removeBtn.innerHTML = '<span class="material-icons text-sm">close</span>';
  575. removeBtn.addEventListener('click', (e) => {
  576. e.stopPropagation();
  577. removePatternFromPlaylist(pattern);
  578. });
  579. card.appendChild(removeBtn);
  580. }
  581. return card;
  582. }
  583. // Search and filter patterns
  584. function searchPatterns(query) {
  585. const normalizedQuery = query.toLowerCase().trim();
  586. if (!normalizedQuery) {
  587. filteredPatterns = [...availablePatterns];
  588. } else {
  589. filteredPatterns = availablePatterns.filter(pattern => {
  590. const patternName = pattern.replace('.thr', '').split('/').pop().toLowerCase();
  591. return patternName.includes(normalizedQuery);
  592. });
  593. }
  594. displayAvailablePatterns();
  595. }
  596. // Handle search input
  597. function handleSearchInput() {
  598. const searchInput = document.getElementById('patternSearchInput');
  599. const clearBtn = document.getElementById('clearSearchBtn');
  600. const query = searchInput.value;
  601. // Show/hide clear button
  602. if (query) {
  603. clearBtn.classList.remove('hidden');
  604. } else {
  605. clearBtn.classList.add('hidden');
  606. }
  607. // Debounce search
  608. if (searchTimeout) {
  609. clearTimeout(searchTimeout);
  610. }
  611. searchTimeout = setTimeout(() => {
  612. searchPatterns(query);
  613. }, 300);
  614. }
  615. // Clear search
  616. function clearSearch() {
  617. const searchInput = document.getElementById('patternSearchInput');
  618. const clearBtn = document.getElementById('clearSearchBtn');
  619. searchInput.value = '';
  620. clearBtn.classList.add('hidden');
  621. searchPatterns('');
  622. }
  623. // Remove pattern from playlist
  624. async function removePatternFromPlaylist(pattern) {
  625. if (!currentPlaylist) return;
  626. if (confirm(`Remove "${pattern.split('/').pop()}" from playlist?`)) {
  627. try {
  628. // Get current playlist data
  629. const response = await fetch(`/get_playlist?name=${encodeURIComponent(currentPlaylist)}`);
  630. if (response.ok) {
  631. const playlistData = await response.json();
  632. const updatedFiles = playlistData.files.filter(file => file !== pattern);
  633. // Update playlist
  634. const updateResponse = await fetch('/modify_playlist', {
  635. method: 'POST',
  636. headers: { 'Content-Type': 'application/json' },
  637. body: JSON.stringify({
  638. playlist_name: currentPlaylist,
  639. files: updatedFiles
  640. })
  641. });
  642. if (updateResponse.ok) {
  643. showStatusMessage('Pattern removed from playlist', 'success');
  644. await loadPlaylistPatterns(currentPlaylist);
  645. } else {
  646. throw new Error('Failed to update playlist');
  647. }
  648. }
  649. } catch (error) {
  650. logMessage(`Error removing pattern: ${error.message}`, LOG_TYPE.ERROR);
  651. showStatusMessage('Failed to remove pattern', 'error');
  652. }
  653. }
  654. }
  655. // Load available patterns for adding (no caching)
  656. async function loadAvailablePatterns(forceRefresh = false) {
  657. const loadingIndicator = document.getElementById('patternsLoadingIndicator');
  658. const grid = document.getElementById('availablePatternsGrid');
  659. const noResultsMessage = document.getElementById('noResultsMessage');
  660. // Always fetch from backend
  661. loadingIndicator.classList.remove('hidden');
  662. grid.classList.add('hidden');
  663. noResultsMessage.classList.add('hidden');
  664. try {
  665. logMessage('Fetching fresh patterns list from server', LOG_TYPE.DEBUG);
  666. const response = await fetch('/list_theta_rho_files');
  667. if (response.ok) {
  668. const patterns = await response.json();
  669. const thrPatterns = patterns.filter(file => file.endsWith('.thr'));
  670. availablePatterns = [...thrPatterns];
  671. filteredPatterns = [...availablePatterns];
  672. // Show patterns immediately - all lazy loading now
  673. displayAvailablePatterns();
  674. if (forceRefresh) {
  675. showStatusMessage('Patterns list refreshed successfully', 'success');
  676. }
  677. } else {
  678. throw new Error('Failed to load available patterns');
  679. }
  680. } catch (error) {
  681. logMessage(`Error loading available patterns: ${error.message}`, LOG_TYPE.ERROR);
  682. showStatusMessage('Failed to load available patterns', 'error');
  683. } finally {
  684. loadingIndicator.classList.add('hidden');
  685. }
  686. }
  687. // Display available patterns in modal
  688. function displayAvailablePatterns() {
  689. const grid = document.getElementById('availablePatternsGrid');
  690. const noResultsMessage = document.getElementById('noResultsMessage');
  691. grid.classList.remove('hidden');
  692. noResultsMessage.classList.add('hidden');
  693. grid.innerHTML = '';
  694. if (filteredPatterns.length === 0) {
  695. grid.classList.add('hidden');
  696. noResultsMessage.classList.remove('hidden');
  697. return;
  698. }
  699. filteredPatterns.forEach((pattern, index) => {
  700. const card = document.createElement('div');
  701. card.className = 'flex flex-col gap-2 cursor-pointer transition-all duration-150 hover:scale-105';
  702. card.dataset.pattern = pattern;
  703. card.innerHTML = `
  704. <div class="w-full bg-center aspect-square bg-cover rounded-full border border-gray-200 dark:border-gray-700 relative pattern-preview">
  705. <div class="absolute top-2 right-2 size-6 rounded-full shadow-md opacity-0 transition-opacity duration-150 flex items-center justify-center">
  706. <span class="material-icons text-sm text-gray-600 dark:text-gray-300">add</span>
  707. </div>
  708. </div>
  709. <p class="text-xs text-gray-700 dark:text-gray-300 font-medium truncate text-center">${pattern.replace('.thr', '').split('/').pop()}</p>
  710. `;
  711. const previewContainer = card.querySelector('.pattern-preview');
  712. const addBtn = card.querySelector('.absolute.top-2');
  713. // Only set preview image if already available in memory cache
  714. const previewData = previewCache.get(pattern);
  715. if (previewData && !previewData.error && previewData.image_data) {
  716. previewContainer.innerHTML = `<img src="${previewData.image_data}" alt="Pattern Preview" class="w-full h-full object-cover rounded-full" />`;
  717. // Re-add the add button
  718. const addBtnContainer = document.createElement('div');
  719. addBtnContainer.className = 'absolute top-2 right-2 size-6 rounded-full bg-white dark:bg-gray-700 shadow-md opacity-0 transition-opacity duration-150 flex items-center justify-center';
  720. addBtnContainer.innerHTML = '<span class="material-icons text-sm text-gray-600 dark:text-gray-300">add</span>';
  721. previewContainer.appendChild(addBtnContainer);
  722. }
  723. // Set up lazy loading for ALL patterns
  724. intersectionObserver.observe(card);
  725. // Handle selection
  726. card.addEventListener('click', () => {
  727. if (selectedPatterns.has(pattern)) {
  728. selectedPatterns.delete(pattern);
  729. card.classList.remove('ring-2', 'ring-blue-500');
  730. addBtn.classList.remove('opacity-100', 'bg-blue-500', 'text-white');
  731. addBtn.classList.add('opacity-0', 'bg-white', 'dark:bg-gray-700');
  732. addBtn.querySelector('.material-icons').textContent = 'add';
  733. } else {
  734. selectedPatterns.add(pattern);
  735. card.classList.add('ring-2', 'ring-blue-500');
  736. addBtn.classList.remove('opacity-0', 'bg-white', 'dark:bg-gray-700');
  737. addBtn.classList.add('opacity-100', 'bg-blue-500', 'text-white');
  738. addBtn.querySelector('.material-icons').textContent = 'check';
  739. }
  740. });
  741. // Show add button on hover
  742. card.addEventListener('mouseenter', () => {
  743. if (!selectedPatterns.has(pattern)) {
  744. addBtn.classList.remove('opacity-0');
  745. addBtn.classList.add('opacity-100');
  746. }
  747. });
  748. card.addEventListener('mouseleave', () => {
  749. if (!selectedPatterns.has(pattern)) {
  750. addBtn.classList.remove('opacity-100');
  751. addBtn.classList.add('opacity-0');
  752. }
  753. });
  754. grid.appendChild(card);
  755. });
  756. // Trigger preview loading for visible patterns after displaying
  757. triggerPreviewLoadingForVisible();
  758. }
  759. // Trigger preview loading for currently visible patterns
  760. function triggerPreviewLoadingForVisible() {
  761. // Get all pattern cards currently in the DOM
  762. const patternCards = document.querySelectorAll('[data-pattern]');
  763. patternCards.forEach(card => {
  764. const pattern = card.dataset.pattern;
  765. const previewContainer = card.querySelector('.pattern-preview');
  766. // Check if this pattern needs preview loading
  767. if (pattern && !previewCache.has(pattern) && !pendingPatterns.has(pattern)) {
  768. // Add to batch for immediate loading
  769. addPatternToBatch(pattern, previewContainer);
  770. }
  771. });
  772. // Process any pending previews immediately
  773. if (pendingPatterns.size > 0) {
  774. processPendingBatch();
  775. }
  776. }
  777. // Add pattern to pending batch for efficient loading
  778. async function addPatternToBatch(pattern, element) {
  779. // Check in-memory cache first
  780. if (previewCache.has(pattern)) {
  781. const previewData = previewCache.get(pattern);
  782. if (previewData && !previewData.error) {
  783. if (element) {
  784. updatePreviewElement(element, previewData.image_data);
  785. }
  786. }
  787. return;
  788. }
  789. // Check IndexedDB cache
  790. const cachedData = await getPreviewFromCache(pattern);
  791. if (cachedData && !cachedData.error) {
  792. // Add to in-memory cache for faster access
  793. previewCache.set(pattern, cachedData);
  794. if (element) {
  795. updatePreviewElement(element, cachedData.image_data);
  796. }
  797. return;
  798. }
  799. // Add loading indicator with better styling
  800. if (element && !element.querySelector('img')) {
  801. element.innerHTML = `
  802. <div class="absolute inset-0 flex items-center justify-center bg-slate-100 rounded-full">
  803. <div class="bg-slate-200 rounded-full h-8 w-8 flex items-center justify-center">
  804. <div class="bg-slate-500 rounded-full h-4 w-4"></div>
  805. </div>
  806. </div>
  807. <div class="absolute inset-0 flex items-center justify-center">
  808. <div class="text-xs text-slate-500 mt-12">Loading...</div>
  809. </div>
  810. `;
  811. }
  812. // Add to pending batch
  813. pendingPatterns.set(pattern, element);
  814. // Process batch immediately if it's full
  815. if (pendingPatterns.size >= BATCH_SIZE) {
  816. processPendingBatch();
  817. }
  818. }
  819. // Update preview element with image
  820. function updatePreviewElement(element, imageData) {
  821. if (element) {
  822. element.innerHTML = `<img src="${imageData}" alt="Pattern Preview" class="w-full h-full object-cover rounded-full" />`;
  823. // Re-add the add button if it exists in the parent card
  824. const card = element.closest('[data-pattern]');
  825. if (card && !selectedPatterns.has(card.dataset.pattern)) {
  826. const addBtnContainer = document.createElement('div');
  827. addBtnContainer.className = 'absolute top-2 right-2 size-6 rounded-full bg-white dark:bg-gray-700 shadow-md opacity-0 transition-opacity duration-150 flex items-center justify-center';
  828. addBtnContainer.innerHTML = '<span class="material-icons text-sm text-gray-600 dark:text-gray-300">add</span>';
  829. element.appendChild(addBtnContainer);
  830. }
  831. }
  832. }
  833. // Add selected patterns to playlist
  834. async function addSelectedPatternsToPlaylist() {
  835. if (selectedPatterns.size === 0 || !currentPlaylist) return;
  836. try {
  837. // Get current playlist data
  838. const response = await fetch(`/get_playlist?name=${encodeURIComponent(currentPlaylist)}`);
  839. if (response.ok) {
  840. const playlistData = await response.json();
  841. const currentFiles = playlistData.files || [];
  842. const newFiles = Array.from(selectedPatterns).filter(pattern => !currentFiles.includes(pattern));
  843. const updatedFiles = [...currentFiles, ...newFiles];
  844. // Update playlist
  845. const updateResponse = await fetch('/modify_playlist', {
  846. method: 'POST',
  847. headers: { 'Content-Type': 'application/json' },
  848. body: JSON.stringify({
  849. playlist_name: currentPlaylist,
  850. files: updatedFiles
  851. })
  852. });
  853. if (updateResponse.ok) {
  854. showStatusMessage(`Added ${newFiles.length} patterns to playlist`, 'success');
  855. selectedPatterns.clear();
  856. document.getElementById('addPatternsModal').classList.add('hidden');
  857. await loadPlaylistPatterns(currentPlaylist);
  858. } else {
  859. throw new Error('Failed to update playlist');
  860. }
  861. }
  862. } catch (error) {
  863. logMessage(`Error adding patterns: ${error.message}`, LOG_TYPE.ERROR);
  864. showStatusMessage('Failed to add patterns', 'error');
  865. }
  866. }
  867. // Run playlist
  868. async function runPlaylist() {
  869. if (!currentPlaylist) return;
  870. const runMode = document.querySelector('input[name="run_playlist"]:checked')?.value || 'single';
  871. const pauseTime = parseInt(document.getElementById('pauseTimeInput').value) || 0;
  872. const clearPattern = document.getElementById('clearPatternSelect').value;
  873. const shuffle = document.getElementById('shuffleCheckbox')?.checked || false;
  874. try {
  875. const response = await fetch('/run_playlist', {
  876. method: 'POST',
  877. headers: { 'Content-Type': 'application/json' },
  878. body: JSON.stringify({
  879. playlist_name: currentPlaylist,
  880. run_mode: runMode,
  881. pause_time: pauseTime,
  882. clear_pattern: clearPattern === 'none' ? null : clearPattern,
  883. shuffle: shuffle
  884. })
  885. });
  886. if (response.ok) {
  887. showStatusMessage(`Started playlist: ${currentPlaylist}`, 'success');
  888. // Show the preview modal when a playlist starts
  889. if (typeof setModalVisibility === 'function') {
  890. setModalVisibility(true, false);
  891. } else if (window.openPlayerPreviewModal) {
  892. window.openPlayerPreviewModal();
  893. }
  894. } else {
  895. let errorMsg = 'Failed to run playlist';
  896. let errorType = 'error';
  897. try {
  898. const data = await response.json();
  899. if (data.detail) {
  900. errorMsg = data.detail;
  901. // Handle specific error cases with appropriate messaging
  902. if (data.detail === 'Connection not established') {
  903. errorMsg = 'Please connect to the device before running a playlist';
  904. errorType = 'warning';
  905. } else if (response.status === 409) {
  906. errorMsg = 'Another pattern is already running. Please stop the current pattern first.';
  907. errorType = 'warning';
  908. } else if (response.status === 404) {
  909. errorMsg = 'Playlist not found. Please refresh the page and try again.';
  910. errorType = 'error';
  911. }
  912. }
  913. } catch (e) {
  914. // If we can't parse the JSON, use status-based messaging
  915. if (response.status === 400) {
  916. errorMsg = 'Invalid request. Please check your settings and try again.';
  917. } else if (response.status === 500) {
  918. errorMsg = 'Server error. Please try again later.';
  919. }
  920. }
  921. showStatusMessage(errorMsg, errorType);
  922. }
  923. } catch (error) {
  924. logMessage(`Error running playlist: ${error.message}`, LOG_TYPE.ERROR);
  925. // Handle network errors specifically
  926. if (error.name === 'TypeError' && error.message.includes('fetch')) {
  927. showStatusMessage('Network error. Please check your connection and try again.', 'error');
  928. } else {
  929. showStatusMessage('Failed to run playlist', 'error');
  930. }
  931. }
  932. }
  933. // Create new playlist
  934. async function createNewPlaylist() {
  935. const playlistName = document.getElementById('newPlaylistName').value.trim();
  936. if (!playlistName) {
  937. showStatusMessage('Please enter a playlist name', 'warning');
  938. return;
  939. }
  940. try {
  941. const response = await fetch('/create_playlist', {
  942. method: 'POST',
  943. headers: { 'Content-Type': 'application/json' },
  944. body: JSON.stringify({
  945. playlist_name: playlistName,
  946. files: []
  947. })
  948. });
  949. if (response.ok) {
  950. showStatusMessage('Playlist created successfully', 'success');
  951. document.getElementById('addPlaylistModal').classList.add('hidden');
  952. document.getElementById('newPlaylistName').value = '';
  953. await loadPlaylists();
  954. } else {
  955. const data = await response.json();
  956. throw new Error(data.detail || 'Failed to create playlist');
  957. }
  958. } catch (error) {
  959. logMessage(`Error creating playlist: ${error.message}`, LOG_TYPE.ERROR);
  960. showStatusMessage('Failed to create playlist', 'error');
  961. }
  962. }
  963. // Delete playlist
  964. async function deletePlaylist(playlistName) {
  965. if (!confirm(`Are you sure you want to delete the playlist "${playlistName}"?`)) {
  966. return;
  967. }
  968. try {
  969. const response = await fetch('/delete_playlist', {
  970. method: 'DELETE',
  971. headers: { 'Content-Type': 'application/json' },
  972. body: JSON.stringify({
  973. playlist_name: playlistName
  974. })
  975. });
  976. if (response.ok) {
  977. showStatusMessage(`Playlist "${playlistName}" deleted successfully`, 'success');
  978. // If the deleted playlist was selected, clear the selection
  979. if (currentPlaylist === playlistName) {
  980. currentPlaylist = null;
  981. const header = document.getElementById('currentPlaylistTitle');
  982. header.innerHTML = '<h1 class="text-gray-900 text-2xl font-semibold leading-tight truncate">Select a Playlist</h1>';
  983. document.getElementById('addPatternsBtn').disabled = true;
  984. document.getElementById('runPlaylistBtn').disabled = true;
  985. document.getElementById('playbackSettings').classList.add('hidden');
  986. document.getElementById('patternsGrid').innerHTML = `
  987. <div class="flex items-center justify-center col-span-full py-12 text-gray-500 dark:text-gray-400">
  988. <span class="text-sm text-center">Select a playlist to view its patterns</span>
  989. </div>
  990. `;
  991. // Return to playlists list on mobile
  992. showPlaylistsList();
  993. }
  994. // Reload playlists
  995. await loadPlaylists();
  996. } else {
  997. const data = await response.json();
  998. throw new Error(data.detail || 'Failed to delete playlist');
  999. }
  1000. } catch (error) {
  1001. logMessage(`Error deleting playlist: ${error.message}`, LOG_TYPE.ERROR);
  1002. showStatusMessage('Failed to delete playlist', 'error');
  1003. }
  1004. }
  1005. // Setup event listeners
  1006. function setupEventListeners() {
  1007. // Mobile back button event listeners
  1008. document.getElementById('mobileBackBtn').addEventListener('click', () => {
  1009. showPlaylistsList();
  1010. });
  1011. // Add playlist button
  1012. document.getElementById('addPlaylistBtn').addEventListener('click', () => {
  1013. const modal = document.getElementById('addPlaylistModal');
  1014. const input = document.getElementById('newPlaylistName');
  1015. // Show modal first
  1016. modal.classList.remove('hidden');
  1017. // Focus handling
  1018. const focusInput = () => {
  1019. if (input) {
  1020. input.focus();
  1021. input.select();
  1022. }
  1023. };
  1024. // Try multiple approaches to ensure focus
  1025. focusInput();
  1026. requestAnimationFrame(focusInput);
  1027. setTimeout(focusInput, 50);
  1028. setTimeout(focusInput, 100);
  1029. });
  1030. // Add patterns button
  1031. document.getElementById('addPatternsBtn').addEventListener('click', async () => {
  1032. await loadAvailablePatterns();
  1033. document.getElementById('addPatternsModal').classList.remove('hidden');
  1034. // Focus search input when modal opens
  1035. setTimeout(() => {
  1036. document.getElementById('patternSearchInput').focus();
  1037. }, 100);
  1038. });
  1039. // Search functionality
  1040. document.getElementById('patternSearchInput').addEventListener('input', handleSearchInput);
  1041. document.getElementById('clearSearchBtn').addEventListener('click', clearSearch);
  1042. // Handle Enter key in search input
  1043. document.getElementById('patternSearchInput').addEventListener('keypress', (e) => {
  1044. if (e.key === 'Enter') {
  1045. e.preventDefault();
  1046. }
  1047. });
  1048. // Run playlist button
  1049. document.getElementById('runPlaylistBtn').addEventListener('click', runPlaylist);
  1050. // Modal controls
  1051. document.getElementById('cancelPlaylistBtn').addEventListener('click', () => {
  1052. document.getElementById('addPlaylistModal').classList.add('hidden');
  1053. });
  1054. document.getElementById('createPlaylistBtn').addEventListener('click', createNewPlaylist);
  1055. document.getElementById('cancelAddPatternsBtn').addEventListener('click', () => {
  1056. selectedPatterns.clear();
  1057. clearSearch();
  1058. document.getElementById('addPatternsModal').classList.add('hidden');
  1059. });
  1060. document.getElementById('confirmAddPatternsBtn').addEventListener('click', addSelectedPatternsToPlaylist);
  1061. // Handle Enter key in new playlist name input
  1062. document.getElementById('newPlaylistName').addEventListener('keypress', (e) => {
  1063. if (e.key === 'Enter') {
  1064. createNewPlaylist();
  1065. }
  1066. });
  1067. // Close modals when clicking outside
  1068. document.getElementById('addPlaylistModal').addEventListener('click', (e) => {
  1069. if (e.target.id === 'addPlaylistModal') {
  1070. document.getElementById('addPlaylistModal').classList.add('hidden');
  1071. }
  1072. });
  1073. document.getElementById('addPatternsModal').addEventListener('click', (e) => {
  1074. if (e.target.id === 'addPatternsModal') {
  1075. selectedPatterns.clear();
  1076. clearSearch();
  1077. document.getElementById('addPatternsModal').classList.add('hidden');
  1078. }
  1079. });
  1080. }
  1081. // Initialize playlists page
  1082. document.addEventListener('DOMContentLoaded', async () => {
  1083. try {
  1084. // Initialize intersection observer for lazy loading
  1085. initializeIntersectionObserver();
  1086. // Initialize IndexedDB preview cache
  1087. await initPreviewCacheDB();
  1088. // Setup event listeners
  1089. setupEventListeners();
  1090. // Initialize mobile view state
  1091. isMobileView = isMobile();
  1092. if (isMobileView) {
  1093. initMobileLayout();
  1094. } else {
  1095. initDesktopLayout();
  1096. }
  1097. // Add window resize listener for responsive behavior
  1098. window.addEventListener('resize', updateMobileView);
  1099. // Restore playback settings
  1100. restorePlaybackSettings();
  1101. setupPlaybackSettingsPersistence();
  1102. // Load playlists
  1103. await loadPlaylists();
  1104. // Check serial connection status
  1105. await checkSerialStatus();
  1106. logMessage('Playlists page initialized successfully', LOG_TYPE.SUCCESS);
  1107. } catch (error) {
  1108. logMessage(`Error during initialization: ${error.message}`, LOG_TYPE.ERROR);
  1109. showStatusMessage('Failed to initialize playlists page', 'error');
  1110. }
  1111. });
  1112. // Check serial connection status
  1113. async function checkSerialStatus() {
  1114. try {
  1115. const response = await fetch('/serial_status');
  1116. if (response.ok) {
  1117. const data = await response.json();
  1118. const statusDot = document.getElementById('connectionStatusDot');
  1119. if (statusDot) {
  1120. statusDot.className = `inline-block size-2 rounded-full ml-2 align-middle ${
  1121. data.connected ? 'bg-green-500' : 'bg-red-500'
  1122. }`;
  1123. }
  1124. }
  1125. } catch (error) {
  1126. logMessage(`Error checking serial status: ${error.message}`, LOG_TYPE.ERROR);
  1127. }
  1128. }
  1129. // Mobile utility functions
  1130. function isMobile() {
  1131. return window.innerWidth <= 768;
  1132. }
  1133. function updateMobileView() {
  1134. const wasMobile = isMobileView;
  1135. isMobileView = isMobile();
  1136. if (wasMobile !== isMobileView) {
  1137. // Mobile state changed, update layout
  1138. if (isMobileView) {
  1139. initMobileLayout();
  1140. } else {
  1141. initDesktopLayout();
  1142. }
  1143. }
  1144. }
  1145. function initMobileLayout() {
  1146. const sidebar = document.getElementById('playlistsSidebar');
  1147. const details = document.getElementById('playlistDetails');
  1148. const mobileBackBtn = document.getElementById('mobileBackBtn');
  1149. if (!currentPlaylist) {
  1150. // Show playlists list, hide details
  1151. sidebar.classList.remove('mobile-hidden');
  1152. details.classList.add('mobile-hidden');
  1153. mobileBackBtn.classList.add('mobile-hidden');
  1154. } else {
  1155. // Show details, hide playlists list
  1156. sidebar.classList.add('mobile-hidden');
  1157. details.classList.remove('mobile-hidden');
  1158. mobileBackBtn.classList.remove('mobile-hidden');
  1159. mobileBackBtn.classList.add('mobile-flex');
  1160. }
  1161. }
  1162. function initDesktopLayout() {
  1163. const sidebar = document.getElementById('playlistsSidebar');
  1164. const details = document.getElementById('playlistDetails');
  1165. const mobileBackBtn = document.getElementById('mobileBackBtn');
  1166. // Show both sidebar and details on desktop
  1167. sidebar.classList.remove('mobile-hidden');
  1168. details.classList.remove('mobile-hidden');
  1169. mobileBackBtn.classList.add('mobile-hidden');
  1170. mobileBackBtn.classList.remove('mobile-flex');
  1171. }
  1172. function showPlaylistDetails() {
  1173. if (isMobileView) {
  1174. const sidebar = document.getElementById('playlistsSidebar');
  1175. const details = document.getElementById('playlistDetails');
  1176. const mobileBackBtn = document.getElementById('mobileBackBtn');
  1177. sidebar.classList.add('mobile-hidden');
  1178. details.classList.remove('mobile-hidden');
  1179. mobileBackBtn.classList.remove('mobile-hidden');
  1180. mobileBackBtn.classList.add('mobile-flex');
  1181. }
  1182. }
  1183. function showPlaylistsList() {
  1184. if (isMobileView) {
  1185. const sidebar = document.getElementById('playlistsSidebar');
  1186. const details = document.getElementById('playlistDetails');
  1187. const mobileBackBtn = document.getElementById('mobileBackBtn');
  1188. sidebar.classList.remove('mobile-hidden');
  1189. details.classList.add('mobile-hidden');
  1190. mobileBackBtn.classList.add('mobile-hidden');
  1191. mobileBackBtn.classList.remove('mobile-flex');
  1192. }
  1193. }