Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Database Notifications - Customizable "Mark All as Read" Action #14157

Open
wants to merge 8 commits into
base: 4.x
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 6 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -1,18 +1,12 @@
@props([
'notifications',
'unreadNotificationsCount',
'markAllNotificationsAsReadAction',
])

<div {{ $attributes->class('mt-2 flex gap-x-3') }}>
@if ($unreadNotificationsCount)
<x-filament::link
color="primary"
tabindex="-1"
tag="button"
wire:click="markAllNotificationsAsRead"
>
{{ __('filament-notifications::database.modal.actions.mark_all_as_read.label') }}
</x-filament::link>
@if ($unreadNotificationsCount && $markAllNotificationsAsReadAction)
{{ ($markAllNotificationsAsReadAction)->isVisible() ? $markAllNotificationsAsReadAction : '' }}
@endif

<x-filament::link
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
@props([
'notifications',
'unreadNotificationsCount',
'markAllNotificationsAsReadAction',
])

@php
Expand Down Expand Up @@ -33,6 +34,7 @@
<x-filament-notifications::database.modal.actions
:notifications="$notifications"
:unread-notifications-count="$unreadNotificationsCount"
:mark-all-notifications-as-read-action="$markAllNotificationsAsReadAction"
/>
</div>
</x-slot>
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
@php
$notifications = $this->getNotifications();
$unreadNotificationsCount = $this->getUnreadNotificationsCount();
$markAllNotificationsAsReadAction = $this->markAllNotificationsAsReadAction();
danharrin marked this conversation as resolved.
Show resolved Hide resolved
@endphp

<div
Expand All @@ -18,6 +19,7 @@ class="flex"
<x-filament-notifications::database.modal
:notifications="$notifications"
:unread-notifications-count="$unreadNotificationsCount"
:mark-all-notifications-as-read-action="$markAllNotificationsAsReadAction"
/>

@if ($broadcastChannel = $this->getBroadcastChannel())
Expand Down
21 changes: 20 additions & 1 deletion packages/notifications/src/Livewire/DatabaseNotifications.php
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,9 @@
namespace Filament\Notifications\Livewire;

use Carbon\CarbonInterface;
use Filament\Actions\Action;
use Filament\Actions\Concerns\InteractsWithActions;
use Filament\Actions\Contracts\HasActions;
use Filament\Notifications\Notification;
use Illuminate\Contracts\Auth\Authenticatable;
use Illuminate\Contracts\Pagination\Paginator;
Expand All @@ -16,8 +19,9 @@
use Livewire\Component;
use Livewire\WithPagination;

class DatabaseNotifications extends Component
class DatabaseNotifications extends Component implements HasActions
{
use InteractsWithActions;
use WithPagination;

public static bool $isPaginated = true;
Expand All @@ -28,6 +32,8 @@ class DatabaseNotifications extends Component

public static ?string $authGuard = null;

public static ?Action $markAllNotificationsAsReadAction = null;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Instead of storing the action, please add HasActions and InteractsWithActions to this component, and then rename getMarkAllNotificationsAsReadAction to markAllNotificationsAsReadAction. Then you can use $this->markAllNotificationsAsReadAction to render

Copy link
Member

@danharrin danharrin Dec 9, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You won't need this line anymore


#[On('databaseNotificationsSent')]
public function refresh(): void {}

Expand Down Expand Up @@ -113,6 +119,19 @@ public function getTrigger(): ?View
return view($viewPath);
}

public function markAllNotificationsAsReadAction(): Action
{
if ($action = static::$markAllNotificationsAsReadAction) {
return $action;
}

Comment on lines +127 to +130
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We don't need this

return Action::make('markAllNotificationsAsRead')
->link()
->label(__('filament-notifications::database.modal.actions.mark_all_as_read.label'))
->extraAttributes(['tabindex' => '-1'])
->action('markAllNotificationsAsRead');
}

public function getUser(): Model | Authenticatable | null
{
return auth(static::$authGuard)->user();
Expand Down
21 changes: 21 additions & 0 deletions packages/panels/docs/06-notifications.md
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,27 @@ public function panel(Panel $panel): Panel
}
```

## Customizing the "Mark All as Read" Action

To customize the "Mark All as Read" button, you can use the `databaseNotificationsMarkAllAsReadAction()` method, passing a closure that returns an action. All methods that are available to [customize action trigger buttons](../../actions/trigger-button) can be used:

```php
use Filament\Actions\Action;
use Filament\Panel;

