Spaces:
Sleeping
Sleeping
| from flask import Flask, request, jsonify, send_file | |
| from flask_cors import CORS | |
| import vtracer | |
| from PIL import Image | |
| import io | |
| import os | |
| import logging | |
| import gradio as gr | |
| # Set up logging | |
| logging.basicConfig(level=logging.INFO) | |
| logger = logging.getLogger(__name__) | |
| app = Flask(__name__) | |
| CORS(app, resources={r"/convert": {"origins": ["https://www.figma.com", "*"]}}) | |
| # VTracer conversion function | |
| def convert_to_vector( | |
| image, | |
| colormode="color", | |
| hierarchical="stacked", | |
| mode="spline", | |
| filter_speckle=4, | |
| color_precision=6, | |
| layer_difference=16, | |
| corner_threshold=60, | |
| length_threshold=4.0, | |
| max_iterations=10, | |
| splice_threshold=45, | |
| path_precision=3 | |
| ): | |
| input_path = "temp_input.jpg" | |
| output_path = "svg_output.svg" | |
| try: | |
| # Save the input image to a temporary file | |
| image.save(input_path) | |
| logger.info(f"Saved image to {input_path}") | |
| # Convert the image to SVG using VTracer | |
| vtracer.convert_image_to_svg_py( | |
| input_path, | |
| output_path, | |
| colormode=colormode, | |
| hierarchical=hierarchical, | |
| mode=mode, | |
| filter_speckle=int(filter_speckle), | |
| color_precision=int(color_precision), | |
| layer_difference=int(layer_difference), | |
| corner_threshold=int(corner_threshold), | |
| length_threshold=float(length_threshold), | |
| max_iterations=int(max_iterations), | |
| splice_threshold=int(splice_threshold), | |
| path_precision=int(path_precision) | |
| ) | |
| logger.info(f"Converted image to SVG at {output_path}") | |
| # Read the SVG output | |
| with open(output_path, "r") as f: | |
| svg_content = f.read() | |
| return svg_content | |
| except Exception as e: | |
| logger.error(f"Error in convert_to_vector: {str(e)}") | |
| raise Exception(f"Conversion failed: {str(e)}") | |
| finally: | |
| # Clean up temporary files | |
| for path in [input_path, output_path]: | |
| if os.path.exists(path): | |
| try: | |
| os.remove(path) | |
| logger.info(f"Removed {path}") | |
| except Exception as e: | |
| logger.warning(f"Failed to remove {path}: {str(e)}") | |
| # Flask endpoint for vector conversion | |
| def convert_image(): | |
| try: | |
| # Handle image upload | |
| if 'file' not in request.files: | |
| return jsonify({'error': 'No image file provided'}), 400 | |
| file = request.files['file'] | |
| image = Image.open(file).convert('RGB') | |
| # Get parameters (with defaults) | |
| colormode = request.form.get('colormode', 'color') | |
| hierarchical = request.form.get('hierarchical', 'stacked') | |
| mode = request.form.get('mode', 'spline') | |
| filter_speckle = int(request.form.get('filter_speckle', 4)) | |
| color_precision = int(request.form.get('color_precision', 6)) | |
| layer_difference = int(request.form.get('layer_difference', 16)) | |
| corner_threshold = int(request.form.get('corner_threshold', 60)) | |
| length_threshold = float(request.form.get('length_threshold', 4.0)) | |
| max_iterations = int(request.form.get('max_iterations', 10)) | |
| splice_threshold = int(request.form.get('splice_threshold', 45)) | |
| path_precision = int(request.form.get('path_precision', 3)) | |
| logger.info("Received request to /convert") | |
| # Convert to SVG | |
| svg_content = convert_to_vector( | |
| image, | |
| colormode=colormode, | |
| hierarchical=hierarchical, | |
| mode=mode, | |
| filter_speckle=filter_speckle, | |
| color_precision=color_precision, | |
| layer_difference=layer_difference, | |
| corner_threshold=corner_threshold, | |
| length_threshold=length_threshold, | |
| max_iterations=max_iterations, | |
| splice_threshold=splice_threshold, | |
| path_precision=path_precision | |
| ) | |
| # Return SVG as JSON | |
| return jsonify({'svg': svg_content}) | |
| except Exception as e: | |
| logger.error(f"Error in convert_image: {str(e)}") | |
| return jsonify({'error': str(e)}), 500 | |
| # Health check endpoint | |
| def health_check(): | |
| logger.info("Health check requested") | |
| return jsonify({'status': 'Image to Vector Converter API is running'}) | |
| # Optional Gradio interface (comment out if not needed) | |
| def handle_color_mode(value): | |
| return value | |
| examples_dir = "examples" | |
| examples = [ | |
| os.path.join(examples_dir, f) for f in ["11.jpg", "02.jpg", "03.jpg"] | |
| if os.path.exists(os.path.join(examples_dir, f)) | |
| ] | |
| css = """ | |
| #col-container { | |
| margin: 0 auto; | |
| max-width: 960px; | |
| } | |
| .generate-btn { | |
| background: linear-gradient(90deg, #4B79A1 0%, #283E51 100%) !important; | |
| border: none !important; | |
| color: white !important; | |
| } | |
| .generate-btn:hover { | |
| transform: translateY(-2px); | |
| box-shadow: 0 5px 15px rgba(0,0,0,0.2); | |
| } | |
| """ | |
| with gr.Blocks(css=css) as gradio_app: | |
| with gr.Column(elem_id="col-container"): | |
| gr.HTML(""" | |
| <div style="text-align: center;"> | |
| <h2>Image to Vector Converter ⚡</h2> | |
| <p>Converts raster images (JPG, PNG, WEBP) to vector graphics (SVG).</p> | |
| </div> | |
| """) | |
| with gr.Row(): | |
| with gr.Column(): | |
| image_input = gr.Image(type="pil", label="Upload Image") | |
| with gr.Accordion("Advanced Settings", open=False): | |
| with gr.Accordion("Clustering", open=False): | |
| colormode = gr.Radio([("COLOR", "color"), ("B/W", "binary")], value="color", label="Color Mode", show_label=False) | |
| filter_speckle = gr.Slider(0, 128, value=4, step=1, label="Filter Speckle", info="Cleaner") | |
| color_precision = gr.Slider(1, 8, value=6, step=1, label="Color Precision", info="More accurate") | |
| layer_difference = gr.Slider(0, 128, value=16, step=1, label="Gradient Step", info="Less layers") | |
| hierarchical = gr.Radio([("STACKED", "stacked"), ("CUTOUT", "cutout")], value="stacked", label="Hierarchical Mode", show_label=False) | |
| with gr.Accordion("Curve Fitting", open=False): | |
| mode = gr.Radio([("SPLINE", "spline"), ("POLYGON", "polygon"), ("PIXEL", "none")], value="spline", label="Mode", show_label=False) | |
| corner_threshold = gr.Slider(0, 180, value=60, step=1, label="Corner Threshold", info="Smoother") | |
| length_threshold = gr.Slider(3.5, 10, value=4.0, step=0.1, label="Segment Length", info="More coarse") | |
| splice_threshold = gr.Slider(0, 180, value=45, step=1, label="Splice Threshold", info="Less accurate") | |
| max_iterations = gr.Slider(1, 20, value=10, step=1, label="Max Iterations", visible=False) | |
| path_precision = gr.Slider(1, 10, value=3, step=1, label="Path Precision", visible=False) | |
| output_text = gr.Textbox(label="Selected Mode", visible=False) | |
| with gr.Row(): | |
| clear_button = gr.Button("Clear") | |
| convert_button = gr.Button("✨ Convert to SVG", variant="primary", elem_classes=["generate-btn"]) | |
| with gr.Column(): | |
| html = gr.HTML(label="SVG Output") | |
| svg_output = gr.File(label="Download SVG") | |
| if examples: | |
| gr.Examples( | |
| examples=examples, | |
| fn=convert_to_vector, | |
| inputs=[image_input], | |
| outputs=[html, svg_output], | |
| cache_examples=False, | |
| run_on_click=True | |
| ) | |
| colormode.change(handle_color_mode, inputs=colormode, outputs=output_text) | |
| hierarchical.change(handle_color_mode, inputs=hierarchical, outputs=output_text) | |
| mode.change(handle_color_mode, inputs=mode, outputs=output_text) | |
| def clear_inputs(): | |
| return ( | |
| gr.Image(value=None), gr.Radio(value="color"), gr.Radio(value="stacked"), | |
| gr.Radio(value="spline"), gr.Slider(value=4), gr.Slider(value=6), | |
| gr.Slider(value=16), gr.Slider(value=60), gr.Slider(value=4.0), | |
| gr.Slider(value=10), gr.Slider(value=45), gr.Slider(value=3) | |
| ) | |
| def update_interactivity_and_visibility(colormode, color_precision_value, layer_difference_value): | |
| is_color_mode = colormode == "color" | |
| return ( | |
| gr.update(interactive=is_color_mode), | |
| gr.update(interactive=is_color_mode), | |
| gr.update(visible=is_color_mode) | |
| ) | |
| colormode.change( | |
| update_interactivity_and_visibility, | |
| inputs=[colormode, color_precision, layer_difference], | |
| outputs=[color_precision, layer_difference, hierarchical] | |
| ) | |
| def update_interactivity_and_visibility_for_mode(mode): | |
| is_spline_mode = mode == "spline" | |
| return ( | |
| gr.update(interactive=is_spline_mode), | |
| gr.update(interactive=is_spline_mode), | |
| gr.update(interactive=is_spline_mode) | |
| ) | |
| mode.change( | |
| update_interactivity_and_visibility_for_mode, | |
| inputs=[mode], | |
| outputs=[corner_threshold, length_threshold, splice_threshold] | |
| ) | |
| clear_button.click( | |
| clear_inputs, | |
| outputs=[ | |
| image_input, colormode, hierarchical, mode, filter_speckle, | |
| color_precision, layer_difference, corner_threshold, length_threshold, | |
| max_iterations, splice_threshold, path_precision | |
| ] | |
| ) | |
| convert_button.click( | |
| convert_to_vector, | |
| inputs=[ | |
| image_input, colormode, hierarchical, mode, filter_speckle, | |
| color_precision, layer_difference, corner_threshold, length_threshold, | |
| max_iterations, splice_threshold, path_precision | |
| ], | |
| outputs=[html, svg_output] | |
| ) | |
| # Mount Gradio app at /gradio (optional) | |
| try: | |
| from gradio import mount_gradio_app | |
| from flask import Flask | |
| app = mount_gradio_app(app, gradio_app, path="/gradio") | |
| logger.info("Gradio app mounted successfully at /gradio") | |
| except Exception as e: | |
| logger.error(f"Failed to mount Gradio app: {str(e)}") | |
| if __name__ == '__main__': | |
| app.run(host='0.0.0.0', port=7860) |