3 namespace BookStack\Entities\Repos;
5 use BookStack\Activity\TagRepo;
6 use BookStack\Entities\Models\Book;
7 use BookStack\Entities\Models\BookChild;
8 use BookStack\Entities\Models\Chapter;
9 use BookStack\Entities\Models\Entity;
10 use BookStack\Entities\Models\CoverImageInterface;
11 use BookStack\Entities\Models\HtmlDescriptionInterface;
12 use BookStack\Entities\Models\HtmlDescriptionTrait;
13 use BookStack\Entities\Queries\PageQueries;
14 use BookStack\Exceptions\ImageUploadException;
15 use BookStack\References\ReferenceStore;
16 use BookStack\References\ReferenceUpdater;
17 use BookStack\Sorting\BookSorter;
18 use BookStack\Uploads\ImageRepo;
19 use BookStack\Util\HtmlDescriptionFilter;
20 use Illuminate\Http\UploadedFile;
24 public function __construct(
25 protected TagRepo $tagRepo,
26 protected ImageRepo $imageRepo,
27 protected ReferenceUpdater $referenceUpdater,
28 protected ReferenceStore $referenceStore,
29 protected PageQueries $pageQueries,
30 protected BookSorter $bookSorter,
35 * Create a new entity in the system.
37 public function create(Entity $entity, array $input)
39 $entity->fill($input);
40 $this->updateDescription($entity, $input);
42 'created_by' => user()->id,
43 'updated_by' => user()->id,
44 'owned_by' => user()->id,
46 $entity->refreshSlug();
49 if (isset($input['tags'])) {
50 $this->tagRepo->saveTagsToEntity($entity, $input['tags']);
54 $entity->rebuildPermissions();
55 $entity->indexForSearch();
56 $this->referenceStore->updateForEntity($entity);
60 * Update the given entity.
62 public function update(Entity $entity, array $input)
64 $oldUrl = $entity->getUrl();
66 $entity->fill($input);
67 $this->updateDescription($entity, $input);
68 $entity->updated_by = user()->id;
70 if ($entity->isDirty('name') || empty($entity->slug)) {
71 $entity->refreshSlug();
76 if (isset($input['tags'])) {
77 $this->tagRepo->saveTagsToEntity($entity, $input['tags']);
81 $entity->indexForSearch();
82 $this->referenceStore->updateForEntity($entity);
84 if ($oldUrl !== $entity->getUrl()) {
85 $this->referenceUpdater->updateEntityReferences($entity, $oldUrl);
90 * Update the given items' cover image, or clear it.
92 * @param Entity&CoverImageInterface $entity
94 * @throws ImageUploadException
97 public function updateCoverImage($entity, ?UploadedFile $coverImage, bool $removeImage = false)
100 $imageType = $entity->coverImageTypeKey();
101 $this->imageRepo->destroyImage($entity->cover()->first());
102 $image = $this->imageRepo->saveNew($coverImage, $imageType, $entity->id, 512, 512, true);
103 $entity->cover()->associate($image);
108 $this->imageRepo->destroyImage($entity->cover()->first());
109 $entity->image_id = 0;
115 * Update the default page template used for this item.
116 * Checks that, if changing, the provided value is a valid template and the user
117 * has visibility of the provided page template id.
119 public function updateDefaultTemplate(Book|Chapter $entity, int $templateId): void
121 $changing = $templateId !== intval($entity->default_template_id);
126 if ($templateId === 0) {
127 $entity->default_template_id = null;
132 $templateExists = $this->pageQueries->visibleTemplates()
133 ->where('id', '=', $templateId)
136 $entity->default_template_id = $templateExists ? $templateId : null;
141 * Sort the parent of the given entity, if any auto sort actions are set for it.
142 * Typically ran during create/update/insert events.
144 public function sortParent(Entity $entity): void
146 if ($entity instanceof BookChild) {
147 $book = $entity->book;
148 $this->bookSorter->runBookAutoSort($book);
152 protected function updateDescription(Entity $entity, array $input): void
154 if (!($entity instanceof HtmlDescriptionInterface)) {
158 if (isset($input['description_html'])) {
159 $entity->setDescriptionHtml(
160 HtmlDescriptionFilter::filterFromString($input['description_html']),
161 html_entity_decode(strip_tags($input['description_html']))
163 } else if (isset($input['description'])) {
164 $entity->setDescriptionHtml('', $input['description']);