playlists.js 53 KB

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