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);
}
});
};
})();