1001 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			1001 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import io
 | |
| import tempfile
 | |
| import subprocess
 | |
| import requests
 | |
| import base64
 | |
| from flask import Flask, render_template_string, request, send_from_directory, session
 | |
| from PIL import Image, ImageDraw, ImageFont
 | |
| import os
 | |
| import re
 | |
| import argparse
 | |
| 
 | |
| app = Flask(__name__)
 | |
| app.secret_key = 'TpfxYMxUJxJMtCCTraBSg1rbd6NLz38JTmIfpBLsotcI47EqXU'
 | |
| 
 | |
| FONT_PATH = "./fonts/ubuntu/Ubuntu-Regular.ttf"
 | |
| FONT_PATH_BOLD = "./fonts/ubuntu/Ubuntu-Bold.ttf"
 | |
| FONT_PATH_OBLIQUE = "./fonts/ubuntu/Ubuntu-Italic.ttf"
 | |
| FONT_PATH_BOLDITALIC = "./fonts/ubuntu/Ubuntu-BoldItalic.ttf"
 | |
| FONT_SIZE = 24
 | |
| HEADER_SIZE_1 = 56
 | |
| HEADER_SIZE_2 = 34
 | |
| HEADER_SIZE_3 = 28
 | |
| BANNER_FONT_SIZE = 300
 | |
| IMAGE_WIDTH = 384
 | |
| BULLET_CHAR = "• "
 | |
| 
 | |
| DITHERING_MODES = {
 | |
|   "Sin dithering": "none",
 | |
|   "Floyd-Steinberg": "floyd",
 | |
|   "Bayer 2x2": "bayer2x2",
 | |
|   "Bayer 4x4": "bayer4x4",
 | |
|   "Bayer 8x8": "bayer8x8",
 | |
|   "Bayer 16x16": "bayer16x16",
 | |
|   "Atkinson": "atkinson",
 | |
|   "Jarvis-Judice-Ninke": "jjn"
 | |
| }
 | |
| 
 | |
