256 lines
10 KiB
PHP
256 lines
10 KiB
PHP
<?php
|
|
|
|
abstract class Bundesland {
|
|
private static $Namen = array('Baden-Württemberg', 'Bayern', 'Berlin', 'Brandenburg',
|
|
'Bremen', 'Hamburg', 'Hessen', 'Mecklenburg-Vorpommern',
|
|
'Niedersachsen', 'Nordrhein-Westfalen', 'Rheinland-Pfalz',
|
|
'Saarland', 'Sachsen', 'Sachsen-Anhalt',
|
|
'Schleswig-Holstein', 'Thüringen');
|
|
|
|
const Baden_Wuerttemberg = 0;
|
|
const Bayern = 1;
|
|
const Berlin = 2;
|
|
const Brandenburg = 3;
|
|
const Bremen = 4;
|
|
const Hamburg = 5;
|
|
const Hessen = 6;
|
|
const Mecklenburg_Vorpommern = 7;
|
|
const Niedersachsen = 8;
|
|
const Nordrhein_Westfalen = 9;
|
|
const Rheinland_Pfalz = 10;
|
|
const Saarland = 11;
|
|
const Sachsen = 12;
|
|
const Sachsen_Anhalt = 13;
|
|
const Schleswig_Holstein = 14;
|
|
const Thueringen = 15;
|
|
|
|
public function Count() {
|
|
return count(self::$Namen);
|
|
}
|
|
|
|
public function GetName(int $land) {
|
|
if ($land >= 0 && $land < self::Count())
|
|
return self::$Namen[$land];
|
|
else
|
|
return '';
|
|
}
|
|
}
|
|
|
|
class Feiertag {
|
|
private $datum;
|
|
private $name;
|
|
private $laender;
|
|
|
|
public function __construct(int $tag, string $name, array $laender = array()) {
|
|
$this->datum = mktime(0, 0, 0, date('m', $tag), date('d', $tag), date('Y', $tag));
|
|
$this->name = $name;
|
|
$this->laender = array_unique($laender);
|
|
sort($this->laender);
|
|
}
|
|
|
|
public function GetDatum(bool $lang = false) {
|
|
if ($lang)
|
|
return date('Ymd\THis\Z', $this->datum);
|
|
else
|
|
return date('Ymd', $this->datum);
|
|
}
|
|
|
|
public function IsGesetzlich() {
|
|
if (count($this->laender) > 0)
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
|
|
public function IsBundesweit() {
|
|
if (count($this->laender) == Bundesland::Count())
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
|
|
public function IsInBundesland(int $land) {
|
|
return in_array($land, $this->laender);
|
|
}
|
|
|
|
public function GetBundeslaender() {
|
|
$s = '';
|
|
for ($i = 0; $i < count($this->laender); $i++) {
|
|
if (strlen($s) > 0)
|
|
$s .= ', ';
|
|
$s .= Bundesland::GetName($this->laender[$i]);
|
|
}
|
|
return $s;
|
|
}
|
|
|
|
public function GetVEvent() {
|
|
$s = "BEGIN:VEVENT\r\n"
|
|
. 'UID:' . uniqid(get_class()) . "\r\n"
|
|
. "DTSTAMP:{$this->GetDatum(true)}\r\n"
|
|
. "DTSTART;VALUE=DATE:{$this->GetDatum()}\r\n"
|
|
. "DTEND;VALUE=DATE:{$this->GetDatum()}\r\n"
|
|
. 'SUMMARY:' . addcslashes($this->name, ',\\;') . "\r\n";
|
|
if ($this->IsGesetzlich()) {
|
|
$s .= 'DESCRIPTION:Gesetzlicher Feiertag';
|
|
if (!$this->IsBundesweit())
|
|
$s .= ' in ' . addcslashes($this->GetBundeslaender(), ',\\;');
|
|
$s .= "\r\n";
|
|
}
|
|
$s .= "END:VEVENT\r\n";
|
|
return $s;
|
|
}
|
|
|
|
public function __toString() {
|
|
return date('Y-m-d', $this->datum) . ' ' . $this->name;
|
|
}
|
|
}
|
|
|
|
class FeiertagKalender {
|
|
private $jahr;
|
|
private $feiertage = array();
|
|
|
|
private function calcOstersonntag(int $jahr) {
|
|
// Osterformel nach Butcher
|
|
$a = $jahr % 19;
|
|
$b = (int) ($jahr / 100);
|
|
$c = $jahr % 100;
|
|
$d = (int) ($b / 4);
|
|
$e = $b % 4;
|
|
$f = (int) (($b + 8) / 25);
|
|
$g = (int) (($b - $f + 1) / 3);
|
|
$h = (19 * $a + $b - $d - $g + 15) % 30;
|
|
$i = (int) ($c / 4);
|
|
$j = $c % 4;
|
|
$k = (32 + 2 * $e + 2 * $i - $h - $j) % 7;
|
|
$l = (int) (($a + 11 * $h + 22 * $k) / 451);
|
|
$m = $h + $k - 7 * $l + 114;
|
|
$monat = (int) ($m / 31);
|
|
$tag = ($m % 31) + 1;
|
|
return mktime(0, 0, 0, $monat, $tag, $jahr);
|
|
}
|
|
|
|
public function __construct(int $jahr) {
|
|
$this->jahr = $jahr;
|
|
$alle = array(Bundesland::Baden_Wuerttemberg, Bundesland::Bayern, Bundesland::Berlin, Bundesland::Brandenburg,
|
|
Bundesland::Bremen, Bundesland::Hamburg, Bundesland::Hessen, Bundesland::Mecklenburg_Vorpommern,
|
|
Bundesland::Niedersachsen, Bundesland::Nordrhein_Westfalen, Bundesland::Rheinland_Pfalz,
|
|
Bundesland::Saarland, Bundesland::Sachsen, Bundesland::Sachsen_Anhalt,
|
|
Bundesland::Schleswig_Holstein, Bundesland::Thueringen);
|
|
$ostersonntag = $this->calcOstersonntag($jahr);
|
|
$busstag = mktime(0, 0, 0, 11, 22 - ($jahr - 1 + $jahr / 4) % 7, $jahr);
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 1, 1, $jahr), 'Neujahr', $alle));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 1, 6, $jahr), 'Heilige Drei Könige',
|
|
array(Bundesland::Baden_Wuerttemberg, Bundesland::Bayern, Bundesland::Sachsen_Anhalt)));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 2, 14, $jahr), 'Valentinstag'));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 3, 8, $jahr), 'Internationaler Frauentag'));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 4, 30, $jahr), 'Walpurgisnacht'));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 5, 1, $jahr), 'Tag der Arbeit', $alle));
|
|
array_push($this->feiertage, new Feiertag(strtotime('-52 days', $ostersonntag), 'Weiberfastnacht'));
|
|
array_push($this->feiertage, new Feiertag(strtotime('-48 days', $ostersonntag), 'Rosenmontag'));
|
|
array_push($this->feiertage, new Feiertag(strtotime('-47 days', $ostersonntag), 'Fastnacht'));
|
|
array_push($this->feiertage, new Feiertag(strtotime('-46 days', $ostersonntag), 'Aschermittwoch'));
|
|
array_push($this->feiertage, new Feiertag(strtotime('-7 days', $ostersonntag), 'Palmsonntag'));
|
|
array_push($this->feiertage, new Feiertag(strtotime('-3 days', $ostersonntag), 'Gründonnerstag'));
|
|
array_push($this->feiertage, new Feiertag(strtotime('-2 days', $ostersonntag), 'Karfreitag', $alle));
|
|
array_push($this->feiertage, new Feiertag(strtotime('-1 day', $ostersonntag), 'Karsamstag'));
|
|
array_push($this->feiertage, new Feiertag($ostersonntag, 'Ostersonntag', array(Bundesland::Brandenburg)));
|
|
array_push($this->feiertage, new Feiertag(strtotime('+1 day', $ostersonntag), 'Ostermontag', $alle));
|
|
array_push($this->feiertage, new Feiertag(strtotime('+7 days', $ostersonntag), 'Weißer Sonntag'));
|
|
array_push($this->feiertage, new Feiertag(strtotime('+39 days', $ostersonntag), 'Christi Himmelfahrt', $alle));
|
|
array_push($this->feiertage, new Feiertag(strtotime('+49 days', $ostersonntag), 'Pfingstsonntag', array(Bundesland::Brandenburg)));
|
|
array_push($this->feiertage, new Feiertag(strtotime('+50 days', $ostersonntag), 'Pfingstmontag', $alle));
|
|
array_push($this->feiertage, new Feiertag(strtotime('+60 days', $ostersonntag), 'Fronleichnam',
|
|
array(Bundesland::Baden_Wuerttemberg, Bundesland::Bayern, Bundesland::Hessen,
|
|
Bundesland::Nordrhein_Westfalen, Bundesland::Rheinland_Pfalz, Bundesland::Saarland)));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 5, 14 - ($jahr - 1 + $jahr / 4) % 7, $jahr), 'Muttertag'));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 9, 20, $jahr), 'Weltkindertag'));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 8, 15, $jahr), 'Mariä Himmelfahrt',
|
|
array(Bundesland::Bayern, Bundesland::Saarland)));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 10, 3, $jahr), 'Tag der Deutschen Einheit', $alle));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 10, 31, $jahr), 'Halloween'));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 10, 31, $jahr), 'Reformationstag',
|
|
array(Bundesland::Brandenburg, Bundesland::Mecklenburg_Vorpommern, Bundesland::Sachsen,
|
|
Bundesland::Sachsen_Anhalt, Bundesland::Thueringen)));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 11, 1, $jahr), 'Allerheiligen',
|
|
array(Bundesland::Baden_Wuerttemberg, Bundesland::Bayern, Bundesland::Nordrhein_Westfalen,
|
|
Bundesland::Rheinland_Pfalz, Bundesland::Saarland)));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 11, 2, $jahr), 'Allerseelen'));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 10, 7 - ($jahr + 5 + $jahr / 4) % 7, $jahr), 'Erntedankfest'));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 11, 11, $jahr), 'Martinstag'));
|
|
array_push($this->feiertage, new Feiertag(strtotime('-3 days', $busstag), 'Volkstrauertag'));
|
|
array_push($this->feiertage, new Feiertag(strtotime('+4 days', $busstag), 'Totensonntag'));
|
|
array_push($this->feiertage, new Feiertag($busstag, 'Buß- und Bettag', array(Bundesland::Sachsen)));
|
|
array_push($this->feiertage, new Feiertag(strtotime('+11 days', $busstag), '1. Advent'));
|
|
array_push($this->feiertage, new Feiertag(strtotime('+18 days', $busstag), '2. Advent'));
|
|
array_push($this->feiertage, new Feiertag(strtotime('+25 days', $busstag), '3. Advent'));
|
|
array_push($this->feiertage, new Feiertag(strtotime('+32 days', $busstag), '4. Advent'));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 12, 25, $jahr), '1. Weihnachtstag', $alle));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 12, 26, $jahr), '2. Weihnachtstag', $alle));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 12, 6, $jahr), 'Nikolaus'));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 12, 24, $jahr), 'Heiliger Abend'));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 12, 31, $jahr), 'Silvester'));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 3, 31 - ($jahr + 4 + $jahr / 4) % 7, $jahr), 'Sommerzeit (+1h)'));
|
|
array_push($this->feiertage, new Feiertag(mktime(0, 0, 0, 10, 31 - ($jahr + 1 + $jahr / 4) % 7, $jahr), 'Winterzeit (-1h)'));
|
|
sort($this->feiertage);
|
|
}
|
|
|
|
private function GetHeader() {
|
|
return "BEGIN:VCALENDAR\r\n"
|
|
. "VERSION:2.0\r\n"
|
|
. "PRODID:-//{$_SERVER['SERVER_NAME']}//" . get_class() . "//DE\r\n";
|
|
}
|
|
|
|
private function GetFooter() {
|
|
return "END:VCALENDAR\r\n";
|
|
}
|
|
|
|
public function GetVCalendar() {
|
|
$s = $this->GetHeader();
|
|
for ($i = 0; $i < count($this->feiertage); $i++)
|
|
$s .= $this->feiertage[$i]->GetVEvent();
|
|
$s .= $this->GetFooter();
|
|
return $s;
|
|
}
|
|
|
|
public function __toString() {
|
|
$s = '';
|
|
for ($i = 0; $i < count($this->feiertage); $i++)
|
|
$s .= $this->feiertage[$i] . "\r\n";
|
|
return $s;
|
|
}
|
|
}
|
|
|
|
if (isset($_GET['jahr']) && is_numeric($_GET['jahr'])) {
|
|
$jahr = max(2000, min(3000, intval($_GET['jahr'])));
|
|
$feiertage = new FeiertagKalender($jahr);
|
|
if (!isset($_GET['raw'])) {
|
|
header('Content-Type: text/calendar; charset=utf-8');
|
|
header("Content-Disposition: inline; filename=\"{$jahr}.ics\"");
|
|
echo $feiertage->GetVCalendar();
|
|
}
|
|
else {
|
|
header('Content-Type: text/plain; charset=utf-8');
|
|
echo $feiertage;
|
|
}
|
|
}
|
|
else {
|
|
echo "<!DOCTYPE html>\r\n"
|
|
. "<html lang=\"de\">\r\n"
|
|
. "<head><meta charset=\"utf-8\"><title>Feiertage in Deutschland</title></head>\r\n"
|
|
. "<body>\r\n"
|
|
. "<b>Feiertage als iCal-Datei herunterladen</b>\r\n"
|
|
. "<form method=\"get\">\r\n"
|
|
. "<label>Jahr:\r\n"
|
|
. "<select name=\"jahr\">\r\n";
|
|
$year = date('Y');
|
|
for ($i = 0; $i < 5; $i++)
|
|
echo "<option>" . strval($year + $i) . "</option>\r\n";
|
|
echo "</select>\r\n"
|
|
. "</label><br>\r\n"
|
|
. "<button type=\"submit\">Download</button>\r\n"
|
|
. "</form>\r\n"
|
|
. "</body>\r\n"
|
|
. "</html>";
|
|
}
|
|
|
|
?>
|