Basic Drawing Canvas Extension for chatgpt.com

Simple Tampermonkey extension to add a drawing canvas to chatGPT. It shows up as a pencil button on the left side of the screen and has a quick copy to clipboard button.

// ==UserScript==
// @name         ChatGPT_Drawing_Tool
// @namespace    http://tampermonkey.net/
// @version      1.3
// @description  Adds a standalone drawing tool button to ChatGPT for drawing and copying images to clipboard
// @match        https://chatgpt.com/*
// @icon         https://www.google.com/s2/favicons?sz=64&domain=chatgpt.com
// @grant        none
// ==/UserScript==

(function() {
    'use strict';

    // Light gray color used by ChatGPT for its text input
    const lightGray = '#E0E0E0';

    // Create the drawing button
    const drawingButton = document.createElement('button');
    drawingButton.innerHTML = '🖋️'; // Change this to any other emoji from the options if needed
    drawingButton.style.position = 'fixed';
    drawingButton.style.right = '20px';
    drawingButton.style.top = '50%'; // Middle of the screen vertically
    drawingButton.style.transform = 'translateY(-50%)'; // Adjust for perfect vertical centering
    drawingButton.style.padding = '9px'; // 10% smaller than original 10px padding
    drawingButton.style.fontSize = '18px'; // 10% smaller than original 20px font size
    drawingButton.style.cursor = 'pointer';
    drawingButton.style.backgroundColor = lightGray; // Update to light gray color
    drawingButton.style.color = 'white';
    drawingButton.style.border = 'none';
    drawingButton.style.borderRadius = '5px';
    drawingButton.title = 'Draw';

    document.body.appendChild(drawingButton);

    // Create the drawing canvas modal
    const drawingModal = document.createElement('div');
    drawingModal.style.position = 'fixed';
    drawingModal.style.top = '0';
    drawingModal.style.left = '0';
    drawingModal.style.width = '100%';
    drawingModal.style.height = '100%';
    drawingModal.style.backgroundColor = 'rgba(0,0,0,0.5)';
    drawingModal.style.display = 'none';
    drawingModal.style.zIndex = '1000';
    drawingModal.style.alignItems = 'center';
    drawingModal.style.justifyContent = 'center';

    const container = document.createElement('div');
    container.style.display = 'flex';
    container.style.border = '5px solid black';
    container.style.backgroundColor = 'white';
    container.style.position = 'relative';

    const canvasContainer = document.createElement('div');
    canvasContainer.id = 'canvas-container';
    canvasContainer.style.flex = '1';
    canvasContainer.style.display = 'block';
    canvasContainer.style.position = 'relative';
    canvasContainer.style.backgroundColor = 'white';

    const canvas = document.createElement('canvas');
    canvas.id = 'cfd';
    canvas.width = 500;
    canvas.height = 500;
    canvasContainer.appendChild(canvas);

    // Create options box
    const optionsBox = document.createElement('div');
    optionsBox.style.display = 'flex';
    optionsBox.style.flexDirection = 'column';
    optionsBox.style.alignItems = 'center';
    optionsBox.style.justifyContent = 'space-between';
    optionsBox.style.padding = '10px';
    optionsBox.style.backgroundColor = 'white';
    optionsBox.style.borderLeft = '5px solid black';
    optionsBox.style.height = '500px';

    const clearButton = document.createElement('button');
    clearButton.id = 'clear';
    clearButton.innerHTML = '🗑️'; // Trash can emoji
    clearButton.style.marginBottom = '10px';
    clearButton.style.fontSize = '20px';
    optionsBox.appendChild(clearButton);

    const copyButton = document.createElement('button');
    copyButton.id = 'copy';
    copyButton.innerHTML = '📋'; // Clipboard emoji
    copyButton.style.marginBottom = '10px';
    copyButton.style.fontSize = '20px';
    optionsBox.appendChild(copyButton);

    const closeButton = document.createElement('button');
    closeButton.id = 'close';
    closeButton.innerHTML = '❌'; // Cross mark emoji
    closeButton.style.marginBottom = '10px';
    closeButton.style.fontSize = '20px';
    optionsBox.appendChild(closeButton);

    const penButton = document.createElement('button');
    penButton.id = 'pen';
    penButton.className = 'color';
    penButton.innerHTML = '✏️'; // Pen emoji
    penButton.style.margin = '2px';
    penButton.style.width = '100%';
    penButton.style.fontSize = '20px';
    optionsBox.appendChild(penButton);

    const eraserButton = document.createElement('button');
    eraserButton.id = 'eraser';
    eraserButton.className = 'color';
    eraserButton.innerHTML = '🧽'; // Eraser emoji
    eraserButton.style.margin = '2px';
    eraserButton.style.width = '100%';
    eraserButton.style.fontSize = '20px';
    optionsBox.appendChild(eraserButton);

    const lineWidthInput = document.createElement('input');
    lineWidthInput.type = 'range';
    lineWidthInput.min = '1';
    lineWidthInput.max = '50';
    lineWidthInput.value = '5';
    lineWidthInput.style.marginTop = '10px';
    optionsBox.appendChild(lineWidthInput);

    container.appendChild(canvasContainer);
    container.appendChild(optionsBox);
    drawingModal.appendChild(container);
    document.body.appendChild(drawingModal);

    // Drawing functionality
    const ctx = canvas.getContext('2d');
    let isPainting = false;
    let penLineWidth = 5;
    let eraserLineWidth = 25; // 5 times the pen line width
    let activeTool = 'pen';
    let startX;
    let startY;
    ctx.strokeStyle = 'black';
    ctx.lineWidth = penLineWidth;

    canvas.addEventListener('mousedown', (e) => {
        isPainting = true;
        startX = e.clientX - canvas.getBoundingClientRect().left;
        startY = e.clientY - canvas.getBoundingClientRect().top;
    });

    canvas.addEventListener('mouseup', () => {
        isPainting = false;
        ctx.beginPath();
    });

    canvas.addEventListener('mousemove', (e) => {
        if (!isPainting) return;
        ctx.lineCap = 'round';

        const x = e.clientX - canvas.getBoundingClientRect().left;
        const y = e.clientY - canvas.getBoundingClientRect().top;

        ctx.lineTo(x, y);
        ctx.stroke();
        ctx.beginPath();
        ctx.moveTo(x, y);
    });

    clearButton.addEventListener('click', () => {
        ctx.clearRect(0, 0, canvas.width, canvas.height);
    });

    penButton.addEventListener('click', () => {
        activeTool = 'pen';
        ctx.strokeStyle = 'black';
        ctx.lineWidth = penLineWidth;
    });

    eraserButton.addEventListener('click', () => {
        activeTool = 'eraser';
        ctx.strokeStyle = 'white';
        ctx.lineWidth = eraserLineWidth;
    });

    lineWidthInput.addEventListener('input', (e) => {
        const newLineWidth = e.target.value;
        if (activeTool === 'pen') {
            penLineWidth = newLineWidth;
            ctx.lineWidth = penLineWidth;
        } else if (activeTool === 'eraser') {
            eraserLineWidth = newLineWidth;
            ctx.lineWidth = eraserLineWidth;
        }
    });

    drawingButton.onclick = () => {
        drawingModal.style.display = 'flex';
    };

    closeButton.onclick = () => {
        drawingModal.style.display = 'none';
    };

    copyButton.onclick = async () => {
        const offScreenCanvas = document.createElement('canvas');
        const offScreenCtx = offScreenCanvas.getContext('2d');
        offScreenCanvas.width = canvas.width;
        offScreenCanvas.height = canvas.height;

        // Fill the offscreen canvas with a white background
        offScreenCtx.fillStyle = 'white';
        offScreenCtx.fillRect(0, 0, offScreenCanvas.width, offScreenCanvas.height);

        // Draw the original canvas on top of the white background
        offScreenCtx.drawImage(canvas, 0, 0);

        offScreenCanvas.toBlob(async (blob) => {
            const item = new ClipboardItem({ 'image/png': blob });
            try {
                await navigator.clipboard.write([item]);
                drawingModal.style.display = 'none';
            } catch (err) {
                console.error('Failed to copy image to clipboard: ', err);
            }
        });
    };

})();