| HTML_FORM = '''
 | |
| <!doctype html>
 | |
| <html>
 | |
| <head>
 | |
|   <title>CatNote</title>
 | |
|   <link rel="manifest" href="/static/manifest.json">
 | |
|   <style>
 | |
|     body {
 | |
|       font-family: sans-serif;
 | |
|       background: #181c1f;
 | |
|       color: #ddd;
 | |
|       min-height: 100vh;
 | |
|       margin: 0;
 | |
|     }
 | |
|     .centered-flex {
 | |
|       display: flex;
 | |
|       justify-content: center;
 | |
|       align-items: flex-start;
 | |
|       height: 100vh;
 | |
|       gap: 2.6em;
 | |
|       padding-top: 2em;
 | |
|     }
 | |
|     .form-card {
 | |
|       background: #22282c;
 | |
|       padding: 2em 2em 1em 2em;
 | |
|       border-radius: 1.5em;
 | |
|       box-shadow: 0 0 12px 0 #000a;
 | |
|       min-width: 410px;
 | |
|       max-width: 450px;
 | |
|       margin-right: 0;
 | |
|       box-sizing: border-box;
 | |
|       display: flex;
 | |
|       flex-direction: column;
 | |
|     }
 | |
|     .markdown-ref {
 | |
|       background: #22282c;
 | |
|       border-radius: 1.2em;
 | |
|       box-shadow: 0 0 12px 0 #000a;
 | |
|       padding: 2em 2em 1.5em 2em;
 | |
|       color: #b6c8e0;
 | |
|       font-size: 1.08rem;
 | |
|       font-family: inherit;
 | |
|       box-sizing: border-box;
 | |
|       width: 100%;
 | |
|       max-width: 410px;
 | |
|     }
 | |
|     .markdown-ref h4 {
 | |
|       margin: 0 0 0.4em 0;
 | |
|       color: #88e0ff;
 | |
|       font-size: 1.12rem;
 | |
|       font-weight: 700;
 | |
|       letter-spacing: 0.01em;
 | |
|     }
 | |
|     .markdown-ref ul {
 | |
|       margin: 0.6em 0 0.3em 0.1em;
 | |
|       padding-left: 1.1em;
 | |
|     }
 | |
|     .markdown-ref li {
 | |
|       margin-bottom: 0.36em;
 | |
|     }
 | |
|     .markdown-ref code {
 | |
|       background: #181c1f;
 | |
|       border-radius: 0.5em;
 | |
|       padding: 0.12em 0.46em;
 | |
|       color: #aef6cb;
 | |
|       font-size: 0.98rem;
 | |
|       margin-left: 0.18em;
 | |
|     }
 | |
|     textarea {
 | |
|       width: 100%;
 | |
|       height: 14em;
 | |
|       padding: 0.75em;
 | |
|       border-radius: 0.7em;
 | |
|       border: 1.5px solid #444;
 | |
|       background: #181c1f;
 | |
|       color: #eee;
 | |
|       font-size: 1.06rem;
 | |
|       margin-bottom: 1em;
 | |
|       resize: vertical;
 | |
|       font-family: 'DejaVu Sans Mono', 'Fira Mono', 'monospace';
 | |
|       box-sizing: border-box;
 | |
|       margin-right: 0;
 | |
|       min-width: 0;
 | |
|     }
 | |
|     .buttons {
 | |
|       display: flex;
 | |
|       gap: 1em;
 | |
|       justify-content: center;
 | |
|     }
 | |
|     button[type=submit] {
 | |
|       background: linear-gradient(90deg, #8ee3c1, #35a7ff);
 | |
|       color: #222;
 | |
|       font-weight: bold;
 | |
|       font-size: 1.12rem;
 | |
|       border: none;
 | |
|       border-radius: 0.7em;
 | |
|       padding: 0.5em 1.6em;
 | |
|       margin-top: 0.1em;
 | |
|       box-shadow: 0 2px 12px #1116;
 | |
|       cursor: pointer;
 | |
|       transition: filter 0.2s, box-shadow 0.2s;
 | |
|       filter: brightness(1);
 | |
|     }
 | |
|     button[type=submit]:hover {
 | |
|       filter: brightness(1.12);
 | |
|       box-shadow: 0 4px 18px #2229;
 | |
|     }
 | |
|     button[type=submit][name=print] {
 | |
|       background: linear-gradient(90deg, #ffeb3b, #ff9100);
 | |
|       color: #181c1f;
 | |
|     }
 | |
|     .preview-card {
 | |
|       display: flex;
 | |
|       flex-direction: column;
 | |
|       background: #23282d;
 | |
|       padding: 2em 1.4em 1em 1.4em;
 | |
|       border-radius: 1.5em;
 | |
|       box-shadow: 0 0 12px 0 #0007;
 | |
|       min-width: 400px;
 | |
|       text-align: center;
 | |
|       box-sizing: border-box;
 | |
|       margin-left: 0;
 | |
|       align-items: center;
 | |
|     }
 | |
|     img {
 | |
|       margin-top: 1em;
 | |
|       max-width: 100%;
 | |
|       background: #fff;
 | |
|       box-shadow: 0 0 8px 1px #111a;
 | |
|     }
 | |
|     .status-msg {
 | |
|       margin: 0.6em 0 0.1em 0;
 | |
|       font-size: 1.12rem;
 | |
|       border-radius: 0.7em;
 | |
|       background: #294b3a;
 | |
|       color: #9fffc6;
 | |
|       padding: 0.55em 1.2em;
 | |
|       font-weight: 600;
 | |
|       box-shadow: 0 0 6px #183a24a0;
 | |
|       display: inline-block;
 | |
|       border-left: 4px solid #4ffab0;
 | |
|     }
 | |
|     .status-err {
 | |
|       background: #48202c;
 | |
|       color: #ffdbe4;
 | |
|       border-left: 4px solid #ff6384;
 | |
|       box-shadow: 0 0 8px #22000690;
 | |
|     }
 | |
|     label {
 | |
|       font-size: 1em;
 | |
|       padding: 0 1em 0 1em;
 | |
|     }
 | |
|     input[type=file] {
 | |
|       margin: 0.5em 1em 1.5em 1em;
 | |
|     }
 | |
|     input[type=file]::file-selector-button {
 | |
|       background: linear-gradient(90deg, #8ee3c1, #35a7ff);
 | |
|       color: #222;
 | |
|       font-weight: bold;
 | |
|       border: none;
 | |
|       padding: 0.5em 1.2em;
 | |
|       border-radius: 0.7em;
 | |
|       box-shadow: 0 2px 12px #1116;
 | |
|       cursor: pointer;
 | |
|       transition: filter 0.2s, box-shadow 0.2s;
 | |
|       margin-right: 0.5em;
 | |
|     }
 | |
|     input[type=file]::file-selector-button:hover {
 | |
|       filter: brightness(1.12);
 | |
|       box-shadow: 0 4px 18px #2229;
 | |
|     }
 | |
|     .print-frame {
 | |
|       position: relative;
 | |
|       padding: 10px 0 10px 0;
 | |
|       margin-top: 1em;
 | |
|       overflow: hidden;
 | |
|       max-width: 424px;
 | |
|     }
 | |
|     .print-frame::before,
 | |
|     .print-frame::after {
 | |
|       content: "";
 | |
|       position: absolute;
 | |
|       left: 0;
 | |
|       width: 100%;
 | |
|       height: 10px;
 | |
|       background: url('data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiPz4KPHN2ZyB3aWR0aD0iMTYiIGhlaWdodD0iMTYiIHZlcnNpb249IjEuMSIgdmlld0JveD0iMCAwIDQuMjMzMyA0LjIzMzMiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CjxwYXRoIHRyYW5zZm9ybT0ibWF0cml4KC4wMzYxMzUgMCAwIC4wNDE3MjUgMS4xNTY4IDEuMTI4OSkiIGQ9Im04NS4xNDEgNzQuNDAzLTExNy4xNS0xZS02IDU4LjU3Ny0xMDEuNDZ6IiBmaWxsPSIjZmZmIiBzdHJva2UtbGluZWNhcD0ic3F1YXJlIiBzdHJva2Utb3BhY2l0eT0iLjk3NjQ3IiBzdHJva2Utd2lkdGg9Ii4yNjQ1OCIvPgo8L3N2Zz4K') repeat-x left;
 | |
|       background-size: 10px 384px;
 | |
|       z-index: 1;
 | |
|     }
 | |
|     .print-frame::before {
 | |
|       top: 1px;
 | |
|     }
 | |
|     .print-frame::after {
 | |
|       bottom: 1px;
 | |
|       transform: rotate(180deg);
 | |
|     }
 | |
|     .print-frame img {
 | |
|       background: #fff;
 | |
|       padding: 20px;
 | |
|       display: block;
 | |
|       margin: 0 auto;
 | |
|       box-shadow: none;
 | |
|       position: relative;
 | |
|       width: 100%;
 | |
|       max-width: 384px;
 | |
|       z-index: 2;
 | |
|     }
 | |
|     textarea {
 | |
|       font-size: 1.16em !important;
 | |
|       padding: 1em;
 | |
|       min-height: 12em;
 | |
|     }
 | |
|     .options {
 | |
|       display: flex;
 | |
|       flex-direction: column;
 | |
|       align-items: left;
 | |
|       justify-content: space-between;
 | |
|       margin-bottom: 1.2em;
 | |
|     }
 | |
|     select {
 | |
|       background: #181c1f;
 | |
|       text-align: center;
 | |
|       text-align-last: center;
 | |
|       border: 1.5px solid #444;
 | |
|       color: #eee;
 | |
|       border-radius: 0.7em;
 | |
|       margin: 1em;
 | |
|       padding: 0.5em 1em;
 | |
|     }
 | |
|     select option {
 | |
|       text-align: left;
 | |
|     }
 | |
|     input[type=number] {
 | |
|       background: #181c1f;
 | |
|       text-align: center;
 | |
|       border: 1.5px solid #444;
 | |
|       color: #eee;
 | |
|       border-radius: 0.7em;
 | |
|       margin: 1em;
 | |
|       padding: 0.5em 1em;
 | |
|     }
 | |
|     input[type=number]::-webkit-inner-spin-button,
 | |
|     input[type=number]::-webkit-outer-spin-button {
 | |
|       -webkit-appearance: none;
 | |
|       margin: 0;
 | |
|     }
 | |
|     .options-dithering,
 | |
|     .options-intensity,
 | |
|     .options-rotation,
 | |
|     .options-printmode {
 | |
|       display: flex;
 | |
|       flex-direction: column;
 | |
|     }
 | |
|     #loading-box {
 | |
|       display: none;
 | |
|       flex-direction: column;
 | |
|       text-align: center;
 | |
|       box-sizing: border-box;
 | |
|       width: 100%;
 | |
|       padding: 1em;
 | |
|     }
 | |
|     #printing-text {
 | |
|       color: #eee;
 | |
|       font-size: 1em;
 | |
|       font-weight: bold;
 | |
|       margin-bottom: 1em;
 | |
|     }
 | |
|     .progress-bar {
 | |
|       position: relative;
 | |
|       width: 100%;
 | |
|       height: 18px;
 | |
|       background: #181c1f;
 | |
|       border: 1.5px solid #444;
 | |
|       border-radius: 12px;
 | |
|       overflow: hidden;
 | |
|       box-shadow: 0 2px 12px #1116;
 | |
|     }
 | |
|     .progress-bar-inner {
 | |
|       width: 40%;
 | |
|       height: 100%;
 | |
|       background: linear-gradient(90deg, #8ee3c1, #35a7ff);
 | |
|       border-radius: 12px;
 | |
|       position: absolute;
 | |
|       left: -40%;
 | |
|       animation: indeterminate-bar 1.2s infinite
 | |
|     }
 | |
|     @keyframes indeterminate-bar {
 | |
|       0% { left: 0%; }
 | |
|       50% { left: calc(100% - 40%); }
 | |
|       100% { left: 0%; }
 | |
|     }
 | |
|     /* this next bit is a mess but it works */
 | |
|     .options-bannermode {
 | |
|       display: flex;
 | |
|       align-items: left;
 | |
|       justify-content: space-between;
 | |
|       margin-left: 1em;
 | |
|       margin-right: 1em;
 | |
|     }
 | |
|     .options-bannermode label {
 | |
|       font-size: 1em;
 | |
|       color: #eee;
 | |
|       margin-bottom: 1em;
 | |
|       display: block;
 | |
|     }
 | |
|     .options-bannermode-inner {
 | |
|       display: block;
 | |
|       align-items: center;
 | |
|       position: relative;
 | |
|       padding-left: 35px;
 | |
|       margin-bottom: 12px;
 | |
|       cursor: pointer;
 | |
|       font-size: 1em;
 | |
|       -webkit-user-select: none;
 | |
|       -moz-user-select: none;
 | |
|       -ms-user-select: none;
 | |
|       user-select: none;
 | |
|     }
 | |
| 
 | |
|     /* Hide the browser's default checkbox */
 | |
|     .options-bannermode-inner input {
 | |
|       position: absolute;
 | |
|       opacity: 0;
 | |
|       cursor: pointer;
 | |
|       height: 0;
 | |
|       width: 0;
 | |
|     }
 | |
| 
 | |
|     /* Create a custom checkbox */
 | |
|     .checkmark {
 | |
|       position: absolute;
 | |
|       top: -25%;
 | |
|       left: 0;
 | |
|       height: 25px;
 | |
|       width: 25px;
 | |
|       background-color: #181c1f;
 | |
|       border-radius: 0.35em;
 | |
|       border: 1.5px solid #444;
 | |
|     }
 | |
| 
 | |
|     /* On mouse-over, add a grey background color */
 | |
|     .options-bannermode-inner:hover input ~ .checkmark {
 | |
|       filter: brightness(1.12);
 | |
|     }
 | |
| 
 | |
|     /* When the checkbox is checked, add a blue background */
 | |
|     .options-bannermode-inner input:checked ~ .checkmark {
 | |
|       background: linear-gradient(135deg, #8ee3c1, #35a7ff);
 | |
|     }
 | |
| 
 | |
|     /* Create the checkmark/indicator (hidden when not checked) */
 | |
|     .checkmark:after {
 | |
|       content: "";
 | |
|       position: absolute;
 | |
|       display: none;
 | |
|     }
 | |
| 
 | |
|     /* Show the checkmark when checked */
 | |
|     .options-bannermode-inner input:checked ~ .checkmark:after {
 | |
|       display: block;
 | |
|     }
 | |
| 
 | |
|     /* Style the checkmark/indicator */
 | |
|     .options-bannermode-inner .checkmark:after {
 | |
|       left: 9px;
 | |
|       top: 5px;
 | |
|       width: 5px;
 | |
|       height: 10px;
 | |
|       border: solid white;
 | |
|       border-width: 0 3px 3px 0;
 | |
|       -webkit-transform: rotate(45deg);
 | |
|       -ms-transform: rotate(45deg);
 | |
|       transform: rotate(45deg);
 | |
|     }
 | |
|     @media (max-width: 1000px) {
 | |
|       body {
 | |
|         font-size: 1.28em !important;
 | |
|       }
 | |
|       h2, h3, h4 {
 | |
|         font-size: 2em !important;
 | |
|       }
 | |
|       .centered-flex {
 | |
|         flex-direction: column;
 | |
|         align-items: center;
 | |
|         height: auto;
 | |
|         gap: 1.6em;
 | |
|       }
 | |
|       .form-card,
 | |
|       .preview-card,
 | |
|       .markdown-ref {
 | |
|         font-size: 1.18em !important;
 | |
|         min-width: auto;
 | |
|         max-width: 90vw;
 | |
|         width: 100%;
 | |
|         margin: 0.8em 0;
 | |
|       }
 | |
|       textarea {
 | |
|         font-size: 1.16em !important;
 | |
|         padding: 1em;
 | |
|         min-height: 12em;
 | |
|       }
 | |
|       .status-msg {
 | |
|         font-size: 1.25em !important;
 | |
|         padding: 0.8em 1.6em;
 | |
|       }
 | |
|       button[type=submit] {
 | |
|         font-size: 1.3em !important;
 | |
|         padding: 0.9em 2.2em;
 | |
|       }
 | |
|       input[type=file] {
 | |
|         font-size: 1.16em !important;
 | |
|         padding: inherit 2.2em;
 | |
|       }
 | |
|       label {
 | |
|         font-size: 1.16em !important;
 | |
|         padding: inherit 2.2em;
 | |
|       }
 | |
|       select {
 | |
|         font-size: 1.16em !important;
 | |
|         padding: 0.6em 1.2em;
 | |
|       }
 | |
|       input[type=number] {
 | |
|         font-size: 1.16em !important;
 | |
|         padding: 0.6em 1.2em;
 | |
|       }
 | |
|       /* i'm so sorry, i'm not bothered to find out why or how this works */
 | |
|       .options-bannermode-inner .checkmark {
 | |
|         top: -10%;
 | |
|       }
 | |
|       .options-bannermode-inner .checkmark::after {
 | |
|         left: 13px;
 | |
|         top: 7px;
 | |
|         transform: rotate(45deg) scale(1.66);
 | |
|       }
 | |
|       .checkmark {
 | |
|         width: 1.2em;
 | |
|         height: 1.2em;
 | |
|       }
 | |
|       .options-bannermode-inner {
 | |
|         padding-left: 55px;
 | |
|       }
 | |
|   </style>
 | |
| </head>
 | |
| <body>
 | |
|   <div class="centered-flex">
 | |
|     <div class="form-card">
 | |
|       <center>
 | |
|         <h2 style="margin-top:0.2em;">😺 CatNote 🖨️</h2>
 | |
|       </center>
 | |
|       <form method="post" enctype="multipart/form-data">
 | |
|         <textarea name="md" placeholder="Ingrese Markdown aquí...">{{ default_md }}</textarea><br>
 | |
|         <label for="userimg" style="display: inline-block; margin-bottom: 0.5em;">Subir imagen (opcional):</label><br>
 | |
|         <input type="file" name="userimg" accept="image/png, image/jpeg"><br>
 | |
|         <div class="options">
 | |
|         <div class="options-bannermode">
 | |
|           <label class="options-bannermode-inner">
 | |
|             <input type="checkbox" name="bannermode" value="1" {% if current_bannermode %}checked{% endif %}>
 | |
|             <span class="checkmark"></span>
 | |
|             Modo Banner (vertical)
 | |
|           </label>
 | |
|           </div>
 | |
|           <div class="options-dithering">
 | |
|             <label for="dithering">Dithering:</label>
 | |
|             <select name="dithering">
 | |
|               {% for label, mode in dithering_modes.items() %}
 | |
|                 <option value="{{ mode }}" {% if mode == current_dithering %}selected="selected"{% endif %}>{{ label }}</option>
 | |
|               {% endfor %}
 | |
|             </select>
 | |
|           </div>
 | |
|           <div class="options-intensity">
 | |
|             <label for="intensity">Intensidad:</label>
 | |
|             <input type="number" name="intensity" value="85" min="0" max="100" step="1">
 | |
|           </div>
 | |
|           <div class="options-rotation">
 | |
|             <label for="rotation">Rotar imagen:</label>
 | |
|             <select name="rotation">
 | |
|               <option value="0" {% if 0 == current_rotation %}selected="selected"{% endif %}>0°</option>
 | |
|               <option value="90" {% if 90 == current_rotation %}selected="selected"{% endif %}>90°</option>
 | |
|               <option value="180" {% if 180 == current_rotation %}selected="selected"{% endif %}>180°</option>
 | |
|               <option value="270" {% if 270 == current_rotation %}selected="selected"{% endif %}>270°</option>
 | |
|             </select>
 | |
|           </div>
 | |
|           <div class="options-printmode">
 | |
|             <label for="printmode">Modo de impresión:</label>
 | |
|             <select name="printmode">
 | |
|               <option value="1bpp" {% if "1bpp" == current_printmode %}selected="selected"{% endif %}>1-bit</option>
 | |
|               <option value="4bpp" {% if "4bpp" == current_printmode %}selected="selected"{% endif %}>4-bit</option>
 | |
|             </select>
 | |
|           </div>
 | |
|           <div id="loading-box">
 | |
|             <div id="printing-text">Imprimiendo...</div>
 | |
|             <div class="progress-bar">
 | |
|               <div class="progress-bar-inner"></div>
 | |
|             </div>
 | |
|           </div>
 | |
|         </div>
 | |
|         <div class="buttons">
 | |
|           <button type="submit" name="generate">📷 Generar</button>
 | |
|           <button type="submit" name="print">🖨️ Imprimir</button>
 | |
|         </div>
 | |
|       </form>
 | |
|       {% if printed %}
 | |
|         <div class="status-msg">✅ Enviado a impresora</div>
 | |
|       {% endif %}
 | |
|       {% if error %}
 | |
|         <div class="status-msg status-err">⚠️ {{ error }}</div>
 | |
|       {% endif %}
 | |
|     </div>
 | |
|     <div class="preview-card">
 | |
|       {% if img %}
 | |
|         <h2 style="margin-top:0.1em; margin-bottom:0;">Vista previa</h2>
 | |
|         <div class="print-frame">
 | |
|           <img src="data:image/png;base64,{{ img }}">
 | |
|         </div>
 | |
|       {% else %}
 | |
|         <p style="opacity:.6;">Su vista previa aparecerá aquí</p>
 | |
|       {% endif %}
 | |
|     </div>
 | |
|     <div class="markdown-ref">
 | |
|       <h4>Referencia rápida de Markdown</h4>
 | |
|       <ul>
 | |
|         <li><b>Negrita:</b> <code>**texto**</code></li>
 | |
|         <li><b>Cursiva:</b> <code>*texto*</code></li>
 | |
|         <li><b>Negrita y cursiva:</b> <code>***texto***</code></li>
 | |
|         <li><b>Encabezado grande:</b> <code># Título</code></li>
 | |
|         <li><b>Encabezado mediano:</b> <code>## Título</code></li>
 | |
|         <li><b>Encabezado chico:</b> <code>### Título</code></li>
 | |
|         <li><b>Lista con viñetas:</b> <code>- Elemento</code></li>
 | |
|         <li><b>Lista numerada:</b> <code>1. Elemento</code></li>
 | |
|         <li><b>Imágen:</b> <code></code></li>
 | |
|         <li><b>Salto de línea:</b> Deje una línea vacía</li>
 | |
|         <li><b>Imagen subida:</b> <code>!(img)</code> (usa la imagen cargada abajo)</li>
 | |
|       </ul>
 | |
|     </div>
 | |
|   </div>
 | |
|   <script>
 | |
|     document.querySelector('form').addEventListener('submit', function(e) {
 | |
|       if (document.activeElement && document.activeElement.name === "print") {
 | |
|         document.getElementById('loading-box').style.display = 'flex';
 | |
|       }
 | |
|     });
 | |
|   </script>
 | |
| </body>
 | |
| </html>
 | |
| '''
 | |