public function panel(Panel $panel): Panel
{
return $panel
// ...
->databaseNotifications()
->databaseNotificationsMarkAllAsReadAction(fn (Action $action) => $action
->button()
->color('success')
->label('Read All')
);
}
```

## Setting up websockets in a panel

The Panel Builder comes with a level of inbuilt support for real-time broadcast and database notifications. However there are a number of areas you will need to install and configure to wire everything up and get it working.
Expand Down
1 change: 1 addition & 0 deletions packages/panels/src/Facades/Filament.php
Original file line number Diff line number Diff line change
Expand Up @@ -105,6 +105,7 @@
* @method static string | null getUrl(Model | null $tenant = null)
* @method static string getVerifyEmailUrl(MustVerifyEmail | Model | Authenticatable $user, array $parameters = [])
* @method static array getWidgets()
* @method static Action getDatabaseNotificationsMarkAllAsReadAction()
* @method static bool hasBreadcrumbs()
* @method static bool hasCollapsibleNavigationGroups()
* @method static bool hasDarkMode()
Expand Down
5 changes: 5 additions & 0 deletions packages/panels/src/FilamentManager.php
Original file line number Diff line number Diff line change
Expand Up @@ -589,6 +589,11 @@ public function getWidgets(): array
return $this->getCurrentPanel()->getWidgets();
}

public function getDatabaseNotificationsMarkAllAsReadAction(): Action
{
return $this->currentPanel->getDatabaseNotificationsMarkAllAsReadAction();
}

public function hasBreadcrumbs(): bool
{
return $this->getCurrentPanel()->hasBreadcrumbs();
Expand Down
6 changes: 6 additions & 0 deletions packages/panels/src/Livewire/DatabaseNotifications.php
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

namespace Filament\Livewire;

use Filament\Actions\Action;
use Filament\Facades\Filament;
use Filament\Notifications\Livewire\DatabaseNotifications as BaseComponent;
use Illuminate\Contracts\Auth\Authenticatable;
Expand All @@ -24,4 +25,9 @@ public function getTrigger(): View
{
return view('filament-panels::components.topbar.database-notifications-trigger');
}

public function markAllNotificationsAsReadAction(): Action
{
return Filament::getDatabaseNotificationsMarkAllAsReadAction();
}
}
27 changes: 27 additions & 0 deletions packages/panels/src/Panel/Concerns/HasNotifications.php
Original file line number Diff line number Diff line change
Expand Up @@ -3,13 +3,16 @@
namespace Filament\Panel\Concerns;

use Closure;
use Filament\Actions\Action;

trait HasNotifications
{
protected bool | Closure $hasDatabaseNotifications = false;

protected string | Closure | null $databaseNotificationsPolling = '30s';

protected ?Closure $modifyDatabaseNotificationsMarkAllAsReadUsing = null;

public function databaseNotifications(bool | Closure $condition = true): static
{
$this->hasDatabaseNotifications = $condition;
Expand All @@ -24,6 +27,13 @@ public function databaseNotificationsPolling(string | Closure | null $interval):
return $this;
}

public function databaseNotificationsMarkAllAsReadAction(?Closure $callback): static
{
$this->modifyDatabaseNotificationsMarkAllAsReadUsing = $callback;

return $this;
}

public function hasDatabaseNotifications(): bool
{
return (bool) $this->evaluate($this->hasDatabaseNotifications);
Expand All @@ -33,4 +43,21 @@ public function getDatabaseNotificationsPollingInterval(): ?string
{
return $this->evaluate($this->databaseNotificationsPolling);
}

public function getDatabaseNotificationsMarkAllAsReadAction(): Action
{
$action = Action::make('markAllNotificationsAsRead')
->link()
->label(__('filament-notifications::database.modal.actions.mark_all_as_read.label'))
->extraAttributes(['tabindex' => '-1'])
->action('markAllNotificationsAsRead');

if ($this->modifyDatabaseNotificationsMarkAllAsReadUsing) {
$action = $this->evaluate($this->modifyDatabaseNotificationsMarkAllAsReadUsing, [
'action' => $action,
]) ?? $action;
}

return $action;
}
}