Files
WaifuBoard/src/extensions/load-extensions.js
itsskaiya 04f37218de Updated headless browser to support dynamic sites
Removed tabs and moved over to pages
Updated the rendering system
Fixed multiple pages not loading on scroll and re-rending or not rendering anything or just page 1.
Fixed the search bar not taking in spaces for each query
Updated how extensions are made
Updated how extensions are loaded
2025-11-21 11:48:07 -05:00

135 lines
4.0 KiB
JavaScript

export async function populateSources(sourceList) {
console.log('Requesting sources from main process...');
const sources = await window.api.getSources();
sourceList.innerHTML = '';
let initialSource = '';
console.log("Raw sources received from backend:", sources);
if (sources && sources.length > 0) {
sources.forEach((source) => {
if (source.name.toLowerCase().includes('tenor')) {
if (source.type !== 'image-board') {
console.error(`CRITICAL: Tenor extension found, but type is "${source.type}". It will be hidden.`);
} else {
console.log("SUCCESS: Tenor extension passed type check.");
}
}
if (source.type !== 'image-board') {
return;
}
const button = document.createElement('button');
button.className = 'source-button hover:bg-gray-700 hover:text-white transition-all duration-200';
button.dataset.source = source.name;
button.title = source.name;
let mainDomain = source.url;
try {
const urlToParse = source.url || source.baseUrl || "";
if (urlToParse) {
const hostname = new URL(urlToParse).hostname;
const parts = hostname.split('.');
if (parts.length > 2 && ['api', 'www'].includes(parts[0])) {
mainDomain = parts.slice(1).join('.');
} else {
mainDomain = hostname;
}
} else {
mainDomain = source.name;
}
} catch (e) {
mainDomain = source.name;
}
const favicon = document.createElement('img');
favicon.className = 'source-icon rounded';
favicon.src = `https://www.google.com/s2/favicons?domain=${mainDomain}&sz=32`;
favicon.alt = source.name;
const textWrapper = document.createElement('div');
textWrapper.className = 'source-text-wrapper';
const nameSpan = document.createElement('span');
nameSpan.className = 'source-name';
nameSpan.textContent = source.name;
const urlSpan = document.createElement('span');
urlSpan.className = 'source-url';
urlSpan.textContent = mainDomain;
textWrapper.appendChild(nameSpan);
textWrapper.appendChild(urlSpan);
favicon.onerror = () => {
favicon.remove();
const fallbackIcon = document.createElement('div');
fallbackIcon.className = 'source-icon fallback';
fallbackIcon.textContent = source.name.substring(0, 1).toUpperCase();
button.insertBefore(fallbackIcon, textWrapper);
};
button.appendChild(favicon);
button.appendChild(textWrapper);
sourceList.appendChild(button);
});
if (sourceList.children.length > 0) {
const firstButton = sourceList.children[0];
firstButton.classList.add('active');
initialSource = firstButton.dataset.source;
} else {
console.warn("All sources were filtered out. Check 'type' property in your extensions.");
}
setupCarousel(sourceList);
} else {
console.warn('No sources loaded from API.');
}
return initialSource;
}
function setupCarousel(element) {
element.addEventListener('wheel', (evt) => {
if (evt.deltaY !== 0) {
if (element.scrollWidth > element.clientWidth) {
evt.preventDefault();
element.scrollLeft += evt.deltaY;
}
}
});
let isDown = false;
let startX;
let scrollLeft;
element.addEventListener('mousedown', (e) => {
isDown = true;
element.style.cursor = 'grabbing';
startX = e.pageX - element.offsetLeft;
scrollLeft = element.scrollLeft;
});
element.addEventListener('mouseleave', () => {
isDown = false;
element.style.cursor = 'grab';
});
element.addEventListener('mouseup', () => {
isDown = false;
element.style.cursor = 'grab';
});
element.addEventListener('mousemove', (e) => {
if (!isDown) return;
e.preventDefault();
const x = e.pageX - element.offsetLeft;
const walk = (x - startX) * 2;
element.scrollLeft = scrollLeft - walk;
});
element.style.cursor = 'grab';
}