PHPで3ヶ月カレンダーを作る

(1/1)
PHP を使って、今月から再来月までの 3 ヶ月カレンダーを作る。
PHP で祝日を求める」「PHP で二十四節気一覧を作成」に、七十二候や旧暦、六曜、干支の計算を加えた。

(2019 年 6 月 30 日)makeCalendar の不具合を修正、リファラチェックを追加した。
(2019 年 2 月 9 日)pahooCalendar::makeLunarCalendar の不具合を修正、getSolarTerm72 の漢字表記を変更した。

サンプル・プログラムの実行例

PHPで3ヶ月カレンダーを作る

サンプル・プログラム

各種暦計算、太陽や月の位置計算は、クラス pahooCalendar として分離している。
コマンドラインから、year(西暦年),month(月)を直接指定してやると、カレンダーだけを表示する。

サンプル・プログラムの解説:旧暦の計算

1400: /**
1401:  * グレゴオリオ暦=旧暦テーブル 作成
1402:  * @param int $year 西暦年
1403:  * @return double 太陽の黄経(視黄経)
1404: */
1405: function makeLunarCalendar($year) {
1406:     unset($this->tblmoon);
1407:     $this->tblmoon = array();
1408: 
1409:     //前年の冬至を求める
1410:     for ($day = 1; $day <= 31; $day++) {
1411:         $lsun = $this->longitude_sun($year - 1, 12, $day, 0, 0, 0);
1412:         if (floor($lsun / 15.0) > 17)    break;
1413:     }
1414:     $d1 = $day - 1;   //冬至
1415: 
1416:     //翌年の雨水を求める
1417:     for ($day = 1; $day <= 31; $day++) {
1418:         $lsun = $this->longitude_sun($year + 1, 2, $day, 0, 0, 0);
1419:         if (floor($lsun / 15.0) > 22)    break;
1420:     }
1421:     $d2 = $day - 1;   //雨水
1422: 
1423:     //朔の日を求める
1424:     $cnt = 0;
1425:     $dd = $d1;
1426:     $mm = 12;
1427:     $yy = $year - 1;
1428:     while ($yy <= $year + 1) {
1429:         $dm = $this->getDaysInMonth($yy$mm);
1430:         while ($dd <= $dm) {
1431:             $age1 = $this->moon_age($yy$mm$dd,  0 - $this->TDIFF, 0, 0);    //Ver.3.11 bug-fix
1432:             $age2 = $this->moon_age($yy$mm$dd, 23 - $this->TDIFF, 59, 59);  //Ver.3.11 bug-fix
1433:             if ($age2 <= $age1) {
1434:                 $this->tblmoon[$cnt]['year']  = $yy;
1435:                 $this->tblmoon[$cnt]['month'] = $mm;
1436:                 $this->tblmoon[$cnt]['day']   = $dd;
1437:                 $this->tblmoon[$cnt]['age']   = $age1;
1438:                 $this->tblmoon[$cnt]['jd']    = $this->Gregorian2JD($yy$mm$dd, 0, 0, 0);
1439:                 $cnt++;
1440:             }
1441:             $dd++;
1442:         }
1443:         $mm++;
1444:         $dd = 1;
1445:         if ($mm > 12) {
1446:             $yy++;
1447:             $mm = 1;
1448:         }
1449:     }
1450: 
1451:     //二十四節気(中)を求める
1452:     $tblsun = array();
1453:     $cnt = 0;
1454:     $dd = $d1;
1455:     $mm = 12;
1456:     $yy = $year - 1;
1457:     while ($yy <= $year + 1) {
1458:         $dm = $this->getDaysInMonth($yy$mm);
1459:         while ($dd <= $dm) {
1460:             $l1 = $this->longitude_sun($yy$mm$dd,  0, 0, 0);
1461:             $l2 = $this->longitude_sun($yy$mm$dd, 24, 0, 0);
1462:             $n1 = floor($l1 / 15.0);
1463:             $n2 = floor($l2 / 15.0);
1464:             if (($n2 != $n1) && ($n2 % 2 == 0)) {
1465:                 $tblsun[$cnt]['jd'] = $this->Gregorian2JD($yy$mm$dd, 0, 0, 0);
1466:                 $oldmonth = floor($n2 / 2) + 2;
1467:                 if ($oldmonth > 12) $oldmonth -= 12;
1468:                 $tblsun[$cnt]['oldmonth']  = $oldmonth;
1469:                 $cnt++;
1470:             }
1471:             $dd++;
1472:         }
1473:         $mm++;
1474:         $dd = 1;
1475:         if ($mm > 12) {
1476:             $yy++;
1477:             $mm = 1;
1478:         }
1479:     }
1480: 
1481:     //月の名前を決める
1482:     $n1 = count($this->tblmoon);
1483:     $n2 = count($tblsun);
1484:     for ($i = 0; $i < $n1 - 1; $i++) {
1485:         for ($j = 0; $j < $n2$j++) {
1486:             if (($this->tblmoon[$i]['jd'] <= $tblsun[$j]['jd'])
1487:                 && ($this->tblmoon[$i + 1]['jd'] > $tblsun[$j]['jd'])) {
1488:                 $this->tblmoon[$i]['oldmonth'] = $tblsun[$j]['oldmonth'];
1489:                 $this->tblmoon[$i]['oldleap']  = FALSE;
1490:                 $this->tblmoon[$i + 1]['oldmonth'] = $tblsun[$j]['oldmonth'];
1491:                 $this->tblmoon[$i + 1]['oldleap']  = TRUE;
1492:                 break;
1493:             }
1494:         }
1495:     }
1496: }

