3 namespace BookStack\Entities\Repos;
5 use BookStack\Activity\ActivityType;
6 use BookStack\Entities\Models\Book;
7 use BookStack\Entities\Models\Chapter;
8 use BookStack\Entities\Queries\EntityQueries;
9 use BookStack\Entities\Tools\BookContents;
10 use BookStack\Entities\Tools\ParentChanger;
11 use BookStack\Entities\Tools\TrashCan;
12 use BookStack\Exceptions\MoveOperationException;
13 use BookStack\Exceptions\PermissionsException;
14 use BookStack\Facades\Activity;
15 use BookStack\Permissions\Permission;
16 use BookStack\Util\DatabaseTransaction;
21 public function __construct(
22 protected BaseRepo $baseRepo,
23 protected EntityQueries $entityQueries,
24 protected TrashCan $trashCan,
25 protected ParentChanger $parentChanger,
30 * Create a new chapter in the system.
32 public function create(array $input, Book $parentBook): Chapter
34 return (new DatabaseTransaction(function () use ($input, $parentBook) {
35 $chapter = new Chapter();
36 $chapter->book_id = $parentBook->id;
37 $chapter->priority = (new BookContents($parentBook))->getLastPriority() + 1;
39 $chapter = $this->baseRepo->create($chapter, $input);
40 $chapter->defaultTemplate()->setFromId(intval($input['default_template_id'] ?? null));
43 Activity::add(ActivityType::CHAPTER_CREATE, $chapter);
45 $this->baseRepo->sortParent($chapter);
52 * Update the given chapter.
54 public function update(Chapter $chapter, array $input): Chapter
56 $chapter = $this->baseRepo->update($chapter, $input);
58 if (array_key_exists('default_template_id', $input)) {
59 $chapter->defaultTemplate()->setFromId(intval($input['default_template_id']));
63 Activity::add(ActivityType::CHAPTER_UPDATE, $chapter);
65 $this->baseRepo->sortParent($chapter);
71 * Remove a chapter from the system.
75 public function destroy(Chapter $chapter): void
77 $this->trashCan->softDestroyChapter($chapter);
78 Activity::add(ActivityType::CHAPTER_DELETE, $chapter);
79 $this->trashCan->autoClearOld();
83 * Move the given chapter into a new parent book.
84 * The $parentIdentifier must be a string of the following format:
85 * 'book:<id>' (book:5).
87 * @throws MoveOperationException
88 * @throws PermissionsException
90 public function move(Chapter $chapter, string $parentIdentifier): Book
92 $parent = $this->entityQueries->findVisibleByStringIdentifier($parentIdentifier);
93 if (!$parent instanceof Book) {
94 throw new MoveOperationException('Book to move chapter into not found');
97 if (!userCan(Permission::ChapterCreate, $parent)) {
98 throw new PermissionsException('User does not have permission to create a chapter within the chosen book');
101 return (new DatabaseTransaction(function () use ($chapter, $parent) {
102 $this->parentChanger->changeBook($chapter, $parent->id);
103 $chapter->rebuildPermissions();
104 Activity::add(ActivityType::CHAPTER_MOVE, $chapter);
106 $this->baseRepo->sortParent($chapter);