craftApprentice
craftApprentice

Reputation: 2777

Javascript function to convert decimal years value into years, months and days

I need a function to transform decimal values of years in years, months and days. Ex: 1.5 years = 1 year, 6 month. 2.2527397260273973 years = 2 years, 3 months and 1 day.

I started with this function:

function yearsToYearsMonthsDays(value){
var years = Math.floor(value);
var months =  Math.floor( (value - years) /(1/12) );
var days = Math.floor ( ( (value - years) /(1/12) - Math.floor((value - years) /(1/12)) ) / (1/365) );
var result = years + " years, " + months + " months, " + days + " days";
    Logger.log(result);
}

But sometimes it doesn't work (ex: 0.75 doesn't produces 9 months).

Any help?

Upvotes: 3

Views: 3657

Answers (4)

jorgeregidor
jorgeregidor

Reputation: 208

I think the best way would be to change all to days, for example:

function yearsToYearsMonthsDays(value)
{
    var totalDays = value * 365;
    var years = Math.floor(totalDays/365);
    var months = Math.floor((totalDays-(years *365))/30);
    var days = Math.floor(totalDays - (years*365) - (months * 30));
    var result = years + " years, " + months + " months, " + days + " days";
    Logger.log(result);
}

Upvotes: 6

stallingOne
stallingOne

Reputation: 4006

I stated using jorgeregidor's answer and I generalized it a bit.

function years_to_ymdh(value) {
    console.log('initialy : '+ value + ' years');

    var results=[], rest=value;
    var units=['years', 'months', 'days', 'hours', 'minutes'];
    var converters=[1, 12, 365.25, 365.25*24, 365.25*24*60];

    units.forEach(function(d,i){
        if (i==0) results[i] = Math.floor(rest);
        else results[i] = Math.floor(rest * converters[i]);
        if (results[i] != 0) rest = rest % (results[i]/converters[i]);
        console.log('-'+results[i]+' '+d+' -> rest', rest);
    });

    var text = results.map( (d,i) => d+' '+units[i] ).join(', ');
    console.log(text);

    // return text;
    return results;
}

years_to_ymdh(2.2527397260273973)
-2 years -> rest 0.25273972602739736
-3 months -> rest 0.00273972602739736
-1 days -> rest 0.000001875240265258784
-0 hours -> rest 0.000001875240265258784
-0 minutes -> rest 0.000001875240265258784
2 years, 3 months, 1 days, 0 hours, 0 minutes

To get seconds it's pretty easy to add 'seconds' to the units array and 365.25*24*60*60 to the converters array.

Upvotes: 0

Mac
Mac

Reputation: 1566

My contribution converts decimal years into time units ranging from eons (billion years) to nanoseconds (billionths of a second), and is output formatted with commas in a slightly chatty style.

There are many ways to convert time, and the presumptions have to be understood. Here are mine:

This converts based on 1) an average month having 30.436875 days, per the Gregorian calendar; 2) a month having 730.485 hours average for a 4-year period which includes 1 leap year; and 3) there being 8765.82 hours per year (not 8760). These internal constants won't fit every application, of course, but should be good enough for general use. For example, my function adds about a minute of time to the OP's example of:

2.2527397260273973 years = 2 years, 3 months and 1 day

where

getTimeFromYears( 2.2527397260273973 );

outputs "2 years, 3 months, 1 day and 57 seconds".

Some units seemed unnecessary, such as weeks when you have days and months, and decades when you have years and centuries, but those can easily be added into the logic. Run the snippet to see the examples.

