Skip to content

Commit

Permalink
Notifications: added user preference UI & logic
Browse files Browse the repository at this point in the history
Includes testing to cover.
Also added file missing from previous commit.
  • Loading branch information
ssddanbrown committed Jul 25, 2023
1 parent 45e75ed commit 634e74e
Show file tree
Hide file tree
Showing 8 changed files with 211 additions and 7 deletions.
1 change: 1 addition & 0 deletions app/Access/Guards/LdapSessionGuard.php
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
use BookStack\Users\Models\User;
use Illuminate\Contracts\Auth\UserProvider;
use Illuminate\Contracts\Session\Session;
use Illuminate\Support\Facades\Hash;
use Illuminate\Support\Str;

class LdapSessionGuard extends ExternalBaseSessionGuard
Expand Down
49 changes: 49 additions & 0 deletions app/Activity/Notifications/NotificationManager.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
<?php

namespace BookStack\Activity\Notifications;

use BookStack\Activity\ActivityType;
use BookStack\Activity\Models\Loggable;
use BookStack\Activity\Notifications\Handlers\CommentCreationNotificationHandler;
use BookStack\Activity\Notifications\Handlers\NotificationHandler;
use BookStack\Activity\Notifications\Handlers\PageCreationNotificationHandler;
use BookStack\Activity\Notifications\Handlers\PageUpdateNotificationHandler;

class NotificationManager
{
/**
* @var class-string<NotificationHandler>[]
*/
protected array $handlers = [];

public function handle(string $activityType, string|Loggable $detail): void
{
$handlersToRun = $this->handlers[$activityType] ?? [];
foreach ($handlersToRun as $handlerClass) {
/** @var NotificationHandler $handler */
$handler = app()->make($handlerClass);
$handler->handle($activityType, $detail);
}
}

/**
* @param class-string<NotificationHandler> $handlerClass
*/
public function registerHandler(string $activityType, string $handlerClass): void
{
if (!isset($this->handlers[$activityType])) {
$this->handlers[$activityType] = [];
}

if (!in_array($handlerClass, $this->handlers[$activityType])) {
$this->handlers[$activityType][] = $handlerClass;
}
}

public function loadDefaultHandlers(): void
{
$this->registerHandler(ActivityType::PAGE_CREATE, PageCreationNotificationHandler::class);
$this->registerHandler(ActivityType::PAGE_UPDATE, PageUpdateNotificationHandler::class);
$this->registerHandler(ActivityType::COMMENT_CREATE, CommentCreationNotificationHandler::class);
}
}
46 changes: 46 additions & 0 deletions app/Settings/UserNotificationPreferences.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
<?php

namespace BookStack\Settings;

use BookStack\Users\Models\User;

class UserNotificationPreferences
{
public function __construct(
protected User $user
) {
}

public function notifyOnOwnPageChanges(): bool
{
return $this->getNotificationSetting('own-page-changes');
}

public function notifyOnOwnPageComments(): bool
{
return $this->getNotificationSetting('own-page-comments');
}

public function notifyOnCommentReplies(): bool
{
return $this->getNotificationSetting('comment-replies');
}

public function updateFromSettingsArray(array $settings)
{
$allowList = ['own-page-changes', 'own-page-comments', 'comment-replies'];
foreach ($settings as $setting => $status) {
if (!in_array($setting, $allowList)) {
continue;
}

$value = $status === 'true' ? 'true' : 'false';
setting()->putUser($this->user, 'notifications#' . $setting, $value);
}
}

protected function getNotificationSetting(string $key): bool
{
return setting()->getUser($this->user, 'notifications#' . $key);
}
}
40 changes: 34 additions & 6 deletions app/Users/Controllers/UserPreferencesController.php
Original file line number Diff line number Diff line change
Expand Up @@ -3,17 +3,16 @@
namespace BookStack\Users\Controllers;

use BookStack\Http\Controller;
use BookStack\Settings\UserNotificationPreferences;
use BookStack\Settings\UserShortcutMap;
use BookStack\Users\UserRepo;
use Illuminate\Http\Request;

