Summarize with AI

Adds a button or keyboard shortcut to summarize articles, news, and similar content using the OpenAI API (gpt-4o-mini model). The summary is displayed in an overlay with enhanced styling and a loading animation.

As of 16. 10. 2024. See the latest version.

// ==UserScript==
// @name         Summarize with AI
// @namespace    https://github.com/insign/summarize-with-ai
// @version      2024.10.11.1526
// @description  Adds a button or keyboard shortcut to summarize articles, news, and similar content using the OpenAI API (gpt-4o-mini model). The summary is displayed in an overlay with enhanced styling and a loading animation.
// @author       Hélio <[email protected]>
// @license      WTFPL
// @match        *://*/*
// @grant        GM.addStyle
// @grant        GM.xmlHttpRequest
// @grant        GM.setValue
// @grant        GM.getValue
// @connect      api.openai.com
// @require      https://cdnjs.cloudflare.com/ajax/libs/readability/0.5.0/Readability.min.js
// ==/UserScript==

(function () {
    'use strict';

    /*** Constants ***/
    const BUTTON_ID = 'summarize-button';
    const OVERLAY_ID = 'summarize-overlay';
    const CLOSE_BUTTON_ID = 'summarize-close';
    const CONTENT_ID = 'summarize-content';
    const ERROR_ID = 'summarize-error';
    const API_URL = 'https://api.openai.com/v1/chat/completions';

    /*** Initialization ***/
    let isArticle = false;
    let articleTitle = '';
    let articleContent = '';

    initialize();

    /**
     * Initializes the Userscript by detecting article content,
     * setting up the summarize button and keyboard shortcuts,
     * and adding necessary event listeners.
     */
    async function initialize() {
        try {
            const articleData = getArticleData();
            if (articleData) {
                isArticle = true;
                ({ title: articleTitle, content: articleContent } = articleData);
                addSummarizeButton();
                setupKeyboardShortcuts();
            } else {
                isArticle = false;
                hideElement(BUTTON_ID);
                disableKeyboardShortcuts();
            }

            setupFocusListeners();
        } catch (error) {
            console.error('Initialization error:', error);
            showErrorNotification('Error during initialization.');
        }
    }

    /*** Function Definitions ***/

    /**
     * Extracts the article's title and content using Readability.
     * @returns {Object|null} Article data or null if not found.
     */
    function getArticleData() {
        try {
            const clonedDoc = document.cloneNode(true);
            // Remove unwanted tags to prevent parsing issues
            clonedDoc.querySelectorAll('script, style').forEach(el => el.remove());

            const reader = new Readability(clonedDoc);
            const article = reader.parse();

            if (article && article.content && article.title) {
                return {
                    title: article.title,
                    content: article.textContent // Plain text for API
                };
            }
            return null;
        } catch (error) {
            console.error('Readability parsing error:', error);
            return null;
        }
    }

    /**
     * Adds the summarize button to the page with predefined styles and event listeners.
     */
    function addSummarizeButton() {
        if (document.getElementById(BUTTON_ID)) return; // Prevent duplicates

        const button = document.createElement('div');
        button.id = BUTTON_ID;
        button.innerText = 'S';
        document.body.appendChild(button);

        button.addEventListener('click', processSummarization);
        button.addEventListener('dblclick', resetApiKey);

        injectStyles();
    }

    /**
     * Injects CSS styles for the summarize button, overlay, and notifications.
     */
    function injectStyles() {
        GM.addStyle(`
            /* Summarize Button Styling */
            #${BUTTON_ID} {
                position: fixed;
                bottom: 20px;
                right: 20px;
                width: 60px;
                height: 60px;
                background-color: rgba(0, 123, 255, 0.9);
                color: white;
                font-size: 28px;
                font-weight: bold;
                text-align: center;
                line-height: 60px;
                border-radius: 50%;
                cursor: pointer;
                z-index: 99999;
                box-shadow: 0 2px 5px rgba(0,0,0,0.3);
                transition: background-color 0.3s, transform 0.3s;
                user-select: none;
                font-family: Arial, sans-serif;
            }
            #${BUTTON_ID}:hover {
                background-color: rgba(0, 123, 255, 1);
                transform: scale(1.1);
            }

            /* Summary Overlay Styling */
            #${OVERLAY_ID} {
                position: fixed;
                top: 0;
                left: 0;
                width: 100%;
                height: 100%;
                background-color: rgba(0, 0, 0, 0.5);
                z-index: 100000;
                display: flex;
                align-items: center;
                justify-content: center;
                overflow: auto;
                font-family: Arial, sans-serif;
            }
            #${CONTENT_ID} {
                background-color: #fff;
                padding: 30px;
                border-radius: 10px;
                box-shadow: 0 0 15px rgba(0,0,0,0.5);
                max-width: 90%;
                max-height: 90%;
                overflow: auto;
                position: relative;
                font-size: 1.2em;
                color: #333;
            }
            #${CLOSE_BUTTON_ID} {
                position: absolute;
                top: 15px;
                right: 15px;
                cursor: pointer;
                font-size: 26px;
                color: #555;
                transition: transform 0.2s;
            }
            #${CLOSE_BUTTON_ID}:hover::after {
                content: "ESC";
                position: absolute;
                top: -12px;
                right: -10px;
                background: rgba(0, 0, 0, 0.7);
                color: white;
                padding: 2px 6px;
                border-radius: 4px;
                font-size: 12px;
                white-space: nowrap;
            }

            /* Error Notification Styling */
            #${ERROR_ID} {
                position: fixed;
                bottom: 20px;
                left: 20px;
                background-color: rgba(255,0,0,0.8);
                color: white;
                padding: 10px 20px;
                border-radius: 5px;
                z-index: 100001;
                font-size: 14px;
                font-family: Arial, sans-serif;
            }

            /* Loading Animation */
            .glow {
                font-size: 1.5em;
                color: #333;
                text-align: center;
                animation: glow 2s ease-in-out infinite alternate;
            }
            @keyframes glow {
                from {
                    color: #4b6cb7;
                    text-shadow: 0 0 10px #4b6cb7, 0 0 20px #4b6cb7, 0 0 30px #4b6cb7;
                }
                to {
                    color: #182848;
                    text-shadow: 0 0 20px #8e2de2, 0 0 30px #8e2de2, 0 0 40px #8e2de2;
                }
            }

            /* Responsive Design */
            @media (max-width: 768px) {
                #${BUTTON_ID} {
                    width: 70px;
                    height: 70px;
                    font-size: 32px;
                    line-height: 70px;
                    bottom: 15px;
                    right: 15px;
                }
                #${CONTENT_ID} {
                    padding: 25px;
                }
                #${ERROR_ID} {
                    bottom: 15px;
                    left: 15px;
                    font-size: 12px;
                }
            }
            @media (max-width: 480px) {
                #${BUTTON_ID} {
                    width: 80px;
                    height: 80px;
                    font-size: 36px;
                    line-height: 80px;
                    bottom: 10px;
                    right: 10px;
                }
                #${CONTENT_ID} {
                    padding: 20px;
                }
                #${ERROR_ID} {
                    padding: 8px 16px;
                    font-size: 11px;
                }
            }

            /* Clean List Styles */
            #${CONTENT_ID} ul, #${CONTENT_ID} p {
                list-style: none;
                padding: 0;
                margin: 0;
            }
        `);
    }

    /**
     * Sets up keyboard shortcuts for summarization.
     * 'S' key triggers summarization unless an input is focused.
     */
    function setupKeyboardShortcuts() {
        document.addEventListener('keydown', handleKeyDown);
    }

    /**
     * Disables keyboard shortcuts by removing the keydown event listener.
     */
    function disableKeyboardShortcuts() {
        document.removeEventListener('keydown', handleKeyDown);
    }

    /**
     * Handles keydown events to trigger summarization with 'S' key.
     * @param {KeyboardEvent} e
     */
    function handleKeyDown(e) {
        const active = document.activeElement;
        const isInput = active && (['INPUT', 'TEXTAREA', 'SELECT'].includes(active.tagName) || active.isContentEditable);

        if (isInput) {
            hideElement(BUTTON_ID);
            return;
        }

        if (e.key.toLowerCase() === 's') {
            e.preventDefault();
            triggerSummarization();
        } else {
            showElement(BUTTON_ID);
        }
    }

    /**
     * Monitors focus changes to show or hide the summarize button accordingly.
     */
    function setupFocusListeners() {
        document.addEventListener('focusin', toggleButtonVisibility);
        document.addEventListener('focusout', toggleButtonVisibility);
    }

    /**
     * Toggles the visibility of the summarize button based on focused element.
     */
    function toggleButtonVisibility() {
        const active = document.activeElement;
        const isInput = active && (['INPUT', 'TEXTAREA', 'SELECT'].includes(active.tagName) || active.isContentEditable);

        if (isInput) {
            hideElement(BUTTON_ID);
        } else if (isArticle) {
            showElement(BUTTON_ID);
        }
    }

    /**
     * Shows an element by its ID.
     * @param {string} id
     */
    function showElement(id) {
        const el = document.getElementById(id);
        if (el) el.style.display = 'block';
    }

    /**
     * Hides an element by its ID.
     * @param {string} id
     */
    function hideElement(id) {
        const el = document.getElementById(id);
        if (el) el.style.display = 'none';
    }

    /**
     * Triggers the summarization process, handling both button clicks and keyboard shortcuts.
     */
    function triggerSummarization() {
        if (!isArticle) {
            alert('This page may not be an article. Proceeding to summarize anyway.');
        }
        processSummarization();
    }

    /**
     * Initiates the summarization process by fetching the API key,
     * displaying the loading overlay, and sending data to OpenAI API.
     */
    async function processSummarization() {
        try {
            const apiKey = await getApiKey();
            if (!apiKey) return;

            const payload = { title: articleTitle, content: articleContent };

            showSummaryOverlay('<p class="glow">Summarizing</p>');
            await summarizeContent(apiKey, payload);
        } catch (error) {
            console.error('Summarization process error:', error);
            showErrorNotification('Error: Failed to initiate summarization.');
            updateSummaryOverlay('<p>Error: Failed to initiate summarization.</p>');
        }
    }

    /**
     * Handles double-click on the summarize button to reset the API key.
     */
    function resetApiKey() {
        const newKey = prompt('Please enter your OpenAI API key:', '');
        if (newKey) {
            GM.setValue('openai_api_key', newKey.trim())
                .then(() => alert('API key successfully updated.'))
                .catch(error => {
                    alert('Error updating the API key.');
                    console.error('API key update error:', error);
                });
        }
    }

    /**
     * Retrieves the OpenAI API key from storage or prompts the user to input it.
     * @returns {Promise<string|null>} The API key or null if not provided.
     */
    async function getApiKey() {
        try {
            let apiKey = (await GM.getValue('openai_api_key'))?.trim();
            if (!apiKey) {
                const userInput = prompt('Please enter your OpenAI API key:', '');
                if (userInput) {
                    apiKey = userInput.trim();
                    await GM.setValue('openai_api_key', apiKey);
                } else {
                    alert('An API key is required to generate a summary.');
                    return null;
                }
            }
            return apiKey;
        } catch (error) {
            console.error('API key retrieval error:', error);
            alert('Failed to retrieve the API key.');
            return null;
        }
    }

    /**
     * Displays the summary overlay with specified content.
     * Sets up event listeners for closing the overlay.
     * @param {string} content - HTML content to display.
     */
    function showSummaryOverlay(content) {
        if (document.getElementById(OVERLAY_ID)) {
            updateSummaryOverlay(content);
            return;
        }

        const overlay = document.createElement('div');
        overlay.id = OVERLAY_ID;
        overlay.innerHTML = `
            <div id="${CONTENT_ID}">
                <div id="${CLOSE_BUTTON_ID}">&times;</div>
                ${content}
            </div>
        `;
        document.body.appendChild(overlay);
        document.body.style.overflow = 'hidden';

        const contentContainer = document.getElementById(CONTENT_ID);
        document.getElementById(CLOSE_BUTTON_ID).addEventListener('click', closeOverlay);
        overlay.addEventListener('click', (e) => {
            if (!contentContainer.contains(e.target)) {
                closeOverlay();
            }
        });
        document.addEventListener('keydown', onEscapePress);

        /**
         * Closes the overlay and restores page scrolling.
         */
        function closeOverlay() {
            document.getElementById(OVERLAY_ID)?.remove();
            document.body.style.overflow = '';
            document.removeEventListener('keydown', onEscapePress);
        }

        /**
         * Handles 'Escape' key press to close the overlay.
         * @param {KeyboardEvent} e
         */
        function onEscapePress(e) {
            if (e.key === 'Escape') closeOverlay();
        }
    }

    /**
     * Updates the content within the summary overlay.
     * @param {string} content - New HTML content to display.
     */
    function updateSummaryOverlay(content) {
        const contentDiv = document.getElementById(CONTENT_ID);
        if (contentDiv) {
            contentDiv.innerHTML = `<div id="${CLOSE_BUTTON_ID}">&times;</div>` + content;
            document.getElementById(CLOSE_BUTTON_ID).addEventListener('click', closeOverlay);
        }

        /**
         * Closes the summary overlay and re-enables background scrolling.
         */
        function closeOverlay() {
            document.getElementById(OVERLAY_ID)?.remove();
            document.body.style.overflow = '';
            document.removeEventListener('keydown', onEscapePress);
        }

        /**
         * Handles the 'Escape' key press to close the overlay.
         * @param {KeyboardEvent} e
         */
        function onEscapePress(e) {
            if (e.key === 'Escape') closeOverlay();
        }
    }

    /**
     * Displays an error notification on the page.
     * @param {string} message - The error message to display.
     */
    function showErrorNotification(message) {
        if (document.getElementById(ERROR_ID)) {
            document.getElementById(ERROR_ID).innerText = message;
            return;
        }

        const errorDiv = document.createElement('div');
        errorDiv.id = ERROR_ID;
        errorDiv.innerText = message;
        document.body.appendChild(errorDiv);

        setTimeout(() => {
            errorDiv.remove();
        }, 4000);
    }

    /**
     * Sends the article data to the OpenAI API to generate a summary.
     * Handles the response and updates the overlay accordingly.
     * @param {string} apiKey - The OpenAI API key.
     * @param {Object} payload - Contains title and content of the article.
     */
    async function summarizeContent(apiKey, payload) {
        try {
            const userLanguage = navigator.language || 'en-US';

            const requestData = {
                model: 'gpt-4o-mini',
                messages: [
                    {
                        role: 'system',
                        content: `You are a helpful assistant that provides clear and affirmative explanations of the content within an article based on its title and content. Your summary should convey exactly what is contained in the article and each of its sections, ensuring that all crucial and fundamental points are covered so that the reader does not miss any important details without needing to read the entire article. Additionally, ensure that the summary addresses the proposition of the title while encompassing a broader overview of the article's content, as the full article answers more than what is suggested by the title.

You should generate a concise summary that includes a brief introduction followed by a list of topics. For the topics, use appropriate emojis as bullet points, and each topic should consist of descriptive and affirmative statements summarizing its subject.

You must always use HTML tags to structure the summary text. You must always use the user's language in addition to the article's original language. The generated HTML should be ready to be injected into the target location, and you must never use markdown.

Required structure:
- Do not add any title
- Use a maximum of 2 sentences for the introduction.
- Use appropriate emojis for topics
- Do not add text like "Article Summary" or "Summary of the article" in the summary, nor "Introduction", "Topics", "Conclusion", etc.

User language: ${userLanguage}.
Adapt the text to be short, concise, and informative without losing important details.`
                    },
                    {
                        role: 'user',
                        content: `Title: ${payload.title}\n\nContent: ${payload.content}`
                    }
                ],
                max_tokens: 500,
                temperature: 0.5,
                n: 1,
                stream: false
            };

            GM.xmlHttpRequest({
                method: 'POST',
                url: API_URL,
                headers: {
                    'Content-Type': 'application/json',
                    'Authorization': `Bearer ${apiKey}`
                },
                data: JSON.stringify(requestData),
                onload: function (response) {
                    handleApiResponse(response);
                },
                onerror: () => handleRequestError('Network issue.'),
                onabort: () => handleRequestError('Request aborted.')
            });
        } catch (error) {
            console.error('API communication error:', error);
            showErrorNotification('Error: Failed to communicate with the API.');
            updateSummaryOverlay('<p>Error: Failed to communicate with the API.</p>');
        }
    }

    /**
     * Handles the API response, updating the overlay or showing errors as needed.
     * @param {Object} response - The API response object.
     */
    function handleApiResponse(response) {
        if (response.status !== 200) {
            let errorMessage = `Error: Failed to retrieve summary. Status: ${response.status || 'N/A'}`;
            if (response.status === 401) {
                errorMessage = 'Error: Invalid API key.';
            } else if (response.status === undefined) {
                errorMessage = 'Error: Unexpected API response.';
            }
            showErrorNotification(errorMessage);
            updateSummaryOverlay(`<p>${errorMessage}</p>`);
            console.error('API response error:', response);
            return;
        }

        try {
            const resData = JSON.parse(response.responseText);
            const summary = resData?.choices?.[0]?.message?.content;
            if (summary) {
                const formattedSummary = summary.replace(/\n+/g, '<br>');
                updateSummaryOverlay(formattedSummary);
            } else {
                throw new Error('Invalid API response structure.');
            }
        } catch (parseError) {
            console.error('Error parsing API response:', parseError);
            showErrorNotification('Error: Failed to parse API response.');
            updateSummaryOverlay('<p>Error: Failed to parse API response.</p>');
        }
    }

    /**
     * Handles request errors by displaying notifications and updating the overlay.
     * @param {string} message - The error message to display.
     */
    function handleRequestError(message) {
        showErrorNotification(`Error: ${message}`);
        updateSummaryOverlay(`<p>Error: ${message}</p>`);
    }

})();