Greasy Fork

GitLab Total Time

Add total time to header of lists on GitLab boards

// ==UserScript==
// @name            GitLab Total Time
// @namespace       https://github.com/LibreCodeCoop/gitlab-time-userscript/
// @version         0.3
// @description     Add total time to header of lists on GitLab boards
// @author          Vitor Mattos
// @supportURL      https://github.com/LibreCodeCoop/gitlab-time-userscript/issues
// @contributionURL https://patreon.com/librecode
// @license         AGPL-3.0-or-later
// @match           https://gitlab.com/*/*/boards/*
// @match           http*://*/*/boards
// @match           http*://*/*/boards?*
// @match           http*://*/*/boards/*
// @icon            
// @require         https://code.jquery.com/jquery-3.6.0.min.js
// ==/UserScript==

function listTotalTime() {
    $('header.board-header div.issue-count-badge').each(function() {
        var container = $(this).children();
        var timeElement = container.find('span.list-total-time');
        if(timeElement.length === 0) {
            container.append('<span class="gl-display-inline-flex gl-ml-3 list-total-time"></span>')
        }
        timeElement = container.find('span.list-total-time')

        timeElement.attr('time_m', 0)
        timeElement.attr('time_h', 0)
        timeElement.attr('time_d', 0)
    });

    $('time:not([datetime]').each(function() {
        var countElement = $(this).closest('div.board').find('span.list-total-time');
        var time = $(this).text()
        var timeArray = time.split(" ")
        for (var i = 0; i < timeArray.length; i++) {
            var type = timeArray[i].slice(-1);
            time = timeArray[i].slice(0, -1);
            if (type === "d") {
                countElement.attr(
                    "time_d",
                    parseInt(countElement.attr("time_d")) + parseInt(time)
                );
            } else if (type === "h") {
                countElement.attr(
                    "time_h",
                    parseInt(countElement.attr("time_h")) + parseInt(time)
                );
            } else if (type === "m") {
                countElement.attr(
                    "time_m",
                    parseInt(countElement.attr("time_m")) + parseInt(time)
                );
            }
        }
    });

    $('span.list-total-time').each(function() {
        var min = parseInt($(this).attr('time_m'))
        var hour = parseInt($(this).attr('time_h'));
        var day = parseInt($(this).attr('time_d')) * 8;
        hour += day;
        if (min >= 60) {
            hour += min / 60;
            min = min % 60;
        }
        $(this).attr('time_h', hour)
        $(this).attr('time_m', min)
        var minPad = $(this).attr('time_m')
        minPad = ('00' + minPad).slice(-2)
        $(this).html(
            '<svg role="img" aria-hidden="true" class="gl-mr-2 gl-icon s14" data-testid="hourglass-icon"><use href="/assets/icons-ffa14d1d14478de17bd5c7220bf466194ad3bc99589858dae76a86bc89017324.svg#hourglass"></use></svg>'
            + $(this).attr('time_h')
            + ':'
            + minPad
        )
    });
}

$(document).ready(function() {
    setTimeout(function () {
        $('<div class="gl-ml-3 gl-display-flex gl-align-items-center"><button title="" data-qa-selector="boards_config_button" type="button" class="btn btn-default btn-md gl-button" id="refresh-list-time"><!----> <!---->  <span class="gl-button-text">Refresh times</span></button></div>').insertBefore($('[data-testid="board-options-dropdown"]'));

        $(document).on("click", "#refresh-list-time", function() {
            listTotalTime();
        });
    }, 1000);
});