【网购小秘】自动查询淘宝、天猫、京东等隐藏的大额优惠券和优惠活动,能省就省,不花冤枉钱!

网购小秘,不花冤枉钱!!!支持平台:京东、淘宝、天猫、天猫超市、天猫国际、京东国际、京东图书、京东大药房、阿里大药房、唯品会等;功能:1、搜索商品时会自动查询标注有优惠券和活动的商品,无需进入详情页,方便快捷;2、浏览商品详情页时脚本会自动查询商品是否有隐藏的优惠券;简单好用,低侵入~

// ==UserScript==
// @name         【网购小秘】自动查询淘宝、天猫、京东等隐藏的大额优惠券和优惠活动,能省就省,不花冤枉钱!
// @namespace    Xanthella_Coupon_Secretary_helper
// @version      3.0.1
// @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
// @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/*
// @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(url = window.location.host){
			let platform = "";
			if(/tmall\./.test(url)){
				platform = "tmall";
			}else if(/(taobao|liangxinyao)\.(com|hk)/.test(url)){
				platform = "taobao";
			}else if(/(jd|yiyaojd|jkcsjd)\.(com|hk)/.test(url)){
				platform = "jd";
			}else if(/(vip|vipglobal)\.(com|hk)/.test(url)){
				platform = "vpinhui";
			}
			return platform;
		}
		
		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", "result":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", "result":text});
				}).catch(error => {
					reject({"code":"exception", "result":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 result = allowEmpty ? !!element : (!!element && !!element.innerHTML);
				if(result){
					resolve(element);
					return;
				}
				const elementInterval = setInterval(()=>{
					if(totalDelay >= maxDelay){
						clearInterval(elementInterval);
						resolve(null);
						return;
					}
					element = target.querySelector(selector);
					result = allowEmpty ? !!element : (!!element && !!element.innerHTML);
					if(result){
						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 result = [];
	        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]);
	            }
	            result.push(row);
	        }
	        return new Matrix(result);
	    }

	    subtract(matrix) {
	        if (this.rows !== matrix.rows || this.cols !== matrix.cols) {
	            throw new Error("error");
	        }
	        const result = [];
	        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]);
	            }
	            result.push(row);
	        }
	        return new Matrix(result);
	    }
	
	    multiply(matrix) {
	        if (this.cols !== matrix.rows) {
	            throw new Error("error");
	        }
	
	        const result = [];
	        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);
	            }
	            result.push(row);
	        }
	        return new Matrix(result);
	    }
	
	    transpose() {
	        const result = [];
	        for (let i = 0; i < this.cols; i++) {
	            const row = [];
	            for (let j = 0; j < this.rows; j++) {
	                row.push(this.data[j][i]);
	            }
	            result.push(row);
	        }
	        return new Matrix(result);
	    }
		
	    print() {
	        this.data.forEach(row => {
	            console.log(row.join(' '));
	        });
	    }
	}
	
	//面向详情相关类
	class DetailPageObject extends BaseObject{
		
		constructor(){
			super();
			this.generateIsResult = 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"
			].map((host)=>currentHost.indexOf(host)!=-1).some((result)=>result);
		}
		
		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;
				}
			}
			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.result){
					const json = JSON.parse(data.result);
					await this.generateCoupon(platform, json.data);
					await this.generateQrcode(platform, json.mscan);
					//开启插入检测
					let heartms = 0;
					const HEART_DELAY = 1500, MAX_MS = 1000*30;  
					const generateResultInterval = setInterval(async ()=>{
						if(this.generateIsResult){
							if(document.querySelector("*[name='exist-llkbccxs-9246-hi']") || heartms>=MAX_MS){
								clearInterval(generateResultInterval);
							}else{
								await this.generateCoupon(platform, json.data);
							}
						}
						heartms += HEART_DELAY;
					}, HEART_DELAY);
				}
			}catch(e){console.log(e);}
		}
		
		async generateCoupon(platform, result){
			try{
				this.generateIsResult = false;
				if(!result || result==="null" || !result.hasOwnProperty("css") || !result.hasOwnProperty("html") || !result.hasOwnProperty("handler")){
					return;
				}
				
				const {css, html, handler, templateId} = result;
				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);
				}
				
				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((privateResultData)=>{
								if(privateResultData.code==="ok" && !!privateResultData.result){
									let url = JSON.parse(privateResultData.result).url;
									if(url){
										this.openInTab(url);
									}
								}
								couponElementA.removeAttribute(clickedTag);
							}).then(()=>{
								couponElementA.removeAttribute(clickedTag);
							});
						}
					});
				}
			
				const canvasElement = document.querySelector("#ca"+templateId);
				if(!canvasElement){
					return;
				}
				const qrcodeResultData = await this.request("GET", goodsPrivateUrl+couponId, null);
				if(!!qrcodeResultData && qrcodeResultData.code==="ok" && !!qrcodeResultData.result){
					let img = JSON.parse(qrcodeResultData.result).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.generateIsResult = 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
			].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.result){
						resolve(data.result);
					}else{
						resolve(null);
					}
				});
			});
		}
		
		pickupElements(confString){
			const visitHref = window.location.href;
			const selectorElementList = new Array();
			let confFilter = confString;
			try{
				confFilter = confFilter.replace(/\\\\/g,"\\");
			}catch(e){}
			const confJson = JSON.parse(confFilter);
			for(let key in confJson){
				if(!confJson.hasOwnProperty(key)){
					continue;
				}
				for(let i=0; i<confJson[key].length; i++){
					const itemJson = confJson[key][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;
			const promises = [];
			items.forEach((item)=>{
				promises.push(this.queryOne(item));
			});
			Promise.all(promises).then((result)=>{
				this.intervalIsRunComplete = true;
			});
		}
		
		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{
					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.result){
						const {tip, encryptLink} = JSON.parse(data.result);
						if(tip){
							element.insertAdjacentHTML('beforeend', tip);
						}
						if(encryptLink){
							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);
							});
						}
					});
				}
			}catch(e){
				console.log(e);
			}
		}
		
		start(){
			if(this.isRun()){
				this.requestConf().then((confString)=>{
					const selectors = this.pickupElements(confString);
					if(this.intervalIsRunComplete){
						this.transformElements(selectors);
					}
					setInterval(()=>{
						if(this.intervalIsRunComplete){
							this.transformElements(selectors);
						}
					}, 999);
				});
			}
		}
	}
	
	(new DetailPageObject()).start();
	(new SearchPageObject()).start();
})();