// ==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
// @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*
// @namespace https://gf.qytechs.cn/users/302262
// ==/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);
})();