Notifications: User watch list and differnt page watch options

- Adds option filtering and alternative text for page watch options.
- Adds "Watched & Ignored Items" list to user notification preferences
  page to show existing watched items.
This commit is contained in:
Dan Brown 2023-08-14 13:11:18 +01:00
parent c47b3f805a
commit d9fdecd902
No known key found for this signature in database
GPG key ID: 46D9F943C24A2EF9
10 changed files with 99 additions and 16 deletions

View file

@ -2,10 +2,12 @@
namespace BookStack\Activity\Models; namespace BookStack\Activity\Models;
use BookStack\Activity\WatchLevels;
use BookStack\Permissions\Models\JointPermission; use BookStack\Permissions\Models\JointPermission;
use Carbon\Carbon; use Carbon\Carbon;
use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\HasMany; use Illuminate\Database\Eloquent\Relations\HasMany;
use Illuminate\Database\Eloquent\Relations\MorphTo;
/** /**
* @property int $id * @property int $id
@ -20,14 +22,24 @@ class Watch extends Model
{ {
protected $guarded = []; protected $guarded = [];
public function watchable() public function watchable(): MorphTo
{ {
$this->morphTo(); return $this->morphTo();
} }
public function jointPermissions(): HasMany public function jointPermissions(): HasMany
{ {
return $this->hasMany(JointPermission::class, 'entity_id', 'watchable_id') return $this->hasMany(JointPermission::class, 'entity_id', 'watchable_id')
->whereColumn('favourites.watchable_type', '=', 'joint_permissions.entity_type'); ->whereColumn('watches.watchable_type', '=', 'joint_permissions.entity_type');
}
public function getLevelName(): string
{
return WatchLevels::levelValueToName($this->level);
}
public function ignoring(): bool
{
return $this->level === WatchLevels::IGNORE;
} }
} }

View file

@ -2,6 +2,10 @@
namespace BookStack\Activity; namespace BookStack\Activity;
use BookStack\Entities\Models\Bookshelf;
use BookStack\Entities\Models\Entity;
use BookStack\Entities\Models\Page;
class WatchLevels class WatchLevels
{ {
/** /**
@ -32,6 +36,7 @@ class WatchLevels
/** /**
* Get all the possible values as an option_name => value array. * Get all the possible values as an option_name => value array.
* @returns array<string, int>
*/ */
public static function all(): array public static function all(): array
{ {
@ -43,11 +48,36 @@ class WatchLevels
return $options; return $options;
} }
public static function levelNameToValue(string $level): int /**
* Get the watch options suited for the given entity.
* @returns array<string, int>
*/
public static function allSuitedFor(Entity $entity): array
{ {
return static::all()[$level] ?? -1; $options = static::all();
if ($entity instanceof Page) {
unset($options['new']);
} elseif ($entity instanceof Bookshelf) {
return [];
}
return $options;
} }
/**
* Convert the given name to a level value.
* Defaults to default value if the level does not exist.
*/
public static function levelNameToValue(string $level): int
{
return static::all()[$level] ?? static::DEFAULT;
}
/**
* Convert the given int level value to a level name.
* Defaults to 'default' level name if not existing.
*/
public static function levelValueToName(int $level): string public static function levelValueToName(int $level): string
{ {
foreach (static::all() as $name => $value) { foreach (static::all() as $name => $value) {

View file

@ -10,7 +10,7 @@ class TopFavourites extends EntityQuery
public function run(int $count, int $skip = 0) public function run(int $count, int $skip = 0)
{ {
$user = user(); $user = user();
if (is_null($user) || $user->isDefault()) { if ($user->isDefault()) {
return collect(); return collect();
} }

View file

@ -2,7 +2,9 @@
namespace BookStack\Users\Controllers; namespace BookStack\Users\Controllers;
use BookStack\Activity\Models\Watch;
use BookStack\Http\Controller; use BookStack\Http\Controller;
use BookStack\Permissions\PermissionApplicator;
use BookStack\Settings\UserNotificationPreferences; use BookStack\Settings\UserNotificationPreferences;
use BookStack\Settings\UserShortcutMap; use BookStack\Settings\UserShortcutMap;
use BookStack\Users\UserRepo; use BookStack\Users\UserRepo;
@ -49,12 +51,17 @@ class UserPreferencesController extends Controller
/** /**
* Show the notification preferences for the current user. * Show the notification preferences for the current user.
*/ */
public function showNotifications() public function showNotifications(PermissionApplicator $permissions)
{ {
$preferences = (new UserNotificationPreferences(user())); $preferences = (new UserNotificationPreferences(user()));
$query = Watch::query()->where('user_id', '=', user()->id);
$query = $permissions->restrictEntityRelationQuery($query, 'watches', 'watchable_id', 'watchable_type');
$watches = $query->with('watchable')->paginate(20);
return view('users.preferences.notifications', [ return view('users.preferences.notifications', [
'preferences' => $preferences, 'preferences' => $preferences,
'watches' => $watches,
]); ]);
} }

View file

@ -414,8 +414,10 @@ return [
'watch_desc_new' => 'Notify when any new page is created within this item.', 'watch_desc_new' => 'Notify when any new page is created within this item.',
'watch_title_updates' => 'All Page Updates', 'watch_title_updates' => 'All Page Updates',
'watch_desc_updates' => 'Notify upon all new pages and page changes.', 'watch_desc_updates' => 'Notify upon all new pages and page changes.',
'watch_desc_updates_page' => 'Notify upon all page changes.',
'watch_title_comments' => 'All Page Updates & Comments', 'watch_title_comments' => 'All Page Updates & Comments',
'watch_desc_comments' => 'Notify upon all new pages, page changes and new comments.', 'watch_desc_comments' => 'Notify upon all new pages, page changes and new comments.',
'watch_desc_comments_page' => 'Notify upon page changes and new comments.',
'watch_change_default' => 'Change default notification preferences', 'watch_change_default' => 'Change default notification preferences',
'watch_detail_ignore' => 'Ignoring notifications', 'watch_detail_ignore' => 'Ignoring notifications',
'watch_detail_new' => 'Watching for new pages', 'watch_detail_new' => 'Watching for new pages',

View file

@ -23,4 +23,6 @@ return [
'notifications_opt_comment_replies' => 'Notify upon replies to my comments', 'notifications_opt_comment_replies' => 'Notify upon replies to my comments',
'notifications_save' => 'Save Preferences', 'notifications_save' => 'Save Preferences',
'notifications_update_success' => 'Notification preferences have been updated!', 'notifications_update_success' => 'Notification preferences have been updated!',
'notifications_watched' => 'Watched & Ignored Items',
'notifications_watched_desc' => ' Below are the items that have custom watch preferences applied. To update your preferences for these, view the item then find the watch options in the sidebar.',
]; ];

View file

@ -0,0 +1,7 @@
<a href="{{ $entity->getUrl() }}" class="flex-container-row items-center">
<span role="presentation"
class="icon flex-none text-{{$entity->getType()}}">@icon($entity->getType())</span>
<div class="flex text-{{ $entity->getType() }}">
{{ $entity->name }}
</div>
</a>

View file

@ -11,7 +11,7 @@
<input type="hidden" name="id" value="{{ $entity->id }}"> <input type="hidden" name="id" value="{{ $entity->id }}">
<ul refs="dropdown@menu" class="dropdown-menu xl-limited anchor-left pb-none"> <ul refs="dropdown@menu" class="dropdown-menu xl-limited anchor-left pb-none">
@foreach(\BookStack\Activity\WatchLevels::all() as $option => $value) @foreach(\BookStack\Activity\WatchLevels::allSuitedFor($entity) as $option => $value)
<li> <li>
<button name="level" value="{{ $option }}" class="icon-item"> <button name="level" value="{{ $option }}" class="icon-item">
@if($watchLevel === $option) @if($watchLevel === $option)
@ -23,7 +23,11 @@
<div class="break-text"> <div class="break-text">
<div class="mb-xxs"><strong>{{ trans('entities.watch_title_' . $option) }}</strong></div> <div class="mb-xxs"><strong>{{ trans('entities.watch_title_' . $option) }}</strong></div>
<div class="text-muted text-small"> <div class="text-muted text-small">
{{ trans('entities.watch_desc_' . $option) }} @if(trans()->has('entities.watch_desc_' . $option . '_' . $entity->getMorphClass()))
{{ trans('entities.watch_desc_' . $option . '_' . $entity->getMorphClass()) }}
@else
{{ trans('entities.watch_desc_' . $option) }}
@endif
</div> </div>
</div> </div>
</button> </button>

View file

@ -95,13 +95,7 @@
:</strong> {{ $activity->type }}</div> :</strong> {{ $activity->type }}</div>
<div class="flex-3 px-m py-xxs min-width-l"> <div class="flex-3 px-m py-xxs min-width-l">
@if($activity->entity) @if($activity->entity)
<a href="{{ $activity->entity->getUrl() }}" class="flex-container-row items-center"> @include('entities.icon-link', ['entity' => $activity->entity])
<span role="presentation"
class="icon flex-none text-{{$activity->entity->getType()}}">@icon($activity->entity->getType())</span>
<div class="flex text-{{ $activity->entity->getType() }}">
{{ $activity->entity->name }}
</div>
</a>
@elseif($activity->detail && $activity->isForEntity()) @elseif($activity->detail && $activity->isForEntity())
<div> <div>
{{ trans('settings.audit_deleted_item') }} <br> {{ trans('settings.audit_deleted_item') }} <br>

View file

@ -41,5 +41,30 @@
</form> </form>
</section> </section>
<section class="card content-wrap auto-height">
<h2 class="list-heading">{{ trans('preferences.notifications_watched') }}</h2>
<p class="text-small text-muted">{{ trans('preferences.notifications_watched_desc') }}</p>
@if($watches->isEmpty())
<p class="text-muted italic">{{ trans('common.no_items') }}</p>
@else
<div class="item-list">
@foreach($watches as $watch)
<div class="flex-container-row justify-space-between item-list-row items-center wrap px-m py-s">
<div class="py-xs px-s min-width-m">
@include('entities.icon-link', ['entity' => $watch->watchable])
</div>
<div class="py-xs min-width-m text-m-right px-m">
@icon('watch' . ($watch->ignoring() ? '-ignore' : ''))
{{ trans('entities.watch_title_' . $watch->getLevelName()) }}
</div>
</div>
@endforeach
</div>
@endif
<div class="my-m">{{ $watches->links() }}</div>
</section>
</div> </div>
@stop @stop