playlists.js 66 KB

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