网购小秘,不花冤枉钱!!!支持平台:京东、淘宝、天猫、天猫超市、天猫国际、京东国际、京东图书、京东大药房、阿里大药房、唯品会等;功能:1、搜索商品时会自动查询标注有优惠券和活动的商品,无需进入详情页,方便快捷;2、浏览商品详情页时脚本会自动查询商品是否有隐藏的优惠券;简单好用,低侵入~
// ==UserScript== // @name 【网购小秘】自动查询淘宝、天猫、京东等隐藏的大额优惠券和优惠活动,能省就省,不花冤枉钱! // @namespace Xanthella_Coupon_Secretary_helper // @version 3.0.2 // @description 网购小秘,不花冤枉钱!!!支持平台:京东、淘宝、天猫、天猫超市、天猫国际、京东国际、京东图书、京东大药房、阿里大药房、唯品会等;功能:1、搜索商品时会自动查询标注有优惠券和活动的商品,无需进入详情页,方便快捷;2、浏览商品详情页时脚本会自动查询商品是否有隐藏的优惠券;简单好用,低侵入~ // @icon  // @author Xanthella,huahuacat // @match *://*.taobao.com/* // @match *://*.tmall.com/* // @match *://www.vipglobal.hk/detail-* // @match *://*.tmall.hk/* // @match *://chaoshi.detail.tmall.com/* // @match *://*.liangxinyao.com/* // @match *://pages.tmall.com/wow/an/cs/search** // @match *://detail.vip.com/detail-* // @match *://*.jd.com/* // @match *://*.jd.hk/* // @match *://category.vip.com/suggest.php** // @match *://item.jkcsjd.com/* // @match *://*.yiyaojd.com/* // @match *://*.vip.com/* // @match *://www.vipglobal.hk // @match *://list.vip.com/*.html // @match *://*.suning.com/* // @exclude *://jianghu.taobao.com/* // @exclude *://login.taobao.com/* // @exclude *://map.taobao.com/* // @exclude *://uland.taobao.com/* // @exclude *://creator.guanghe.taobao.com/* // @exclude *://myseller.taobao.com/* // @exclude *://qn.taobao.com/* // @exclude *://jingfen.jd.com/* // @exclude *://jmw.jd.com/* // @exclude *://passport.jd.com/* // @exclude *://passport.shop.jd.com/* // @exclude *://passport.vip.com/* // @exclude *://huodong.taobao.com/wow/z/guang/gg_publish/* // @exclude *://passport.suning.com/* // @grant GM_info // @grant GM_download // @grant GM_getValue // @grant GM_setValue // @grant GM_openInTab // @grant GM_xmlhttpRequest // @grant GM_addStyle // @run-at document-end // @license AGPL License // @noframes // @copyright Xanthella,huahuacat // ==/UserScript== (function() { 'use strict'; /** * AGPL 协议全称为 GNU Affero General Public License(GNU Affero 通用公共许可证),它是一种自由软件许可证。 * AGPL 协议的主要特点是在传统的开源软件许可证(如 GPL)的基础上,加强了对软件在网络环境中使用的限制。 * GPL 是 GNU General Public License(GNU 通用公共许可证)的缩写,本脚本遵循AGPL协议!满足其开源原则、Copyleft原则、附带源代码、允许收费等原则 * * 脚本中部分工具类搜索自互联网,并没有明确的出处。因此在此说明!如有侵权请留言告知!以上~ * 领券部分源码借鉴自:https://github.com/huahuacatTX/greasyfork/blob/main/pro_huahuacat_union.js * 特此声明! * * 在原基础上去掉了一些高侵入的设计,优化了部分代码 */ //基础工具类 class BaseObject { getParamterBySuffix(url=window.location.href, suffix="html"){ if(url.indexOf("?")!=-1){ url = url.split("?")[0]; } if(url.indexOf("#")!=-1){ url = url.split("#")[0]; } var splitText = url.split("/"); var idText = splitText[splitText.length-1]; idText = idText.replace(".html",""); return idText; } getPlatform(host = window.location.host){ let platform = ""; if(host.indexOf(".taobao.")!=-1 || host.indexOf(".liangxinyao.")!=-1){ platform = "taobao"; }else if(host.indexOf(".tmall.")!=-1){ platform = "tmall"; }else if(host.indexOf(".jd.")!=-1 || host.indexOf(".yiyaojd.")!=-1 || host.indexOf(".jkcsjd.")!=-1){ platform = "jd"; }else if(host.indexOf(".vip.")!=-1 || host.indexOf(".vipglobal.")!=-1){ platform = "vpinhui"; }else if(host.indexOf(".suning.")!=-1){ platform = "suning"; } return platform; } suningParameter(url){ const regex = /product\.suning\.com\/(\d+\/\d+)\.html/; const match = url.match(regex); if(match){ return match[1].replace(/\//g, '-'); } return null; } getParamterBySearch(paramsString=window.location.href, tag){ if(paramsString.indexOf("?")!=-1){ paramsString = paramsString.split('?')[1]; // Extract the query string } const params = new URLSearchParams(paramsString); return params.get(tag); } request(method, url, param) { if(!method){ method = "get"; } if(!url){ return new Promise(function(resolve, reject){ reject({"code":"exception", "r###lt":null}); }); } if(!param){ param = {}; } method = method.toUpperCase(); let config = { method: method }; if (method === 'POST') { config.headers['Content-Type'] = 'application/json'; config.body = JSON.stringify(param); } return new Promise(function(resolve, reject){ fetch(url, config).then(response => response.text()).then(text => { resolve({"code":"ok", "r###lt":text}); }).catch(error => { reject({"code":"exception", "r###lt":null}); }); }); } querySelectorAsync(selector, target = document.body, allowEmpty = true, delay=10, maxDelay=10 * 1000){ return new Promise((resolve,reject) =>{ if(selector.toUpperCase()=="BODY"){ resolve(document.body); return; } if(selector.toUpperCase()=="HTML"){ resolve(document.html); return; } let totalDelay = 0; let element = target.querySelector(selector); let r###lt = allowEmpty ? !!element : (!!element && !!element.innerHTML); if(r###lt){ resolve(element); return; } const elementInterval = setInterval(()=>{ if(totalDelay >= maxDelay){ clearInterval(elementInterval); resolve(null); return; } element = target.querySelector(selector); r###lt = allowEmpty ? !!element : (!!element && !!element.innerHTML); if(r###lt){ clearInterval(elementInterval); resolve(element); }else{ totalDelay += delay; } }, delay); }); } openInTab(url, options={"active":true, "insert":true, "setParent":true}){ if (typeof GM_openInTab === "function") { GM_openInTab(url, options); } else { GM.openInTab(url, options); } } encodeText(text){ if(!text){ return ""; } text = text.replace(/\t|\r/g, ""); return encodeURIComponent(text); } isElementDisplayed(element) { if (element.offsetParent!== null) { return true; } const style = window.getComputedStyle(element); return style.display!== "none"; } } class Matrix { constructor(data) { this.data = data; this.rows = data.length; this.cols = data[0].length; } add(matrix) { if (this.rows !== matrix.rows || this.cols !== matrix.cols) { throw new Error("error"); } const r###lt = []; for (let i = 0; i < this.rows; i++) { const row = []; for (let j = 0; j < this.cols; j++) { row.push(this.data[i][j] + matrix.data[i][j]); } r###lt.push(row); } return new Matrix(r###lt); } subtract(matrix) { if (this.rows !== matrix.rows || this.cols !== matrix.cols) { throw new Error("error"); } const r###lt = []; for (let i = 0; i < this.rows; i++) { const row = []; for (let j = 0; j < this.cols; j++) { row.push(this.data[i][j] - matrix.data[i][j]); } r###lt.push(row); } return new Matrix(r###lt); } multiply(matrix) { if (this.cols !== matrix.rows) { throw new Error("error"); } const r###lt = []; for (let i = 0; i < this.rows; i++) { const row = []; for (let j = 0; j < matrix.cols; j++) { let sum = 0; for (let k = 0; k < this.cols; k++) { sum += this.data[i][k] * matrix.data[k][j]; } row.push(sum); } r###lt.push(row); } return new Matrix(r###lt); } transpose() { const r###lt = []; for (let i = 0; i < this.cols; i++) { const row = []; for (let j = 0; j < this.rows; j++) { row.push(this.data[j][i]); } r###lt.push(row); } return new Matrix(r###lt); } print() { this.data.forEach(row => { console.log(row.join(' ')); }); } } //面向详情相关类 class DetailPageObject extends BaseObject{ constructor(){ super(); this.generateIsR###lt = true; this.baseUrl = "https://t.jtm.pub"; } isRun(){ const currentHost = window.location.host; return ["detail.tmall.com", "item.taobao.com", "item.jd.com", "item.yiyaojd.com", "npcitem.jd.hk", "detail.tmall.hk", "detail.vip.com", "item.jkcsjd.com", "product.suning.com" ].map((host)=>currentHost.indexOf(host)!=-1).some((r###lt)=>r###lt); } getProductsInfo(platform){ var goodsId = ""; var goodsName = ""; const href = window.location.href; if(platform=="taobao"){ goodsId = this.getParamterBySearch(window.location.search, "id"); try{ const titleObj = document.querySelector("[class^='ItemTitle--']"); if(!!titleObj){ goodsName = titleObj.textContent; } }catch(e){} }else if(platform=="tmall"){ goodsId = this.getParamterBySearch(window.location.search, "id"); try{ const titleObj = document.querySelector("[class^='ItemTitle--']"); if(!!titleObj){ goodsName = titleObj.textContent; } }catch(e){} }else if(platform=="jd"){ goodsId = this.getParamterBySuffix(href); try{ const titleObj = document.querySelector("[class='sku-name']"); if(!!titleObj){ goodsName = titleObj.textContent; } }catch(e){} }else if(platform=="vpinhui"){ goodsId = this.getParamterBySuffix(href).replace("detail-",""); const titleObj = document.querySelector("[class='pib-title-detail']"); if(!!titleObj){ goodsName = titleObj.textContent; } }else if(platform=="suning"){ goodsId = this.suningParameter(href); const titleObj = document.querySelector("#itemDisplayName"); if(!!titleObj){ goodsName = titleObj.textContent; } } return {"goodsId":goodsId, "goodsName":this.encodeText(goodsName)}; } async getHandlerElement(handler){ const getElement = async (handler)=>{ const promiseArray = []; const handlers = handler.split("@"); for(let i=0; i<handlers.length; i++){ const eleName = handlers[i]; if(!eleName){ continue; } if(eleName=="body"){ promiseArray.push( new Promise((resolve,reject) =>{ resolve(document.body) }) ); }else if(eleName=="html"){ promiseArray.push( new Promise((resolve,reject) =>{ resolve(document.html) }) ); }else{ promiseArray.push(this.querySelectorAsync(eleName, document.body, true, 10, 1500)); } } const element = await Promise.race(promiseArray); return element ? element : null; } const element = await getElement(handler); return new Promise((resolve,reject) =>{ resolve(element); }); } async generateHtml(platform, goodsId, goodsName){ if(!platform || !goodsId){ return "kong"; } let addition = ""; if(platform=="vpinhui"){ const vip = goodsId.split("-"); addition = vip[0]; goodsId = vip[1]; } const goodsCouponUrl = this.baseUrl+"/api/coupon/query?no=6&version=1.0.2&platform="+platform+"&id="+goodsId+"&q="+goodsName+"&addition="+addition; try{ const data = await this.request("GET", goodsCouponUrl, null); if(data.code=="ok" && !!data.r###lt){ const json = JSON.parse(data.r###lt); await this.generateCoupon(platform, json.data); await this.generateQrcode(platform, json.mscan); //开启插入检测 let heartms = 0; const HEART_DELAY = 1500, MAX_MS = 1000*30; const generateR###ltInterval = setInterval(async ()=>{ if(this.generateIsR###lt){ if(document.querySelector("*[name='exist-llkbccxs-9246-hi']") || heartms>=MAX_MS){ clearInterval(generateR###ltInterval); }else{ await this.generateCoupon(platform, json.data); } } heartms += HEART_DELAY; }, HEART_DELAY); } }catch(e){console.log(e);} } async generateCoupon(platform, r###lt){ try{ this.generateIsR###lt = false; if(!r###lt || r###lt==="null" || !r###lt.hasOwnProperty("css") || !r###lt.hasOwnProperty("html") || !r###lt.hasOwnProperty("handler")){ return; } const {css, html, handler, templateId} = r###lt; if(!css || !html || !handler){ return; } GM_addStyle(css); // 添加HTML, 需要动态检测元素 const handlerElement = await this.getHandlerElement(handler); if(!handlerElement){ return; } if(platform=="taobao"){ handlerElement.parentNode.insertAdjacentHTML('afterend', html); }else if(platform=="tmall"){ handlerElement.parentNode.insertAdjacentHTML('afterend', html); }else if(platform=="jd"){ handlerElement.insertAdjacentHTML('afterend', html); }else if(platform=="vpinhui"){ handlerElement.insertAdjacentHTML('afterend', html); }else if(platform=="suning"){ handlerElement.insertAdjacentHTML('afterend', html); } const templateElement = document.querySelector("div[id='"+templateId+"']"); if(!templateElement){ return; } const couponId = templateElement.getAttribute("data-id"); const goodsPrivateUrl = this.baseUrl+"/api/private/change/coupon?no=6&v=1.0.2&platform="+platform+"&id="; if(!/\d/.test(couponId)){ return; } setInterval(()=>{ templateElement.querySelectorAll("*").forEach((element)=>{ element.removeAttribute("data-spm-anchor-id"); }); },400); const couponElementA = templateElement.querySelector("a[name='cpShUrl']"), clickedTag = "aclicked"; if(couponElementA){ couponElementA.addEventListener("click",()=>{ event.stopPropagation(); event.preventDefault(); if(couponElementA.getAttribute(clickedTag)){ return; } couponElementA.setAttribute(clickedTag, "true"); const href = couponElementA.getAttribute("href"); if(href && href.indexOf("https://")!=-1){ this.openInTab(href); couponElementA.removeAttribute(clickedTag); }else{ this.request("GET", goodsPrivateUrl+couponId, null).then((privateR###ltData)=>{ if(privateR###ltData.code==="ok" && !!privateR###ltData.r###lt){ let url = JSON.parse(privateR###ltData.r###lt).url; if(url){ this.openInTab(url); } } couponElementA.removeAttribute(clickedTag); }).then(()=>{ couponElementA.removeAttribute(clickedTag); }); } }); } const canvasElement = document.querySelector("#ca"+templateId); if(!canvasElement){ return; } const qrcodeR###ltData = await this.request("GET", goodsPrivateUrl+couponId, null); if(!!qrcodeR###ltData && qrcodeR###ltData.code==="ok" && !!qrcodeR###ltData.r###lt){ let img = JSON.parse(qrcodeR###ltData.r###lt).img; if(!!img){ let cxt = canvasElement.getContext("2d"); let imgData = new Image(); imgData.src = img; imgData.onload=function(){ cxt.drawImage(imgData, 0, 0, imgData.width, imgData.height); } } } }catch(e){ console.log(e); }finally{ this.generateIsR###lt = true; } } async generateQrcode(platform, mscan){ if(!mscan || mscan==="null" || !mscan.hasOwnProperty("mount") || !mscan.hasOwnProperty("html")|| !mscan.hasOwnProperty("qrcode")){ return; } const {mount, html, qrcode, iden} = mscan; if(!!mount && !!html && !!qrcode && !!iden){ const mountElement = await this.querySelectorAsync(mount, document.body, true, 10, 1500); if(mountElement){ mountElement.insertAdjacentHTML('afterend', html); let canvasElement = document.getElementById("mscan"+iden); let width = canvasElement.getAttribute("width"); let height = canvasElement.getAttribute("height"); let cxt = canvasElement.getContext("2d"); let imgData = new Image(); imgData.src = qrcode; imgData.onload=function(){ cxt.drawImage(imgData, 0, 0, width, height); } } } } start(){ if(!this.isRun()){ return; } const platform = this.getPlatform(); if(platform){ const goodsData = this.getProductsInfo(platform); this.generateHtml(platform, goodsData.goodsId, goodsData.goodsName); } } } //面向搜索相关类 class SearchPageObject extends BaseObject{ constructor(){ super(); this.intervalIsRunComplete = true; this.searchAttribute = "query-good-job"; this.baseUrl = "https://t.jtm.pub"; } isRun(){ const visitHref = window.location.href; return [ /^https:\/\/www\.(taobao|jd|tmall)\.(com|hk)(\/|\/\?)?/i,//淘宝/天猫/京东首页 /^https:\/\/(shop(\d+)|s)\.taobao\.com/i, //搜索或者未装修主页 /(pages|list)\.tmall\.(com|hk)/i, /tmall\.com\/(category|search|shop|\?q=)/i, /maiyao\.liangxinyao\.com/i, /search\.jd\.(com|hk)/i, /pro\.jd\.com\/mall/i, /jd\.com\/view_search/i, //商店主页 /category\.vip\.com/i, /(list|category)\.vip\.com/i, /^https:\/\/(?!product|dfp\.)([^\/]+)\.suning\.com\//i ].map((reg)=>(new RegExp(reg)).test(visitHref)).some((res)=>res); } requestConf(){ return new Promise((resolve, reject) => { this.request("GET", this.baseUrl+"/api/plugin/load/conf", null).then((data)=>{ if(data.code=="ok" && !!data.r###lt){ resolve(data.r###lt); }else{ resolve(null); } }); }); } pickupElements(confString, platform){ const visitHref = window.location.href; const selectorElementList = new Array(); let confFilter = confString; try{ confFilter = confFilter.replace(/\\\\/g,"\\"); }catch(e){} const confJson = JSON.parse(confFilter); if(confJson.hasOwnProperty(platform)){ const platformConfJson = confJson[platform]; for(let i=0; i<platformConfJson.length; i++){ const itemJson = platformConfJson[i]; if(!itemJson.hasOwnProperty("elements") || !itemJson.hasOwnProperty("matches")){ continue; } const {elements, matches} = itemJson; const isMatch = matches.map((reg)=>(new RegExp(reg, "i")).test(visitHref)).some((res)=>res); if(isMatch){ for(let j=0; j<elements.length; j++){ selectorElementList.push({ "element":elements[j]["element"], "findA":elements[j]["findA"], "page":elements[j]["page"] }); } } } } return selectorElementList; } transformElements(selectors){ const items = []; selectors.forEach((elementObj)=>{ if(elementObj.element){ const elements = document.querySelectorAll(elementObj.element+":not(["+this.searchAttribute+"='true'])"); //console.log("elements",elements.length); elements.forEach((element)=>{ if(element){ items.push({"element":element, "findA": elementObj.findA, "page":elementObj.page}); } }); } }); if(items.length>0){ this.queryAll(items); } } queryAll(items){ this.intervalIsRunComplete = false; this.processLinksInBatches(items, 18).then((r###lt)=>{ this.intervalIsRunComplete = true; }); } async processLinksInBatches(items, batchSize) { const r###lts = []; for (let i = 0; i < items.length; i += batchSize) { const batch = items.slice(i, i + batchSize); // 获取当前批次的链接 const batchR###lts = await Promise.all( // 同时处理当前批次中的所有请求 batch.map(item => this.queryOne(item)) ); r###lts.push(...batchR###lts); // 保存批次结果 } return r###lts; // 返回所有结果 } queryOne(item){ const { element, page, findA} = item; const self = this; return new Promise(function(resolve, reject){ if(!self.isElementDisplayed(element)){ //如果元素已经隐藏,则不执行 resolve("cacel"); return; } if(element.getAttribute(self.searchAttribute)){ //当存在时,说明已经处理过了 resolve("processed"); return; } element.setAttribute(self.searchAttribute, "true"); element.style.position = "relative"; element.addEventListener("click", function(e){ element.insertAdjacentHTML('beforeend', self.browsedHtml); }); let goodsDetailUrl = null; if(findA==="this"){ //说明本身就是A标签 goodsDetailUrl = element.getAttribute("href"); }else if(/^child@/.test(findA)){ const elementA = element.querySelector(findA.replace(/^child@/,"")); if(elementA){ goodsDetailUrl = elementA.getAttribute("href"); } } if(!goodsDetailUrl){ resolve("exception-url-null"); return; } let analysisData = null; if(/^jd_/.test(page)){ let jdId = self.getParamterBySuffix(goodsDetailUrl); if(!!jdId) analysisData = {"id":jdId, "platform":"jd"}; }else if(/^vpinhui_/.test(page)){ let vipId = self.getParamterBySuffix(goodsDetailUrl).replace("detail-",""); if(!!vipId){ analysisData = {"id":vipId.split("-")[1], "platform":"vpinhui"}; } }else if(/^suning_/.test(page)){ let suningId = self.suningParameter(goodsDetailUrl); if(!!suningId){ analysisData = {"id":suningId, "platform":"suning"}; } }else{ let platform = self.getPlatform(goodsDetailUrl); let id = self.getParamterBySearch(goodsDetailUrl, "id"); if(platform && id){ analysisData = {"id":id, "platform":platform}; } } if(!analysisData){ resolve("exception-data-null"); return; } const searchUrl = self.baseUrl+"/api/ebusiness/q/c?p="+analysisData.platform+"&id="+analysisData.id+"&no=6"; self.request("GET", searchUrl, null).then((data)=>{ if(data.code=="ok" && !!data.r###lt){ const {id, tip, encryptLink} = JSON.parse(data.r###lt); if(tip){ //console.log("coupon exist", id); element.insertAdjacentHTML('beforeend', tip); } if(encryptLink){ //console.log("jood job!", id); let decryptUrl = null; try{ const decryptLink = atob(encryptLink); decryptUrl = decryptLink.split('').reverse().join(''); }catch(e){} if(decryptUrl){ self.relativeJu(page, element, decryptUrl); } } } resolve("success"); }).catch(()=>{ resolve("error"); }); }); } relativeJu(page, element, decryptUrl){ const self = this; try{ if(page.indexOf("jd_")!=-1){ element.querySelectorAll("a").forEach((element_a)=>{ if(element_a.getAttribute("href").indexOf("item.jd.com")!=-1){ element_a.removeAttribute(onclick); element_a.addEventListener("click", function(e){ e.preventDefault(); e.stopPropagation(); self.openInTab(decryptUrl); }); } }); } else if(page.indexOf("taobao_")!=-1 || page.indexOf("tmall_")!=-1){ element.addEventListener("click", function(e){ const target = e.target const tagName = target.tagName.toUpperCase(); let isPreventDefault = false; if(tagName==="A"){ //只有点击A标签才去判断 const href = target.getAttribute("href"); const isDetail = /(detail|item)\.(tmall|taobao)\.com\/item\.htm/.test(href); if(isDetail){ isPreventDefault = true; } }else{ isPreventDefault = true; } if(isPreventDefault){ e.preventDefault(); e.stopPropagation(); self.openInTab(decryptUrl); } }); } else if(page.indexOf("vpinhui_")!=-1){ element.querySelectorAll("a").forEach((element_a)=>{ if(element_a.getAttribute("href").indexOf("detail.vip.com/detail-")!=-1){ element_a.addEventListener("click", function(e){ e.preventDefault(); e.stopPropagation(); self.openInTab(decryptUrl); }); } }); } else if(page.indexOf("suning_")!=-1){ element.querySelectorAll("a").forEach((element_a)=>{ if(element_a.getAttribute("href").indexOf("product.suning.com")!=-1){ element_a.addEventListener("click", function(e){ e.preventDefault(); e.stopPropagation(); self.openInTab(decryptUrl); }); } }); } }catch(e){ console.log(e); } } start(){ if(this.isRun()){ const platform = this.getPlatform(); this.requestConf().then((confString)=>{ const selectors = this.pickupElements(confString, (platform=="tmall"? "taobao" : platform)); if(this.intervalIsRunComplete){ this.transformElements(selectors); } setInterval(()=>{ if(this.intervalIsRunComplete){ this.transformElements(selectors); } }, 999); }); } } } (new DetailPageObject()).start(); (new SearchPageObject()).start(); })();