A modern, object-oriented calendar management library for PHP 8.2+.
CalendR provides a clean, immutable, and iterable API to manipulate time periods (Years, Months, Weeks, Days...) and manage associated events.
- Object-Oriented Periods: Manipulate
Year,Month,Week,Dayas objects, not strings or timestamps. - Fully Iterable: Iterate over a Year to get Months, or a Month to get Days, using native
foreachloops. - Immutable by Design: Based on
DateTimeImmutable, ensuring safe date manipulations. - Event Management: Fetch and aggregate events from multiple sources (Doctrine, API, etc.) for any period.
- Zero Dependencies: The core library has no external dependencies.
- Framework Integrations: Includes a Symfony Bundle and Twig extensions.
composer require yohang/calendrThe Calendar class is your main entry point. It acts as a factory to create periods configured with your preferences (e.g., first day of the week).
<?php
use CalendR\Calendar;
$calendar = new Calendar();
// Get a specific year
$year = $calendar->getYear(2025);
foreach ($year as $month) {
echo $month->format('F Y') . "\n";
// Iterate over days in that month
foreach ($month as $day) {
// ...
}
}Every period object implements PeriodInterface, providing powerful methods:
<?php
$month = $calendar->getMonth(2025, 1); // January 2025
// Check containment
if ($month->contains(new \DateTimeImmutable('2025-01-15'))) {
echo "We are in the middle of the month!";
}
// Navigation
$nextMonth = $month->getNext(); // February 2025
$prevMonth = $month->getPrevious(); // December 2024
// DatePeriod compatibility
foreach ($month->getDatePeriod() as $date) {
// $date is a DateTimeImmutable
}CalendR can attach events to any period. You need to configure an EventManager with one or more ProviderInterface.
<?php
use CalendR\Event\AbstractEvent;
class MyEvent extends AbstractEvent
{
public function __construct(
private \DateTimeImmutable $begin,
private \DateTimeImmutable $end
) {}
public function getBegin(): \DateTimeInterface { return $this->begin; }
public function getEnd(): \DateTimeInterface { return $this->end; }
public function getUid(): string { return uniqid(); }
}<?php
use CalendR\Event\Provider\ProviderInterface;
class MyProvider implements ProviderInterface
{
public function getEvents(\DateTimeInterface $begin, \DateTimeInterface $end, array $options = []): array
{
// Query your database or API here using $begin and $end
return [
new MyEvent($begin, $end),
];
}
}<?php
$manager = $calendar->getEventManager();
$manager->addProvider('my_source', new MyProvider());
$month = $calendar->getMonth(2025, 1);
$events = $manager->find($month);
foreach ($events as $event) {
// ...
}If you use Symfony, the bundle is automatically configured.
# config/packages/calendr.yaml
calendr:
periods:
default_first_weekday: 1 # MondayYou can access periods directly in your templates:
{# Iterate over days of the current month #}
{% set month = calendr_month(2025, 1) %}
<table>
{% for week in month %}
<tr>
{% for day in week %}
<td>
{{ day.begin|date('d') }}
{# Fetch events for this specific day #}
{% for event in calendr_events(day) %}
<span class="event">{{ event.uid }}</span>
{% endfor %}
</td>
{% endfor %}
</tr>
{% endfor %}
</table>