hand-written / index.html
Mustafa7assan's picture
Add 2 files
cb10101 verified
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Handwriting Canvas</title>
<script src="https://cdn.tailwindcss.com"></script>
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css">
<style>
@import url('https://fonts.googleapis.com/css2?family=Caveat:wght@400;700&display=swap');
.handwriting {
font-family: 'Caveat', cursive;
}
canvas {
touch-action: none;
box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1), 0 2px 4px -1px rgba(0, 0, 0, 0.06);
}
.tool-btn {
transition: all 0.2s ease;
}
.tool-btn.active {
transform: scale(1.1);
box-shadow: 0 0 0 2px rgba(59, 130, 246, 0.5);
}
#drawingCanvas {
background-image: url('');
}
.ink-effect {
position: relative;
}
.ink-effect::after {
content: '';
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: radial-gradient(circle, transparent 0%, rgba(0,0,0,0.05) 100%);
pointer-events: none;
}
</style>
</head>
<body class="bg-gray-100 min-h-screen">
<div class="container mx-auto px-4 py-8">
<header class="mb-8 text-center">
<h1 class="text-4xl font-bold text-indigo-700 handwriting mb-2">Handwriting Canvas</h1>
<p class="text-gray-600">Draw or write naturally on images or blank paper</p>
</header>
<div class="flex flex-col lg:flex-row gap-6">
<!-- Tools Panel -->
<div class="bg-white rounded-lg shadow-md p-4 lg:w-1/4">
<h2 class="text-xl font-semibold mb-4 text-gray-800 handwriting">Tools</h2>
<div class="space-y-4">
<!-- Color Picker -->
<div>
<label class="block text-sm font-medium text-gray-700 mb-1 handwriting">Pen Color</label>
<div class="flex flex-wrap gap-2">
<button data-color="#000000" class="tool-btn w-8 h-8 rounded-full bg-black border-2 border-gray-300"></button>
<button data-color="#e53e3e" class="tool-btn w-8 h-8 rounded-full bg-red-600 border-2 border-gray-300"></button>
<button data-color="#3182ce" class="tool-btn w-8 h-8 rounded-full bg-blue-600 border-2 border-gray-300"></button>
<button data-color="#38a169" class="tool-btn w-8 h-8 rounded-full bg-green-600 border-2 border-gray-300"></button>
<button data-color="#805ad5" class="tool-btn w-8 h-8 rounded-full bg-purple-600 border-2 border-gray-300"></button>
</div>
</div>
<!-- Brush Size -->
<div>
<label class="block text-sm font-medium text-gray-700 mb-1 handwriting">Brush Size</label>
<input type="range" id="brushSize" min="1" max="20" value="3" class="w-full h-2 bg-gray-200 rounded-lg appearance-none cursor-pointer">
<div class="flex justify-between text-xs text-gray-500 mt-1">
<span>Thin</span>
<span>Thick</span>
</div>
</div>
<!-- Background Options -->
<div>
<label class="block text-sm font-medium text-gray-700 mb-1 handwriting">Background</label>
<div class="flex flex-wrap gap-2">
<button data-bg="blank" class="tool-btn px-3 py-1 bg-white border border-gray-300 rounded-md handwriting">White Paper</button>
<button data-bg="grid" class="tool-btn px-3 py-1 bg-white border border-gray-300 rounded-md handwriting">Grid Paper</button>
<button id="uploadImageBtn" class="tool-btn px-3 py-1 bg-indigo-100 text-indigo-700 border border-indigo-200 rounded-md handwriting">
<i class="fas fa-image mr-1"></i> Upload Image
</button>
<input type="file" id="imageUpload" accept="image/*" class="hidden">
</div>
</div>
<!-- Actions -->
<div class="pt-4 border-t border-gray-200">
<button id="clearCanvas" class="w-full bg-red-100 text-red-700 py-2 rounded-md hover:bg-red-200 transition handwriting">
<i class="fas fa-trash-alt mr-2"></i> Clear Canvas
</button>
<button id="downloadCanvas" class="w-full mt-2 bg-indigo-100 text-indigo-700 py-2 rounded-md hover:bg-indigo-200 transition handwriting">
<i class="fas fa-download mr-2"></i> Download
</button>
</div>
<!-- Handwriting Toggle -->
<div class="pt-4 border-t border-gray-200">
<label class="inline-flex items-center cursor-pointer">
<input type="checkbox" id="handwritingEffect" class="sr-only peer" checked>
<div class="relative w-11 h-6 bg-gray-200 peer-focus:outline-none peer-focus:ring-4 peer-focus:ring-indigo-300 rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:border-gray-300 after:border after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-indigo-600"></div>
<span class="ml-3 text-sm font-medium text-gray-700 handwriting">Handwriting Effect</span>
</label>
</div>
</div>
</div>
<!-- Canvas Area -->
<div class="flex-1">
<div class="bg-white rounded-lg shadow-md p-4">
<div class="relative overflow-hidden rounded-md ink-effect">
<canvas id="drawingCanvas" class="bg-white w-full h-96 lg:h-[500px] border border-gray-200"></canvas>
</div>
<div class="mt-4 flex justify-center space-x-4">
<button id="undoBtn" class="px-4 py-2 bg-gray-100 text-gray-700 rounded-md hover:bg-gray-200 transition handwriting">
<i class="fas fa-undo mr-2"></i> Undo
</button>
<button id="redoBtn" class="px-4 py-2 bg-gray-100 text-gray-700 rounded-md hover:bg-gray-200 transition handwriting">
<i class="fas fa-redo mr-2"></i> Redo
</button>
</div>
</div>
</div>
</div>
</div>
<script>
document.addEventListener('DOMContentLoaded', function() {
const canvas = document.getElementById('drawingCanvas');
const ctx = canvas.getContext('2d');
const colorButtons = document.querySelectorAll('[data-color]');
const bgButtons = document.querySelectorAll('[data-bg]');
const brushSizeInput = document.getElementById('brushSize');
const clearCanvasBtn = document.getElementById('clearCanvas');
const downloadCanvasBtn = document.getElementById('downloadCanvas');
const uploadImageBtn = document.getElementById('uploadImageBtn');
const imageUpload = document.getElementById('imageUpload');
const handwritingEffect = document.getElementById('handwritingEffect');
const undoBtn = document.getElementById('undoBtn');
const redoBtn = document.getElementById('redoBtn');
let isDrawing = false;
let currentColor = '#000000';
let currentBrushSize = 3;
let handwritingEnabled = true;
let canvasHistory = [];
let historyIndex = -1;
// Set canvas size
function resizeCanvas() {
const container = canvas.parentElement;
canvas.width = container.clientWidth;
canvas.height = container.clientHeight;
redrawCanvas();
}
// Initialize canvas
function initCanvas() {
resizeCanvas();
setBackground('blank');
// Add event listeners for window resize
window.addEventListener('resize', () => {
resizeCanvas();
});
// Save initial state
saveCanvasState();
}
// Set background
function setBackground(type) {
if (type === 'blank') {
canvas.style.backgroundImage = 'none';
canvas.style.backgroundColor = '#ffffff';
} else if (type === 'grid') {
canvas.style.backgroundImage = 'url("")';
}
}
// Save canvas state to history
function saveCanvasState() {
// Remove any states after current index (for redo)
if (historyIndex < canvasHistory.length - 1) {
canvasHistory = canvasHistory.slice(0, historyIndex + 1);
}
const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);
canvasHistory.push(imageData);
historyIndex = canvasHistory.length - 1;
updateUndoRedoButtons();
}
// Update undo/redo button states
function updateUndoRedoButtons() {
undoBtn.disabled = historyIndex <= 0;
redoBtn.disabled = historyIndex >= canvasHistory.length - 1;
}
// Undo action
function undo() {
if (historyIndex > 0) {
historyIndex--;
redrawCanvas();
}
}
// Redo action
function redo() {
if (historyIndex < canvasHistory.length - 1) {
historyIndex++;
redrawCanvas();
}
}
// Redraw canvas from history
function redrawCanvas() {
if (canvasHistory.length > 0) {
ctx.putImageData(canvasHistory[historyIndex], 0, 0);
}
}
// Start drawing
function startDrawing(e) {
isDrawing = true;
draw(e);
}
// Stop drawing
function stopDrawing() {
isDrawing = false;
ctx.beginPath();
saveCanvasState();
}
// Draw on canvas
function draw(e) {
if (!isDrawing) return;
ctx.lineCap = 'round';
ctx.lineJoin = 'round';
ctx.strokeStyle = currentColor;
ctx.lineWidth = currentBrushSize;
// Get position (handling both mouse and touch events)
let x, y;
if (e.type.includes('touch')) {
const rect = canvas.getBoundingClientRect();
x = e.touches[0].clientX - rect.left;
y = e.touches[0].clientY - rect.top;
} else {
x = e.offsetX;
y = e.offsetY;
}
// Apply handwriting effect by adding slight randomness
if (handwritingEnabled) {
ctx.lineTo(
x + (Math.random() - 0.5) * currentBrushSize * 0.5,
y + (Math.random() - 0.5) * currentBrushSize * 0.5
);
} else {
ctx.lineTo(x, y);
}
ctx.stroke();
ctx.beginPath();
ctx.moveTo(x, y);
}
// Clear canvas
function clearCanvas() {
ctx.clearRect(0, 0, canvas.width, canvas.height);
saveCanvasState();
}
// Download canvas as image
function downloadCanvas() {
const link = document.createElement('a');
link.download = 'handwriting-' + new Date().toISOString().slice(0, 10) + '.png';
link.href = canvas.toDataURL('image/png');
link.click();
}
// Handle image upload
function handleImageUpload(e) {
const file = e.target.files[0];
if (!file) return;
const reader = new FileReader();
reader.onload = function(event) {
const img = new Image();
img.onload = function() {
// Clear canvas and draw the image
ctx.clearRect(0, 0, canvas.width, canvas.height);
// Calculate dimensions to maintain aspect ratio
const ratio = Math.min(
canvas.width / img.width,
canvas.height / img.height
);
const width = img.width * ratio;
const height = img.height * ratio;
const x = (canvas.width - width) / 2;
const y = (canvas.height - height) / 2;
ctx.drawImage(img, x, y, width, height);
saveCanvasState();
};
img.src = event.target.result;
};
reader.readAsDataURL(file);
}
// Event listeners
canvas.addEventListener('mousedown', startDrawing);
canvas.addEventListener('mousemove', draw);
canvas.addEventListener('mouseup', stopDrawing);
canvas.addEventListener('mouseout', stopDrawing);
// Touch events for mobile
canvas.addEventListener('touchstart', (e) => {
e.preventDefault();
startDrawing(e);
});
canvas.addEventListener('touchmove', (e) => {
e.preventDefault();
draw(e);
});
canvas.addEventListener('touchend', (e) => {
e.preventDefault();
stopDrawing();
});
// Color buttons
colorButtons.forEach(button => {
button.addEventListener('click', () => {
currentColor = button.dataset.color;
// Update active state
colorButtons.forEach(btn => btn.classList.remove('active'));
button.classList.add('active');
});
});
// Background buttons
bgButtons.forEach(button => {
button.addEventListener('click', () => {
setBackground(button.dataset.bg);
// Update active state
bgButtons.forEach(btn => btn.classList.remove('active'));
button.classList.add('active');
});
});
// Brush size
brushSizeInput.addEventListener('input', () => {
currentBrushSize = brushSizeInput.value;
});
// Clear canvas
clearCanvasBtn.addEventListener('click', clearCanvas);
// Download
downloadCanvasBtn.addEventListener('click', downloadCanvas);
// Image upload
uploadImageBtn.addEventListener('click', () => {
imageUpload.click();
});
imageUpload.addEventListener('change', handleImageUpload);
// Handwriting effect toggle
handwritingEffect.addEventListener('change', () => {
handwritingEnabled = handwritingEffect.checked;
});
// Undo/redo buttons
undoBtn.addEventListener('click', undo);
redoBtn.addEventListener('click', redo);
// Initialize
initCanvas();
// Set black as default active color
document.querySelector('[data-color="#000000"]').classList.add('active');
document.querySelector('[data-bg="blank"]').classList.add('active');
});
</script>
<p style="border-radius: 8px; text-align: center; font-size: 12px; color: #fff; margin-top: 16px;position: fixed; left: 8px; bottom: 8px; z-index: 10; background: rgba(0, 0, 0, 0.8); padding: 4px 8px;">Made with <img src="https://enzostvs-deepsite.hf.space/logo.svg" alt="DeepSite Logo" style="width: 16px; height: 16px; vertical-align: middle;display:inline-block;margin-right:3px;filter:brightness(0) invert(1);"><a href="https://enzostvs-deepsite.hf.space" style="color: #fff;text-decoration: underline;" target="_blank" >DeepSite</a> - 🧬 <a href="https://enzostvs-deepsite.hf.space?remix=Mustafa7assan/hand-written" style="color: #fff;text-decoration: underline;" target="_blank" >Remix</a></p></body>
</html>