function getTimeFromYears( _years ) {
// to omit a unit from the output, multiply its value by zero; see gyears

    "use strict";

    var _hoursInEon            = 8765820000000     // eon =         1,000,000,000 years (billion)
        , _hoursInGalacticYear = 2016138600000     // galactic year = 230,000,000 years (230 million)
        , _hoursInEpoch        =    8765820000     // epoch =           1,000,000 years (million)
        , _hoursInMillenium    =       8765820     // millenium =           1,000 years
        , _hoursInCentury      =        876582   
        , _hoursInYear         =          8765.82  // often cited as 8760, or 730 * 12  // 365.2425 days in year
        , _hoursInMonth        =           730.485 // hours average per month for a 4-year period which includes 1 leap year
                                                   // average month has 30.436875 days, Gregorian calendar
        , _hoursInDay          =            24
        , _hoursInHour         =             1
        , _hoursInMinute       =             0.0166666666666667
        , _hoursInSecond       =             0.000277778
        , _hoursInMillisecond  =             0.000000277778       // A millisecond is one-thousandth of a second
        , _hoursInNanosecond   =             0.000000000000277778 // A nanosecond is one-billionth of a second


        , totalHours = _years * _hoursInYear
        
        , eons   = Math.floor(   totalHours
                    / _hoursInEon )

        , gyears = Math.floor( ( totalHours - ( eons * _hoursInEon ) )
                    / _hoursInGalacticYear ) * 0

        , epochs = Math.floor( ( totalHours - ( eons * _hoursInEon ) - ( gyears * _hoursInGalacticYear ) )
                    / _hoursInEpoch )

        , mills  = Math.floor( ( totalHours - ( eons * _hoursInEon ) - ( gyears * _hoursInGalacticYear ) - ( epochs * _hoursInEpoch ) )
                    / _hoursInMillenium )

        , cents  = Math.floor( ( totalHours - ( eons * _hoursInEon ) - ( gyears * _hoursInGalacticYear ) - ( epochs * _hoursInEpoch ) - ( mills * _hoursInMillenium ) )
                    / _hoursInCentury )

        , years  = Math.floor( ( totalHours - ( eons * _hoursInEon ) - ( gyears * _hoursInGalacticYear ) - ( epochs * _hoursInEpoch ) - ( mills * _hoursInMillenium ) - ( cents * _hoursInCentury ) )
                    / _hoursInYear )
    
        , months = Math.floor( ( totalHours - ( eons * _hoursInEon ) - ( gyears * _hoursInGalacticYear ) - ( epochs * _hoursInEpoch ) - ( mills * _hoursInMillenium ) - ( cents * _hoursInCentury ) - ( years * _hoursInYear ) )
                    / _hoursInMonth )
    
        , days   = Math.floor( ( totalHours - ( eons * _hoursInEon ) - ( gyears * _hoursInGalacticYear ) - ( epochs * _hoursInEpoch ) - ( mills * _hoursInMillenium ) - ( cents * _hoursInCentury ) - ( years * _hoursInYear ) - ( months * _hoursInMonth ) )
                    / _hoursInDay )
    
        , hours  = Math.floor( ( totalHours - ( eons * _hoursInEon ) - ( gyears * _hoursInGalacticYear ) - ( epochs * _hoursInEpoch ) - ( mills * _hoursInMillenium ) - ( cents * _hoursInCentury ) - ( years * _hoursInYear ) - ( months * _hoursInMonth ) - ( days * _hoursInDay ) )
                    / _hoursInHour )
    
        , mins  = Math.floor( ( totalHours - ( eons * _hoursInEon ) - ( gyears * _hoursInGalacticYear ) - ( epochs * _hoursInEpoch ) - ( mills * _hoursInMillenium ) - ( cents * _hoursInCentury ) - ( years * _hoursInYear ) - ( months * _hoursInMonth ) - ( days * _hoursInDay ) - ( hours * _hoursInHour ) )
                    / _hoursInMinute ) 

        , secs  = Math.floor( ( totalHours - ( eons * _hoursInEon ) - ( gyears * _hoursInGalacticYear ) - ( epochs * _hoursInEpoch ) - ( mills * _hoursInMillenium ) - ( cents * _hoursInCentury ) - ( years * _hoursInYear ) - ( months * _hoursInMonth ) - ( days * _hoursInDay ) - ( hours * _hoursInHour ) - ( mins * _hoursInMinute ) )
                    / _hoursInSecond )

        , msecs = Math.floor( ( totalHours - ( eons * _hoursInEon ) - ( gyears * _hoursInGalacticYear ) - ( epochs * _hoursInEpoch ) - ( mills * _hoursInMillenium ) - ( cents * _hoursInCentury ) - ( years * _hoursInYear ) - ( months * _hoursInMonth ) - ( days * _hoursInDay ) - ( hours * _hoursInHour ) - ( mins * _hoursInMinute ) - ( secs * _hoursInSecond ) )
                    / _hoursInMillisecond )

        , nsecs = Math.floor( ( totalHours - ( eons * _hoursInEon ) - ( gyears * _hoursInGalacticYear ) - ( epochs * _hoursInEpoch ) - ( mills * _hoursInMillenium ) - ( cents * _hoursInCentury ) - ( years * _hoursInYear ) - ( months * _hoursInMonth ) - ( days * _hoursInDay ) - ( hours * _hoursInHour ) - ( mins * _hoursInMinute ) - ( secs * _hoursInSecond ) - ( msecs * _hoursInMillisecond ) )
                    / _hoursInNanosecond )


        , _eonsPhrase   = ( eons   < 1 ? '' : eons   === 1 ? '1 eon'           : addCommas( eons   ) + ' eons'           )
        , _gyearsPhrase = ( gyears < 1 ? '' : gyears === 1 ? '1 galactic year' : addCommas( gyears ) + ' galactic years' )
        , _epochsPhrase = ( epochs < 1 ? '' : epochs === 1 ? '1 epoch'         : addCommas( epochs ) + ' epochs'         )
        , _millsPhrase  = ( mills  < 1 ? '' : mills  === 1 ? '1 millenium'     : mills  + ' millenia'  )
        , _centsPhrase  = ( cents  < 1 ? '' : cents  === 1 ? '1 century'       : cents  + ' centuries' )
        , _yearsPhrase  = ( years  < 1 ? '' : years  === 1 ? '1 year'          : years  + ' years'     )
        , _monthsPhrase = ( months < 1 ? '' : months === 1 ? '1 month'         : months + ' months'    )
        , _daysPhrase   = ( days   < 1 ? '' : days   === 1 ? '1 day'           : days   + ' days'      )
        , _hoursPhrase  = ( hours  < 1 ? '' : hours  === 1 ? '1 hour'          : hours  + ' hours'     )
        , _minsPhrase   = ( mins   < 1 ? '' : mins   === 1 ? '1 minute'        : mins   + ' minutes'   )
        , _secsPhrase   = ( secs   < 1 ? '' : secs   === 1 ? '1 second'        : secs   + ' seconds'   )
        , _msecsPhrase  = ( msecs  < 1 ? '' : msecs  === 1 ? '1 millisecond'   : addCommas( msecs ) + ' milliseconds' )
        , _nsecsPhrase  = ( nsecs  < 1 ? '' : nsecs  === 1 ? '1 nanosecond'    : addCommas( nsecs ) + ' nanoseconds'  )

        , _phrasePart = new Array(13)
        , _phrasesInUse = 0
        , _phrasesUsed  = 0
        , _joiner = ','
        , _result = ''
    ;


    ////////////////////////////////////////////////////
    // cosmetic adjustments
    if( eons > 999999 ) { _joiner = ';'; }
    if( secs > 0 || mins > 0 ) { _msecsPhrase = _nsecsPhrase = ''; } 
    ////////////////////////////////////////////////////

    _phrasePart[  0 ] = _eonsPhrase;
    _phrasePart[  1 ] = _gyearsPhrase;
    _phrasePart[  2 ] = _epochsPhrase;
    _phrasePart[  3 ] = _millsPhrase;
    _phrasePart[  4 ] = _centsPhrase;
    _phrasePart[  5 ] = _yearsPhrase;
    _phrasePart[  6 ] = _monthsPhrase;
    _phrasePart[  7 ] = _daysPhrase;
    _phrasePart[  8 ] = _hoursPhrase;
    _phrasePart[  9 ] = _minsPhrase;
    _phrasePart[ 10 ] = _secsPhrase;
    _phrasePart[ 11 ] = _msecsPhrase;
    _phrasePart[ 12 ] = _nsecsPhrase;

    // count the phrase pieces to use
    for( var i = 0; i < _phrasePart.length; i++ ) {
        if( _phrasePart[ i ].length ) { _phrasesInUse++; }
    }

    // assemble the output
    for( var i = 0; i < _phrasePart.length; i++ ) {
        if( _phrasePart[ i ].length ) {
            _result += _phrasePart[ i ];
            _phrasesUsed++;

            // only for the last phrase
            if( _phrasesInUse - _phrasesUsed === 1 ) {
                _result += ' and ';
            
            } else
            if( _phrasesInUse - _phrasesUsed > 0 ) {
                _result += ( _joiner + ' ' );
            }
        }
    }

    return _result;
};


