LoginSignup
20
16

More than 3 years have passed since last update.

JavaScriptで休日判定

Last updated at Posted at 2017-08-25

日付から休日を判定するスクリプト

追記

以下を対応しました。

「国民の祝日」について

国民の祝日に関する法律(昭和23年法律第178号)の一部改正について

天皇の退位等に関する皇室典範特例法(平成29年法律第63号。以下「皇室典範特例法」という。)附則第10条により、国民の祝日に関する法律が一部改正されるため、皇室典範特例法の施行の日(平成31年4月30日)の翌日より、天皇誕生日が12月23日から2月23日になります。

「天皇の即位の日及び即位礼正殿の儀の行われる日を休日とする法律」について

天皇の即位の日の平成31年(2019年)5月1日及び即位礼正殿の儀が行われる日の平成31年(2019年)10月22日は、休日となります。また、これらの休日は国民の祝日扱いとなるため、平成31年(2019年)4月30日と5月2日も休日となります

「スポーツの日」について

平成32年(2020年)以降、「体育の日」は「スポーツの日」になります。

国民の祝日に関する法律(昭和23年法律第178号)の特例について

平成32年(2020年)に限り、「海の日」は7月23日に、「体育の日(スポーツの日)は7月24日に、「山の日」は8月10日になります。

参考

https://ja.wikipedia.org/wiki/国民の祝日に関する法律
https://ja.wikipedia.org/wiki/国民の祝日
https://ja.wikipedia.org/wiki/国民の休日

判定順序

国民の祝日を判定⇒振替休日を判定⇒国民の休日を判定

