Merge branch 'drawing_updates'

This commit is contained in:
Dan Brown 2018-05-27 19:42:25 +01:00
commit 6b84a76af1
No known key found for this signature in database
GPG Key ID: 46D9F943C24A2EF9
35 changed files with 538 additions and 223 deletions

View File

@ -0,0 +1,83 @@
<?php
namespace BookStack\Console\Commands;
use BookStack\Services\ImageService;
use Illuminate\Console\Command;
use Symfony\Component\Console\Output\OutputInterface;
class CleanupImages extends Command
{
/**
* The name and signature of the console command.
*
* @var string
*/
protected $signature = 'bookstack:cleanup-images
{--a|all : Include images that are used in page revisions}
{--f|force : Actually run the deletions}
';
/**
* The console command description.
*
* @var string
*/
protected $description = 'Cleanup images and drawings';
protected $imageService;
/**
* Create a new command instance.
* @param ImageService $imageService
*/
public function __construct(ImageService $imageService)
{
$this->imageService = $imageService;
parent::__construct();
}
/**
* Execute the console command.
*
* @return mixed
*/
public function handle()
{
$checkRevisions = $this->option('all') ? false : true;
$dryRun = $this->option('force') ? false : true;
if (!$dryRun) {
$proceed = $this->confirm("This operation is destructive and is not guaranteed to be fully accurate.\nEnsure you have a backup of your images.\nAre you sure you want to proceed?");
if (!$proceed) {
return;
}
}
$deleted = $this->imageService->deleteUnusedImages($checkRevisions, $dryRun);
$deleteCount = count($deleted);
if ($dryRun) {
$this->comment('Dry run, No images have been deleted');
$this->comment($deleteCount . ' images found that would have been deleted');
$this->showDeletedImages($deleted);
$this->comment('Run with -f or --force to perform deletions');
return;
}
$this->showDeletedImages($deleted);
$this->comment($deleteCount . ' images deleted');
}
protected function showDeletedImages($paths)
{
if ($this->getOutput()->getVerbosity() <= OutputInterface::VERBOSITY_NORMAL) return;
if (count($paths) > 0) {
$this->line('Images to delete:');
}
foreach ($paths as $path) {
$this->line($path);
}
}
}

View File

@ -164,32 +164,6 @@ class ImageController extends Controller
return response()->json($image); return response()->json($image);
} }
/**
* Replace the data content of a drawing.
* @param string $id
* @param Request $request
* @return \Illuminate\Contracts\Routing\ResponseFactory|\Illuminate\Http\JsonResponse|\Symfony\Component\HttpFoundation\Response
*/
public function replaceDrawing(string $id, Request $request)
{
$this->validate($request, [
'image' => 'required|string'
]);
$this->checkPermission('image-create-all');
$imageBase64Data = $request->get('image');
$image = $this->imageRepo->getById($id);
$this->checkOwnablePermission('image-update', $image);
try {
$image = $this->imageRepo->replaceDrawingContent($image, $imageBase64Data);
} catch (ImageUploadException $e) {
return response($e->getMessage(), 500);
}
return response()->json($image);
}
/** /**
* Get the content of an image based64 encoded. * Get the content of an image based64 encoded.
* @param $id * @param $id
@ -245,26 +219,29 @@ class ImageController extends Controller
} }
/** /**
* Deletes an image and all thumbnail/image files * Show the usage of an image on pages.
* @param EntityRepo $entityRepo * @param EntityRepo $entityRepo
* @param Request $request * @param $id
* @param int $id
* @return \Illuminate\Http\JsonResponse * @return \Illuminate\Http\JsonResponse
*/ */
public function destroy(EntityRepo $entityRepo, Request $request, $id) public function usage(EntityRepo $entityRepo, $id)
{
$image = $this->imageRepo->getById($id);
$pageSearch = $entityRepo->searchForImage($image->url);
return response()->json($pageSearch);
}
/**
* Deletes an image and all thumbnail/image files
* @param int $id
* @return \Illuminate\Http\JsonResponse
* @throws \Exception
*/
public function destroy($id)
{ {
$image = $this->imageRepo->getById($id); $image = $this->imageRepo->getById($id);
$this->checkOwnablePermission('image-delete', $image); $this->checkOwnablePermission('image-delete', $image);
// Check if this image is used on any pages
$isForced = in_array($request->get('force', ''), [true, 'true']);
if (!$isForced) {
$pageSearch = $entityRepo->searchForImage($image->url);
if ($pageSearch !== false) {
return response()->json($pageSearch, 400);
}
}
$this->imageRepo->destroyImage($image); $this->imageRepo->destroyImage($image);
return response()->json(trans('components.images_deleted')); return response()->json(trans('components.images_deleted'));
} }

View File

@ -1,5 +1,6 @@
<?php namespace BookStack\Http\Controllers; <?php namespace BookStack\Http\Controllers;
use BookStack\Services\ImageService;
use Illuminate\Http\Request; use Illuminate\Http\Request;
use Illuminate\Http\Response; use Illuminate\Http\Response;
use Setting; use Setting;
@ -13,7 +14,7 @@ class SettingController extends Controller
public function index() public function index()
{ {
$this->checkPermission('settings-manage'); $this->checkPermission('settings-manage');
$this->setPageTitle('Settings'); $this->setPageTitle(trans('settings.settings'));
// Get application version // Get application version
$version = trim(file_get_contents(base_path('version'))); $version = trim(file_get_contents(base_path('version')));
@ -43,4 +44,48 @@ class SettingController extends Controller
session()->flash('success', trans('settings.settings_save_success')); session()->flash('success', trans('settings.settings_save_success'));
return redirect('/settings'); return redirect('/settings');
} }
/**
* Show the page for application maintenance.
* @return \Illuminate\Contracts\View\Factory|\Illuminate\View\View
*/
public function showMaintenance()
{
$this->checkPermission('settings-manage');
$this->setPageTitle(trans('settings.maint'));
// Get application version
$version = trim(file_get_contents(base_path('version')));
return view('settings/maintenance', ['version' => $version]);
}
/**
* Action to clean-up images in the system.
* @param Request $request
* @param ImageService $imageService
* @return \Illuminate\Http\RedirectResponse|\Illuminate\Routing\Redirector
*/
public function cleanupImages(Request $request, ImageService $imageService)
{
$this->checkPermission('settings-manage');
$checkRevisions = !($request->get('ignore_revisions', 'false') === 'true');
$dryRun = !($request->has('confirm'));
$imagesToDelete = $imageService->deleteUnusedImages($checkRevisions, $dryRun);
$deleteCount = count($imagesToDelete);
if ($deleteCount === 0) {
session()->flash('warning', trans('settings.maint_image_cleanup_nothing_found'));
return redirect('/settings/maintenance')->withInput();
}
if ($dryRun) {
session()->flash('cleanup-images-warning', trans('settings.maint_image_cleanup_warning', ['count' => $deleteCount]));
} else {
session()->flash('success', trans('settings.maint_image_cleanup_success', ['count' => $deleteCount]));
}
return redirect('/settings/maintenance#image-cleanup')->withInput();
}
} }

View File

@ -9,13 +9,15 @@ class Image extends Ownable
/** /**
* Get a thumbnail for this image. * Get a thumbnail for this image.
* @param int $width * @param int $width
* @param int $height * @param int $height
* @param bool|false $keepRatio * @param bool|false $keepRatio
* @return string * @return string
* @throws \Exception
*/ */
public function getThumb($width, $height, $keepRatio = false) public function getThumb($width, $height, $keepRatio = false)
{ {
return Images::getThumbnail($this, $width, $height, $keepRatio); return Images::getThumbnail($this, $width, $height, $keepRatio);
} }
} }

View File

