mirror of
				https://github.com/BookStackApp/BookStack.git
				synced 2025-11-03 02:13:16 +03:00 
			
		
		
		
	* Started mass-refactoring of the current entity repos * Rewrote book tree logic - Now does two simple queries instead of one really complex one. - Extracted logic into its own class. - Remove model-level akward union field listing. - Logic now more readable than being large separate query and compilation functions. * Extracted and split book sort logic * Finished up Book controller/repo organisation * Refactored bookshelves controllers and repo parts * Fixed issues found via phpunit * Refactored Chapter controller * Updated Chapter export controller * Started Page controller/repo refactor * Refactored another chunk of PageController * Completed initial pagecontroller refactor pass * Fixed tests and continued reduction of old repos * Removed old page remove and further reduced entity repo * Removed old entity repo, split out page controller * Ran phpcbf and split out some page content methods * Tidied up some EntityProvider elements * Fixed issued caused by viewservice change
		
			
				
	
	
		
			187 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			187 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php namespace BookStack\Actions;
 | 
						|
 | 
						|
use BookStack\Auth\Permissions\PermissionService;
 | 
						|
use BookStack\Entities\Book;
 | 
						|
use BookStack\Entities\Entity;
 | 
						|
 | 
						|
class ActivityService
 | 
						|
{
 | 
						|
    protected $activity;
 | 
						|
    protected $user;
 | 
						|
    protected $permissionService;
 | 
						|
 | 
						|
    /**
 | 
						|
     * ActivityService constructor.
 | 
						|
     * @param Activity $activity
 | 
						|
     * @param PermissionService $permissionService
 | 
						|
     */
 | 
						|
    public function __construct(Activity $activity, PermissionService $permissionService)
 | 
						|
    {
 | 
						|
        $this->activity = $activity;
 | 
						|
        $this->permissionService = $permissionService;
 | 
						|
        $this->user = user();
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Add activity data to database.
 | 
						|
     * @param \BookStack\Entities\Entity $entity
 | 
						|
     * @param string $activityKey
 | 
						|
     * @param int $bookId
 | 
						|
     */
 | 
						|
    public function add(Entity $entity, string $activityKey, int $bookId = null)
 | 
						|
    {
 | 
						|
        $activity = $this->newActivityForUser($activityKey, $bookId);
 | 
						|
        $entity->activity()->save($activity);
 | 
						|
        $this->setNotification($activityKey);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Adds a activity history with a message, without binding to a entity.
 | 
						|
     * @param string $activityKey
 | 
						|
     * @param string $message
 | 
						|
     * @param int $bookId
 | 
						|
     */
 | 
						|
    public function addMessage(string $activityKey, string $message, int $bookId = null)
 | 
						|
    {
 | 
						|
        $this->newActivityForUser($activityKey, $bookId)->forceFill([
 | 
						|
            'extra' => $message
 | 
						|
        ])->save();
 | 
						|
 | 
						|
        $this->setNotification($activityKey);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Get a new activity instance for the current user.
 | 
						|
     * @param string $key
 | 
						|
     * @param int|null $bookId
 | 
						|
     * @return Activity
 | 
						|
     */
 | 
						|
    protected function newActivityForUser(string $key, int $bookId = null)
 | 
						|
    {
 | 
						|
        return $this->activity->newInstance()->forceFill([
 | 
						|
            'key' => strtolower($key),
 | 
						|
            'user_id' => $this->user->id,
 | 
						|
            'book_id' => $bookId ?? 0,
 | 
						|
        ]);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Removes the entity attachment from each of its activities
 | 
						|
     * and instead uses the 'extra' field with the entities name.
 | 
						|
     * Used when an entity is deleted.
 | 
						|
     * @param \BookStack\Entities\Entity $entity
 | 
						|
     * @return mixed
 | 
						|
     */
 | 
						|
    public function removeEntity(Entity $entity)
 | 
						|
    {
 | 
						|
        // TODO - Rewrite to db query.
 | 
						|
        $activities = $entity->activity;
 | 
						|
        foreach ($activities as $activity) {
 | 
						|
            $activity->extra = $entity->name;
 | 
						|
            $activity->entity_id = 0;
 | 
						|
            $activity->entity_type = null;
 | 
						|
            $activity->save();
 | 
						|
        }
 | 
						|
        return $activities;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Gets the latest activity.
 | 
						|
     * @param int $count
 | 
						|
     * @param int $page
 | 
						|
     * @return array
 | 
						|
     */
 | 
						|
    public function latest($count = 20, $page = 0)
 | 
						|
    {
 | 
						|
        $activityList = $this->permissionService
 | 
						|
            ->filterRestrictedEntityRelations($this->activity, 'activities', 'entity_id', 'entity_type')
 | 
						|
            ->orderBy('created_at', 'desc')
 | 
						|
            ->with('user', 'entity')
 | 
						|
            ->skip($count * $page)
 | 
						|
            ->take($count)
 | 
						|
            ->get();
 | 
						|
 | 
						|
        return $this->filterSimilar($activityList);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Gets the latest activity for an entity, Filtering out similar
 | 
						|
     * items to prevent a message activity list.
 | 
						|
     * @param \BookStack\Entities\Entity $entity
 | 
						|
     * @param int $count
 | 
						|
     * @param int $page
 | 
						|
     * @return array
 | 
						|
     */
 | 
						|
    public function entityActivity($entity, $count = 20, $page = 1)
 | 
						|
    {
 | 
						|
        if ($entity->isA('book')) {
 | 
						|
            $query = $this->activity->where('book_id', '=', $entity->id);
 | 
						|
        } else {
 | 
						|
            $query = $this->activity->where('entity_type', '=', $entity->getMorphClass())
 | 
						|
                ->where('entity_id', '=', $entity->id);
 | 
						|
        }
 | 
						|
        
 | 
						|
        $activity = $this->permissionService
 | 
						|
            ->filterRestrictedEntityRelations($query, 'activities', 'entity_id', 'entity_type')
 | 
						|
            ->orderBy('created_at', 'desc')
 | 
						|
            ->with(['entity', 'user.avatar'])
 | 
						|
            ->skip($count * ($page - 1))
 | 
						|
            ->take($count)
 | 
						|
            ->get();
 | 
						|
 | 
						|
        return $this->filterSimilar($activity);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Get latest activity for a user, Filtering out similar
 | 
						|
     * items.
 | 
						|
     * @param $user
 | 
						|
     * @param int $count
 | 
						|
     * @param int $page
 | 
						|
     * @return array
 | 
						|
     */
 | 
						|
    public function userActivity($user, $count = 20, $page = 0)
 | 
						|
    {
 | 
						|
        $activityList = $this->permissionService
 | 
						|
            ->filterRestrictedEntityRelations($this->activity, 'activities', 'entity_id', 'entity_type')
 | 
						|
            ->orderBy('created_at', 'desc')->where('user_id', '=', $user->id)->skip($count * $page)->take($count)->get();
 | 
						|
        return $this->filterSimilar($activityList);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Filters out similar activity.
 | 
						|
     * @param Activity[] $activities
 | 
						|
     * @return array
 | 
						|
     */
 | 
						|
    protected function filterSimilar($activities)
 | 
						|
    {
 | 
						|
        $newActivity = [];
 | 
						|
        $previousItem = false;
 | 
						|
        foreach ($activities as $activityItem) {
 | 
						|
            if ($previousItem === false) {
 | 
						|
                $previousItem = $activityItem;
 | 
						|
                $newActivity[] = $activityItem;
 | 
						|
                continue;
 | 
						|
            }
 | 
						|
            if (!$activityItem->isSimilarTo($previousItem)) {
 | 
						|
                $newActivity[] = $activityItem;
 | 
						|
            }
 | 
						|
            $previousItem = $activityItem;
 | 
						|
        }
 | 
						|
        return $newActivity;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Flashes a notification message to the session if an appropriate message is available.
 | 
						|
     * @param $activityKey
 | 
						|
     */
 | 
						|
    protected function setNotification($activityKey)
 | 
						|
    {
 | 
						|
        $notificationTextKey = 'activities.' . $activityKey . '_notification';
 | 
						|
        if (trans()->has($notificationTextKey)) {
 | 
						|
            $message = trans($notificationTextKey);
 | 
						|
            session()->flash('success', $message);
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |