您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
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 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 = '';