commit 829996b41eb8084b483de4d67f05e558078b0bd6 Author: renato97 Date: Tue Nov 4 04:25:25 2025 +0000 Initial commit: Manga Mass Downloader Chrome Extension ✨ Features: - Multi-selection checkboxes on manga listings - Batch download selected manga or all manga from page - Optimized parallel downloading (20ms delays, 5 concurrent) - Visual progress tracking - Popup UI for easy control - Fixed duplicate checkbox issue with deduplication logic 📁 Files: - manifest.json: Extension configuration - content.js: Checkbox injection & manga detection - background.js: Optimized download engine - popup.html/js: User interface - README.md: Complete documentation diff --git a/README.md b/README.md new file mode 100644 index 0000000..2f29bb6 --- /dev/null +++ b/README.md @@ -0,0 +1,107 @@ +# Manga Mass Downloader + +Chrome Extension para descarga masiva de manga desde listados de e-hentai.org + +## ✨ Características + +- ✅ **Selección múltiple**: Agrega checkboxes a cada miniatura de manga +- ✅ **Descarga individual**: Selecciona 5-10 manga y descárgalos uno por uno +- ✅ **Descarga masiva**: Descarga TODOS los manga de una página de una vez +- ✅ **Basado en addon optimizado**: Utiliza el motor de descarga paralelo ultra-rápido +- ✅ **Barra de progreso**: Visualiza el progreso en tiempo real +- ✅ **Interfaz intuitiva**: Popup con controles fáciles de usar + +## 🚀 Cómo usar + +### 1. Instalación +- Ve a `chrome://extensions/` +- Activa "Modo de desarrollador" +- Carga la carpeta del addon + +### 2. Seleccionar Manga +- Ve a https://e-hentai.org/?f_cats=1019&f_search=spanish (o cualquier listado) +- Las miniaturas tendrán un checkbox "Descargar" en la esquina superior izquierda +- Haz clic para seleccionar los manga que quieres +- Usa el botón "Seleccionar Todos" en la esquina inferior izquierda + +### 3. Descargar +- Haz clic en el ícono del addon en la barra de herramientas +- **Opción 1**: "Descargar Seleccionados" - Descarga solo los marcados +- **Opción 2**: "Descargar TODOS de la Página" - Descarga todos los manga visibles +- **Opción 3**: "Limpiar Selección" - Deselecciona todo + +## 📁 Estructura del proyecto + +``` +mass-downloader/ +├── manifest.json # Configuración de la extensión +├── content.js # Content script para detectar miniaturas +├── background.js # Service worker para descargas (optimizado) +├── popup.html # Interfaz del popup +├── popup.js # Lógica del popup +├── jszip.min.js # Librería para crear archivos ZIP +└── README.md # Este archivo +``` + +## 🎯 Funcionamiento + +### Content Script (`content.js`) +- Detecta miniaturas de manga en la página +- Agrega checkboxes interactivos +- Extrae metadata (título, ID, token, URL) +- Permite selección múltiple + +### Background Script (`background.js`) +- Descarga imágenes en lotes paralelos (5 simultáneas) +- Delay fijo de 20ms entre descargas +- Crea archivos ZIP optimizados +- Basado en el addon original ultra-optimizado + +### Popup (`popup.html` + `popup.js`) +- Contador de manga seleccionados +- 3 botones principales de descarga +- Barra de progreso visual +- Feedback de estado en tiempo real + +## ⚡ Optimizaciones + +- **Velocidad extrema**: Descarga en paralelo con delays mínimos +- **Rate limiting controlado**: 20ms delay fijo +- **Validación robusta**: Filtra imágenes válidas +- **Compresión ZIP**: Nivel 6 DEFLATE para mejor ratio +- **Nombres limpios**: Sanitización de títulos + +## 🔧 Diferencias con el addon original + +| Característica | Manga Downloader | Manga Mass Downloader | +|---|---|---| +| **Página objetivo** | Páginas de galería (`/g/`) | Listados de manga | +| **Selección** | Un manga a la vez | Múltiples manga | +| **Descarga** | Individual | Individual y masiva | +| **UI** | Botón flotante | Popup + checkboxes | +| **Scope** | Manga específico | Lista de manga | + +## 📝 Notas + +- Solo funciona en e-hentai.org y exhentai.org +- Requiere credenciales de sesión (cookie de login) +- Cada descarga abre un diálogo de "Guardar como" +- Las descargas se procesan secuencialmente para evitar saturar el servidor + +## 🐛 Solución de problemas + +**No aparecen los checkboxes:** +- Recarga la página +- Verifica que estés en una página de listado (no en `/g/`) + +**Error al descargar:** +- Verifica que estés logueado en e-hentai +- Revisa la consola (F12) para logs detallados + +**Descarga muy lenta:** +- Es normal, se procesa secuencialmente +- Las primeras descargas tardan más + +## 📄 Licencia + +MIT License - Uso personal diff --git a/background.js b/background.js new file mode 100644 index 0000000..a93002c --- /dev/null +++ b/background.js @@ -0,0 +1,278 @@ +// Background Service Worker para Manga Mass Downloader +// Basado en el addon original optimizado + +importScripts('jszip.min.js'); + +chrome.runtime.onMessage.addListener((request, sender, sendResponse) => { + if (request.action === 'downloadManga') { + downloadManga(request.metadata, request.imageUrls) + .then(result => sendResponse({ success: true, data: result })) + .catch(error => sendResponse({ success: false, error: error.message })); + return true; + } +}); + +// Función principal para descargar el manga +async function downloadManga(metadata, imageUrls) { + console.log('Iniciando descarga del manga:', metadata.title); + + try { + // 1. Descargar todas las imágenes + const images = await downloadAllImages(imageUrls, metadata.title); + + // 2. Crear archivo ZIP + const zipBlob = await createZipArchive(images, metadata.title); + + // 3. Descargar el archivo + await downloadZipFile(zipBlob, metadata.title); + + return { + success: true, + title: metadata.title, + totalImages: images.length + }; + + } catch (error) { + console.error('Error en descarga:', error); + throw error; + } +} + +// Descargar todas las imágenes en PARALELO (máxima velocidad) +async function downloadAllImages(imageUrls, title) { + console.log(`🚀 Descargando ${imageUrls.length} imágenes en paralelo...`); + + // Procesar en lotes de 5 imágenes simultáneas + const batchSize = 5; + const allImages = []; + + for (let i = 0; i < imageUrls.length; i += batchSize) { + const batch = imageUrls.slice(i, i + batchSize); + console.log(`\n=== Procesando lote ${Math.floor(i/batchSize) + 1}: imágenes ${i + 1}-${i + batch.length}`); + + // Crear promesas para el lote actual + const batchPromises = batch.map(async (imageInfo, batchIndex) => { + const globalIndex = i + batchIndex; + try { + // Delay fijo de 20ms + await new Promise(resolve => setTimeout(resolve, 20)); + + const imageData = await downloadImage(imageInfo, title); + console.log(`✓ Lote ${Math.floor(i/batchSize) + 1}: Descargada ${globalIndex + 1}/${imageUrls.length}`); + + // Enviar progreso + chrome.runtime.sendMessage({ + action: 'downloadProgress', + progress: { + current: globalIndex + 1, + total: imageUrls.length, + filename: imageData.filename, + errors: 0 + } + }).catch(() => {}); + + return imageData; + } catch (error) { + console.error(`✗ Error en imagen ${globalIndex + 1}:`, error.message); + return { error: true, index: globalIndex, message: error.message }; + } + }); + + // Esperar a que termine todo el lote + const batchResults = await Promise.all(batchPromises); + const successful = batchResults.filter(r => !r.error); + + allImages.push(...successful); + + console.log(`✓ Lote ${Math.floor(i/batchSize) + 1} completado: ${successful.length}/${batch.length} imágenes`); + } + + console.log(`\n✅ Descarga paralela completada: ${allImages.length}/${imageUrls.length} imágenes`); + return allImages; +} + +// Descargar una imagen individual +async function downloadImage(imageInfo, title) { + const { url, index } = imageInfo; + + try { + // Hacer fetch de la página /s/ + const response = await fetch(url, { + credentials: 'include' + }); + + if (!response.ok) { + throw new Error(`Error ${response.status} al acceder a la página ${index + 1}`); + } + + const html = await response.text(); + + // Extraer URL real de imagen + const imageUrl = extractImageUrlFromHtml(html); + + if (!imageUrl) { + throw new Error(`No se encontró imagen en la página ${index + 1}`); + } + + // Hacer fetch de la imagen real + const imageResponse = await fetch(imageUrl, { + credentials: 'include' + }); + + if (!imageResponse.ok) { + throw new Error(`Error ${imageResponse.status} al descargar imagen ${index + 1}`); + } + + const blob = await imageResponse.blob(); + + // Validación + if (blob.size < 500) { + throw new Error(`Imagen demasiado pequeña: ${blob.size} bytes`); + } + + if (!blob.type || !blob.type.startsWith('image/')) { + throw new Error(`Tipo de archivo inválido: ${blob.type}`); + } + + const extension = 'jpg'; + const paddedIndex = String(index + 1).padStart(3, '0'); + + return { + blob: blob, + filename: `page_${paddedIndex}.${extension}`, + index: index + }; + + } catch (error) { + console.error(`Error en imagen ${index + 1}:`, error.message); + throw error; + } +} + +// Extraer URL de imagen desde HTML +function extractImageUrlFromHtml(html) { + try { + // Método 1: img#img + let match = html.match(/]+id=["']img["'][^>]*>/i); + if (match) { + const srcMatch = match[0].match(/src=["']([^"']+)["']/i); + if (srcMatch && srcMatch[1]) return srcMatch[1]; + } + + // Método 2: img con hath + match = html.match(/]*src=["']([^"']*hath[^"']*)["'][^>]*>/i); + if (match) { + const srcMatch = match[0].match(/src=["']([^"']+)["']/i); + if (srcMatch && srcMatch[1]) return srcMatch[1]; + } + + // Método 3: Cualquier img + match = html.match(/]+src=["']([^"']+)["'][^>]*>/i); + if (match) { + const srcMatch = match[0].match(/src=["']([^"']+)["']/i); + if (srcMatch && srcMatch[1]) return srcMatch[1]; + } + + // Método 4: data-src + match = html.match(/<[^>]+data-src=["']([^"']+)["'][^>]*>/i); + if (match) { + const dataSrcMatch = match[0].match(/data-src=["']([^"']+)["']/i); + if (dataSrcMatch && dataSrcMatch[1]) return dataSrcMatch[1]; + } + + // Método 5: style background + match = html.match(/<[^>]+style=["'][^"']*background[^"']*url\(["']?([^"')]+)["']?\)[^"']*["'][^>]*>/i); + if (match) { + const imageUrl = match[1]; + const isValidImage = imageUrl.includes('hath') || imageUrl.match(/\.(jpg|jpeg|png|gif|webp)/i); + if (isValidImage) return imageUrl; + } + + // Método 6: srcset + match = html.match(/]+srcset=["']([^"']+)["'][^>]*>/i); + if (match) { + const urls = match[1].split(',')[0].trim(); + const urlMatch = urls.match(/(https?:\/\/\S+)/); + if (urlMatch && urlMatch[1]) return urlMatch[1]; + } + + // Método 7: URLs directas hath + match = html.match(/(https?:\/\/[^"'\s]*hath\.network[^"'\s]*)/i); + if (match && match[1]) return match[1]; + + // Método 8: Fallback + match = html.match(/(https?:\/\/[^"'\s]*\.(jpg|jpeg|png|gif|webp))/i); + if (match && match[1]) return match[1]; + + return null; + } catch (error) { + console.error('Error extrayendo URL de imagen:', error); + return null; + } +} + +// Crear archivo ZIP +async function createZipArchive(images, title) { + console.log('\n========== CREANDO ZIP =========='); + console.log('Total de imágenes a comprimir:', images.length); + + const cleanTitle = title + .replace(/[<>:"/\\|?*]/g, '_') + .replace(/\s+/g, '_') + .substring(0, 100); + + if (images.length === 0) { + throw new Error('No hay imágenes para comprimir'); + } + + const zip = new JSZip(); + + images.forEach((imageData, i) => { + if (imageData.blob && imageData.blob.size > 0) { + zip.file(imageData.filename, imageData.blob); + } + }); + + const zipBlob = await zip.generateAsync({ + type: 'blob', + compression: 'DEFLATE', + compressionOptions: { level: 6 } + }); + + console.log(`✓ ZIP generado: ${zipBlob.size} bytes`); + return zipBlob; +} + +// Descargar archivo ZIP +async function downloadZipFile(zipBlob, title) { + let cleanTitle = title || 'Manga'; + cleanTitle = cleanTitle.replace(/[<>:"/\\|?*]/g, '_'); + cleanTitle = cleanTitle.replace(/[\x00-\x1F\x7F]/g, ''); + cleanTitle = cleanTitle.replace(/\s+/g, '_'); + cleanTitle = cleanTitle.replace(/_+/g, '_'); + cleanTitle = cleanTitle.substring(0, 80); + + if (!cleanTitle || cleanTitle.length === 0 || cleanTitle === '.') { + cleanTitle = 'Manga'; + } + if (cleanTitle.startsWith('.')) { + cleanTitle = 'Manga_' + cleanTitle; + } + + const filename = `${cleanTitle}.zip`; + + const arrayBuffer = await zipBlob.arrayBuffer(); + const uint8Array = new Uint8Array(arrayBuffer); + let binary = ''; + for (let i = 0; i < uint8Array.length; i++) { + binary += String.fromCharCode(uint8Array[i]); + } + const base64 = btoa(binary); + const dataUrl = `data:application/zip;base64,${base64}`; + + await chrome.downloads.download({ + url: dataUrl, + filename: filename, + saveAs: true + }); +} diff --git a/content.js b/content.js new file mode 100644 index 0000000..78174c2 --- /dev/null +++ b/content.js @@ -0,0 +1,310 @@ +// Content Script para Manga Mass Downloader +// Basado en addon original - SIMPLIFICADO + +(function() { + 'use strict'; + + let selectedMangas = new Set(); + let checkboxesAdded = false; + + // Escuchar mensajes del popup + chrome.runtime.onMessage.addListener((request, sender, sendResponse) => { + if (request.action === 'getSelectedMangas') { + const selected = Array.from(selectedMangas); + sendResponse({ mangas: selected }); + } else if (request.action === 'clearSelection') { + selectedMangas.clear(); + updateSelectedCount(); + sendResponse({ success: true }); + } else if (request.action === 'getImageUrls') { + getImageUrlsForManga(request.manga) + .then(imageUrls => sendResponse({ imageUrls })) + .catch(error => sendResponse({ error: error.message })); + return true; + } else if (request.action === 'extractAllMangas') { + const allMangas = extractAllMangasFromPage(); + sendResponse({ mangas: allMangas }); + } + return true; + }); + + // Extraer TODOS los manga de la página + function extractAllMangasFromPage() { + const mangas = []; + const links = document.querySelectorAll('a[href*="/g/"]'); + + links.forEach(link => { + const href = link.href; + const match = href.match(/\/g\/(\d+)\/([a-f0-9]+)/); + if (match) { + const title = link.textContent.trim() || 'Untitled'; + mangas.push({ + id: match[1], + token: match[2], + title: title, + url: href, + baseUrl: `https://e-hentai.org/g/${match[1]}/${match[2]}/` + }); + } + }); + + return mangas; + } + + // Obtener URLs de imágenes para un manga + async function getImageUrlsForManga(manga) { + const imageUrls = []; + const baseUrl = manga.baseUrl; + + try { + const response = await fetch(baseUrl, { + credentials: 'include' + }); + + if (!response.ok) { + throw new Error(`Error ${response.status}`); + } + + const html = await response.text(); + const parser = new DOMParser(); + const doc = parser.parseFromString(html, 'text/html'); + + // Detectar páginas + let actualTotalPages = 1; + const pageInfo = doc.querySelector('.gpc, .gt, #gdn + span'); + if (pageInfo) { + const pageText = pageInfo.textContent.trim(); + const pageMatch = pageText.match(/Showing\s+1\s*-\s*\d+\s+of\s+(\d+)\s+images/i); + if (pageMatch) { + const totalImages = parseInt(pageMatch[1]); + actualTotalPages = Math.ceil(totalImages / 20); + } + } + + // Procesar todas las páginas + for (let page = 1; page <= actualTotalPages; page++) { + const pageUrl = page === 1 ? baseUrl : `${baseUrl}?p=${page}`; + const pageResponse = await fetch(pageUrl, { + credentials: 'include' + }); + + if (!pageResponse.ok) continue; + + const pageHtml = await pageResponse.text(); + const pageDoc = parser.parseFromString(pageHtml, 'text/html'); + + const links = pageDoc.querySelectorAll('a[href*="/s/"]'); + links.forEach(link => { + const href = link.href; + if (href && href.includes('/s/')) { + imageUrls.push({ + url: href, + index: imageUrls.length + }); + } + }); + + if (page < actualTotalPages) { + await new Promise(resolve => setTimeout(resolve, 20)); + } + } + + return imageUrls; + } catch (error) { + console.error('Error:', error); + throw error; + } + } + + // Agregar checkbox a enlaces de galerías + function addCheckboxes() { + console.log('\n==== addCheckboxes() INICIADO ===='); + console.log('Timestamp:', new Date().toLocaleTimeString()); + + const existingCheckboxes = document.querySelectorAll('.mass-downloader-checkbox').length; + console.log('📊 Checkboxes existentes antes de limpiar:', existingCheckboxes); + + // SOLUCIÓN DEFINITIVA: Limpiar TODOS los checkboxes existentes + document.querySelectorAll('.mass-downloader-checkbox').forEach(cb => cb.remove()); + selectedMangas.clear(); + + console.log('🧹 Checkboxes limpiados'); + + // Obtener TODOS los enlaces + const allLinks = document.querySelectorAll('a[href*="/g/"]'); + console.log('🔍 Total enlaces encontrados:', allLinks.length); + + // Log detallado de todos los enlaces + const allLinkData = []; + allLinks.forEach((link, index) => { + const match = link.href.match(/\/g\/(\d+)\/([a-f0-9]+)/); + if (match) { + allLinkData.push({ + index: index, + id: match[1], + href: link.href.substring(0, 60) + '...', + parentTag: link.parentElement.tagName, + parentClass: link.parentElement.className + }); + } + }); + + console.log('📋 Detalle de enlaces (primeros 5):', allLinkData.slice(0, 5)); + + // DEDUPLICACIÓN ROBUSTA: Trackear IDs únicos durante la iteración + const seenIds = new Set(); + const uniqueLinks = []; + + allLinks.forEach(link => { + const match = link.href.match(/\/g\/(\d+)/); + if (match) { + const galleryId = match[1]; + if (!seenIds.has(galleryId)) { + seenIds.add(galleryId); + uniqueLinks.push(link); + console.log(` ➕ ID único encontrado: ${galleryId} (href: ${link.href.substring(0, 50)}...)`); + } else { + console.log(` ⏭️ ID duplicado omitido: ${galleryId}`); + } + } + }); + + console.log('✅ Total IDs únicos (sin duplicados):', uniqueLinks.length); + console.log('✅ Set "seenIds" size:', seenIds.size); + + // Crear checkboxes SOLO para enlaces únicos + let count = 0; + console.log('\n🎯 CREANDO CHECKBOXES:'); + uniqueLinks.forEach((link, index) => { + const match = link.href.match(/\/g\/(\d+)\/([a-f0-9]+)/); + const parent = link.parentElement; + + // Verificar que este parent no tenga ya un checkbox + if (parent.querySelector('.mass-downloader-checkbox')) { + console.log(` ⚠️ [${index}] Parent ya tiene checkbox, saltando...`); + return; + } + + console.log(` 📝 [${index}] Creando checkbox para manga ID: ${match ? match[1] : 'N/A'}`); + + // Crear checkbox + const container = document.createElement('div'); + container.className = 'mass-downloader-checkbox'; + container.style.cssText = ` + position: absolute; + top: 5px; + left: 5px; + z-index: 9999; + background: rgba(0,0,0,0.8); + padding: 4px 8px; + border-radius: 4px; + `; + + const checkbox = document.createElement('input'); + checkbox.type = 'checkbox'; + checkbox.style.cssText = 'width: 16px; height: 16px;'; + + const label = document.createElement('label'); + label.textContent = 'Select'; + label.style.cssText = 'color: white; font-size: 11px; margin-left: 4px;'; + + checkbox.addEventListener('change', (e) => { + e.stopPropagation(); + const match = link.href.match(/\/g\/(\d+)\/([a-f0-9]+)/); + if (match) { + if (checkbox.checked) { + selectedMangas.add(match[1]); + } else { + selectedMangas.delete(match[1]); + } + updateSelectedCount(); + } + }); + + container.appendChild(checkbox); + container.appendChild(label); + + // Agregar al parent + if (getComputedStyle(parent).position === 'static') { + parent.style.position = 'relative'; + } + parent.appendChild(container); + count++; + console.log(` ✅ Checkbox ${count} agregado y appendeado al parent <${parent.tagName}>`); + }); + + console.log('\n==== RESUMEN FINAL ===='); + console.log('🎉 Total checkboxes CREADOS:', count); + console.log('📊 Total checkboxes en DOM:', document.querySelectorAll('.mass-downloader-checkbox').length); + console.log('📊 Set selectedMangas size:', selectedMangas.size); + console.log('==== addCheckboxes() COMPLETADO ====\n'); + + checkboxesAdded = true; + } + + function updateSelectedCount() { + chrome.runtime.sendMessage({ + action: 'updateSelectedCount', + count: selectedMangas.size + }).catch(() => {}); + } + + function addSelectAllButton() { + if (document.querySelector('.mass-downloader-select-all')) return; + + const button = document.createElement('button'); + button.className = 'mass-downloader-select-all'; + button.textContent = 'Select All'; + button.style.cssText = ` + position: fixed; + bottom: 20px; + left: 20px; + padding: 10px 20px; + background: #667eea; + color: white; + border: none; + border-radius: 6px; + cursor: pointer; + z-index: 10000; + `; + + let allSelected = false; + button.addEventListener('click', () => { + allSelected = !allSelected; + const checkboxes = document.querySelectorAll('.mass-downloader-checkbox input'); + + if (allSelected) { + checkboxes.forEach(cb => { + cb.checked = true; + const match = cb.closest('div').parentElement.querySelector('a[href*="/g/"]').href.match(/\/g\/(\d+)/); + if (match) selectedMangas.add(match[1]); + }); + button.textContent = 'Deselect All'; + } else { + checkboxes.forEach(cb => cb.checked = false); + selectedMangas.clear(); + button.textContent = 'Select All'; + } + + updateSelectedCount(); + }); + + document.body.appendChild(button); + } + + function init() { + if (window.location.href.includes('/g/')) return; + + setTimeout(() => { + addCheckboxes(); + addSelectAllButton(); + }, 1000); + } + + if (document.readyState === 'loading') { + document.addEventListener('DOMContentLoaded', init); + } else { + init(); + } + +})(); diff --git a/jszip.min.js b/jszip.min.js new file mode 100644 index 0000000..ff4cfd5 --- /dev/null +++ b/jszip.min.js @@ -0,0 +1,13 @@ +/*! + +JSZip v3.10.1 - A JavaScript class for generating and reading zip files + + +(c) 2009-2016 Stuart Knightley +Dual licenced under the MIT license or GPLv3. See https://raw.github.com/Stuk/jszip/main/LICENSE.markdown. + +JSZip uses the library pako released under the MIT license : +https://github.com/nodeca/pako/blob/main/LICENSE +*/ + +!function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{("undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self?self:this).JSZip=e()}}(function(){return function s(a,o,h){function u(r,e){if(!o[r]){if(!a[r]){var t="function"==typeof require&&require;if(!e&&t)return t(r,!0);if(l)return l(r,!0);var n=new Error("Cannot find module '"+r+"'");throw n.code="MODULE_NOT_FOUND",n}var i=o[r]={exports:{}};a[r][0].call(i.exports,function(e){var t=a[r][1][e];return u(t||e)},i,i.exports,s,a,o,h)}return o[r].exports}for(var l="function"==typeof require&&require,e=0;e>2,s=(3&t)<<4|r>>4,a=1>6:64,o=2>4,r=(15&i)<<4|(s=p.indexOf(e.charAt(o++)))>>2,n=(3&s)<<6|(a=p.indexOf(e.charAt(o++))),l[h++]=t,64!==s&&(l[h++]=r),64!==a&&(l[h++]=n);return l}},{"./support":30,"./utils":32}],2:[function(e,t,r){"use strict";var n=e("./external"),i=e("./stream/DataWorker"),s=e("./stream/Crc32Probe"),a=e("./stream/DataLengthProbe");function o(e,t,r,n,i){this.compressedSize=e,this.uncompressedSize=t,this.crc32=r,this.compression=n,this.compressedContent=i}o.prototype={getContentWorker:function(){var e=new i(n.Promise.resolve(this.compressedContent)).pipe(this.compression.uncompressWorker()).pipe(new a("data_length")),t=this;return e.on("end",function(){if(this.streamInfo.data_length!==t.uncompressedSize)throw new Error("Bug : uncompressed data size mismatch")}),e},getCompressedWorker:function(){return new i(n.Promise.resolve(this.compressedContent)).withStreamInfo("compressedSize",this.compressedSize).withStreamInfo("uncompressedSize",this.uncompressedSize).withStreamInfo("crc32",this.crc32).withStreamInfo("compression",this.compression)}},o.createWorkerFrom=function(e,t,r){return e.pipe(new s).pipe(new a("uncompressedSize")).pipe(t.compressWorker(r)).pipe(new a("compressedSize")).withStreamInfo("compression",t)},t.exports=o},{"./external":6,"./stream/Crc32Probe":25,"./stream/DataLengthProbe":26,"./stream/DataWorker":27}],3:[function(e,t,r){"use strict";var n=e("./stream/GenericWorker");r.STORE={magic:"\0\0",compressWorker:function(){return new n("STORE compression")},uncompressWorker:function(){return new n("STORE decompression")}},r.DEFLATE=e("./flate")},{"./flate":7,"./stream/GenericWorker":28}],4:[function(e,t,r){"use strict";var n=e("./utils");var o=function(){for(var e,t=[],r=0;r<256;r++){e=r;for(var n=0;n<8;n++)e=1&e?3988292384^e>>>1:e>>>1;t[r]=e}return t}();t.exports=function(e,t){return void 0!==e&&e.length?"string"!==n.getTypeOf(e)?function(e,t,r,n){var i=o,s=n+r;e^=-1;for(var a=n;a>>8^i[255&(e^t[a])];return-1^e}(0|t,e,e.length,0):function(e,t,r,n){var i=o,s=n+r;e^=-1;for(var a=n;a>>8^i[255&(e^t.charCodeAt(a))];return-1^e}(0|t,e,e.length,0):0}},{"./utils":32}],5:[function(e,t,r){"use strict";r.base64=!1,r.binary=!1,r.dir=!1,r.createFolders=!0,r.date=null,r.compression=null,r.compressionOptions=null,r.comment=null,r.unixPermissions=null,r.dosPermissions=null},{}],6:[function(e,t,r){"use strict";var n=null;n="undefined"!=typeof Promise?Promise:e("lie"),t.exports={Promise:n}},{lie:37}],7:[function(e,t,r){"use strict";var n="undefined"!=typeof Uint8Array&&"undefined"!=typeof Uint16Array&&"undefined"!=typeof Uint32Array,i=e("pako"),s=e("./utils"),a=e("./stream/GenericWorker"),o=n?"uint8array":"array";function h(e,t){a.call(this,"FlateWorker/"+e),this._pako=null,this._pakoAction=e,this._pakoOptions=t,this.meta={}}r.magic="\b\0",s.inherits(h,a),h.prototype.processChunk=function(e){this.meta=e.meta,null===this._pako&&this._createPako(),this._pako.push(s.transformTo(o,e.data),!1)},h.prototype.flush=function(){a.prototype.flush.call(this),null===this._pako&&this._createPako(),this._pako.push([],!0)},h.prototype.cleanUp=function(){a.prototype.cleanUp.call(this),this._pako=null},h.prototype._createPako=function(){this._pako=new i[this._pakoAction]({raw:!0,level:this._pakoOptions.level||-1});var t=this;this._pako.onData=function(e){t.push({data:e,meta:t.meta})}},r.compressWorker=function(e){return new h("Deflate",e)},r.uncompressWorker=function(){return new h("Inflate",{})}},{"./stream/GenericWorker":28,"./utils":32,pako:38}],8:[function(e,t,r){"use strict";function A(e,t){var r,n="";for(r=0;r>>=8;return n}function n(e,t,r,n,i,s){var a,o,h=e.file,u=e.compression,l=s!==O.utf8encode,f=I.transformTo("string",s(h.name)),c=I.transformTo("string",O.utf8encode(h.name)),d=h.comment,p=I.transformTo("string",s(d)),m=I.transformTo("string",O.utf8encode(d)),_=c.length!==h.name.length,g=m.length!==d.length,b="",v="",y="",w=h.dir,k=h.date,x={crc32:0,compressedSize:0,uncompressedSize:0};t&&!r||(x.crc32=e.crc32,x.compressedSize=e.compressedSize,x.uncompressedSize=e.uncompressedSize);var S=0;t&&(S|=8),l||!_&&!g||(S|=2048);var z=0,C=0;w&&(z|=16),"UNIX"===i?(C=798,z|=function(e,t){var r=e;return e||(r=t?16893:33204),(65535&r)<<16}(h.unixPermissions,w)):(C=20,z|=function(e){return 63&(e||0)}(h.dosPermissions)),a=k.getUTCHours(),a<<=6,a|=k.getUTCMinutes(),a<<=5,a|=k.getUTCSeconds()/2,o=k.getUTCFullYear()-1980,o<<=4,o|=k.getUTCMonth()+1,o<<=5,o|=k.getUTCDate(),_&&(v=A(1,1)+A(B(f),4)+c,b+="up"+A(v.length,2)+v),g&&(y=A(1,1)+A(B(p),4)+m,b+="uc"+A(y.length,2)+y);var E="";return E+="\n\0",E+=A(S,2),E+=u.magic,E+=A(a,2),E+=A(o,2),E+=A(x.crc32,4),E+=A(x.compressedSize,4),E+=A(x.uncompressedSize,4),E+=A(f.length,2),E+=A(b.length,2),{fileRecord:R.LOCAL_FILE_HEADER+E+f+b,dirRecord:R.CENTRAL_FILE_HEADER+A(C,2)+E+A(p.length,2)+"\0\0\0\0"+A(z,4)+A(n,4)+f+b+p}}var I=e("../utils"),i=e("../stream/GenericWorker"),O=e("../utf8"),B=e("../crc32"),R=e("../signature");function s(e,t,r,n){i.call(this,"ZipFileWorker"),this.bytesWritten=0,this.zipComment=t,this.zipPlatform=r,this.encodeFileName=n,this.streamFiles=e,this.accumulate=!1,this.contentBuffer=[],this.dirRecords=[],this.currentSourceOffset=0,this.entriesCount=0,this.currentFile=null,this._sources=[]}I.inherits(s,i),s.prototype.push=function(e){var t=e.meta.percent||0,r=this.entriesCount,n=this._sources.length;this.accumulate?this.contentBuffer.push(e):(this.bytesWritten+=e.data.length,i.prototype.push.call(this,{data:e.data,meta:{currentFile:this.currentFile,percent:r?(t+100*(r-n-1))/r:100}}))},s.prototype.openedSource=function(e){this.currentSourceOffset=this.bytesWritten,this.currentFile=e.file.name;var t=this.streamFiles&&!e.file.dir;if(t){var r=n(e,t,!1,this.currentSourceOffset,this.zipPlatform,this.encodeFileName);this.push({data:r.fileRecord,meta:{percent:0}})}else this.accumulate=!0},s.prototype.closedSource=function(e){this.accumulate=!1;var t=this.streamFiles&&!e.file.dir,r=n(e,t,!0,this.currentSourceOffset,this.zipPlatform,this.encodeFileName);if(this.dirRecords.push(r.dirRecord),t)this.push({data:function(e){return R.DATA_DESCRIPTOR+A(e.crc32,4)+A(e.compressedSize,4)+A(e.uncompressedSize,4)}(e),meta:{percent:100}});else for(this.push({data:r.fileRecord,meta:{percent:0}});this.contentBuffer.length;)this.push(this.contentBuffer.shift());this.currentFile=null},s.prototype.flush=function(){for(var e=this.bytesWritten,t=0;t=this.index;t--)r=(r<<8)+this.byteAt(t);return this.index+=e,r},readString:function(e){return n.transformTo("string",this.readData(e))},readData:function(){},lastIndexOfSignature:function(){},readAndCheckSignature:function(){},readDate:function(){var e=this.readInt(4);return new Date(Date.UTC(1980+(e>>25&127),(e>>21&15)-1,e>>16&31,e>>11&31,e>>5&63,(31&e)<<1))}},t.exports=i},{"../utils":32}],19:[function(e,t,r){"use strict";var n=e("./Uint8ArrayReader");function i(e){n.call(this,e)}e("../utils").inherits(i,n),i.prototype.readData=function(e){this.checkOffset(e);var t=this.data.slice(this.zero+this.index,this.zero+this.index+e);return this.index+=e,t},t.exports=i},{"../utils":32,"./Uint8ArrayReader":21}],20:[function(e,t,r){"use strict";var n=e("./DataReader");function i(e){n.call(this,e)}e("../utils").inherits(i,n),i.prototype.byteAt=function(e){return this.data.charCodeAt(this.zero+e)},i.prototype.lastIndexOfSignature=function(e){return this.data.lastIndexOf(e)-this.zero},i.prototype.readAndCheckSignature=function(e){return e===this.readData(4)},i.prototype.readData=function(e){this.checkOffset(e);var t=this.data.slice(this.zero+this.index,this.zero+this.index+e);return this.index+=e,t},t.exports=i},{"../utils":32,"./DataReader":18}],21:[function(e,t,r){"use strict";var n=e("./ArrayReader");function i(e){n.call(this,e)}e("../utils").inherits(i,n),i.prototype.readData=function(e){if(this.checkOffset(e),0===e)return new Uint8Array(0);var t=this.data.subarray(this.zero+this.index,this.zero+this.index+e);return this.index+=e,t},t.exports=i},{"../utils":32,"./ArrayReader":17}],22:[function(e,t,r){"use strict";var n=e("../utils"),i=e("../support"),s=e("./ArrayReader"),a=e("./StringReader"),o=e("./NodeBufferReader"),h=e("./Uint8ArrayReader");t.exports=function(e){var t=n.getTypeOf(e);return n.checkSupport(t),"string"!==t||i.uint8array?"nodebuffer"===t?new o(e):i.uint8array?new h(n.transformTo("uint8array",e)):new s(n.transformTo("array",e)):new a(e)}},{"../support":30,"../utils":32,"./ArrayReader":17,"./NodeBufferReader":19,"./StringReader":20,"./Uint8ArrayReader":21}],23:[function(e,t,r){"use strict";r.LOCAL_FILE_HEADER="PK",r.CENTRAL_FILE_HEADER="PK",r.CENTRAL_DIRECTORY_END="PK",r.ZIP64_CENTRAL_DIRECTORY_LOCATOR="PK",r.ZIP64_CENTRAL_DIRECTORY_END="PK",r.DATA_DESCRIPTOR="PK\b"},{}],24:[function(e,t,r){"use strict";var n=e("./GenericWorker"),i=e("../utils");function s(e){n.call(this,"ConvertWorker to "+e),this.destType=e}i.inherits(s,n),s.prototype.processChunk=function(e){this.push({data:i.transformTo(this.destType,e.data),meta:e.meta})},t.exports=s},{"../utils":32,"./GenericWorker":28}],25:[function(e,t,r){"use strict";var n=e("./GenericWorker"),i=e("../crc32");function s(){n.call(this,"Crc32Probe"),this.withStreamInfo("crc32",0)}e("../utils").inherits(s,n),s.prototype.processChunk=function(e){this.streamInfo.crc32=i(e.data,this.streamInfo.crc32||0),this.push(e)},t.exports=s},{"../crc32":4,"../utils":32,"./GenericWorker":28}],26:[function(e,t,r){"use strict";var n=e("../utils"),i=e("./GenericWorker");function s(e){i.call(this,"DataLengthProbe for "+e),this.propName=e,this.withStreamInfo(e,0)}n.inherits(s,i),s.prototype.processChunk=function(e){if(e){var t=this.streamInfo[this.propName]||0;this.streamInfo[this.propName]=t+e.data.length}i.prototype.processChunk.call(this,e)},t.exports=s},{"../utils":32,"./GenericWorker":28}],27:[function(e,t,r){"use strict";var n=e("../utils"),i=e("./GenericWorker");function s(e){i.call(this,"DataWorker");var t=this;this.dataIsReady=!1,this.index=0,this.max=0,this.data=null,this.type="",this._tickScheduled=!1,e.then(function(e){t.dataIsReady=!0,t.data=e,t.max=e&&e.length||0,t.type=n.getTypeOf(e),t.isPaused||t._tickAndRepeat()},function(e){t.error(e)})}n.inherits(s,i),s.prototype.cleanUp=function(){i.prototype.cleanUp.call(this),this.data=null},s.prototype.resume=function(){return!!i.prototype.resume.call(this)&&(!this._tickScheduled&&this.dataIsReady&&(this._tickScheduled=!0,n.delay(this._tickAndRepeat,[],this)),!0)},s.prototype._tickAndRepeat=function(){this._tickScheduled=!1,this.isPaused||this.isFinished||(this._tick(),this.isFinished||(n.delay(this._tickAndRepeat,[],this),this._tickScheduled=!0))},s.prototype._tick=function(){if(this.isPaused||this.isFinished)return!1;var e=null,t=Math.min(this.max,this.index+16384);if(this.index>=this.max)return this.end();switch(this.type){case"string":e=this.data.substring(this.index,t);break;case"uint8array":e=this.data.subarray(this.index,t);break;case"array":case"nodebuffer":e=this.data.slice(this.index,t)}return this.index=t,this.push({data:e,meta:{percent:this.max?this.index/this.max*100:0}})},t.exports=s},{"../utils":32,"./GenericWorker":28}],28:[function(e,t,r){"use strict";function n(e){this.name=e||"default",this.streamInfo={},this.generatedError=null,this.extraStreamInfo={},this.isPaused=!0,this.isFinished=!1,this.isLocked=!1,this._listeners={data:[],end:[],error:[]},this.previous=null}n.prototype={push:function(e){this.emit("data",e)},end:function(){if(this.isFinished)return!1;this.flush();try{this.emit("end"),this.cleanUp(),this.isFinished=!0}catch(e){this.emit("error",e)}return!0},error:function(e){return!this.isFinished&&(this.isPaused?this.generatedError=e:(this.isFinished=!0,this.emit("error",e),this.previous&&this.previous.error(e),this.cleanUp()),!0)},on:function(e,t){return this._listeners[e].push(t),this},cleanUp:function(){this.streamInfo=this.generatedError=this.extraStreamInfo=null,this._listeners=[]},emit:function(e,t){if(this._listeners[e])for(var r=0;r "+e:e}},t.exports=n},{}],29:[function(e,t,r){"use strict";var h=e("../utils"),i=e("./ConvertWorker"),s=e("./GenericWorker"),u=e("../base64"),n=e("../support"),a=e("../external"),o=null;if(n.nodestream)try{o=e("../nodejs/NodejsStreamOutputAdapter")}catch(e){}function l(e,o){return new a.Promise(function(t,r){var n=[],i=e._internalType,s=e._outputType,a=e._mimeType;e.on("data",function(e,t){n.push(e),o&&o(t)}).on("error",function(e){n=[],r(e)}).on("end",function(){try{var e=function(e,t,r){switch(e){case"blob":return h.newBlob(h.transformTo("arraybuffer",t),r);case"base64":return u.encode(t);default:return h.transformTo(e,t)}}(s,function(e,t){var r,n=0,i=null,s=0;for(r=0;r>>6:(r<65536?t[s++]=224|r>>>12:(t[s++]=240|r>>>18,t[s++]=128|r>>>12&63),t[s++]=128|r>>>6&63),t[s++]=128|63&r);return t}(e)},s.utf8decode=function(e){return h.nodebuffer?o.transformTo("nodebuffer",e).toString("utf-8"):function(e){var t,r,n,i,s=e.length,a=new Array(2*s);for(t=r=0;t>10&1023,a[r++]=56320|1023&n)}return a.length!==r&&(a.subarray?a=a.subarray(0,r):a.length=r),o.applyFromCharCode(a)}(e=o.transformTo(h.uint8array?"uint8array":"array",e))},o.inherits(a,n),a.prototype.processChunk=function(e){var t=o.transformTo(h.uint8array?"uint8array":"array",e.data);if(this.leftOver&&this.leftOver.length){if(h.uint8array){var r=t;(t=new Uint8Array(r.length+this.leftOver.length)).set(this.leftOver,0),t.set(r,this.leftOver.length)}else t=this.leftOver.concat(t);this.leftOver=null}var n=function(e,t){var r;for((t=t||e.length)>e.length&&(t=e.length),r=t-1;0<=r&&128==(192&e[r]);)r--;return r<0?t:0===r?t:r+u[e[r]]>t?r:t}(t),i=t;n!==t.length&&(h.uint8array?(i=t.subarray(0,n),this.leftOver=t.subarray(n,t.length)):(i=t.slice(0,n),this.leftOver=t.slice(n,t.length))),this.push({data:s.utf8decode(i),meta:e.meta})},a.prototype.flush=function(){this.leftOver&&this.leftOver.length&&(this.push({data:s.utf8decode(this.leftOver),meta:{}}),this.leftOver=null)},s.Utf8DecodeWorker=a,o.inherits(l,n),l.prototype.processChunk=function(e){this.push({data:s.utf8encode(e.data),meta:e.meta})},s.Utf8EncodeWorker=l},{"./nodejsUtils":14,"./stream/GenericWorker":28,"./support":30,"./utils":32}],32:[function(e,t,a){"use strict";var o=e("./support"),h=e("./base64"),r=e("./nodejsUtils"),u=e("./external");function n(e){return e}function l(e,t){for(var r=0;r>8;this.dir=!!(16&this.externalFileAttributes),0==e&&(this.dosPermissions=63&this.externalFileAttributes),3==e&&(this.unixPermissions=this.externalFileAttributes>>16&65535),this.dir||"/"!==this.fileNameStr.slice(-1)||(this.dir=!0)},parseZIP64ExtraField:function(){if(this.extraFields[1]){var e=n(this.extraFields[1].value);this.uncompressedSize===s.MAX_VALUE_32BITS&&(this.uncompressedSize=e.readInt(8)),this.compressedSize===s.MAX_VALUE_32BITS&&(this.compressedSize=e.readInt(8)),this.localHeaderOffset===s.MAX_VALUE_32BITS&&(this.localHeaderOffset=e.readInt(8)),this.diskNumberStart===s.MAX_VALUE_32BITS&&(this.diskNumberStart=e.readInt(4))}},readExtraFields:function(e){var t,r,n,i=e.index+this.extraFieldsLength;for(this.extraFields||(this.extraFields={});e.index+4>>6:(r<65536?t[s++]=224|r>>>12:(t[s++]=240|r>>>18,t[s++]=128|r>>>12&63),t[s++]=128|r>>>6&63),t[s++]=128|63&r);return t},r.buf2binstring=function(e){return l(e,e.length)},r.binstring2buf=function(e){for(var t=new h.Buf8(e.length),r=0,n=t.length;r>10&1023,o[n++]=56320|1023&i)}return l(o,n)},r.utf8border=function(e,t){var r;for((t=t||e.length)>e.length&&(t=e.length),r=t-1;0<=r&&128==(192&e[r]);)r--;return r<0?t:0===r?t:r+u[e[r]]>t?r:t}},{"./common":41}],43:[function(e,t,r){"use strict";t.exports=function(e,t,r,n){for(var i=65535&e|0,s=e>>>16&65535|0,a=0;0!==r;){for(r-=a=2e3>>1:e>>>1;t[r]=e}return t}();t.exports=function(e,t,r,n){var i=o,s=n+r;e^=-1;for(var a=n;a>>8^i[255&(e^t[a])];return-1^e}},{}],46:[function(e,t,r){"use strict";var h,c=e("../utils/common"),u=e("./trees"),d=e("./adler32"),p=e("./crc32"),n=e("./messages"),l=0,f=4,m=0,_=-2,g=-1,b=4,i=2,v=8,y=9,s=286,a=30,o=19,w=2*s+1,k=15,x=3,S=258,z=S+x+1,C=42,E=113,A=1,I=2,O=3,B=4;function R(e,t){return e.msg=n[t],t}function T(e){return(e<<1)-(4e.avail_out&&(r=e.avail_out),0!==r&&(c.arraySet(e.output,t.pending_buf,t.pending_out,r,e.next_out),e.next_out+=r,t.pending_out+=r,e.total_out+=r,e.avail_out-=r,t.pending-=r,0===t.pending&&(t.pending_out=0))}function N(e,t){u._tr_flush_block(e,0<=e.block_start?e.block_start:-1,e.strstart-e.block_start,t),e.block_start=e.strstart,F(e.strm)}function U(e,t){e.pending_buf[e.pending++]=t}function P(e,t){e.pending_buf[e.pending++]=t>>>8&255,e.pending_buf[e.pending++]=255&t}function L(e,t){var r,n,i=e.max_chain_length,s=e.strstart,a=e.prev_length,o=e.nice_match,h=e.strstart>e.w_size-z?e.strstart-(e.w_size-z):0,u=e.window,l=e.w_mask,f=e.prev,c=e.strstart+S,d=u[s+a-1],p=u[s+a];e.prev_length>=e.good_match&&(i>>=2),o>e.lookahead&&(o=e.lookahead);do{if(u[(r=t)+a]===p&&u[r+a-1]===d&&u[r]===u[s]&&u[++r]===u[s+1]){s+=2,r++;do{}while(u[++s]===u[++r]&&u[++s]===u[++r]&&u[++s]===u[++r]&&u[++s]===u[++r]&&u[++s]===u[++r]&&u[++s]===u[++r]&&u[++s]===u[++r]&&u[++s]===u[++r]&&sh&&0!=--i);return a<=e.lookahead?a:e.lookahead}function j(e){var t,r,n,i,s,a,o,h,u,l,f=e.w_size;do{if(i=e.window_size-e.lookahead-e.strstart,e.strstart>=f+(f-z)){for(c.arraySet(e.window,e.window,f,f,0),e.match_start-=f,e.strstart-=f,e.block_start-=f,t=r=e.hash_size;n=e.head[--t],e.head[t]=f<=n?n-f:0,--r;);for(t=r=f;n=e.prev[--t],e.prev[t]=f<=n?n-f:0,--r;);i+=f}if(0===e.strm.avail_in)break;if(a=e.strm,o=e.window,h=e.strstart+e.lookahead,u=i,l=void 0,l=a.avail_in,u=x)for(s=e.strstart-e.insert,e.ins_h=e.window[s],e.ins_h=(e.ins_h<=x&&(e.ins_h=(e.ins_h<=x)if(n=u._tr_tally(e,e.strstart-e.match_start,e.match_length-x),e.lookahead-=e.match_length,e.match_length<=e.max_lazy_match&&e.lookahead>=x){for(e.match_length--;e.strstart++,e.ins_h=(e.ins_h<=x&&(e.ins_h=(e.ins_h<=x&&e.match_length<=e.prev_length){for(i=e.strstart+e.lookahead-x,n=u._tr_tally(e,e.strstart-1-e.prev_match,e.prev_length-x),e.lookahead-=e.prev_length-1,e.prev_length-=2;++e.strstart<=i&&(e.ins_h=(e.ins_h<e.pending_buf_size-5&&(r=e.pending_buf_size-5);;){if(e.lookahead<=1){if(j(e),0===e.lookahead&&t===l)return A;if(0===e.lookahead)break}e.strstart+=e.lookahead,e.lookahead=0;var n=e.block_start+r;if((0===e.strstart||e.strstart>=n)&&(e.lookahead=e.strstart-n,e.strstart=n,N(e,!1),0===e.strm.avail_out))return A;if(e.strstart-e.block_start>=e.w_size-z&&(N(e,!1),0===e.strm.avail_out))return A}return e.insert=0,t===f?(N(e,!0),0===e.strm.avail_out?O:B):(e.strstart>e.block_start&&(N(e,!1),e.strm.avail_out),A)}),new M(4,4,8,4,Z),new M(4,5,16,8,Z),new M(4,6,32,32,Z),new M(4,4,16,16,W),new M(8,16,32,32,W),new M(8,16,128,128,W),new M(8,32,128,256,W),new M(32,128,258,1024,W),new M(32,258,258,4096,W)],r.deflateInit=function(e,t){return Y(e,t,v,15,8,0)},r.deflateInit2=Y,r.deflateReset=K,r.deflateResetKeep=G,r.deflateSetHeader=function(e,t){return e&&e.state?2!==e.state.wrap?_:(e.state.gzhead=t,m):_},r.deflate=function(e,t){var r,n,i,s;if(!e||!e.state||5>8&255),U(n,n.gzhead.time>>16&255),U(n,n.gzhead.time>>24&255),U(n,9===n.level?2:2<=n.strategy||n.level<2?4:0),U(n,255&n.gzhead.os),n.gzhead.extra&&n.gzhead.extra.length&&(U(n,255&n.gzhead.extra.length),U(n,n.gzhead.extra.length>>8&255)),n.gzhead.hcrc&&(e.adler=p(e.adler,n.pending_buf,n.pending,0)),n.gzindex=0,n.status=69):(U(n,0),U(n,0),U(n,0),U(n,0),U(n,0),U(n,9===n.level?2:2<=n.strategy||n.level<2?4:0),U(n,3),n.status=E);else{var a=v+(n.w_bits-8<<4)<<8;a|=(2<=n.strategy||n.level<2?0:n.level<6?1:6===n.level?2:3)<<6,0!==n.strstart&&(a|=32),a+=31-a%31,n.status=E,P(n,a),0!==n.strstart&&(P(n,e.adler>>>16),P(n,65535&e.adler)),e.adler=1}if(69===n.status)if(n.gzhead.extra){for(i=n.pending;n.gzindex<(65535&n.gzhead.extra.length)&&(n.pending!==n.pending_buf_size||(n.gzhead.hcrc&&n.pending>i&&(e.adler=p(e.adler,n.pending_buf,n.pending-i,i)),F(e),i=n.pending,n.pending!==n.pending_buf_size));)U(n,255&n.gzhead.extra[n.gzindex]),n.gzindex++;n.gzhead.hcrc&&n.pending>i&&(e.adler=p(e.adler,n.pending_buf,n.pending-i,i)),n.gzindex===n.gzhead.extra.length&&(n.gzindex=0,n.status=73)}else n.status=73;if(73===n.status)if(n.gzhead.name){i=n.pending;do{if(n.pending===n.pending_buf_size&&(n.gzhead.hcrc&&n.pending>i&&(e.adler=p(e.adler,n.pending_buf,n.pending-i,i)),F(e),i=n.pending,n.pending===n.pending_buf_size)){s=1;break}s=n.gzindexi&&(e.adler=p(e.adler,n.pending_buf,n.pending-i,i)),0===s&&(n.gzindex=0,n.status=91)}else n.status=91;if(91===n.status)if(n.gzhead.comment){i=n.pending;do{if(n.pending===n.pending_buf_size&&(n.gzhead.hcrc&&n.pending>i&&(e.adler=p(e.adler,n.pending_buf,n.pending-i,i)),F(e),i=n.pending,n.pending===n.pending_buf_size)){s=1;break}s=n.gzindexi&&(e.adler=p(e.adler,n.pending_buf,n.pending-i,i)),0===s&&(n.status=103)}else n.status=103;if(103===n.status&&(n.gzhead.hcrc?(n.pending+2>n.pending_buf_size&&F(e),n.pending+2<=n.pending_buf_size&&(U(n,255&e.adler),U(n,e.adler>>8&255),e.adler=0,n.status=E)):n.status=E),0!==n.pending){if(F(e),0===e.avail_out)return n.last_flush=-1,m}else if(0===e.avail_in&&T(t)<=T(r)&&t!==f)return R(e,-5);if(666===n.status&&0!==e.avail_in)return R(e,-5);if(0!==e.avail_in||0!==n.lookahead||t!==l&&666!==n.status){var o=2===n.strategy?function(e,t){for(var r;;){if(0===e.lookahead&&(j(e),0===e.lookahead)){if(t===l)return A;break}if(e.match_length=0,r=u._tr_tally(e,0,e.window[e.strstart]),e.lookahead--,e.strstart++,r&&(N(e,!1),0===e.strm.avail_out))return A}return e.insert=0,t===f?(N(e,!0),0===e.strm.avail_out?O:B):e.last_lit&&(N(e,!1),0===e.strm.avail_out)?A:I}(n,t):3===n.strategy?function(e,t){for(var r,n,i,s,a=e.window;;){if(e.lookahead<=S){if(j(e),e.lookahead<=S&&t===l)return A;if(0===e.lookahead)break}if(e.match_length=0,e.lookahead>=x&&0e.lookahead&&(e.match_length=e.lookahead)}if(e.match_length>=x?(r=u._tr_tally(e,1,e.match_length-x),e.lookahead-=e.match_length,e.strstart+=e.match_length,e.match_length=0):(r=u._tr_tally(e,0,e.window[e.strstart]),e.lookahead--,e.strstart++),r&&(N(e,!1),0===e.strm.avail_out))return A}return e.insert=0,t===f?(N(e,!0),0===e.strm.avail_out?O:B):e.last_lit&&(N(e,!1),0===e.strm.avail_out)?A:I}(n,t):h[n.level].func(n,t);if(o!==O&&o!==B||(n.status=666),o===A||o===O)return 0===e.avail_out&&(n.last_flush=-1),m;if(o===I&&(1===t?u._tr_align(n):5!==t&&(u._tr_stored_block(n,0,0,!1),3===t&&(D(n.head),0===n.lookahead&&(n.strstart=0,n.block_start=0,n.insert=0))),F(e),0===e.avail_out))return n.last_flush=-1,m}return t!==f?m:n.wrap<=0?1:(2===n.wrap?(U(n,255&e.adler),U(n,e.adler>>8&255),U(n,e.adler>>16&255),U(n,e.adler>>24&255),U(n,255&e.total_in),U(n,e.total_in>>8&255),U(n,e.total_in>>16&255),U(n,e.total_in>>24&255)):(P(n,e.adler>>>16),P(n,65535&e.adler)),F(e),0=r.w_size&&(0===s&&(D(r.head),r.strstart=0,r.block_start=0,r.insert=0),u=new c.Buf8(r.w_size),c.arraySet(u,t,l-r.w_size,r.w_size,0),t=u,l=r.w_size),a=e.avail_in,o=e.next_in,h=e.input,e.avail_in=l,e.next_in=0,e.input=t,j(r);r.lookahead>=x;){for(n=r.strstart,i=r.lookahead-(x-1);r.ins_h=(r.ins_h<>>=y=v>>>24,p-=y,0===(y=v>>>16&255))C[s++]=65535&v;else{if(!(16&y)){if(0==(64&y)){v=m[(65535&v)+(d&(1<>>=y,p-=y),p<15&&(d+=z[n++]<>>=y=v>>>24,p-=y,!(16&(y=v>>>16&255))){if(0==(64&y)){v=_[(65535&v)+(d&(1<>>=y,p-=y,(y=s-a)>3,d&=(1<<(p-=w<<3))-1,e.next_in=n,e.next_out=s,e.avail_in=n>>24&255)+(e>>>8&65280)+((65280&e)<<8)+((255&e)<<24)}function s(){this.mode=0,this.last=!1,this.wrap=0,this.havedict=!1,this.flags=0,this.dmax=0,this.check=0,this.total=0,this.head=null,this.wbits=0,this.wsize=0,this.whave=0,this.wnext=0,this.window=null,this.hold=0,this.bits=0,this.length=0,this.offset=0,this.extra=0,this.lencode=null,this.distcode=null,this.lenbits=0,this.distbits=0,this.ncode=0,this.nlen=0,this.ndist=0,this.have=0,this.next=null,this.lens=new I.Buf16(320),this.work=new I.Buf16(288),this.lendyn=null,this.distdyn=null,this.sane=0,this.back=0,this.was=0}function a(e){var t;return e&&e.state?(t=e.state,e.total_in=e.total_out=t.total=0,e.msg="",t.wrap&&(e.adler=1&t.wrap),t.mode=P,t.last=0,t.havedict=0,t.dmax=32768,t.head=null,t.hold=0,t.bits=0,t.lencode=t.lendyn=new I.Buf32(n),t.distcode=t.distdyn=new I.Buf32(i),t.sane=1,t.back=-1,N):U}function o(e){var t;return e&&e.state?((t=e.state).wsize=0,t.whave=0,t.wnext=0,a(e)):U}function h(e,t){var r,n;return e&&e.state?(n=e.state,t<0?(r=0,t=-t):(r=1+(t>>4),t<48&&(t&=15)),t&&(t<8||15=s.wsize?(I.arraySet(s.window,t,r-s.wsize,s.wsize,0),s.wnext=0,s.whave=s.wsize):(n<(i=s.wsize-s.wnext)&&(i=n),I.arraySet(s.window,t,r-n,i,s.wnext),(n-=i)?(I.arraySet(s.window,t,r-n,n,0),s.wnext=n,s.whave=s.wsize):(s.wnext+=i,s.wnext===s.wsize&&(s.wnext=0),s.whave>>8&255,r.check=B(r.check,E,2,0),l=u=0,r.mode=2;break}if(r.flags=0,r.head&&(r.head.done=!1),!(1&r.wrap)||(((255&u)<<8)+(u>>8))%31){e.msg="incorrect header check",r.mode=30;break}if(8!=(15&u)){e.msg="unknown compression method",r.mode=30;break}if(l-=4,k=8+(15&(u>>>=4)),0===r.wbits)r.wbits=k;else if(k>r.wbits){e.msg="invalid window size",r.mode=30;break}r.dmax=1<>8&1),512&r.flags&&(E[0]=255&u,E[1]=u>>>8&255,r.check=B(r.check,E,2,0)),l=u=0,r.mode=3;case 3:for(;l<32;){if(0===o)break e;o--,u+=n[s++]<>>8&255,E[2]=u>>>16&255,E[3]=u>>>24&255,r.check=B(r.check,E,4,0)),l=u=0,r.mode=4;case 4:for(;l<16;){if(0===o)break e;o--,u+=n[s++]<>8),512&r.flags&&(E[0]=255&u,E[1]=u>>>8&255,r.check=B(r.check,E,2,0)),l=u=0,r.mode=5;case 5:if(1024&r.flags){for(;l<16;){if(0===o)break e;o--,u+=n[s++]<>>8&255,r.check=B(r.check,E,2,0)),l=u=0}else r.head&&(r.head.extra=null);r.mode=6;case 6:if(1024&r.flags&&(o<(d=r.length)&&(d=o),d&&(r.head&&(k=r.head.extra_len-r.length,r.head.extra||(r.head.extra=new Array(r.head.extra_len)),I.arraySet(r.head.extra,n,s,d,k)),512&r.flags&&(r.check=B(r.check,n,d,s)),o-=d,s+=d,r.length-=d),r.length))break e;r.length=0,r.mode=7;case 7:if(2048&r.flags){if(0===o)break e;for(d=0;k=n[s+d++],r.head&&k&&r.length<65536&&(r.head.name+=String.fromCharCode(k)),k&&d>9&1,r.head.done=!0),e.adler=r.check=0,r.mode=12;break;case 10:for(;l<32;){if(0===o)break e;o--,u+=n[s++]<>>=7&l,l-=7&l,r.mode=27;break}for(;l<3;){if(0===o)break e;o--,u+=n[s++]<>>=1)){case 0:r.mode=14;break;case 1:if(j(r),r.mode=20,6!==t)break;u>>>=2,l-=2;break e;case 2:r.mode=17;break;case 3:e.msg="invalid block type",r.mode=30}u>>>=2,l-=2;break;case 14:for(u>>>=7&l,l-=7&l;l<32;){if(0===o)break e;o--,u+=n[s++]<>>16^65535)){e.msg="invalid stored block lengths",r.mode=30;break}if(r.length=65535&u,l=u=0,r.mode=15,6===t)break e;case 15:r.mode=16;case 16:if(d=r.length){if(o>>=5,l-=5,r.ndist=1+(31&u),u>>>=5,l-=5,r.ncode=4+(15&u),u>>>=4,l-=4,286>>=3,l-=3}for(;r.have<19;)r.lens[A[r.have++]]=0;if(r.lencode=r.lendyn,r.lenbits=7,S={bits:r.lenbits},x=T(0,r.lens,0,19,r.lencode,0,r.work,S),r.lenbits=S.bits,x){e.msg="invalid code lengths set",r.mode=30;break}r.have=0,r.mode=19;case 19:for(;r.have>>16&255,b=65535&C,!((_=C>>>24)<=l);){if(0===o)break e;o--,u+=n[s++]<>>=_,l-=_,r.lens[r.have++]=b;else{if(16===b){for(z=_+2;l>>=_,l-=_,0===r.have){e.msg="invalid bit length repeat",r.mode=30;break}k=r.lens[r.have-1],d=3+(3&u),u>>>=2,l-=2}else if(17===b){for(z=_+3;l>>=_)),u>>>=3,l-=3}else{for(z=_+7;l>>=_)),u>>>=7,l-=7}if(r.have+d>r.nlen+r.ndist){e.msg="invalid bit length repeat",r.mode=30;break}for(;d--;)r.lens[r.have++]=k}}if(30===r.mode)break;if(0===r.lens[256]){e.msg="invalid code -- missing end-of-block",r.mode=30;break}if(r.lenbits=9,S={bits:r.lenbits},x=T(D,r.lens,0,r.nlen,r.lencode,0,r.work,S),r.lenbits=S.bits,x){e.msg="invalid literal/lengths set",r.mode=30;break}if(r.distbits=6,r.distcode=r.distdyn,S={bits:r.distbits},x=T(F,r.lens,r.nlen,r.ndist,r.distcode,0,r.work,S),r.distbits=S.bits,x){e.msg="invalid distances set",r.mode=30;break}if(r.mode=20,6===t)break e;case 20:r.mode=21;case 21:if(6<=o&&258<=h){e.next_out=a,e.avail_out=h,e.next_in=s,e.avail_in=o,r.hold=u,r.bits=l,R(e,c),a=e.next_out,i=e.output,h=e.avail_out,s=e.next_in,n=e.input,o=e.avail_in,u=r.hold,l=r.bits,12===r.mode&&(r.back=-1);break}for(r.back=0;g=(C=r.lencode[u&(1<>>16&255,b=65535&C,!((_=C>>>24)<=l);){if(0===o)break e;o--,u+=n[s++]<>v)])>>>16&255,b=65535&C,!(v+(_=C>>>24)<=l);){if(0===o)break e;o--,u+=n[s++]<>>=v,l-=v,r.back+=v}if(u>>>=_,l-=_,r.back+=_,r.length=b,0===g){r.mode=26;break}if(32&g){r.back=-1,r.mode=12;break}if(64&g){e.msg="invalid literal/length code",r.mode=30;break}r.extra=15&g,r.mode=22;case 22:if(r.extra){for(z=r.extra;l>>=r.extra,l-=r.extra,r.back+=r.extra}r.was=r.length,r.mode=23;case 23:for(;g=(C=r.distcode[u&(1<>>16&255,b=65535&C,!((_=C>>>24)<=l);){if(0===o)break e;o--,u+=n[s++]<>v)])>>>16&255,b=65535&C,!(v+(_=C>>>24)<=l);){if(0===o)break e;o--,u+=n[s++]<>>=v,l-=v,r.back+=v}if(u>>>=_,l-=_,r.back+=_,64&g){e.msg="invalid distance code",r.mode=30;break}r.offset=b,r.extra=15&g,r.mode=24;case 24:if(r.extra){for(z=r.extra;l>>=r.extra,l-=r.extra,r.back+=r.extra}if(r.offset>r.dmax){e.msg="invalid distance too far back",r.mode=30;break}r.mode=25;case 25:if(0===h)break e;if(d=c-h,r.offset>d){if((d=r.offset-d)>r.whave&&r.sane){e.msg="invalid distance too far back",r.mode=30;break}p=d>r.wnext?(d-=r.wnext,r.wsize-d):r.wnext-d,d>r.length&&(d=r.length),m=r.window}else m=i,p=a-r.offset,d=r.length;for(hd?(m=R[T+a[v]],A[I+a[v]]):(m=96,0),h=1<>S)+(u-=h)]=p<<24|m<<16|_|0,0!==u;);for(h=1<>=1;if(0!==h?(E&=h-1,E+=h):E=0,v++,0==--O[b]){if(b===w)break;b=t[r+a[v]]}if(k>>7)]}function U(e,t){e.pending_buf[e.pending++]=255&t,e.pending_buf[e.pending++]=t>>>8&255}function P(e,t,r){e.bi_valid>d-r?(e.bi_buf|=t<>d-e.bi_valid,e.bi_valid+=r-d):(e.bi_buf|=t<>>=1,r<<=1,0<--t;);return r>>>1}function Z(e,t,r){var n,i,s=new Array(g+1),a=0;for(n=1;n<=g;n++)s[n]=a=a+r[n-1]<<1;for(i=0;i<=t;i++){var o=e[2*i+1];0!==o&&(e[2*i]=j(s[o]++,o))}}function W(e){var t;for(t=0;t>1;1<=r;r--)G(e,s,r);for(i=h;r=e.heap[1],e.heap[1]=e.heap[e.heap_len--],G(e,s,1),n=e.heap[1],e.heap[--e.heap_max]=r,e.heap[--e.heap_max]=n,s[2*i]=s[2*r]+s[2*n],e.depth[i]=(e.depth[r]>=e.depth[n]?e.depth[r]:e.depth[n])+1,s[2*r+1]=s[2*n+1]=i,e.heap[1]=i++,G(e,s,1),2<=e.heap_len;);e.heap[--e.heap_max]=e.heap[1],function(e,t){var r,n,i,s,a,o,h=t.dyn_tree,u=t.max_code,l=t.stat_desc.static_tree,f=t.stat_desc.has_stree,c=t.stat_desc.extra_bits,d=t.stat_desc.extra_base,p=t.stat_desc.max_length,m=0;for(s=0;s<=g;s++)e.bl_count[s]=0;for(h[2*e.heap[e.heap_max]+1]=0,r=e.heap_max+1;r<_;r++)p<(s=h[2*h[2*(n=e.heap[r])+1]+1]+1)&&(s=p,m++),h[2*n+1]=s,u>=7;n>>=1)if(1&r&&0!==e.dyn_ltree[2*t])return o;if(0!==e.dyn_ltree[18]||0!==e.dyn_ltree[20]||0!==e.dyn_ltree[26])return h;for(t=32;t>>3,(s=e.static_len+3+7>>>3)<=i&&(i=s)):i=s=r+5,r+4<=i&&-1!==t?J(e,t,r,n):4===e.strategy||s===i?(P(e,2+(n?1:0),3),K(e,z,C)):(P(e,4+(n?1:0),3),function(e,t,r,n){var i;for(P(e,t-257,5),P(e,r-1,5),P(e,n-4,4),i=0;i>>8&255,e.pending_buf[e.d_buf+2*e.last_lit+1]=255&t,e.pending_buf[e.l_buf+e.last_lit]=255&r,e.last_lit++,0===t?e.dyn_ltree[2*r]++:(e.matches++,t--,e.dyn_ltree[2*(A[r]+u+1)]++,e.dyn_dtree[2*N(t)]++),e.last_lit===e.lit_bufsize-1},r._tr_align=function(e){P(e,2,3),L(e,m,z),function(e){16===e.bi_valid?(U(e,e.bi_buf),e.bi_buf=0,e.bi_valid=0):8<=e.bi_valid&&(e.pending_buf[e.pending++]=255&e.bi_buf,e.bi_buf>>=8,e.bi_valid-=8)}(e)}},{"../utils/common":41}],53:[function(e,t,r){"use strict";t.exports=function(){this.input=null,this.next_in=0,this.avail_in=0,this.total_in=0,this.output=null,this.next_out=0,this.avail_out=0,this.total_out=0,this.msg="",this.state=null,this.data_type=2,this.adler=0}},{}],54:[function(e,t,r){(function(e){!function(r,n){"use strict";if(!r.setImmediate){var i,s,t,a,o=1,h={},u=!1,l=r.document,e=Object.getPrototypeOf&&Object.getPrototypeOf(r);e=e&&e.setTimeout?e:r,i="[object process]"==={}.toString.call(r.process)?function(e){process.nextTick(function(){c(e)})}:function(){if(r.postMessage&&!r.importScripts){var e=!0,t=r.onmessage;return r.onmessage=function(){e=!1},r.postMessage("","*"),r.onmessage=t,e}}()?(a="setImmediate$"+Math.random()+"$",r.addEventListener?r.addEventListener("message",d,!1):r.attachEvent("onmessage",d),function(e){r.postMessage(a+e,"*")}):r.MessageChannel?((t=new MessageChannel).port1.onmessage=function(e){c(e.data)},function(e){t.port2.postMessage(e)}):l&&"onreadystatechange"in l.createElement("script")?(s=l.documentElement,function(e){var t=l.createElement("script");t.onreadystatechange=function(){c(e),t.onreadystatechange=null,s.removeChild(t),t=null},s.appendChild(t)}):function(e){setTimeout(c,0,e)},e.setImmediate=function(e){"function"!=typeof e&&(e=new Function(""+e));for(var t=new Array(arguments.length-1),r=0;r"] + } + ] +} diff --git a/popup.html b/popup.html new file mode 100644 index 0000000..b2f5126 --- /dev/null +++ b/popup.html @@ -0,0 +1,177 @@ + + + + + + + +
+

📦 Manga Mass Downloader

+
+ +
+
0
+
Manga Seleccionados
+
+ + + + + + + +
+
+
+
+
+
+ +
+ 💡 Selecciona los manga en la página y usa este popup para descargarlos +
+ + + + diff --git a/popup.js b/popup.js new file mode 100644 index 0000000..3282c40 --- /dev/null +++ b/popup.js @@ -0,0 +1,224 @@ +// Popup JavaScript para Manga Mass Downloader + +document.addEventListener('DOMContentLoaded', async () => { + const selectedCountElement = document.getElementById('selectedCount'); + const downloadSelectedBtn = document.getElementById('downloadSelected'); + const downloadAllBtn = document.getElementById('downloadAll'); + const clearSelectionBtn = document.getElementById('clearSelection'); + const progressDiv = document.getElementById('progress'); + const progressFill = document.getElementById('progressFill'); + const statusElement = document.getElementById('status'); + + let selectedMangas = []; + const [tab] = await chrome.tabs.query({ active: true, currentWindow: true }); + + // Obtener manga seleccionados del content script + async function getSelectedMangas() { + if (!tab.url.includes('e-hentai.org')) { + selectedCountElement.textContent = '0'; + return []; + } + + return new Promise((resolve) => { + chrome.tabs.sendMessage(tab.id, { action: 'getSelectedMangas' }, (response) => { + if (chrome.runtime.lastError) { + console.error('Error:', chrome.runtime.lastError); + selectedCountElement.textContent = '0'; + resolve([]); + return; + } + + selectedMangas = response ? response.mangas : []; + selectedCountElement.textContent = selectedMangas.length; + resolve(selectedMangas); + }); + }); + } + + // Actualizar contador + async function updateSelectedCount() { + await getSelectedMangas(); + } + + // Mostrar progreso + function showProgress(current, total, status) { + progressDiv.style.display = 'block'; + const percentage = total > 0 ? Math.round((current / total) * 100) : 0; + progressFill.style.width = percentage + '%'; + statusElement.textContent = status || `${current}/${total}`; + } + + // Ocultar progreso + function hideProgress() { + progressDiv.style.display = 'none'; + progressFill.style.width = '0%'; + statusElement.textContent = ''; + } + + // Descargar manga seleccionados + downloadSelectedBtn.addEventListener('click', async () => { + await getSelectedMangas(); + + if (selectedMangas.length === 0) { + alert('No hay manga seleccionados. Ve a la página y selecciona algunos manga.'); + return; + } + + downloadSelectedBtn.disabled = true; + downloadAllBtn.disabled = true; + clearSelectionBtn.disabled = true; + + showProgress(0, selectedMangas.length, 'Iniciando descargas...'); + + try { + // Descargar cada manga secuencialmente + for (let i = 0; i < selectedMangas.length; i++) { + const manga = selectedMangas[i]; + showProgress(i + 1, selectedMangas.length, `Descargando: ${manga.title.substring(0, 30)}...`); + + try { + // Obtener URLs de imágenes + const imageUrls = await getImageUrls(manga); + + // Enviar al background para descarga + const response = await chrome.runtime.sendMessage({ + action: 'downloadManga', + metadata: manga, + imageUrls: imageUrls + }); + + if (!response.success) { + console.error(`Error descargando ${manga.title}:`, response.error); + } + } catch (error) { + console.error(`Error en manga ${manga.title}:`, error.message); + } + } + + showProgress(selectedMangas.length, selectedMangas.length, '¡Completado!'); + statusElement.className = 'status success'; + + // Limpiar selección después de 3 segundos + setTimeout(() => { + chrome.tabs.sendMessage(tab.id, { action: 'clearSelection' }, async () => { + await updateSelectedCount(); + }); + }, 3000); + + } catch (error) { + console.error('Error general:', error); + statusElement.textContent = 'Error: ' + error.message; + statusElement.className = 'status error'; + } finally { + downloadSelectedBtn.disabled = false; + downloadAllBtn.disabled = false; + clearSelectionBtn.disabled = false; + } + }); + + // Descargar TODOS los manga de la página + downloadAllBtn.addEventListener('click', async () => { + if (!tab.url.includes('e-hentai.org')) { + alert('Esta extensión solo funciona en e-hentai.org'); + return; + } + + if (!confirm('¿Descargar TODOS los manga de esta página?')) { + return; + } + + downloadSelectedBtn.disabled = true; + downloadAllBtn.disabled = true; + clearSelectionBtn.disabled = true; + + showProgress(0, 100, 'Obteniendo lista de manga...'); + + try { + // Pedir al content script que extraiga todos los manga + const result = await new Promise((resolve) => { + chrome.tabs.sendMessage(tab.id, { action: 'extractAllMangas' }, (response) => { + resolve(response || { mangas: [] }); + }); + }); + + const allMangas = result.mangas || []; + showProgress(0, allMangas.length, `Encontrados ${allMangas.length} manga`); + + // Descargar cada manga + for (let i = 0; i < allMangas.length; i++) { + const manga = allMangas[i]; + showProgress(i + 1, allMangas.length, `Descargando: ${manga.title.substring(0, 30)}...`); + + try { + const imageUrls = await getImageUrls(manga); + + const response = await chrome.runtime.sendMessage({ + action: 'downloadManga', + metadata: manga, + imageUrls: imageUrls + }); + + if (!response.success) { + console.error(`Error descargando ${manga.title}:`, response.error); + } + } catch (error) { + console.error(`Error en manga ${manga.title}:`, error.message); + } + } + + showProgress(allMangas.length, allMangas.length, '¡Completado!'); + statusElement.className = 'status success'; + + } catch (error) { + console.error('Error general:', error); + statusElement.textContent = 'Error: ' + error.message; + statusElement.className = 'status error'; + } finally { + downloadSelectedBtn.disabled = false; + downloadAllBtn.disabled = false; + clearSelectionBtn.disabled = false; + } + }); + + // Limpiar selección + clearSelectionBtn.addEventListener('click', async () => { + chrome.tabs.sendMessage(tab.id, { action: 'clearSelection' }, async () => { + await updateSelectedCount(); + hideProgress(); + }); + }); + + // Obtener URLs para un manga específico + async function getImageUrls(manga) { + return new Promise((resolve, reject) => { + chrome.tabs.sendMessage( + tab.id, + { action: 'getImageUrls', manga: manga }, + (response) => { + if (chrome.runtime.lastError) { + reject(chrome.runtime.lastError); + return; + } + if (response.error) { + reject(new Error(response.error)); + return; + } + resolve(response.imageUrls); + } + ); + }); + } + + // Escuchar mensajes del background + chrome.runtime.onMessage.addListener((request, sender, sendResponse) => { + if (request.action === 'updateSelectedCount') { + selectedCountElement.textContent = request.count; + } + }); + + // Inicializar + updateSelectedCount(); + + // Actualizar contador cada 2 segundos + setInterval(updateSelectedCount, 2000); +});