NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name Show Price as Hours of your Life // @description Gets the price displayed on shopping sites and shows the equivalent hours of your life spent to earn that money. // @author navchandar // @version 2.0 // @license MIT // @run-at document-end // @homepage https://navchandar.github.io/ // @homepage https://github.com/navchandar/ // @homepageURL https://navchandar.github.io/ // @copyright 2019, navchandar (https://openuserjs.org/users/navchandar) // @supportURL https://openuserjs.org/scripts/navchandar/Show_Price_as_Hours_of_your_Life/issues // @downloadURL https://openuserjs.org/install/navchandar/Show_Price_as_Hours_of_your_Life.user.js // @updateURL https://openuserjs.org/meta/navchandar/Show_Price_as_Hours_of_your_Life.meta.js // @setupURL https://openuserjs.org/install/navchandar/Show_Price_as_Hours_of_your_Life.user.js // @icon  // @grant GM_addStyle // @grant GM_getValue // @grant GM_setValue // @include https://play.google.com/store/* // @include https://play.google.com/store* // @include http://play.google.com/store* // @match https://www.google.com/*shop* // @match https://google.com/*shop* // @match https://www.google.co.in/*shop* // @match https://google.co.in/*shop* // @include http://www.amazon.com // @include https://www.amazon.com // @include http://www.amazon.in // @match https://www.amazon.in/* // @include https://www.flipkart.com* // @include http://www.flipkart.com* // @include https://paytm.com* // @include http://paytm.com* // @include https://paytmmall.com* // @include http://paytmmall.com* // ==/UserScript== function has(String, search) { try { if (String.indexOf(search) > -1) { return true; } } catch (err) {} return false; } function updateValuesINR() { var approxHourlySalary = Number(GM_getValue("approxAnnualSalary", "")) / (9 * 22 * 12); if (!approxHourlySalary) { console.log("Error calculating approxHourlySalary.") } if (approxHourlySalary > 0) { var items, separator; // These xpaths cover most of the price elements shown. Feel free to comment/fork to update if required. var amazonINRxpath = "//span[contains(@class, 'CurrencyINR')]//following-sibling::span" + " | " + "//span[@class='currencyINR']/parent::*" + " | " + "//*[contains(@class, 'rice')][contains(text(), '₹')]" + " | " + "//*[contains(@class, 'price')]//*[contains(text(), '₹')]" + " | " + "//*[contains(@class, 'price-whole')]" + " | " + "//h2/span[contains(text(), '₹')]" + " | " + "//div[@data-id]//h3" + " | " + "//*[text() = 'Select Seats']//..//..//..//*[contains(text(), ',')]"; var amazonUSDxpath = "//*[@class='dealPrice']" + " | " + "//*[@class='buyingPrice']" + " | " + "//*[contains(@class,'-price-whole')]" + " | " + "(//span[@class='dv-conditional-linebreak']//following-sibling::text())[1]" + " | " + "(//span[@class='dv-conditional-linebreak']//following-sibling::text())[2]" + " | " + "//span[contains(@class, 'price')][contains(text(), '$')]" + " | " + "//h2[contains(@class, 'headline') and contains(text(), '₹')]" + " | " + "//a[@id='dealTitle']" + " | " + "//div[contains(@class, 'priceBlock')]"; var flipkartINRxpath = "//*[contains(@title, '₹')]/div[3]" + " | " + "//div[contains(text(), '₹')]"; var paytmINRxpath = "//span/span[contains(text(), 'Rs')]" + " | " + "//a//*[contains(@class, 'iconRupess')]//.." + " | " + "//*[contains(@class, 'iconRupess')]//.."; var playStoreINRxpath = "//*[@class='display-price' and contains(text(), '₹')]" + " | " + "//button[@jsmodel]//span[contains(text(), '₹')]" + " | " + "//button[contains(text(), '₹')]//text()" + " | " + "//meta[@itemprop='price']"; var gShopINRxpath = "//div[contains(@class, 'content')]//*[contains(text(), '₹')]" + " | " + "//div[contains(@class, 'result')]//*[contains(text(), '₹')]" + " | " + "//a//div[contains(text(), '₹')]"; var hostURL = window.location.href; if (has(hostURL, 'amazon.in')) { items = document.evaluate(amazonINRxpath, document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); separator = " "; } else if (has(hostURL, 'amazon.com')) { items = document.evaluate(amazonUSDxpath, document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); separator = " "; } else if (has(hostURL, 'flipkart')) { items = document.evaluate(flipkartINRxpath, document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); separator = " "; } else if (has(hostURL, 'paytm')) { items = document.evaluate(paytmINRxpath, document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); separator = "<br/>"; } else if ((has(hostURL, 'google')) && (has(hostURL, 'details'))) { items = document.evaluate(playStoreINRxpath, document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); separator = " "; } else if ((has(hostURL, 'google')) && (has(hostURL, 'store'))) { items = document.evaluate(playStoreINRxpath, document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); separator = "<br/>"; } else if ((has(hostURL, 'google.co')) && (has(hostURL, 'shop'))) { items = document.evaluate(gShopINRxpath, document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); separator = " "; } else { console.log("Show Price Hours script didnt match with the current site."); } if (!items) { console.log("No items found with the xpath in this page."); } else if (items.snapshotLength <= 0) { console.log("No items found with the xpath in this page."); } else { var passCount = 0; for (let i = 0; i < items.snapshotLength; i++) { var thisitem = items.snapshotItem(i); var txt = thisitem.innerText; var htm = thisitem.innerHTML; if (txt == null | txt == "" | htm == null | htm == "") { txt = thisitem.textContent; } var alreadyUpdated = false; try { if (has(txt, 'hrs') | has(txt, 'Day') | has(txt, 'Month') | has(txt, 'YEARS')) { alreadyUpdated = true; } } catch (err) {} try { if (has(htm, 'hrs') | has(htm, 'Day') | has(htm, 'Month') | has(htm, 'YEARS')) { alreadyUpdated = true; } } catch (err) {} if (!alreadyUpdated) { try { // to remove +10% offers from flipkart if (has(txt, '+')) { txt = txt.split('+')[0]; } // to remove hidden text from paytm if (has(txt, '%') && has(txt, '-')) { txt = txt.split('-')[0] } // to remove 100-200 range from amazon if (has(txt, '-')) { txt = txt.split('-')[1]; } if (has(txt, 'to')) { txt = txt.split('to')[1]; } if (has(txt, 'Buy')) { txt = txt.split('Buy')[0]; } // remove unneccessary text and convert to number. txt = (txt.replace(/\d+ Used/i, "").replace(/\d+ New/i, "")).replace(/\(\d+%\)/g, ""); txt = txt.replace(/[a-z]/gi, ""); txt = txt.replace(",", "").replace("Rs.", "").replace("₹", "").replace("$", "").replace(".00", "").replace(" ", ""); var amount = Number(txt); if (!amount || amount <= 0) { amount = Number(((txt.replace(".00", "").replace(/\.\d+/, "")).replace(/\.\d+\./, "")).replace(/\D+/g, "")); } var appender = ""; if (amount > 0) { var calculated = (amount / approxHourlySalary); if (calculated) { if (calculated <= 10) { appender = (separator + "<b>(" + (calculated).toFixed(1).toString() + " hrs)</b>"); thisitem.innerHTML += appender; } else if (calculated < 190) { appender = (separator + "<b>(" + (calculated / 9).toFixed(1).toString() + " Days)</b>"); thisitem.innerHTML += appender; } else if (calculated < 2380) { appender = (separator + "<b>(" + (calculated / 198).toFixed(1).toString() + " Months)</b>"); thisitem.innerHTML += appender; } else { appender = (separator + "<b>(" + (calculated / 2376).toFixed(1).toString() + " YEARS)</b>"); thisitem.innerHTML += appender; } passCount++; } else { console.log("Error calculating amount : " + amount); } } } catch (err) {} } } console.log("Updated hours to " + passCount + " items."); items = ""; } } } function ButtonClickAction(zEvent) { GM_setValue("approxAnnualSalary", ""); var approxAnnualSalary = prompt('To calculate how many hours of your life, each item will cost, add an approximate Annual Salary/Wage Amount :', ''); if (approxAnnualSalary) { GM_setValue("approxAnnualSalary", approxAnnualSalary); console.log("Stored approxAnnualSalary : " + approxAnnualSalary); GM_setValue("UserCancelledPrompt", false); self.location.assign(location); } else { GM_setValue("UserCancelledPrompt", true); console.log("User Cancelled prompt to update approxAnnualSalary."); } } function AddButton() { // Add a button element on div var zNode = document.createElement('div'); zNode.innerHTML = '<button id="myButton" title="To update your stored wage/salary, Click me" type="button">Reset</button>'; zNode.setAttribute('id', 'myContainer'); document.body.appendChild(zNode); //--- Activate the newly added button. document.getElementById("myButton").addEventListener("click", ButtonClickAction, false); //--- Style our newly added element using CSS. GM_addStyle(` #myContainer{position:fixed;bottom:0;left:0;font-size:11px;margin:0;opacity:.75}#myButton{cursor:pointer;background-color:lightgrey;color:black} #myButton:hover{background-color:#555;color:white;box-shadow:0 2px 6px 0 rgba(0,0,0,.24),0 5px 10px 0 rgba(0,0,0,.19);opacity:1} `); } (function () { 'use strict'; //GM_setValue("approxAnnualSalary", ""); // Reset here for test. var approxAnnualSalary = GM_getValue("approxAnnualSalary", ""); var checkPriorCancellation = GM_getValue("UserCancelledPrompt"); if ((approxAnnualSalary == "" || !approxAnnualSalary) && !checkPriorCancellation) { approxAnnualSalary = prompt('To calculate how many hours of your life, each item will cost, add an approximate Annual Salary/Wage Amount :', ''); if (approxAnnualSalary) { GM_setValue("approxAnnualSalary", approxAnnualSalary); GM_setValue("UserCancelledPrompt", false); console.log("Stored approxAnnualSalary : " + approxAnnualSalary); } else { GM_setValue("UserCancelledPrompt", true); console.log("User Cancelled prompt to update approxAnnualSalary."); } } // call this function once AddButton(); // call this every 2 seconds to update according to page load setInterval(function () { updateValuesINR(); }, 2000); })();