1 <?php namespace BookStack\Actions;
3 use BookStack\Auth\Permissions\PermissionService;
4 use BookStack\Entities\Models\Book;
5 use BookStack\Entities\Models\Entity;
6 use BookStack\Entities\EntityProvider;
8 use Illuminate\Support\Collection;
13 protected $permissionService;
14 protected $entityProvider;
17 * ViewService constructor.
19 * @param PermissionService $permissionService
20 * @param EntityProvider $entityProvider
22 public function __construct(View $view, PermissionService $permissionService, EntityProvider $entityProvider)
25 $this->permissionService = $permissionService;
26 $this->entityProvider = $entityProvider;
30 * Add a view to the given entity.
31 * @param \BookStack\Entities\Models\Entity $entity
34 public function add(Entity $entity)
37 if ($user === null || $user->isDefault()) {
40 $view = $entity->views()->where('user_id', '=', $user->id)->first();
41 // Add view if model exists
43 $view->increment('views');
47 // Otherwise create new view count
48 $entity->views()->save($this->view->newInstance([
49 'user_id' => $user->id,
57 * Get the entities with the most views.
60 * @param string|array $filterModels
61 * @param string $action - used for permission checking
64 public function getPopular(int $count = 10, int $page = 0, array $filterModels = null, string $action = 'view')
66 $skipCount = $count * $page;
67 $query = $this->permissionService
68 ->filterRestrictedEntityRelations($this->view, 'views', 'viewable_id', 'viewable_type', $action)
69 ->select('*', 'viewable_id', 'viewable_type', DB::raw('SUM(views) as view_count'))
70 ->groupBy('viewable_id', 'viewable_type')
71 ->orderBy('view_count', 'desc');
74 $query->whereIn('viewable_type', $this->entityProvider->getMorphClasses($filterModels));
77 return $query->with('viewable')
86 * Get all recently viewed entities for the current user.
88 public function getUserRecentlyViewed(int $count = 10, int $page = 1)
91 if ($user === null || $user->isDefault()) {
96 /** @var Entity $instance */
97 foreach ($this->entityProvider->all() as $name => $instance) {
98 $items = $instance::visible()->withLastView()
99 ->orderBy('last_viewed_at', 'desc')
100 ->skip($count * ($page - 1))
103 $all = $all->concat($items);
106 return $all->sortByDesc('last_viewed_at')->slice(0, $count);
110 * Reset all view counts by deleting all views.
112 public function resetAll()
114 $this->view->truncate();