Gallery: add delete functionality for gallery items

This commit is contained in:
Cohee
2025-06-04 23:53:18 +03:00
parent f646c9416a
commit 96fb85457f
5 changed files with 102 additions and 6 deletions

View File

@@ -2439,7 +2439,7 @@ export function appendMediaToMessage(mes, messageElement, adjustScroll = true) {
const image = messageElement.find('.mes_img');
const text = messageElement.find('.mes_text');
const isInline = !!mes.extra?.inline_image;
image.off('load').on('load', function () {
const doAdjustScroll = () => {
if (!adjustScroll) {
return;
}
@@ -2447,6 +2447,16 @@ export function appendMediaToMessage(mes, messageElement, adjustScroll = true) {
const newChatHeight = $('#chat').prop('scrollHeight');
const diff = newChatHeight - chatHeight;
$('#chat').scrollTop(scrollPosition + diff);
};
image.off('load').on('load', function () {
image.removeAttr('alt');
image.removeClass('error');
doAdjustScroll();
});
image.off('error').on('error', function () {
image.attr('alt', '');
image.addClass('error');
doAdjustScroll();
});
image.attr('src', mes.extra?.image);
image.attr('title', mes.extra?.title || mes.title || '');

View File

@@ -15,10 +15,12 @@ import { ARGUMENT_TYPE, SlashCommandNamedArgument } from '../../slash-commands/S
import { DragAndDropHandler } from '../../dragdrop.js';
import { commonEnumProviders } from '../../slash-commands/SlashCommandCommonEnumsProvider.js';
import { t, translate } from '../../i18n.js';
import { Popup } from '../../popup.js';
const extensionName = 'gallery';
const extensionFolderPath = `scripts/extensions/${extensionName}/`;
let firstTime = true;
let deleteModeActive = false;
// Exposed defaults for future tweaking
let thumbnailHeight = 150;
@@ -146,6 +148,29 @@ async function getGalleryFolders() {
}
}
/**
* Deletes a gallery item based on the provided URL.
* @param {string} url - The URL of the image to be deleted.
*/
async function deleteGalleryItem(url) {
try {
const response = await fetch('/api/images/delete', {
method: 'POST',
headers: getRequestHeaders(),
body: JSON.stringify({ path: url }),
});
if (!response.ok) {
throw new Error(`HTTP error. Status: ${response.status}`);
}
toastr.success(t`Image deleted successfully.`);
} catch (error) {
console.error('Failed to delete the image:', error);
toastr.error(t`Failed to delete the image. Check the console for details.`);
}
}
/**
* Sets the sort order for the gallery.
* @param {string} order Sort order
@@ -260,7 +285,7 @@ async function initGallery(items, url) {
*
* @returns {Promise<void>} - Promise representing the completion of the gallery display process.
*/
async function showCharGallery() {
async function showCharGallery(deleteModeState = false) {
// Load necessary files if it's the first time calling the function
if (firstTime) {
await loadFileToDocument(
@@ -276,6 +301,7 @@ async function showCharGallery() {
}
try {
deleteModeActive = deleteModeState;
let url = selected_group || this_chid;
if (!selected_group && this_chid !== undefined) {
url = getGalleryFolder(characters[this_chid]);
@@ -429,6 +455,18 @@ async function makeMovable(url) {
galleryFolderAccept.title = t`Change gallery folder`;
galleryFolderAccept.addEventListener('click', onChangeFolder);
const galleryDeleteMode = document.createElement('div');
galleryDeleteMode.classList.add('right_menu_button', 'fa-solid', 'fa-trash', 'fa-fw');
galleryDeleteMode.classList.toggle('warning', deleteModeActive);
galleryDeleteMode.title = t`Delete mode`;
galleryDeleteMode.addEventListener('click', () => {
deleteModeActive = !deleteModeActive;
galleryDeleteMode.classList.toggle('warning', deleteModeActive);
if (deleteModeActive) {
toastr.info(t`Delete mode is ON. Click on images you want to delete.`);
}
});
const galleryFolderRestore = document.createElement('div');
galleryFolderRestore.classList.add('right_menu_button', 'fa-solid', 'fa-recycle', 'fa-fw');
galleryFolderRestore.title = t`Restore gallery folder`;
@@ -436,6 +474,7 @@ async function makeMovable(url) {
topBarElement.appendChild(galleryFolderInput);
topBarElement.appendChild(galleryFolderAccept);
topBarElement.appendChild(galleryDeleteMode);
topBarElement.appendChild(galleryFolderRestore);
newElement.append(topBarElement);
@@ -635,11 +674,21 @@ function sanitizeHTMLId(id) {
function viewWithDragbox(items) {
if (items && items.length > 0) {
const url = items[0].responsiveURL(); // Get the URL of the clicked image/video
if (deleteModeActive) {
Popup.show.confirm(t`Are you sure you want to delete this image?`, url)
.then(async (confirmed) => {
if (!confirmed) {
return;
}
deleteGalleryItem(url).then(() => showCharGallery(deleteModeActive));
});
} else {
// ID should just be the last part of the URL, removing the extension
const id = sanitizeHTMLId(url.substring(url.lastIndexOf('/') + 1, url.lastIndexOf('.')));
makeDragImg(id, url);
}
}
}
// Registers a simple command for opening the char gallery.

View File

@@ -43,3 +43,8 @@
#dragGallery {
min-height: 25dvh;
}
#gallery .right_menu_button.warning {
opacity: 1;
filter: unset;
}

View File

@@ -5059,6 +5059,12 @@ a:hover {
cursor: pointer;
}
.mes_img.error {
visibility: hidden;
min-height: 100px;
min-width: 120px;
}
.mes_img_swipes,
.mes_img_controls {
position: absolute;
@@ -5099,6 +5105,8 @@ a:hover {
filter: brightness(150%);
}
.mes_img_container:has(.mes_img.error) .mes_img_swipes,
.mes_img_container:has(.mes_img.error) .mes_img_controls,
.mes_img_container:hover .mes_img_swipes,
.mes_img_container:focus-within .mes_img_swipes,
.mes_img_container:hover .mes_img_controls,

View File

@@ -5,7 +5,7 @@ import { Buffer } from 'node:buffer';
import express from 'express';
import sanitize from 'sanitize-filename';
import { clientRelativePath, removeFileExtension, getImages } from '../util.js';
import { clientRelativePath, removeFileExtension, getImages, isPathUnderParent } from '../util.js';
/**
* Ensure the directory for the provided file path exists.
@@ -126,3 +126,27 @@ router.post('/folders', (request, response) => {
return response.status(500).send({ error: 'Unable to retrieve folders' });
}
});
router.post('/delete', async (request, response) => {
try {
if (!request.body.path) {
return response.status(400).send('No path specified');
}
const pathToDelete = path.join(request.user.directories.root, request.body.path);
if (!isPathUnderParent(request.user.directories.userImages, pathToDelete)) {
return response.status(400).send('Invalid path');
}
if (!fs.existsSync(pathToDelete)) {
return response.status(404).send('File not found');
}
fs.unlinkSync(pathToDelete);
console.info(`Deleted image: ${request.body.path} from ${request.user.profile.handle}`);
return response.sendStatus(200);
} catch (error) {
console.error(error);
return response.sendStatus(500);
}
});