NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name VTPortal total time calculator // @description Make VTPortal calculate the total worked time. Written by @alopez // @copyright 2023, Aritz // @license MIT // @version 5 // @author Aritz Lopez // @collaborator Marcos Ruiz // @grant none // @match https://sign3910.visualtime.net/* // @match https://vtportal.visualtime.net/* // @namespace https://greasyfork.org/users/855840 // ==/UserScript== /* jshint esversion: 10 */ function update_total_time() { if (!document.querySelector("div#punchesList")) return; if (window.eval('i18nextko.i18n.lng();') !== last_language_code) { update_language_data().then(() => { update_total_time(); }); return; } let totalElement = document.querySelector("span#totalTimeElement"); if (!totalElement) { totalElement = document.createElement("span"); totalElement.id = "totalTimeElement" totalElement.style.fontSize = "1.5rem"; if (document.querySelector('div[data-options*="punchesHome"]').nextElementSibling) { totalElement.style.marginLeft = '20%'; totalElement.style.top = '1rem'; totalElement.style.position = 'relative'; const divider = document.querySelector('div[data-options*="punchesHome"]').nextElementSibling; divider.parentNode.insertBefore(totalElement, divider.nextSibling); } else if (document.querySelector("div#punchesList")) { document.querySelector("div#punchesList").appendChild(totalElement); } } let remainingTimeElement = document.querySelector("span#remainingTime"); if (!remainingTimeElement) { remainingTimeElement = document.createElement("span"); remainingTimeElement.id = "remainingTime" remainingTimeElement.style.fontSize = "1.5rem"; remainingTimeElement.style.top = totalElement.style.top; remainingTimeElement.style.position = 'relative'; totalElement.parentNode.insertBefore(remainingTimeElement, totalElement.nextSibling); } const punches = Array.prototype.map.call( document.querySelectorAll('div#punchesList div[data-bind="text: $data.Name"]'), function (d) { return d.innerHTML } ) let totalTime = 0; let lastEntry = 0; for (let punch of punches) { const punchParts = punch.split(":"); const time = parseInt(punchParts[1].trim()) * 60 + parseInt(punchParts[2]); if (all_enter_options.includes(punchParts[0])) { if (lastEntry != 0) { totalElement.innerHTML = "Error: Two consecutive entries"; return } else { lastEntry = time; } } else { if (lastEntry > time) { // Previous entry was the day before totalTime += 24 * 60 - lastEntry + time; } else { // If there was no last entry, assume it was the day before, and so calculate since midnight, by subtracting 0 in lastEntry totalTime += (time - lastEntry); } lastEntry = 0; } } // If last entry was not exited, calculate until now if (lastEntry != 0) { const current = new Date(); const exitTime = current.getHours() * 60 + current.getMinutes(); totalTime += (exitTime - lastEntry); } let remainingTime = totalTime - theoretical_minutes; const remaining_hours_str = Math.floor(Math.abs(remainingTime) / 60).toString().padStart(2, "0"); const remaining_minutes_str = (Math.abs(remainingTime) % 60).toString().padStart(2, "0"); if (remainingTime < 0) { remainingTimeElement.style.color = "red"; remainingTimeElement.innerHTML = `${remaining_message}: -${remaining_hours_str}:${remaining_minutes_str} (${remainingTime} min.)` } else { remainingTimeElement.style.color = "green"; remainingTimeElement.innerHTML = `${remaining_message}: ${remaining_hours_str}:${remaining_minutes_str} 🍺 (${remainingTime} min.)` } const hours_str = Math.floor(totalTime / 60).toString().padStart(2, "0"); const minutes_str = (totalTime % 60).toString().padStart(2, "0"); totalElement.innerHTML = `${total_message}: ${hours_str}:${minutes_str} - `; } let all_enter_options = []; let total_message = "Total"; let remaining_message = "Saldo"; let last_language_code = "en"; let theoretical_minutes = 8.5 * 60; async function update_language_data() { const language_code = window.eval('i18nextko.i18n.lng();') last_language_code = language_code; const punch_lang_response = await fetch(`https://vtportal.visualtime.net/2/js/localization/vtportal.i18n.${language_code}.json`); const punch_lang_data = await punch_lang_response.json(); const generic_lang_response = await fetch(`https://vtportal.visualtime.net/2/js/localization/dx.all.${language_code}.json`); const generic_lang_data = await generic_lang_response.json(); all_enter_options = [ punch_lang_data.roPunches_TA_in, punch_lang_data.roPunches_TA_in_cause, punch_lang_data.roPunches_TA_in_causeHome, punch_lang_data.roEntry ]; total_message = generic_lang_data[language_code]["dxPivotGrid-total"]; parts = total_message.split(' '); total_message = parts[parts.length - 1]; total_message = total_message.charAt(0).toUpperCase() + total_message.slice(1); remaining_message = punch_lang_data.roAccrualLbl; } const get_current_day_info_promise = () => { return window.eval('new Promise((resolve, reject) => {new WebServiceRobotics(function (t) {resolve(t);}).getEmployeeDayInfo(undefined, - 1);})') }; async function get_theoretical_hours() { const day_info = await get_current_day_info_promise(); // The request name says "Hours" but is in fact minutes :( theoretical_minutes = day_info.DayInfo.DayData[0].MainShift.PlannedHours; } function prepare() { Promise.all([ update_language_data(), get_theoretical_hours(), ]).then(() => { update_total_time(); setInterval(update_total_time, 5000); }); } // Wait for the #punchesList element to be present, at that point, it is ready to calculate const observer = new MutationObserver(function(mutations_list) { mutations_list.forEach(function(mutation) { mutation.addedNodes.forEach(function(added_node) { if(added_node.id == 'punchesList') { observer.disconnect(); setTimeout(prepare, 100); } }); }); }); observer.observe(document.documentElement, { subtree: true, childList: true }); var link = document.querySelector("link[rel~='icon']"); if (!link) { link = document.createElement('link'); link.rel = 'icon'; document.getElementsByTagName('head')[0].appendChild(link); } link.href = '';