Skip to main content

High-performance tool for improving AI-generated pixel art

Project description

unfake

Improve AI-generated pixel art through scale detection, color quantization, and smart downscaling. Features Rust acceleration for critical operations, achieving a 10-20% speedup over the original JavaScript implementation.

Based on the excellent work by:

Examples

Click each image to view the original, processed result, and results of two different naïve fixed-size-nearest-neighbor methods. comparison_grid comparison_grid comparison_grid comparison_grid comparison_grid

Images taken from examples for AI pixel art models like Pixel Art XL, FLUX.1-Dev Pixel LoRA, and FLUX.1-Kontext Pixel LoRA.

Features

  • Automatic Scale Detection: Detects the inherent scale of pixel art using both runs-based and edge-aware methods
  • Advanced Color Quantization: Wu color quantization algorithm with Rust acceleration
  • Smart Downscaling: Multiple methods including dominant color, median, mode, and content-adaptive
  • Image Cleanup: Alpha binarization, morphological operations, and jaggy edge removal
  • Grid Snapping: Automatic alignment to pixel grid for clean results
  • Flexible API: Both synchronous and asynchronous interfaces
  • Fast: Process a 1-megapixel image in as fast as half a second.

Upcoming

  • Vectorization

Installation

From PyPI (recommended)

pip install unfake

From Source

# Clone the repository
git clone https://github.com/painebenjamin/unfake.py.git
cd unfake

# Install with pip (includes Rust compilation)
pip install .

# Or for development
pip install -e .

Requirements

  • Python 3.8+
  • Rust toolchain (for building from source)
  • OpenCV Python bindings
  • Pillow
  • NumPy

Usage

Command Line

# Basic usage with auto-detection
unfake input.png

# Specify output file
unfake input.png -o output.png

# Control color palette size
unfake input.png -c 16                    # Maximum 16 colors
unfake input.png --auto-colors            # Auto-detect optimal color count

# Force specific scale
unfake input.png --scale 4                # Force 4x downscaling

# Choose downscaling method
unfake input.png -m dominant              # Dominant color (default, best for pixel art)
unfake input.png -m median                # Median color
unfake input.png -m content-adaptive      # High quality but slower

# Enable cleanup operations
unfake input.png --cleanup morph,jaggy    # Morphological + jaggy edge cleanup

# Use fixed color palette
unfake input.png --palette palette.txt    # File with hex colors, one per line

# Adjust processing parameters
unfake input.png --alpha-threshold 200    # Higher threshold for alpha binarization
unfake input.png --threshold 0.1          # Dominant color threshold (0.0-1.0)
unfake input.png --no-snap                # Disable grid snapping

# Verbose output
unfake input.png -v                       # Show detailed processing info

Python API

import unfake

# Basic processing with defaults
result = unfake.process_image_sync(
    "input.png",
    max_colors=32,                       # Maximum colors in output
    detect_method="auto",                # Scale detection: "auto", "runs", "edge"
    downscale_method="dominant",         # Method: "dominant", "median", "mode", "mean", "content-adaptive"
    cleanup={"morph": False, "jaggy": False},
    snap_grid=True                       # Align to pixel grid
)

# Access results
processed_image = result['image']        # PIL Image
palette = result['palette']              # List of hex colors
manifest = result['manifest']            # Processing metadata

# Auto-detect optimal colors
result = unfake.process_image_sync(
    "input.png",
    max_colors=None,                     # Auto-detect
    auto_color_detect=True
)

# Use fixed palette
fixed_colors = ['#000000', '#ffffff', '#ff0000', '#00ff00', '#0000ff']
result = unfake.process_image_sync(
    "input.png",
    fixed_palette=fixed_colors
)

Asynchronous API

import asyncio
import unfake

async def process_image_async():
    result = await unfake.process_image(
        "input.png",
        max_colors=16,
        detect_method="runs",
        downscale_method="median",
        cleanup={"morph": True, "jaggy": False},
        snap_grid=True
    )
    result["image"].save("output.png")

asyncio.run(process_image_async())

Processing Options

Scale Detection Methods

  • auto (default): Tries runs-based first, falls back to edge-aware
  • runs: Analyzes color run lengths (fast, works well for clean pixel art)
  • edge: Uses edge detection (slower but handles anti-aliased images)

