playlists.js 67 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780
  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. case 'favorite':
  659. // Check if patterns are in favorites (access global favoritePatterns)
  660. const aIsFavorite = window.favoritePatterns ? window.favoritePatterns.has(a.path) : false;
  661. const bIsFavorite = window.favoritePatterns ? window.favoritePatterns.has(b.path) : false;
  662. if (aIsFavorite && !bIsFavorite) return sortDirection === 'asc' ? -1 : 1;
  663. if (!aIsFavorite && bIsFavorite) return sortDirection === 'asc' ? 1 : -1;
  664. // Both have same favorite status, sort by name as secondary sort
  665. aVal = a.name.toLowerCase();
  666. bVal = b.name.toLowerCase();
  667. break;
  668. default:
  669. aVal = a.name.toLowerCase();
  670. bVal = b.name.toLowerCase();
  671. }
  672. let result = 0;
  673. if (aVal < bVal) result = -1;
  674. else if (aVal > bVal) result = 1;
  675. return sortDirection === 'asc' ? result : -result;
  676. });
  677. }
  678. // Filter patterns based on current filters
  679. function filterPatterns(patterns, filters, searchQuery = '') {
  680. return patterns.filter(pattern => {
  681. // Category filter
  682. if (filters.category !== 'all' && pattern.category !== filters.category) {
  683. return false;
  684. }
  685. // Search query filter
  686. if (searchQuery.trim()) {
  687. const normalizedQuery = searchQuery.toLowerCase().trim();
  688. const patternName = pattern.name.toLowerCase();
  689. const category = pattern.category.toLowerCase();
  690. return patternName.includes(normalizedQuery) || category.includes(normalizedQuery);
  691. }
  692. return true;
  693. });
  694. }
  695. // Apply sorting and filtering to patterns
  696. function applyPatternsFilteringAndSorting() {
  697. const searchQuery = document.getElementById('patternSearchInput')?.value || '';
  698. // Check if enhanced metadata is available
  699. if (!availablePatternsWithMetadata || availablePatternsWithMetadata.length === 0) {
  700. // Fallback to basic search if metadata not loaded yet
  701. if (searchQuery.trim()) {
  702. filteredPatterns = availablePatterns.filter(pattern =>
  703. pattern.toLowerCase().includes(searchQuery.toLowerCase())
  704. );
  705. } else {
  706. filteredPatterns = [...availablePatterns];
  707. }
  708. displayAvailablePatterns();
  709. return;
  710. }
  711. // Start with all available patterns with metadata
  712. let patterns = [...availablePatternsWithMetadata];
  713. // Apply filters
  714. patterns = filterPatterns(patterns, currentFilters, searchQuery);
  715. // Apply sorting
  716. patterns = sortPatterns(patterns, currentSort.field, currentSort.direction);
  717. // Update filtered patterns (convert back to path format for compatibility)
  718. filteredPatterns = patterns.map(p => p.path);
  719. // Update display
  720. displayAvailablePatterns();
  721. updateSortAndFilterUI();
  722. }
  723. // Search and filter patterns (updated to work with metadata)
  724. function searchPatterns(query) {
  725. applyPatternsFilteringAndSorting();
  726. }
  727. // Update sort and filter UI to reflect current state
  728. function updateSortAndFilterUI() {
  729. // Update sort direction icon
  730. const sortDirectionIcon = document.getElementById('sortDirectionIcon');
  731. if (sortDirectionIcon) {
  732. sortDirectionIcon.textContent = currentSort.direction === 'asc' ? 'arrow_upward' : 'arrow_downward';
  733. }
  734. // Update sort field select
  735. const sortFieldSelect = document.getElementById('sortFieldSelect');
  736. if (sortFieldSelect) {
  737. sortFieldSelect.value = currentSort.field;
  738. }
  739. // Update filter selects
  740. const categorySelect = document.getElementById('categoryFilterSelect');
  741. if (categorySelect) {
  742. categorySelect.value = currentFilters.category;
  743. }
  744. }
  745. // Populate category filter dropdown with available categories (subfolders)
  746. function updateCategoryFilter() {
  747. const categorySelect = document.getElementById('categoryFilterSelect');
  748. if (!categorySelect) return;
  749. // Check if metadata is available
  750. if (!availablePatternsWithMetadata || availablePatternsWithMetadata.length === 0) {
  751. // Show basic options if metadata not loaded
  752. categorySelect.innerHTML = '<option value="all">All Folders (loading...)</option>';
  753. return;
  754. }
  755. // Get unique categories (subfolders)
  756. const categories = [...new Set(availablePatternsWithMetadata.map(p => p.category))].sort();
  757. // Clear existing options except "All"
  758. categorySelect.innerHTML = '<option value="all">All Folders</option>';
  759. // Add category options
  760. categories.forEach(category => {
  761. if (category) {
  762. const option = document.createElement('option');
  763. option.value = category;
  764. // Display friendly names for full paths
  765. if (category === 'root') {
  766. option.textContent = 'Root Folder';
  767. } else {
  768. // For full paths, show the path but make it more readable
  769. const displayName = category
  770. .split('/')
  771. .map(part => part.charAt(0).toUpperCase() + part.slice(1).replace('_', ' '))
  772. .join(' › ');
  773. option.textContent = displayName;
  774. }
  775. categorySelect.appendChild(option);
  776. }
  777. });
  778. }
  779. // Handle sort field change
  780. function handleSortFieldChange() {
  781. const sortFieldSelect = document.getElementById('sortFieldSelect');
  782. if (sortFieldSelect) {
  783. currentSort.field = sortFieldSelect.value;
  784. applyPatternsFilteringAndSorting();
  785. }
  786. }
  787. // Handle sort direction toggle
  788. function handleSortDirectionToggle() {
  789. currentSort.direction = currentSort.direction === 'asc' ? 'desc' : 'asc';
  790. applyPatternsFilteringAndSorting();
  791. }
  792. // Handle category filter change
  793. function handleCategoryFilterChange() {
  794. const categorySelect = document.getElementById('categoryFilterSelect');
  795. if (categorySelect) {
  796. currentFilters.category = categorySelect.value;
  797. applyPatternsFilteringAndSorting();
  798. }
  799. }
  800. // Handle search input
  801. function handleSearchInput() {
  802. const searchInput = document.getElementById('patternSearchInput');
  803. const clearBtn = document.getElementById('clearSearchBtn');
  804. const query = searchInput.value;
  805. // Show/hide clear button
  806. if (query) {
  807. clearBtn.classList.remove('hidden');
  808. } else {
  809. clearBtn.classList.add('hidden');
  810. }
  811. // Debounce search
  812. if (searchTimeout) {
  813. clearTimeout(searchTimeout);
  814. }
  815. searchTimeout = setTimeout(() => {
  816. searchPatterns(query);
  817. }, 300);
  818. }
  819. // Clear search
  820. function clearSearch() {
  821. const searchInput = document.getElementById('patternSearchInput');
  822. const clearBtn = document.getElementById('clearSearchBtn');
  823. searchInput.value = '';
  824. clearBtn.classList.add('hidden');
  825. searchPatterns('');
  826. }
  827. // Remove pattern from playlist
  828. async function removePatternFromPlaylist(pattern) {
  829. if (!currentPlaylist) return;
  830. if (confirm(`Remove "${pattern.split('/').pop()}" from playlist?`)) {
  831. try {
  832. // Get current playlist data
  833. const response = await fetch(`/get_playlist?name=${encodeURIComponent(currentPlaylist)}`);
  834. if (response.ok) {
  835. const playlistData = await response.json();
  836. const updatedFiles = playlistData.files.filter(file => file !== pattern);
  837. // Update playlist
  838. const updateResponse = await fetch('/modify_playlist', {
  839. method: 'POST',
  840. headers: { 'Content-Type': 'application/json' },
  841. body: JSON.stringify({
  842. playlist_name: currentPlaylist,
  843. files: updatedFiles
  844. })
  845. });
  846. if (updateResponse.ok) {
  847. showStatusMessage('Pattern removed from playlist', 'success');
  848. await loadPlaylistPatterns(currentPlaylist);
  849. } else {
  850. throw new Error('Failed to update playlist');
  851. }
  852. }
  853. } catch (error) {
  854. logMessage(`Error removing pattern: ${error.message}`, LOG_TYPE.ERROR);
  855. showStatusMessage('Failed to remove pattern', 'error');
  856. }
  857. }
  858. }
  859. // Load available patterns for adding (with metadata for sorting/filtering)
  860. async function loadAvailablePatterns(forceRefresh = false) {
  861. const loadingIndicator = document.getElementById('patternsLoadingIndicator');
  862. const grid = document.getElementById('availablePatternsGrid');
  863. const noResultsMessage = document.getElementById('noResultsMessage');
  864. // Always fetch from backend
  865. loadingIndicator.classList.remove('hidden');
  866. grid.classList.add('hidden');
  867. noResultsMessage.classList.add('hidden');
  868. try {
  869. // First load basic patterns list for fast initial display
  870. logMessage('Fetching basic patterns list from server', LOG_TYPE.DEBUG);
  871. const basicResponse = await fetch('/list_theta_rho_files');
  872. if (!basicResponse.ok) {
  873. throw new Error('Failed to load available patterns');
  874. }
  875. const patterns = await basicResponse.json();
  876. const thrPatterns = patterns.filter(file => file.endsWith('.thr'));
  877. availablePatterns = [...thrPatterns];
  878. filteredPatterns = [...availablePatterns];
  879. // Show patterns immediately for fast loading
  880. displayAvailablePatterns();
  881. // Then load metadata in background
  882. setTimeout(async () => {
  883. try {
  884. logMessage('Loading enhanced metadata in background', LOG_TYPE.DEBUG);
  885. const metadataResponse = await fetch('/list_theta_rho_files_with_metadata');
  886. if (metadataResponse.ok) {
  887. const patternsWithMetadata = await metadataResponse.json();
  888. availablePatternsWithMetadata = [...patternsWithMetadata];
  889. // Update category filter dropdown now that we have metadata
  890. updateCategoryFilter();
  891. logMessage(`Enhanced metadata loaded for ${patternsWithMetadata.length} patterns`, LOG_TYPE.DEBUG);
  892. } else {
  893. logMessage('Failed to load enhanced metadata - using basic functionality', LOG_TYPE.WARNING);
  894. }
  895. } catch (metadataError) {
  896. logMessage(`Error loading enhanced metadata: ${metadataError.message}`, LOG_TYPE.WARNING);
  897. // Continue with basic functionality
  898. }
  899. }, 100);
  900. if (forceRefresh) {
  901. showStatusMessage('Patterns list refreshed successfully', 'success');
  902. }
  903. } catch (error) {
  904. logMessage(`Error loading available patterns: ${error.message}`, LOG_TYPE.ERROR);
  905. showStatusMessage('Failed to load available patterns', 'error');
  906. } finally {
  907. loadingIndicator.classList.add('hidden');
  908. }
  909. }
  910. // Update selection count display
  911. function updateSelectionCount() {
  912. const countElement = document.getElementById('selectionCount');
  913. if (countElement) {
  914. const count = selectedPatterns.size;
  915. countElement.textContent = `${count} selected`;
  916. }
  917. updateToggleSelectAllButton();
  918. }
  919. // Smart toggle for Select All / Deselect All
  920. function toggleSelectAll() {
  921. const patterns = filteredPatterns.length > 0 ? filteredPatterns : availablePatterns;
  922. const allSelected = patterns.length > 0 && patterns.every(pattern => selectedPatterns.has(pattern));
  923. if (allSelected) {
  924. // Deselect all
  925. selectedPatterns.clear();
  926. } else {
  927. // Select all
  928. patterns.forEach(pattern => {
  929. selectedPatterns.add(pattern);
  930. });
  931. }
  932. updatePatternSelection();
  933. updateSelectionCount();
  934. }
  935. // Update the toggle button text and icon based on selection state
  936. function updateToggleSelectAllButton() {
  937. const patterns = filteredPatterns.length > 0 ? filteredPatterns : availablePatterns;
  938. const allSelected = patterns.length > 0 && patterns.every(pattern => selectedPatterns.has(pattern));
  939. const icon = document.getElementById('toggleSelectAllIcon');
  940. const text = document.getElementById('toggleSelectAllText');
  941. if (icon && text) {
  942. if (allSelected) {
  943. icon.textContent = 'check_box';
  944. text.textContent = 'Deselect All';
  945. } else {
  946. icon.textContent = 'check_box_outline_blank';
  947. text.textContent = 'Select All';
  948. }
  949. }
  950. }
  951. // Select all visible patterns (legacy function - keep for compatibility)
  952. function selectAllPatterns() {
  953. const patterns = filteredPatterns.length > 0 ? filteredPatterns : availablePatterns;
  954. patterns.forEach(pattern => {
  955. selectedPatterns.add(pattern);
  956. });
  957. updatePatternSelection();
  958. updateSelectionCount();
  959. }
  960. // Deselect all patterns (legacy function - keep for compatibility)
  961. function deselectAllPatterns() {
  962. selectedPatterns.clear();
  963. updatePatternSelection();
  964. updateSelectionCount();
  965. }
  966. // Update visual selection state for all pattern cards
  967. function updatePatternSelection() {
  968. const cards = document.querySelectorAll('#availablePatternsGrid .group');
  969. cards.forEach(card => {
  970. const patternName = card.dataset.pattern;
  971. if (selectedPatterns.has(patternName)) {
  972. card.classList.add('ring-2', 'ring-blue-500');
  973. } else {
  974. card.classList.remove('ring-2', 'ring-blue-500');
  975. }
  976. });
  977. }
  978. // Display available patterns in modal
  979. function displayAvailablePatterns() {
  980. const grid = document.getElementById('availablePatternsGrid');
  981. const noResultsMessage = document.getElementById('noResultsMessage');
  982. grid.classList.remove('hidden');
  983. noResultsMessage.classList.add('hidden');
  984. grid.innerHTML = '';
  985. if (filteredPatterns.length === 0) {
  986. grid.classList.add('hidden');
  987. noResultsMessage.classList.remove('hidden');
  988. return;
  989. }
  990. filteredPatterns.forEach((pattern) => {
  991. const card = document.createElement('div');
  992. const isSelected = selectedPatterns.has(pattern);
  993. // Add blue ring if already selected
  994. card.className = `group flex flex-col gap-2 cursor-pointer transition-all duration-150 hover:scale-105 ${isSelected ? 'ring-2 ring-blue-500' : ''}`;
  995. card.dataset.pattern = pattern;
  996. card.innerHTML = `
  997. <div class="w-full bg-center aspect-square bg-cover rounded-full border border-gray-200 dark:border-gray-700 relative pattern-preview">
  998. </div>
  999. <p class="text-xs text-gray-700 dark:text-gray-300 font-medium truncate text-center">${pattern.replace('.thr', '').split('/').pop()}</p>
  1000. `;
  1001. const previewContainer = card.querySelector('.pattern-preview');
  1002. // Check in-memory cache first
  1003. const previewData = previewCache.get(pattern);
  1004. if (previewData && !previewData.error && previewData.image_data) {
  1005. previewContainer.innerHTML = `<img src="${previewData.image_data}" alt="Pattern Preview" class="w-full h-full object-cover rounded-full" />`;
  1006. } else {
  1007. // Try to load from IndexedDB cache asynchronously
  1008. loadPreviewFromCache(pattern, previewContainer);
  1009. }
  1010. // Set up lazy loading for ALL patterns
  1011. intersectionObserver.observe(card);
  1012. // Handle selection
  1013. card.addEventListener('click', () => {
  1014. if (selectedPatterns.has(pattern)) {
  1015. selectedPatterns.delete(pattern);
  1016. card.classList.remove('ring-2', 'ring-blue-500');
  1017. } else {
  1018. selectedPatterns.add(pattern);
  1019. card.classList.add('ring-2', 'ring-blue-500');
  1020. }
  1021. updateSelectionCount();
  1022. });
  1023. grid.appendChild(card);
  1024. });
  1025. // Trigger immediate preview loading for visible patterns in modal
  1026. requestAnimationFrame(() => {
  1027. setTimeout(() => {
  1028. loadVisibleModalPreviews();
  1029. }, 50); // Small delay to ensure modal layout is complete
  1030. });
  1031. }
  1032. // Load previews for patterns currently visible in the modal
  1033. async function loadVisibleModalPreviews() {
  1034. const visiblePatterns = new Map();
  1035. const patternCards = document.querySelectorAll('#availablePatternsGrid [data-pattern]');
  1036. patternCards.forEach(card => {
  1037. const pattern = card.dataset.pattern;
  1038. const previewContainer = card.querySelector('.pattern-preview');
  1039. // Skip if pattern is already displayed (has an img element) or if already in memory cache
  1040. if (!pattern || previewCache.has(pattern) || previewContainer.querySelector('img')) return;
  1041. // Check if card is visible in viewport
  1042. const rect = card.getBoundingClientRect();
  1043. const isVisible = rect.top < window.innerHeight && rect.bottom > 0;
  1044. if (isVisible) {
  1045. visiblePatterns.set(pattern, card);
  1046. // Remove from intersection observer since we're loading it immediately
  1047. intersectionObserver.unobserve(card);
  1048. }
  1049. });
  1050. if (visiblePatterns.size > 0) {
  1051. logMessage(`Loading ${visiblePatterns.size} visible modal previews not found in cache`, LOG_TYPE.DEBUG);
  1052. // Add visible patterns to pending batch
  1053. for (const [pattern, element] of visiblePatterns) {
  1054. pendingPatterns.set(pattern, element);
  1055. }
  1056. // Process batch immediately for visible patterns
  1057. await processPendingBatch();
  1058. }
  1059. }
  1060. // Add pattern to pending batch for efficient loading
  1061. async function addPatternToBatch(pattern, element) {
  1062. // Check in-memory cache first
  1063. if (previewCache.has(pattern)) {
  1064. const previewData = previewCache.get(pattern);
  1065. if (previewData && !previewData.error) {
  1066. if (element) {
  1067. updatePreviewElement(element, previewData.image_data);
  1068. }
  1069. }
  1070. return;
  1071. }
  1072. // Check IndexedDB cache
  1073. const cachedData = await getPreviewFromCache(pattern);
  1074. if (cachedData && !cachedData.error) {
  1075. // Add to in-memory cache for faster access
  1076. previewCache.set(pattern, cachedData);
  1077. if (element) {
  1078. updatePreviewElement(element, cachedData.image_data);
  1079. }
  1080. return;
  1081. }
  1082. // Add loading indicator with better styling
  1083. if (element && !element.querySelector('img')) {
  1084. element.innerHTML = `
  1085. <div class="absolute inset-0 flex items-center justify-center bg-slate-100 rounded-full">
  1086. <div class="bg-slate-200 rounded-full h-8 w-8 flex items-center justify-center">
  1087. <div class="bg-slate-500 rounded-full h-4 w-4"></div>
  1088. </div>
  1089. </div>
  1090. <div class="absolute inset-0 flex items-center justify-center">
  1091. <div class="text-xs text-slate-500 mt-12">Loading...</div>
  1092. </div>
  1093. `;
  1094. }
  1095. // Add to pending batch
  1096. pendingPatterns.set(pattern, element);
  1097. // Process batch immediately if it's full
  1098. if (pendingPatterns.size >= BATCH_SIZE) {
  1099. processPendingBatch();
  1100. }
  1101. }
  1102. // Update preview element with image
  1103. function updatePreviewElement(element, imageData) {
  1104. if (element) {
  1105. element.innerHTML = `<img src="${imageData}" alt="Pattern Preview" class="w-full h-full object-cover rounded-full" />`;
  1106. // Re-add the add button if it exists in the parent card
  1107. const card = element.closest('[data-pattern]');
  1108. if (card && !selectedPatterns.has(card.dataset.pattern)) {
  1109. const addBtnContainer = document.createElement('div');
  1110. 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';
  1111. addBtnContainer.innerHTML = '<span class="material-icons text-sm text-gray-600 dark:text-gray-300">add</span>';
  1112. element.appendChild(addBtnContainer);
  1113. }
  1114. }
  1115. }
  1116. // Save selected patterns to playlist (replaces entire playlist)
  1117. async function addSelectedPatternsToPlaylist() {
  1118. if (!currentPlaylist) return;
  1119. try {
  1120. // Simply replace the playlist with the selected patterns
  1121. const updatedFiles = Array.from(selectedPatterns);
  1122. // Update playlist
  1123. const updateResponse = await fetch('/modify_playlist', {
  1124. method: 'POST',
  1125. headers: { 'Content-Type': 'application/json' },
  1126. body: JSON.stringify({
  1127. playlist_name: currentPlaylist,
  1128. files: updatedFiles
  1129. })
  1130. });
  1131. if (updateResponse.ok) {
  1132. showStatusMessage(`Playlist "${currentPlaylist}" saved successfully`, 'success');
  1133. selectedPatterns.clear();
  1134. document.getElementById('addPatternsModal').classList.add('hidden');
  1135. await loadPlaylistPatterns(currentPlaylist);
  1136. } else {
  1137. throw new Error('Failed to update playlist');
  1138. }
  1139. } catch (error) {
  1140. logMessage(`Error saving playlist: ${error.message}`, LOG_TYPE.ERROR);
  1141. showStatusMessage('Failed to save playlist', 'error');
  1142. }
  1143. }
  1144. // Run playlist
  1145. async function runPlaylist() {
  1146. if (!currentPlaylist) return;
  1147. const runMode = document.querySelector('input[name="run_playlist"]:checked')?.value || 'single';
  1148. const pauseTime = parseInt(document.getElementById('pauseTimeInput').value) || 0;
  1149. const clearPattern = document.getElementById('clearPatternSelect').value;
  1150. const shuffle = document.getElementById('shuffleCheckbox')?.checked || false;
  1151. try {
  1152. const response = await fetch('/run_playlist', {
  1153. method: 'POST',
  1154. headers: { 'Content-Type': 'application/json' },
  1155. body: JSON.stringify({
  1156. playlist_name: currentPlaylist,
  1157. run_mode: runMode,
  1158. pause_time: pauseTime,
  1159. clear_pattern: clearPattern === 'none' ? null : clearPattern,
  1160. shuffle: shuffle
  1161. })
  1162. });
  1163. if (response.ok) {
  1164. showStatusMessage(`Started playlist: ${currentPlaylist}`, 'success');
  1165. // Show the preview modal when a playlist starts
  1166. if (typeof setModalVisibility === 'function') {
  1167. setModalVisibility(true, false);
  1168. } else if (window.openPlayerPreviewModal) {
  1169. window.openPlayerPreviewModal();
  1170. }
  1171. } else {
  1172. let errorMsg = 'Failed to run playlist';
  1173. let errorType = 'error';
  1174. try {
  1175. const data = await response.json();
  1176. if (data.detail) {
  1177. errorMsg = data.detail;
  1178. // Handle specific error cases with appropriate messaging
  1179. if (data.detail === 'Connection not established') {
  1180. errorMsg = 'Please connect to the device before running a playlist';
  1181. errorType = 'warning';
  1182. } else if (response.status === 409) {
  1183. errorMsg = 'Another pattern is already running. Please stop the current pattern first.';
  1184. errorType = 'warning';
  1185. } else if (response.status === 404) {
  1186. errorMsg = 'Playlist not found. Please refresh the page and try again.';
  1187. errorType = 'error';
  1188. }
  1189. }
  1190. } catch (e) {
  1191. // If we can't parse the JSON, use status-based messaging
  1192. if (response.status === 400) {
  1193. errorMsg = 'Invalid request. Please check your settings and try again.';
  1194. } else if (response.status === 500) {
  1195. errorMsg = 'Server error. Please try again later.';
  1196. }
  1197. }
  1198. showStatusMessage(errorMsg, errorType);
  1199. }
  1200. } catch (error) {
  1201. logMessage(`Error running playlist: ${error.message}`, LOG_TYPE.ERROR);
  1202. // Handle network errors specifically
  1203. if (error.name === 'TypeError' && error.message.includes('fetch')) {
  1204. showStatusMessage('Network error. Please check your connection and try again.', 'error');
  1205. } else {
  1206. showStatusMessage('Failed to run playlist', 'error');
  1207. }
  1208. }
  1209. }
  1210. // Create new playlist
  1211. async function createNewPlaylist() {
  1212. const playlistName = document.getElementById('newPlaylistName').value.trim();
  1213. if (!playlistName) {
  1214. showStatusMessage('Please enter a playlist name', 'warning');
  1215. return;
  1216. }
  1217. try {
  1218. const response = await fetch('/create_playlist', {
  1219. method: 'POST',
  1220. headers: { 'Content-Type': 'application/json' },
  1221. body: JSON.stringify({
  1222. playlist_name: playlistName,
  1223. files: []
  1224. })
  1225. });
  1226. if (response.ok) {
  1227. showStatusMessage('Playlist created successfully', 'success');
  1228. document.getElementById('addPlaylistModal').classList.add('hidden');
  1229. document.getElementById('newPlaylistName').value = '';
  1230. await loadPlaylists();
  1231. } else {
  1232. const data = await response.json();
  1233. throw new Error(data.detail || 'Failed to create playlist');
  1234. }
  1235. } catch (error) {
  1236. logMessage(`Error creating playlist: ${error.message}`, LOG_TYPE.ERROR);
  1237. showStatusMessage('Failed to create playlist', 'error');
  1238. }
  1239. }
  1240. // Delete playlist
  1241. async function deletePlaylist(playlistName) {
  1242. if (!confirm(`Are you sure you want to delete the playlist "${playlistName}"?`)) {
  1243. return;
  1244. }
  1245. try {
  1246. const response = await fetch('/delete_playlist', {
  1247. method: 'DELETE',
  1248. headers: { 'Content-Type': 'application/json' },
  1249. body: JSON.stringify({
  1250. playlist_name: playlistName
  1251. })
  1252. });
  1253. if (response.ok) {
  1254. showStatusMessage(`Playlist "${playlistName}" deleted successfully`, 'success');
  1255. // If the deleted playlist was selected, clear the selection
  1256. if (currentPlaylist === playlistName) {
  1257. currentPlaylist = null;
  1258. const header = document.getElementById('currentPlaylistTitle');
  1259. header.innerHTML = '<h1 class="text-gray-900 text-2xl font-semibold leading-tight truncate">Select a Playlist</h1>';
  1260. document.getElementById('addPatternsBtn').disabled = true;
  1261. document.getElementById('runPlaylistBtn').disabled = true;
  1262. document.getElementById('playbackSettings').classList.add('hidden');
  1263. document.getElementById('patternsGrid').innerHTML = `
  1264. <div class="flex items-center justify-center col-span-full py-12 text-gray-500 dark:text-gray-400">
  1265. <span class="text-sm text-center">Select a playlist to view its patterns</span>
  1266. </div>
  1267. `;
  1268. // Return to playlists list on mobile
  1269. showPlaylistsList();
  1270. }
  1271. // Reload playlists
  1272. await loadPlaylists();
  1273. } else {
  1274. const data = await response.json();
  1275. throw new Error(data.detail || 'Failed to delete playlist');
  1276. }
  1277. } catch (error) {
  1278. logMessage(`Error deleting playlist: ${error.message}`, LOG_TYPE.ERROR);
  1279. showStatusMessage('Failed to delete playlist', 'error');
  1280. }
  1281. }
  1282. // Setup event listeners
  1283. function setupEventListeners() {
  1284. // Mobile back button event listeners
  1285. document.getElementById('mobileBackBtn').addEventListener('click', () => {
  1286. showPlaylistsList();
  1287. });
  1288. // Add playlist button
  1289. document.getElementById('addPlaylistBtn').addEventListener('click', () => {
  1290. const modal = document.getElementById('addPlaylistModal');
  1291. const input = document.getElementById('newPlaylistName');
  1292. // Show modal first
  1293. modal.classList.remove('hidden');
  1294. // Focus handling
  1295. const focusInput = () => {
  1296. if (input) {
  1297. input.focus();
  1298. input.select();
  1299. }
  1300. };
  1301. // Try multiple approaches to ensure focus
  1302. focusInput();
  1303. requestAnimationFrame(focusInput);
  1304. setTimeout(focusInput, 50);
  1305. setTimeout(focusInput, 100);
  1306. });
  1307. // Add patterns button
  1308. document.getElementById('addPatternsBtn').addEventListener('click', async () => {
  1309. // Load current playlist patterns first
  1310. if (currentPlaylist) {
  1311. const response = await fetch(`/get_playlist?name=${encodeURIComponent(currentPlaylist)}`);
  1312. if (response.ok) {
  1313. const playlistData = await response.json();
  1314. const currentFiles = playlistData.files || [];
  1315. // Pre-select current patterns
  1316. selectedPatterns.clear();
  1317. currentFiles.forEach(pattern => selectedPatterns.add(pattern));
  1318. }
  1319. }
  1320. await loadAvailablePatterns();
  1321. updatePatternSelection();
  1322. updateSelectionCount();
  1323. // Update modal title
  1324. const modalTitle = document.getElementById('modalTitle');
  1325. if (modalTitle) {
  1326. modalTitle.textContent = currentPlaylist ? `Edit Patterns for "${currentPlaylist}"` : 'Add Patterns to Playlist';
  1327. }
  1328. document.getElementById('addPatternsModal').classList.remove('hidden');
  1329. // Focus search input when modal opens
  1330. setTimeout(() => {
  1331. document.getElementById('patternSearchInput').focus();
  1332. }, 100);
  1333. });
  1334. // Search functionality
  1335. document.getElementById('patternSearchInput').addEventListener('input', handleSearchInput);
  1336. document.getElementById('clearSearchBtn').addEventListener('click', clearSearch);
  1337. // Sort and filter controls
  1338. document.getElementById('sortFieldSelect').addEventListener('change', handleSortFieldChange);
  1339. document.getElementById('sortDirectionBtn').addEventListener('click', handleSortDirectionToggle);
  1340. document.getElementById('categoryFilterSelect').addEventListener('change', handleCategoryFilterChange);
  1341. // Handle Enter key in search input
  1342. document.getElementById('patternSearchInput').addEventListener('keypress', (e) => {
  1343. if (e.key === 'Enter') {
  1344. e.preventDefault();
  1345. }
  1346. });
  1347. // Run playlist button
  1348. document.getElementById('runPlaylistBtn').addEventListener('click', runPlaylist);
  1349. // Modal controls
  1350. document.getElementById('cancelPlaylistBtn').addEventListener('click', () => {
  1351. document.getElementById('addPlaylistModal').classList.add('hidden');
  1352. });
  1353. document.getElementById('createPlaylistBtn').addEventListener('click', createNewPlaylist);
  1354. document.getElementById('cancelAddPatternsBtn').addEventListener('click', () => {
  1355. selectedPatterns.clear();
  1356. clearSearch();
  1357. updateSelectionCount();
  1358. document.getElementById('addPatternsModal').classList.add('hidden');
  1359. });
  1360. document.getElementById('confirmAddPatternsBtn').addEventListener('click', addSelectedPatternsToPlaylist);
  1361. // Smart Toggle Select All button
  1362. const toggleSelectBtn = document.getElementById('toggleSelectAllBtn');
  1363. if (toggleSelectBtn) {
  1364. toggleSelectBtn.addEventListener('click', toggleSelectAll);
  1365. }
  1366. // Handle Enter key in new playlist name input
  1367. document.getElementById('newPlaylistName').addEventListener('keypress', (e) => {
  1368. if (e.key === 'Enter') {
  1369. createNewPlaylist();
  1370. }
  1371. });
  1372. // Close modals when clicking outside
  1373. document.getElementById('addPlaylistModal').addEventListener('click', (e) => {
  1374. if (e.target.id === 'addPlaylistModal') {
  1375. document.getElementById('addPlaylistModal').classList.add('hidden');
  1376. }
  1377. });
  1378. document.getElementById('addPatternsModal').addEventListener('click', (e) => {
  1379. if (e.target.id === 'addPatternsModal') {
  1380. selectedPatterns.clear();
  1381. clearSearch();
  1382. document.getElementById('addPatternsModal').classList.add('hidden');
  1383. }
  1384. });
  1385. }
  1386. // Initialize playlists page
  1387. document.addEventListener('DOMContentLoaded', async () => {
  1388. try {
  1389. // Initialize intersection observer for lazy loading
  1390. initializeIntersectionObserver();
  1391. // Initialize IndexedDB preview cache
  1392. await initPreviewCacheDB();
  1393. // Setup event listeners
  1394. setupEventListeners();
  1395. // Initialize mobile view state
  1396. isMobileView = isMobile();
  1397. if (isMobileView) {
  1398. initMobileLayout();
  1399. } else {
  1400. initDesktopLayout();
  1401. }
  1402. // Add window resize listener for responsive behavior
  1403. window.addEventListener('resize', updateMobileView);
  1404. // Restore playback settings
  1405. restorePlaybackSettings();
  1406. setupPlaybackSettingsPersistence();
  1407. // Load playlists
  1408. await loadPlaylists();
  1409. // Check serial connection status
  1410. await checkSerialStatus();
  1411. logMessage('Playlists page initialized successfully', LOG_TYPE.SUCCESS);
  1412. } catch (error) {
  1413. logMessage(`Error during initialization: ${error.message}`, LOG_TYPE.ERROR);
  1414. showStatusMessage('Failed to initialize playlists page', 'error');
  1415. }
  1416. });
  1417. // Check serial connection status
  1418. async function checkSerialStatus() {
  1419. try {
  1420. const response = await fetch('/serial_status');
  1421. if (response.ok) {
  1422. const data = await response.json();
  1423. const statusDot = document.getElementById('connectionStatusDot');
  1424. if (statusDot) {
  1425. statusDot.className = `inline-block size-2 rounded-full ml-2 align-middle ${
  1426. data.connected ? 'bg-green-500' : 'bg-red-500'
  1427. }`;
  1428. }
  1429. }
  1430. } catch (error) {
  1431. logMessage(`Error checking serial status: ${error.message}`, LOG_TYPE.ERROR);
  1432. }
  1433. }
  1434. // Mobile utility functions
  1435. function isMobile() {
  1436. return window.innerWidth <= 768;
  1437. }
  1438. function updateMobileView() {
  1439. const wasMobile = isMobileView;
  1440. isMobileView = isMobile();
  1441. if (wasMobile !== isMobileView) {
  1442. // Mobile state changed, update layout
  1443. if (isMobileView) {
  1444. initMobileLayout();
  1445. } else {
  1446. initDesktopLayout();
  1447. }
  1448. }
  1449. }
  1450. function initMobileLayout() {
  1451. const sidebar = document.getElementById('playlistsSidebar');
  1452. const details = document.getElementById('playlistDetails');
  1453. const mobileBackBtn = document.getElementById('mobileBackBtn');
  1454. if (!currentPlaylist) {
  1455. // Show playlists list, hide details
  1456. sidebar.classList.remove('mobile-hidden');
  1457. details.classList.add('mobile-hidden');
  1458. mobileBackBtn.classList.add('mobile-hidden');
  1459. } else {
  1460. // Show details, hide playlists list
  1461. sidebar.classList.add('mobile-hidden');
  1462. details.classList.remove('mobile-hidden');
  1463. mobileBackBtn.classList.remove('mobile-hidden');
  1464. mobileBackBtn.classList.add('mobile-flex');
  1465. }
  1466. }
  1467. function initDesktopLayout() {
  1468. const sidebar = document.getElementById('playlistsSidebar');
  1469. const details = document.getElementById('playlistDetails');
  1470. const mobileBackBtn = document.getElementById('mobileBackBtn');
  1471. // Show both sidebar and details on desktop
  1472. sidebar.classList.remove('mobile-hidden');
  1473. details.classList.remove('mobile-hidden');
  1474. mobileBackBtn.classList.add('mobile-hidden');
  1475. mobileBackBtn.classList.remove('mobile-flex');
  1476. }
  1477. function showPlaylistDetails() {
  1478. if (isMobileView) {
  1479. const sidebar = document.getElementById('playlistsSidebar');
  1480. const details = document.getElementById('playlistDetails');
  1481. const mobileBackBtn = document.getElementById('mobileBackBtn');
  1482. sidebar.classList.add('mobile-hidden');
  1483. details.classList.remove('mobile-hidden');
  1484. mobileBackBtn.classList.remove('mobile-hidden');
  1485. mobileBackBtn.classList.add('mobile-flex');
  1486. }
  1487. }
  1488. function showPlaylistsList() {
  1489. if (isMobileView) {
  1490. const sidebar = document.getElementById('playlistsSidebar');
  1491. const details = document.getElementById('playlistDetails');
  1492. const mobileBackBtn = document.getElementById('mobileBackBtn');
  1493. sidebar.classList.remove('mobile-hidden');
  1494. details.classList.add('mobile-hidden');
  1495. mobileBackBtn.classList.add('mobile-hidden');
  1496. mobileBackBtn.classList.remove('mobile-flex');
  1497. }
  1498. }