With spatie/opening-hours
you create an object that describes a business' opening hours, which you can query for open
or closed
on days or specific dates, or use to present the times per day.
A set of opening hours is created by passing in a regular schedule, and a list of exceptions.
// Add the use at the top of each file where you want to use the OpeningHours class:
use Spatie\OpeningHours\OpeningHours;
$openingHours = OpeningHours::create([
'monday' => ['09:00-12:00', '13:00-18:00'],
'tuesday' => ['09:00-12:00', '13:00-18:00'],
'wednesday' => ['09:00-12:00'],
'thursday' => ['09:00-12:00', '13:00-18:00'],
'friday' => ['09:00-12:00', '13:00-20:00'],
'saturday' => ['09:00-12:00', '13:00-16:00'],
'sunday' => [],
'exceptions' => [
'2016-11-11' => ['09:00-12:00'],
'2016-12-25' => [],
'01-01' => [], // Recurring on each 1st of January
'12-25' => ['09:00-12:00'], // Recurring on each 25th of December
],
]);
The object can be queried for a day in the week, which will return a result based on the regular schedule:
// Open on Mondays:
$openingHours->isOpenOn('monday'); // true
// Closed on Sundays:
$openingHours->isOpenOn('sunday'); // false
It can also be queried for a specific date and time:
// Closed because it's after hours:
$openingHours->isOpenAt(new DateTime('2016-09-26 19:00:00')); // false
// Closed because Christmas was set as an exception
$openingHours->isOpenAt(new DateTime('2016-12-25')); // false
It can also return arrays of opening hours for a week or a day:
// OpeningHoursForDay object for the regular schedule
$openingHours->forDay('monday');
// OpeningHoursForDay[] for the regular schedule, keyed by day name
$openingHours->forWeek();
// Array of day with same schedule for the regular schedule, keyed by day name, days combined by working hours
$openingHours->forWeekCombined();
// OpeningHoursForDay object for a specific day
$openingHours->forDate(new DateTime('2016-12-25'));
// OpeningHoursForDay[] of all exceptions, keyed by date
$openingHours->exceptions();
You can add data in definitions then retrieve them:
$openingHours = OpeningHours::create([
'monday' => [
'data' => 'Typical Monday',
'09:00-12:00',
'13:00-18:00',
],
'tuesday' => [
'09:00-12:00',
'13:00-18:00',
[
'19:00-21:00',
'data' => 'Extra on Tuesday evening',
],
],
'exceptions' => [
'2016-12-25' => [
'data' => 'Closed for Christmas',
],
],
]);
echo $openingHours->forDay('monday')->getData(); // Typical Monday
echo $openingHours->forDate(new DateTime('2016-12-25'))->getData(); // Closed for Christmas
echo $openingHours->forDay('tuesday')[2]->getData(); // Extra on Tuesday evening
In the example above, data are strings but it can be any kind of value. So you can embed multiple properties in an array.
For structure convenience, the data-hours couple can be a fully-associative array, so the example above is strictly equivalent to the following:
$openingHours = OpeningHours::create([
'monday' => [
'hours' => [
'09:00-12:00',
'13:00-18:00',
],
'data' => 'Typical Monday',
],
'tuesday' => [
['hours' => '09:00-12:00'],
['hours' => '13:00-18:00'],
['hours' => '19:00-21:00', 'data' => 'Extra on Tuesday evening'],
],
// Open by night from Wednesday 22h to Thursday 7h:
'wednesday' => ['22:00-24:00'], // use the special "24:00" to reach midnight included
'thursday' => ['00:00-07:00'],
'exceptions' => [
'2016-12-25' => [
'hours' => [],
'data' => 'Closed for Christmas',
],
],
]);
The last structure tool is the filter, it allows you to pass closures (or callable function/method reference) that take a date as a parameter and returns the settings for the given date.
$openingHours = OpeningHours::create([
'monday' => [
'09:00-12:00',
],
'filters' => [
function ($date) {
$year = intval($date->format('Y'));
$easterMonday = new DateTimeImmutable('2018-03-21 +'.(easter_days($year) + 1).'days');
if ($date->format('m-d') === $easterMonday->format('m-d')) {
return []; // Closed on Easter Monday
// Any valid exception-array can be returned here (range of hours, with or without data)
}
// Else the filter does not apply to the given date
},
],
]);
If a callable is found in the "exceptions"
property, it will be added automatically to filters so you can mix filters and exceptions both in the exceptions array. The first filter that returns a non-null value will have precedence over the next filters and the filters array has precedence over the filters inside the exceptions array.
Warning: We will loop on all filters for each date from which we need to retrieve opening hours and can neither predicate nor cache the result (can be a random function) so you must be careful with filters, too many filters or long process inside filters can have a significant impact on the performance.
It can also return the next open or close DateTime
from a given DateTime
.
// The next open datetime is tomorrow morning, because we’re closed on 25th of December.
$nextOpen = $openingHours->nextOpen(new DateTime('2016-12-25 10:00:00')); // 2016-12-26 09:00:00
// The next open datetime is this afternoon, after the lunch break.
$nextOpen = $openingHours->nextOpen(new DateTime('2016-12-24 11:00:00')); // 2016-12-24 13:00:00
// The next close datetime is at noon.
$nextClose = $openingHours->nextClose(new DateTime('2016-12-24 10:00:00')); // 2016-12-24 12:00:00
// The next close datetime is tomorrow at noon, because we’re closed on 25th of December.
$nextClose = $openingHours->nextClose(new DateTime('2016-12-25 15:00:00')); // 2016-12-26 12:00:00
Read the usage section for the full api.
Spatie is a webdesign agency based in Antwerp, Belgium. You'll find an overview of all our open source projects on our website.
You can install the package via composer:
composer require spatie/opening-hours
The package should only be used through the OpeningHours
class. There are also three value object classes used throughout, Time
, which represents a single time, TimeRange
, which represents a period with a start and an end, and openingHoursForDay
, which represents a set of TimeRange
s which can't overlap.
Static factory method to fill the set of opening hours.
$openingHours = OpeningHours::create([
'monday' => ['09:00-12:00', '13:00-18:00'],
// ...
]);
For safety sake, creating OpeningHours
object with overlapping ranges will throw an exception. But you can explicitly merge them.
$ranges = [
'monday' => ['08:00-11:00', '10:00-12:00'],
];
$mergedRanges = OpeningHours::mergeOverlappingRanges($ranges); // Monday becomes ['08:00-12:00']
OpeningHours::create($mergedRanges);
// Or use the following shortcut to create from ranges that possibly overlap:
OpeningHours::createAndMergeOverlappingRanges($ranges);
Not all days are mandatory, if a day is missing, it will be set as closed.
The same as create
, but non-static.
$openingHours = (new OpeningHours)->fill([
'monday' => ['09:00-12:00', '13:00-18:00'],
// ...
]);
Returns an array of OpeningHoursForDay
objects for a regular week.
$openingHours->forWeek();
Returns an array of days. Array key is first day with same hours, array values are days that have the same working hours and OpeningHoursForDay
object.
$openingHours->forWeekCombined();
Returns an OpeningHoursForDay
object for a regular day. A day is lowercase string of the english day name.
$openingHours->forDay('monday');
Returns an OpeningHoursForDay
object for a specific date. It looks for an exception on that day, and otherwise it returns the opening hours based on the regular schedule.
$openingHours->forDate(new DateTime('2016-12-25'));
Returns an array of all OpeningHoursForDay
objects for exceptions, keyed by a Y-m-d
date string.
$openingHours->exceptions();
Checks if the business is open on a day in the regular schedule.
$openingHours->isOpenOn('saturday');
Checks if the business is closed on a day in the regular schedule.
$openingHours->isClosedOn('sunday');
Checks if the business is open on a specific day, at a specific time.
$openingHours->isOpenAt(new DateTime('2016-26-09 20:00'));
Checks if the business is closed on a specific day, at a specific time.
$openingHours->isClosedAt(new DateTime('2016-26-09 20:00'));
Checks if the business is open right now.
$openingHours->isOpen();
Checks if the business is closed right now.
$openingHours->isClosed();
Returns next open DateTime from the given DateTime
$openingHours->nextOpen(new DateTime('2016-12-24 11:00:00'));
Returns next close DateTime from the given DateTime
$openingHours->nextClose(new DateTime('2016-12-24 11:00:00'));
This class is meant as read-only. It implements ArrayAccess
, Countable
and IteratorAggregate
so you can process the list of TimeRange
s in an array-like way.
Value object describing a period with a start and an end time. Can be cast to a string in a H:i-H:i
format.
Value object describing a single time. Can be cast to a string in a H:i
format.
Please see CHANGELOG for more information about what has changed recently.
composer test
Please see CONTRIBUTING for details.
If you discover any security related issues, please email [email protected] instead of using the issue tracker.
You're free to use this package, but if it makes it to your production environment we highly appreciate you sending us a postcard from your hometown, mentioning which of our package(s) you are using.
Our address is: Spatie, Samberstraat 69D, 2060 Antwerp, Belgium.
We publish all received postcards on our company website.
Spatie is a webdesign agency based in Antwerp, Belgium. You'll find an overview of all our open source projects on our website.
Does your business depend on our contributions? Reach out and support us on Patreon. All pledges will be dedicated to allocating workforce on maintenance and new awesome stuff.
The MIT License (MIT). Please see License File for more information.