您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Converts selected tags on GoodReads into rating images (such as tags with half-star ratings)
当前为
// ==UserScript== // @name Goodreads tags to images // @namespace http://www.goodreads.com/* // @description Converts selected tags on GoodReads into rating images (such as tags with half-star ratings) // @include /^https?://.*\.goodreads\.com/.*$/ // @grant none // @version 1.0.0 // ==/UserScript== // TODO : // // * Make another attempt at using jquery without goodreads version conflict - load just within tamper/greasemonkey? // * Flag elements that have already been updated and don't update them again (necessary if text is not removed). Consider dataset attribute // TODO-MAYBE : // // * Consider making text removal optional // * Improve hover background color // // Load tag image data into keyed hash // function initImageData() { /* // GoodReads style stars tagImages['stars-on' ] = { imgData: "" }; tagImages['stars-half'] = { imgData: "" }; tagImages['stars-off' ] = { imgData: "" }; */ // Readinglist style stars tagImages['stars-on' ] = { imgData: "" }; tagImages['stars-half'] = { imgData: "" }; tagImages['stars-off' ] = { imgData: "" }; // Readinglist style clouds tagImages['clouds-on' ] = { imgData: "" }; tagImages['clouds-half'] = { imgData: "" }; tagImages['clouds-off' ] = { imgData: "" }; } // // Installs a mutation observer to catch dynamic content updates to any tag shelfs (such as when a user edits tags via a pop-up) // -> Called on page load and by installInfiniteScrollHook() to capture new tag shelfs that get added dynamically // function installTagShelfUpdateHook() { // Cross browser mutation observer support var MutationObserver = window.MutationObserver || window.WebKitMutationObserver || window.MozMutationObserver; // Watch all of the various types of tag shelfs (on the book list page, review edit page, book page, etc) var objShelfList = document.querySelectorAll('[id*=shelfList]'); // Matches shelfList..., review_shelfList..., etc // Make sure the required elements were found, otherwise don't install the observer if ((objShelfList != null) && (MutationObserver != null)) { // Create an observer and callback var observer = new MutationObserver( // This is the callback function function(mutations) { // Convert newly loaded anchor tag content convertTagsToImages(); } ); // Start observing the target element(s) // Note : Repeated observe calls on the same target just replace the previous observe, so it's // ok to re-observe the same target in the future without first disconnecting from it for(var i = 0; i < objShelfList.length; ++i) { observer.observe(objShelfList[i], { attributes: true, childList: true, characterData: true }); } } } // // Installs a mutation observer to catch content updates generated by Infinite Scroll mode on book list pages // -> Depends on these two divs existing : #infiniteLoading and #infiniteStatus // -> There are other ways this could be accomplished (timer, observe other divs, waitForKeyElements, etc) // function installInfiniteScrollHook() { // Cross browser mutation observer support var MutationObserver = window.MutationObserver || window.WebKitMutationObserver || window.MozMutationObserver; // Check infinite status text area to identify if pending updates are complete var objInfiniteStatusDiv = document.querySelector('#infiniteStatus'); // Watch the infinite scroll throbber div for changes var objInfiniteScrollThrobberDiv = document.querySelector('#infiniteLoading'); // Make sure the required elements were found, otherwise don't install the observer if ((objInfiniteScrollThrobberDiv != null) && (objInfiniteStatusDiv != null) && (MutationObserver != null)) { // Initialize the status text monitoring string strInfiniteScrollStatusLast = objInfiniteStatusDiv.textContent; // Create an observer and callback var observer = new MutationObserver( // This is the callback function function(mutations) { // Determine whether an update is complete or still in progress // by checking the infinite scroll "X of Y loaded" display text if (objInfiniteStatusDiv) { // Check to see if the update is complete, if so then convert newly loaded anchor tags if (objInfiniteStatusDiv.textContent != strInfiniteScrollStatusLast) { convertTagsToImages(); // Update the shelf hook to capture and monitor new content (i.e. new book tag shelves) installTagShelfUpdateHook() } // Record the status text for comparison next time strInfiniteScrollStatusLast = objInfiniteStatusDiv.textContent; } } ); // Start observing the target element observer.observe(objInfiniteScrollThrobberDiv, { attributes: true, childList: true, characterData: true }); } } // // Append an <img> tag with the given image data to an element // function appendImage(parentObj, imgData) { var tagImg = document.createElement('img'); tagImg.src = imgData; parentObj.appendChild(tagImg); } // // Prepend an <img> tag with the given image data to an element (via modifying it's html) // function prependImage(parentObj, imgData) { parentObj.innerHTML = "<img src=" + imgData + " alt=''/>" + parentObj.innerHTML; } // // Render a tag rating based on a type ("stars","clouds) and a numeric value in tag format ("4-0","1-5", etc) // function renderTagImages(parentObj, imgType, imgValue) { var valMax = 5.0; var valOn = parseFloat( imgValue.replace("-", ".") ); var valOff = valMax - valOn; if ((imgType == "stars") || (imgType == "clouds")) { // Render whole "on" icons first while (valOn > 0.5) { appendImage(parentObj, tagImages[ imgType + '-on' ].imgData ); valOn -= 1.0; } // Render half "on" icon if needed for 0.5 values if (valOn == 0.5) { appendImage(parentObj, tagImages[ imgType + '-half' ].imgData ); } // Render the remaining slots as placeholders ("off") while (valOff >= 1) { appendImage(parentObj, tagImages[ imgType + '-off' ].imgData ); valOff -= 1.0; } } } // // Find links with matching tag text and convert them to the paired images // (non-jquery version to avoid GoodReads breakage with jquery version conflict) // function convertTagsToImages() { var myRegexp = /rating-(stars|clouds)-(\d-\d)/; var objText; var elAnchor; var elLinks = document.getElementsByTagName( 'a' ); // Walk through all the anchor tags on the page for ( var i = 0; i < elLinks.length; i++ ) { elAnchor = elLinks[ i ]; var match = myRegexp.exec(elAnchor.text); // Note : match[0] = full match text, [1] = "stars" or "clouds", [2] = "N1-N2" where (ideally) N1 is a digit 0-9 and N2 is 0 or 5 if (match != null) { // Strip out tag name and save off any trailing text (trailing text gets re-appended later) objText = elAnchor.text.replace(match[0], ""); // Remove tag text temporarily elAnchor.innerHTML = ""; // Render tag image renderTagImages(elAnchor, match[1], match[2]); // prevent line breaks elAnchor.style.whiteSpace="nowrap"; // Restore trailing text elAnchor.innerHTML = elAnchor.innerHTML + objText; } } } // A couple globals var tagImages = Object.create(null); // Hash for storing tag image data by key name var strInfiniteScrollStatusLast; var objInfiniteStatusDiv; // Initialize our tag images initImageData(); // Convert any tags found on the page convertTagsToImages(); // Install hooks for converting dynamic content that appears after initial page load installInfiniteScrollHook(); installTagShelfUpdateHook();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址