🏠 Home 

Greasy Fork is available in English.

Google Infinite Scroll

-

// ==UserScript==
// @name:ko           구글 무한 스크롤
// @name              Google Infinite Scroll
// @description:ko    -
// @description       -
// @namespace         https://ndaesik.tistory.com/
// @version           2025.01.10.08.44
// @author            ndaesik
// @icon              https://t1.gstatic.com/faviconV2?client=SOCIAL&type=FAVICON&fallback_opts=TYPE,SIZE,URL&url=https://www.google.com
// @match             *://www.google.com/search*
// @grant             GM_xmlhttpRequest
// @run-at            document-end
// @connect           *
// ==/UserScript==
if(new URL(window.location.href).searchParams.has('udm')) return;
document.head.appendChild(Object.assign(document.createElement('style'), {
textContent: `
#botstuff [role="navigation"] { display: none !important; }
.youtube-thumbnail {
object-fit: cover !important;
width:100%;
height:100%;
}
img[src="data:text/plain;base64,"] {
opacity: 0 !important;
}
`
}));
const PLACEHOLDER_SELECTOR = '[src=""]';
const getYoutubeVideoId = url => {
try {
const urlObj = new URL(url);
return urlObj.hostname.includes('youtube.com') ? urlObj.searchParams.get('v') : null;
} catch (error) {
return (console.error('Error parsing YouTube URL:', error), null);
}
};
async function replaceYoutubeThumbnail(imgElement) {
try {
const container = imgElement.closest('[data-curl*="youtube.com/watch"]');
if (!container) return;
const videoId = getYoutubeVideoId(container.getAttribute('data-curl'));
if (!videoId) return;
const dataUrl = await convertImageToDataUrl(`https://img.youtube.com/vi/${videoId}/sddefault.jpg`);
dataUrl && (imgElement.src = dataUrl, imgElement.classList.add('youtube-thumbnail'));
} catch (error) {
console.error('Error replacing YouTube thumbnail:', error);
imgElement.src.startsWith('data:text/plain;base64,') && (imgElement.style.opacity = '0');
}
}
async function convertImageToDataUrl(imageUrl) {
return new Promise(resolve =>
GM_xmlhttpRequest({
method: "GET",
url: imageUrl,
responseType: "blob",
onload: ({response}) => {
const reader = new FileReader();
reader.onloadend = () => resolve(reader.r###lt);
reader.readAsDataURL(response);
},
onerror: error => (console.error('Error fetching image:', error), resolve(null))
})
).catch(error => (console.error('Error converting image:', error), null));
}
async function getOGImage(url) {
return new Promise((resolve) =>
GM_xmlhttpRequest({
method: "GET",
url,
onload: ({responseText}) => {
const doc = new DOMParser().parseFromString(responseText, "text/html");
resolve(doc.querySelector('meta[property="og:image"]')?.content || doc.querySelector('img[src^="http"]')?.src || null);
},
onerror: (error) => (console.error('Error fetching page:', error), resolve(null))
})
).catch(() => null);
}
async function replacePlaceholderImage(imgElement) {
try {
let url, ogImageUrl, dataUrl;
const youtubeContainer = imgElement.closest('[data-curl*="youtube.com/watch"]');
youtubeContainer ? (await replaceYoutubeThumbnail(imgElement), imgElement.classList.add('youtube-thumbnail')) :
(url = imgElement.parentElement?.parentElement?.parentElement?.parentElement?.querySelector('a[data-ved]')?.href) ?
(ogImageUrl = await getOGImage(url)) ?
(dataUrl = await convertImageToDataUrl(ogImageUrl)) ? imgElement.src = dataUrl : null
: null
: null;
} catch (error) { console.debug(error); }
}
const fetchNextPage = async pageNumber => {
const baseUrl = new URL(window.location.href);
const text = await (await fetch(`${baseUrl.origin}${baseUrl.pathname}?q=${baseUrl.searchParams.get('q')}&start=${pageNumber * 10}`)).text();
const newDoc = new DOMParser().parseFromString(text, 'text/html');
const container = document.createElement('div');
container.id = `page-${pageNumber}`;
container.style.cssText = 'margin-top: 20px;';
newDoc.querySelectorAll('#rso > div').forEach(r###lt => container.appendChild(r###lt.cloneNode(true)));
const lastAddedPage = document.querySelector(`#page-${pageNumber - 1}`) || document.querySelector('#botstuff');
lastAddedPage.after(container);
const newPlaceholders = container.querySelectorAll(PLACEHOLDER_SELECTOR);
const youtubeR###lts = container.querySelectorAll('[data-curl*="youtube.com/watch"]');
if (youtubeR###lts.length > 0) {
newPlaceholders.forEach(replacePlaceholderImage);
}
return !!newDoc.querySelector('#pnnext');
};
let [pageNumber, isLoading, hasMore] = [1, false, true];
window.addEventListener('scroll', async () => {
if (!isLoading && hasMore && window.innerHeight + window.pageYOffset >= document.documentElement.offsetHeight - 1000) {
isLoading = true;
hasMore = await fetchNextPage(pageNumber);
pageNumber += hasMore ? 1 : 0;
isLoading = false;
}
});