| 
 | |
| def remove_uploaded_img():
 | |
|     path = session.pop('uploaded_img_path', None)
 | |
|     if path and os.path.exists(path):
 | |
|         try:
 | |
|             os.remove(path)
 | |
|         except Exception:
 | |
|             pass
 | |
| 
 | |
| def bleh_image_from_url(url, dithering, mode):
 | |
|     resp = requests.get(url, stream=True)
 | |
|     resp.raise_for_status()
 | |
|     bleh = subprocess.Popen(
 | |
|         ["./bleh", "-o", "-", "-mode", f"{mode}", "-d", f"{dithering}", "-"], 
 | |
|         stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
 | |
|     out, err = bleh.communicate(resp.content)
 | |
|     if bleh.returncode != 0:
 | |
|         raise RuntimeError(f"Driver failed: {err.decode()}")
 | |
|     img = Image.open(io.BytesIO(out)).convert("L")
 | |
|     # Optionally check width, pad/resize if needed
 | |
|     if img.width != IMAGE_WIDTH:
 | |
|         img = img.resize((IMAGE_WIDTH, img.height), Image.LANCZOS)
 | |
|     return img
 | |
| 
 | |
| def bleh_image_from_bytes(image_bytes, dithering, mode):
 | |
|     bleh = subprocess.Popen(
 | |
|         ["./bleh", "-o", "-", "-mode", f"{mode}", "-d", f"{dithering}", "-"],
 | |
|         stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE
 | |
|     )
 | |
|     out, err = bleh.communicate(image_bytes)
 | |
|     if bleh.returncode != 0:
 | |
|         raise RuntimeError(f"Driver failed: {err.decode()}")
 | |
|     img = Image.open(io.BytesIO(out)).convert("L")
 | |
|     if img.width != IMAGE_WIDTH:
 | |
|         img = img.resize((IMAGE_WIDTH, img.height), Image.LANCZOS)
 | |
|     return img
 | |
| 
 | |
| def rotate_image_bytes(image_bytes, rotation):
 | |
|     if rotation == 0:
 | |
|         return image_bytes
 | |
|     img = Image.open(io.BytesIO(image_bytes))
 | |
|     img = img.rotate(-rotation, expand=True)  # PIL rotates counterclockwise, negative for clockwise
 | |
|     buf = io.BytesIO()
 | |
|     img.save(buf, format="PNG")
 | |
|     return buf.getvalue()
 | |
| 
 | |
| def parse_line(line):
 | |
|   if re.match(r"!\(img\)", line) or re.match(r"!\[.*\]\(img\)", line):
 | |
|       return ('userimage', None)
 | |
|   image_match = re.match(r"^!\[(.*?)\]\((.+?)\)", line)
 | |
|   if image_match:
 | |
|       alt = image_match.group(1)
 | |
|       url = image_match.group(2)
 | |
|       return ('image', url, alt)
 | |
|   if re.match(r'^\s*---\s*$', line):
 | |
|       return ('hr', [])
 | |
|   header_match = re.match(r"^(#{1,3}) +(.*)", line)
 | |
|   if header_match:
 | |
|       header_level = len(header_match.group(1))
 | |
|       line = header_match.group(2)
 | |
|       return ('header', header_level, parse_segments(line))
 | |
|   bullet_match = re.match(r"^\s*([-*\u2022]) +(.*)", line)
 | |
|   if bullet_match:
 | |
|       return ('bullet', parse_segments(bullet_match.group(2)))
 | |
|   ordered_match = re.match(r"^\s*(\d+)\. +(.*)", line)
 | |
|   if ordered_match:
 | |
|       return ('ordered', int(ordered_match.group(1)), parse_segments(ordered_match.group(2)))
 | |
|   return ('text', parse_segments(line))
 | |
| 
 | |
| def parse_segments(line):
 | |
|     # Handle escaped asterisks: replace them with a placeholder
 | |
|     line = line.replace(r'\*', '\x07')
 | |
| 
 | |
|     # Apply formatting
 | |
|     bi = re.findall(r"\*\*\*(.+?)\*\*\*", line)
 | |
|     for x in bi:
 | |
|         line = line.replace(f"***{x}***", f"\x01{x}\x02")
 | |
|     b = re.findall(r"\*\*(.+?)\*\*", line)
 | |
|     for x in b:
 | |
|         line = line.replace(f"**{x}**", f"\x03{x}\x04")
 | |
|     i = re.findall(r"\*(.+?)\*", line)
 | |
|     for x in i:
 | |
|         line = line.replace(f"*{x}*", f"\x05{x}\x06")
 | |
| 
 | |
|     # Split into styled segments
 | |
|     segments = []
 | |
|     i = 0
 | |
|     while i < len(line):
 | |
|         if line[i] == '\x01':
 | |
|             i += 1
 | |
|             start = i
 | |
|             while i < len(line) and line[i] != '\x02': i += 1
 | |
|             segments.append(('bolditalic', line[start:i]))
 | |
|             i += 1
 | |
|         elif line[i] == '\x03':
 | |
|             i += 1
 | |
|             start = i
 | |
|             while i < len(line) and line[i] != '\x04': i += 1
 | |
|             segments.append(('bold', line[start:i]))
 | |
|             i += 1
 | |
|         elif line[i] == '\x05':
 | |
|             i += 1
 | |
|             start = i
 | |
|             while i < len(line) and line[i] != '\x06': i += 1
 | |
|             segments.append(('italic', line[start:i]))
 | |
|             i += 1
 | |
|         else:
 | |
|             start = i
 | |
|             while i < len(line) and line[i] not in '\x01\x03\x05': i += 1
 | |
|             if i > start:
 | |
|                 segments.append(('text', line[start:i]))
 | |
| 
 | |
|     # Restore literal asterisks
 | |
|     segments = [(style, text.replace('\x07', '*')) for style, text in segments]
 | |
|     return segments
 | |
| 
 | |
| def font_for_style(style, font, font_bold, font_italic, font_bolditalic):
 | |
|     if style == 'bolditalic':
 | |
|         return font_bolditalic or font_bold or font_italic or font
 | |
|     elif style == 'bold':
 | |
|         return font_bold
 | |
|     elif style == 'italic':
 | |
|         return font_italic
 | |
|     else:
 | |
|         return font
 | |
| 
 | |
| def wrap_segments(segments, font, font_bold, font_italic, font_bolditalic, max_width, start_x=0):
 | |
|     line = []
 | |
|     x = start_x
 | |
|     for style, text in segments:
 | |
|         words = re.split(r'(\s+)', text)
 | |
|         for word in words:
 | |
|             if word == '':
 | |
|                 continue
 | |
|             f = font_for_style(style, font, font_bold, font_italic, font_bolditalic)
 | |
|             w = f.getbbox(word)[2] - f.getbbox(word)[0]
 | |
|             if x + w > max_width and line:
 | |
|                 yield line
 | |
|                 line = []
 | |
|                 x = start_x
 | |
|             line.append((style, word))
 | |
|             x += w
 | |
|     if line:
 | |
|         yield line
 | |
| 
 | |
| def render(md, dithering, printmode, uploaded_img_bytes=None, bannermode=False):
 | |
|     font = ImageFont.truetype(FONT_PATH, FONT_SIZE)
 | |
|     font_bold = ImageFont.truetype(FONT_PATH_BOLD, FONT_SIZE)
 | |
|     font_italic = ImageFont.truetype(FONT_PATH_OBLIQUE, FONT_SIZE)
 | |
|     font_banner = ImageFont.truetype(FONT_PATH_BOLD, BANNER_FONT_SIZE)
 | |
|     try:
 | |
|         font_bolditalic = ImageFont.truetype(FONT_PATH_BOLDITALIC, FONT_SIZE)
 | |
|     except:
 | |
|         font_bolditalic = None
 | |
|     font_h1 = ImageFont.truetype(FONT_PATH_BOLD, HEADER_SIZE_1)
 | |
|     font_h2 = ImageFont.truetype(FONT_PATH_BOLD, HEADER_SIZE_2)
 | |
|     font_h3 = ImageFont.truetype(FONT_PATH_BOLD, HEADER_SIZE_3)
 | |
| 
 | |
|     if bannermode:
 | |
|         # Remove line breaks for single-line banner
 | |
|         md = md.replace('\r\n', ' ') # should change this if we're ever on windows
 | |
|         # Only render as a single text line (ignore markdown except bold/italic)
 | |
|         segments = parse_segments(md)
 | |
|         # Calculate total width needed
 | |
|         x = 0
 | |
|         for style, text in segments:
 | |
|             f = font_banner
 | |
|             x += f.getbbox(text)[2] - f.getbbox(text)[0]
 | |
|         width = max(x, 1)
 | |
|         # Create image: width x 384 (height is printer width)
 | |
|         image = Image.new("L", (width, IMAGE_WIDTH), 255)
 | |
|         draw = ImageDraw.Draw(image)
 | |
|         x = 0
 | |
|         y = (IMAGE_WIDTH - BANNER_FONT_SIZE) // 2  # Center vertically
 | |
|         for style, text in segments:
 | |
|             f = font_banner
 | |
|             draw.text((x, y), text, font=f, fill=0)
 | |
|             x += f.getbbox(text)[2] - f.getbbox(text)[0]
 | |
|         # Rotate so text is vertical
 | |
|         image = image.rotate(270, expand=True)
 | |
|         return image
 | |
|     
 | |
|     lines_out = []
 | |
|     
 | |
|     for src_line in md.splitlines():
 | |
|         if src_line.strip() == '':
 | |
|             lines_out.append(('blank', []))
 | |
|             continue
 | |
|         tag = parse_line(src_line)            
 | |
|         if tag[0] == 'image':
 | |
|             try:
 | |
|                 image = bleh_image_from_url(tag[1], dithering, printmode)
 | |
|                 lines_out.append(('image', image))
 | |
|             except Exception as e:
 | |
|                 lines_out.append(('text', [('text', f"[Imagen inválida: {e}]")], font, FONT_SIZE))
 | |
|         elif tag[0] == 'userimage':
 | |
|           if uploaded_img_bytes:
 | |
|             try:
 | |
|                 image = bleh_image_from_bytes(uploaded_img_bytes, dithering, printmode)
 | |
|                 lines_out.append(('image', image))
 | |
|             except Exception as e:
 | |
|                 lines_out.append(('text', [('text', f"[Error al procesar imagen]")], font, FONT_SIZE))
 | |
|                 print(f"Image processing error: {e}")
 | |
|           else:
 | |
|             lines_out.append(('text', [('text', "[No se subió una imagen]")], font, FONT_SIZE))
 | |
|         elif tag[0] == 'hr':
 | |
|             lines_out.append(('hr',))
 | |
|         elif tag[0] == 'header':
 | |
|             header_level = tag[1]
 | |
|             segments = tag[2]
 | |
|             if header_level == 1:
 | |
|                 font_h = font_h1
 | |
|                 size_h = HEADER_SIZE_1
 | |
|             elif header_level == 2:
 | |
|                 font_h = font_h2
 | |
|                 size_h = HEADER_SIZE_2
 | |
|             else:
 | |
|                 font_h = font_h3
 | |
|                 size_h = HEADER_SIZE_3
 | |
|             for wrapped in wrap_segments(segments, font_h, font_h, font_h, font_h, IMAGE_WIDTH):
 | |
|                 lines_out.append(('header', wrapped, font_h, size_h))
 | |
|         elif tag[0] == 'bullet':
 | |
|             segments = tag[1]
 | |
|             bullet_font = font
 | |
|             bullet_w = bullet_font.getbbox(BULLET_CHAR)[2] - bullet_font.getbbox(BULLET_CHAR)[0]
 | |
|             wrapped_lines = list(wrap_segments(segments, font, font_bold, font_italic, font_bolditalic, IMAGE_WIDTH - bullet_w, start_x=0))
 | |
|             for i, wrapped in enumerate(wrapped_lines):
 | |
|                 if i == 0:
 | |
|                     lines_out.append(('bullet', wrapped, bullet_font, FONT_SIZE, True, bullet_w))
 | |
|                 else:
 | |
|                     lines_out.append(('bullet', wrapped, bullet_font, FONT_SIZE, False, bullet_w))
 | |
|         elif tag[0] == 'ordered':
 | |
|             idx, segments = tag[1], tag[2]
 | |
|             num_str = f"{idx}. "
 | |
|             number_font = font
 | |
|             num_w = number_font.getbbox(num_str)[2] - number_font.getbbox(num_str)[0]
 | |
|             wrapped_lines = list(wrap_segments(segments, font, font_bold, font_italic, font_bolditalic, IMAGE_WIDTH - num_w, start_x=0))
 | |
|             for i, wrapped in enumerate(wrapped_lines):
 | |
|                 if i == 0:
 | |
|                     lines_out.append(('ordered', wrapped, number_font, FONT_SIZE, num_str, True, num_w))
 | |
|                 else:
 | |
|                     lines_out.append(('ordered', wrapped, number_font, FONT_SIZE, num_str, False, num_w))
 | |
|         else:  # normal text
 | |
|             segments = tag[1]
 | |
|             for wrapped in wrap_segments(segments, font, font_bold, font_italic, font_bolditalic, IMAGE_WIDTH):
 | |
|                 lines_out.append(('text', wrapped, font, FONT_SIZE))
 | |
| 
 | |
|     # Compute total height, including images
 | |
|     height = 10  # Top margin
 | |
|     for item in lines_out:
 | |
|         if item[0] in ('header', 'text', 'bullet', 'ordered'):
 | |
|             height += item[3]
 | |
|         elif item[0] == 'hr':
 | |
|             height += 10
 | |
|         elif item[0] == 'blank':
 | |
|             height += FONT_SIZE
 | |
|         elif item[0] == 'image':
 | |
|             img = item[1]
 | |
|             height += img.height + 10  # add margin below image
 | |
| 
 | |
|     image = Image.new("L", (IMAGE_WIDTH, height), 255)
 | |
|     draw = ImageDraw.Draw(image)
 | |
|     y = 0
 | |
|     for item in lines_out:
 | |
|         if item[0] == 'blank':
 | |
|             y += FONT_SIZE
 | |
|         elif item[0] == 'hr':
 | |
|             draw.line((0, y + 5, IMAGE_WIDTH, y + 5), fill=0, width=2)
 | |
|             y += 10
 | |
|         elif item[0] == 'header':
 | |
|             segments, fnt, sz = item[1], item[2], item[3]
 | |
|             x = 0
 | |
|             for style, text in segments:
 | |
|                 draw.text((x, y), text, font=fnt, fill=0)
 | |
|                 x += fnt.getbbox(text)[2] - fnt.getbbox(text)[0]
 | |
|             y += sz
 | |
|         elif item[0] == 'bullet':
 | |
|             segments, bullet_font, sz, show_bullet, bullet_w = item[1], item[2], item[3], item[4], item[5]
 | |
|             x = 0
 | |
|             if show_bullet:
 | |
|                 draw.text((x, y), BULLET_CHAR, font=bullet_font, fill=0)
 | |
|                 x += bullet_w
 | |
|             else:
 | |
|                 x += bullet_w
 | |
|             for style, text in segments:
 | |
|                 f = font_for_style(style, bullet_font, font_bold, font_italic, font_bolditalic)
 | |
|                 draw.text((x, y), text, font=f, fill=0)
 | |
|                 x += f.getbbox(text)[2] - f.getbbox(text)[0]
 | |
|             y += sz
 | |
|         elif item[0] == 'ordered':
 | |
|             segments, number_font, sz, num_str, show_num, num_w = item[1], item[2], item[3], item[4], item[5], item[6]
 | |
|             x = 0
 | |
|             if show_num:
 | |
|                 draw.text((x, y), num_str, font=number_font, fill=0)
 | |
|                 x += num_w
 | |
|             else:
 | |
|                 x += num_w
 | |
|             for style, text in segments:
 | |
|                 f = font_for_style(style, number_font, font_bold, font_italic, font_bolditalic)
 | |
|                 draw.text((x, y), text, font=f, fill=0)
 | |
|                 x += f.getbbox(text)[2] - f.getbbox(text)[0]
 | |
|             y += sz
 | |
|         elif item[0] == 'text':
 | |
|             segments, fnt, sz = item[1], item[2], item[3]
 | |
|             x = 0
 | |
|             for style, text in segments:
 | |
|                 f = font_for_style(style, font, font_bold, font_italic, font_bolditalic)
 | |
|                 draw.text((x, y), text, font=f, fill=0)
 | |
|                 x += f.getbbox(text)[2] - f.getbbox(text)[0]
 | |
|             y += sz
 | |
|         elif item[0] == 'image':
 | |
|             img = item[1]
 | |
|             # Center image horizontally if narrower
 | |
|             img_x = (IMAGE_WIDTH - img.width) // 2 if img.width < IMAGE_WIDTH else 0
 | |
|             image.paste(img, (img_x, y))
 | |
|             y += img.height + 10  # vertical margin after image
 | |
|     if bannermode:
 | |
|         image = image.rotate(270, expand=True)
 | |
|     return image
 | |
| 
 | |
| @app.route("/", methods=["GET", "POST"])
 | |
| def index():
 | |
|     img_data = None
 | |
|     md = ""
 | |
|     printed = False
 | |
|     error = None
 | |
|     if request.method == "POST":
 | |
|         userimg = request.files.get("userimg")
 | |
|         uploaded_img_bytes = None
 | |
|         rotation = int(request.form.get("rotation", "0"))
 | |
|         if userimg and userimg.filename:
 | |
|             # Remove old temp file if present
 | |
|             remove_uploaded_img()
 | |
|             # Save new file to a temp location
 | |
|             with tempfile.NamedTemporaryFile(delete=False, suffix=".catnote") as tmpf:
 | |
|                 userimg.save(tmpf)
 | |
|                 tmpf.flush()
 | |
|                 session['uploaded_img_path'] = tmpf.name
 | |
|                 tmpf.seek(0)
 | |
|                 uploaded_img_bytes = tmpf.read()
 | |
|         elif 'uploaded_img_path' in session:
 | |
|             path = session['uploaded_img_path']
 | |
|             if os.path.exists(path):
 | |
|                 with open(path, "rb") as f:
 | |
|                     uploaded_img_bytes = f.read()
 | |
|             else:
 | |
|                 # If file was deleted or missing, remove from session
 | |
|                 session.pop('uploaded_img_path', None)
 | |
|         else:
 | |
|             remove_uploaded_img()
 | |
|         if uploaded_img_bytes and rotation in (90, 180, 270):
 | |
|             try:
 | |
|                 uploaded_img_bytes = rotate_image_bytes(uploaded_img_bytes, rotation)
 | |
|             except Exception as e:
 | |
|                 error = f"Error rotating image: {e}"
 | |
|                 uploaded_img_bytes = None
 | |
|         md = request.form["md"]
 | |
|         dithering = request.form.get("dithering", "floyd")
 | |
|         printmode = request.form.get("printmode", "1bpp")
 | |
|         bannermode = bool(request.form.get("bannermode"))
 | |
|         image = render(md, dithering, printmode, uploaded_img_bytes, bannermode=bannermode)
 | |
|         session['dithering'] = dithering
 | |
|         session['printmode'] = printmode
 | |
|         session['rotation'] = rotation
 | |
|         session['bannermode'] = bannermode
 | |
|         buf = io.BytesIO()
 | |
|         image.save(buf, format="PNG")
 | |
|         buf.seek(0)
 | |
|         img_data = base64.b64encode(buf.getvalue()).decode()
 | |
|         intensity = request.form.get("intensity", "85")
 | |
|         try:
 | |
|             intensity = int(intensity)
 | |
|             if intensity < 0 or intensity > 100:
 | |
|                 raise ValueError("Intensity must be between 0 and 100")
 | |
|         except ValueError:
 | |
|             error = "Intensidad debe ser un número entre 0 y 100"
 | |
|             intensity = 85 
 | |
|         session['intensity'] = intensity
 | |
|         # If print button pressed, send to driver
 | |
|         if "print" in request.form:
 | |
|             try:
 | |
|                 with tempfile.NamedTemporaryFile(suffix=".png", delete=True) as tmpfile:
 | |
|                     image.save(tmpfile, format="PNG")
 | |
|                     tmpfile.flush()
 | |
|                     # Run the bleh command
 | |
|                     result = subprocess.run([
 | |
|                         "./bleh",
 | |
|                         "-mode", f"{printmode}",
 | |
|                         "-intensity", f"{intensity}",
 | |
|                         tmpfile.name
 | |
|                     ], capture_output=True, text=True, timeout=90)
 | |
|                     if result.returncode != 0:
 | |
|                         error = f"Printer error: {result.stderr or result.stdout}"
 | |
|                     else:
 | |
|                         printed = True
 | |
|             except Exception as e:
 | |
|                 error = f"Failed to print: {e}"
 | |
|     return render_template_string(HTML_FORM, dithering_modes=DITHERING_MODES, img=img_data, default_md=md,
 | |
|         printed=printed, error=error, current_dithering=session.get('dithering', 'floyd'), current_rotation=session.get('rotation', 0), current_printmode=session.get('printmode', '1bpp'), current_bannermode=session.get('bannermode', False))
 | |
| 
 | |
| @app.route('/manifest.json')
 | |
| def manifest():
 | |
|     return send_from_directory('static', 'manifest.json')
 | |
| 
 | |
| @app.route('/icon512_maskable.png')
 | |
| def icon_maskable():
 | |
|     return send_from_directory('static', 'icon512_maskable.png')
 | |
| 
 | |
| @app.route('/icon512_rounded.png')
 | |
| def icon_rounded():
 | |
|     return send_from_directory('static', 'icon512_rounded.png')
 | |
| 
 | |
| if __name__ == "__main__":
 | |
|     parser = argparse.ArgumentParser(description='CatNote: Server for Markdown to MXW01 Cat Printer')
 | |
|     parser.add_argument('-p', '--port', type=int, default=5000, help='Port to run the server on (default: 5000)')
 | |
|     args = parser.parse_args()
 | |
|     app.run(host='0.0.0.0', port=args.port)
 | |
|   
 |