how to get date for holidays using php

23,494

Solution 1

You can leverage this function of php. strtotime

$currentYear = date("Y");

MLK Day -

echo date('Y-m-d', strtotime("third monday of january $currentYear"));

Presidents Day -

echo date('Y-m-d', strtotime("third monday/OD February $currentYear"));

Easter -

echo date('Y-m-d', strtotime("last sunday of march $currentYear"));

Memorial Day -

echo date('Y-m-d', strtotime("last monday of may $currentYear"));

Solution 2

The strtotime function is useful here but it seems to have some peculiarities on how it interprets the English verbiage. Be sure to check your wording carefully and check your results.

For instance this seems like it should work, but it will return the last Monday in April ($currentYear = 2014)!

echo "Memorial Day " . date('F d, Y', strtotime("may $currentYear last monday")) . "<br />";

However, this phrasing will return the correct holiday for 2014.

echo "Memorial Day " . date('F d, Y', strtotime("last monday of May $currentYear")) . "<br />"; 

And this will give you the correct date for the holiday next year.

echo "Memorial Day " . date('F d, Y', strtotime("last monday of May $currentYear+1")) . "<br />";

The PHP doc page has additional comments on its use. Take care

Solution 3

This way worked for me.

function observed_date($holiday){
    $day = date("w", strtotime($holiday));
    if($day == 6) {
        $observed_date = $holiday -1;
    } elseif ($day == 0) {
        $observed_date = $holiday +1;
    } else {
        $observed_date = $holiday;
    }
    return $observed_date;
}

function get_holiday($holiday_name) {

    $currentYear = date('Y');

    switch ($holiday_name) {
        // New Years Day
        case "new_year":
            $holiday = observed_date(date('Ymd', strtotime("first day of january $currentYear")));
            break;
        // Martin Luther King, Jr. Day
        case "mlk_day":
            $holiday = date('Ymd', strtotime("january $currentYear third monday"));
            break;
        // President's Day
        case "presidents_day":
            $holiday = date('Ymd', strtotime("february $currentYear third monday"));
            break;
        // Memorial Day
        case "memorial_day":
            $holiday = (new DateTime("Last monday of May"))->format("Ymd");
            break;
        // Independence Day
        case "independence_day":
            $holiday = observed_date(date('Ymd', strtotime("july 4 $currentYear")));
            break;
        // Labor Day
        case "labor_day":
            $holiday = date('Ymd', strtotime("september $currentYear first monday"));
            break;
        // Columbus Day
        case "columbus_day":
            $holiday = date('Ymd', strtotime("october $currentYear second monday"));
            break;
        // Veteran's Day
        case "veterans_day":
            $holiday = observed_date(date('Ymd', strtotime("november 11 $currentYear")));
            break;
        // Thanksgiving Day
        case "thanksgiving_day":
            $holiday = date('Ymd', strtotime("november $currentYear fourth thursday"));
            break;
        // Christmas Day
        case "christmas_day":
        $holiday = observed_date(date('Ymd', strtotime("december 25 $currentYear")));
            break;

        default:
            $holiday = "";
            break;
    }
    return $holiday;

}

Then you can print out the holiday by calling the function echo get_holiday('new_year');

And the result for 2015 should be 20150101

Solution 4

Solution for moving holydays in Germany: returns an array of 'Y-m-d' formated strings including good friday, easter monday, ascencion, and pentecost monday. A new DateTime object for easter ist created for each day to enhance readability.

function beweglicheFeiertage( int $year ):array
{
    //Karfreitag - good friday
    $return = array();
    $easterDate = DateTime::createFromFormat('U', easter_date($year) );
    $easterDate->modify('- 1 days');
    $return[] = $easterDate->format('Y-m-d');
    //Ostermontag easter monday
    $easterDate = DateTime::createFromFormat('U', easter_date($year) );
    $easterDate->modify('+ 2 day');
    $return[] = $easterDate->format('Y-m-d');
    //Himmelfahrt ascencion
    $easterDate = DateTime::createFromFormat('U', easter_date($year) );
    $easterDate->modify('+ 40 days');//go to Ascencionday
    $return[] = $easterDate->format('Y-m-d');
    //Pfingstmontag - pentecost monday
    $easterDate = DateTime::createFromFormat('U', easter_date($year) );
    $easterDate->modify('+ 51 days');//go to Pentecost Monday
    $return[] = $easterDate->format('Y-m-d');

    return $return;
}

Solution 5