function addCommas(t) {
    t += ""; var x = t.split("."), x1 = x[0], x2 = x.length > 1 ? "." + x[1] : "";
    for (var r = /(\d+)(\d{3})/; r.test(x1); ) x1 = x1.replace(r, "$1,$2");
    return x1 + x2;
};


console.log( 'getTimeFromYears( .75 ) -> ' + getTimeFromYears( .75 ) ); // 9 months
console.log( 'getTimeFromYears( 9/12 ) -> ' + getTimeFromYears( 9/12 ) ); // 9 months
console.log( 'getTimeFromYears( 1/365.2425 ) -> ' + getTimeFromYears( 1/365.2425 ) ); // 1 day
console.log( 'getTimeFromYears( 1/365.2425 * 14 ) -> ' + getTimeFromYears( 1/365.2425 * 14 ) ); // 14 days
console.log( 'getTimeFromYears( 1/365.2425 / 24 ) -> ' + getTimeFromYears( 1/365.2425 / 24 ) ); // 1 hour
console.log( 'getTimeFromYears( 1/365.2425 / 24 * 730.485 ) -> ' + getTimeFromYears( 1/365.2425 / 24 * 730.485 ) ); // 1 month
console.log( 'getTimeFromYears( 1/365.2425 * 0.0000000000000115741 ) -> ' + getTimeFromYears( 1/365.2425 * 0.0000000000000115741 ) ); // 1 nanosecond
console.log( 'getTimeFromYears( 1/365.2425 / 24 * 0.000000000000277778 ) -> ' + getTimeFromYears( 1/365.2425 / 24 * 0.000000000000277778 ) ); // 1 nanosecond
console.log( 'getTimeFromYears( 1/365.2425 / 24 * 0.000000000000277778 * 12000 ) -> ' + getTimeFromYears( 1/365.2425 / 24 * 0.000000000000277778 * 12000 ) ); // 12,000 nanoseconds
console.log( 'getTimeFromYears( 1/365.2425 / 24 * 0.000000000000277778 * 1000000 ) -> ' + getTimeFromYears( 1/365.2425 / 24 * 0.000000000000277778 * 1000000 ) ); // 1 millisecond
console.log( 'getTimeFromYears( 1/365.2425 / 24 * 0.000000000000277778 * 12345678 ) -> ' + getTimeFromYears( 1/365.2425 / 24 * 0.000000000000277778 * 12345678 ) ); // 12 milliseconds and 345,678 nanoseconds
console.log( 'getTimeFromYears( 123456789 ) -> ' + getTimeFromYears( 123456789 ) ); // 123 epochs, 456 millenia, 7 centuries, 88 years, 11 months, 30 days, 10 hours, 29 minutes and 5 seconds
console.log( 'getTimeFromYears( 9876543210 ) -> ' + getTimeFromYears( 9876543210 ) ); // 9 eons, 876 epochs, 543 millenia, 2 centuries, 10 years and 11 seconds
console.log( 'getTimeFromYears( 2.2527397260273973 ) -> ' + getTimeFromYears( 2.2527397260273973 ) ); // 2 years, 3 months, 1 day and 57 seconds

Upvotes: 2

Lo&#239;c
Lo&#239;c

Reputation: 11943

Your script works well here (tested in firebug console).

A call to your function with 0.75 as a parameter produces :

0 years, 9 months, 0 days

What do you exactly want it to do?

If you want it to remove the "0" values, then you need to test them and don't append them into the string if they are equals to 0, eg :

var result = "";
if(years !=0){result += year + " years"}

Upvotes: 0

Related Questions