Amazon - Goodreads metadata

Shows the ratings from Goodreads on Amazon book pages

Na nainštalovanie skriptu si budete musieť nainštalovať rozšírenie, ako napríklad Tampermonkey, Greasemonkey alebo Violentmonkey.

Na inštaláciu tohto skriptu je potrebné nainštalovať rozšírenie, ako napríklad Tampermonkey.

Na nainštalovanie skriptu si budete musieť nainštalovať rozšírenie, ako napríklad Tampermonkey, % alebo Violentmonkey.

Na nainštalovanie skriptu si budete musieť nainštalovať rozšírenie, ako napríklad Tampermonkey alebo Userscripts.

Na inštaláciu tohto skriptu je potrebné nainštalovať rozšírenie, ako napríklad Tampermonkey.

Na inštaláciu tohto skriptu je potrebné nainštalovať rozšírenie správcu používateľských skriptov.

(Už mám správcu používateľských skriptov, nechajte ma ho nainštalovať!)

Advertisement:

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie, ako napríklad Stylus.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie, ako napríklad Stylus.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie, ako napríklad Stylus.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie správcu používateľských štýlov.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie správcu používateľských štýlov.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie správcu používateľských štýlov.

(Už mám správcu používateľských štýlov, nechajte ma ho nainštalovať!)

Advertisement:

// ==UserScript==
// @name         Amazon - Goodreads metadata
// @namespace    https://github.com/bricemciver/GreasemonekeyScripts
// @version      1.0.2
// @author       Brice McIver
// @description  Shows the ratings from Goodreads on Amazon book pages
// @license      MIT
// @icon         https://www.google.com/s2/favicons?sz=64&domain=amazon.com
// @match        https://amazon.com/*
// @match        https://*.amazon.com/*
// @match        https://amazon.co.uk/*
// @match        https://*.amazon.co.uk/*
// @match        https://amazon.ca/*
// @match        https://*.amazon.ca/*
// @match        https://amazon.de/*
// @match        https://*.amazon.de/*
// @match        https://amazon.fr/*
// @match        https://*.amazon.fr/*
// @match        https://amazon.es/*
// @match        https://*.amazon.es/*
// @match        https://amazon.it/*
// @match        https://*.amazon.it/*
// @match        https://amazon.co.jp/*
// @match        https://*.amazon.co.jp/*
// @match        https://amazon.cn/*
// @match        https://*.amazon.cn/*
// @match        https://amazon.com.br/*
// @match        https://*.amazon.com.br/*
// @match        https://amazon.in/*
// @match        https://*.amazon.in/*
// @match        https://amazon.com.mx/*
// @match        https://*.amazon.com.mx/*
// @match        https://amazon.com.au/*
// @match        https://*.amazon.com.au/*
// @grant        GM.xmlHttpRequest
// @grant        GM_xmlhttpRequest
// ==/UserScript==

(function() {
var asinRegex = /^[A-Z0-9]{10}$/;
	var goodreadsRegex = /"aggregateRating":({"@type":"AggregateRating","ratingValue":.*?,"ratingCount":.*?,"reviewCount":.*?})/;
	var extractASINs = () => {
		const asins = [];
		const books = document.querySelectorAll("bds-unified-book-faceout");
		for (const item of Array.from(books)) {
			const asin = item.dataset.csaCItemId;
			if (asin && asinRegex.test(asin)) asins.push(asin);
		}
		const asinMeta = document.querySelector("div[data-asin]");
		if (asinMeta) {
			const asin = asinMeta.dataset.asin;
			if (asin && asinRegex.test(asin)) asins.push(asin);
		}
		return asins;
	};
	var fetchGoodreadsDataForASIN = (asin) => {
		return GM.xmlHttpRequest({
			method: "GET",
			url: `https://www.goodreads.com/book/isbn/${asin}`
		});
	};
	var insertGoodreadsData = (asin, goodreadsData) => {
		const container = document.createElement("div");
		container.style.padding = "6px";
		container.style.margin = "5px 0";
		container.style.backgroundColor = "#f8f8f8";
		container.style.border = "1px solid #ddd";
		container.style.borderRadius = "3px";
		let content = `<div style="display: flex; flex-direction: column; gap: 4px; margin-bottom: 2px;">
          <span><img src="https://www.goodreads.com/favicon.ico" style="width: 16px; height: 16px; margin-right: 3px;" alt="Goodreads" />
          <a href="${goodreadsData.bookUrl}" target="_blank" style="font-weight: bold;">Goodreads</a></span>`;
		if (goodreadsData.rating) content += `<span style="color: #000">${goodreadsData.rating} stars</span>`;
		if (goodreadsData.ratingCount) content += `<span style="white-space: nowrap;">${goodreadsData.ratingCount} ratings</span>`;
		if (goodreadsData.reviewCount) content += `<span style="white-space: nowrap;">${goodreadsData.reviewCount} reviews</span>`;
		content += "</div>";
		container.innerHTML = content;
		const currentBooks = document.querySelectorAll("bds-unified-book-faceout");
		for (const book of Array.from(currentBooks)) {
			const bookInfoDiv = book.shadowRoot?.querySelector("div[data-csa-c-item-id]");
			if (bookInfoDiv) {
				const bookAsin = bookInfoDiv.dataset.csaCItemId;
				if (bookAsin && bookAsin === asin) {
					const ratings = book.shadowRoot?.querySelector("div.star-rating");
					if (ratings) {
						ratings.parentNode?.insertBefore(container, ratings.nextSibling);
						break;
					}
				}
			}
		}
		const reviewElement = document.getElementById("reviewFeatureGroup");
		if (reviewElement) reviewElement.parentNode?.insertBefore(container, reviewElement.nextSibling);
	};
	var processAsins = async (asins) => {
		for (const asin of asins) try {
			const goodreadsData = await fetchGoodreadsDataForASIN(asin);
			const url = goodreadsData.finalUrl;
			const aggregateMatch = goodreadsRegex.exec(goodreadsData.responseText);
			if (aggregateMatch && aggregateMatch.length > 1) {
				const aggregateData = JSON.parse(aggregateMatch[1]);
				insertGoodreadsData(asin, {
					rating: aggregateData.ratingValue,
					ratingCount: aggregateData.ratingCount,
					reviewCount: aggregateData.reviewCount,
					bookUrl: url
				});
			}
		} catch (error) {
			console.error("Error fetching Goodreads data:", error);
		}
	};
	var init = async () => {
		const asins = extractASINs();
		if (!asins || asins.length === 0) return;
		try {
			await processAsins(asins);
		} catch (error) {
			console.error("Error in Goodreads script:", error);
		}
	};
	init();
})();