Custom tags and improved rating display for eBird media.
// ==UserScript==
// @name eBird Acornizer
// @namespace https://github.com/balagansky/
// @version 2025-11-15
// @description Custom tags and improved rating display for eBird media.
// @author Ruslan Balagansky
// @license MIT
// @match https://media.ebird.org/catalog*
// @match https://macaulaylibrary.org/asset/*
// @icon 
// @grant GM.getValue
// @grant GM.setValue
// @grant GM.listValues
// @grant GM.registerMenuCommand
// @grant GM.setClipboard
// @run-at document-end
// ==/UserScript==
var settings;
var favorites;
var goods;
var alternates;
var funnies;
var stares;
const cMaxAutoLoad = 500;
// import/export functions
function importFavorites() {
var dataStr = prompt("Enter data (exported with Export function): ");
try {
var data = JSON.parse(dataStr);
for (const key in data) {
GM.setValue(key, data[key]);
console.log("Saved data for key: " + key);
}
alert("Data imported! Refresh the page to see changes.");
} catch (e) {
alert("Failed to import data. Please check the format and try again.");
}
}
function exportFavorites() {
GM.listValues().then(async function(keys) {
var data = {};
for (const key of keys) {
data[key] = await GM.getValue(key);
console.log("Loaded data for key: " + key);
}
GM.setClipboard(JSON.stringify(data), "text").then(function() {
alert("Exported data to clipboard.");
});
});
}
GM.registerMenuCommand("Import Tags", importFavorites);
GM.registerMenuCommand("Export Tags", exportFavorites);
async function readFromStorage(key)
{
try {
// GM.getValue returns the stored value directly; don't JSON.parse it
var readResult = await GM.getValue(key);
var valueStr = JSON.stringify(readResult);
if (valueStr)
valueStr = valueStr.slice(0, 100) + "...";
console.log("read " + key + ": " + valueStr);
return readResult;
} catch (e) {
console.log("error reading " + key + ". Maybe not written yet?");
return null;
}
}
async function saveToStorage(key, value)
{
// stringify once for logging only
var valueStr = JSON.stringify(value);
if (valueStr)
valueStr = valueStr.slice(0, 100) + "...";
console.log("saving " + key + ": " + valueStr);
await GM.setValue(key, value);
console.log(key + " saved");
}
async function readSettings()
{
var readSettings = await readFromStorage("settings");
settings = readSettings || {
maxResults: 100
};
}
async function saveSettings()
{
await saveToStorage("settings", settings);
}
async function readFavorites()
{
var readFavorites = await readFromStorage("favorites");
favorites = new Set();
if (readFavorites) {
try {
favorites = new Set(readFavorites);
} catch {}
}
var readGoods = await readFromStorage("goods");
goods = new Set();
if (readGoods) {
try {
goods = new Set(readGoods);
} catch {}
}
var readAlternates = await readFromStorage("alternates");
alternates = new Set();
if (readAlternates) {
try {
alternates = new Set(readAlternates);
} catch {}
}
var readFunnies = await readFromStorage("funnies");
funnies = new Set();
if (readFunnies) {
try {
funnies = new Set(readFunnies);
} catch {}
}
var readStares = await readFromStorage("stares");
stares = new Set();
if (readStares) {
try {
stares = new Set(readStares);
} catch {}
}
}
async function saveFavorites()
{
await saveToStorage("favorites", Array.from(favorites));
await saveToStorage("goods", Array.from(goods));
await saveToStorage("alternates", Array.from(alternates));
await saveToStorage("funnies", Array.from(funnies));
await saveToStorage("stares", Array.from(stares));
}
async function readStorage()
{
await readSettings();
await readFavorites();
}
function isViewSupported() {
var resultsGrid = document.getElementsByClassName("ResultsGrid");
if (resultsGrid.length == 0) {
console.log("Only grid views are supported.");
return false;
}
return true;
}
var results = [];
var resultIds = new Set();
var resultOrigOrder = {};
var resultRatings = {};
function clearResults() {
results = [];
resultIds = new Set();
resultOrigOrder = {};
}
var additionalLoadCount = 0;
const cImagesPerLoad = 30;
function loadMoreResults() {
if (results.length == 0)
return;
var pagination = document.getElementsByClassName("pagination")[0];
for (var pagChild of pagination.childNodes) {
if (pagChild.type == "button") {
if (results.length >= settings.maxResults) {
console.log("Result limit reached.");
} else if (additionalLoadCount > settings.maxResults / cImagesPerLoad) {
console.log("Safety additional load limit reached");
} else if (isViewSupported()) {
console.log("loading more results");
pagChild.click();
additionalLoadCount += 1;
}
break;
}
}
}
function getResultId(result) {
return result.querySelector("[data-asset-id]").getAttribute("data-asset-id");
}
function getNumRatings(result) {
var ratings = result.querySelector(".RatingStars-count");
if (!ratings)
return 0;
return Number(result.querySelector(".RatingStars-count").textContent.match(/\d+/));
}
function getStarRating(result) {
var stars = result.querySelector(".RatingStars");
if (!stars)
return 0;
return Number(stars.querySelector("[class=is-visuallyHidden]").textContent.match(/\d+/));
}
function getCheckboxState(result, checkboxClassName) {
var checkBox = result.getElementsByClassName(checkboxClassName)[0];
return checkBox && checkBox.checked;
}
function getAcornRating(result) {
if (getCheckboxState(result, "favCheck"))
return 3;
if (getCheckboxState(result, "goodCheck"))
return 2;
return 0;
}
function getAlternateRating(result) { return getCheckboxState(result, "altCheck");}
function getFunnyRating(result) { return getCheckboxState(result, "funnyCheck");}
function getStareRating(result) { return getCheckboxState(result, "stareCheck");}
function getOriginalOrder(result) {
return resultOrigOrder[getResultId(result)];
}
const cEncodedIcon = ''
var iconStyles = new Set();
function createAcornizerIconElement(size)
{
if (!iconStyles[size]) {
iconStyles.add(size);
const iconStyle = document.createElement('style')
iconStyle.textContent = `
span.acornizer-icon-${size}::before {
background-image: url("${cEncodedIcon}");
content: "";
background-repeat: no-repeat;
background-size: ${size}px ${size}px;
width: ${size}px;
height: ${size}px;
margin-right: 2px;
display: inline-block;
}`;
document.head.appendChild(iconStyle);
}
var span = document.createElement("span");
span.classList.add(`acornizer-icon-${size}`);
return span;
}
function readNewCards() {
var resultItems = document.getElementsByClassName("ResultsGrid-card");
var gotNewResult = false;
var cardOrder = 1;
for (var result of resultItems) {
const resultId = getResultId(result);
if (!resultIds.has(resultId))
{
resultOrigOrder[resultId] = cardOrder;
gotNewResult = true;
//console.log("num ratings " + getNumRatings(result));
//console.log("rating " + getStarRating(result));
resultIds.add(resultId);
// NOTE: cloning breaks site code. Have to manipulate in place.
results.push(result);
// add acornizer controls
var capDiv = result.getElementsByClassName("ResultsGrid-caption")[0];
if (capDiv.getElementsByClassName("favDiv").length == 0) {
// add fav div
var userDiv = capDiv.getElementsByClassName("userDateLoc")[0];
var favDiv = document.createElement("div");
favDiv.classList.add("favDiv");
capDiv.insertBefore(favDiv, userDiv);
favDiv.appendChild(createAcornizerIconElement(20));
var favCheck = document.createElement("input");
favCheck.classList.add("favCheck");
favCheck.setAttribute("type", "checkbox");
favCheck.checked = favorites.has(resultId);
favCheck.addEventListener("change", (e) => {
if (e.target.checked) {
favorites.add(resultId);
goods.delete(resultId);
alternates.delete(resultId);
e.target.parentElement.getElementsByClassName("goodCheck")[0].checked = false;
e.target.parentElement.getElementsByClassName("altCheck")[0].checked = false;
} else {
favorites.delete(resultId);
}
saveFavorites();
updateOrdering();
});
favDiv.appendChild(favCheck);
favDiv.appendChild(document.createTextNode("Favorite "));
var goodCheck = document.createElement("input");
goodCheck.classList.add("goodCheck");
goodCheck.setAttribute("type", "checkbox");
goodCheck.checked = goods.has(resultId);
goodCheck.addEventListener("change", (e) => {
if (e.target.checked) {
goods.add(resultId);
favorites.delete(resultId);
alternates.delete(resultId);
e.target.parentElement.getElementsByClassName("favCheck")[0].checked = false;
e.target.parentElement.getElementsByClassName("altCheck")[0].checked = false;
} else {
goods.delete(resultId);
}
saveFavorites();
updateOrdering();
});
favDiv.appendChild(goodCheck);
favDiv.appendChild(document.createTextNode("Good "));
var altCheck = document.createElement("input");
altCheck.classList.add("altCheck");
altCheck.setAttribute("type", "checkbox");
altCheck.checked = alternates.has(resultId);
altCheck.addEventListener("change", (e) => {
if (e.target.checked) {
alternates.add(resultId);
goods.delete(resultId);
favorites.delete(resultId);
e.target.parentElement.getElementsByClassName("goodCheck")[0].checked = false;
e.target.parentElement.getElementsByClassName("favCheck")[0].checked = false;
} else {
alternates.delete(resultId);
}
saveFavorites();
updateOrdering();
});
favDiv.appendChild(altCheck);
favDiv.appendChild(document.createTextNode("Alternate "));
var funnyCheck = document.createElement("input");
funnyCheck.classList.add("funnyCheck");
funnyCheck.setAttribute("type", "checkbox");
funnyCheck.checked = funnies.has(resultId);
funnyCheck.addEventListener("change", (e) => {
if (e.target.checked) {
funnies.add(resultId);
} else {
funnies.delete(resultId);
}
saveFavorites();
updateOrdering();
});
favDiv.appendChild(document.createTextNode(" | "));
favDiv.appendChild(funnyCheck);
favDiv.appendChild(document.createTextNode("Funny "));
var stareCheck = document.createElement("input");
stareCheck.classList.add("stareCheck");
stareCheck.setAttribute("type", "checkbox");
stareCheck.checked = stares.has(resultId);
stareCheck.addEventListener("change", (e) => {
if (e.target.checked) {
stares.add(resultId);
} else {
stares.delete(resultId);
}
saveFavorites();
updateOrdering();
});
favDiv.appendChild(stareCheck);
favDiv.appendChild(document.createTextNode("Staring"));
// add image url
var modifiedLibraryDiv = document.createElement("div");
modifiedLibraryDiv.style = "display: flex; justify-content: space-between";
var libraryAnchor = capDiv.lastChild;
capDiv.appendChild(modifiedLibraryDiv);
modifiedLibraryDiv.appendChild(libraryAnchor);
var customLibraryDiv = document.createElement("div");
customLibraryDiv.appendChild(createAcornizerIconElement(18));
var libraryImageUrl = document.createElement("a");
libraryImageUrl.href = `https://cdn.download.ams.birds.cornell.edu/api/v2/asset/${resultId}/2400`;
libraryImageUrl.target = "_blank";
libraryImageUrl.innerText = "Image Link";
customLibraryDiv.appendChild(libraryImageUrl);
modifiedLibraryDiv.appendChild(customLibraryDiv);
// add average rating (query api)
try {
// for result item
var modifiedRatingDiv = document.createElement("div");
modifiedRatingDiv.style = "display: flex; justify-content: space-between";
capDiv.insertBefore(modifiedRatingDiv, userDiv);
var ratingAnchor = capDiv.getElementsByClassName("RatingStars")[0];
modifiedRatingDiv.appendChild(ratingAnchor);
var customRatingDiv = document.createElement("div");
customRatingDiv.style = "display: flex";
customRatingDiv.appendChild(createAcornizerIconElement(18));
var avgRatingDiv = document.createElement("div");
avgRatingDiv.id = `avg${resultId}`;
avgRatingDiv.innerHTML = 'Avg: (loading...)';
customRatingDiv.appendChild(avgRatingDiv);
modifiedRatingDiv.appendChild(customRatingDiv);
fetch(`https://media.ebird.org/internal/v1/get-rating/${resultId}`)
.then(r => {
if (r.ok) {
return r.json();
}
throw new Error('rating query failed');
})
.then(data => {
var ratingDivToUpdate = document.getElementById(`avg${resultId}`);
var resultRatings = data[resultId];
avgRatingTxt = 'Avg: ' + parseFloat(resultRatings.ratingAverage.toFixed(3)).toString();
myRatingTxt = '';
if ("myRating" in resultRatings && resultRatings.myRating > 0) {
myRatingTxt = "My: " + resultRatings.myRating.toString();
}
ratingDivToUpdate.innerHTML = myRatingTxt + " | " + avgRatingTxt;
})
.catch(err => {
console.error('rating query failed');
});
} catch (e) {
// ignoring missing ratings, etc
}
}
}
cardOrder += 1;
}
return gotNewResult;
}
function acornSorted() {
return results.sort(function(a, b) {
if (settings.sortByFunny)
{
let aa = getFunnyRating(a);
let ba = getFunnyRating(b);
if (aa > ba)
return -1;
if (aa < ba)
return 1;
}
if (settings.sortByStare)
{
let aa = getStareRating(a);
let ba = getStareRating(b);
if (aa > ba)
return -1;
if (aa < ba)
return 1;
}
if (settings.sortByFavorites)
{
let aa = getAcornRating(a);
let ba = getAcornRating(b);
if (aa > ba)
return -1;
if (aa < ba)
return 1;
}
if (settings.sortByAlternates)
{
let aa = getAlternateRating(a);
let ba = getAlternateRating(b);
if (aa > ba)
return -1;
if (aa < ba)
return 1;
}
if (settings.sortByNumRatings)
{
let ar = getNumRatings(a);
let br = getNumRatings(b);
if (ar > br)
return -1;
if (ar < br)
return 1;
}
let as = getOriginalOrder(a);
let bs = getOriginalOrder(b);
if (as < bs)
return -1;
if (as > bs)
return 1;
console.log("oops?");
return 0;
});
}
function applyOrdering(containerElem, orderedElems) {
for (var elem of orderedElems.toReversed()) {
containerElem.insertBefore(elem, containerElem.firstChild);
}
}
function updateOrdering() {
console.log("reordering");
// rebuild results grid from saved results
var resultsGrid = document.getElementsByClassName("ResultsGrid")[0];
applyOrdering(resultsGrid, acornSorted());
}
function applyAcorns() {
if (!readNewCards())
return;
console.log("# results: " + results.length);
updateOrdering();
}
var resultsObserver = null;
function observeResults() {
if (!resultsObserver) {
resultsObserver = new MutationObserver(function(mutations) {
mutations.forEach(function(mutation) {
for (var addedNode of mutation.addedNodes) {
applyAcorns();
if (addedNode.type == "li") {
applyAcorns();
}
}
})
});
var resultsGrid = document.getElementsByClassName("ResultsGrid")[0];
resultsObserver.observe(resultsGrid, { childList: true });
}
}
function processSearchResults() {
if (!isViewSupported())
return;
observeResults();
applyAcorns();
loadMoreResults();
}
function refreshView() {
if (!isViewSupported())
return;
clearResults();
processSearchResults();
updateOrdering();
}
var wasViewSupported = isViewSupported();
function observePageChanges() {
var pagination = document.getElementsByClassName("pagination")[0];
var paginationObserver = new MutationObserver(function(mutations) {
mutations.forEach(function(mutation) {
for (var addedNode of mutation.addedNodes) {
if (addedNode.type == "button") {
if (!isViewSupported())
return;
loadMoreResults();
}
}
})
});
paginationObserver.observe(pagination, { childList: true });
var viewObserver = new MutationObserver(function(mutations) {
if (wasViewSupported != isViewSupported())
{
console.log("view change");
if (isViewSupported()) {
refreshView();
}
addSettings();
}
wasViewSupported = isViewSupported();
});
viewObserver.observe(document, { childList: true, subtree: true });
function observeFilterElement(element, observeAttributes = false) {
//console.log("observing " + element.textContent);
var filterSpanObserver = new MutationObserver(function (mutations) {
console.log("filter change");
clearResults();
loadMoreResults();
});
filterSpanObserver.observe(element, {
characterData: true, attributes: observeAttributes, childList: false, subtree: true
});
filterSpanObservers.push(filterSpanObserver);
}
var activeFiltersDiv = document.getElementsByClassName("ActiveFilters")[0];
var filterSpanObservers = [];
var filterObserver = new MutationObserver(function(mutations) {
console.log("resetting results");
for (var mutation of mutations) {
for (var addedNode of mutation.addedNodes) {
for (var span of addedNode.getElementsByTagName("span")) {
observeFilterElement(span);
}
}
}
clearResults();
loadMoreResults();
});
filterObserver.observe(activeFiltersDiv, { childList: true, subtree: true });
for (let span of activeFiltersDiv.getElementsByTagName("span")) {
observeFilterElement(span);
}
var filtersDiv = document.getElementsByClassName("filters")[0];
var currentSortDiv = filtersDiv.getElementsByClassName("filterSection--last")[0];
for (let span of currentSortDiv.getElementsByTagName("span")) {
observeFilterElement(span);
}
// update when switching media type (birds vs habitats etc)
var tabsDiv = document.getElementsByClassName("tabs")[0];
for (let button of tabsDiv.getElementsByTagName("button")) {
observeFilterElement(button, true);
}
}
function addSettings()
{
var existingSettingsDiv = document.getElementById("settingsDiv");
if (existingSettingsDiv)
existingSettingsDiv.parentElement.removeChild(existingSettingsDiv);
var resultsGrid = document.getElementsByClassName("ResultsGrid");
if (resultsGrid.length == 0)
return;
resultsGrid = resultsGrid[0];
var settingsDiv = document.createElement("div");
settingsDiv.id = "settingsDiv";
resultsGrid.parentElement.insertBefore(settingsDiv, resultsGrid);
var maxResultsInput = document.createElement("input");
maxResultsInput.setAttribute("type", "number");
maxResultsInput.id = "maxResultsInput";
maxResultsInput.min = 1;
maxResultsInput.max = cMaxAutoLoad;
maxResultsInput.value = settings.maxResults;
maxResultsInput.addEventListener("change", (e) => {
var input = document.getElementById("maxResultsInput");
input.value = Math.min(input.max, Math.max(input.min, input.value));
updateSettings().then(loadMoreResults);
});
settingsDiv.appendChild(createAcornizerIconElement(25));
settingsDiv.appendChild(document.createTextNode("Auto-Load Results: "));
settingsDiv.appendChild(maxResultsInput);
settingsDiv.appendChild(document.createTextNode(" "));
var favSortCheck = document.createElement("input");
favSortCheck.id = "favSortCheck";
favSortCheck.setAttribute("type", "checkbox");
favSortCheck.checked = settings.sortByFavorites;
favSortCheck.addEventListener("change", () => {
updateSettings().then(updateOrdering);
});
settingsDiv.appendChild(document.createTextNode("Sort by "));
settingsDiv.appendChild(favSortCheck);
settingsDiv.appendChild(document.createTextNode(" Favorites"));
var altSortCheck = document.createElement("input");
altSortCheck.id = "altSortCheck";
altSortCheck.setAttribute("type", "checkbox");
altSortCheck.checked = settings.sortByAlternates;
altSortCheck.addEventListener("change", () => {
updateSettings().then(updateOrdering);
});
settingsDiv.appendChild(document.createTextNode(", then by "));
settingsDiv.appendChild(altSortCheck);
settingsDiv.appendChild(document.createTextNode(" Alternates"));
var ratingCountSortCheck = document.createElement("input");
ratingCountSortCheck.id = "numRatingsSortCheck";
ratingCountSortCheck.setAttribute("type", "checkbox");
ratingCountSortCheck.checked = settings.sortByNumRatings;
ratingCountSortCheck.addEventListener("change", () => {
updateSettings().then(updateOrdering);
});
settingsDiv.appendChild(document.createTextNode(", then by "));
settingsDiv.appendChild(ratingCountSortCheck);
settingsDiv.appendChild(document.createTextNode(" # of ratings"));
settingsDiv.appendChild(document.createTextNode(". Show on top: "));
var funnySortcheck = document.createElement("input");
funnySortcheck.id = "funnySortCheck";
funnySortcheck.setAttribute("type", "checkbox");
funnySortcheck.checked = settings.sortByFunny;
funnySortcheck.addEventListener("change", (e) => {
if (e.target.checked) {
document.getElementById("stareSortCheck").checked = false;
}
updateSettings().then(updateOrdering);
});
settingsDiv.appendChild(funnySortcheck);
settingsDiv.appendChild(document.createTextNode(" Funny"));
var stareSortCheck = document.createElement("input");
stareSortCheck.id = "stareSortCheck";
stareSortCheck.setAttribute("type", "checkbox");
stareSortCheck.checked = settings.sortByStare;
stareSortCheck.addEventListener("change", (e) => {
if (e.target.checked) {
document.getElementById("funnySortCheck").checked = false;
}
updateSettings().then(updateOrdering);
});
settingsDiv.appendChild(document.createTextNode(", or "));
settingsDiv.appendChild(stareSortCheck);
settingsDiv.appendChild(document.createTextNode(" Staring."));
}
async function updateSettings()
{
await readSettings();
settings.maxResults = document.getElementById("maxResultsInput").value;
settings.sortByFavorites = document.getElementById("favSortCheck").checked;
settings.sortByAlternates = document.getElementById("altSortCheck").checked;
settings.sortByNumRatings = document.getElementById("numRatingsSortCheck").checked;
settings.sortByFunny = document.getElementById("funnySortCheck").checked;
settings.sortByStare = document.getElementById("stareSortCheck").checked;
await saveSettings();
}
function acornize() {
processSearchResults();
observePageChanges();
addSettings();
}
function displayRatingsOnAssetPage() {
var assetId = window.location.href.split("/asset/")[1].split("/")[0];
var ratingDiv = document.getElementsByClassName("Rating")[0];
var customRatingDiv = document.createElement("div");
customRatingDiv.appendChild(createAcornizerIconElement(20));
customRatingDiv.style = "display: flex; justify-content: space-between; margin-right: 4rem;";
var avgRatingDiv = document.createElement("div");
avgRatingDiv.id = "avgRating";
avgRatingDiv.innerHTML = 'Avg: (loading...)';
customRatingDiv.appendChild(avgRatingDiv);
ratingDiv.insertBefore(customRatingDiv, ratingDiv.firstChild);
fetch(`https://macaulaylibrary.org/internal/v1/get-rating/${assetId}`)
.then(r => {
if (r.ok) {
return r.json();
}
throw new Error('rating query failed');
})
.then(data => {
var ratingDivToUpdate = document.getElementById("avgRating");
var resultRatings = data[assetId];
var avgText = 'None';
if (resultRatings.ratingAverage > 0) {
avgText = parseFloat(resultRatings.ratingAverage.toFixed(3)).toString();
}
ratingDivToUpdate.innerHTML = 'Avg: ' + avgText;
})
.catch(err => {
console.error('rating query failed');
});
}
window.onload = function() {
if (window.location.href.includes("media.ebird.org/catalog")) {
readStorage().then(() => acornize());
} else if (window.location.href.includes("macaulaylibrary.org/asset/")) {
displayRatingsOnAssetPage();
}
};