mobile ux
This commit is contained in:
@@ -238,12 +238,14 @@
|
||||
}
|
||||
function trackerFavicon(domain){
|
||||
if(!trackerFaviconsEnabled || !domain) return '<i class="fa-solid fa-bullseye"></i>';
|
||||
const src=`https://${encodeURIComponent(domain).replace(/%2E/g,'.')}/favicon.ico`;
|
||||
// Note: Favicony trackerów idą przez lokalny cache backendu, więc przeglądarka nie odpytuje tysięcy domen bezpośrednio.
|
||||
const src=`/api/trackers/favicon/${encodeURIComponent(domain)}`;
|
||||
return `<img class="tracker-favicon" src="${esc(src)}" alt="" loading="lazy" onerror="this.classList.add('d-none')"><i class="fa-solid fa-bullseye tracker-fallback-icon"></i>`;
|
||||
}
|
||||
function trackerFilterPlaceholder(){
|
||||
if(trackerSummaryStatus==='loading') return '<div class="tracker-filter-empty"><span class="spinner-border spinner-border-xs"></span> Loading trackers...</div>';
|
||||
if(trackerSummaryStatus==='error') return '<div class="tracker-filter-empty text-warning"><i class="fa-solid fa-triangle-exclamation"></i> Tracker list unavailable</div>';
|
||||
if(Number(trackerSummary.pending||0)) return `<div class="tracker-filter-empty"><span class="spinner-border spinner-border-xs"></span> Scanning cache: ${esc(trackerSummary.cached||0)}/${esc(trackerSummary.scanned||0)}</div>`;
|
||||
if(hasTorrentSnapshot && torrents.size) return '<div class="tracker-filter-empty">No trackers found</div>';
|
||||
return '<div class="tracker-filter-empty">Waiting for torrents...</div>';
|
||||
}
|
||||
@@ -261,22 +263,24 @@
|
||||
}
|
||||
async function refreshTrackerSummary(force=false){
|
||||
const hashes=[...torrents.keys()].sort();
|
||||
const sig=`${hashes.length}:${hashes.slice(0,2000).join(',')}:${trackerFaviconsEnabled?1:0}`;
|
||||
if(!force && sig===trackerSummarySignature) return;
|
||||
const sig=`${hashes.length}:${hashes[0]||''}:${hashes[hashes.length-1]||''}:${trackerFaviconsEnabled?1:0}`;
|
||||
if(!force && sig===trackerSummarySignature && !Number(trackerSummary.pending||0)) return;
|
||||
trackerSummarySignature=sig;
|
||||
if(!hashes.length){ trackerSummary={hashes:{},trackers:[],scanned:0,errors:[]}; trackerSummaryStatus='empty'; renderTrackerFilters(); return; }
|
||||
trackerSummaryStatus='loading';
|
||||
if(!hashes.length){ trackerSummary={hashes:{},trackers:[],scanned:0,errors:[],pending:0,cached:0}; trackerSummaryStatus='empty'; renderTrackerFilters(); return; }
|
||||
trackerSummaryStatus=(trackerSummary.trackers||[]).length?'ready':'loading';
|
||||
renderTrackerFilters();
|
||||
try{
|
||||
const qs=new URLSearchParams({limit:'2000'});
|
||||
// Note: Browser sends currently visible torrent hashes, avoiding an empty cache race on the backend.
|
||||
hashes.slice(0,2000).forEach(h=>qs.append('hash',h));
|
||||
const j=await (await fetch(`/api/trackers/summary?${qs.toString()}`)).json();
|
||||
if(!j.ok) throw new Error(j.error||'Tracker summary failed');
|
||||
trackerSummary=j.summary||{hashes:{},trackers:[],scanned:0,errors:[]};
|
||||
trackerSummaryStatus=(trackerSummary.trackers||[]).length?'ready':'empty';
|
||||
// Note: Nie wysyłamy 13k hashy w URL; backend bierze lokalny snapshot i doczytuje cache małymi porcjami.
|
||||
const j=await (await fetch('/api/trackers/summary?scan_limit=200')).json();
|
||||
if(!j.ok && !j.summary) throw new Error(j.error||'Tracker summary failed');
|
||||
trackerSummary=j.summary||{hashes:{},trackers:[],scanned:0,errors:[],pending:0,cached:0};
|
||||
trackerSummaryStatus=(trackerSummary.trackers||[]).length?'ready':Number(trackerSummary.pending||0)?'loading':'empty';
|
||||
renderTrackerFilters();
|
||||
scheduleRender(true);
|
||||
if(Number(trackerSummary.pending||0)>0){
|
||||
clearTimeout(trackerSummaryTimer);
|
||||
trackerSummaryTimer=setTimeout(()=>refreshTrackerSummary(true).catch(()=>{}), 3500);
|
||||
}
|
||||
}catch(e){ trackerSummaryStatus='error'; renderTrackerFilters(); console.warn('Tracker summary failed', e); }
|
||||
}
|
||||
function scheduleTrackerSummary(force=false){
|
||||
|
||||
Reference in New Issue
Block a user