@ -3,6 +3,7 @@
namespace BookStack\Providers; namespace BookStack\Providers;
use BookStack\Activity; use BookStack\Activity;
use BookStack\Image;
use BookStack\Services\ImageService; use BookStack\Services\ImageService;
use BookStack\Services\PermissionService; use BookStack\Services\PermissionService;
use BookStack\Services\ViewService; use BookStack\Services\ViewService;
@ -57,6 +58,7 @@ class CustomFacadeProvider extends ServiceProvider
$this->app->bind('images', function () { $this->app->bind('images', function () {
return new ImageService( return new ImageService(
$this->app->make(Image::class),
$this->app->make(ImageManager::class), $this->app->make(ImageManager::class),
$this->app->make(Factory::class), $this->app->make(Factory::class),
$this->app->make(Repository::class) $this->app->make(Repository::class)

View File

@ -153,17 +153,6 @@ class ImageRepo
return $image; return $image;
} }
/**
* Replace the image content of a drawing.
* @param Image $image
* @param string $base64Uri
* @return Image
* @throws \BookStack\Exceptions\ImageUploadException
*/
public function replaceDrawingContent(Image $image, string $base64Uri)
{
return $this->imageService->replaceImageDataFromBase64Uri($image, $base64Uri);
}
/** /**
* Update the details of an image via an array of properties. * Update the details of an image via an array of properties.
@ -183,13 +172,14 @@ class ImageRepo
/** /**
* Destroys an Image object along with its files and thumbnails. * Destroys an Image object along with its revisions, files and thumbnails.
* @param Image $image * @param Image $image
* @return bool * @return bool
* @throws \Exception
*/ */
public function destroyImage(Image $image) public function destroyImage(Image $image)
{ {
$this->imageService->destroyImage($image); $this->imageService->destroy($image);
return true; return true;
} }
@ -200,7 +190,7 @@ class ImageRepo
* @throws \BookStack\Exceptions\ImageUploadException * @throws \BookStack\Exceptions\ImageUploadException
* @throws \Exception * @throws \Exception
*/ */
private function loadThumbs(Image $image) protected function loadThumbs(Image $image)
{ {
$image->thumbs = [ $image->thumbs = [
'gallery' => $this->getThumbnail($image, 150, 150), 'gallery' => $this->getThumbnail($image, 150, 150),
@ -250,7 +240,7 @@ class ImageRepo
*/ */
public function isValidType($type) public function isValidType($type)
{ {
$validTypes = ['drawing', 'gallery', 'cover', 'system', 'user']; $validTypes = ['gallery', 'cover', 'system', 'user'];
return in_array($type, $validTypes); return in_array($type, $validTypes);
} }
} }

View File

@ -166,7 +166,7 @@ class UserRepo
// Delete user profile images // Delete user profile images
$profileImages = $images = Image::where('type', '=', 'user')->where('created_by', '=', $user->id)->get(); $profileImages = $images = Image::where('type', '=', 'user')->where('created_by', '=', $user->id)->get();
foreach ($profileImages as $image) { foreach ($profileImages as $image) {
Images::destroyImage($image); Images::destroy($image);
} }
} }

View File

@ -3,11 +3,11 @@
use BookStack\Exceptions\ImageUploadException; use BookStack\Exceptions\ImageUploadException;
use BookStack\Image; use BookStack\Image;
use BookStack\User; use BookStack\User;
use DB;
use Exception; use Exception;
use Intervention\Image\Exception\NotSupportedException; use Intervention\Image\Exception\NotSupportedException;
use Intervention\Image\ImageManager; use Intervention\Image\ImageManager;
use Illuminate\Contracts\Filesystem\Factory as FileSystem; use Illuminate\Contracts\Filesystem\Factory as FileSystem;
use Illuminate\Contracts\Filesystem\Filesystem as FileSystemInstance;
use Illuminate\Contracts\Cache\Repository as Cache; use Illuminate\Contracts\Cache\Repository as Cache;
use Symfony\Component\HttpFoundation\File\UploadedFile; use Symfony\Component\HttpFoundation\File\UploadedFile;
@ -17,15 +17,18 @@ class ImageService extends UploadService
protected $imageTool; protected $imageTool;
protected $cache; protected $cache;
protected $storageUrl; protected $storageUrl;
protected $image;
/** /**
* ImageService constructor. * ImageService constructor.
* @param $imageTool * @param Image $image
* @param $fileSystem * @param ImageManager $imageTool
* @param $cache * @param FileSystem $fileSystem
* @param Cache $cache
*/ */
public function __construct(ImageManager $imageTool, FileSystem $fileSystem, Cache $cache) public function __construct(Image $image, ImageManager $imageTool, FileSystem $fileSystem, Cache $cache)
{ {
$this->image = $image;
$this->imageTool = $imageTool; $this->imageTool = $imageTool;
$this->cache = $cache; $this->cache = $cache;
parent::__construct($fileSystem); parent::__construct($fileSystem);
@ -82,31 +85,6 @@ class ImageService extends UploadService
return $this->saveNew($name, $data, $type, $uploadedTo); return $this->saveNew($name, $data, $type, $uploadedTo);
} }
/**
* Replace the data for an image via a Base64 encoded string.
* @param Image $image
* @param string $base64Uri
* @return Image
* @throws ImageUploadException
*/
public function replaceImageDataFromBase64Uri(Image $image, string $base64Uri)
{
$splitData = explode(';base64,', $base64Uri);
if (count($splitData) < 2) {
throw new ImageUploadException("Invalid base64 image data provided");
}
$data = base64_decode($splitData[1]);
$storage = $this->getStorage();
try {
$storage->put($image->path, $data);
} catch (Exception $e) {
throw new ImageUploadException(trans('errors.path_not_writable', ['filePath' => $image->path]));
}
return $image;
}
/** /**
* Gets an image from url and saves it to the database. * Gets an image from url and saves it to the database.
* @param $url * @param $url
@ -140,16 +118,16 @@ class ImageService extends UploadService
$secureUploads = setting('app-secure-images'); $secureUploads = setting('app-secure-images');
$imageName = str_replace(' ', '-', $imageName); $imageName = str_replace(' ', '-', $imageName);
if ($secureUploads) {
$imageName = str_random(16) . '-' . $imageName;
}
$imagePath = '/uploads/images/' . $type . '/' . Date('Y-m-M') . '/'; $imagePath = '/uploads/images/' . $type . '/' . Date('Y-m-M') . '/';
while ($storage->exists($imagePath . $imageName)) { while ($storage->exists($imagePath . $imageName)) {
$imageName = str_random(3) . $imageName; $imageName = str_random(3) . $imageName;
} }
$fullPath = $imagePath . $imageName; $fullPath = $imagePath . $imageName;
if ($secureUploads) {
$fullPath = $imagePath . str_random(16) . '-' . $imageName;
}
try { try {
$storage->put($fullPath, $imageData); $storage->put($fullPath, $imageData);
@ -172,20 +150,11 @@ class ImageService extends UploadService
$imageDetails['updated_by'] = $userId; $imageDetails['updated_by'] = $userId;
} }
$image = (new Image()); $image = $this->image->newInstance();
$image->forceFill($imageDetails)->save(); $image->forceFill($imageDetails)->save();
return $image; return $image;
} }
/**
* Get the storage path, Dependant of storage type.
* @param Image $image
* @return mixed|string
*/
protected function getPath(Image $image)
{
return $image->path;
}
/** /**
* Checks if the image is a gif. Returns true if it is, else false. * Checks if the image is a gif. Returns true if it is, else false.
@ -194,7 +163,7 @@ class ImageService extends UploadService
*/ */
protected function isGif(Image $image) protected function isGif(Image $image)
{ {
return strtolower(pathinfo($this->getPath($image), PATHINFO_EXTENSION)) === 'gif'; return strtolower(pathinfo($image->path, PATHINFO_EXTENSION)) === 'gif';
} }
/** /**
@ -212,11 +181,11 @@ class ImageService extends UploadService
public function getThumbnail(Image $image, $width = 220, $height = 220, $keepRatio = false) public function getThumbnail(Image $image, $width = 220, $height = 220, $keepRatio = false)
{ {
if ($keepRatio && $this->isGif($image)) { if ($keepRatio && $this->isGif($image)) {
return $this->getPublicUrl($this->getPath($image)); return $this->getPublicUrl($image->path);
} }
$thumbDirName = '/' . ($keepRatio ? 'scaled-' : 'thumbs-') . $width . '-' . $height . '/'; $thumbDirName = '/' . ($keepRatio ? 'scaled-' : 'thumbs-') . $width . '-' . $height . '/';
$imagePath = $this->getPath($image); $imagePath = $image->path;
$thumbFilePath = dirname($imagePath) . $thumbDirName . basename($imagePath); $thumbFilePath = dirname($imagePath) . $thumbDirName . basename($imagePath);
if ($this->cache->has('images-' . $image->id . '-' . $thumbFilePath) && $this->cache->get('images-' . $thumbFilePath)) { if ($this->cache->has('images-' . $image->id . '-' . $thumbFilePath) && $this->cache->get('images-' . $thumbFilePath)) {
@ -262,43 +231,51 @@ class ImageService extends UploadService
*/ */
public function getImageData(Image $image) public function getImageData(Image $image)
{ {
$imagePath = $this->getPath($image); $imagePath = $image->path;
$storage = $this->getStorage(); $storage = $this->getStorage();
return $storage->get($imagePath); return $storage->get($imagePath);
} }
/** /**
* Destroys an Image object along with its files and thumbnails. * Destroy an image along with its revisions, thumbnails and remaining folders.
* @param Image $image * @param Image $image
* @return bool
* @throws Exception * @throws Exception
*/ */
public function destroyImage(Image $image) public function destroy(Image $image)
{
$this->destroyImagesFromPath($image->path);
$image->delete();
}
/**
* Destroys an image at the given path.
* Searches for image thumbnails in addition to main provided path..
* @param string $path
* @return bool
*/
protected function destroyImagesFromPath(string $path)
{ {
$storage = $this->getStorage(); $storage = $this->getStorage();
$imageFolder = dirname($this->getPath($image)); $imageFolder = dirname($path);
$imageFileName = basename($this->getPath($image)); $imageFileName = basename($path);
$allImages = collect($storage->allFiles($imageFolder)); $allImages = collect($storage->allFiles($imageFolder));
// Delete image files
$imagesToDelete = $allImages->filter(function ($imagePath) use ($imageFileName) { $imagesToDelete = $allImages->filter(function ($imagePath) use ($imageFileName) {
$expectedIndex = strlen($imagePath) - strlen($imageFileName); $expectedIndex = strlen($imagePath) - strlen($imageFileName);
return strpos($imagePath, $imageFileName) === $expectedIndex; return strpos($imagePath, $imageFileName) === $expectedIndex;
}); });
$storage->delete($imagesToDelete->all()); $storage->delete($imagesToDelete->all());
// Cleanup of empty folders // Cleanup of empty folders
foreach ($storage->directories($imageFolder) as $directory) { $foldersInvolved = array_merge([$imageFolder], $storage->directories($imageFolder));
foreach ($foldersInvolved as $directory) {
if ($this->isFolderEmpty($directory)) { if ($this->isFolderEmpty($directory)) {
$storage->deleteDirectory($directory); $storage->deleteDirectory($directory);
} }
} }
if ($this->isFolderEmpty($imageFolder)) {
$storage->deleteDirectory($imageFolder);
}
$image->delete();
return true; return true;
} }
@ -321,6 +298,46 @@ class ImageService extends UploadService
return $image; return $image;
} }
/**
* Delete gallery and drawings that are not within HTML content of pages or page revisions.
* Checks based off of only the image name.
* Could be much improved to be more specific but kept it generic for now to be safe.
*
* Returns the path of the images that would be/have been deleted.
* @param bool $checkRevisions
* @param bool $dryRun
* @param array $types
* @return array
*/
public function deleteUnusedImages($checkRevisions = true, $dryRun = true, $types = ['gallery', 'drawio'])
{
$types = array_intersect($types, ['gallery', 'drawio']);
$deletedPaths = [];
$this->image->newQuery()->whereIn('type', $types)
->chunk(1000, function($images) use ($types, $checkRevisions, &$deletedPaths, $dryRun) {
foreach ($images as $image) {
$searchQuery = '%' . basename($image->path) . '%';
$inPage = DB::table('pages')
->where('html', 'like', $searchQuery)->count() > 0;
$inRevision = false;
if ($checkRevisions) {
$inRevision = DB::table('page_revisions')
->where('html', 'like', $searchQuery)->count() > 0;
}
if (!$inPage && !$inRevision) {
$deletedPaths[] = $image->path;
if (!$dryRun) {
$this->destroy($image);
}
}
}
});
return $deletedPaths;
}
/** /**
* Convert a image URI to a Base64 encoded string. * Convert a image URI to a Base64 encoded string.
* Attempts to find locally via set storage method first. * Attempts to find locally via set storage method first.

View File

@ -0,0 +1,4 @@
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
<path clip-rule="evenodd" fill="none" d="M0 0h24v24H0z"/>
<path d="M22.7 19l-9.1-9.1c.9-2.3.4-5-1.5-6.9-2-2-5-2.4-7.4-1.3L9 6 6 9 1.6 4.7C.4 7.1.9 10.1 2.9 12.1c1.9 1.9 4.6 2.4 6.9 1.5l9.1 9.1c.4.4 1 .4 1.4 0l2.3-2.3c.5-.4.5-1.1.1-1.4z"/>
</svg>

After

Width:  |  Height:  |  Size: 315 B

View File

@ -52,6 +52,10 @@ class MarkdownEditor {
let action = button.getAttribute('data-action'); let action = button.getAttribute('data-action');
if (action === 'insertImage') this.actionInsertImage(); if (action === 'insertImage') this.actionInsertImage();
if (action === 'insertLink') this.actionShowLinkSelector(); if (action === 'insertLink') this.actionShowLinkSelector();
if (action === 'insertDrawing' && event.ctrlKey) {
this.actionShowImageManager();
return;
}
if (action === 'insertDrawing') this.actionStartDrawing(); if (action === 'insertDrawing') this.actionStartDrawing();
}); });
@ -293,7 +297,14 @@ class MarkdownEditor {
this.cm.focus(); this.cm.focus();
this.cm.replaceSelection(newText); this.cm.replaceSelection(newText);
this.cm.setCursor(cursorPos.line, cursorPos.ch + newText.length); this.cm.setCursor(cursorPos.line, cursorPos.ch + newText.length);
}); }, 'gallery');
}
actionShowImageManager() {
let cursorPos = this.cm.getCursor('from');
window.ImageManager.show(image => {
this.insertDrawing(image, cursorPos);
}, 'drawio');
} }
// Show the popup link selector and insert a link when finished // Show the popup link selector and insert a link when finished
@ -324,10 +335,7 @@ class MarkdownEditor {
}; };
window.$http.post(window.baseUrl('/images/drawing/upload'), data).then(resp => { window.$http.post(window.baseUrl('/images/drawing/upload'), data).then(resp => {
let newText = `<div drawio-diagram="${resp.data.id}"><img src="${resp.data.url}"></div>`; this.insertDrawing(resp.data, cursorPos);
this.cm.focus();
this.cm.replaceSelection(newText);
this.cm.setCursor(cursorPos.line, cursorPos.ch + newText.length);
DrawIO.close(); DrawIO.close();
}).catch(err => { }).catch(err => {
window.$events.emit('error', trans('errors.image_upload_error')); window.$events.emit('error', trans('errors.image_upload_error'));
@ -336,6 +344,13 @@ class MarkdownEditor {
}); });
} }
insertDrawing(image, originalCursor) {
let newText = `<div drawio-diagram="${image.id}"><img src="${image.url}"></div>`;
this.cm.focus();
this.cm.replaceSelection(newText);
this.cm.setCursor(originalCursor.line, originalCursor.ch + newText.length);
}
// Show draw.io if enabled and handle save. // Show draw.io if enabled and handle save.
actionEditDrawing(imgContainer) { actionEditDrawing(imgContainer) {
if (document.querySelector('[drawio-enabled]').getAttribute('drawio-enabled') !== 'true') return; if (document.querySelector('[drawio-enabled]').getAttribute('drawio-enabled') !== 'true') return;
@ -353,8 +368,8 @@ class MarkdownEditor {
uploaded_to: Number(document.getElementById('page-editor').getAttribute('page-id')) uploaded_to: Number(document.getElementById('page-editor').getAttribute('page-id'))
}; };
window.$http.put(window.baseUrl(`/images/drawing/upload/${drawingId}`), data).then(resp => { window.$http.post(window.baseUrl(`/images/drawing/upload`), data).then(resp => {
let newText = `<div drawio-diagram="${resp.data.id}"><img src="${resp.data.url + `?updated=${Date.now()}`}"></div>`; let newText = `<div drawio-diagram="${resp.data.id}"><img src="${resp.data.url}"></div>`;
let newContent = this.cm.getValue().split('\n').map(line => { let newContent = this.cm.getValue().split('\n').map(line => {
if (line.indexOf(`drawio-diagram="${drawingId}"`) !== -1) { if (line.indexOf(`drawio-diagram="${drawingId}"`) !== -1) {
return newText; return newText;

View File

@ -221,8 +221,6 @@ function codePlugin() {
function drawIoPlugin() { function drawIoPlugin() {
const drawIoUrl = 'https://www.draw.io/?embed=1&ui=atlas&spin=1&proto=json';
let iframe = null;
let pageEditor = null; let pageEditor = null;
let currentNode = null; let currentNode = null;
@ -230,6 +228,22 @@ function drawIoPlugin() {
return node.hasAttribute('drawio-diagram'); return node.hasAttribute('drawio-diagram');
} }
function showDrawingManager(mceEditor, selectedNode = null) {
pageEditor = mceEditor;
currentNode = selectedNode;
// Show image manager
window.ImageManager.show(function (image) {
if (selectedNode) {
let imgElem = selectedNode.querySelector('img');
pageEditor.dom.setAttrib(imgElem, 'src', image.url);
pageEditor.dom.setAttrib(selectedNode, 'drawio-diagram', image.id);
} else {
let imgHTML = `<div drawio-diagram="${image.id}" contenteditable="false"><img src="${image.url}"></div>`;
pageEditor.insertContent(imgHTML);
}
}, 'drawio');
}
function showDrawingEditor(mceEditor, selectedNode = null) { function showDrawingEditor(mceEditor, selectedNode = null) {
pageEditor = mceEditor; pageEditor = mceEditor;
currentNode = selectedNode; currentNode = selectedNode;
@ -248,9 +262,9 @@ function drawIoPlugin() {
if (currentNode) { if (currentNode) {
DrawIO.close(); DrawIO.close();
let imgElem = currentNode.querySelector('img'); let imgElem = currentNode.querySelector('img');
let drawingId = currentNode.getAttribute('drawio-diagram'); window.$http.post(window.baseUrl(`/images/drawing/upload`), data).then(resp => {
window.$http.put(window.baseUrl(`/images/drawing/upload/${drawingId}`), data).then(resp => { pageEditor.dom.setAttrib(imgElem, 'src', resp.data.url);
pageEditor.dom.setAttrib(imgElem, 'src', `${resp.data.url}?updated=${Date.now()}`); pageEditor.dom.setAttrib(currentNode, 'drawio-diagram', resp.data.id);
}).catch(err => { }).catch(err => {
window.$events.emit('error', trans('errors.image_upload_error')); window.$events.emit('error', trans('errors.image_upload_error'));
console.log(err); console.log(err);
@ -287,13 +301,24 @@ function drawIoPlugin() {
window.tinymce.PluginManager.add('drawio', function(editor, url) { window.tinymce.PluginManager.add('drawio', function(editor, url) {
editor.addCommand('drawio', () => { editor.addCommand('drawio', () => {
showDrawingEditor(editor); let selectedNode = editor.selection.getNode();
showDrawingEditor(editor, isDrawing(selectedNode) ? selectedNode : null);
}); });
editor.addButton('drawio', { editor.addButton('drawio', {
type: 'splitbutton',
tooltip: 'Drawing', tooltip: 'Drawing',
image: ` dy53My5vcmcvMjAwMC9zdmciPgogICAgPHBhdGggZD0iTTIzIDdWMWgtNnYySDdWMUgxdjZoMnYx MEgxdjZoNnYtMmgxMHYyaDZ2LTZoLTJWN2gyek0zIDNoMnYySDNWM3ptMiAxOEgzdi0yaDJ2Mnpt MTItMkg3di0ySDVWN2gyVjVoMTB2MmgydjEwaC0ydjJ6bTQgMmgtMnYtMmgydjJ6TTE5IDVWM2gy djJoLTJ6bS01LjI3IDloLTMuNDlsLS43MyAySDcuODlsMy40LTloMS40bDMuNDEgOWgtMS42M2wt Ljc0LTJ6bS0zLjA0LTEuMjZoMi42MUwxMiA4LjkxbC0xLjMxIDMuODN6Ii8+CiAgICA8cGF0aCBk PSJNMCAwaDI0djI0SDB6IiBmaWxsPSJub25lIi8+Cjwvc3ZnPg==`, image: ` dy53My5vcmcvMjAwMC9zdmciPgogICAgPHBhdGggZD0iTTIzIDdWMWgtNnYySDdWMUgxdjZoMnYx MEgxdjZoNnYtMmgxMHYyaDZ2LTZoLTJWN2gyek0zIDNoMnYySDNWM3ptMiAxOEgzdi0yaDJ2Mnpt MTItMkg3di0ySDVWN2gyVjVoMTB2MmgydjEwaC0ydjJ6bTQgMmgtMnYtMmgydjJ6TTE5IDVWM2gy djJoLTJ6bS01LjI3IDloLTMuNDlsLS43MyAySDcuODlsMy40LTloMS40bDMuNDEgOWgtMS42M2wt Ljc0LTJ6bS0zLjA0LTEuMjZoMi42MUwxMiA4LjkxbC0xLjMxIDMuODN6Ii8+CiAgICA8cGF0aCBk PSJNMCAwaDI0djI0SDB6IiBmaWxsPSJub25lIi8+Cjwvc3ZnPg==`,
cmd: 'drawio' cmd: 'drawio',
menu: [
{
text: 'Drawing Manager',
onclick() {
let selectedNode = editor.selection.getNode();
showDrawingManager(editor, isDrawing(selectedNode) ? selectedNode : null);
}
}
]
}); });
editor.on('dblclick', event => { editor.on('dblclick', event => {
@ -443,7 +468,7 @@ class WysiwygEditor {
html += `<img src="${image.thumbs.display}" alt="${image.name}">`; html += `<img src="${image.thumbs.display}" alt="${image.name}">`;
html += '</a>'; html += '</a>';
win.tinyMCE.activeEditor.execCommand('mceInsertContent', false, html); win.tinyMCE.activeEditor.execCommand('mceInsertContent', false, html);
}); }, 'gallery');
} }
}, },
@ -522,7 +547,7 @@ class WysiwygEditor {
html += `<img src="${image.thumbs.display}" alt="${image.name}">`; html += `<img src="${image.thumbs.display}" alt="${image.name}">`;
html += '</a>'; html += '</a>';
editor.execCommand('mceInsertContent', false, html); editor.execCommand('mceInsertContent', false, html);
}); }, 'gallery');
} }
}); });

View File

@ -26,25 +26,32 @@ const data = {
imageUpdateSuccess: false, imageUpdateSuccess: false,
imageDeleteSuccess: false, imageDeleteSuccess: false,
deleteConfirm: false,
}; };
const methods = { const methods = {
show(providedCallback) { show(providedCallback, imageType = null) {
callback = providedCallback; callback = providedCallback;
this.showing = true; this.showing = true;
this.$el.children[0].components.overlay.show(); this.$el.children[0].components.overlay.show();
// Get initial images if they have not yet been loaded in. // Get initial images if they have not yet been loaded in.
if (dataLoaded) return; if (dataLoaded && imageType === this.imageType) return;
if (imageType) {
this.imageType = imageType;
this.resetState();
}
this.fetchData(); this.fetchData();
dataLoaded = true; dataLoaded = true;
}, },
hide() { hide() {
if (this.$refs.dropzone) {
this.$refs.dropzone.onClose();
}
this.showing = false; this.showing = false;
this.selectedImage = false; this.selectedImage = false;
this.$refs.dropzone.onClose();
this.$el.children[0].components.overlay.hide(); this.$el.children[0].components.overlay.hide();
}, },
@ -62,13 +69,18 @@ const methods = {
}, },
setView(viewName) { setView(viewName) {
this.view = viewName;
this.resetState();
this.fetchData();
},
resetState() {
this.cancelSearch(); this.cancelSearch();
this.images = []; this.images = [];
this.hasMore = false; this.hasMore = false;
this.deleteConfirm = false;
page = 0; page = 0;
this.view = viewName; baseUrl = window.baseUrl(`/images/${this.imageType}/${this.view}/`);
baseUrl = window.baseUrl(`/images/${this.imageType}/${viewName}/`);
this.fetchData();
}, },
searchImages() { searchImages() {
@ -89,6 +101,7 @@ const methods = {
}, },
cancelSearch() { cancelSearch() {
if (!this.searching) return;
this.searching = false; this.searching = false;
this.searchTerm = ''; this.searchTerm = '';
this.images = preSearchImages; this.images = preSearchImages;
@ -105,6 +118,7 @@ const methods = {
this.callbackAndHide(image); this.callbackAndHide(image);
} else { } else {
this.selectedImage = image; this.selectedImage = image;
this.deleteConfirm = false;
this.dependantPages = false; this.dependantPages = false;
} }
@ -134,17 +148,22 @@ const methods = {
}, },
deleteImage() { deleteImage() {
let force = this.dependantPages !== false;
let url = window.baseUrl('/images/' + this.selectedImage.id); if (!this.deleteConfirm) {
if (force) url += '?force=true'; let url = window.baseUrl(`/images/usage/${this.selectedImage.id}`);
this.$http.delete(url).then(response => { this.$http.get(url).then(resp => {
this.dependantPages = resp.data;
}).catch(console.error).then(() => {
this.deleteConfirm = true;
});
return;
}
this.$http.delete(`/images/${this.selectedImage.id}`).then(resp => {
this.images.splice(this.images.indexOf(this.selectedImage), 1); this.images.splice(this.images.indexOf(this.selectedImage), 1);
this.selectedImage = false; this.selectedImage = false;
this.$events.emit('success', trans('components.image_delete_success')); this.$events.emit('success', trans('components.image_delete_success'));
}).catch(error=> { this.deleteConfirm = false;
if (error.response.status === 400) {
this.dependantPages = error.response.data;
}
}); });
}, },

View File

@ -146,7 +146,8 @@ body.flexbox-support #entity-selector-wrap .popup-body .form-group {
.dropzone-container { .dropzone-container {
position: relative; position: relative;
border: 3px dashed #DDD; background-color: #EEE;
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='4' height='4' viewBox='0 0 4 4'%3E%3Cpath fill='%23a9a9a9' fill-opacity='0.52' d='M1 3h1v1H1V3zm2-2h1v1H3V1z'%3E%3C/path%3E%3C/svg%3E");
} }
.image-manager-list .image { .image-manager-list .image {
@ -163,8 +164,10 @@ body.flexbox-support #entity-selector-wrap .popup-body .form-group {
transition: all cubic-bezier(.4, 0, 1, 1) 160ms; transition: all cubic-bezier(.4, 0, 1, 1) 160ms;
overflow: hidden; overflow: hidden;
&.selected { &.selected {
transform: scale3d(0.92, 0.92, 0.92); //transform: scale3d(0.92, 0.92, 0.92);
border: 1px solid #444; border: 4px solid #FFF;
overflow: hidden;
border-radius: 8px;
box-shadow: 0 1px 2px 0 rgba(0, 0, 0, 0.2); box-shadow: 0 1px 2px 0 rgba(0, 0, 0, 0.2);
} }
img { img {
@ -210,12 +213,30 @@ body.flexbox-support #entity-selector-wrap .popup-body .form-group {
.image-manager-sidebar { .image-manager-sidebar {
width: 300px; width: 300px;
margin-left: 1px; margin-left: 1px;
padding: $-m $-l;
overflow-y: auto; overflow-y: auto;
overflow-x: hidden; overflow-x: hidden;
border-left: 1px solid #DDD; border-left: 1px solid #DDD;
.inner {
padding: $-m;
}
img {
max-width: 100%;
max-height: 180px;
display: block;
margin: 0 auto $-m auto;
box-shadow: 0 1px 21px 1px rgba(76, 76, 76, 0.3);
}
.image-manager-viewer {
height: 196px;
display: flex;
align-items: center;
justify-content: center;
a {
display: inline-block;
}
}
.dropzone-container { .dropzone-container {
margin-top: $-m; border-bottom: 1px solid #DDD;
} }
} }
@ -242,10 +263,10 @@ body.flexbox-support #entity-selector-wrap .popup-body .form-group {
* Copyright (c) 2012 Matias Meno <m@tias.me> * Copyright (c) 2012 Matias Meno <m@tias.me>
*/ */
.dz-message { .dz-message {
font-size: 1.2em; font-size: 1em;
line-height: 1.1; line-height: 2.35;
font-style: italic; font-style: italic;
color: #aaa; color: #888;
text-align: center; text-align: center;
cursor: pointer; cursor: pointer;
padding: $-l $-m; padding: $-l $-m;

View File

@ -154,6 +154,7 @@ $btt-size: 40px;
} }
input { input {
flex: 5; flex: 5;
padding: $-xs $-s;
&:focus, &:active { &:focus, &:active {
outline: 0; outline: 0;
} }

View File

@ -12,7 +12,8 @@ return [
'image_uploaded' => 'Hochgeladen am :uploadedDate', 'image_uploaded' => 'Hochgeladen am :uploadedDate',
'image_load_more' => 'Mehr', 'image_load_more' => 'Mehr',
'image_image_name' => 'Bildname', 'image_image_name' => 'Bildname',
'image_delete_confirm' => 'Dieses Bild wird auf den folgenden Seiten benutzt. Bitte klicken Sie erneut auf löschen, wenn Sie dieses Bild wirklich entfernen möchten.', 'image_delete_used' => 'Dieses Bild wird auf den folgenden Seiten benutzt. ',
'image_delete_confirm' => 'Bitte klicken Sie erneut auf löschen, wenn Sie dieses Bild wirklich entfernen möchten.',
'image_select_image' => 'Bild auswählen', 'image_select_image' => 'Bild auswählen',
'image_dropzone' => 'Ziehen Sie Bilder hierher oder klicken Sie, um ein Bild auszuwählen', 'image_dropzone' => 'Ziehen Sie Bilder hierher oder klicken Sie, um ein Bild auszuwählen',
'images_deleted' => 'Bilder gelöscht', 'images_deleted' => 'Bilder gelöscht',

View File

@ -13,7 +13,8 @@ return [
'image_uploaded' => 'Uploaded :uploadedDate', 'image_uploaded' => 'Uploaded :uploadedDate',
'image_load_more' => 'Load More', 'image_load_more' => 'Load More',
'image_image_name' => 'Image Name', 'image_image_name' => 'Image Name',
'image_delete_confirm' => 'This image is used in the pages below, Click delete again to confirm you want to delete this image.', 'image_delete_used' => 'This image is used in the pages below.',
'image_delete_confirm' => 'Click delete again to confirm you want to delete this image.',
'image_select_image' => 'Select Image', 'image_select_image' => 'Select Image',
'image_dropzone' => 'Drop images or click here to upload', 'image_dropzone' => 'Drop images or click here to upload',
'images_deleted' => 'Images Deleted', 'images_deleted' => 'Images Deleted',

View File

@ -51,6 +51,19 @@ return [
'reg_confirm_restrict_domain_desc' => 'Enter a comma separated list of email domains you would like to restrict registration to. Users will be sent an email to confirm their address before being allowed to interact with the application. <br> Note that users will be able to change their email addresses after successful registration.', 'reg_confirm_restrict_domain_desc' => 'Enter a comma separated list of email domains you would like to restrict registration to. Users will be sent an email to confirm their address before being allowed to interact with the application. <br> Note that users will be able to change their email addresses after successful registration.',
'reg_confirm_restrict_domain_placeholder' => 'No restriction set', 'reg_confirm_restrict_domain_placeholder' => 'No restriction set',
/**
* Maintenance settings
*/
'maint' => 'Maintenance',
'maint_image_cleanup' => 'Cleanup Images',
'maint_image_cleanup_desc' => "Scans page & revision content to check which images and drawings are currently in use and which images are redundant. Ensure you create a full database and image backup before running this.",
'maint_image_cleanup_ignore_revisions' => 'Ignore images in revisions',
'maint_image_cleanup_run' => 'Run Cleanup',
'maint_image_cleanup_warning' => ':count potentially unused images were found. Are you sure you want to delete these images?',
'maint_image_cleanup_success' => ':count potentially unused images found and deleted!',
'maint_image_cleanup_nothing_found' => 'No unused images found, Nothing deleted!',
/** /**
* Role settings * Role settings
*/ */

View File

@ -13,7 +13,8 @@ return [
'image_uploaded' => 'Subido el :uploadedDate', 'image_uploaded' => 'Subido el :uploadedDate',
'image_load_more' => 'Cargar más', 'image_load_more' => 'Cargar más',
'image_image_name' => 'Nombre de imagen', 'image_image_name' => 'Nombre de imagen',
'image_delete_confirm' => 'Esta imagen está siendo utilizada en las páginas mostradas a continuación, haga click de nuevo para confirmar que quiere borrar esta imagen.', 'image_delete_used' => 'Esta imagen está siendo utilizada en las páginas mostradas a continuación.',
'image_delete_confirm' => 'Haga click de nuevo para confirmar que quiere borrar esta imagen.',
'image_select_image' => 'Seleccionar Imagen', 'image_select_image' => 'Seleccionar Imagen',
'image_dropzone' => 'Arrastre las imágenes o hacer click aquí para Subir', 'image_dropzone' => 'Arrastre las imágenes o hacer click aquí para Subir',
'images_deleted' => 'Imágenes borradas', 'images_deleted' => 'Imágenes borradas',

View File

@ -13,7 +13,8 @@ return [
'image_uploaded' => 'Subido el :uploadedDate', 'image_uploaded' => 'Subido el :uploadedDate',
'image_load_more' => 'Cargar más', 'image_load_more' => 'Cargar más',
'image_image_name' => 'Nombre de imagen', 'image_image_name' => 'Nombre de imagen',
'image_delete_confirm' => 'Esta imagen esta siendo utilizada en las páginas a continuación, haga click de nuevo para confirmar que quiere borrar esta imagen.', 'image_delete_used' => 'Esta imagen esta siendo utilizada en las páginas a continuación.',
'image_delete_confirm' => 'Haga click de nuevo para confirmar que quiere borrar esta imagen.',
'image_select_image' => 'Seleccionar Imagen', 'image_select_image' => 'Seleccionar Imagen',
'image_dropzone' => 'Arrastre las imágenes o hacer click aquí para Subir', 'image_dropzone' => 'Arrastre las imágenes o hacer click aquí para Subir',
'images_deleted' => 'Imágenes borradas', 'images_deleted' => 'Imágenes borradas',

View File

@ -13,7 +13,8 @@ return [
'image_uploaded' => 'Ajoutée le :uploadedDate', 'image_uploaded' => 'Ajoutée le :uploadedDate',
'image_load_more' => 'Charger plus', 'image_load_more' => 'Charger plus',
'image_image_name' => 'Nom de l\'image', 'image_image_name' => 'Nom de l\'image',
'image_delete_confirm' => 'Cette image est utilisée dans les pages ci-dessous. Confirmez que vous souhaitez bien supprimer cette image.', 'image_delete_used' => 'Cette image est utilisée dans les pages ci-dessous.',
'image_delete_confirm' => 'Confirmez que vous souhaitez bien supprimer cette image.',
'image_select_image' => 'Selectionner l\'image', 'image_select_image' => 'Selectionner l\'image',
'image_dropzone' => 'Glissez les images ici ou cliquez pour les ajouter', 'image_dropzone' => 'Glissez les images ici ou cliquez pour les ajouter',
'images_deleted' => 'Images supprimées', 'images_deleted' => 'Images supprimées',

View File

@ -13,7 +13,8 @@ return [
'image_uploaded' => 'Uploaded :uploadedDate', 'image_uploaded' => 'Uploaded :uploadedDate',
'image_load_more' => 'Carica Altre', 'image_load_more' => 'Carica Altre',
'image_image_name' => 'Nome Immagine', 'image_image_name' => 'Nome Immagine',
'image_delete_confirm' => 'Questa immagine è usata nelle pagine elencate, clicca elimina nuovamente per confermare.', 'image_delete_used' => 'Questa immagine è usata nelle pagine elencate.',
'image_delete_confirm' => 'Clicca elimina nuovamente per confermare.',
'image_select_image' => 'Seleziona Immagine', 'image_select_image' => 'Seleziona Immagine',
'image_dropzone' => 'Rilascia immagini o clicca qui per caricarle', 'image_dropzone' => 'Rilascia immagini o clicca qui per caricarle',
'images_deleted' => 'Immagini Eliminate', 'images_deleted' => 'Immagini Eliminate',

View File

@ -13,7 +13,8 @@ return [
'image_uploaded' => 'アップロード日時: :uploadedDate', 'image_uploaded' => 'アップロード日時: :uploadedDate',
'image_load_more' => 'さらに読み込む', 'image_load_more' => 'さらに読み込む',
'image_image_name' => '画像名', 'image_image_name' => '画像名',
'image_delete_confirm' => 'この画像は以下のページで利用されています。削除してもよろしければ、再度ボタンを押して下さい。', 'image_delete_used' => 'この画像は以下のページで利用されています。',
'image_delete_confirm' => '削除してもよろしければ、再度ボタンを押して下さい。',
'image_select_image' => '選択', 'image_select_image' => '選択',
'image_dropzone' => '画像をドロップするか、クリックしてアップロード', 'image_dropzone' => '画像をドロップするか、クリックしてアップロード',
'images_deleted' => '画像を削除しました', 'images_deleted' => '画像を削除しました',

View File

@ -13,7 +13,8 @@ return [
'image_uploaded' => 'Uploaded :uploadedDate', 'image_uploaded' => 'Uploaded :uploadedDate',
'image_load_more' => 'Meer Laden', 'image_load_more' => 'Meer Laden',
'image_image_name' => 'Afbeeldingsnaam', 'image_image_name' => 'Afbeeldingsnaam',
'image_delete_confirm' => 'Deze afbeeldingen is op onderstaande pagina\'s in gebruik, Klik opnieuw op verwijderen om de afbeelding echt te verwijderen.', 'image_delete_used' => 'Deze afbeeldingen is op onderstaande pagina\'s in gebruik.',
'image_delete_confirm' => 'Klik opnieuw op verwijderen om de afbeelding echt te verwijderen.',
'image_select_image' => 'Kies Afbeelding', 'image_select_image' => 'Kies Afbeelding',
'image_dropzone' => 'Sleep afbeeldingen hier of klik hier om te uploaden', 'image_dropzone' => 'Sleep afbeeldingen hier of klik hier om te uploaden',
'images_deleted' => 'Verwijderde Afbeeldingen', 'images_deleted' => 'Verwijderde Afbeeldingen',

View File

@ -13,7 +13,8 @@ return [
'image_uploaded' => 'Udostępniono :uploadedDate', 'image_uploaded' => 'Udostępniono :uploadedDate',
'image_load_more' => 'Wczytaj więcej', 'image_load_more' => 'Wczytaj więcej',
'image_image_name' => 'Nazwa obrazka', 'image_image_name' => 'Nazwa obrazka',
'image_delete_confirm' => 'Ten obrazek jest używany na stronach poniżej, kliknij ponownie Usuń by potwierdzić usunięcie obrazka.', 'image_delete_used' => 'Ten obrazek jest używany na stronach poniżej.',
'image_delete_confirm' => 'Kliknij ponownie Usuń by potwierdzić usunięcie obrazka.',
'image_select_image' => 'Wybierz obrazek', 'image_select_image' => 'Wybierz obrazek',
'image_dropzone' => 'Upuść obrazki tutaj lub kliknij by wybrać obrazki do udostępnienia', 'image_dropzone' => 'Upuść obrazki tutaj lub kliknij by wybrać obrazki do udostępnienia',
'images_deleted' => 'Usunięte obrazki', 'images_deleted' => 'Usunięte obrazki',

View File

@ -13,7 +13,8 @@ return [
'image_uploaded' => 'Carregado :uploadedDate', 'image_uploaded' => 'Carregado :uploadedDate',
'image_load_more' => 'Carregar Mais', 'image_load_more' => 'Carregar Mais',
'image_image_name' => 'Nome da Imagem', 'image_image_name' => 'Nome da Imagem',
'image_delete_confirm' => 'Essa imagem é usada nas páginas abaixo. Clique em Excluir novamente para confirmar que você deseja mesmo eliminar a imagem.', 'image_delete_used' => 'Essa imagem é usada nas páginas abaixo.',
'image_delete_confirm' => 'Clique em Excluir novamente para confirmar que você deseja mesmo eliminar a imagem.',
'image_select_image' => 'Selecionar Imagem', 'image_select_image' => 'Selecionar Imagem',
'image_dropzone' => 'Arraste imagens ou clique aqui para fazer upload', 'image_dropzone' => 'Arraste imagens ou clique aqui para fazer upload',
'images_deleted' => 'Imagens excluídas', 'images_deleted' => 'Imagens excluídas',

View File

@ -13,7 +13,8 @@ return [
'image_uploaded' => 'Загруженно :uploadedDate', 'image_uploaded' => 'Загруженно :uploadedDate',
'image_load_more' => 'Загрузить ещё', 'image_load_more' => 'Загрузить ещё',
'image_image_name' => 'Имя изображения', 'image_image_name' => 'Имя изображения',
'image_delete_confirm' => 'Это изображение используется на странице ниже. Снова кликните удалить для подтверждения того что вы хотите удалить.', 'image_delete_used' => 'Это изображение используется на странице ниже.',
'image_delete_confirm' => 'Снова кликните удалить для подтверждения того что вы хотите удалить.',
'image_select_image' => 'Выбрать изображение', 'image_select_image' => 'Выбрать изображение',
'image_dropzone' => 'Перетащите изображение или кликните для загрузки', 'image_dropzone' => 'Перетащите изображение или кликните для загрузки',
'images_deleted' => 'Изображения удалены', 'images_deleted' => 'Изображения удалены',

View File

@ -13,7 +13,8 @@ return [
'image_uploaded' => 'Nahrané :uploadedDate', 'image_uploaded' => 'Nahrané :uploadedDate',
'image_load_more' => 'Načítať viac', 'image_load_more' => 'Načítať viac',
'image_image_name' => 'Názov obrázka', 'image_image_name' => 'Názov obrázka',
'image_delete_confirm' => 'Tento obrázok je použitý na stránkach uvedených nižšie, kliknite znova na zmazať pre potvrdenie zmazania tohto obrázka.', 'image_delete_used' => 'Tento obrázok je použitý na stránkach uvedených nižšie.',
'image_delete_confirm' => 'Kliknite znova na zmazať pre potvrdenie zmazania tohto obrázka.',
'image_select_image' => 'Vybrať obrázok', 'image_select_image' => 'Vybrať obrázok',
'image_dropzone' => 'Presuňte obrázky sem alebo kliknite sem pre nahranie', 'image_dropzone' => 'Presuňte obrázky sem alebo kliknite sem pre nahranie',
'images_deleted' => 'Obrázky zmazané', 'images_deleted' => 'Obrázky zmazané',

View File

@ -13,7 +13,8 @@ return [
'image_uploaded' => 'Laddades upp :uploadedDate', 'image_uploaded' => 'Laddades upp :uploadedDate',
'image_load_more' => 'Ladda fler', 'image_load_more' => 'Ladda fler',
'image_image_name' => 'Bildnamn', 'image_image_name' => 'Bildnamn',
'image_delete_confirm' => 'Den här bilden används på nedanstående sidor, klicka på "ta bort" en gång till för att bekräfta att du vill ta bort bilden.', 'image_delete_used' => 'Den här bilden används på nedanstående sidor.',
'image_delete_confirm' => 'Klicka på "ta bort" en gång till för att bekräfta att du vill ta bort bilden.',
'image_select_image' => 'Välj bild', 'image_select_image' => 'Välj bild',
'image_dropzone' => 'Släpp bilder här eller klicka för att ladda upp', 'image_dropzone' => 'Släpp bilder här eller klicka för att ladda upp',
'images_deleted' => 'Bilder borttagna', 'images_deleted' => 'Bilder borttagna',

View File

@ -13,7 +13,8 @@ return [
'image_uploaded' => '上传于 :uploadedDate', 'image_uploaded' => '上传于 :uploadedDate',
'image_load_more' => '显示更多', 'image_load_more' => '显示更多',
'image_image_name' => '图片名称', 'image_image_name' => '图片名称',
'image_delete_confirm' => '该图像用于以下页面,如果你想删除它,请再次按下按钮。', 'image_delete_used' => '该图像用于以下页面。',
'image_delete_confirm' => '如果你想删除它,请再次按下按钮。',
'image_select_image' => '选择图片', 'image_select_image' => '选择图片',
'image_dropzone' => '拖放图片或点击此处上传', 'image_dropzone' => '拖放图片或点击此处上传',
'images_deleted' => '图片已删除', 'images_deleted' => '图片已删除',

View File

@ -13,7 +13,8 @@ return [
'image_uploaded' => '上傳於 :uploadedDate', 'image_uploaded' => '上傳於 :uploadedDate',
'image_load_more' => '載入更多', 'image_load_more' => '載入更多',
'image_image_name' => '圖片名稱', 'image_image_name' => '圖片名稱',
'image_delete_confirm' => '所使用圖片目前用於以下頁面,如果你想刪除它,請再次按下按鈕。', 'image_delete_used' => '所使用圖片目前用於以下頁面。',
'image_delete_confirm' => '如果你想刪除它,請再次按下按鈕。',
'image_select_image' => '選擇圖片', 'image_select_image' => '選擇圖片',
'image_dropzone' => '拖曳圖片或點選這裡上傳', 'image_dropzone' => '拖曳圖片或點選這裡上傳',
'images_deleted' => '圖片已刪除', 'images_deleted' => '圖片已刪除',

View File

@ -1,5 +1,5 @@
<div id="image-manager" image-type="{{ $imageType }}" uploaded-to="{{ $uploaded_to or 0 }}"> <div id="image-manager" image-type="{{ $imageType }}" uploaded-to="{{ $uploaded_to or 0 }}">
<div overlay v-cloak @click="hide()"> <div overlay v-cloak @click="hide">
<div class="popup-body" @click.stop=""> <div class="popup-body" @click.stop="">
<div class="popup-header primary-background"> <div class="popup-header primary-background">
@ -40,46 +40,53 @@
</div> </div>
<div class="image-manager-sidebar"> <div class="image-manager-sidebar">
<dropzone v-if="imageType !== 'drawio'" ref="dropzone" placeholder="{{ trans('components.image_dropzone') }}" :upload-url="uploadUrl" :uploaded-to="uploadedTo" @success="uploadSuccess"></dropzone>
<div class="inner"> <div class="inner">
<div class="image-manager-details anim fadeIn" v-if="selectedImage"> <div class="image-manager-details anim fadeIn" v-if="selectedImage">
<form @submit.prevent="saveImageDetails"> <form @submit.prevent="saveImageDetails">
<div> <div class="image-manager-viewer">
<a :href="selectedImage.url" target="_blank" style="display: block;"> <a :href="selectedImage.url" target="_blank" style="display: block;">
<img :src="selectedImage.thumbs.gallery" :alt="selectedImage.title" <img :src="selectedImage.thumbs.display" :alt="selectedImage.name"
:title="selectedImage.name"> :title="selectedImage.name">
</a> </a>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="name">{{ trans('components.image_image_name') }}</label> <label for="name">{{ trans('components.image_image_name') }}</label>
<input id="name" name="name" v-model="selectedImage.name"> <input id="name" class="input-base" name="name" v-model="selectedImage.name">
</div> </div>
</form> </form>
<div v-show="dependantPages">
<p class="text-neg text-small">
{{ trans('components.image_delete_confirm') }}
</p>
<ul class="text-neg">
<li v-for="page in dependantPages">
<a :href="page.url" target="_blank" class="text-neg" v-text="page.name"></a>
</li>
</ul>
</div>
<div class="clearfix"> <div class="clearfix">
<form class="float left" @submit.prevent="deleteImage"> <div class="float left">
<button class="button icon neg">@icon('delete')</button> <button type="button" class="button icon outline" @click="deleteImage">@icon('delete')</button>
</form>
<button class="button pos anim fadeIn float right" v-show="selectedImage" @click="callbackAndHide(selectedImage)"> </div>
<button class="button anim fadeIn float right" v-show="selectedImage" @click="callbackAndHide(selectedImage)">
{{ trans('components.image_select_image') }} {{ trans('components.image_select_image') }}
</button> </button>
<div class="clearfix"></div>
<div v-show="dependantPages">
<p class="text-neg text-small">
{{ trans('components.image_delete_used') }}
</p>
<ul class="text-neg">
<li v-for="page in dependantPages">
<a :href="page.url" target="_blank" class="text-neg" v-text="page.name"></a>
</li>
</ul>
</div>
<div v-show="deleteConfirm" class="text-neg text-small">
{{ trans('components.image_delete_confirm') }}
</div>
</div> </div>
</div> </div>
<dropzone ref="dropzone" placeholder="{{ trans('components.image_dropzone') }}" :upload-url="uploadUrl" :uploaded-to="uploadedTo" @success="uploadSuccess"></dropzone>
</div> </div>
</div> </div>

View File

@ -0,0 +1,49 @@
@extends('simple-layout')
@section('toolbar')
@include('settings/navbar', ['selected' => 'maintenance'])
@stop
@section('body')
<div class="container small">
<div class="text-right text-muted container">
<br>
BookStack @if(strpos($version, 'v') !== 0) version @endif {{ $version }}
</div>
<div class="card" id="image-cleanup">
<h3>@icon('images') {{ trans('settings.maint_image_cleanup') }}</h3>
<div class="body">
<div class="row">
<div class="col-sm-6">
<p class="small muted">{{ trans('settings.maint_image_cleanup_desc') }}</p>
</div>
<div class="col-sm-6">
<form method="POST" action="{{ baseUrl('/settings/maintenance/cleanup-images') }}">
{!! csrf_field() !!}
<input type="hidden" name="_method" value="DELETE">
<div>
@if(session()->has('cleanup-images-warning'))
<p class="text neg">
{{ session()->get('cleanup-images-warning') }}
</p>
<input type="hidden" name="ignore_revisions" value="{{ session()->getOldInput('ignore_revisions', 'false') }}">
<input type="hidden" name="confirm" value="true">
@else
<label>
<input type="checkbox" name="ignore_revisions" value="true">
{{ trans('settings.maint_image_cleanup_ignore_revisions') }}
</label>
@endif
</div>
<button class="button outline">{{ trans('settings.maint_image_cleanup_run') }}</button>
</form>
</div>
</div>
</div>
</div>
</div>
@stop

View File

@ -2,6 +2,7 @@
<div class="col-md-12 setting-nav nav-tabs"> <div class="col-md-12 setting-nav nav-tabs">
@if($currentUser->can('settings-manage')) @if($currentUser->can('settings-manage'))
<a href="{{ baseUrl('/settings') }}" @if($selected == 'settings') class="selected text-button" @endif>@icon('settings'){{ trans('settings.settings') }}</a> <a href="{{ baseUrl('/settings') }}" @if($selected == 'settings') class="selected text-button" @endif>@icon('settings'){{ trans('settings.settings') }}</a>
<a href="{{ baseUrl('/settings/maintenance') }}" @if($selected == 'maintenance') class="selected text-button" @endif>@icon('spanner'){{ trans('settings.maint') }}</a>
@endif @endif
@if($currentUser->can('users-manage')) @if($currentUser->can('users-manage'))
<a href="{{ baseUrl('/settings/users') }}" @if($selected == 'users') class="selected text-button" @endif>@icon('users'){{ trans('settings.users') }}</a> <a href="{{ baseUrl('/settings/users') }}" @if($selected == 'users') class="selected text-button" @endif>@icon('users'){{ trans('settings.users') }}</a>

View File

@ -95,7 +95,7 @@ Route::group(['middleware' => 'auth'], function () {
Route::get('/base64/{id}', 'ImageController@getBase64Image'); Route::get('/base64/{id}', 'ImageController@getBase64Image');
Route::put('/update/{imageId}', 'ImageController@update'); Route::put('/update/{imageId}', 'ImageController@update');
Route::post('/drawing/upload', 'ImageController@uploadDrawing'); Route::post('/drawing/upload', 'ImageController@uploadDrawing');
Route::put('/drawing/upload/{id}', 'ImageController@replaceDrawing'); Route::get('/usage/{id}', 'ImageController@usage');
Route::post('/{type}/upload', 'ImageController@uploadByType'); Route::post('/{type}/upload', 'ImageController@uploadByType');
Route::get('/{type}/all', 'ImageController@getAllByType'); Route::get('/{type}/all', 'ImageController@getAllByType');
Route::get('/{type}/all/{page}', 'ImageController@getAllByType'); Route::get('/{type}/all/{page}', 'ImageController@getAllByType');
@ -151,6 +151,10 @@ Route::group(['middleware' => 'auth'], function () {
Route::get('/', 'SettingController@index')->name('settings'); Route::get('/', 'SettingController@index')->name('settings');
Route::post('/', 'SettingController@update'); Route::post('/', 'SettingController@update');
// Maintenance
Route::get('/maintenance', 'SettingController@showMaintenance');
Route::delete('/maintenance/cleanup-images', 'SettingController@cleanupImages');
// Users // Users
Route::get('/users', 'UserController@index'); Route::get('/users', 'UserController@index');
Route::get('/users/create', 'UserController@create'); Route::get('/users/create', 'UserController@create');

View File

@ -2,6 +2,8 @@
use BookStack\Image; use BookStack\Image;
use BookStack\Page; use BookStack\Page;
use BookStack\Repos\EntityRepo;
use BookStack\Services\ImageService;
class ImageTest extends TestCase class ImageTest extends TestCase
{ {
@ -210,38 +212,6 @@ class ImageTest extends TestCase
$this->assertTrue($testImageData === $uploadedImageData, "Uploaded image file data does not match our test image as expected"); $this->assertTrue($testImageData === $uploadedImageData, "Uploaded image file data does not match our test image as expected");
} }
public function test_drawing_replacing()
{
$page = Page::first();
$editor = $this->getEditor();
$this->actingAs($editor);
$this->postJson('images/drawing/upload', [
'uploaded_to' => $page->id,
'image' => 'image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAUAAAAFCAIAAAACDbGyAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH4gEcDQ4S1RUeKwAAAB1pVFh0Q29tbWVudAAAAAAAQ3JlYXRlZCB3aXRoIEdJTVBkLmUHAAAAFElEQVQI12NctNWSAQkwMaACUvkAfCkBmjyhGl4AAAAASUVORK5CYII='
]);
$image = Image::where('type', '=', 'drawio')->first();
$replace = $this->putJson("images/drawing/upload/{$image->id}", [
'image' => 'image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAUAAAAFCAIAAAACDbGyAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH4gEcDCo5iYNs+gAAAB1pVFh0Q29tbWVudAAAAAAAQ3JlYXRlZCB3aXRoIEdJTVBkLmUHAAAAFElEQVQI12O0jN/KgASYGFABqXwAZtoBV6Sl3hIAAAAASUVORK5CYII='
]);
$replace->assertStatus(200);
$replace->assertJson([
'type' => 'drawio',
'uploaded_to' => $page->id,
'created_by' => $editor->id,
'updated_by' => $editor->id,
]);
$this->assertTrue(file_exists(public_path($image->path)), 'Uploaded image not found at path: '. public_path($image->path));
$testImageData = file_get_contents($this->getTestImageFilePath());
$uploadedImageData = file_get_contents(public_path($image->path));
$this->assertTrue($testImageData === $uploadedImageData, "Uploaded image file data does not match our test image as expected");
}
public function test_user_images_deleted_on_user_deletion() public function test_user_images_deleted_on_user_deletion()
{ {
$editor = $this->getEditor(); $editor = $this->getEditor();
@ -266,4 +236,59 @@ class ImageTest extends TestCase
]); ]);
} }
public function test_deleted_unused_images()
{
$page = Page::first();
$admin = $this->getAdmin();
$this->actingAs($admin);
$imageName = 'unused-image.png';
$relPath = $this->getTestImagePath('gallery', $imageName);
$this->deleteImage($relPath);
$upload = $this->uploadImage($imageName, $page->id);
$upload->assertStatus(200);
$image = Image::where('type', '=', 'gallery')->first();
$entityRepo = app(EntityRepo::class);
$entityRepo->updatePage($page, $page->book_id, [
'name' => $page->name,
'html' => $page->html . "<img src=\"{$image->url}\">",
'summary' => ''
]);
// Ensure no images are reported as deletable
$imageService = app(ImageService::class);
$toDelete = $imageService->deleteUnusedImages(true, true);
$this->assertCount(0, $toDelete);
// Save a revision of our page without the image;
$entityRepo->updatePage($page, $page->book_id, [
'name' => $page->name,
'html' => "<p>Hello</p>",
'summary' => ''
]);
// Ensure revision images are picked up okay
$imageService = app(ImageService::class);
$toDelete = $imageService->deleteUnusedImages(true, true);
$this->assertCount(0, $toDelete);
$toDelete = $imageService->deleteUnusedImages(false, true);
$this->assertCount(1, $toDelete);
// Check image is found when revisions are destroyed
$page->revisions()->delete();
$toDelete = $imageService->deleteUnusedImages(true, true);
$this->assertCount(1, $toDelete);
// Check the image is deleted
$absPath = public_path($relPath);
$this->assertTrue(file_exists($absPath), "Existing uploaded file at path {$absPath} exists");
$toDelete = $imageService->deleteUnusedImages(true, false);
$this->assertCount(1, $toDelete);
$this->assertFalse(file_exists($absPath));
$this->deleteImage($relPath);
}
} }