ascii/ascii.py

361 lines
10 KiB
Python
Raw Normal View History

2018-01-20 20:44:04 -05:00
#!/usr/bin/env python3
"""
ASCII
"""
from io import BytesIO
import requests
from PIL import Image, ImageFont, ImageDraw
#import imageio
import numpy as np
import numpngw
ASCII_CHARS = "$@%#*+=-:. "
2017-12-07 00:04:04 -05:00
BRAIL_CHARS = "⠿⠾⠼⠸⠰⠠ "
2017-12-04 05:23:17 -05:00
HEADERS = {'User-Agent': 'Gimme ascii.'}
2018-01-20 20:44:04 -05:00
def scale_image(image, maxDim=(100,100)):
"""
Resizes an image while preserving the aspect ratio. Chooses the
dimension to scale by based on whichever is larger, ensuring that
2017-12-04 05:23:17 -05:00
neither width or height is ever larger than the maxDim argument.
Because text characters are typically pretty close to a 1:2 rectangle,
we weight the width twice as much.
"""
original_width, original_height = image.size
original_width = original_width * 2
2018-01-20 20:44:04 -05:00
if original_width <= maxDim[0] and original_height <= maxDim[1]:
2017-12-07 00:04:04 -05:00
new_width, new_height = image.size
elif original_width > original_height:
2018-01-20 20:44:04 -05:00
new_width = maxDim[0]
2017-12-07 00:04:04 -05:00
aspect_ratio = original_height/float(original_width)
new_height = int(aspect_ratio * new_width)
else:
2018-01-20 20:44:04 -05:00
new_height = maxDim[1]
2017-12-07 00:04:04 -05:00
aspect_ratio = original_width/float(original_height)
new_width = int(aspect_ratio * new_height)
2017-12-16 01:57:48 -05:00
image = image.resize((new_width, new_height))
return image
2017-12-07 00:04:04 -05:00
def pixels_to_chars(image, scale="ascii", color_code=None):
"""
Maps each pixel to an ascii char based on where it falls in the range
2017-12-07 00:04:04 -05:00
0-255 normalized to the length of the chosen scale.
"""
2017-12-07 00:04:04 -05:00
scales = {"ascii": ASCII_CHARS,
"ascii_reverse": "".join(reversed(ASCII_CHARS)),
"brail": BRAIL_CHARS,
"brail_reverse": "".join(reversed(BRAIL_CHARS))}
2017-12-07 00:04:04 -05:00
color_prefix = {"irc": "\03", "ansi":"\033"}
range_width = int(255 / len(scales[scale])) + (255 % len(scales[scale]) > 0)
2017-12-07 00:04:04 -05:00
pixels = list(image.getdata())
pixels = [pixels[i:i + image.size[0]] for i in range(0, len(pixels),
image.size[0])]
2017-12-07 00:04:04 -05:00
chars = []
for row in pixels:
new_row = ""
for pixel in row:
R, G, B = pixel
L = R * 299/1000 + G * 587/1000 + B * 114/1000
index = int(L/range_width)
char = scales[scale][index]
if color_code and char is not " ":
prefix = color_prefix[color_code] + char_color(pixel,color_code)
char = prefix + char
new_row += char
chars.append(new_row)
if color_code == "ansi":
chars.append("\033[0m")
return "\n".join(chars)
def char_color(pixel, code="irc"):
"""
Maps a color to a character based on the original pixel color.
Calculates the distance from the provided color to each of the 16
colors in the IRC and ANSI color codes and selects the closest match.
"""
colors_index = ["white", "black", "blue", "green", "red", "brown", "purple",
"orange", "yellow", "light green", "teal", "cyan", "light blue", "pink",
"grey", "silver"]
colors_rgb = {"white": (0,0,0), "black": (255,255,255), "blue": (0,0,255),
"green": (0,255,0), "red": (255,0,0), "brown": (150,75,0),
"purple": (128,0,128), "orange": (255,128,0), "yellow": (255,255,0),
"light green": (191,255,0), "teal": (0,128,128), "cyan": (0,255,255),
"light blue": (65,105,225), "pink":(255,192,203), "grey": (128,128,128),
"silver": (192,192,192)}
2017-11-30 00:04:45 -05:00
colors_irc = {"white": "0", "black": "1", "blue": "2", "green": "3",
"red": "4", "brown": "5", "purple": "6", "orange": "7", "yellow": "8",
"light green": "9", "teal": "10", "cyan": "11", "light blue": "12",
"pink": "13", "grey": "14", "silver": "15"}
colors_ansi = {"white": "[1;37m", "black": "[0;30m", "blue": "[0;34m",
"green": "[0;32m", "red": "[0;31m", "brown": "[0;33m",
"purple": "[0;35m", "orange": "[1;31m", "yellow": "[1;33m",
"light green": "[1;32m", "teal": "[0;36m", "cyan": "[1;36m",
"light blue": "[1;34m", "pink": "[1;35m", "grey": "[1;30m",
"silver": "[0;37m"}
dist = [(abs(pixel[0] - colors_rgb[color][0])**2
+ abs(pixel[1] - colors_rgb[color][1])**2
+ abs(pixel[2] - colors_rgb[color][2])**2)**0.5
for color in colors_index]
color = colors_index[dist.index(min(dist))]
if code == "irc":
return colors_irc[color]
elif code == "ansi":
return colors_ansi[color]
def open_image(imagePath):
"""
Opens the image at the supplied file path in PIL. If an internet URL
is supplied, it will download the image and then open it. Returns a
PIL image object.
"""
try:
if imagePath.startswith("http"):
res = requests.get(imagePath, headers=HEADERS, verify=True,
timeout=20)
2017-11-30 00:04:45 -05:00
if res.status_code == 404:
return "404: file not found."
res.raise_for_status()
image = Image.open(BytesIO(res.content))
else:
image = Image.open(imagePath)
2017-11-30 00:04:45 -05:00
except FileNotFoundError:
return f"File not found: {imagePath}"
except Exception as e:
return(f"Error opening image: {imagePath}\n{e}")
return image
2017-12-16 01:57:48 -05:00
def alpha_composite(image, color=(255, 255, 255)):
"""
2017-12-16 01:57:48 -05:00
Alpha composite an RGBA Image with a specified color.
Source: http://stackoverflow.com/a/9166671/284318
"""
2017-12-16 01:57:48 -05:00
image.load() # needed for split()
background = Image.new('RGB', image.size, color)
background.paste(image, mask=image.split()[3]) # 3 is the alpha channel
return background
2017-12-08 11:34:54 -05:00
def image_to_ascii(image=None, reverse=False, brail=False, color=None,**kwargs):
"""
Reads an image file and converts it to ascii art. Returns a
newline-delineated string. If reverse is True, the ascii scale is
reversed.
"""
2017-12-07 00:04:04 -05:00
if not image:
image = open_image(kwargs["imagePath"])
if image.mode == "P":
image = image.convert(image.palette.mode)
if image.mode == "RGBA":
2017-12-16 01:57:48 -05:00
image = alpha_composite(image).convert("RGB")
2017-12-07 00:04:04 -05:00
image = scale_image(image)
2017-12-08 11:34:54 -05:00
if brail:
2017-12-07 00:04:04 -05:00
scale = "brail"
else:
scale = "ascii"
2017-12-08 11:34:54 -05:00
if reverse:
2017-12-07 00:04:04 -05:00
scale += "_reverse"
2017-12-08 11:34:54 -05:00
chars = pixels_to_chars(image, scale, color)
image.close()
del(image)
return chars
def ascii_to_image(image_ascii):
"""
Creates a plain image and draws text on it.
"""
2017-11-30 00:04:45 -05:00
# TODO: make font type, size and color non-fixed
width = len(image_ascii[:image_ascii.index("\n")]) * 8
height = (image_ascii.count("\n")+1) * 12 + 4
font = ImageFont.truetype("LiberationMono-Regular.ttf", 14)
image = Image.new("RGB", (width, height), (255,255,255))
draw = ImageDraw.Draw(image)
draw.text((0,0), image_ascii, (0,0,0), font=font, spacing=0)
return image
2017-12-08 11:34:54 -05:00
def handle_gif(imagePath, **kwargs):
"""
Handle gifs seperately.
"""
2017-12-08 11:34:54 -05:00
image = open_image(imagePath)
ascii_seq = []
2017-12-07 00:04:04 -05:00
new_image = ascii_to_image(image_to_ascii(image, **kwargs))
image.seek(1)
while True:
try:
2017-12-07 00:04:04 -05:00
im = ascii_to_image(image_to_ascii(image, **kwargs))
ascii_seq.append(im)
image.seek(image.tell()+1)
except EOFError:
break # end of sequence
2017-11-30 00:04:45 -05:00
# new_image.save(output, save_all=True, append_images=ascii_seq,
# duration=60, loop=0, optimize=True)
ascii_seq = [new_image] + ascii_seq
np_ascii_seq = [np.array(im) for im in ascii_seq]
2017-12-07 00:04:04 -05:00
with open(kwargs["output"], "wb") as file:
numpngw.write_apng(file, np_ascii_seq)
2018-01-20 20:44:04 -05:00
def brail_char(chunk, threshold):
"""
Accepts a numpy matrix and spits out a brail character.
"""
chunk = np.array_split(chunk, 3, axis=0)
chunk = np.concatenate(chunk, axis=1)
chunk = np.array_split(chunk, 6, axis=1)
dots = ""
for sub_chunk in chunk:
if np.mean(sub_chunk) < threshold:
dots += "1"
else:
dots += "0"
char = chr(int(dots, base=2)+10240)
return char
def image_to_brail(image, fontSize=(8,15)):
"""
An alternative method of generating brail ascii art.
"""
if not image:
image = open_image(kwargs["imagePath"])
if image.mode == "P":
image = image.convert(image.palette.mode)
if image.mode == "RGBA":
image = alpha_composite(image).convert("RGB")
image = image.convert("L")
matSize = (image.size[0] // fontSize[0], image.size[1] // fontSize[1])
if image.size[0] > fontSize[0]*100 or image.size[1] > fontSize[1]*100:
image = scale_image(image, (fontSize[0]*100, fontSize[1]*100))
image = image.crop((0, 0, matSize[0]*fontSize[0], matSize[1]*fontSize[1]))
threshold = np.mean(image)
grid = np.array(image)
grid = np.split(grid, matSize[1], axis=0)
grid = np.concatenate(grid, axis=1)
grid = np.split(grid, matSize[0]*matSize[1], axis=1)
for n, chunk in enumerate(grid):
char = brail_char(chunk, threshold)
grid[n] = char
grid = "".join(grid)
grid = [grid[n : n + matSize[0]] for n in range(0, len(grid), matSize[0])]
return "\n".join(grid)
if __name__=='__main__':
import argparse
parser = argparse.ArgumentParser(
description="Converts an image file to ascii art.")
parser.add_argument(
"imagePath",
2017-12-04 05:23:17 -05:00
help="The path to the image file. May be a local path or internet URL.")
parser.add_argument(
"-r",
"--reverse",
action="store_true",
help="Reverses the ascii scale.")
parser.add_argument(
"-o",
"--output",
help="Outputs the ascii art into a file at the specified path.")
parser.add_argument(
"-i",
"--image",
2017-12-07 00:04:04 -05:00
dest="drawImage",
action="store_true",
2017-11-30 00:04:45 -05:00
help="Outputs the ascii art as an image rather than text. Requires \
--output.")
parser.add_argument(
"-a",
"--animated",
action="store_true",
help="Handles animated GIFs. Includes --image.")
parser.add_argument(
"-c",
"--color",
type=str,
2017-11-30 00:04:45 -05:00
help="Colorizes the ascii matrix. Currently supported modes are 'irc' \
2017-12-04 05:23:17 -05:00
and 'ansi' for generating color codes compliant with those standards.")
2017-11-30 00:04:45 -05:00
parser.add_argument(
"--ansi",
dest="color",
action="store_const",
const="ansi",
help="Shortcut for '--color ansi'.")
parser.add_argument(
"--irc",
dest="color",
action="store_const",
const="irc",
help="Shortcut for '--color irc'.")
2017-12-07 00:04:04 -05:00
parser.add_argument(
"-b",
"--brail",
action="store_true",
help="Uses brail unicode characters instead of ascii characters.")
2018-01-20 20:44:04 -05:00
parser.add_argument(
"-B",
"--brail2",
action="store_true",
help="A better braille algorithm for a better you.")
args = parser.parse_args()
if args.animated: # --animated includes --image
2017-12-07 00:04:04 -05:00
args.drawImage = True
if args.drawImage: # --image requires --output
if not args.output:
parser.error("--image requires --output")
if args.animated:
2017-12-07 00:04:04 -05:00
handle_gif(**vars(args))
exit()
2018-01-20 20:44:04 -05:00
if args.brail2:
image = open_image(args.imagePath)
chars = image_to_brail(image)
print(chars)
exit()
2017-12-07 00:04:04 -05:00
image_ascii = image_to_ascii(None, **vars(args))
if args.drawImage:
image = ascii_to_image(image_ascii)
image.save(args.output, "PNG")
elif args.output:
with open(args.output, "w+") as file:
file.write(image_ascii)
else:
print(image_ascii)