assets = $assets; $this->pathResolver = $pathResolver; $this->urlGenerator = $urlGenerator; $this->attachmentHelper = $attachmentHelper; $this->filterService = $filterService; //Determine a normalized path to the public folder (assets are relative to this folder) $this->public_path = $this->pathResolver->parameterToAbsolutePath('public'); } /** * Converts the absolute file path to a version relative to the public folder, that can be passed to asset * Asset Component functions. * * @param string $absolute_path the absolute path that should be converted * @param string|null $public_path The public path to which the relative pathes should be created. * The path must NOT have a trailing slash! * If this is set to null, the global public/ folder is used. * * @return string|null The relative version of the string. Null if the absolute path was not a child folder * of public path */ public function absolutePathToAssetPath(string $absolute_path, ?string $public_path = null): ?string { if (null === $public_path) { $public_path = $this->public_path; } //Our absolute path must begin with public path or we can not use it for asset pathes. if (0 !== strpos($absolute_path, $public_path)) { return null; } //Return the part relative after public path. return substr($absolute_path, strlen($public_path) + 1); } /** * Returns a URL under which the attachment file can be viewed. */ public function getViewURL(Attachment $attachment): string { $absolute_path = $this->attachmentHelper->toAbsoluteFilePath($attachment); if (null === $absolute_path) { throw new RuntimeException('The given attachment is external or has no valid file, so no URL can get generated for it! Use Attachment::getURL() to get the external URL!'); } $asset_path = $this->absolutePathToAssetPath($absolute_path); //If path is not relative to public path or marked as secure, serve it via controller if (null === $asset_path || $attachment->isSecure()) { return $this->urlGenerator->generate('attachment_view', ['id' => $attachment->getID()]); } //Otherwise we can serve the relative path via Asset component return $this->assets->getUrl($asset_path); } /** * Returns a URL to an thumbnail of the attachment file. */ public function getThumbnailURL(Attachment $attachment, string $filter_name = 'thumbnail_sm'): string { if (! $attachment->isPicture()) { throw new InvalidArgumentException('Thumbnail creation only works for picture attachments!'); } if ($attachment->isExternal()) { return $attachment->getURL(); } $absolute_path = $this->attachmentHelper->toAbsoluteFilePath($attachment); if (null === $absolute_path) { throw new RuntimeException('The given attachment is external or has no valid file, so no URL can get generated for it!'); } $asset_path = $this->absolutePathToAssetPath($absolute_path); //If path is not relative to public path or marked as secure, serve it via controller if (null === $asset_path || $attachment->isSecure()) { return $this->urlGenerator->generate('attachment_view', ['id' => $attachment->getID()]); } //For builtin ressources it is not useful to create a thumbnail //because the footprints images are small and highly optimized already. if ('thumbnail_md' === $filter_name && $attachment->isBuiltIn()) { return $this->assets->getUrl($asset_path); } //Otherwise we can serve the relative path via Asset component return $this->filterService->getUrlOfFilteredImage($asset_path, $filter_name); } /** * Returns a download link to the file associated with the attachment. */ public function getDownloadURL(Attachment $attachment): string { //Redirect always to download controller, which sets the correct headers for downloading: return $this->urlGenerator->generate('attachment_download', ['id' => $attachment->getID()]); } }