Skip to main content

A high-performance, GPU-accelerated Bayer matrix dithering library and CLI for images, GIFs, and videos.

Project description

Dithered Image

Bayer Dithering

Python Version from PEP 621 TOML License GitHub Repo stars

A high-performance, GPU-accelerated Python library and Command-Line Interface (CLI) that applies Bayer matrix dithering to images, GIFs, and videos.

It offers a variety of customizable options such as matrix size, custom color filters, sharpness, contrast, and downscaling, powered by parallel processing for blazing-fast media generation.

Features

  • Hardware Acceleration: Choose between CPU or GPU (taichi backend) for massive performance gains, especially on videos.
  • Universal Media Support: Seamlessly process PNG, JPG, GIF, and MP4 files with automatic format detection.
  • Global CLI: Install once and use the dither command from anywhere in your terminal.
  • Customizable Filters: Apply beautiful retro color palettes easily.
  • Pre-Processing Pipeline: Built-in options for downscaling, contrast adjustment, and sharpening before the dithering effect is applied.

Installation

  1. Clone the repository:
    git clone https://github.com/madmattp/Bayer-Dithering.git
    cd Bayer-Dithering
    
  2. Install the package and its dependencies:
    pip install -e .
    
    (This will install the required libraries and link the dither command to your system).

Usage

Once installed, you can use the dither command directly in your terminal.

Command Line Options:

  • -i, --input: (Required) Specifies the input file (image, gif, or video).

  • -a, --arch: Processing hardware. Options: cpu, gpu (default: cpu).

  • -m, --matrix: Selects the Bayer matrix size. Options: 2x2, 4x4, 8x8 (default: 4x4).

  • -o, --output: Specifies the output file path. If not provided, a default name will be automatically generated.

  • -f, --filter: Applies a custom color filter to the output image (e.g., Matrix, Orange, Vapor).

  • -s, --sharpness: Adjusts the sharpness (default: 1.6).

  • -c, --contrast: Adjusts the contrast (default: 1.5).

  • -d, --downscale: Downscales the image by a factor before dithering (default: 2).

  • -u, --upscale: Upscales the image back to its original size after dithering (default: True). Set -u false to disable.

  • -q, --quiet: Runs the script in quiet mode, suppressing terminal output.

Recommended Settings

For more visually pleasing retro results, it is recommended to use the following settings:

  • Contrast: 1.5

  • Sharpness: 1.6

  • Downscaling: >= 2

Examples

1. Dithering an Image:

dither -i media/cat.jpg -d 6

Silly Cat Dithered Image

2. Dithering a Video with a Color Filter (GPU Accelerated):

dither -i media/huh.mp4 -a gpu -m 4x4 -c 1.5 -s 1.6 -f Cyan -u False -d 4

Python API Usage

You can also import BayerDithering directly into your own Python scripts to build custom graphics pipelines or integrate the effect into other applications.

1. Basic Image Processing (NumPy / OpenCV)

If you already have an image loaded as a NumPy array, you can process it directly:

import cv2
from BayerDithering import BayerDither, CPUProcessor, DitherConfig, matrices

# Create the pipeline configuration
config = DitherConfig(
    b_matrix=matrices["4x4"],
    contrast=1.5,
    sharpness=1.6,
    downscale_factor=2,
    upscale=True,
    filter=None  # Pass a tuple of RGB colors or None for grayscale
)

# Initialize the processor (CPU or GPUProcessor) and the ditherer
processor = CPUProcessor(config)
ditherer = BayerDither(processor=processor, verbose=True)

# Apply dithering to a NumPy array (BGR image from OpenCV)
image = cv2.imread("media/cat.jpg")
dithered_image = ditherer.apply(image)

# Save the result
cv2.imwrite("media/cat_dithered.png", dithered_image)

2. High-Performance Video Processing (GPU Accelerated)

For processing videos, pass a cv2.VideoCapture object directly into the apply method. The pipeline leverages the taichi backend for parallel GPU frame processing and returns a ProcessedVideo context manager:

