Skip to main content

Convert images to beautiful ANSI escape codes

Project description

CLImage 📷

Convert images to beautiful ANSI escape codes for display in command line interfaces.

Available as both a CLI application and a Python library.

demo

Features

  • Custom sized images
  • ASCII or Unicode support
    • Unicode enables 4x more detail
  • 8/16/256/Truecolor support, for a wider gamut of colors
  • Selectable system palettes to adjust for user terminal themes
  • Fast color lookup with KDTrees & memoization

Usage

CLImage is available as both a standalone CLI program, or available for import as a Python3 library.

CLI Program

By default converting an image will output in 256 color, as 80 columns, and ASCII (for reasonable compatibility).

$ climage image.png

demo

A nicer image can be obtained when enabling unicode and truecolor flags.

$ climage --unicode --truecolour image.png

demo

For display in TTYs such as the Linux terminal before starting X11 (also accessible by Ctrl-Alt-F3, etc), you should restrict yourself to 8 color, and ASCII only. As 8 and 16 colors are solely constructed out of system colors (often set by the terminal theme), it is recommended to select a palette for these, depending on what you have selected.

$ climage --8color warhol.png --palette solarized

demo

Further options may be found by running climage --help

Python Library

It is recommended that you use the Python library if converting a large number of images. Simple usage:

import climage

# Convert an image to a 50 character wide image.
output = climage.convert('image.png', is_unicode=True, width=50)
print(output)

# Convert an image using 8 color mode 100 columns wide, and write to file.
climage.to_file('image.png', 'out.txt', is_8color=True, width=100)

Formats

The API also supports supplying a Pillow Image object, or a numpy array representing an image.

These formats may be useful if you have already loaded from file, or are applying complex filters to the images prior to rendering.

Pillow Images can be converted via the convert_pil function:

import climage
from PIL import Image

# It is good practice to convert to RGB, before conversion. Climage supports
# RGBA also, but will render as if the alpha channel is completely opaque.
# Attempting to convert greyscale or other formats (e.g. RBG) will fail, or
# present an invalid result.
img = Image.open('image.png').convert('RGB')
output = climage.convert_pil(img, is_unicode=True)
print(output)

Similarly, numpy arrays are converted using the convert_arr function. In the example below we use Pillow to generate the numpy array, but any row-major array whose element represents an RGB triplet. Each colour is represented by a number in the range [0,255].

import climage
from PIL import Image

# Convert the image into 50px * 50px, as the convert_array function does not
# perform resizing.
img = Image.open('image.png').convert('RGB').resize((50, 50))
arr = np.array(img)
output = climage.convert_array(arr, is_unicode=True)
print(output)

Python lists are also accepted:

from climage import convert_array

# Simple 2x2 image showing raw python lists are usable too.
print(
  convert_array([
    [[128, 128, 0], [0, 128, 128]],
    [[0, 128, 0], [0, 0, 128]],
  ], is_unicode=True)
)

Output:

python list example

Helpers

To provide you more expressiveness and flexibility, various helper methods are exposed.

from climage import color_to_flags, color_types, convert

# color_to_flags allows you to easily define the color option required by
# convert. Accepts a value from the `color_types` enum, which has the following
# options:
#   - color_types.truecolor
#   - color_types.color256
#   - color_types.color16
#   - color_types.color8
print(convert('teapot.png', is_unicode=True, **color_to_flags(color_types.truecolor)))

Output:

teapot example

from climage import get_ansi_pixel, get_reset_code, color_types

# get_ansi_pixel lets you to convert an RGB color to the ANSI escape code
# sequence for representing that colour as a pixel. Similar to the convert
# functions, by default converts using a 256 color palette, and does not use
# the unicode characters.
# This call will print 2 characters wide, as it is not unicode.
pix = get_ansi_pixel(
  # Violet
  (148, 0, 211),
  ctype=color_types.truecolor,
)
# If you are printing this pixel singularly, you will want to reset the color
# back to normal. If you are wishing to print a bunch sequentially (as in, on
# the same line), you should append the reset code at the end.
print(pix + get_reset_code())

Output:

single pixel example

from climage import get_dual_unicode_ansi_pixels, get_reset_code, color_types

# get_dual_unicode_ansi_pixels is the unicode character equivalent of
# get_ansi_pixel. This accepts two colours, one for the upper pixel, and one
# for the bottom pixel.
pixels = get_dual_unicode_ansi_pixels(
  # Green
  (0, 128, 0),
  # Red
  (128, 0, 0),
  ctype=color_types.color256,
  palette="solarized",
)
print(pixels + get_reset_code())

Output:

dual pixel unicode example

Additional interfacing

As climage can convert arbitrary images in memory, converting images on the Internet is possible using the requests library:

import requests
import climage
from io import BytesIO

response = requests.get('https://www.python.org/static/community_logos/python-logo-master-v3-TM-flattened.png')
# Convert to RGB, as files on the Internet may be greyscale, which are not
# supported.
img = Image.open(BytesIO(response.content)).convert('RGB')
# Convert the image to 80col, in 256 color mode, using unicode for higher def.
converted = climage.convert_pil(img, is_unicode=True)
print(converted)

Output:

internet conversion

Note that caching the image may be more efficient for multiple invocations.

Future:

  • Improve performance in 256 color mapping, see here
  • Ideas? Let me know by filing an issue. :)

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

climage-0.2.1.tar.gz (13.6 kB view details)

Uploaded Source

Built Distribution

climage-0.2.1-py3-none-any.whl (12.1 kB view details)

Uploaded Python 3

File details

Details for the file climage-0.2.1.tar.gz.

File metadata

  • Download URL: climage-0.2.1.tar.gz
  • Upload date:
  • Size: 13.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.9.2

File hashes

Hashes for climage-0.2.1.tar.gz
Algorithm Hash digest
SHA256 a9b5728983af54f36357cf5eff824c7c5899d36ba6a0df1cc00af718691d7968
MD5 acb90d787678c5d42aef98d3da3320ec
BLAKE2b-256 a03d6bfb71da5f63956adda8df1d943d68949e0f101394074f1f2046bd38e0c5

See more details on using hashes here.

File details

Details for the file climage-0.2.1-py3-none-any.whl.

File metadata

  • Download URL: climage-0.2.1-py3-none-any.whl
  • Upload date:
  • Size: 12.1 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.9.2

File hashes

Hashes for climage-0.2.1-py3-none-any.whl
Algorithm Hash digest
SHA256 6a09c6f9d278d5cb8ff5c49cdaff7586fcf82e08cb8d25578d0a71ec0bfb60bc
MD5 3d139ec8ffa12e5e9e73e64b8f6dc76a
BLAKE2b-256 a0300baceedc7782ae300935652e142b1caa4c4080ef6abd6e69563b7bb443fd

See more details on using hashes here.

Supported by

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