Notion Invoices Bulk Downloader

Adds bulk download for Notion invoices and receipts.

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         Notion Invoices Bulk Downloader
// @name:ru      Notion Invoices Bulk Downloader
// @namespace    http://tampermonkey.net/
// @version      1.0.1
// @description  Adds bulk download for Notion invoices and receipts.
// @description:ru Массовое скачивание инвойсов и квитанций в Notion.
// @author       DayDve
// @license      MIT
// @icon         
// @match        https://www.notion.so/*
// @connect      stripe.com
// @connect      amazonaws.com
// @grant        GM_xmlhttpRequest
// @grant        GM_addStyle
// @run-at       document-start
// @keywords     notion, invoices, download, pdf
// ==/UserScript==

(function() {
    'use strict';

    // --- STYLES ---
    GM_addStyle(`
        /* Download Selected Button */
        #nir-mass-download-btn {
            user-select: none; transition: background 20ms ease-in, opacity 200ms;
            display: flex; align-items: center; justify-content: center; height: 32px;
            padding-inline: 12px; border-radius: 6px; white-space: nowrap;
            font-size: 14px; font-weight: 500; line-height: 1.2;
            margin-inline-start: 12px;
            cursor: not-allowed;

            opacity: 0.6;
            color: #888;
            border: 1px solid #555;
            background: #333;
            flex-shrink: 0;
        }

        #nir-mass-download-btn.active {
            opacity: 1;
            cursor: pointer;
            color: #2383e2;
            border: 1px solid rgba(35, 131, 226, 0.5);
            background: rgba(35, 131, 226, 0.1);
        }
        #nir-mass-download-btn.active:hover {
            background: rgba(35, 131, 226, 0.25);
        }

        #nir-mass-download-btn.loading {
            opacity: 0.7;
            cursor: wait !important;
            pointer-events: none;
        }

        /* Invoice Header Wrapper: Makes original header and new button inline */
        .nir-invoices-header-flex {
            display: flex;
            align-items: center;
            justify-content: flex-start;
        }

        /* Original Invoices Title (hidden to use modified copy in the flex container) */
        .nir-original-hidden {
            display: none !important;
        }

        /* Checkbox Container (inserted into row) */
        .nir-checkbox-container {
            display: flex;
            align-items: center;
            justify-content: center;
            width: 20px;
            margin-right: 8px;
            flex-shrink: 0;
            flex-grow: 0;
        }

        /* Modified Invoice Row: Enables checkbox placement */
        .nir-invoice-row-modified {
            display: flex !important;
            align-items: center !important;
            transition: background-color 0.3s ease;
        }

        /* Status Colors */
        .nir-row-downloading {
            background-color: rgba(255, 255, 0, 0.1) !important;
        }
        .nir-row-success {
            background-color: rgba(0, 128, 0, 0.1) !important;
        }
        .nir-row-error {
            background-color: rgba(255, 0, 0, 0.1) !important;
        }
    `);

    // --- GLOBAL STATE ---
    const STATE = {
        headers: null,
        invoiceMap: {}, // "Month D, YYYY" -> "in_XXXXXX"
        rawEvents: [],
        selectedInvoices: new Set(),
        processing: false
    };

    // --- 1. INTERCEPTOR (Capture Billing Data) ---
    function injectInterceptor() {
        const script = document.createElement('script');
        script.textContent = `
            (() => {
                const originalFetch = window.fetch;
                window.fetch = function(...args) {
                    const [resource, config] = args;
                    if (typeof resource === 'string' && resource.includes('getBillingHistory')) {
                        if (config && config.body) {
                            const p = originalFetch.apply(this, args);
                            p.then(res => res.clone().json()).then(json => {
                                window.dispatchEvent(new CustomEvent('NotionInvoiceCaptured', {
                                    detail: { headers: config.headers, data: json }
                                }));
                            });
                            return p;
                        }
                    }
                    return originalFetch.apply(this, args);
                };
            })();
        `;
        (document.head || document.documentElement).appendChild(script);
        script.remove();
    }
    injectInterceptor();

    // --- 2. DATA LISTENER ---
    window.addEventListener('NotionInvoiceCaptured', (e) => {
        const events = e.detail.data.events || [];
        const invoices = events.filter(x => x.type === 'invoice');

        STATE.headers = e.detail.headers;
        STATE.rawEvents = invoices;
        STATE.selectedInvoices.clear();

        // Build map for DOM injection lookup
        const formatter = new Intl.DateTimeFormat('en-US', { month: 'long', day: 'numeric', year: 'numeric' });

        invoices.forEach(inv => {
            const m = inv.url.match(/(in_[a-zA-Z0-9]+)/);
            if (m) {
                const id = m[1];
                const dateText = formatter.format(new Date(inv.timestamp));
                STATE.invoiceMap[dateText] = id;
            }
        });

        startObserver();
    });

    // --- 3. DOM OBSERVER (Injection Logic) ---
    let observerTimer = null;
    function startObserver() {
        if (observerTimer) clearInterval(observerTimer);

        observerTimer = setInterval(() => {
            injectMassDownloadButton();

            // Find invoice rows based on unique styling
            const invoiceRows = document.querySelectorAll('div[style*="display: flex;"][style*="gap: 40px;"][style*="border-bottom: 1px solid var(--ca-regDivCol);"][style*="padding: 11px 0px;"]');

            invoiceRows.forEach(row => {
                if (row.classList.contains('nir-invoice-row-modified')) return;

                const infoCol = row.children[0];
                if (!infoCol) return;

                // Look for date text in the info column
                const dateMatch = infoCol.innerText.match(/([A-Z][a-z]+ \d{1,2}, \d{4})/);

                if (dateMatch) {
                    const dateString = dateMatch[0];
                    const invoiceId = STATE.invoiceMap[dateString];

                    if (invoiceId) {
                        row.classList.add('nir-invoice-row-modified');
                        injectCheckbox(row, invoiceId);
                    }
                }
            });
        }, 500);
    }

    // --- 4. UI INJECTION FUNCTIONS ---

    function injectCheckbox(rowElement, invoiceId) {
        const checkContainer = document.createElement('div');
        checkContainer.className = 'nir-checkbox-container';

        const checkbox = document.createElement('input');
        checkbox.type = 'checkbox';
        checkbox.style.cursor = 'pointer';
        checkbox.id = `nir-check-${invoiceId}`;

        checkbox.onchange = () => {
            if (checkbox.checked) {
                STATE.selectedInvoices.add(invoiceId);
            } else {
                STATE.selectedInvoices.delete(invoiceId);
            }
            updateMassDownloadButtonState();
        };
        checkContainer.appendChild(checkbox);

        rowElement.insertBefore(checkContainer, rowElement.firstChild);

        // Status indicator insertion (empty div for applying color classes)
        const actionsCol = rowElement.children[2];
        if (actionsCol) {
            const statusIndicator = document.createElement('div');
            statusIndicator.id = `nir-status-${invoiceId}`;
            // Preserve the original margin-end of Notion's default button column
            statusIndicator.style.cssText = 'width: 0px; margin-inline-end: 10px;';

            const viewButton = actionsCol.querySelector('[role="button"]');
            if (viewButton) {
                actionsCol.insertBefore(statusIndicator, viewButton);
            }
        }
    }

    function injectMassDownloadButton() {
        const xpathResult = document.evaluate(
            '//div[text()="Invoices" and @style]',
            document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null
        );
        const invoicesTitle = xpathResult.singleNodeValue;

        if (!invoicesTitle) return;
        if (invoicesTitle.classList.contains('nir-processed')) return;
        invoicesTitle.classList.add('nir-processed');

        const titleCopy = invoicesTitle.cloneNode(true);

        let massBtn = document.createElement('div');
        massBtn.id = 'nir-mass-download-btn';
        massBtn.innerHTML = 'Download Selected (0)';
        massBtn.onclick = runMassDownload;

        const newContainer = document.createElement('div');
        newContainer.className = 'nir-invoices-header-flex';

        // Transfer styles (border-bottom, padding-bottom, margin-bottom) to the new container
        const originalStyle = invoicesTitle.getAttribute('style');
        newContainer.setAttribute('style', originalStyle);

        // Clear conflicting styles from the title copy
        titleCopy.style.borderBottom = 'none';
        titleCopy.style.paddingBottom = '0px';
        titleCopy.style.marginBottom = '0px';
        titleCopy.style.flexShrink = '0';

        newContainer.appendChild(titleCopy);
        newContainer.appendChild(massBtn);

        // Insert the new container before the original title
        const originalParent = invoicesTitle.parentElement;
        originalParent.insertBefore(newContainer, invoicesTitle);

        // Hide the original title element
        invoicesTitle.classList.add('nir-original-hidden');

        updateMassDownloadButtonState(massBtn);
    }

    function updateMassDownloadButtonState(btn = document.getElementById('nir-mass-download-btn')) {
        if (!btn || STATE.processing) return;

        const count = STATE.selectedInvoices.size;
        btn.innerHTML = `Download Selected (${count})`;

        if (count > 0) {
            btn.classList.add('active');
        } else {
            btn.classList.remove('active');
        }
    }

    // --- 5. DOWNLOAD LOGIC ---

    async function runMassDownload() {
        const btn = document.getElementById('nir-mass-download-btn');
        if (!btn || !btn.classList.contains('active')) return;

        STATE.processing = true;
        btn.classList.add('loading');

        const headers = { ...STATE.headers, 'Cookie': document.cookie };
        delete headers['content-length'];

        const filesCollection = [];
        const queue = Array.from(STATE.selectedInvoices);

        try {
            for (let i = 0; i < queue.length; i++) {
                const invoiceId = queue[i];
                btn.innerHTML = `⏳ ${i + 1}/${queue.length}`;

                const rowEl = getRowElement(invoiceId);
                if (rowEl) {
                    rowEl.classList.add('nir-row-downloading');
                    rowEl.classList.remove('nir-row-success', 'nir-row-error');
                }

                try {
                    const result = await processViaNotionAPI(invoiceId, headers);
                    if (result.invoice.data) filesCollection.push(result.invoice);
                    if (result.receipt.data) filesCollection.push(result.receipt);

                    if (rowEl) {
                        rowEl.classList.remove('nir-row-downloading');
                        rowEl.classList.add('nir-row-success');
                    }
                } catch(e) {
                    if (rowEl) {
                        rowEl.classList.remove('nir-row-downloading');
                        rowEl.classList.add('nir-row-error');
                    }
                    console.warn(`Error processing ${invoiceId}:`, e);
                }

                await sleep(200);
            }

            btn.innerHTML = "Packing ZIP...";
            const zipBlob = createZip(filesCollection);
            saveBlob(zipBlob, `Notion_Invoices.zip`);

            STATE.selectedInvoices.clear();

            btn.innerHTML = "✅ Done!";
        } catch (e) {
            btn.innerHTML = "❌ Error";
            console.error(e);
        } finally {
            STATE.processing = false;
            // Reset status and checkboxes
            setTimeout(() => {
                btn.classList.remove('loading');
                updateMassDownloadButtonState();
                queue.forEach(id => {
                    const rowEl = getRowElement(id);
                    if (rowEl) {
                        rowEl.classList.remove('nir-row-success', 'nir-row-error');
                    }
                    const checkbox = document.getElementById(`nir-check-${id}`);
                    if (checkbox) checkbox.checked = false;
                });
            }, 3000);
        }
    }

    // --- CORE API WORKER (Native) ---
    async function processViaNotionAPI(invoiceId, authHeaders) {
        const notionRes = await fetch('https://www.notion.so/api/v3/getInvoiceData', {
            method: 'POST', headers: authHeaders, body: JSON.stringify({ type: 'invoice', invoiceId: invoiceId })
        });
        const nJson = await notionRes.json();
        const data = nJson.invoiceData;

        const dateStr = new Date(data.date).toISOString().split('T')[0];
        const num = data.invoiceNumber || invoiceId;

        const stripeUrl = data.hostedInvoiceUrl;
        const m = stripeUrl.match(/invoice\.stripe\.com\/i\/([^\/]+)\/([^\?]+)/);
        if (!m) throw new Error("Bad Stripe URL");

        const acct = m[1];
        const id = m[2];

        const [invData, recData] = await Promise.all([
            fetchDataViaJson(`https://invoicedata.stripe.com/invoice_pdf_file_url/${acct}/${id}`),
            fetchDataViaJson(`https://invoicedata.stripe.com/invoice_receipt_file_url/${acct}/${id}`)
        ]);

        return {
            invoice: { name: `${dateStr}__Invoice-${num}.pdf`, data: invData },
            receipt: { name: `${dateStr}__Receipt-${num}.pdf`, data: recData }
        };
    }

    async function fetchDataViaJson(apiUrl) {
        try {
            const json = await gmGetJSON(apiUrl);
            if (!json.file_url) return null;
            return await gmGetBytes(json.file_url);
        } catch (e) { return null; }
    }

    // --- ZIP ENGINE (No Lib) ---
    function createZip(files) {
        const parts = [];
        let offset = 0;
        const centralDirectory = [];
        const crcTable = new Int32Array(256);
        for (let i = 0; i < 256; i++) {
            let c = i;
            for (let k = 0; k < 8; k++) c = (c & 1) ? (0xEDB88320 ^ (c >>> 1)) : (c >>> 1);
            crcTable[i] = c;
        }
        function crc32(u8arr) {
            let crc = -1;
            for (let i = 0; i < u8arr.length; i++) crc = (crc >>> 8) ^ crcTable[(crc ^ u8arr[i]) & 0xFF];
            return (crc ^ -1) >>> 0;
        }
        const encoder = new TextEncoder();

        for (const file of files) {
            if(!file.data) continue;
            const nameBytes = encoder.encode(file.name);
            const data = file.data;
            const crc = crc32(data);
            const header = new Uint8Array(30 + nameBytes.length);
            const view = new DataView(header.buffer);
            view.setUint32(0, 0x04034b50, true);
            view.setUint16(4, 0x000a, true);
            view.setUint16(6, 0x0000, true);
            view.setUint16(8, 0x0000, true);
            view.setUint32(14, crc, true);
            view.setUint32(18, data.length, true);
            view.setUint32(22, data.length, true);
            view.setUint16(26, nameBytes.length, true);
            view.setUint16(28, 0, true);
            header.set(nameBytes, 30);
            parts.push(header);
            parts.push(data);
            centralDirectory.push({ nameBytes, crc, size: data.length, offset });
            offset += header.length + data.length;
        }
        const cdStartOffset = offset;
        for (const cd of centralDirectory) {
            const header = new Uint8Array(46 + cd.nameBytes.length);
            const view = new DataView(header.buffer);
            view.setUint32(0, 0x02014b50, true);
            view.setUint16(4, 0x000a, true);
            view.setUint16(6, 0x000a, true);
            view.setUint16(8, 0x0000, true);
            view.setUint16(10, 0x0000, true);
            view.setUint32(16, cd.crc, true);
            view.setUint32(20, cd.size, true);
            view.setUint32(24, cd.size, true);
            view.setUint16(28, cd.nameBytes.length, true);
            view.setUint16(30, 0, true);
            view.setUint16(32, 0, true);
            view.setUint16(34, 0, true);
            view.setUint16(36, 0, true);
            view.setUint32(38, 0, true);
            view.setUint32(42, cd.offset, true);
            header.set(cd.nameBytes, 46);
            parts.push(header);
            offset += header.length;
        }
        const eocd = new Uint8Array(22);
        const view = new DataView(eocd.buffer);
        view.setUint32(0, 0x06054b50, true);
        view.setUint16(4, 0, true);
        view.setUint16(6, 0, true);
        view.setUint16(8, centralDirectory.length, true);
        view.setUint16(10, centralDirectory.length, true);
        view.setUint32(12, offset - cdStartOffset, true);
        view.setUint32(16, cdStartOffset, true);
        parts.push(eocd);
        return new Blob(parts, { type: 'application/zip' });
    }

    // --- HELPERS ---
    function getRowElement(invoiceId) {
        const checkbox = document.getElementById(`nir-check-${invoiceId}`);
        return checkbox ? checkbox.closest('.nir-invoice-row-modified') : null;
    }
    function gmGetJSON(url) {
        return new Promise((resolve, reject) => {
            GM_xmlhttpRequest({
                method: "GET", url,
                onload: r => { try { resolve(JSON.parse(r.responseText)) } catch { reject() } },
                onerror: reject
            });
        });
    }
    function gmGetBytes(url) {
        return new Promise((resolve, reject) => {
            GM_xmlhttpRequest({
                method: "GET", url, responseType: 'arraybuffer',
                onload: r => { if(r.status===200) resolve(new Uint8Array(r.response)); else reject(); },
                onerror: reject
            });
        });
    }
    function saveBlob(blob, name) {
        const a = document.createElement("a");
        a.href = URL.createObjectURL(blob);
        a.download = name;
        a.style.display = 'none';
        document.body.appendChild(a);
        a.click();
        setTimeout(() => { URL.revokeObjectURL(a.href); document.body.removeChild(a); }, 1000);
    }
    function sleep(ms) { return new Promise(r => setTimeout(r, ms)); }
    // No-op log function for clean code
    function log() {}

})();