var Holiday = (function () {

    /**
     * 休日取得
     * @param Date date
     * @return string
     */
    var getHolidayName = function (date) {

        var holidayName = "";

        if (typeof date != "object") {
            return holidayName;
        }

        try {
            date.getTime();
        } catch (error) {
            return holidayName;
        }

        holidayName = _getPublicHolidayName(date);

        if (holidayName !== "") {
            return holidayName;
        }

        var year = date.getFullYear();
        var month = date.getMonth() + 1;
        var day = date.getDate();
        var weekday = date.getDay();

        // 振替休日の判定

        var timeStamp = date.getTime();
        // 前日(dateをベースに別インスタンスで作成)
        var previousDate = new Date(timeStamp);
        previousDate.setDate(date.getDate() - 1);
        // 前日(dateをベースに別インスタンスで作成)
        var previousDateForLoop = new Date(timeStamp);
        previousDateForLoop.setDate(date.getDate() - 1);

        // 1986-2006の5月4日の特殊処理
        if (1986 <= year && year <= 2006 && month == 5 && day == 4) {
            if (weekday == 1) {
                holidayName = "振替休日";
            }
        }
        if (holidayName !== "") {
            return holidayName;
        }

        if (new Date("1973/04/12") <= date && date <= new Date("2006/12/31")) {
            // 前日が日曜かつ国民の祝日のとき本日は振替休日
            if (previousDate.getDay() === 0 && _getPublicHolidayName(previousDate) !== "") {
                holidayName = "振替休日";
            }
        }
        if (holidayName !== "") {
            return holidayName;
        }

        if (new Date("2007/01/01") <= date) {
            // 前日に国民の祝日が連続しており、その中に日曜が含まれるとき本日は振替休日
            while (true) {
                if (_getPublicHolidayName(previousDateForLoop) === "") {
                    // 前日が国民の祝日では無い場合は終了
                    break;
                }
                if (previousDateForLoop.getDay() === 0) {
                    // 前日が日曜の場合は終了
                    holidayName = "振替休日";
                    break;
                }
                previousDateForLoop.setDate(previousDateForLoop.getDate() - 1);
            }
        }
        if (holidayName !== "") {
            return holidayName;
        }

        // 国民の休日

        // 1986-2006の5月4日の特殊処理
        if (1986 <= year && year <= 2006 && month === 5 && day === 4) {
            if (weekday >= 2 && weekday <= 6) {
                holidayName = "国民の休日";
            }
        }
        if (holidayName !== "") {
            return holidayName;
        }

        // 前後が国民の祝日である平日は国民の休日となる
        // 次の日(dateをベースに別インスタンスで作成)
        var nextDate = new Date(timeStamp);
        nextDate.setDate(date.getDate() + 1);

        if (new Date("1985/12/27") <= date) {
            if (_getPublicHolidayName(previousDate) !== "" && _getPublicHolidayName(nextDate) !== "" && weekday >= 2 && weekday <= 6) {
                holidayName = "国民の休日";
            }
        }

        return holidayName;
    }

    var _fixHolidayObjectList = [
        {"name": "元日", "month": 1, "day": 1, "start": 1949, "end": null},
        {"name": "成人の日", "month": 1, "day": 15, "start": 1949, "end": 1999},
        {"name": "建国記念の日", "month": 2, "day": 11, "start": 1967, "end": null},
        {"name": "昭和天皇の大喪の礼", "month": 2, "day": 24, "start": 1989, "end": 1989},
        {"name": "皇太子・明仁親王の結婚の儀", "month": 4, "day": 10, "start": 1959, "end": 1959},
        {"name": "天皇誕生日", "month": 4, "day": 29, "start": 1949, "end": 1988},
        {"name": "みどりの日", "month": 4, "day": 29, "start": 1989, "end": 2006},
        {"name": "昭和の日", "month": 4, "day": 29, "start": 2007, "end": null},
        {"name": "憲法記念日", "month": 5, "day": 3, "start": 1949, "end": null},
        {"name": "みどりの日", "month": 5, "day": 4, "start": 2007, "end": null},
        {"name": "こどもの日", "month": 5, "day": 5, "start": 1949, "end": null},
        {"name": "皇太子・皇太子徳仁親王の結婚の儀", "month": 6, "day": 9, "start": 1993, "end": 1993},
        {"name": "海の日", "month": 7, "day": 20, "start": 1996, "end": 2002},
        {"name": "海の日", "month": 7, "day": 23, "start": 2020, "end": 2020},
        {"name": "山の日", "month": 8, "day": 11, "start": 2016, "end": 2019},
        {"name": "山の日", "month": 8, "day": 10, "start": 2020, "end": 2020},
        {"name": "山の日", "month": 8, "day": 11, "start": 2021, "end": null},
        {"name": "敬老の日", "month": 9, "day": 15, "start": 1966, "end": 2002},
        {"name": "体育の日", "month": 10, "day": 10, "start": 1966, "end": 1999},
        {"name": "文化の日", "month": 11, "day": 3, "start": 1948, "end": null},
        {"name": "即位礼正殿の儀", "month": 11, "day": 12, "start": 1990, "end": 1990},
        {"name": "勤労感謝の日", "month": 11, "day": 23, "start": 1948, "end": null},
        {"name": "天皇誕生日", "month": 12, "day": 23, "start": 1989, "end": 2018},
        {"name": "天皇の即位の日", "month": 5, "day": 1, "start": 2019, "end": 2019},
        {"name": "即位礼正殿の儀", "month": 10, "day": 22, "start": 2019, "end": 2019},
        {"name": "天皇誕生日", "month": 2, "day": 23, "start": 2020, "end": null},
        {"name": "スポーツの日", "month": 7, "day": 24, "start": 2020, "end": 2020},
    ];

    var _happyMondayObjectList = [
        {"name": "成人の日", "month": 1, "weekNumber": 2, "start": 2000, "end": null},
        {"name": "海の日", "month": 7, "weekNumber": 3, "start": 2003, "end": 2019},
        {"name": "海の日", "month": 7, "weekNumber": 3, "start": 2021, "end": null},
        {"name": "敬老の日", "month": 9, "weekNumber": 3, "start": 2003, "end": null},
        {"name": "体育の日", "month": 10, "weekNumber": 2, "start": 2000, "end": 2019},
        {"name": "スポーツの日", "month": 10, "weekNumber": 2, "start": 2021, "end": null},
    ];

    /**
     * 国民の祝日取得
     * @param Date date
     * @return string
     */
    var _getPublicHolidayName = function (date) {

        var publicHolidayName = "";

        var year = date.getFullYear();
        var month = date.getMonth() + 1;
        var day = date.getDate();
        var weekday = date.getDay();

        // 第何週
        var weekNumber = null;

        if (day % 7 == 0) {
            weekNumber = day / 7;
        } else {
            weekNumber = parseInt(day / 7, 10) + 1;
        }

        // 固定祝日
        _fixHolidayObjectList.forEach(function (object) {

            if (object.end !== null) {
                if (object.start <= year && year <= object.end && month === object.month && day === object.day) {
                    publicHolidayName = object.name;
                }
            } else {
                if (object.start <= year && month === object.month && day === object.day) {
                    publicHolidayName = object.name;
                }
            }

        });

        if (publicHolidayName !== "") {
            return publicHolidayName;
        }

        // ハッピーマンデー制度
        _happyMondayObjectList.forEach(function (object) {

            if (weekday !== 1) {
                // 月曜ではない場合
                return;
            }

            if (object.end !== null) {
                if (object.start <= year && year <= object.end && month === object.month && weekNumber === object.weekNumber) {
                    publicHolidayName = object.name;
                }
            } else {
                if (object.start <= year && month === object.month && weekNumber === object.weekNumber) {
                    publicHolidayName = object.name;
                }
            }

        });

        if (publicHolidayName !== "") {
            return publicHolidayName;
        }

        // 春分の日・秋分の日
        if (1900 <= year && year < 1980) {
            if (month === 3) {
                if (day === parseInt(20.8357 + 0.242194 * ( year - 1980), 10) - parseInt((year - 1983) / 4, 10)) {
                    publicHolidayName = "春分の日";
                }
            }
            if (month === 9) {
                if (day === parseInt(23.2588 + 0.242194 * ( year - 1980), 10) - parseInt((year - 1983) / 4, 10)) {
                    publicHolidayName = "秋分の日";
                }
            }
        }
        if (1980 <= year && year < 2100) {
            if (month === 3) {
                if (day === parseInt(20.8431 + 0.242194 * ( year - 1980), 10) - parseInt((year - 1980) / 4, 10)) {
                    publicHolidayName = "春分の日";
                }
            }
            if (month === 9) {
                if (day === parseInt(23.2488 + 0.242194 * ( year - 1980), 10) - parseInt((year - 1980) / 4, 10)) {
                    publicHolidayName = "秋分の日";
                }
            }
        }
        if (2100 <= year && year < 2150) {
            if (month === 3) {
                if (day === parseInt(20.8510 + 0.242194 * ( year - 1980), 10) - parseInt((year - 1980) / 4, 10)) {
                    publicHolidayName = "春分の日";
                }
            }
            if (month === 9) {
                if (day === parseInt(24.2488 + 0.242194 * ( year - 1980), 10) - parseInt((year - 1980) / 4, 10)) {
                    publicHolidayName = "秋分の日";
                }
            }
        }

        return publicHolidayName;
    }

    return {
        getHolidayName: getHolidayName
    }
})();
20
16
3

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
20
16