491 lines
19 KiB
JavaScript

// Image management functionality
let currentPage = 1;
let totalPages = 1;
let imageBlobCache = new Map(); // Cache for blob URLs
// Load image with authentication and return blob URL
async function loadImageBlob(imageId) {
// Check cache first
if (imageBlobCache.has(imageId)) {
return imageBlobCache.get(imageId);
}
try {
const blobUrl = await apiClient.downloadImageBlob(imageId);
imageBlobCache.set(imageId, blobUrl);
return blobUrl;
} catch (error) {
console.error(`Failed to load image ${imageId}:`, error);
throw error; // Let the caller handle the error
}
}
// Clean up blob URLs to prevent memory leaks
function cleanupBlobCache() {
for (const [imageId, blobUrl] of imageBlobCache.entries()) {
if (blobUrl.startsWith('blob:')) {
URL.revokeObjectURL(blobUrl);
}
}
imageBlobCache.clear();
}
// Load images with pagination
async function loadImages(page = 1, tags = null) {
if (!config.isConfigured()) {
showAlert('Please configure your API settings first.', 'warning');
return;
}
const container = document.getElementById('imagesContainer');
container.innerHTML = '<div class="text-center"><div class="loading-spinner"></div> Loading images...</div>';
try {
const response = await apiClient.getImages(page, 20, tags);
currentPage = page;
totalPages = Math.ceil(response.total / (response.limit || 20));
// Handle structured response - extract images array from response object
const images = response.images || response;
displayImages(images);
displayPagination(response);
} catch (error) {
handleApiError(error, 'loading images');
container.innerHTML = '<div class="alert alert-danger">Failed to load images</div>';
}
}
// Display images in grid
async function displayImages(images) {
const container = document.getElementById('imagesContainer');
if (!images || images.length === 0) {
container.innerHTML = `
<div class="text-center py-5">
<i class="fas fa-images fa-3x text-muted mb-3"></i>
<h4>No images found</h4>
<p class="text-muted">Upload your first image to get started!</p>
<button class="btn btn-primary" onclick="showUploadModal()">
<i class="fas fa-plus me-1"></i>Upload Image
</button>
</div>
`;
return;
}
// Create image cards with placeholder images first
const imagesHtml = images.map(image => `
<div class="image-card card">
<div class="image-container" style="position: relative; height: 200px; background-color: #f8f9fa; display: flex; align-items: center; justify-content: center;">
<img id="img-${image.id}"
src=""
alt="${escapeHtml(image.description || 'Image')}"
onclick="viewImage('${image.id}')"
style="cursor: pointer; opacity: 0; width: 100%; height: 200px; object-fit: cover; position: absolute; top: 0; left: 0;">
<div class="loading-overlay" style="position: absolute; top: 50%; left: 50%; transform: translate(-50%, -50%); z-index: 2;">
<div class="loading-spinner large"></div>
<div class="mt-2 text-muted small">Loading...</div>
</div>
</div>
<div class="card-body">
<h6 class="card-title">${escapeHtml(truncateText(image.description || 'Untitled', 50))}</h6>
<p class="card-text small text-muted">
<i class="fas fa-calendar me-1"></i>${formatDate(image.upload_date)}
</p>
${image.tags && image.tags.length > 0 ? `
<div class="mb-2">
${image.tags.map(tag => `<span class="badge bg-secondary me-1">${escapeHtml(tag)}</span>`).join('')}
</div>
` : ''}
<div class="btn-group w-100" role="group">
<button class="btn btn-sm btn-outline-primary" onclick="viewImage('${image.id}')">
<i class="fas fa-eye"></i>
</button>
<button class="btn btn-sm btn-outline-secondary" onclick="editImage('${image.id}')">
<i class="fas fa-edit"></i>
</button>
<button class="btn btn-sm btn-outline-danger" onclick="deleteImage('${image.id}')">
<i class="fas fa-trash"></i>
</button>
</div>
</div>
</div>
`).join('');
container.innerHTML = `<div class="image-grid">${imagesHtml}</div>`;
// Load actual images asynchronously
for (const image of images) {
loadImageBlob(image.id).then(blobUrl => {
const imgElement = document.getElementById(`img-${image.id}`);
const loadingOverlay = imgElement.parentElement.querySelector('.loading-overlay');
if (imgElement) {
imgElement.src = blobUrl;
imgElement.style.opacity = '1';
if (loadingOverlay) {
loadingOverlay.style.display = 'none';
}
}
}).catch(error => {
console.error(`Failed to load image ${image.id}:`, error);
const imgElement = document.getElementById(`img-${image.id}`);
const loadingOverlay = imgElement.parentElement.querySelector('.loading-overlay');
if (imgElement && loadingOverlay) {
// Show error state instead of broken image
loadingOverlay.innerHTML = `
<div class="text-center text-muted">
<i class="fas fa-exclamation-triangle fa-2x mb-2"></i>
<div class="small">Failed to load</div>
</div>
`;
imgElement.style.display = 'none'; // Hide the img element completely
}
});
}
}
// Display pagination
function displayPagination(response) {
const container = document.getElementById('imagesContainer');
if (totalPages <= 1) return;
const paginationHtml = `
<nav class="mt-4">
<ul class="pagination justify-content-center">
<li class="page-item ${currentPage === 1 ? 'disabled' : ''}">
<a class="page-link" href="#" onclick="loadImages(${currentPage - 1})">Previous</a>
</li>
${Array.from({length: Math.min(totalPages, 5)}, (_, i) => {
const page = i + 1;
return `
<li class="page-item ${page === currentPage ? 'active' : ''}">
<a class="page-link" href="#" onclick="loadImages(${page})">${page}</a>
</li>
`;
}).join('')}
<li class="page-item ${currentPage === totalPages ? 'disabled' : ''}">
<a class="page-link" href="#" onclick="loadImages(${currentPage + 1})">Next</a>
</li>
</ul>
</nav>
`;
container.insertAdjacentHTML('beforeend', paginationHtml);
}
// Show upload modal
function showUploadModal() {
const modalBody = `
<form id="uploadForm">
<div class="mb-3">
<label class="form-label">Select Image</label>
<div class="upload-area" id="uploadArea">
<i class="fas fa-cloud-upload-alt fa-3x text-primary mb-3"></i>
<h5>Drag & drop an image here</h5>
<p class="text-muted">or click to select a file</p>
<input type="file" id="imageFile" accept="image/*" style="display: none;">
</div>
<div id="imagePreview" class="mt-3" style="display: none;">
<img id="previewImg" class="img-fluid rounded" style="max-height: 200px;">
</div>
</div>
<div class="mb-3">
<label for="imageDescription" class="form-label">Description</label>
<textarea class="form-control" id="imageDescription" rows="3"
placeholder="Describe this image..."></textarea>
</div>
<div class="mb-3">
<label for="imageTags" class="form-label">Tags</label>
<input type="text" class="form-control" id="imageTags"
placeholder="Enter tags separated by commas">
<div class="form-text">e.g., nature, landscape, sunset</div>
</div>
</form>
`;
const modalFooter = `
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Cancel</button>
<button type="button" class="btn btn-primary" onclick="uploadImage()">
<i class="fas fa-upload me-1"></i>Upload
</button>
`;
const modal = createModal('uploadModal', 'Upload Image', modalBody, modalFooter);
// Setup file input and drag & drop
const uploadArea = document.getElementById('uploadArea');
const fileInput = document.getElementById('imageFile');
uploadArea.addEventListener('click', () => fileInput.click());
fileInput.addEventListener('change', (e) => {
if (e.target.files.length > 0) {
handleFileSelection(e.target.files[0]);
}
});
setupFileDropZone(uploadArea, (files) => {
if (files.length > 0) {
handleFileSelection(files[0]);
}
});
modal.show();
}
// Handle file selection
async function handleFileSelection(file) {
try {
validateFile(file);
const preview = await createImagePreview(file);
const previewContainer = document.getElementById('imagePreview');
const previewImg = document.getElementById('previewImg');
previewImg.src = preview;
previewContainer.style.display = 'block';
// Store file for upload
document.getElementById('uploadForm').selectedFile = file;
} catch (error) {
showAlert(error.message, 'danger');
}
}
// Upload image
async function uploadImage() {
const form = document.getElementById('uploadForm');
const file = form.selectedFile;
if (!file) {
showAlert('Please select an image file', 'danger');
return;
}
const description = document.getElementById('imageDescription').value.trim();
const tagsInput = document.getElementById('imageTags').value.trim();
const tags = tagsInput ? tagsInput.split(',').map(tag => tag.trim()).filter(tag => tag) : [];
const uploadButton = document.querySelector('#uploadModal .btn-primary');
setLoadingState(uploadButton);
try {
const formData = new FormData();
formData.append('file', file);
formData.append('description', description);
if (tags.length > 0) {
formData.append('tags', tags.join(','));
}
await apiClient.uploadImage(formData);
showAlert('Image uploaded successfully!', 'success');
// Close modal and refresh images
bootstrap.Modal.getInstance(document.getElementById('uploadModal')).hide();
removeModal('uploadModal');
loadImages(currentPage);
} catch (error) {
handleApiError(error, 'uploading image');
} finally {
setLoadingState(uploadButton, false);
}
}
// View image details
async function viewImage(imageId) {
try {
// Close any existing modals first
const existingModals = ['viewImageModal', 'editImageModal'];
existingModals.forEach(modalId => {
const existingModal = document.getElementById(modalId);
if (existingModal) {
const bsModal = bootstrap.Modal.getInstance(existingModal);
if (bsModal) {
bsModal.hide();
}
removeModal(modalId);
}
});
const image = await apiClient.getImage(imageId);
// Load the image with authentication
const imageSrc = await loadImageBlob(imageId);
// Use unique modal ID to prevent conflicts
const modalId = `viewImageModal-${imageId}`;
const modalBody = `
<div class="text-center mb-3">
<img src="${imageSrc}" class="img-fluid rounded"
style="max-height: 400px;" alt="${escapeHtml(image.description || 'Image')}">
</div>
<div class="row">
<div class="col-md-6">
<h6>Description</h6>
<p>${escapeHtml(image.description || 'No description')}</p>
</div>
<div class="col-md-6">
<h6>Details</h6>
<p><strong>Created:</strong> ${formatDate(image.upload_date)}</p>
<p><strong>Size:</strong> ${formatFileSize(image.file_size)}</p>
<p><strong>Type:</strong> ${image.content_type}</p>
</div>
</div>
${image.tags && image.tags.length > 0 ? `
<div class="mt-3">
<h6>Tags</h6>
${image.tags.map(tag => `<span class="badge bg-secondary me-1">${escapeHtml(tag)}</span>`).join('')}
</div>
` : ''}
`;
const modalFooter = `
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Close</button>
<button type="button" class="btn btn-primary" onclick="editImage('${imageId}')">
<i class="fas fa-edit me-1"></i>Edit
</button>
<button type="button" class="btn btn-danger" onclick="deleteImage('${imageId}')">
<i class="fas fa-trash me-1"></i>Delete
</button>
`;
const modal = createModal(modalId, 'Image Details', modalBody, modalFooter);
// Add cleanup when modal is hidden
const modalElement = document.getElementById(modalId);
modalElement.addEventListener('hidden.bs.modal', () => {
removeModal(modalId);
});
modal.show();
} catch (error) {
handleApiError(error, 'loading image details');
}
}
// Edit image
async function editImage(imageId) {
try {
// Close any existing modals first
const existingModals = document.querySelectorAll('.modal');
existingModals.forEach(modal => {
const bsModal = bootstrap.Modal.getInstance(modal);
if (bsModal) {
bsModal.hide();
}
modal.remove();
});
const image = await apiClient.getImage(imageId);
// Load the image with authentication
const imageSrc = await loadImageBlob(imageId);
// Use unique modal ID to prevent conflicts
const modalId = `editImageModal-${imageId}`;
const modalBody = `
<form id="editImageForm-${imageId}">
<div class="mb-3 text-center">
<img src="${imageSrc}" class="img-fluid rounded"
style="max-height: 200px;" alt="${escapeHtml(image.description || 'Image')}">
</div>
<div class="mb-3">
<label for="editDescription-${imageId}" class="form-label">Description</label>
<textarea class="form-control" id="editDescription-${imageId}" rows="3">${escapeHtml(image.description || '')}</textarea>
</div>
<div class="mb-3">
<label for="editTags-${imageId}" class="form-label">Tags</label>
<input type="text" class="form-control" id="editTags-${imageId}"
value="${image.tags ? image.tags.join(', ') : ''}">
<div class="form-text">Enter tags separated by commas</div>
</div>
</form>
`;
const modalFooter = `
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Cancel</button>
<button type="button" class="btn btn-primary" onclick="saveImageChanges('${imageId}')">
<i class="fas fa-save me-1"></i>Save Changes
</button>
`;
const modal = createModal(modalId, 'Edit Image', modalBody, modalFooter);
// Add cleanup when modal is hidden
const modalElement = document.getElementById(modalId);
modalElement.addEventListener('hidden.bs.modal', () => {
removeModal(modalId);
});
modal.show();
} catch (error) {
handleApiError(error, 'loading image for editing');
}
}
// Save image changes
async function saveImageChanges(imageId) {
const description = document.getElementById(`editDescription-${imageId}`).value.trim();
const tagsInput = document.getElementById(`editTags-${imageId}`).value.trim();
const tags = tagsInput ? tagsInput.split(',').map(tag => tag.trim()).filter(tag => tag) : [];
const saveButton = document.querySelector(`#editImageModal-${imageId} .btn-primary`);
setLoadingState(saveButton);
try {
await apiClient.updateImage(imageId, {
description,
tags
});
showAlert('Image updated successfully!', 'success');
// Close modal and refresh images
bootstrap.Modal.getInstance(document.getElementById(`editImageModal-${imageId}`)).hide();
removeModal(`editImageModal-${imageId}`);
loadImages(currentPage);
} catch (error) {
handleApiError(error, 'updating image');
} finally {
setLoadingState(saveButton, false);
}
}
// Delete image
function deleteImage(imageId) {
confirmAction('Are you sure you want to delete this image? This action cannot be undone.', async () => {
try {
await apiClient.deleteImage(imageId);
showAlert('Image deleted successfully!', 'success');
loadImages(currentPage);
// Close any open modals for this image
const modalsToClose = [`viewImageModal-${imageId}`, `editImageModal-${imageId}`];
modalsToClose.forEach(modalId => {
const modalElement = document.getElementById(modalId);
if (modalElement) {
const bsModal = bootstrap.Modal.getInstance(modalElement);
if (bsModal) {
bsModal.hide();
}
removeModal(modalId);
}
});
} catch (error) {
handleApiError(error, 'deleting image');
}
});
}