Coppermine Gallery Downloader

Download galleries powered by Coppermine Photo Gallery

// ==UserScript==
// @name         Coppermine Gallery Downloader
// @namespace    https://github.com/xanthisafk/coppermine-gallery-downloader
// @version      1.3
// @description  Download galleries powered by Coppermine Photo Gallery
// @author       Abhinav
// @license      MIT; https://github.com/xanthisafk/coppermine-gallery-downloader/blob/main/LICENSE
// @match        *://*/thumbnails.php?*
// @match        *://*/displayimage.php?*
// @grant        GM_xmlhttpRequest
// @require      https://cdnjs.cloudflare.com/ajax/libs/jszip/3.1.5/jszip.min.js
// @require      https://unpkg.com/[email protected]/dist/FileSaver.min.js
// @icon         
// ==/UserScript==

// Icon: "Simple pickaxe" by ramaskrik is in the Public Domain, CC0. (https://opengameart.org/content/simple-pickaxe)

/* globals JSZip saveAs */
(function () {
    'use strict';

    // Create and style the download button
    const downloadButton = document.createElement('button');
    downloadButton.textContent = '💾 Download';
    downloadButton.style.position = 'fixed';
    downloadButton.style.bottom = '10px';
    downloadButton.style.left = '10px';
    downloadButton.style.zIndex = '1000';
    downloadButton.style.padding = '10px';
    downloadButton.style.backgroundColor = '#ff4081';
    downloadButton.style.color = '#fff';
    downloadButton.style.border = 'none';
    downloadButton.style.borderRadius = '5px';
    downloadButton.style.cursor = 'pointer';
    downloadButton.style.fontSize = '16px';
    document.body.appendChild(downloadButton);

    // Create and style a status indicator
    const statusIndicator = document.createElement('div');
    statusIndicator.style.position = 'fixed';
    statusIndicator.style.bottom = '60px';
    statusIndicator.style.left = '10px';
    statusIndicator.style.zIndex = '1000';
    statusIndicator.style.padding = '5px';
    statusIndicator.style.backgroundColor = 'rgba(0, 0, 0, 0.7)';
    statusIndicator.style.color = '#fff';
    statusIndicator.style.borderRadius = '3px';
    statusIndicator.style.display = 'none';
    document.body.appendChild(statusIndicator);

    /**
     * Utility to update the status message on the UI.
     * @param {string} message - The status message to display.
     */
    function updateStatus(message) {
        statusIndicator.textContent = message;
        statusIndicator.style.display = 'block';
        console.log(message);
    }

    /**
     * Fetch an image using GM_xmlhttpRequest and return it as a Promise.
     * @param {string} imageUrl - The URL of the image to fetch.
     * @returns {Promise<ArrayBuffer>} - A Promise resolving to the image data.
     */
    function fetchImageAsArrayBuffer(imageUrl) {
        return new Promise((resolve, reject) => {
            GM_xmlhttpRequest({
                method: 'GET',
                url: imageUrl,
                responseType: 'arraybuffer',
                onload(response) {
                    if (response.status === 200) {
                        resolve(response.response);
                    } else {
                        reject(new Error(`Failed to fetch image from ${imageUrl}`));
                    }
                },
                onerror() {
                    reject(new Error(`Network error while fetching ${imageUrl}`));
                }
            });
        });
    }

    /**
     * Download a single image on the page.
     */
    async function downloadSingleImage() {
        try {
            downloadButton.disabled = true;
            updateStatus('Preparing to download image...');

            const imageElement = document.querySelector('.image');
            if (!imageElement) {
                throw new Error('Image element not found on this page.');
            }

            const rawImageUrl = imageElement.src.replace('normal_', '');
            const imageFileName = rawImageUrl.split('/').pop();

            // Naming convention: hostname-[image file name].[ext]
            const hostname = window.location.hostname.split('.')[0];
            const formattedFileName = `${hostname}-${imageFileName}`;

            updateStatus('Downloading image...');
            const imageData = await fetchImageAsArrayBuffer(rawImageUrl);

            // Save the downloaded image
            const imageBlob = new Blob([imageData], { type: 'image/jpeg' });
            saveAs(imageBlob, formattedFileName);

            updateStatus('Image download complete!');
        } catch (error) {
            updateStatus(`Error: ${error.message}`);
        } finally {
            setTimeout(() => {
                statusIndicator.style.display = 'none';
                downloadButton.disabled = false;
            }, 3000);
        }
    }

    /**
     * Download all images in a gallery as a ZIP file.
     */
    async function downloadGalleryAsZip() {
        try {
            downloadButton.disabled = true;
            updateStatus('Preparing to download gallery...');

            const galleryTitleElement = document.querySelector('.maintable h2');
            const galleryTitle = galleryTitleElement ? galleryTitleElement.textContent.trim() : 'Gallery';
            const hostname = window.location.hostname.split('.')[0];
            const zipFileName = `${hostname}-${galleryTitle}.zip`;

            const zip = new JSZip();
            const thumbnailElements = document.querySelectorAll('.thumbnail');
            const imageUrls = [];

            // Extract and clean up image URLs
            thumbnailElements.forEach(thumbnail => {
                let rawImageUrl = thumbnail.src;
                if (rawImageUrl.includes('thumb_')) {
                    rawImageUrl = rawImageUrl.replace('thumb_', '');
                    const imageFileName = rawImageUrl.split('/').pop();
                    const formattedFileName = `${hostname}-${imageFileName}`;
                    imageUrls.push({ url: rawImageUrl, fileName: formattedFileName });
                }
            });

            if (imageUrls.length === 0) {
                throw new Error('No images found in the gallery.');
            }

            updateStatus(`Found ${imageUrls.length} images. Starting download...`);

            // Download each image and add to ZIP
            for (let i = 0; i < imageUrls.length; i++) {
                const { url, fileName } = imageUrls[i];
                updateStatus(`Downloading image ${i + 1} of ${imageUrls.length}...`);

                try {
                    const imageData = await fetchImageAsArrayBuffer(url);
                    zip.file(fileName, imageData, { binary: true });
                } catch (error) {
                    console.error(`Failed to download ${fileName}:`, error);
                }
            }

            updateStatus('Creating ZIP file...');
            const zipBlob = await zip.generateAsync({
                type: 'blob',
                compression: 'DEFLATE',
                compressionOptions: { level: 6 }
            });

            updateStatus('Downloading ZIP file...');
            saveAs(zipBlob, zipFileName);

            updateStatus('Gallery download complete!');
        } catch (error) {
            updateStatus(`Error: ${error.message}`);
        } finally {
            setTimeout(() => {
                statusIndicator.style.display = 'none';
                downloadButton.disabled = false;
            }, 3000);
        }
    }

    // Attach appropriate click handlers based on the page type
    const isSingleImagePage = window.location.href.includes('displayimage.php');
    if (isSingleImagePage) {
        downloadButton.textContent = '💾 Download Image';
        downloadButton.addEventListener('click', downloadSingleImage);
    } else {
        downloadButton.textContent = '💾 Download Gallery';
        downloadButton.addEventListener('click', downloadGalleryAsZip);
    }
})();