// ==UserScript==
// @name Amazon Kindle Deals Goodreads Ratings (Per Section)
// @license MIT-0
// @namespace http://tampermonkey.net/
// @version 2.5.2
// @description Add Goodreads ratings to Amazon Kindle deals page for specific sections with highlighting
// @match https://www.amazon.com/*
// @grant GM_xmlhttpRequest
// @run-at document-idle
// ==/UserScript==
(function() {
'use strict';
// Configurable variables
let debugMode = true;
// Rating thresholds and colors
const highRatingThresholds = [4.00, 4.30, 4.50];
const highRatingColors = ['#e6ffe6', '#ccffcc', '#99ff99']; // Light to dark green
const lowRatingThresholds = [3.60, 3.30];
const lowRatingColors = ['#ffe6e6', '#ffcccc']; // Light to medium red
// Review count thresholds and colors
const highRatingsCountThresholds = [1000, 5000, 10000];
const highRatingsCountColors = ['#e6ffe6', '#ccffcc', '#99ff99']; // Light to dark green
const lowRatingsCountThresholds = [100, 10];
const lowRatingsCountColors = ['#ffe6e6', '#ffcccc']; // Light to medium red
const longTitleLength = 42;
let bookData = [];
let processedASINs = new Set();
let linksToProcess = [];
let currentLinkIndex = 0;
let isPaused = false;
let isProcessing = false;
function log(message) {
if (debugMode) {
console.log(`[Goodreads Ratings Debug]: ${message}`);
}
}
function decodeHTMLEntities(text) {
const textArea = document.createElement('textarea');
textArea.innerHTML = text;
return textArea.value;
}
function getASIN(url) {
const match = url.match(/\/([A-Z0-9]{10})(?:\/|\?|$)/);
return match ? match[1] : null;
}
function extractYear(text) {
const yearRegex = /\b\d{4}\b/; // Regular expression to match a four-digit year
const match = text.match(yearRegex);
return match ? parseInt(match[0]) : '-';
}
function extractNumberOfPages(text) {
const pagesRegex = /(\d{1,3}(?:,\d{3})*)\s*(?:page|pages)/i; // This regex looks for one or more digits (with optional commas) followed by "page" or "pages"
const match = text.match(pagesRegex);
return match ? parseInt(match[1].replace(/,/g, ''), 10) : '-';
}
function extractLastNumber(str) {
// First, try to find a number after '#'
const hashMatch = str.match(/#\s*(\d+)\s*$/);
if (hashMatch) {
return parseInt(hashMatch[1]);
}
// If no '#' found, find the last number in the string
const allNumbers = str.match(/\d+/g);
if (allNumbers) {
return parseInt(allNumbers[allNumbers.length - 1]);
}
// If no numbers found at all
return null;
}
function isShelved(container) {
const buttons = container.querySelectorAll('button');
return Array.from(buttons).some(button => {
const label = button.getAttribute('aria-label');
return label && label.includes('Shelved');
});
}
function getLiteraryAwards(doc) {
if (doc) {
const scripts = doc.getElementsByTagName('script');
let awardsData = null;
// Iterate through scripts to find the one containing awards data
for (let script of scripts) {
const content = script.textContent || script.innerText;
if (content.includes('"awards":')) {
// This script likely contains our data
const match = content.match(/"awards":\s*"([^"]*)"/);
if (match) {
try {
awardsData = match[1];
break;
} catch (e) {
console.error("Error parsing awards data:", e);
}
}
}
}
return awardsData;
} else {
return null;
}
}
function getUniqueBookLinks(container) {
const uniqueLinks = [];
const seenHrefs = new Set();
// Kindle Deals page
const sections = container.querySelectorAll('div[data-testid="asin-face"], .ubf-book-info');
sections.forEach(asinFace => {
const link = asinFace.querySelector('a');
if (link && !seenHrefs.has(link.href)) {
seenHrefs.add(link.href);
uniqueLinks.push(link);
}
});
// Regular Kindle page
const bookFaceouts = container.querySelectorAll('bds-unified-book-faceout');
bookFaceouts.forEach(faceout => {
const shadowRoot = faceout.shadowRoot;
if (shadowRoot) {
const link = shadowRoot.querySelector('a');
if (link && !seenHrefs.has(link.href)) {
seenHrefs.add(link.href);
uniqueLinks.push(link);
}
}
});
log(`Unique link count: ${uniqueLinks.length}`);
return uniqueLinks;
}
function fetchGoodreadsData(asin) {
return new Promise((resolve, reject) => {
setTimeout(() => {
if (isPaused) {
resolve(null);
return;
}
log(`Fetching data for ASIN: ${asin}`);
GM_xmlhttpRequest({
method: "GET",
url: `https://www.goodreads.com/book/isbn/${asin}`,
onload: function(response) {
log(`Received response for ASIN: ${asin}`);
const parser = new DOMParser();
const doc = parser.parseFromString(response.responseText, "text/html");
// log(doc.documentElement.outerHTML);
const h1Element = doc.querySelector('h1[data-testid="bookTitle"]');
const metadataElement = doc.querySelector('.RatingStatistics__rating');
if (!h1Element || !metadataElement) {
log(`No results found for ASIN: ${asin}`);
resolve(null);
return;
}
const fullTitle = h1Element.textContent.trim();
const title = fullTitle.length > longTitleLength ? fullTitle.slice(0, longTitleLength) + '...' : fullTitle;
const rating = metadataElement.textContent.trim().replace(/\s*stars/, '');
const ratingsCountElement = doc.querySelector('[data-testid="ratingsCount"]');
const ratingsCount = ratingsCountElement ? ratingsCountElement.textContent.trim().split(' ')[0] : '0';
const reviewsCountElement = doc.querySelector('[data-testid="reviewsCount"]');
const reviewsCount = reviewsCountElement ? reviewsCountElement.textContent.trim().split(' ')[0] : '0';
// Is this part of a series?
const seriesElement = doc.querySelector('.BookPageTitleSection h3 a');
const series = seriesElement ? seriesElement.textContent.trim() : null;
const seriesLink = seriesElement ? seriesElement.href : null;
// Show a small image of the cover
const coverImageElement = doc.querySelector('.BookCover__image img');
const coverImage = coverImageElement ? coverImageElement.src : 'https://dryofg8nmyqjw.cloudfront.net/images/no-cover.png';
// Extract the first author
const authorElement = doc.querySelector('span.ContributorLink__name');
const author = authorElement ? authorElement.textContent.trim() : '-';
// Extract the first genre
const genreElement = doc.querySelector('.BookPageMetadataSection__genreButton a');
const genre = genreElement ? genreElement.textContent.trim() : '-';
// Extract the publication year
const publicationElement = doc.querySelector('p[data-testid="publicationInfo"]');
const publicationYear = publicationElement ? extractYear(publicationElement.textContent.trim()) : '';
// Extract the number of pages
const pagesElement = doc.querySelector('p[data-testid="pagesFormat"]');
const numberOfPages = pagesElement ? extractNumberOfPages(pagesElement.textContent.trim()) : '';
// Is it on a shelf already?
const actionsElement = doc.querySelector('.BookPageMetadataSection__mobileBookActions');
const onShelf = isShelved(actionsElement);
const awards = getLiteraryAwards(doc);
const data = {
asin: asin,
coverImage: coverImage,
title: title || "Unknown Title",
fullTitle: fullTitle,
longTitle: fullTitle.length > longTitleLength,
author: author,
series: series,
seriesLink: seriesLink,
rating: rating,
ratingsCount: ratingsCount,
reviewsCount: reviewsCount,
genre: genre,
goodreadsUrl: `https://www.goodreads.com/book/isbn/${asin}`,
publicationYear: publicationYear,
numberOfPages: numberOfPages,
onShelf: onShelf,
awards: awards
};
log(`Parsed data for ${data.title}: ${JSON.stringify(data)}`);
resolve(data);
},
onerror: function(error) {
log(`Error fetching data for ASIN: ${asin}`);
reject(error);
}
});
}, 250);
});
}
function getRatingColor(rating) {
rating = parseFloat(rating);
for (let i = highRatingThresholds.length - 1; i >= 0; i--) {
if (rating >= highRatingThresholds[i]) {
return highRatingColors[i];
}
}
for (let i = lowRatingThresholds.length - 1; i >= 0; i--) {
if (rating < lowRatingThresholds[i]) {
return lowRatingColors[i];
}
}
return '';
}
function getRatingsCountColor(count) {
count = parseInt(count.replace(/,/g, ''));
for (let i = highRatingsCountThresholds.length - 1; i >= 0; i--) {
if (count >= highRatingsCountThresholds[i]) {
return highRatingsCountColors[i];
}
}
for (let i = lowRatingsCountThresholds.length - 1; i >= 0; i--) {
if (count < lowRatingsCountThresholds[i]) {
return lowRatingsCountColors[i];
}
}
return '';
}
function addUIElement(books, isLoading = false) {
let container = document.getElementById('goodreads-ratings');
if (!container) {
container = document.createElement('div');
container.id = 'goodreads-ratings';
container.style.position = 'fixed';
container.style.top = '10px';
container.style.right = '10px';
container.style.backgroundColor = 'white';
container.style.padding = '10px';
container.style.border = '1px solid black';
container.style.zIndex = '9999';
container.style.maxHeight = '80vh';
container.style.overflowY = 'auto';
document.body.appendChild(container);
}
// Clear previous content
container.innerHTML = '';
const headerContainer = document.createElement('div');
headerContainer.style.display = 'flex';
headerContainer.style.justifyContent = 'space-between';
headerContainer.style.alignItems = 'center';
headerContainer.style.marginBottom = '10px';
const title = document.createElement('h3');
title.textContent = 'Goodreads Ratings';
title.style.margin = '0';
headerContainer.appendChild(title);
// Add pause/resume button only if there are links to process
if (linksToProcess.length > 0) {
const pauseResumeButton = document.createElement('button');
pauseResumeButton.textContent = isPaused ? 'Resume' : 'Pause';
pauseResumeButton.addEventListener('click', togglePauseResume);
headerContainer.appendChild(pauseResumeButton);
}
container.appendChild(headerContainer);
const statusMessage = document.createElement('p');
if (isPaused) {
statusMessage.textContent = 'Processing paused';
} else if (isLoading) {
statusMessage.textContent = `Processing... (${currentLinkIndex} of ${linksToProcess.length} books processed)`;
} else if (books.length === 0) {
statusMessage.textContent = 'No books processed yet';
} else {
statusMessage.textContent = 'Processing finished!';
}
container.appendChild(statusMessage);
const table = document.createElement('table');
table.style.borderCollapse = 'collapse';
table.style.width = '100%';
//table.style.tableLayout = 'fixed'; // This helps maintain consistent column widths
// Add a style for all cells
const cellStyle = `
border: 1px solid gray;
padding: 5px;
vertical-align: middle;
`;
// Create table header
const thead = document.createElement('thead');
const headerRow = document.createElement('tr');
['Cover', 'Title', 'Author', 'Price', 'Rating', 'Rating Count', 'Review Count', 'Genre', 'Year', 'Pages'].forEach(headerText => {
const th = document.createElement('th');
th.textContent = headerText;
th.style.cssText = cellStyle + `
font-weight: bold;
background-color: #f2f2f2;
`;
headerRow.appendChild(th);
});
thead.appendChild(headerRow);
table.appendChild(thead);
// Create table body
const tbody = document.createElement('tbody');
books.forEach(book => {
if (book) {
const row = document.createElement('tr');
// Cover image cell
const coverCell = document.createElement('td');
coverCell.style.cssText = cellStyle + `
text-align: center;
width: 30px;
`;
const coverImg = document.createElement('img');
coverImg.src = book.coverImage;
coverImg.alt = `${book.title} cover`;
coverImg.style.width = '28px';
coverImg.style.height = 'auto';
coverImg.style.display = 'block';
coverImg.style.margin = '0 auto'; // Centers the image horizontally
coverImg.onerror = function() {
this.onerror = null;
this.src = '';
this.alt = 'Cover not available';
};
coverCell.appendChild(coverImg);
row.appendChild(coverCell);
// Title cell
const titleCell = document.createElement('td');
if (book.series) {
const seriesNumber = extractLastNumber(book.series);
const seriesLink = document.createElement('a');
seriesLink.href = book.seriesLink;
seriesLink.target = '_blank';
seriesLink.textContent = seriesNumber ? `📚${seriesNumber}` : '📚';
seriesLink.title = book.series;
titleCell.appendChild(seriesLink);
// Add small span separator
const span = document.createElement('span');
span.textContent = ' | ';
titleCell.appendChild(span);
}
const link = document.createElement('a');
link.href = book.goodreadsUrl;
link.target = '_blank';
link.textContent = book.onShelf ? `⭐ ${book.title}` : book.title;
if (book.longTitle) {
link.title = decodeHTMLEntities(book.fullTitle);
}
titleCell.appendChild(link);
titleCell.style.cssText = cellStyle;
row.appendChild(titleCell);
// Author cell
const authorCell = document.createElement('td');
authorCell.textContent = book.author;
authorCell.style.cssText = cellStyle;
row.appendChild(authorCell);
// Price cell
const priceCell = document.createElement('td');
const priceLink = document.createElement('a');
priceLink.href = `https://www.amazon.com/dp/${book.asin}`;
priceLink.target = '_blank';
priceLink.textContent = book.price.replace(/^(?!\$)/, '$') || 'N/A'; // Add leading $ sign
if (book.awards) {
priceLink.textContent = `🏅 ${priceLink.textContent}`;
priceLink.title = decodeHTMLEntities(book.awards);
}
priceCell.appendChild(priceLink);
priceCell.style.cssText = cellStyle + 'text-align: right;';
row.appendChild(priceCell);
// Rating cell
const ratingCell = document.createElement('td');
ratingCell.textContent = book.rating;
ratingCell.style.cssText = cellStyle + `
text-align: right;
background-color: ${getRatingColor(book.rating)};
`;
row.appendChild(ratingCell);
// Ratings count cell
const ratingsCountCell = document.createElement('td');
ratingsCountCell.textContent = book.ratingsCount;
ratingsCountCell.style.cssText = cellStyle + `
text-align: right;
background-color: ${getRatingsCountColor(book.ratingsCount)};
`;
row.appendChild(ratingsCountCell);
// Reviews count cell
const reviewsCountCell = document.createElement('td');
reviewsCountCell.textContent = book.reviewsCount;
reviewsCountCell.style.cssText = cellStyle + 'text-align: right;';
row.appendChild(reviewsCountCell);
// Genre cell
const genreCell = document.createElement('td');
genreCell.textContent = book.genre;
genreCell.style.cssText = cellStyle;
row.appendChild(genreCell);
// Genre cell
const publicationCell = document.createElement('td');
publicationCell.textContent = book.publicationYear;
publicationCell.style.cssText = cellStyle;
row.appendChild(publicationCell);
// Pages cell
const pagesCell = document.createElement('td');
pagesCell.textContent = book.numberOfPages.toLocaleString('en-US');
pagesCell.style.cssText = cellStyle + 'text-align: right;';
row.appendChild(pagesCell);
tbody.appendChild(row);
}
});
table.appendChild(tbody);
container.appendChild(table);
}
function addBookAndSort(newBook) {
if (newBook && !processedASINs.has(newBook.asin)) {
bookData.push(newBook);
processedASINs.add(newBook.asin);
bookData.sort((a, b) => parseFloat(b.rating) - parseFloat(a.rating));
addUIElement(bookData, linksToProcess.length > currentLinkIndex);
}
}
async function processBooks() {
while (currentLinkIndex < linksToProcess.length && !isPaused) {
const link = linksToProcess[currentLinkIndex];
const asin = getASIN(link.href);
currentLinkIndex++;
if (asin && !processedASINs.has(asin)) {
try {
log(`---- Processing book ${currentLinkIndex} of ${linksToProcess.length} ----`);
const data = await fetchGoodreadsData(asin);
if (data) {
// Kindle Deals pages
const asinFace = link.closest('[data-testid="asin-face"]');
if (asinFace) {
const priceElement = asinFace && asinFace.querySelector('[data-testid="price"]');
if (priceElement) {
const priceTextContent = priceElement.textContent;
const priceMatch = priceTextContent.match(/Deal price: \$(\d+\.\d+)/);
data.price = priceMatch ? priceMatch[1] : 'N/A';
}
}
// Regular Kindle page
if (!data.price) {
const sibling = link.nextElementSibling;
if (sibling) {
const bookPrice = sibling.querySelector('bds-book-price');
if (bookPrice) {
// data.price = bookPrice.getAttribute('unstylizedprice'); -- old way
const shadowRoot = bookPrice.shadowRoot;
if (shadowRoot) {
const priceDiv = shadowRoot.querySelector('.offscreen'); // .price -> .offscreen
if (priceDiv) {
data.price = priceDiv.textContent.trim();
}
}
}
}
}
if (!data.price) {
data.price = 'N/A';
}
addBookAndSort(data);
}
} catch (error) {
console.error('Error fetching Goodreads data:', error);
}
}
}
if (currentLinkIndex >= linksToProcess.length) {
log('All books processed');
addUIElement(bookData, false);
}
}
function togglePauseResume() {
isPaused = !isPaused;
if (!isPaused) {
processBooks();
}
addUIElement(bookData, !isPaused);
}
function addButtonToSection(section) {
const button = document.createElement('button');
button.textContent = 'Get Goodreads Ratings';
button.style.margin = '10px';
button.addEventListener('click', async function() {
this.disabled = true;
const newLinks = getUniqueBookLinks(section);
linksToProcess.push(...newLinks.filter(link => !processedASINs.has(getASIN(link.href))));
if (!isProcessing) {
isProcessing = true;
addUIElement(bookData, true);
try {
await processBooks();
} finally {
isProcessing = false;
}
}
});
section.insertBefore(button, section.firstChild);
}
function initializeScript() {
const addButtonsToSections = () => {
const sections = document.querySelectorAll('div[data-testid="asin-faceout-shoveler.card-cont"]:not([data-goodreads-processed]), div[data-testid="mfs-container.hor-scroll"]:not([data-goodreads-processed])');
sections.forEach(section => {
addButtonToSection(section);
section.setAttribute('data-goodreads-processed', 'true');
});
if (sections.length > 0) {
log(`Buttons added to ${sections.length} new sections`);
}
};
// Initial run
addButtonsToSections();
// Set up a MutationObserver to watch for new sections
const observer = new MutationObserver((mutations) => {
mutations.forEach((mutation) => {
if (mutation.type === 'childList') {
addButtonsToSections();
}
});
});
observer.observe(document.body, {
childList: true,
subtree: true
});
log('Script initialized and watching for new sections');
}
// Run the script when the page is fully loaded
if (document.readyState === 'complete') {
initializeScript();
} else {
window.addEventListener('load', initializeScript);
}
log('Script setup complete');
})();