diff --git a/README.md b/README.md
index 6959137..83dab11 100644
--- a/README.md
+++ b/README.md
@@ -66,7 +66,9 @@ use Saade\FilamentFullCalendar\Widgets\FullCalendarWidget;
class CalendarWidget extends FullCalendarWidget
{
-
+ /**
+ * Return events that should be rendered statically on calendar first render.
+ */
public function getViewData(): array
{
return [
@@ -84,10 +86,22 @@ class CalendarWidget extends FullCalendarWidget
]
];
}
+
+ /**
+ * FullCalendar will call this function whenever it needs new event data.
+ * This is triggered when the user clicks prev/next or switches views on the calendar.
+ */
+ public function fetchEvents(array $fetchInfo, array $ignorableIds): array
+ {
+ // You can use $fetchInfo to filter events by date, and $ignorableIds to ignore already displayed events.
+ return [];
+ }
}
```
-> You should return an array of FullCalendar [EventObject](https://fullcalendar.io/docs/event-object).
+You can use one or both methods to fetch events.
+
+> Both methods should retun an array of [EventObject](https://fullcalendar.io/docs/event-object).
@@ -132,7 +146,7 @@ return [
# Listening for events
-The only events supported right now are: [EventClick](https://fullcalendar.io/docs/eventClick) and [EventDrop](https://fullcalendar.io/docs/eventDrop)
+The only event-related events supported right now are: [EventClick](https://fullcalendar.io/docs/eventClick) and [EventDrop](https://fullcalendar.io/docs/eventDrop)
They're commented out by default so livewire does not spam requests without they being used. You are free to paste them in your `CalendarWidget` class. See: [FiresEvents](https://github.com/saade/filament-fullcalendar/blob/main/src/Widgets/Concerns/FiresEvents.php)
@@ -171,7 +185,14 @@ The process of saving and editing the event is up to you, since this plugin does
## Creating Events:
-Override the `createEvent` function in your widget class, and you are ready to go!
+Events can be created in two ways.
+
+- Clicking on a day (default)
+- Selecting a date range (click and drag across calendar days) (you need to opt-in for this, set `selectable => true` in the config file.)
+
+This will open the Create Event modal.
+
+When the create form gets submitted, it will call the `createEvent` function on your widget. Be sure to add the snippet below to your calendar class.
```php
public function createEvent(array $data): void
@@ -200,7 +221,11 @@ protected static function getCreateEventFormSchema(): array
## Editing Events:
-Override the `editEvent` function in your widget class, and you are ready to go!
+Events can be edited by clicking on an event on the calendar.
+
+This will open the Edit Event modal.
+
+When the edit form gets submitted, it will call the `editEvent` function on your widget. Be sure to add the snippet below to your calendar class.
```php
public function editEvent(array $data): void
@@ -271,6 +296,50 @@ public function yourMethod(): void
+# Filtering events based on the calendar view
+
+If you want to filter your events based on the days that are currently shown in the calendar, you can implement the `fetchInfo()` method from the [CanFetchEvents](https://github.com/saade/filament-fullcalendar/blob/main/src/Widgets/Concerns/CanFetchEvents.php) trait. Add the following code to your calendar widget:
+
+```php
+/**
+ * FullCalendar will call this function whenever it needs new event data.
+ * This is triggered when the user clicks prev/next or switches views.
+ *
+ * @see https://fullcalendar.io/docs/events-function
+ * @param array $fetchInfo start and end date of the current view
+ * @param array $ignorableIds ids of the events that are already loaded and should be ignored
+ */
+public function fetchEvents(array $fetchInfo, array $ignorableIds): array
+{
+ return [];
+}
+```
+
+you can filter events based on the timespan `$fetchInfo['start']` and `$fetchInfo['end']`.
+> **Warning**
+>
+> Keep in mind that returning events that are already in the calendar, can cause duplicates. You should filter them out using the `$ignorableIds` ids.
+
+example:
+```php
+public function fetchEvents(array $fetchInfo, array $ignorableIds): array
+{
+ $schedules = Appointment::query()
+ ->where([
+ ['start_at', '>=', $fetchInfo['start']],
+ ['end_at', '<', $fetchInfo['end']],
+ ])
+ ->whereNotIn('id', $ignorableIds)
+ ->get();
+
+ $data = $schedules->map( ... );
+
+ return $data;
+}
+```
+
+
+
## Testing
```bash
diff --git a/resources/views/fullcalendar.blade.php b/resources/views/fullcalendar.blade.php
index 8daae89..032f68e 100644
--- a/resources/views/fullcalendar.blade.php
+++ b/resources/views/fullcalendar.blade.php
@@ -1,4 +1,4 @@
-@php($locale = strtolower(str_replace('_', '-', $this->getConfig()['locale'])))
+@php($locale = strtolower(str_replace('_', '-', $this->config('locale', config('app.locale')))))
@@ -7,13 +7,12 @@
x-data=""
x-init='
document.addEventListener("DOMContentLoaded", function() {
- var initial = true;
const config = @json($this->getConfig());
- const eventsData = @json($events);
const locale = "{{ $locale }}";
- @if($this->isLazyLoad())
- const cachedEventIds = [];
- @endif
+ const events = @json($events);
+ const cachedEventIds = [
+ ...events.map(event => event.id),
+ ];
const eventClick = function ({ event, jsEvent }) {
if( event.url ) {
@@ -45,6 +44,21 @@
@endif
}
+ const fetchEvents = function ({ start, end }, successCallback, failureCallback) {
+ @if( $this::canFetchEvents() )
+ return $wire.fetchEvents({ start, end }, cachedEventIds)
+ .then(events => {
+ // Cache fetched events
+ cachedEventIds.push(...events.map(event => event.id));
+
+ return successCallback(events);
+ })
+ .catch( failureCallback );
+ @else
+ return successCallback([]);
+ @endif
+ }
+
const calendar = new FullCalendar.Calendar($el, {
...config,
locale,
@@ -52,46 +66,18 @@
eventDrop,
dateClick,
select,
- @if($this->isLazyLoad())
- events: function(fetchInfo, successCallback, failureCallback) {
- if(initial){
- initial = false
-
- if(eventsData[0]?.id){
- eventsData.forEach((event) => cachedEventIds.push(event.id))
- }
-
- successCallback(eventsData)
- }else{
- $wire.lazyLoadViewData(fetchInfo)
- .then(result => {
- if(result.length == 0) return
-
- if(result[0].id){
- result.forEach((event) => cachedEventIds.indexOf(event.id) != -1 ? null : cachedEventIds.push(event.id) && eventsData.push(event))
- successCallback(eventsData)
- }else{
- successCallback(result)
- }
- })
- }
- },
- @else
- events: eventsData,
- @endif
+ eventSources:[
+ { events },
+ fetchEvents
+ ]
});
calendar.render();
- window.addEventListener("filament-fullcalendar:refresh", (event) => {
+ window.addEventListener("filament-fullcalendar:refresh", () => {
calendar.removeAllEvents();
- @unless($this->isLazyLoad())
- event.detail.data.map(event => calendar.addEvent(event));
- @else
- cachedEventIds.splice(0, cachedEventIds.length)
- calendar.refetchEvents()
- eventsData.splice(0, eventsData.length)
- @endunless
+ cachedEventIds.length = 0;
+ calendar.refetchEvents();
});
})
'>
diff --git a/src/Widgets/Concerns/CanFetchEvents.php b/src/Widgets/Concerns/CanFetchEvents.php
new file mode 100644
index 0000000..9a6b724
--- /dev/null
+++ b/src/Widgets/Concerns/CanFetchEvents.php
@@ -0,0 +1,25 @@
+dispatchBrowserEvent('filament-fullcalendar:refresh', $this->isLazyLoad() ? null : ['data' => $this->getViewData()]);
+ $this->dispatchBrowserEvent('filament-fullcalendar:refresh');
}
}
diff --git a/src/Widgets/Concerns/UsesLazyLoad.php b/src/Widgets/Concerns/UsesLazyLoad.php
deleted file mode 100644
index dc20224..0000000
--- a/src/Widgets/Concerns/UsesLazyLoad.php
+++ /dev/null
@@ -1,13 +0,0 @@
-with([
- 'events' => $this->isLazyLoad() ? $this->lazyLoadViewData() : $this->getViewData(),
+ 'events' => $this->getViewData(),
]);
}
}