import cv2
from BayerDithering import BayerDither, GPUProcessor, DitherConfig, matrices
from BayerDithering.utils import ProcessedVideo

config = DitherConfig(
    b_matrix=matrices["8x8"],
    contrast=1.3,
    sharpness=1.5,
    downscale_factor=2
)

# Use GPUProcessor for hardware acceleration
processor = GPUProcessor(config)
ditherer = BayerDither(processor=processor)

# Open the video stream using OpenCV
video_capture = cv2.VideoCapture("media/huh.mp4")

# Pass the video capture object to the ditherer.
# The router will automatically return a ProcessedVideo context.
# Always use 'with' to ensure secure handling and cleanup of temporary files.
with ditherer.apply(video_capture) as result:
    result.save_with_audio(original_video_path="media/huh.mp4", path="media/huh_dithered.mp4")

# Remember to release the video hardware resource
video_capture.release()

3. Animated GIF Processing

GIF processing relies on multi-frame arrays or lists of images. The apply method processes the frame sequence and returns a ProcessedGIF context manager to gracefully handle saving the output stream:

import imageio as iio
from BayerDithering import BayerDither, CPUProcessor, DitherConfig, matrices
from BayerDithering.utils import ProcessedGIF

config = DitherConfig(
    b_matrix=matrices["4x4"],
    contrast=1.6,
    sharpness=1.4,
    downscale_factor=3,
    upscale=True
)

# Initialize using CPU or GPU (both support GIF frame-by-frame processing)
processor = CPUProcessor(config)
ditherer = BayerDither(processor=processor)

# Load the GIF frames as a single multi-frame NumPy array using imageio
with iio.get_reader("media/cat-shocked.gif") as gif_frames:

  # Pass the frame sequence object to the ditherer. 
  # The router will return a ProcessedGIF instance
  with ditherer.apply(gif_frames) as result:
    result.save(dest_path="media/cat_shocked_dithered.gif")

4. Loading Custom Filters Programmatically

If you want to use the color palettes defined in your filters.toml dynamically inside a Python script:

from BayerDithering.utils import load_filters
from BayerDithering import DitherConfig, matrices

# Load all filters as a dictionary
filters = load_filters()

# Extract the RGB data for a specific palette (e.g., 'Cyan')
cyan_palette = filters.get("Cyan")

config = DitherConfig(
    b_matrix=matrices["4x4"],
    filter=cyan_palette  # Pass the loaded palette data to the configuration
)

Contributions

Feel free to open issues or contribute via pull requests. Contributions are welcome!

Project details


Download files

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

Source Distribution

bayerdithering-0.3.0.tar.gz (16.4 kB view details)

Uploaded Source

Built Distribution

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

bayerdithering-0.3.0-py3-none-any.whl (20.2 kB view details)

Uploaded Python 3

File details

Details for the file bayerdithering-0.3.0.tar.gz.

File metadata

  • Download URL: bayerdithering-0.3.0.tar.gz
  • Upload date:
  • Size: 16.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.13.13

File hashes

Hashes for bayerdithering-0.3.0.tar.gz
Algorithm Hash digest
SHA256 24e575ba63be1b979869d6f95da656d49db977f5d54066b751a97190d6223e1a
MD5 f3ab9f5319cd5238f6e2eeacf6f2e64d
BLAKE2b-256 09ef4435e20e67deda0a0b70ec08830d5f69cd6f8303f5aaba50d23241d2e92c

See more details on using hashes here.

File details

Details for the file bayerdithering-0.3.0-py3-none-any.whl.

File metadata

  • Download URL: bayerdithering-0.3.0-py3-none-any.whl
  • Upload date:
  • Size: 20.2 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.13.13

File hashes

Hashes for bayerdithering-0.3.0-py3-none-any.whl
Algorithm Hash digest
SHA256 a4354e7192b5efb41d7e63cecb7ce5e220b89e6c02b34033ad2c16f351fdebc7
MD5 d8f54aa7bdf0de327ef1c8cecd5e2d25
BLAKE2b-256 aae5de7034740398bfde93f46ce9f824192862b63b5654e02a03f1e981e1f925

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