グレゴリオ暦から旧暦を求める手順は、「旧暦と六曜を作りましょう」を参考にした。

変換のための方程式を導出できないため、あらかじめ計算したい西暦年から変換テーブルを作る必要がある。これを行うのがユーザー関数 makeLunarCalendar である。
  1. 前年の冬至を求める。
  2. 翌年の雨水を求める。
  3. 1~2 の期間中の二十四節気(中)を配列 $tblsun に格納する。
  4. 朔日と中を比較して、月の名前と閏月を決めてゆく。

1498: /**
1499:  * 旧暦を求める
1500:  * @param int $year  西暦年
1501:  * @param int $month 月
1502:  * @param int $day   日
1503:  * @return array(旧暦月,日,閏月フラグ)
1504: */
1505: function Gregorian2Lunar($year$month$day) {
1506:     $jd = $this->Gregorian2JD($year$month$day, 0, 0, 0);
1507:     $str = '';
1508:     $n1 = count($this->tblmoon);
1509:     for ($i = 0; $i < $n1 - 1; $i++) {
1510:         if ($jd < $this->tblmoon[$i + 1]['jd']) {
1511:             $day = floor($jd - $this->tblmoon[$i]['jd']) + 1;
1512:             $items = array($this->tblmoon[$i]['oldmonth'], $day$this->tblmoon[$i]['oldleap']);
1513:             break;
1514:         }
1515:     }
1516:     return $items;
1517: }

変換テーブルが用意できたら、ユーザー関数 Gregorian2Lunar を使って旧暦を求める。

この旧暦計算法は天保暦に基づいているが、天保暦には 2033 年問題がある。西暦 2033 年(令和 15 年)10 月は旧暦 9 月だが、11 月に入ると旧暦 11 月と、10 月が無くなってしまう問題である。
これは、1844 年(天保 14 年)に天保暦が導入されてから初めて起きる事態だが、天保暦は 1872 年(明治 5 年)に廃止されているために正式な解決法は用意されていない。
いくつかの解決案が提示されているが、本プログラムには組み込んでいない。

サンプル・プログラムの解説:干支の計算

