NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name Flatten Zimbra Calendar Background Colors and Grey Elapsed Time and Events DEV // @version 1.3 // @author You // @match https://partage.unistra.fr/* // @grant none // @license MIT // ==/UserScript== //Observer that checks if the calendar node exists var calendarObserver = new MutationObserver(function (mutations, observer) { console.log("Document mutation detected"); //If the calendar node exists if (document.getElementsByClassName("ZmCalViewMgr").length > 0) { console.log("Calendar detected"); //regex to identify if the correct style is applied afterwards var re = /rgb\((\d{1,3}), (\d{1,3}), (\d{1,3})\) none repeat scroll 0% 0%/; //Create an events observer that will change all backgrounds when triggered var eventsObserver = new MutationObserver(function (mutations, observer) { console.log("Event mutation detected"); changeBackgroundStyle(eventsObserver, re); }); //Events observer is triggered when background attributes of calendar node or any of its children are modified eventsObserver.observe(document.getElementsByClassName("ZmCalViewMgr")[0], { subtree: true, attributes: true, attributesFilter: ['background', 'opacity'], childList: true }); //We disconnect the calendar observer since the events observer is now up and running calendarObserver.disconnect(); } }); //Calendar observer is triggered whenever a change is made in the document calendarObserver.observe(document, { subtree: true, attributes: true }); //Function to change the backgrounds to a flatter, easier to read format function changeBackgroundStyle(eventsObserver, re) { //Get current date, day, hours and minutes var currentDate = new Date(); var currentDay = currentDate.getDay(); var currentHour = currentDate.getHours() + ':' + currentDate.getMinutes(); var lengthOfDay = Date.parse("01/01/1970 17:00") - Date.parse("01/01/1970 8:00"); var advancementOfDay = ((Date.parse("01/01/1970 " + currentHour) - Date.parse("01/01/1970 8:00")) / lengthOfDay * 100).toFixed(1); //console.log(advancementOfDay); //Temporarily disconnect the events observer otherwise our background manipulations will trigger an infinite loop (not needed for firefox 66+, how come ?) eventsObserver.disconnect(); //Retrieve all day grids var events = document.querySelectorAll('*[id$="_tableBody"]'); var dayGrids = Array.from(document.getElementsByClassName('ImgCalendarDayGrid')); dayGrids.splice(0, 1); var dayGridDay = 0; //For all day grids dayGrids.forEach(function (dayGrid) { //console.log(dayGrid); if (dayGrid.style.width != "" && dayGrid.style.height != "") { dayGridDay++; if (dayGridDay < currentDay) { dayGrid.parentNode.style.background = "rgba(227,227,220,1)"; } else if (dayGridDay === currentDay) { //console.log("linear-gradient(180deg, rgba(227,227,220,1) 0%, rgba(227,227,220,1) " + advancementOfDay + "%, rgba(255,255,255,1) " + (parseInt(advancementOfDay) + 1) + "%, rgba(255,255,255,1) 100%);") dayGrid.parentNode.style.background = "linear-gradient(180deg, rgba(227,227,220,1) 0%, rgba(227,227,220,1) " + advancementOfDay + "%, rgba(255,255,255,1) " + (parseInt(advancementOfDay) + 1) + "%,rgba(255,255,255,1) 100%)"; } } }); //For all events events.forEach(function (event) { //Delete the very small and useless border between the vertical bar and the event box itself event.style.border = 0; //If event has end date node, get its value otherwise build it based on the height of the event if (event.firstChild.childNodes && event.firstChild.childNodes.length > 2) { var eventEndTime = event.firstChild.childNodes[2].textContent; } else { let eventHeight = event.parentNode.style.height; let eventDuration; if (eventHeight === "15.4px") { eventDuration = 15; } else if (eventHeight === "21px") { eventDuration = 30; } else if (eventHeight === "31.5px") { eventDuration = 45; } else if (eventHeight === "42px") { eventDuration = 60; } let eventStartTimeHours = event.firstChild.firstChild.textContent.split(':')[0]; let eventStartTimeMinutes = event.firstChild.firstChild.textContent.split(':')[1]; let eventEndTimeHours = eventStartTimeHours; let eventEndTimeMinutes = (eventStartTimeMinutes + eventDuration) % 60; if (eventEndTimeMinutes <= eventStartTimeMinutes) { eventEndTimeHours += 1; } eventEndTime = eventEndTimeHours + ':' + eventEndTimeMinutes; } //Get number of day of the event of event in work week display mode, 1 is monday, 2 is tuesday etc let eventDay = Math.floor(parseInt(event.parentNode.parentNode.style.left.split('p')[0]) / 294 + 1); // console.log(event); // console.log(eventDay); // console.log(currentDay); // console.log(eventEndTime); // console.log(currentHour); //If event was on an earlier day or ended earlier today make it transparent if (eventDay < currentDay || (eventDay === currentDay && Date.parse("01/01/1970 " + currentHour) > Date.parse("01/01/1970 " + eventEndTime))) { event.style.opacity = 0.5; } //If the correct style is not applied to the event background if (!re.test(event.style.background)) { //Retrieve the color of the event let eventColor = event.style.background.split("rgb(")[2].split(")")[0]; //Set the background style to a flat version of that color event.style.background = "rgba(" + eventColor + ", 1) repeat scroll 0% 0%"; //Set the background style of the events for which the previous line didn't work because their apperance is linked to the style of a child node (no idea why) event.firstChild.firstChild.style.background = "rgba(" + eventColor + ", 1)"; } //If the correct style is not applied to the state side bar if (event.firstChild.firstChild.firstChild.style.background != "" && !re.test(event.firstChild.firstChild.firstChild.style.background)) { //Set the background style of the small "state" vertical bar let eventStateColor = event.firstChild.firstChild.firstChild.style.background.split("rgb(")[2].split(")")[0]; event.firstChild.firstChild.firstChild.style.background = "rgba(" + eventStateColor + ", 1) repeat scroll 0% 0%"; } }); //Reconnect the events observer eventsObserver.observe(document.getElementsByClassName("ZmCalViewMgr")[0], { subtree: true, attributes: true, attributesFilter: ['background', 'opacity'], childList: true }); //Stop timer //let time = new Date() - start; //Log time it took in ms //console.log(time + "ms"); }