Greasy Fork is available in English.

VTPortal total time calculator

Make VTPortal calculate the total worked time. Written by @alopez

// ==UserScript==
// @name     VTPortal total time calculator
// @description Make VTPortal calculate the total worked time. Written by @alopez
// @copyright 2023, Aritz
// @license MIT
// @version  4
// @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 (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 = 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 new Promise((resolve, reject) => {
        new WebServiceRobotics(function (t) {return 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 = '';