I created a function that calculates all the federal holidays (US), for any given year and returns them inside an array. I've tested it with several different years (from 2016 to 2024). Here is the code, please let me know if you find any miscalculations:

    //function that checks if a holiday lands on saturday/sunday and so we can move them to a friday/monday respectively
    private function getObservedDate($holidayDate){

        $dayofweek = date("w", strtotime($holidayDate));

        if ($dayofweek == 6) $holidayDate = date('m/d/Y', strtotime("$holidayDate - 1 days")); //saturday moves to friday
        else if ($dayofweek == 0)  $holidayDate = date('m/d/Y', strtotime("$holidayDate + 1 days"));  //sunday moves monday

        return $holidayDate;
    }


    //function that calculates the holidays for any given year
    private function getFederalHolidaysForYear($year){

        $NY = $this->getObservedDate( date('m/d/Y', strtotime("1/1/$year")) ); //new years day

        $MLK = $this->getObservedDate( date('m/d/Y', strtotime("third monday of january $year")) );  //martin luther king day

        $PD = $this->getObservedDate( date('m/d/Y', strtotime("third monday of february $year")) ); ; //presidents day

        $MDay = $this->getObservedDate( date('m/d/Y', strtotime("last monday of May $year")) ); //memorial day  

        $IDay = $this->getObservedDate( date('m/d/Y', strtotime("7/4/$year")) );  // independence day

        $LD = $this->getObservedDate( date('m/d/Y', strtotime("first monday of september $year")) ); //labor day

        $VD = $this->getObservedDate( date('m/d/Y', strtotime("11/11/$year")) ); //veterans day

        $ColD =$this->getObservedDate( date('m/d/Y', strtotime("second monday of october $year")) ); //columbus day   

        $TG = $this->getObservedDate( date('m/d/Y', strtotime("last thursday of november $year")) ); // thanksgiving       

        $CD = $this->getObservedDate( date('m/d/Y', strtotime("12/25/$year")) );  //christmas day 

        $nonWorkingDays = array();

        array_push($nonWorkingDays, $NY, $MLK, $PD, $MDay, $IDay, $LD, $ColD, $VD, $TG, $CD);

        return $nonWorkingDays;
    }
Share:
23,494
telexper
Author by

telexper

Updated on July 09, 2022

Comments

  • telexper
    telexper almost 2 years

    For example the:

    1. 3rd Monday of January for Martin Luther King Day
    2. 3rd Monday of February for Presidents' Day (Washington's Birthday)
    3. Last Sunday of March for Easter
    4. Last Monday of May for Memorial Day

    I am trying to get these dates, so that I can mark it on my calendar without manually putting everything for the years to come.


    MODIFIED ANSWER!!

    $curYir = date("Y");//current year
    
    $MLK = date('Y-m-d', strtotime("january $curYir third monday")); //marthin luthor king day
    $PD = date('Y-m-d', strtotime("february $curYir third monday")); //presidents day
    $Est =  date('Y-m-d', easter_date($curYir))); // easter 
    $MDay = date('Y-m-d', strtotime("may $curYir first monday")); // memorial day
    //("may $curYir last monday") will give you the last monday of may 1967
    //much better to determine it by a loop
          $eMDay = explode("-",$MDay);
          $year = $eMDay[0];
          $month = $eMDay[1];
          $day = $eMDay[2];
    
          while($day <= 31){
              $day = $day + 7;
          }
          if($day > 31)
          $day = $day - 7;
    
          $MDay = $year.'-'.$month.'-'.$day;
    $LD = date('Y-m-d', strtotime("september $curYir first monday"));  //labor day
    $CD = date('Y-m-d', strtotime("october $curYir third monday")); //columbus day
    $TH = date('Y-m-d', strtotime("november $curYir first thursday")); // thanks giving 
    //("november $curYir last thursday") will give you the last thursday of november 1967
    //much better to determine it by a loop
          $eTH = explode("-",$TH);
          $year = $eTH[0];
          $month = $eTH[1];
          $day = $eTH[2];
    
          while($day <= 30){
              $day = $day + 7;
          }
          if($day > 30)
          //watch out for the days in the month November only have 30
          $day = $day - 7;
    
          $TH = $year.'-'.$month.'-'.$day;
    
  • Matt James
    Matt James over 8 years
    Easter isn't always the last Monday in March. Better to use PHP's built in easter_date() function: php.net/manual/en/function.easter-date.php
  • Etienne Noël
    Etienne Noël about 8 years
    Are you forgetting easter?
  • Javis
    Javis almost 8 years
    This doesn't always returns the desired date. Check @Steve-Isenberg comment
  • TGates
    TGates about 6 years
    Optionally, if you just want the day for the current year you can shorten the above like so: Memorial Day - echo date('m-d', strtotime("last monday of march"));
  • luenib
    luenib over 5 years
    Memorial Day = last monday of may
  • kmoser
    kmoser over 3 years
    Adding and subtracting 1 from dates that are in YYYYMMDD format is a bad idea: if the date happens to be the first day of the month, rolling back by 1 day will give you an invalid date. For example, 1/1/2011 (Near Year's Day) is a Saturday, and subtracting 1 from 20110101 gives you 20110100. Same if the holiday falls on the last day of the month which happens to be a Sunday: if you add 1 day, you'll end up with an invalid date.
  • Max
    Max over 3 years
    That might be true for the date() functions. In the modify() method of Datetime however this is taken care of.
  • kmoser
    kmoser over 3 years
    Adding and subtracting 1 from dates that are in YYYYMMDD format is a bad idea: if the date happens to be the first day of the month, rolling back by 1 day will give you an invalid date. For example, 1/1/2011 (Near Year's Day) is a Saturday, and subtracting 1 from 20110101 gives you 20110100. Same if the holiday falls on the last day of the month which happens to be a Sunday: if you add 1 day, you'll end up with an invalid date.
  • kmoser
    kmoser over 3 years
    My mistake, my comment was meant for the post above you by @Amen Ra.