1538: /**
1539:  * 干支を求める(下請け関数)
1540:  * @param int $a1 十干の基準値
1541:  * @param int $a2 十二支の基準値
1542:  * @param int $n  計算したい値
1543:  * @return string 干支
1544: */
1545: function __eto($a1$a2$n) {
1546: //十干
1547: static $table1 = array(
1548:  0 =>'',
1549:  1 =>'',
1550:  2 =>'',
1551:  3 =>'',
1552:  4 =>'',
1553:  5 =>'',
1554:  6 =>'',
1555:  7 =>'',
1556:  8 =>'',
1557:  9 =>''
1558: );
1559: 
1560: //十二支
1561: static $table2 = array(
1562:  0 =>'',
1563:  1 =>'',
1564:  2 =>'',
1565:  3 =>'',
1566:  4 =>'',
1567:  5 =>'',
1568:  6 =>'',
1569:  7 =>'',
1570:  8 =>'',
1571:  9 =>'',
1572: 10 =>'',
1573: 11 =>''
1574: );
1575: 
1576:     return $table1[abs($n - $a1) % 10] . $table2[abs($n - $a2) % 12];
1577: }

干支には、年の干支月の干支日の干支 の 3種類がある。
干支は、甲・乙・丙・丁・戊・己・庚・辛・壬・癸の 10要素からなる十干 (じっかん) と、子・丑・寅・卯・辰・巳・午・未・申・酉・戌・亥の 12要素からなる十二支 (じゅうにし) の組み合わせからなる。
つまり、十干は年/月/日が 10 回ごとに繰り返され、十二支は年/月/日が 12 回ごとに繰り返されることになる。これを計算するのが __eto である。

次に、基準となる年月日を調べておく。
甲の年は西暦 1904 年(明治 37 年)、子の年は西暦 1900 年(明治 33 年)である。年の干支は、この 2 つを基準に計算する。
甲子の月は 1903 年(明治 36 年)11 月である。月の干支は、ここを基準に計算する。
甲子の日は 1902 年(明治 35 年)4 月 11 日である。日の干支は、ここを基準に計算する。

サンプル・プログラムの解説:カレンダー作成

