您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
View Amazon review images in a zoomable wall
// ==UserScript== // @name amazonReviewWall // @namespace https://github.com/deltabravozulu/usefulUserScripts // @version 420.69.2022-05-12T14:28:42 // @author DeltaBravoZulu // @description View Amazon review images in a zoomable wall // @homepage https://github.com/deltabravozulu/usefulUserScripts/tree/main/amazonReviewWall // @icon https://raw.githubusercontent.com/deltabravozulu/usefulUserScripts/main/amazonReviewWall/images/amazonReviewWall_small.png // @icon64 https://raw.githubusercontent.com/deltabravozulu/usefulUserScripts/main/amazonReviewWall/images/amazonReviewWall_small.png // @supportURL https://github.com/deltabravozulu/usefulUserScripts // @include /^https?:\/\/(www|smile)\.amazon\.(cn|in|co\.jp|sg|ae|fr|de|it|nl|es|co\.uk|ca|com(\.(mx|au|br|tr))?)\/.*(dp|gp\/(product|video)|exec\/obidos\/ASIN|o\/ASIN)\/.*$/ // @grant GM_setValue // @grant GM_getValue // @grant window.close // @grant window.focus // @run-at document-idle // @license PayMe // ==/UserScript== /////////////////////////////////////// // Amazon Review Photowall // /////////////////////////////////////// /* ** Makes a zoomable photo wall from Amazon review pictures */ async function theStuff() { function sleep(ms) { return new Promise((resolve) => setTimeout(resolve, ms)); } function appendHTML() { var wrapper = document.createElement("body"); wrapper.innerHTML = '\ <style>\ html,body {\ margin: 0;\ padding: 0;\ }\ body { background-color: black; }\ header {\ padding: 1em;\ background-color: #333;\ text-align: center;\ }\ h1 { color:#fff;}\ a:link,\ a:visited {\ text-decoration: none;\ font-family: sans-serif;\ color: white;\ }\ a:hover { color: #ddd; }\ #author { float: left; }\ #repo { float: right; }\ #title {\ display: inline-block;\ font-weight: bold;\ }\ .zoomwall {\ font-size: 0;\ overflow: hidden; \ }\ .zoomwall img {\ height: 15vw;\ opacity: 1;\ vertical-align: top;\ \ transform-origin: 0% 0%;\ transition-property: transform, opacity;\ transition-duration: 0.3s;\ transition-timing-function: ease-out;\ -webkit-transform-origin: 0% 0%;\ -webkit-transition-property: transform, opacity;\ -webkit-transition-duration: 0.3s;\ -webkit-transition-timing-function: ease-out;\ }\ .zoomwall.lightbox img {\ transition-timing-function: ease-in;\ -webkit-transition-timing-function: ease-in;\ }\ .zoomwall.lightbox img {\ opacity: 0.3;\ }\ .zoomwall.lightbox img.active {\ opacity: 1;\ }\ </style>\ <div id="zoomwall" class="zoomwall"></div>\ '; document.appendChild(wrapper); } await sleep(1000); //https://stackoverflow.com/a/58316317/8398127 function getChunks(selector, strStart, strEnd) { const html = document.querySelector(selector).innerHTML; const start = html.indexOf(strStart); const end = html.indexOf(strEnd, start); if (start == -1 || end == -1) return; return { before: html.substr(0, start), match: html.substr(start, end - start + strEnd.length), after: html.substr(end + strEnd.length, html.length - end), }; } let chunkSelector = "#reviews-image-gallery-container > script"; let chunkStart = "data, "; let chunkEnd = '");'; chunks = getChunks(chunkSelector, chunkStart, chunkEnd).match; dataAndCSRF = chunks .replaceAll(");", "") .replaceAll('"', "") .replaceAll(" ", ""); dataAndCSRFArray = dataAndCSRF.split(","); data = dataAndCSRFArray[1]; firstReviewIds = dataAndCSRFArray[3]; asin = data; csrfToken = firstReviewIds; let text = ""; let date = new Date().toISOString(); function AppendToLocalStorage(name, value) { let n = 0; while (localStorage.getItem(name + "-" + n)) { n++; } localStorage.setItem(name + "-" + n, value); //add item at first available index number } function GetLocalStorage(name) { let n = 0; let data = []; while (localStorage.getItem(name + "-" + n)) { data.push(localStorage.getItem(name + "-" + n++)); } return data; } function AppendToGMStorage(name, value) { let n = 0; while (GM_getValue(name + "-" + n, "")) { n++; } GM_setValue(name + "-" + n, value); //add item at first available index number } function GetGMStorage(name) { let n = 0; let data = []; while (GM_getValue(name + "-" + n, "")) { data.push(GM_getValue(name + "-" + n++)); } return data; } //https://www.amazon.com/hz/reviews-render/get-reviews-with-media?mediaType=image&asin=B08DKK2KD1&csrfToken=guMT%2B0nYuIuyIIE6DbwfM0%2BcGnpEsbBr8SAKSRIAAAABAAAAAGJ5vEhyYXcAAAAA%2B4kUEk%2F7iMGR3xPcX6iU function loadReviews() { url = "https://www.amazon.com/hz/reviews-render/get-reviews-with-media?mediaType=image&asin=" + asin + "&csrfToken=" + csrfToken; function getJSON(url) { var resp; var xmlHttp; resp = ""; xmlHttp = new XMLHttpRequest(); if (xmlHttp != null) { xmlHttp.open("GET", url, false); xmlHttp.send(null); resp = xmlHttp.responseText; } return resp; } jsonGot = getJSON(url); jsonParsed = JSON.parse(jsonGot); //need to keep doing this until all pages are hit csrfToken = jsonParsed.csrfToken; reviews = jsonParsed.reviewsWithMediaList; reviewKeys = Object.keys(reviews); reviewCount = reviewKeys.length; for (var r = 0; r < reviewCount; r++) { reviewKey = reviewKeys[r]; review = reviews[reviewKey]; //console.log(review) reviewId = review.reviewId; reviewUrl = "https://www.amazon.com/gp/customer-reviews/" + reviewId; profileUrl = "https://www.amazon.com" + review.customerProfileLink; AppendToGMStorage(asin, profileUrl); AppendToLocalStorage(asin, profileUrl); for (var j = 0; j < review.images.length; j++) { imgUrl = review.images[j].source; text += '<img src="' + imgUrl + '" reviewurl="' + reviewUrl + '" profileurl="' + profileUrl + '"></img>'; console.log( "imgUrl: " + imgUrl + ", reviewUrl: " + reviewUrl + ", profileUrl: " + profileUrl ); //console.log(csrfToken) } } } loadReviews(); await sleep(3000); var newDoc = document.open("text/html", "replace"); appendHTML(); await sleep(1000); document.getElementById("zoomwall").innerHTML = text; await sleep(1000); var zoomwall = { create: function (blocks, enableKeys) { zoomwall.resize(blocks.children); blocks.classList.remove("loading"); // shrink blocks if an empty space is clicked blocks.addEventListener("click", function () { if (this.children && this.children.length > 0) { zoomwall.shrink(this.children[0]); } }); // add click listeners to blocks for (var i = 0; i < blocks.children.length; i++) { blocks.children[i].addEventListener("click", zoomwall.animate); } // add key down listener if (enableKeys) { var keyPager = function (event) { which = event.which; keyCode = event.keyCode; shiftKey = event.shiftKey; altKey = event.altKey; ctrlKey = event.ctrlKey; metaKey = event.metaKey; key = event.key; defaultPrevented = event.defaultPrevented; //[prevent key codes from working when shift,alt,ctrl,cmd,windows,super,etc. keys are working ] if ( defaultPrevented || shiftKey || altKey || ctrlKey || metaKey ) { console.log(key); return; } //[ESC] = zoom out else if (keyCode === 27) { console.log(key); if (blocks.children && blocks.children.length > 0) { zoomwall.shrink(blocks.children[0]); } event.preventDefault(); } //[⬅] = previous else if (keyCode === 37) { console.log(key); zoomwall.page(blocks, false); event.preventDefault(); } //[➡] = next else if (keyCode === 39) { console.log(key); zoomwall.page(blocks, true); event.preventDefault(); } //[space]||[⬆]||[r] = open review else if ( keyCode === 32 || keyCode === 38 || keyCode === 82 ) { console.log(key); zoomwall.reviewUrl(blocks); event.preventDefault(); } //[⬇]||[p] = open profile else if (keyCode === 40 || keyCode === 80) { console.log(key); zoomwall.profileUrl(blocks); event.preventDefault(); } else { console.log(key); return; } }; document.addEventListener("keydown", keyPager); } }, resizeRow: function (row, width) { if (row && row.length > 1) { for (var i in row) { row[i].style.width = (parseInt(window.getComputedStyle(row[i]).width, 10) / width) * 100 + "%"; row[i].style.height = "auto"; } } }, calcRowWidth: function (row) { var width = 0; for (var i in row) { width += parseInt(window.getComputedStyle(row[i]).width, 10); } return width; }, resize: function (blocks) { var row = []; var top = -1; for (var c = 0; c < blocks.length; c++) { var block = blocks[c]; if (block) { if (top == -1) { top = block.offsetTop; } else if (block.offsetTop != top) { zoomwall.resizeRow(row, zoomwall.calcRowWidth(row)); row = []; top = block.offsetTop; } row.push(block); } } zoomwall.resizeRow(row, zoomwall.calcRowWidth(row)); }, reset: function (block) { block.style.transform = "translate(0, 0) scale(1)"; block.style.webkitTransform = "translate(0, 0) scale(1)"; block.classList.remove("active"); }, shrink: function (block) { block.parentNode.classList.remove("lightbox"); // reset all blocks zoomwall.reset(block); var prev = block.previousElementSibling; while (prev) { zoomwall.reset(prev); prev = prev.previousElementSibling; } var next = block.nextElementSibling; while (next) { zoomwall.reset(next); next = next.nextElementSibling; } // swap images if (block.dataset.lowres) { block.src = block.dataset.lowres; } }, expand: function (block) { block.classList.add("active"); block.parentNode.classList.add("lightbox"); // parent dimensions var parentStyle = window.getComputedStyle(block.parentNode); var parentWidth = parseInt(parentStyle.width, 10); var parentHeight = parseInt(parentStyle.height, 10); var parentTop = block.parentNode.getBoundingClientRect().top; // block dimensions var blockStyle = window.getComputedStyle(block); var blockWidth = parseInt(blockStyle.width, 10); var blockHeight = parseInt(blockStyle.height, 10); // determine maximum height var targetHeight = window.innerHeight; if (parentHeight < window.innerHeight) { targetHeight = parentHeight; } else if (parentTop > 0) { targetHeight -= parentTop; } // swap images if (block.dataset.highres) { if ( block.src != block.dataset.highres && block.dataset.lowres === undefined ) { block.dataset.lowres = block.src; } block.src = block.dataset.highres; } // determine what blocks are on this row var row = []; row.push(block); var next = block.nextElementSibling; while (next && next.offsetTop == block.offsetTop) { row.push(next); next = next.nextElementSibling; } var prev = block.previousElementSibling; while (prev && prev.offsetTop == block.offsetTop) { row.unshift(prev); prev = prev.previousElementSibling; } // calculate scale var scale = targetHeight / blockHeight; if (blockWidth * scale > parentWidth) { scale = parentWidth / blockWidth; } // determine offset var offsetY = parentTop - block.parentNode.offsetTop + block.offsetTop; if (offsetY > 0) { if (parentHeight < window.innerHeight) { offsetY -= targetHeight / 2 - (blockHeight * scale) / 2; } if (parentTop > 0) { offsetY -= parentTop; } } var leftOffsetX = 0; // shift in current row for (var i = 0; i < row.length && row[i] != block; i++) { leftOffsetX += parseInt(window.getComputedStyle(row[i]).width, 10) * scale; } leftOffsetX = parentWidth / 2 - (blockWidth * scale) / 2 - leftOffsetX; var rightOffsetX = 0; // shift in current row for (var i = row.length - 1; i >= 0 && row[i] != block; i--) { rightOffsetX += parseInt(window.getComputedStyle(row[i]).width, 10) * scale; } rightOffsetX = parentWidth / 2 - (blockWidth * scale) / 2 - rightOffsetX; // transform current row var itemOffset = 0; // offset due to scaling of previous items var prevWidth = 0; for (var i = 0; i < row.length; i++) { itemOffset += prevWidth * scale - prevWidth; prevWidth = parseInt(window.getComputedStyle(row[i]).width, 10); var percentageOffsetX = ((itemOffset + leftOffsetX) / prevWidth) * 100; var percentageOffsetY = (-offsetY / parseInt(window.getComputedStyle(row[i]).height, 10)) * 100; row[i].style.transformOrigin = "0% 0%"; row[i].style.webkitTransformOrigin = "0% 0%"; row[i].style.transform = "translate(" + percentageOffsetX.toFixed(8) + "%, " + percentageOffsetY.toFixed(8) + "%) scale(" + scale.toFixed(8) + ")"; row[i].style.webkitTransform = "translate(" + percentageOffsetX.toFixed(8) + "%, " + percentageOffsetY.toFixed(8) + "%) scale(" + scale.toFixed(8) + ")"; } // transform items after var nextOffsetY = blockHeight * (scale - 1) - offsetY; var prevHeight; itemOffset = 0; // offset due to scaling of previous items prevWidth = 0; var next = row[row.length - 1].nextElementSibling; var nextRowTop = -1; while (next) { var curTop = next.offsetTop; if (curTop == nextRowTop) { itemOffset += prevWidth * scale - prevWidth; } else { if (nextRowTop != -1) { itemOffset = 0; nextOffsetY += prevHeight * (scale - 1); } nextRowTop = curTop; } prevWidth = parseInt(window.getComputedStyle(next).width, 10); prevHeight = parseInt(window.getComputedStyle(next).height, 10); var percentageOffsetX = ((itemOffset + leftOffsetX) / prevWidth) * 100; var percentageOffsetY = (nextOffsetY / prevHeight) * 100; next.style.transformOrigin = "0% 0%"; next.style.webkitTransformOrigin = "0% 0%"; next.style.transform = "translate(" + percentageOffsetX.toFixed(8) + "%, " + percentageOffsetY.toFixed(8) + "%) scale(" + scale.toFixed(8) + ")"; next.style.webkitTransform = "translate(" + percentageOffsetX.toFixed(8) + "%, " + percentageOffsetY.toFixed(8) + "%) scale(" + scale.toFixed(8) + ")"; next = next.nextElementSibling; } // transform items before var prevOffsetY = -offsetY; itemOffset = 0; // offset due to scaling of previous items prevWidth = 0; var prev = row[0].previousElementSibling; var prevRowTop = -1; while (prev) { var curTop = prev.offsetTop; if (curTop == prevRowTop) { itemOffset -= prevWidth * scale - prevWidth; } else { itemOffset = 0; prevOffsetY -= parseInt(window.getComputedStyle(prev).height, 10) * (scale - 1); prevRowTop = curTop; } prevWidth = parseInt(window.getComputedStyle(prev).width, 10); var percentageOffsetX = ((itemOffset - rightOffsetX) / prevWidth) * 100; var percentageOffsetY = (prevOffsetY / parseInt(window.getComputedStyle(prev).height, 10)) * 100; prev.style.transformOrigin = "100% 0%"; prev.style.webkitTransformOrigin = "100% 0%"; prev.style.transform = "translate(" + percentageOffsetX.toFixed(8) + "%, " + percentageOffsetY.toFixed(8) + "%) scale(" + scale.toFixed(8) + ")"; prev.style.webkitTransform = "translate(" + percentageOffsetX.toFixed(8) + "%, " + percentageOffsetY.toFixed(8) + "%) scale(" + scale.toFixed(8) + ")"; prev = prev.previousElementSibling; } }, animate: function (e) { if (this.classList.contains("active")) { zoomwall.shrink(this); } else { var actives = this.parentNode.getElementsByClassName("active"); for (var i = 0; i < actives.length; i++) { actives[i].classList.remove("active"); } zoomwall.expand(this); } e.stopPropagation(); }, reviewUrl: function (blocks) { var actives = blocks.getElementsByClassName("active"); if (actives && actives.length > 0) { var current = actives[0]; } window.open(current.attributes.reviewurl.value, "_blank").blur(); self.focus(); }, profileUrl: function (blocks) { var actives = blocks.getElementsByClassName("active"); if (actives && actives.length > 0) { var current = actives[0]; } window.open(current.attributes.profileurl.value, "_blank").blur(); self.focus(); }, page: function (blocks, isNext) { var actives = blocks.getElementsByClassName("active"); if (actives && actives.length > 0) { var current = actives[0]; var next; if (isNext) { next = current.nextElementSibling; } else { next = current.previousElementSibling; } if (next) { current.classList.remove("active"); // swap images if (current.dataset.lowres) { current.src = current.dataset.lowres; } zoomwall.expand(next); } } }, }; await sleep(1000); zoomwall.create(document.getElementById("zoomwall"), true); await sleep(1000); newDoc.close(); } /////////////////////////////////////// // Button Injections // /////////////////////////////////////// /* ** Adds button to process start */ function addButts() { console.log("Adding Button"); var oldButtons = document.querySelector("#productTitle"); var newButtons = oldButtons.parentElement; var iconHtml = '<button type="button" class="default icon-only" id="ImageWallButton" title="See review images in a wall"><img height="40px" width="40px" src=""></img></button>'; newButtons.insertAdjacentHTML("beforeend", iconHtml); document .getElementById("ImageWallButton") .addEventListener("click", theStuff); console.log("Added Button"); } addButts();