class UserPreferencesController extends Controller
{
protected UserRepo $userRepo;

public function __construct(UserRepo $userRepo)
{
$this->userRepo = $userRepo;
public function __construct(
protected UserRepo $userRepo
) {
}

/**
Expand Down Expand Up @@ -47,6 +46,35 @@ public function updateShortcuts(Request $request)
return redirect('/preferences/shortcuts');
}

/**
* Show the notification preferences for the current user.
*/
public function showNotifications()
{
$preferences = (new UserNotificationPreferences(user()));

return view('users.preferences.notifications', [
'preferences' => $preferences,
]);
}

/**
* Update the notification preferences for the current user.
*/
public function updateNotifications(Request $request)
{
$data = $this->validate($request, [
'preferences' => ['required', 'array'],
'preferences.*' => ['required', 'string'],
]);

$preferences = (new UserNotificationPreferences(user()));
$preferences->updateFromSettingsArray($data['preferences']);
$this->showSuccessNotification(trans('preferences.notifications_update_success'));

return redirect('/preferences/notifications');
}

/**
* Update the preferred view format for a list view of the given type.
*/
Expand Down Expand Up @@ -123,7 +151,7 @@ public function updateCodeLanguageFavourite(Request $request)
{
$validated = $this->validate($request, [
'language' => ['required', 'string', 'max:20'],
'active' => ['required', 'bool'],
'active' => ['required', 'bool'],
]);

$currentFavoritesStr = setting()->getForCurrentUser('code-language-favourites', '');
Expand Down
10 changes: 9 additions & 1 deletion lang/en/preferences.php
Original file line number Diff line number Diff line change
Expand Up @@ -15,4 +15,12 @@
'shortcuts_save' => 'Save Shortcuts',
'shortcuts_overlay_desc' => 'Note: When shortcuts are enabled a helper overlay is available via pressing "?" which will highlight the available shortcuts for actions currently visible on the screen.',
'shortcuts_update_success' => 'Shortcut preferences have been updated!',
];

'notifications' => 'Notification Preferences',
'notifications_desc' => 'Control the email notifications you receive when certain activity is performed within the system.',
'notifications_opt_own_page_changes' => 'Notify upon changes to pages I own',
'notifications_opt_own_page_comments' => 'Notify upon comments on pages I own',
'notifications_opt_comment_replies' => 'Notify upon replies to my comments',
'notifications_save' => 'Save Preferences',
'notifications_update_success' => 'Notification preferences have been updated!',
];
45 changes: 45 additions & 0 deletions resources/views/users/preferences/notifications.blade.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
@extends('layouts.simple')

@section('body')
<div class="container small my-xl">

<section class="card content-wrap auto-height">
<form action="{{ url('/preferences/notifications') }}" method="post">
{{ method_field('put') }}
{{ csrf_field() }}

<h1 class="list-heading">{{ trans('preferences.notifications') }}</h1>
<p class="text-small text-muted">{{ trans('preferences.notifications_desc') }}</p>

<div class="toggle-switch-list">
<div>
@include('form.toggle-switch', [
'name' => 'preferences[own-page-changes]',
'value' => $preferences->notifyOnOwnPageChanges(),
'label' => trans('preferences.notifications_opt_own_page_changes'),
])
</div>
<div>
@include('form.toggle-switch', [
'name' => 'preferences[own-page-comments]',
'value' => $preferences->notifyOnOwnPageComments(),
'label' => trans('preferences.notifications_opt_own_page_comments'),
])
</div>
<div>
@include('form.toggle-switch', [
'name' => 'preferences[comment-replies]',
'value' => $preferences->notifyOnCommentReplies(),
'label' => trans('preferences.notifications_opt_comment_replies'),
])
</div>
</div>

<div class="form-group text-right">
<button class="button">{{ trans('preferences.notifications_save') }}</button>
</div>
</form>
</section>

</div>
@stop
2 changes: 2 additions & 0 deletions routes/web.php
Original file line number Diff line number Diff line change
Expand Up @@ -231,6 +231,8 @@
Route::redirect('/preferences', '/');
Route::get('/preferences/shortcuts', [UserControllers\UserPreferencesController::class, 'showShortcuts']);
Route::put('/preferences/shortcuts', [UserControllers\UserPreferencesController::class, 'updateShortcuts']);
Route::get('/preferences/notifications', [UserControllers\UserPreferencesController::class, 'showNotifications']);
Route::put('/preferences/notifications', [UserControllers\UserPreferencesController::class, 'updateNotifications']);
Route::patch('/preferences/change-view/{type}', [UserControllers\UserPreferencesController::class, 'changeView']);
Route::patch('/preferences/change-sort/{type}', [UserControllers\UserPreferencesController::class, 'changeSort']);
Route::patch('/preferences/change-expansion/{type}', [UserControllers\UserPreferencesController::class, 'changeExpansion']);
Expand Down
25 changes: 25 additions & 0 deletions tests/User/UserPreferencesTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,31 @@ public function test_body_has_shortcuts_component_when_active()
$this->withHtml($this->get('/'))->assertElementExists('body[component="shortcuts"]');
}

public function test_notification_preferences_updating()
{
$this->asEditor();

// View preferences with defaults
$resp = $this->get('/preferences/notifications');
$resp->assertSee('Notification Preferences');

$html = $this->withHtml($resp);
$html->assertFieldHasValue('preferences[comment-replies]', 'false');

// Update preferences
$resp = $this->put('/preferences/notifications', [
'preferences' => ['comment-replies' => 'true'],
]);

$resp->assertRedirect('/preferences/notifications');
$resp->assertSessionHas('success', 'Notification preferences have been updated!');

// View updates to preferences page
$resp = $this->get('/preferences/notifications');
$html = $this->withHtml($resp);
$html->assertFieldHasValue('preferences[comment-replies]', 'true');
}

public function test_update_sort_preference()
{
$editor = $this->users->editor();
Expand Down

0 comments on commit 634e74e

Please sign in to comment.