0174: /**
0175:  * 1ヶ月分のカレンダーを作成
0176:  * @param object $pcl  pahooCalendarオブジェクト
0177:  * @param int $start 週の開始曜日(0:日曜日, 1:月曜日...6:土曜日)
0178:  * @param int $year  西暦年
0179:  * @param int $month 月
0180:  * @return string HTMLコンテンツ/FALSE:エラー
0181: */
0182: function makeCalendar($pcl$start$year$month) {
0183:     if ($month < 1 && $month > 12)      return FALSE;
0184: 
0185:     $eto_year  = $pcl->eto_year($year);
0186:     $eto_month = $pcl->eto_month($year$month);
0187: 
0188: $html =<<< EOT
0189: <table class="calendar">
0190: <tr>
0191: <th colspan="7"><span class="large">{$year}年({$eto_year}) {$month}月({$eto_month})</span></th>
0192: </tr>
0193: <tr>
0194: 
0195: EOT;
0196: 
0197:     //曜日の行
0198:     for ($i = 0; $i < 7; $i++) {
0199:         $n  = ($start + $i) % 7;
0200:         if ($n == 6)        $class = 'blue';
0201:         else if ($n == 0)   $class = 'red';
0202:         else                $class = 'black';
0203:         $str = $pcl->__getWeekString($n);
0204:         $html .= "<th><span class=\"{$class}\">{$str}</span></th>";
0205:     }
0206:     $html .= "</tr>\n";
0207: 
0208:     //カレンダー本体
0209:     $wn1 = $pcl->getWeekNumber($year$month, 1); //月の最初の曜日
0210:     $dim = $pcl->getDaysInMonth($year$month);   //月の最後の日
0211:     $cnt = 0;
0212:     $flag = FALSE;
0213:     $n = $start;
0214:     $day = 1;
0215:     while (1) {
0216:         if ($cnt % 7 == 0)      $html .= "<tr>\n";
0217:         //曜日の色
0218:         if ($n % 7 == 6)        $class = 'blue';
0219:         else if ($n % 7 == 0)   $class = 'red';
0220:         else                    $class = 'black';
0221:         if ($n % 7 == $wn1$flag = TRUE;
0222:         //表示開始
0223:         if ($flag) {
0224:             //祝日
0225:             $holiday = $pcl->getHoliday($year$month$day);
0226:             if ($holiday != '') $class = 'red';
0227:             $holiday = ($holiday == FALSE) ? '<br />' :
0228:                 "<span class=\"small red\">{$holiday}</span><br />";
0229:             //二十四節気
0230:             $solarterm = $pcl->getSolarTerm($year$month$day);
0231:             $solarterm = ($solarterm == '') ? '<br />' :
0232:                 "<span class=\"small\">{$solarterm}</span><br />";
0233:             //七十二候
0234:             $solarterm72 = $pcl->getSolarTerm72($year$month$day);
0235:             $solarterm72 = ($solarterm72 == '') ? '<br />' :
0236:                 "<span class=\"small\">{$solarterm72}</span><br />";
0237:             //旧暦
0238:             list($oldmonth$oldday$oldleap) = $pcl->Gregorian2Lunar($year$month$day);
0239:             $oldleap = $oldleap ? '' : '';
0240:             $rokuyou = $pcl->rokuyou($oldmonth$oldday);
0241:             $oldcal = sprintf('%s%d月%d日<br />(%s)', $oldleap$oldmonth$oldday$rokuyou);
0242:             //日の干支
0243:             $eto_day = $pcl->eto_day($year$month$day);
0244:             //表示
0245: $html .=<<< EOT
0246: <td>
0247: <span class="large {$class}">{$day}</span><br />
0248: {$holiday}
0249: {$solarterm}
0250: {$solarterm72}
0251: <span class="small">{$eto_day}<br /></span>
0252: <span class="small">{$oldcal}</span>
0253: </td>
0254: 
0255: EOT;
0256:             $day++;
0257:             if ($day > $dim)    break;
0258:         } else {
0259:             $html .= "<td>&nbsp;</td>";
0260:         }
0261:         $cnt++;
0262:         $n++;
0263:         if ($cnt % 7 == 0)  $html .= "</tr>\n";
0264:     }
0265:     //最後の日以降の処理
0266:     $cnt++;
0267:     while ($cnt % 7 != 0) {
0268:         $html .= "<td>&nbsp;</td>";
0269:         $cnt++;
0270:         if ($cnt % 7 == 0)  $html .= "</tr>\n";
0271:     }
0272: 
0273:     $html .= "</table>\n";
0274: 
0275:     return $html;
0276: }

ユーザー関数 makeCalendar は 1 ヶ月分のカレンダーを作成する。
週の開始曜日を自由に設定できるようにするための工夫をしている。

カレンダー本体を作成する処理では、まず、月の最初の曜日 $wn1 と、月の最後の日 $dim を計算しておく。
$wn1 と週の開始曜日 $start が一致するまでは変数 $flag が FALSE のままで、この時は日付を入れずに空のままにしておく。
$flag が TRUE になったら日付を入れてゆく。あわせて、二十四節気、旧暦の計算を行う。

$dim に達したらループを抜け出し、行末まで空白のセルを追加してゆく。

質疑応答

【質問】

「PHP で 3 ヶ月カレンダーを作る」のページのソースなのですが「サンプル・プログラムの解説:カレンダー作成」のソースの 0171行目に「<tr>」の開始タグを入れたほうがいいと思います。曜日部分の行部分 TR の開始タグが無いようです。
間違えであればすいません。


【回答】

ご指摘ありがとうございます。<tr> タグが抜けていました。追加しました。

活用例

二十四節気・七十二候・干支・旧暦・六曜カレンダー:みんなの知識 ちょっと便利帳」では、このサンプル・プログラムを活用している。ありがとうございます。

参考サイト

(この項おわり)
header