// ==UserScript== // @name Cornucopia Enhancement Suite // @namespace github.com/lundgren // @version 0.0.8 // @description Variuos enhancements to Cornucopia.se // @author You // @match https://cornucopia.se/* // @icon https://www.google.com/s2/favicons?sz=64&domain=cornucopia.se // @grant GM.setValue // @grant GM.getValue // ==/UserScript== const DEFAULT_PREFERENCES = { hideFirstVisit: true, highlightParagraphs: true, highlightAdmin: true, favoriteMe: true, autoHideByDefault: false, jumpToFirstUnread: false, colorParagraphs: "#84dc23", colorComments: "#dc2328", colorFavorites: "#23dcd7", colorNavigation: "#7b23dc", favoriteWords: [], favoriteAuthors: [], hiddenAuthors: [], }; const SETTINGS_ICN = ``; const HIDE_COMMENTS_ICN = ``; const FAVORITE_COMMENTS_ICN = ``; const EMPTY_STAR_BASE64 = ""; const FILLED_STAR_BASE64 = ""; const ADMIN_COLOR = "#d3d3d3"; const StateFlags = { Read: 1 << 0, Hidden: 1 << 1, Favorite: 1 << 2, }; (async () => { const pageId = window.location.pathname; if (pageId === "/") { // We only enhance the article page for now return; } // Fetch info about previous visits const pageState = await readStorage(pageId, { readBefore: false }); const preferences = { ...DEFAULT_PREFERENCES, ...(await readStorage("preferences", {})) }; const articleReadBefore = pageState.readBefore; // Setup a state object to simplify working with flags const state = { isUnread: (id) => pageState[id] === undefined, isHidden: (id) => pageState[id] & StateFlags.Hidden, isFavorite: (id) => pageState[id] & StateFlags.Favorite, setAsRead: (id) => (pageState[id] |= StateFlags.Read), setHidden: (id, hidden) => { if (hidden) { pageState[id] |= StateFlags.Hidden; } else { pageState[id] &= ~StateFlags.Hidden; } }, setFavorite: (id, favorite) => { if (favorite) { pageState[id] |= StateFlags.Favorite; } else { pageState[id] &= ~StateFlags.Favorite; } }, isHideReadThreads: () => pageState.hideReadThreads, setHideReadThreads: (hide) => (pageState.hideReadThreads = hide), }; // Try to find the logged in user so we can automatically favorite their comments const loggedInUser = findLoggedInUser(); // Setup a canvas to draw the minimap on const $canvas = document.createElement("canvas"); document.body.appendChild($canvas); // Get a list of all comments and paragraphs const comments = getComments(); const paragraphs = getParagraphs(); const allItems = [...paragraphs, ...comments]; const redrawHighlightsAndMinimap = () => { drawHighlights(allItems); drawMinimap($canvas, allItems); }; // Add state to all comments and make them clickable for (const comment of comments) { comment.unread = state.isUnread(comment.id); comment.hasFavoriteWords = containsAny(comment.$el, preferences.favoriteWords); state.setAsRead(comment.id); const shouldAutoHideAuthor = preferences.hiddenAuthors.includes(comment.author); if (comment.unread && shouldAutoHideAuthor) { state.setHidden(comment.id, true); comment.hidden = true; } if (comment.unread && comment.author === loggedInUser && preferences.favoriteMe) { state.setFavorite(comment.id, true); } if (comment.unread && comment.hasFavoriteWords) { highlightWords(comment.$el, preferences.favoriteWords, preferences.colorFavorites); } comment.isHidden = () => state.isHidden(comment.id); comment.isFavorite = () => state.isFavorite(comment.id); comment.isParentFavorite = () => { let node = comment; while (node) { if (node.isFavorite()) { return true; } node = node.parent; } return false; }; // Return how the comment should be highlighted, undefined if not at all comment.highlightColor = () => { let color; if (comment.fromAdmin && preferences.highlightAdmin) { color = ADMIN_COLOR; } if (comment.unread) { if (articleReadBefore || !preferences.hideFirstVisit) { color = preferences.colorComments; } const isFavoriteAuthor = preferences.favoriteAuthors.includes(comment.author); if (isFavoriteAuthor || comment.isParentFavorite() || comment.hasFavoriteWords) { color = preferences.colorFavorites; } } return color; }; comment.setHidden = (hidden) => { state.setHidden(comment.id, hidden); writeStorage(pageId, pageState); }; comment.setFavorite = (favorite) => { state.setFavorite(comment.id, favorite); writeStorage(pageId, pageState); redrawHighlightsAndMinimap(); }; } // We cant count unread children until all comments have been processed let totalUnread = 0; for (const comment of comments) { totalUnread += comment.unread ? 1 : 0; comment.count = getCommentCount(comment); makeCommentClickable(comment); } // Add unread count to the comment count header updateCommentCount(totalUnread); // Add state to all paragraphs for (const paragraph of paragraphs) { paragraph.unread = state.isUnread(paragraph.id); state.setAsRead(paragraph.id); paragraph.highlightColor = () => { const hightlight = preferences.highlightParagraphs && paragraph.unread && (articleReadBefore || !preferences.hideFirstVisit); return hightlight ? preferences.colorParagraphs : undefined; }; } // Highlight all new content and comments drawHighlights(allItems); // Setup comments options const toggleReadThreads = (hide) => { state.setHideReadThreads(hide); writeStorage(pageId, pageState); for (const comment of comments) { const topCommentWithoutUnread = !comment.parent && comment.count.unread == 0; if (topCommentWithoutUnread) { if (hide) { comment.$el.temporarilyHideIt(); } else { comment.$el.temporarilyShowIt(); } } } }; const hideReadComments = state.isHideReadThreads() || (preferences.autoHideByDefault && !articleReadBefore); if (hideReadComments) { toggleReadThreads(true); } addOptionsControls(state.isHideReadThreads(), toggleReadThreads); // Add hotkeys to navigate between paragraphs and comments const jumpToFirstUnread = preferences.jumpToFirstUnread && articleReadBefore; enableHotkeyNavigationFor(allItems, preferences.colorNavigation, jumpToFirstUnread); // Make sure the minimap is redrawn when the window is resized const redrawMinimap = () => drawMinimap($canvas, allItems); new ResizeObserver(redrawMinimap).observe(document.body); addEventListener("resize", redrawMinimap); // Save the info about this visit pageState.lastReadTs = Date.now(); pageState.readBefore = true; writeStorage(pageId, pageState); })(); async function readStorage(key, defaultVal) { // Firefox if (typeof browser !== "undefined") { return (await browser.storage.local.get(key))[key] || defaultVal; } // Chromium based browsers if (typeof chrome !== "undefined") { return (await chrome.storage.local.get(key))[key] || defaultVal; } // Userscripts if (typeof GM !== "undefined") { return await GM.getValue(key, defaultVal); } throw new Error("No storage available"); } async function writeStorage(key, value) { // Firefox if (typeof browser !== "undefined") { return await browser.storage.local.set({ [key]: value }); } // Chromium based browsers if (typeof chrome !== "undefined") { return await chrome.storage.local.set({ [key]: value }); } // Userscripts if (typeof GM !== "undefined") { return await GM.setValue(key, value); } throw new Error("No storage available"); } function findLoggedInUser() { const $wpAdmin = document.getElementById("wp-admin-bar-my-account"); if ($wpAdmin) { return ($wpAdmin.getElementsByClassName("display-name")[0]?.innerText || "") .trim() .toLowerCase(); } return null; } function getDepth(element) { for (const className of element.classList) { if (className.startsWith("depth-")) { return parseInt(className.split("-")[1], 10); } } return 0; } function getCommentsChildren($el) { const $children = $el.getElementsByClassName(`comment depth-${getDepth($el) + 1}`); const children = []; for (let $child of $children) { children.push({ $el: $child, children: getCommentsChildren($child), }); } return children; } // getComments will return a list of all comments on the page function getComments() { const comments = []; const traverseAndSave = ($el, parent) => { const id = $el.id; const author = ($el.getElementsByClassName("author")[0]?.innerText || "").trim().toLowerCase(); const $commentText = $el.getElementsByClassName("comment-text")[0]; $commentText.style.transition = "color 500ms"; const comment = { id: id, $el: $commentText, type: "comment", author: author, fromAdmin: $el.classList.contains("bypostauthor") || $el.classList.contains("comment-author-admin"), children: [], parent: parent, }; comments.push(comment); const $children = $el.getElementsByClassName(`comment depth-${getDepth($el) + 1}`); for (let $child of $children) { comment.children.push(traverseAndSave($child, comment)); } return comment; }; const $commentsContainer = document.getElementById("comments"); if ($commentsContainer) { const $children = $commentsContainer.getElementsByClassName(`comment depth-1`); for (let $child of $children) { traverseAndSave($child, undefined); } } return comments; } // getParagraphs will return a list of all paragraphs function getParagraphs() { const paragraphs = []; const $post = document.getElementById("penci-post-entry-inner"); for (let $paragraph of $post.children) { if ($paragraph.nodeName === "P" || $paragraph.nodeName === "BLOCKQUOTE") { if ($paragraph.nodeName === "BLOCKQUOTE") { // Blockquotes uses the left border to indicate quotes, so we wrap it in a div we can style const $wrapper = document.createElement("div"); $paragraph.replaceWith($wrapper); $wrapper.append($paragraph); $paragraph = $wrapper; } paragraphs.push({ id: hashCode($paragraph.outerHTML), type: "paragraph", $el: $paragraph, }); } } // Add next post link as a paragraph const $nextPost = document.getElementsByClassName("post-pagination")[0]; if ($nextPost) { paragraphs.push({ id: hashCode($nextPost.innerText), type: "nextpost", $el: $nextPost, }); } return paragraphs; } function drawHighlights(items) { for (let item of items) { const color = item.highlightColor(); if (color) { item.$el.style.borderLeft = `3px solid ${color}`; item.$el.style.paddingLeft = "6px"; } else if (item.$el.style.borderLeft != "") { // Remove possible highlights item.$el.style.borderLeft = ""; item.$el.style.paddingLeft = ""; } } } function containsAny($el, words = []) { const $commentText = $el.getElementsByClassName("comment-content")[0]; if ($commentText) { const lowerText = $commentText.textContent.toLowerCase(); return words.some((word) => lowerText.includes(word)); } return false; } function highlightWords($el, words = [], color) { const $commentText = $el.getElementsByClassName("comment-content")[0]; for (let i = 0; i < $commentText.children.length; i++) { const $child = $commentText.children[i]; if ($child.tagName === "P") { for (let word of words) { $child.innerHTML = $child.innerHTML.replace( new RegExp(`(${word})`, "ig"), `$1` ); } } } } function updateCommentCount(count) { const $comments = document.getElementById("comments"); if ($comments) { const $commentCount = $comments.getElementsByClassName("post-box-title")[0]; if ($commentCount) { $commentCount.textContent = `${$commentCount.textContent} (${count} olästa)`; } } } function addOptionsControls(isChecked, onHideThreadsChanged) { const $comments = document.getElementById("comments"); var $commentList = $comments.getElementsByClassName("comments")[0]; if ($commentList) { // Settings button to open the options page const $settingsBtn = strToElement(SETTINGS_ICN); $settingsBtn.href = chrome.runtime.getURL("options.html"); $settingsBtn.addEventListener("click", () => { chrome.runtime.sendMessage({ action: "openOptionsPage" }); }); const $spacing = document.createElement("span"); $spacing.style.padding = "0 10px"; $spacing.appendChild(document.createTextNode(" | ")); // Checkbox to hide read threads const $checkbox = document.createElement("input"); $checkbox.type = "checkbox"; $checkbox.checked = isChecked; $checkbox.id = "hide-read-threads"; $checkbox.addEventListener("change", (e) => onHideThreadsChanged(e.target.checked)); const $label = document.createElement("label"); $label.htmlFor = "hide-read-threads"; $label.title = "This will hide all top level comments that have no unread replies to minimize distractions.\nIf a thread gets a reply in your next visit it will show up again."; $label.appendChild(document.createTextNode(" Hide threads without unread replies")); // Add the options to the DOM const $container = document.createElement("div"); $container.style.padding = "10px"; $container.appendChild($settingsBtn); $container.appendChild($spacing); $container.appendChild($checkbox); $container.appendChild($label); $comments.insertBefore($container, $commentList); } } // enableHotkeyNavigationFor will // - allow the user to navigate new comments and paragraphs using the J and K keys // - allow the user to navigate to the first new comment or paragraph using the I key // - allow the user to minimize navigated comments by pressing the M key function enableHotkeyNavigationFor(allItems, navigationColor, jumpToFirstUnread) { const entries = allItems.filter((item) => item.unread); let navigatedEntry; let styleBefore; addEventListener("scroll", (e) => { if (navigatedEntry) { navigatedEntry.$el.style.borderLeft = styleBefore; navigatedEntry = null; } }); // Scroll to center of element or top of element if it's too tall // Also highlight the element for a short time const scrollTo = (entry) => { const rect = entry.$el.getBoundingClientRect(); if (rect.height > window.innerHeight) { window.scrollTo(0, rect.top + window.scrollY - 50); } else { entry.$el.scrollIntoView({ block: "center" }); } setTimeout(() => { navigatedEntry = entry; styleBefore = entry.$el.style.borderLeft; entry.$el.style.borderLeft = `3px solid ${navigationColor}`; }, 50); }; const jumpToFirst = () => { for (let i = 0; i < entries.length; i++) { const isMinimized = entries[i].$el.getBoundingClientRect().height === 0; if (!isMinimized) { scrollTo(entries[i]); return; } } // If no unread, jump to comment section const $comments = document.getElementById("comments"); if ($comments) { let position = $comments.getBoundingClientRect(); window.scrollTo(0, position.top + window.scrollY - 50); } }; addEventListener("keydown", (e) => { const focused = document.activeElement && document.activeElement.nodeName; if (focused === "INPUT" || focused === "TEXTAREA") { return; } const center = window.innerHeight / 2; if (e.code === "KeyJ") { for (let i = 0; i < entries.length; i++) { const rect = entries[i].$el.getBoundingClientRect(); const isMinimized = rect.height === 0; const isBelowCenter = rect.top + rect.height / 2 - 50 > center; const isAlreadyNavigated = entries[i] === navigatedEntry; if (isBelowCenter && !isMinimized && !isAlreadyNavigated) { scrollTo(entries[i]); return; } } } else if (e.code === "KeyK") { for (let i = entries.length - 1; i >= 0; i--) { const rect = entries[i].$el.getBoundingClientRect(); const isMinimized = rect.height === 0; const isAboveCenter = rect.top + rect.height / 2 + 50 < center; const isAlreadyNavigated = entries[i] === navigatedEntry; if (isAboveCenter && !isMinimized && !isAlreadyNavigated) { scrollTo(entries[i]); return; } } } else if (e.code === "KeyI") { jumpToFirst(); } else if (e.code === "KeyM") { if (navigatedEntry && navigatedEntry.type === "comment") { navigatedEntry.$el.hideIt(); } } else if (e.code === "KeyN") { if (navigatedEntry && navigatedEntry.type === "comment") { navigatedEntry.$el.favoriteIt(); } } }); if (jumpToFirstUnread) { if ("scrollRestoration" in history) { history.scrollRestoration = "manual"; } jumpToFirst(); } } // drawMinimap will // - draw a minimap on the right side of the screen function drawMinimap($canvas, entries) { const scrollOffset = window.innerWidth - document.documentElement.clientWidth; $canvas.style = `position:fixed; left: ${ document.body.clientWidth - 5 }px; top: ${scrollOffset}px; width: 5px; height: ${ window.innerHeight - scrollOffset * 2 }px; z-index: 9999;`; $canvas.width = 5; $canvas.height = window.innerHeight; const q = window.innerHeight / document.documentElement.scrollHeight; var ctx = $canvas.getContext("2d"); ctx.clearRect(0, 0, $canvas.width, $canvas.height); for (const item of entries) { const color = item.highlightColor(); if (color) { const rect = item.$el.getBoundingClientRect(); ctx.fillStyle = color; ctx.fillRect(0, (window.scrollY + rect.top) * q, 5, rect.height * q); } } } function makeCommentClickable(comment) { const $container = comment.$el.parentElement.parentElement; const $hiddenContainer = createHiddenThreadContainer($container, comment); $container.parentNode.insertBefore($hiddenContainer, $container); const hideComments = (e) => { e?.preventDefault(); $container.style.display = "none"; $hiddenContainer.style.display = ""; comment.setHidden(true); }; const temporarilyHideComments = (e) => { e?.preventDefault(); $container.style.display = "none"; $hiddenContainer.style.display = ""; }; const showComments = (e) => { e?.preventDefault(); $container.style.display = ""; $hiddenContainer.style.display = "none"; comment.setHidden(false); }; const temporarilyShowComments = (e) => { e?.preventDefault(); $container.style.display = ""; $hiddenContainer.style.display = "none"; comment.setHidden(false); }; if (comment.isHidden()) { $container.style.display = "none"; } else { $hiddenContainer.style.display = "none"; } // Find author element and hide comments when it's clicked const $author = comment.$el.getElementsByClassName("author")[0]; $author.title = "Hide comments"; $author.style.cursor = "pointer"; $author.prepend(strToElement(HIDE_COMMENTS_ICN)); $author.addEventListener("click", hideComments); comment.$el.hideIt = hideComments; comment.$el.temporarilyHideIt = temporarilyHideComments; // Show comments again when hidden container is clicked $hiddenContainer.addEventListener("click", showComments); comment.$el.showIt = showComments; comment.$el.temporarilyShowIt = temporarilyShowComments; // Add favorite button const $favoriteBtn = strToElement(FAVORITE_COMMENTS_ICN); $favoriteBtn.title = "Favorite comments"; $favoriteBtn.style.cursor = "pointer"; $favoriteBtn.src = comment.isFavorite() ? FILLED_STAR_BASE64 : EMPTY_STAR_BASE64; $author.prepend($favoriteBtn); const toggleFavorite = (e) => { e?.stopImmediatePropagation(); if (comment.isFavorite()) { $favoriteBtn.src = EMPTY_STAR_BASE64; comment.setFavorite(false); } else { $favoriteBtn.src = FILLED_STAR_BASE64; comment.setFavorite(true); } }; $favoriteBtn.addEventListener("click", toggleFavorite); comment.$el.favoriteIt = toggleFavorite; } function createHiddenThreadContainer($container, comment) { const $hiddenContainer = document.createElement("div"); const $text = document.createElement("p"); $hiddenContainer.appendChild($text); const $authorContent = comment.$el.getElementsByClassName("author")[0]; const $commentContent = comment.$el.getElementsByClassName("comment-content")[0]; const summary = getSummary($authorContent, $commentContent); if (comment.count.unread > 0) { $text.innerText = `[${comment.count.total} comments, ${comment.count.unread} unread] - ${summary} (${comment.count.unread} unread)`; } else { $text.innerText = `[${comment.count.total} comments] - ${summary}`; } $hiddenContainer.className = $container.className; $hiddenContainer.title = "Show comments"; $hiddenContainer.style.cursor = "pointer"; $hiddenContainer.style.height = "30px"; $hiddenContainer.style.borderTop = "1px solid #dedede"; if ($container.className.includes("depth-1")) { $hiddenContainer.style.backgroundColor = "#fff9ec"; } return $hiddenContainer; } function getSummary($author, $comment) { const author = $author.innerText || $author.textContent || ""; const comment = $comment.innerText || $comment.textContent || ""; return `${author}: ${comment}`.replace(/(\r\n|\n|\r)/gm, " ").substring(0, 75) + "..."; } function getCommentCount(comment) { let total = 1; let unread = comment.unread ? 1 : 0; for (const child of comment.children) { const count = getCommentCount(child); total += count.total; unread += count.unread; } return { total, unread }; } function strToElement(str) { const temp = document.createElement("div"); temp.innerHTML = str; return temp.firstChild; } // https://stackoverflow.com/a/52171480 function hashCode(str) { let h1 = 0xdeadbeef ^ 0, h2 = 0x41c6ce57 ^ 0; for (let i = 0, ch; i < str.length; i++) { ch = str.charCodeAt(i); h1 = Math.imul(h1 ^ ch, 2654435761); h2 = Math.imul(h2 ^ ch, 1597334677); } h1 = Math.imul(h1 ^ (h1 >>> 16), 2246822507) ^ Math.imul(h2 ^ (h2 >>> 13), 3266489909); h2 = Math.imul(h2 ^ (h2 >>> 16), 2246822507) ^ Math.imul(h1 ^ (h1 >>> 13), 3266489909); return 4294967296 * (2097151 & h2) + (h1 >>> 0); }