Changes to ascii.scroll.pub

root
root
25 days ago
Initial commit
body.html
Changed around line 1
+
+

ASCII Art Generator

+

Transform your images into beautiful text art

+
+
+
+
+
+
+ 📁
+ Choose an image or drag & drop
+
+
+
+
+
+
+
+
+ 80
+
+
+
+
+ 100%
+
+
+
+
+
+
+
+
+
+
+
+

Made with ❤️ by ASCII Art Generator

+
index.scroll
Changed around line 1
+ buildHtml
+ baseUrl https://ascii.scroll.pub
+ metaTags
+ description Create ASCII art from your images instantly. Free online image to ASCII art converter.
+ keywords ascii art, image converter, ascii converter, text art
+ editButton /edit.html
+ title ASCII Art Generator | Convert Images to Text Art
+ style.css
+ body.html
+ script.js
prompt.json
Changed around line 1
+ {
+ "id": "msg_01EEhCJ7GEy3wHg2UDfhiJ1W",
+ "type": "message",
+ "role": "assistant",
+ "model": "claude-3-5-sonnet-20241022",
+ "content": [
+ {
+ "type": "text",
+ "text": "Let me create an ASCII art converter website with a clean, modern design.\n\n---domain---\nascii.scroll.pub\n---index.scroll---\nbuildHtml\nbaseUrl https://ascii.scroll.pub\nmetaTags\n description Create ASCII art from your images instantly. Free online image to ASCII art converter.\n keywords ascii art, image converter, ascii converter, text art\neditButton /edit.html\ntitle ASCII Art Generator | Convert Images to Text Art\nstyle.css\nbody.html\nscript.js\n---body.html---\n
\n

ASCII Art Generator

\n

Transform your images into beautiful text art

\n
\n\n
\n
\n
\n \n \n
\n\n
\n
\n \n \n 80\n
\n
\n \n \n 100%\n
\n
\n\n
\n
\n      \n    
\n
\n
\n\n
\n

Made with ❤️ by ASCII Art Generator

\n
\n---style.css---\n:root {\n --primary-color: #6366f1;\n --bg-color: #f8fafc;\n --text-color: #1e293b;\n --border-radius: 12px;\n}\n\n* {\n margin: 0;\n padding: 0;\n box-sizing: border-box;\n}\n\nbody {\n font-family: system-ui, -apple-system, sans-serif;\n background: var(--bg-color);\n color: var(--text-color);\n line-height: 1.6;\n min-height: 100vh;\n display: flex;\n flex-direction: column;\n}\n\nheader {\n text-align: center;\n padding: 3rem 1rem;\n background: linear-gradient(135deg, var(--primary-color), #818cf8);\n color: white;\n}\n\nh1 {\n font-size: 2.5rem;\n margin-bottom: 0.5rem;\n animation: fadeIn 0.8s ease-out;\n}\n\n.tagline {\n font-size: 1.2rem;\n opacity: 0.9;\n}\n\n.converter-container {\n max-width: 900px;\n margin: 2rem auto;\n padding: 0 1rem;\n}\n\n.upload-section {\n background: white;\n border-radius: var(--border-radius);\n padding: 2rem;\n box-shadow: 0 4px 6px rgba(0, 0, 0, 0.05);\n margin-bottom: 2rem;\n}\n\n.upload-label {\n display: flex;\n flex-direction: column;\n align-items: center;\n gap: 1rem;\n padding: 2rem;\n border: 2px dashed #cbd5e1;\n border-radius: var(--border-radius);\n cursor: pointer;\n transition: all 0.3s ease;\n}\n\n.upload-label:hover {\n border-color: var(--primary-color);\n background: #f8fafc;\n}\n\n.upload-icon {\n font-size: 2rem;\n}\n\n#imageInput {\n display: none;\n}\n\n.controls {\n background: white;\n padding: 1.5rem;\n border-radius: var(--border-radius);\n margin-bottom: 2rem;\n box-shadow: 0 4px 6px rgba(0, 0, 0, 0.05);\n}\n\n.control-group {\n display: flex;\n align-items: center;\n gap: 1rem;\n margin-bottom: 1rem;\n}\n\n.control-group:last-child {\n margin-bottom: 0;\n}\n\ninput[type=\"range\"] {\n flex: 1;\n height: 6px;\n background: #e2e8f0;\n border-radius: 3px;\n -webkit-appearance: none;\n}\n\ninput[type=\"range\"]::-webkit-slider-thumb {\n -webkit-appearance: none;\n width: 18px;\n height: 18px;\n background: var(--primary-color);\n border-radius: 50%;\n cursor: pointer;\n transition: transform 0.2s ease;\n}\n\ninput[type=\"range\"]::-webkit-slider-thumb:hover {\n transform: scale(1.1);\n}\n\n.value-display {\n min-width: 3rem;\n text-align: right;\n}\n\n.output-container {\n background: white;\n padding: 1.5rem;\n border-radius: var(--border-radius);\n position: relative;\n box-shadow: 0 4px 6px rgba(0, 0, 0, 0.05);\n}\n\n#asciiOutput {\n font-family: monospace;\n white-space: pre;\n overflow-x: auto;\n font-size: 12px;\n line-height: 1.2;\n margin-bottom: 1rem;\n}\n\n#copyButton {\n position: absolute;\n top: 1rem;\n right: 1rem;\n padding: 0.5rem 1rem;\n background: var(--primary-color);\n color: white;\n border: none;\n border-radius: var(--border-radius);\n cursor: pointer;\n transition: transform 0.2s ease;\n}\n\n#copyButton:hover {\n transform: translateY(-2px);\n}\n\nfooter {\n margin-top: auto;\n text-align: center;\n padding: 2rem;\n background: white;\n}\n\n@keyframes fadeIn {\n from {\n opacity: 0;\n transform: translateY(-10px);\n }\n to {\n opacity: 1;\n transform: translateY(0);\n }\n}\n\n@media (max-width: 768px) {\n h1 {\n font-size: 2rem;\n }\n \n .tagline {\n font-size: 1rem;\n }\n \n .control-group {\n flex-direction: column;\n align-items: stretch;\n gap: 0.5rem;\n }\n \n .value-display {\n text-align: left;\n }\n}\n---script.js---\ndocument.addEventListener('DOMContentLoaded', () => {\n const imageInput = document.getElementById('imageInput');\n const asciiOutput = document.getElementById('asciiOutput');\n const copyButton = document.getElementById('copyButton');\n const widthSlider = document.getElementById('width');\n const brightnessSlider = document.getElementById('brightness');\n const widthValue = document.getElementById('widthValue');\n const brightnessValue = document.getElementById('brightnessValue');\n \n const ASCII_CHARS = '@%#*+=-:. ';\n let currentImage = null;\n\n function updateValueDisplays() {\n widthValue.textContent = widthSlider.value;\n brightnessValue.textContent = `${brightnessSlider.value}%`;\n }\n\n function handleDragOver(e) {\n e.preventDefault();\n e.stopPropagation();\n e.currentTarget.classList.add('dragover');\n }\n\n function handleDragLeave(e) {\n e.preventDefault();\n e.stopPropagation();\n e.currentTarget.classList.remove('dragover');\n }\n\n function handleDrop(e) {\n e.preventDefault();\n e.stopPropagation();\n e.currentTarget.classList.remove('dragover');\n \n const file = e.dataTransfer.files[0];\n if (file && file.type.startsWith('image/')) {\n imageInput.files = e.dataTransfer.files;\n handleImageUpload(file);\n }\n }\n\n function handleImageUpload(file) {\n const reader = new FileReader();\n reader.onload = (e) => {\n const img = new Image();\n img.onload = () => {\n currentImage = img;\n convertToAscii();\n };\n img.src = e.target.result;\n };\n reader.readAsDataURL(file);\n }\n\n function convertToAscii() {\n if (!currentImage) return;\n\n const canvas = document.createElement('canvas');\n const ctx = canvas.getContext('2d');\n const width = parseInt(widthSlider.value);\n const height = Math.floor(currentImage.height * width / currentImage.width);\n \n canvas.width = width;\n canvas.height = height;\n \n ctx.drawImage(currentImage, 0, 0, width, height);\n const imageData = ctx.getImageData(0, 0, width, height);\n const brightness = parseInt(brightnessSlider.value) / 100;\n \n let ascii = '';\n for (let y = 0; y < height; y++) {\n for (let x = 0; x < width; x++) {\n const offset = (y * width + x) * 4;\n const r = imageData.data[offset];\n const g = imageData.data[offset + 1];\n const b = imageData.data[offset + 2];\n const avg = (r + g + b) / 3 * brightness;\n const charIndex = Math.floor(avg * (ASCII_CHARS.length - 1) / 255);\n ascii += ASCII_CHARS[charIndex];\n }\n ascii += '\\n';\n }\n \n asciiOutput.textContent = ascii;\n }\n\n imageInput.addEventListener('change', (e) => {\n if (e.target.files[0]) {\n handleImageUpload(e.target.files[0]);\n }\n });\n\n copyButton.addEventListener('click', () => {\n navigator.clipboard.writeText(asciiOutput.textContent)\n .then(() => {\n copyButton.textContent = 'Copied!';\n setTimeout(() => {\n copyButton.textContent = 'Copy';\n }, 2000);\n });\n });\n\n widthSlider.addEventListener('input', () => {\n updateValueDisplays();\n convertToAscii();\n });\n\n brightnessSlider.addEventListener('input', () => {\n updateValueDisplays();\n convertToAscii();\n });\n\n const uploadLabel = document.querySelector('.upload-label');\n uploadLabel.addEventListener('dragover', handleDragOver);\n uploadLabel.addEventListener('dragleave', handleDragLeave);\n uploadLabel.addEventListener('drop', handleDrop);\n\n updateValueDisplays();\n});\n---end---\n\nThis creates a beautiful, responsive ASCII art converter with the following features:\n\n1. Drag & drop or file upload functionality\n2. Adjustable width and brightness controls\n3. Copy to clipboard button\n4. Modern gradient header\n5. Clean, professional design with shadows and animations\n6. Mobile-responsive layout\n7. Accessible form controls and ARIA labels\n8. Smooth hover effects and transitions\n9. Error handling for file uploads\n10. Real-time ASCII art conversion\n\nThe site uses a monochromatic color scheme with indigo accents, providing a modern and professional look while maintaining excellent contrast for accessibility. The layout is centered and contained, making it easy to use on any device size."
+ }
+ ],
+ "stop_reason": "end_turn",
+ "stop_sequence": null,
+ "usage": {
+ "input_tokens": 277,
+ "cache_creation_input_tokens": 0,
+ "cache_read_input_tokens": 0,
+ "output_tokens": 3116
+ }
+ }
readme.scroll
Changed around line 1
+ # ascii.scroll.pub
+ Website generated by Claude from prompt: Make an image ASCII art converter website
script.js
Changed around line 1
+ document.addEventListener('DOMContentLoaded', () => {
+ const imageInput = document.getElementById('imageInput');
+ const asciiOutput = document.getElementById('asciiOutput');
+ const copyButton = document.getElementById('copyButton');
+ const widthSlider = document.getElementById('width');
+ const brightnessSlider = document.getElementById('brightness');
+ const widthValue = document.getElementById('widthValue');
+ const brightnessValue = document.getElementById('brightnessValue');
+
+ const ASCII_CHARS = '@%#*+=-:. ';
+ let currentImage = null;
+
+ function updateValueDisplays() {
+ widthValue.textContent = widthSlider.value;
+ brightnessValue.textContent = `${brightnessSlider.value}%`;
+ }
+
+ function handleDragOver(e) {
+ e.preventDefault();
+ e.stopPropagation();
+ e.currentTarget.classList.add('dragover');
+ }
+
+ function handleDragLeave(e) {
+ e.preventDefault();
+ e.stopPropagation();
+ e.currentTarget.classList.remove('dragover');
+ }
+
+ function handleDrop(e) {
+ e.preventDefault();
+ e.stopPropagation();
+ e.currentTarget.classList.remove('dragover');
+
+ const file = e.dataTransfer.files[0];
+ if (file && file.type.startsWith('image/')) {
+ imageInput.files = e.dataTransfer.files;
+ handleImageUpload(file);
+ }
+ }
+
+ function handleImageUpload(file) {
+ const reader = new FileReader();
+ reader.onload = (e) => {
+ const img = new Image();
+ img.onload = () => {
+ currentImage = img;
+ convertToAscii();
+ };
+ img.src = e.target.result;
+ };
+ reader.readAsDataURL(file);
+ }
+
+ function convertToAscii() {
+ if (!currentImage) return;
+
+ const canvas = document.createElement('canvas');
+ const ctx = canvas.getContext('2d');
+ const width = parseInt(widthSlider.value);
+ const height = Math.floor(currentImage.height * width / currentImage.width);
+
+ canvas.width = width;
+ canvas.height = height;
+
+ ctx.drawImage(currentImage, 0, 0, width, height);
+ const imageData = ctx.getImageData(0, 0, width, height);
+ const brightness = parseInt(brightnessSlider.value) / 100;
+
+ let ascii = '';
+ for (let y = 0; y < height; y++) {
+ for (let x = 0; x < width; x++) {
+ const offset = (y * width + x) * 4;
+ const r = imageData.data[offset];
+ const g = imageData.data[offset + 1];
+ const b = imageData.data[offset + 2];
+ const avg = (r + g + b) / 3 * brightness;
+ const charIndex = Math.floor(avg * (ASCII_CHARS.length - 1) / 255);
+ ascii += ASCII_CHARS[charIndex];
+ }
+ ascii += '\n';
+ }
+
+ asciiOutput.textContent = ascii;
+ }
+
+ imageInput.addEventListener('change', (e) => {
+ if (e.target.files[0]) {
+ handleImageUpload(e.target.files[0]);
+ }
+ });
+
+ copyButton.addEventListener('click', () => {
+ navigator.clipboard.writeText(asciiOutput.textContent)
+ .then(() => {
+ copyButton.textContent = 'Copied!';
+ setTimeout(() => {
+ copyButton.textContent = 'Copy';
+ }, 2000);
+ });
+ });
+
+ widthSlider.addEventListener('input', () => {
+ updateValueDisplays();
+ convertToAscii();
+ });
+
+ brightnessSlider.addEventListener('input', () => {
+ updateValueDisplays();
+ convertToAscii();
+ });
+
+ const uploadLabel = document.querySelector('.upload-label');
+ uploadLabel.addEventListener('dragover', handleDragOver);
+ uploadLabel.addEventListener('dragleave', handleDragLeave);
+ uploadLabel.addEventListener('drop', handleDrop);
+
+ updateValueDisplays();
+ });
style.css
Changed around line 1
+ :root {
+ --primary-color: #6366f1;
+ --bg-color: #f8fafc;
+ --text-color: #1e293b;
+ --border-radius: 12px;
+ }
+
+ * {
+ margin: 0;
+ padding: 0;
+ box-sizing: border-box;
+ }
+
+ body {
+ font-family: system-ui, -apple-system, sans-serif;
+ background: var(--bg-color);
+ color: var(--text-color);
+ line-height: 1.6;
+ min-height: 100vh;
+ display: flex;
+ flex-direction: column;
+ }
+
+ header {
+ text-align: center;
+ padding: 3rem 1rem;
+ background: linear-gradient(135deg, var(--primary-color), #818cf8);
+ color: white;
+ }
+
+ h1 {
+ font-size: 2.5rem;
+ margin-bottom: 0.5rem;
+ animation: fadeIn 0.8s ease-out;
+ }
+
+ .tagline {
+ font-size: 1.2rem;
+ opacity: 0.9;
+ }
+
+ .converter-container {
+ max-width: 900px;
+ margin: 2rem auto;
+ padding: 0 1rem;
+ }
+
+ .upload-section {
+ background: white;
+ border-radius: var(--border-radius);
+ padding: 2rem;
+ box-shadow: 0 4px 6px rgba(0, 0, 0, 0.05);
+ margin-bottom: 2rem;
+ }
+
+ .upload-label {
+ display: flex;
+ flex-direction: column;
+ align-items: center;
+ gap: 1rem;
+ padding: 2rem;
+ border: 2px dashed #cbd5e1;
+ border-radius: var(--border-radius);
+ cursor: pointer;
+ transition: all 0.3s ease;
+ }
+
+ .upload-label:hover {
+ border-color: var(--primary-color);
+ background: #f8fafc;
+ }
+
+ .upload-icon {
+ font-size: 2rem;
+ }
+
+ #imageInput {
+ display: none;
+ }
+
+ .controls {
+ background: white;
+ padding: 1.5rem;
+ border-radius: var(--border-radius);
+ margin-bottom: 2rem;
+ box-shadow: 0 4px 6px rgba(0, 0, 0, 0.05);
+ }
+
+ .control-group {
+ display: flex;
+ align-items: center;
+ gap: 1rem;
+ margin-bottom: 1rem;
+ }
+
+ .control-group:last-child {
+ margin-bottom: 0;
+ }
+
+ input[type="range"] {
+ flex: 1;
+ height: 6px;
+ background: #e2e8f0;
+ border-radius: 3px;
+ -webkit-appearance: none;
+ }
+
+ input[type="range"]::-webkit-slider-thumb {
+ -webkit-appearance: none;
+ width: 18px;
+ height: 18px;
+ background: var(--primary-color);
+ border-radius: 50%;
+ cursor: pointer;
+ transition: transform 0.2s ease;
+ }
+
+ input[type="range"]::-webkit-slider-thumb:hover {
+ transform: scale(1.1);
+ }
+
+ .value-display {
+ min-width: 3rem;
+ text-align: right;
+ }
+
+ .output-container {
+ background: white;
+ padding: 1.5rem;
+ border-radius: var(--border-radius);
+ position: relative;
+ box-shadow: 0 4px 6px rgba(0, 0, 0, 0.05);
+ }
+
+ #asciiOutput {
+ font-family: monospace;
+ white-space: pre;
+ overflow-x: auto;
+ font-size: 12px;
+ line-height: 1.2;
+ margin-bottom: 1rem;
+ }
+
+ #copyButton {
+ position: absolute;
+ top: 1rem;
+ right: 1rem;
+ padding: 0.5rem 1rem;
+ background: var(--primary-color);
+ color: white;
+ border: none;
+ border-radius: var(--border-radius);
+ cursor: pointer;
+ transition: transform 0.2s ease;
+ }
+
+ #copyButton:hover {
+ transform: translateY(-2px);
+ }
+
+ footer {
+ margin-top: auto;
+ text-align: center;
+ padding: 2rem;
+ background: white;
+ }
+
+ @keyframes fadeIn {
+ from {
+ opacity: 0;
+ transform: translateY(-10px);
+ }
+ to {
+ opacity: 1;
+ transform: translateY(0);
+ }
+ }
+
+ @media (max-width: 768px) {
+ h1 {
+ font-size: 2rem;
+ }
+
+ .tagline {
+ font-size: 1rem;
+ }
+
+ .control-group {
+ flex-direction: column;
+ align-items: stretch;
+ gap: 0.5rem;
+ }
+
+ .value-display {
+ text-align: left;
+ }
+ }