Downscaling Methods

  • dominant (default): Uses most frequent color in each block (best for pixel art)
  • median: Median color value (good for photos)
  • mode: Most common color (similar to dominant)
  • mean: Average color (can create new colors)
  • content-adaptive: Advanced algorithm based on Kopf & Lischinski 2011

Cleanup Options

  • morph: Morphological operations to remove noise
  • jaggy: Removes isolated diagonal pixels

Performance

Example processing times for a 1024x1024 image on a high-end Intel desktop CPU using defaults:

  • Pure Python: ~71 seconds
  • With Rust Acceleration: ~700 milliseconds (about 100x speedup!)

Algorithm Details

Scale Detection

The tool uses two methods to detect the inherent scale of pixel art:

  1. Runs-based: Analyzes horizontal and vertical color runs to find the GCD
  2. Edge-aware: Uses Sobel edge detection to find regular grid patterns

Color Quantization

Implements the Wu color quantization algorithm (1992) which:

  • Builds a 3D color histogram
  • Recursively subdivides color space
  • Minimizes variance within each partition
  • Produces high-quality palettes

Downscaling

The dominant color method:

  • Divides image into scale×scale blocks
  • Finds most frequent color in each block
  • Falls back to mean if no color is dominant
  • Preserves original palette colors

Credits

This Python/Rust implementation is based on:

  • unfake.js by Eugeniy Smirnov - The original JavaScript implementation that inspired this project
  • image-quantization by Igor Bezkrovnyi - TypeScript implementation of various color quantization algorithms

Additional references:

  • Wu, Xiaolin. "Efficient Statistical Computations for Optimal Color Quantization" (1992)
  • Kopf, Johannes and Dani Lischinski. "Depixelizing Pixel Art" (2011)

License

MIT License

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distributions

No source distribution files available for this release.See tutorial on generating distribution archives.

Built Distributions

If you're not sure about the file name format, learn more about wheel file names.

unfake-1.0.3-cp38-abi3-win_amd64.whl (269.2 kB view details)

Uploaded CPython 3.8+Windows x86-64

unfake-1.0.3-cp38-abi3-manylinux_2_34_x86_64.whl (351.0 kB view details)

Uploaded CPython 3.8+manylinux: glibc 2.34+ x86-64

unfake-1.0.3-cp38-abi3-macosx_11_0_arm64.whl (306.4 kB view details)

Uploaded CPython 3.8+macOS 11.0+ ARM64

File details

Details for the file unfake-1.0.3-cp38-abi3-win_amd64.whl.

File metadata

  • Download URL: unfake-1.0.3-cp38-abi3-win_amd64.whl
  • Upload date:
  • Size: 269.2 kB
  • Tags: CPython 3.8+, Windows x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.11.12

File hashes

Hashes for unfake-1.0.3-cp38-abi3-win_amd64.whl
Algorithm Hash digest
SHA256 2eecb1dde955ce9af40af714d0ed412ee25a1767517badce394194c7f75fc4d1
MD5 4497a2e7aec9fd07d0f85f1f8e32c10e
BLAKE2b-256 8875a36eca941e98ef5da2cb1d20bfc3c59f34aeff9dd5a4d4426ab4cbb3efbf

See more details on using hashes here.

File details

Details for the file unfake-1.0.3-cp38-abi3-manylinux_2_34_x86_64.whl.

File metadata

File hashes

Hashes for unfake-1.0.3-cp38-abi3-manylinux_2_34_x86_64.whl
Algorithm Hash digest
SHA256 43afa95189c908198dfd414254db0884595c61f8a831b111540d171678f9c1fd
MD5 520bbfe6b700ee15b1bb4f65072edb10
BLAKE2b-256 e54360fc9638598bc28d8c3af64e70cf4b0b864f2a7f564940448e140e761e7b

See more details on using hashes here.

File details

Details for the file unfake-1.0.3-cp38-abi3-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for unfake-1.0.3-cp38-abi3-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 1c69adf2d301583c30a9b390f4e8aac823ae016c00982ca1b510c84406a0a268
MD5 923a5fd4e1cbe0bb6a635e5806782a20
BLAKE2b-256 284e68fda91bd6609cc89f3a6bd4eb2ae434fcdf7622b9ea1ccdcac78f6b74da

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page