| import gradio as gr |
| import qrcode |
| import random |
| import os |
| from datetime import datetime |
| from PIL import Image, ImageDraw |
| from math import cos, sin, radians |
|
|
|
|
| def create_border_decoration(qr_image, decoration_style="Flowers"): |
| |
| qr_image = qr_image.convert('RGB') |
| |
| |
| width, height = qr_image.size |
| |
| |
| padding = 20 |
| new_width = width + (padding * 2) |
| new_height = height + (padding * 2) |
| |
| |
| decorated_image = Image.new('RGB', (new_width, new_height), 'white') |
| |
| |
| decorated_image.paste(qr_image, (padding, padding)) |
| |
| |
| draw = ImageDraw.Draw(decorated_image) |
| |
| |
| deco_size = 8 |
| gap = deco_size * 1.5 |
| |
| |
| border_points = [] |
| |
| |
| for x in range(padding//2, new_width - padding//2, int(gap)): |
| border_points.append((x, padding//2)) |
| |
| |
| for y in range(padding//2, new_height - padding//2, int(gap)): |
| border_points.append((new_width - padding//2, y)) |
| |
| |
| for x in range(new_width - padding//2, padding//2, -int(gap)): |
| border_points.append((x, new_height - padding//2)) |
| |
| |
| for y in range(new_height - padding//2, padding//2, -int(gap)): |
| border_points.append((padding//2, y)) |
|
|
| |
| for x, y in border_points: |
| if decoration_style == "Flowers": |
| for angle in range(0, 360, 45): |
| x1 = x + deco_size * cos(radians(angle)) |
| y1 = y + deco_size * sin(radians(angle)) |
| draw.ellipse([x1-4, y1-4, x1+4, y1+4], fill='pink') |
| draw.ellipse([x-3, y-3, x+3, y+3], fill='yellow') |
| |
| elif decoration_style == "Hearts": |
| heart_size = 6 |
| draw.ellipse([x-heart_size, y-heart_size, x, y], fill='red') |
| draw.ellipse([x, y-heart_size, x+heart_size, y], fill='red') |
| draw.polygon([(x-heart_size, y), (x+heart_size, y), (x, y+heart_size)], fill='red') |
| |
| elif decoration_style == "Waves": |
| wave_size = 10 |
| draw.arc([x-wave_size, y-wave_size//2, x+wave_size, y+wave_size//2], |
| 0, 180, fill='lightblue', width=2) |
| draw.arc([x-wave_size, y, x+wave_size, y+wave_size], |
| 0, 180, fill='blue', width=2) |
| |
| elif decoration_style == "Leaves": |
| leaf_size = 8 |
| draw.ellipse([x-leaf_size, y-leaf_size//2, x+leaf_size, y+leaf_size//2], |
| fill='lightgreen') |
| draw.ellipse([x-leaf_size//2, y-leaf_size, x+leaf_size//2, y+leaf_size], |
| fill='darkgreen') |
| |
| elif decoration_style == "Stars": |
| star_size = 6 |
| points = [] |
| for i in range(5): |
| angle = i * 72 |
| x1 = x + star_size * cos(radians(angle)) |
| y1 = y + star_size * sin(radians(angle)) |
| points.append((x1, y1)) |
| x2 = x + (star_size/2) * cos(radians(angle + 36)) |
| y2 = y + (star_size/2) * sin(radians(angle + 36)) |
| points.append((x2, y2)) |
| draw.polygon(points, fill='gold') |
| |
| elif decoration_style == "Chains": |
| chain_size = 8 |
| draw.ellipse([x-chain_size, y-chain_size//2, x+chain_size, y+chain_size//2], |
| outline='gray', width=2) |
| |
| elif decoration_style == "Bubbles": |
| bubble_sizes = [6, 4, 2] |
| for size in bubble_sizes: |
| draw.ellipse([x-size, y-size, x+size, y+size], |
| outline='skyblue', width=1) |
| |
| elif decoration_style == "Vines": |
| vine_size = 10 |
| draw.arc([x-vine_size, y-vine_size, x+vine_size, y+vine_size], |
| 0, 180, fill='green', width=2) |
| draw.ellipse([x-3, y-3, x+3, y+3], fill='lightgreen') |
| |
| elif decoration_style == "Diamonds": |
| diamond_size = 6 |
| points = [ |
| (x, y-diamond_size), |
| (x+diamond_size, y), |
| (x, y+diamond_size), |
| (x-diamond_size, y) |
| ] |
| draw.polygon(points, outline='purple', width=1) |
| |
| elif decoration_style == "Lace": |
| lace_size = 8 |
| draw.arc([x-lace_size, y-lace_size, x+lace_size, y+lace_size], |
| 0, 180, fill='gray', width=1) |
| draw.arc([x-lace_size//2, y-lace_size//2, x+lace_size//2, y+lace_size//2], |
| 180, 360, fill='gray', width=1) |
| |
| return decorated_image |
|
|
| def rgba_to_rgb(rgba_color): |
| """Convert RGBA color string to RGB hex color""" |
| if rgba_color.startswith('rgba'): |
| values = rgba_color.strip('rgba()').split(',') |
| r = int(float(values[0])) |
| g = int(float(values[1])) |
| b = int(float(values[2])) |
| return f'#{r:02x}{g:02x}{b:02x}' |
| return rgba_color |
|
|
| def create_qr(content, qr_type, fill_color, back_color, box_size, border_size, error_correction, border_decoration="No Decoration"): |
| |
| fill_color = rgba_to_rgb(fill_color) |
| back_color = rgba_to_rgb(back_color) |
| |
| |
| formatted_data = format_data(content, qr_type) |
| |
| |
| error_levels = { |
| "Low (7%)": qrcode.constants.ERROR_CORRECT_L, |
| "Medium (15%)": qrcode.constants.ERROR_CORRECT_M, |
| "Quartile (25%)": qrcode.constants.ERROR_CORRECT_Q, |
| "High (30%)": qrcode.constants.ERROR_CORRECT_H |
| } |
| |
| |
| qr = qrcode.QRCode( |
| version=1, |
| error_correction=error_levels[error_correction], |
| box_size=box_size, |
| border=border_size, |
| ) |
| |
| qr.add_data(formatted_data) |
| qr.make(fit=True) |
| |
| |
| qr_img = qr.make_image(fill_color=fill_color, back_color=back_color) |
| |
| |
| if border_decoration != "No Decoration": |
| qr_img = create_border_decoration(qr_img, border_decoration) |
| |
| |
| timestamp = datetime.now().strftime("%Y%m%d_%H%M%S") |
| random_id = random.randint(1000, 9999) |
| filename = f"qrfile/qr_{timestamp}_{random_id}.png" |
| |
| os.makedirs("qrfile", exist_ok=True) |
| qr_img.save(filename) |
| cleanup_old_files("qrfile/", max_files=100) |
| |
| |
| info_log = f"""โ
QR CODE GENERATED! |
| {'=' * 50} |
| ๐ QR Code Info: |
| โข Type: {qr_type} |
| โข Content: {content[:40]}{'...' if len(content) > 40 else ''} |
| {'=' * 50} |
| ๐จ Style Settings: |
| โข QR Color: {fill_color} |
| โข Background: {back_color} |
| โข Size: {box_size} |
| โข Border: {border_size} |
| โข Decoration: {border_decoration} |
| {'=' * 50} |
| ๐ง Technical: |
| โข Error Correction: {error_correction} |
| โข Formatted Data: {formatted_data[:50]}{'...' if len(formatted_data) > 50 else ''} |
| {'=' * 50} |
| ๐พ Ready to download!""" |
| |
| return filename, formatted_data, info_log |
|
|
| def format_data(content, qr_type): |
| if not content: |
| return "" |
| |
| format_rules = { |
| "URL": lambda x: f"https://{x}" if not x.startswith(('http://', 'https://')) else x, |
| "Email": lambda x: f"mailto:{x}", |
| "Phone": lambda x: f"tel:{x}", |
| "SMS": lambda x: f"sms:{x}", |
| "WhatsApp": lambda x: f"whatsapp://send?text={x}", |
| "Location": lambda x: f"geo:{x}", |
| "Wi-Fi": lambda x: f"WIFI:S:{x};;", |
| "Text": lambda x: x, |
| "vCard": lambda x: f"BEGIN:VCARD\nVERSION:3.0\n{x}\nEND:VCARD" |
| } |
| |
| return format_rules[qr_type](content.strip()) |
|
|
| def cleanup_old_files(directory, max_files): |
| files = [f for f in os.listdir(directory) if f.endswith('.png')] |
| if len(files) > max_files: |
| files.sort(key=lambda x: os.path.getctime(os.path.join(directory, x))) |
| for f in files[:-max_files]: |
| try: |
| os.remove(os.path.join(directory, f)) |
| except: |
| continue |
|
|
| def format_example_text(qr_type): |
| examples = { |
| "URL": "โข Direct URL: https://example.com\nโข Without https: example.com", |
| "Email": "โข Basic: example@email.com\nโข With subject: example@email.com?subject=Hello", |
| "Phone": "โข International: +1234567890\nโข Local: 01012345678", |
| "SMS": "โข Basic: +1234567890\nโข With message: +1234567890?body=Hello", |
| "WhatsApp": "โข Message: Hello World!\nโข With number: +1234567890:Hello", |
| "Location": "โข Coordinates: 37.7749,-122.4194\nโข With zoom: 37.7749,-122.4194,15z", |
| "Wi-Fi": "โข Network name only: MyWiFiNetwork\nโข With password: WIFI:S:MyNetwork;P:password;;", |
| "Text": "โข Simple text: Hello World!\nโข Multiple lines: Line 1\\nLine 2", |
| "vCard": "โข Basic:\nFN:John Doe\nTEL:+1234567890\nEMAIL:john@example.com" |
| } |
| return examples.get(qr_type, "Enter your content here...") |
|
|
|
|
| |
| |
| |
|
|
| css = """ |
| /* ===== ๐จ Google Fonts Import ===== */ |
| @import url('https://fonts.googleapis.com/css2?family=Bangers&family=Comic+Neue:wght@400;700&display=swap'); |
| |
| /* ===== ๐จ Comic Classic ๋ฐฐ๊ฒฝ - ๋นํฐ์ง ํ์ดํผ + ๋ํธ ํจํด ===== */ |
| .gradio-container { |
| background-color: #FEF9C3 !important; |
| background-image: |
| radial-gradient(#1F2937 1px, transparent 1px) !important; |
| background-size: 20px 20px !important; |
| min-height: 100vh !important; |
| font-family: 'Comic Neue', cursive, sans-serif !important; |
| } |
| |
| /* ===== ํ๊น
ํ์ด์ค ์๋จ ์์ ์จ๊น ===== */ |
| .huggingface-space-header, |
| #space-header, |
| .space-header, |
| [class*="space-header"], |
| .svelte-1ed2p3z, |
| .space-header-badge, |
| .header-badge, |
| [data-testid="space-header"], |
| .svelte-kqij2n, |
| .svelte-1ax1toq, |
| .embed-container > div:first-child { |
| display: none !important; |
| visibility: hidden !important; |
| height: 0 !important; |
| width: 0 !important; |
| overflow: hidden !important; |
| opacity: 0 !important; |
| pointer-events: none !important; |
| } |
| |
| /* ===== Footer ์์ ์จ๊น ===== */ |
| footer, |
| .footer, |
| .gradio-container footer, |
| .built-with, |
| [class*="footer"], |
| .gradio-footer, |
| .main-footer, |
| div[class*="footer"], |
| .show-api, |
| .built-with-gradio, |
| a[href*="gradio.app"], |
| a[href*="huggingface.co/spaces"] { |
| display: none !important; |
| visibility: hidden !important; |
| height: 0 !important; |
| padding: 0 !important; |
| margin: 0 !important; |
| } |
| |
| /* ===== ๋ฉ์ธ ์ปจํ
์ด๋ ===== */ |
| #col-container { |
| max-width: 1200px; |
| margin: 0 auto; |
| } |
| |
| /* ===== ๐จ ํค๋ ํ์ดํ - ์ฝ๋ฏน ์คํ์ผ ===== */ |
| .header-text h1 { |
| font-family: 'Bangers', cursive !important; |
| color: #1F2937 !important; |
| font-size: 3.5rem !important; |
| font-weight: 400 !important; |
| text-align: center !important; |
| margin-bottom: 0.5rem !important; |
| text-shadow: |
| 4px 4px 0px #FACC15, |
| 6px 6px 0px #1F2937 !important; |
| letter-spacing: 3px !important; |
| -webkit-text-stroke: 2px #1F2937 !important; |
| } |
| |
| /* ===== ๐จ ์๋ธํ์ดํ ===== */ |
| .subtitle { |
| text-align: center !important; |
| font-family: 'Comic Neue', cursive !important; |
| font-size: 1.2rem !important; |
| color: #1F2937 !important; |
| margin-bottom: 1.5rem !important; |
| font-weight: 700 !important; |
| } |
| |
| /* ===== ๐จ ์นด๋/ํจ๋ - ๋งํ ํ๋ ์ ์คํ์ผ ===== */ |
| .gr-panel, |
| .gr-box, |
| .gr-form, |
| .block, |
| .gr-group { |
| background: #FFFFFF !important; |
| border: 3px solid #1F2937 !important; |
| border-radius: 8px !important; |
| box-shadow: 6px 6px 0px #1F2937 !important; |
| transition: all 0.2s ease !important; |
| } |
| |
| .gr-panel:hover, |
| .block:hover { |
| transform: translate(-2px, -2px) !important; |
| box-shadow: 8px 8px 0px #1F2937 !important; |
| } |
| |
| /* ===== ๐จ ์
๋ ฅ ํ๋ (Textbox) ===== */ |
| textarea, |
| input[type="text"], |
| input[type="number"] { |
| background: #FFFFFF !important; |
| border: 3px solid #1F2937 !important; |
| border-radius: 8px !important; |
| color: #1F2937 !important; |
| font-family: 'Comic Neue', cursive !important; |
| font-size: 1rem !important; |
| font-weight: 700 !important; |
| transition: all 0.2s ease !important; |
| } |
| |
| textarea:focus, |
| input[type="text"]:focus, |
| input[type="number"]:focus { |
| border-color: #3B82F6 !important; |
| box-shadow: 4px 4px 0px #3B82F6 !important; |
| outline: none !important; |
| } |
| |
| textarea::placeholder { |
| color: #9CA3AF !important; |
| font-weight: 400 !important; |
| } |
| |
| /* ===== ๐จ ๋๋กญ๋ค์ด ์คํ์ผ ===== */ |
| .gr-dropdown { |
| background: #FFFFFF !important; |
| border: 3px solid #1F2937 !important; |
| border-radius: 8px !important; |
| box-shadow: 3px 3px 0px #1F2937 !important; |
| } |
| |
| .gr-dropdown > div { |
| background: #FFFFFF !important; |
| border: none !important; |
| } |
| |
| .gr-dropdown input { |
| color: #1F2937 !important; |
| font-family: 'Comic Neue', cursive !important; |
| font-weight: 700 !important; |
| } |
| |
| .gr-dropdown ul { |
| background: #FFFFFF !important; |
| border: 3px solid #1F2937 !important; |
| border-radius: 8px !important; |
| box-shadow: 4px 4px 0px #1F2937 !important; |
| } |
| |
| .gr-dropdown ul li { |
| color: #1F2937 !important; |
| font-family: 'Comic Neue', cursive !important; |
| font-weight: 700 !important; |
| padding: 8px 12px !important; |
| } |
| |
| .gr-dropdown ul li:hover { |
| background: #FACC15 !important; |
| color: #1F2937 !important; |
| } |
| |
| .gr-dropdown ul li.selected { |
| background: #3B82F6 !important; |
| color: #FFFFFF !important; |
| } |
| |
| /* ===== ๐จ Primary ๋ฒํผ - ์ฝ๋ฏน ๋ธ๋ฃจ ===== */ |
| .gr-button-primary, |
| button.primary, |
| .gr-button.primary, |
| .generate-btn { |
| background: #3B82F6 !important; |
| border: 3px solid #1F2937 !important; |
| border-radius: 8px !important; |
| color: #FFFFFF !important; |
| font-family: 'Bangers', cursive !important; |
| font-weight: 400 !important; |
| font-size: 1.3rem !important; |
| letter-spacing: 2px !important; |
| padding: 14px 28px !important; |
| box-shadow: 5px 5px 0px #1F2937 !important; |
| transition: all 0.1s ease !important; |
| text-shadow: 1px 1px 0px #1F2937 !important; |
| } |
| |
| .gr-button-primary:hover, |
| button.primary:hover, |
| .gr-button.primary:hover, |
| .generate-btn:hover { |
| background: #2563EB !important; |
| transform: translate(-2px, -2px) !important; |
| box-shadow: 7px 7px 0px #1F2937 !important; |
| } |
| |
| .gr-button-primary:active, |
| button.primary:active, |
| .gr-button.primary:active, |
| .generate-btn:active { |
| transform: translate(3px, 3px) !important; |
| box-shadow: 2px 2px 0px #1F2937 !important; |
| } |
| |
| /* ===== ๐จ Secondary ๋ฒํผ - ์ฝ๋ฏน ๋ ๋ ===== */ |
| .gr-button-secondary, |
| button.secondary { |
| background: #EF4444 !important; |
| border: 3px solid #1F2937 !important; |
| border-radius: 8px !important; |
| color: #FFFFFF !important; |
| font-family: 'Bangers', cursive !important; |
| font-weight: 400 !important; |
| font-size: 1.1rem !important; |
| letter-spacing: 1px !important; |
| box-shadow: 4px 4px 0px #1F2937 !important; |
| transition: all 0.1s ease !important; |
| text-shadow: 1px 1px 0px #1F2937 !important; |
| } |
| |
| .gr-button-secondary:hover, |
| button.secondary:hover { |
| background: #DC2626 !important; |
| transform: translate(-2px, -2px) !important; |
| box-shadow: 6px 6px 0px #1F2937 !important; |
| } |
| |
| /* ===== ๐จ ๋ก๊ทธ ์ถ๋ ฅ ์์ญ ===== */ |
| .info-log textarea { |
| background: #1F2937 !important; |
| color: #10B981 !important; |
| font-family: 'Courier New', monospace !important; |
| font-size: 0.9rem !important; |
| font-weight: 400 !important; |
| border: 3px solid #10B981 !important; |
| border-radius: 8px !important; |
| box-shadow: 4px 4px 0px #10B981 !important; |
| } |
| |
| /* ===== ๐จ ์ปฌ๋ฌ ํผ์ปค ์คํ์ผ ===== */ |
| .gr-colorpicker, |
| input[type="color"] { |
| border: 3px solid #1F2937 !important; |
| border-radius: 8px !important; |
| box-shadow: 3px 3px 0px #1F2937 !important; |
| width: 60px !important; |
| height: 40px !important; |
| cursor: pointer !important; |
| } |
| |
| .gr-colorpicker:hover, |
| input[type="color"]:hover { |
| transform: translate(-2px, -2px) !important; |
| box-shadow: 5px 5px 0px #1F2937 !important; |
| } |
| |
| /* ===== ๐จ ์ฌ๋ผ์ด๋ ์คํ์ผ ===== */ |
| input[type="range"] { |
| accent-color: #3B82F6 !important; |
| height: 8px !important; |
| } |
| |
| .gr-slider { |
| background: #FFFFFF !important; |
| } |
| |
| /* ===== ๐จ ์์ ํ
์คํธ ๋ฐ์ค ===== */ |
| .example-box textarea { |
| background: #FEF3C7 !important; |
| border: 2px dashed #F59E0B !important; |
| border-radius: 8px !important; |
| color: #92400E !important; |
| font-family: 'Comic Neue', cursive !important; |
| font-size: 0.9rem !important; |
| } |
| |
| /* ===== ๐จ QR ์ฝ๋ ์ถ๋ ฅ ์์ญ ===== */ |
| .qr-output, |
| .gr-image { |
| border: 4px solid #1F2937 !important; |
| border-radius: 8px !important; |
| box-shadow: 8px 8px 0px #1F2937 !important; |
| overflow: hidden !important; |
| background: #FFFFFF !important; |
| } |
| |
| /* ===== ๐จ ์์ฝ๋์ธ - ๋งํ์ ์คํ์ผ ===== */ |
| .gr-accordion { |
| background: #FACC15 !important; |
| border: 3px solid #1F2937 !important; |
| border-radius: 8px !important; |
| box-shadow: 4px 4px 0px #1F2937 !important; |
| } |
| |
| .gr-accordion-header { |
| color: #1F2937 !important; |
| font-family: 'Comic Neue', cursive !important; |
| font-weight: 700 !important; |
| font-size: 1.1rem !important; |
| } |
| |
| /* ===== ๐จ ๋ผ๋ฒจ ์คํ์ผ ===== */ |
| label, |
| .gr-input-label, |
| .gr-block-label { |
| color: #1F2937 !important; |
| font-family: 'Comic Neue', cursive !important; |
| font-weight: 700 !important; |
| font-size: 1rem !important; |
| } |
| |
| span.gr-label { |
| color: #1F2937 !important; |
| } |
| |
| /* ===== ๐จ ์ ๋ณด ํ
์คํธ ===== */ |
| .gr-info, |
| .info { |
| color: #6B7280 !important; |
| font-family: 'Comic Neue', cursive !important; |
| font-size: 0.9rem !important; |
| } |
| |
| /* ===== ๐จ Instructions ์น์
===== */ |
| .instructions-box { |
| background: linear-gradient(135deg, #EFF6FF 0%, #DBEAFE 100%) !important; |
| border: 3px solid #3B82F6 !important; |
| border-radius: 12px !important; |
| padding: 1.5rem !important; |
| box-shadow: 6px 6px 0px #1F2937 !important; |
| margin-top: 1.5rem !important; |
| } |
| |
| .instructions-box h3 { |
| font-family: 'Bangers', cursive !important; |
| color: #1F2937 !important; |
| font-size: 1.3rem !important; |
| margin-bottom: 0.5rem !important; |
| } |
| |
| .instructions-box ul, |
| .instructions-box ol { |
| font-family: 'Comic Neue', cursive !important; |
| color: #1F2937 !important; |
| font-weight: 700 !important; |
| } |
| |
| /* ===== ๐จ Tips ์น์
===== */ |
| .tips-box { |
| background: linear-gradient(135deg, #FEF3C7 0%, #FDE68A 100%) !important; |
| border: 3px solid #F59E0B !important; |
| border-radius: 12px !important; |
| padding: 1.5rem !important; |
| box-shadow: 6px 6px 0px #1F2937 !important; |
| margin-top: 1rem !important; |
| } |
| |
| /* ===== ๐จ ์คํฌ๋กค๋ฐ - ์ฝ๋ฏน ์คํ์ผ ===== */ |
| ::-webkit-scrollbar { |
| width: 12px; |
| height: 12px; |
| } |
| |
| ::-webkit-scrollbar-track { |
| background: #FEF9C3; |
| border: 2px solid #1F2937; |
| } |
| |
| ::-webkit-scrollbar-thumb { |
| background: #3B82F6; |
| border: 2px solid #1F2937; |
| border-radius: 0px; |
| } |
| |
| ::-webkit-scrollbar-thumb:hover { |
| background: #EF4444; |
| } |
| |
| /* ===== ๐จ ์ ํ ํ์ด๋ผ์ดํธ ===== */ |
| ::selection { |
| background: #FACC15; |
| color: #1F2937; |
| } |
| |
| /* ===== ๐จ ๋งํฌ ์คํ์ผ ===== */ |
| a { |
| color: #3B82F6 !important; |
| text-decoration: none !important; |
| font-weight: 700 !important; |
| } |
| |
| a:hover { |
| color: #EF4444 !important; |
| } |
| |
| /* ===== ๐จ Row/Column ๊ฐ๊ฒฉ ===== */ |
| .gr-row { |
| gap: 1.5rem !important; |
| } |
| |
| .gr-column { |
| gap: 1rem !important; |
| } |
| |
| /* ===== ๐จ ์ปฌ๋ฌ ํผ์ปค Row ===== */ |
| .color-picker-row { |
| display: flex !important; |
| gap: 1rem !important; |
| align-items: center !important; |
| } |
| |
| /* ===== ๋ฐ์ํ ์กฐ์ ===== */ |
| @media (max-width: 768px) { |
| .header-text h1 { |
| font-size: 2.2rem !important; |
| text-shadow: |
| 3px 3px 0px #FACC15, |
| 4px 4px 0px #1F2937 !important; |
| } |
| |
| .gr-button-primary, |
| button.primary { |
| padding: 12px 20px !important; |
| font-size: 1.1rem !important; |
| } |
| |
| .gr-panel, |
| .block { |
| box-shadow: 4px 4px 0px #1F2937 !important; |
| } |
| } |
| |
| /* ===== ๐จ ๋คํฌ๋ชจ๋ ๋นํ์ฑํ ===== */ |
| @media (prefers-color-scheme: dark) { |
| .gradio-container { |
| background-color: #FEF9C3 !important; |
| } |
| } |
| """ |
|
|
|
|
| def create_interface(): |
| with gr.Blocks(fill_height=True, css=css, title="QR Canvas+") as demo: |
| |
| |
| gr.HTML(""" |
| <div style="text-align: center; margin: 20px 0 10px 0;"> |
| <a href="https://www.humangen.ai" target="_blank" style="text-decoration: none;"> |
| <img src="https://img.shields.io/static/v1?label=๐ HOME&message=HUMANGEN.AI&color=0000ff&labelColor=ffcc00&style=for-the-badge" alt="HOME"> |
| </a> |
| </div> |
| """) |
| |
| |
| gr.Markdown( |
| """ |
| # ๐ฏ QR CANVAS+ GENERATOR ๐ฑ |
| """, |
| elem_classes="header-text" |
| ) |
| |
| gr.Markdown( |
| """ |
| <p class="subtitle">๐จ Create customized QR codes with professional styling & decorations! โจ</p> |
| """, |
| ) |
| |
| with gr.Row(equal_height=False): |
| |
| with gr.Column(scale=2, min_width=450): |
| qr_type = gr.Dropdown( |
| choices=["URL", "Email", "Phone", "SMS", "WhatsApp", "Location", "Wi-Fi", "Text", "vCard"], |
| value="URL", |
| label="๐ QR Code Type" |
| ) |
| |
| content = gr.Textbox( |
| label="โ๏ธ Content", |
| placeholder="Enter your content here...", |
| lines=3 |
| ) |
| |
| example_format = gr.Textbox( |
| value=format_example_text("URL"), |
| label="๐ Format Examples", |
| interactive=False, |
| lines=4, |
| elem_classes="example-box" |
| ) |
| |
| with gr.Row(elem_classes="color-picker-row"): |
| fill_color = gr.ColorPicker( |
| label="๐จ QR Code Color", |
| value="#000000" |
| ) |
| back_color = gr.ColorPicker( |
| label="๐ผ๏ธ Background Color", |
| value="#FFFFFF" |
| ) |
| |
| with gr.Row(): |
| box_size = gr.Slider( |
| minimum=1, |
| maximum=30, |
| value=15, |
| step=1, |
| label="๐ QR Code Size" |
| ) |
| border_size = gr.Slider( |
| minimum=0, |
| maximum=5, |
| value=2, |
| step=1, |
| label="๐ฒ Border Size" |
| ) |
|
|
| error_correction = gr.Dropdown( |
| choices=[ |
| "Low (7%)", |
| "Medium (15%)", |
| "Quartile (25%)", |
| "High (30%)" |
| ], |
| value="Medium (15%)", |
| label="๐ง Error Correction Level" |
| ) |
|
|
| border_decoration = gr.Dropdown( |
| choices=[ |
| "No Decoration", |
| "Flowers", |
| "Hearts", |
| "Waves", |
| "Leaves", |
| "Stars", |
| "Chains", |
| "Bubbles", |
| "Vines", |
| "Diamonds", |
| "Lace" |
| ], |
| value="No Decoration", |
| label="๐ธ Border Decoration Style" |
| ) |
|
|
| generate_btn = gr.Button( |
| "๐ฏ GENERATE QR CODE! ๐ฑ", |
| variant="primary", |
| size="lg", |
| elem_classes="generate-btn" |
| ) |
| |
| with gr.Accordion("๐ Generation Log", open=True): |
| info_log = gr.Textbox( |
| label="", |
| placeholder="Enter content and click generate to see info...", |
| lines=14, |
| max_lines=20, |
| interactive=False, |
| elem_classes="info-log" |
| ) |
| |
| |
| with gr.Column(scale=1, min_width=350): |
| output_image = gr.Image( |
| label="๐ผ๏ธ Generated QR Code", |
| type="filepath", |
| height=400, |
| elem_classes="qr-output" |
| ) |
| |
| output_data = gr.Textbox( |
| label="๐ Formatted Data", |
| interactive=False, |
| lines=2 |
| ) |
| |
| gr.Markdown( |
| """ |
| <p style="text-align: center; margin-top: 15px; font-weight: 700; color: #1F2937;"> |
| ๐ก Right-click on the QR code to save! |
| </p> |
| """ |
| ) |
|
|
| |
| gr.Markdown( |
| """ |
| <div class="instructions-box"> |
| <h3>๐ INSTRUCTIONS</h3> |
| <ol> |
| <li>Select the QR code type from the dropdown menu</li> |
| <li>Enter your content following the format examples shown</li> |
| <li>Customize the appearance using the color pickers and sliders</li> |
| <li>Choose a border decoration style (optional)</li> |
| <li>Click 'GENERATE QR CODE' to create your custom QR code</li> |
| </ol> |
| </div> |
| |
| <div class="tips-box"> |
| <h3>๐ก TIPS</h3> |
| <ul> |
| <li>Use higher error correction levels for better scan reliability</li> |
| <li>Ensure sufficient contrast between QR code and background colors</li> |
| <li>Keep the content concise for better readability</li> |
| <li>Follow the format examples for best results</li> |
| </ul> |
| </div> |
| """ |
| ) |
| |
| |
| def update_example(qr_type): |
| return format_example_text(qr_type) |
| |
| qr_type.change( |
| fn=update_example, |
| inputs=[qr_type], |
| outputs=example_format |
| ) |
| |
| generate_btn.click( |
| fn=create_qr, |
| inputs=[ |
| content, |
| qr_type, |
| fill_color, |
| back_color, |
| box_size, |
| border_size, |
| error_correction, |
| border_decoration |
| ], |
| outputs=[output_image, output_data, info_log] |
| ) |
| |
| return demo |
|
|
|
|
| if __name__ == "__main__": |
| try: |
| os.makedirs("qrfile", exist_ok=True) |
| demo = create_interface() |
| demo.launch( |
| server_name="0.0.0.0", |
| server_port=7860, |
| share=True, |
| debug=True |
| ) |
| except Exception as e: |
| print(f"Error starting the application: {e}") |