2023-06-26 06:41:28 +00:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# Scroll IRC Art Bot - Developed by acidvegas in Python (https://git.acid.vegas/scroll)
|
|
|
|
|
2023-06-26 06:44:30 +00:00
|
|
|
'''
|
2023-06-26 06:50:27 +00:00
|
|
|
Props:
|
|
|
|
- forked idea from malcom's img2irc (https://github.com/waveplate/img2irc)
|
|
|
|
- big props to wrk (wr34k) for forking this one
|
2023-06-26 06:44:30 +00:00
|
|
|
|
|
|
|
pull request: https://github.com/ircart/scroll/pull/3
|
|
|
|
|
|
|
|
'''
|
2023-06-26 06:41:28 +00:00
|
|
|
|
|
|
|
import io
|
|
|
|
|
|
|
|
try:
|
|
|
|
from PIL import Image
|
|
|
|
except ImportError:
|
|
|
|
raise SystemExit('missing required \'pillow\' library (https://pypi.org/project/pillow/)')
|
|
|
|
|
2023-06-28 18:08:37 +00:00
|
|
|
palettes = {
|
|
|
|
'RGB88': [0xffffff, 0x000000, 0x00007f, 0x009300, 0xff0000, 0x7f0000, 0x9c009c, 0xfc7f00,
|
|
|
|
0xffff00, 0x00fc00, 0x009393, 0x00ffff, 0x0000fc, 0xff00ff, 0x0, 0x0,
|
|
|
|
0x470000, 0x472100, 0x474700, 0x324700, 0x004700, 0x00472c, 0x004747, 0x002747,
|
|
|
|
0x000047, 0x2e0047, 0x470047, 0x47002a, 0x740000, 0x743a00, 0x747400, 0x517400,
|
|
|
|
0x007400, 0x007449, 0x007474, 0x004074, 0x000074, 0x4b0074, 0x740074, 0x740045,
|
|
|
|
0xb50000, 0xb56300, 0xb5b500, 0x7db500, 0x00b500, 0x00b571, 0x00b5b5, 0x0063b5,
|
|
|
|
0x0000b5, 0x7500b5, 0xb500b5, 0xb5006b, 0xff0000, 0xff8c00, 0xffff00, 0xb2ff00,
|
|
|
|
0x00ff00, 0x00ffa0, 0x00ffff, 0x008cff, 0x0000ff, 0xa500ff, 0xff00ff, 0xff0098,
|
|
|
|
0xff5959, 0xffb459, 0xffff71, 0xcfff60, 0x6fff6f, 0x65ffc9, 0x6dffff, 0x59b4ff,
|
|
|
|
0x5959ff, 0xc459ff, 0xff66ff, 0xff59bc, 0xff9c9c, 0xffd39c, 0xffff9c, 0xe2ff9c,
|
|
|
|
0x9cff9c, 0x9cffdb, 0x9cffff, 0x9cd3ff, 0x9c9cff, 0xdc9cff, 0xff9cff, 0xff94d3]
|
|
|
|
|
|
|
|
'RGB99': [0xffffff, 0x000000, 0x00007f, 0x009300, 0xff0000, 0x7f0000, 0x9c009c, 0xfc7f00,
|
|
|
|
0xffff00, 0x00fc00, 0x009393, 0x00ffff, 0x0000fc, 0xff00ff, 0x7f7f7f, 0xd2d2d2,
|
|
|
|
0x470000, 0x472100, 0x474700, 0x324700, 0x004700, 0x00472c, 0x004747, 0x002747,
|
|
|
|
0x000047, 0x2e0047, 0x470047, 0x47002a, 0x740000, 0x743a00, 0x747400, 0x517400,
|
|
|
|
0x007400, 0x007449, 0x007474, 0x004074, 0x000074, 0x4b0074, 0x740074, 0x740045,
|
|
|
|
0xb50000, 0xb56300, 0xb5b500, 0x7db500, 0x00b500, 0x00b571, 0x00b5b5, 0x0063b5,
|
|
|
|
0x0000b5, 0x7500b5, 0xb500b5, 0xb5006b, 0xff0000, 0xff8c00, 0xffff00, 0xb2ff00,
|
|
|
|
0x00ff00, 0x00ffa0, 0x00ffff, 0x008cff, 0x0000ff, 0xa500ff, 0xff00ff, 0xff0098,
|
|
|
|
0xff5959, 0xffb459, 0xffff71, 0xcfff60, 0x6fff6f, 0x65ffc9, 0x6dffff, 0x59b4ff,
|
|
|
|
0x5959ff, 0xc459ff, 0xff66ff, 0xff59bc, 0xff9c9c, 0xffd39c, 0xffff9c, 0xe2ff9c,
|
|
|
|
0x9cff9c, 0x9cffdb, 0x9cffff, 0x9cd3ff, 0x9c9cff, 0xdc9cff, 0xff9cff, 0xff94d3,
|
|
|
|
0x000000, 0x131313, 0x282828, 0x363636, 0x4d4d4d, 0x656565, 0x818181, 0x9f9f9f,
|
|
|
|
0xbcbcbc, 0xe2e2e2, 0xffffff]
|
|
|
|
}
|
|
|
|
|
|
|
|
def convert(data, max_line_len, img_width=80, palette='RBG88'):
|
|
|
|
if palette not in palettes:
|
|
|
|
raise Exception('invalid palette option')
|
|
|
|
palette = palettes[palette]
|
2023-06-26 06:41:28 +00:00
|
|
|
image = Image.open(io.BytesIO(data))
|
|
|
|
del data
|
2023-06-28 18:08:37 +00:00
|
|
|
return convert_image(image, max_line_len, img_width, palette)
|
2023-06-26 09:32:13 +00:00
|
|
|
|
2023-06-28 18:08:37 +00:00
|
|
|
def convert_image(image, max_line_len, img_width, palette):
|
2023-06-26 06:41:28 +00:00
|
|
|
(width, height) = image.size
|
|
|
|
img_height = img_width / width * height
|
|
|
|
del height, width
|
|
|
|
image.thumbnail((img_width, img_height), Image.Resampling.LANCZOS)
|
2023-06-26 09:32:13 +00:00
|
|
|
del img_height
|
2023-06-26 06:41:28 +00:00
|
|
|
CHAR = '\u2580'
|
2023-06-26 09:32:13 +00:00
|
|
|
buf = list()
|
|
|
|
for i in range(0, image.size[1], 2):
|
|
|
|
if i+1 >= image.size[1]:
|
|
|
|
bitmap = [[rgb_to_hex(image.getpixel((x, i))) for x in range(image.size[0])]]
|
|
|
|
bitmap += [[0 for _ in range(image.size[0])]]
|
|
|
|
else:
|
|
|
|
bitmap = [[rgb_to_hex(image.getpixel((x, y))) for x in range(image.size[0])] for y in [i, i+1]]
|
2023-06-28 18:08:37 +00:00
|
|
|
top_row = [AnsiPixel(px, palette) for px in bitmap[0]]
|
|
|
|
bottom_row = [AnsiPixel(px, palette) for px in bitmap[1]]
|
2023-06-26 09:32:13 +00:00
|
|
|
buf += [""]
|
|
|
|
last_fg = last_bg = -1
|
|
|
|
ansi_row = list()
|
|
|
|
for j in range(image.size[0]):
|
|
|
|
top_pixel = top_row[j]
|
|
|
|
bottom_pixel = bottom_row[j]
|
|
|
|
pixel_pair = AnsiPixelPair(top_pixel, bottom_pixel)
|
2023-06-26 06:41:28 +00:00
|
|
|
fg = pixel_pair.top.irc
|
|
|
|
bg = pixel_pair.bottom.irc
|
2023-06-26 09:32:13 +00:00
|
|
|
if j != 0:
|
2023-06-26 06:41:28 +00:00
|
|
|
if fg == last_fg and bg == last_bg:
|
2023-06-26 09:32:13 +00:00
|
|
|
buf[-1] += CHAR
|
2023-06-26 06:41:28 +00:00
|
|
|
elif bg == last_bg:
|
2023-06-26 09:32:13 +00:00
|
|
|
buf[-1] += f'\x03{fg}{CHAR}'
|
2023-06-26 06:41:28 +00:00
|
|
|
else:
|
2023-06-26 09:32:13 +00:00
|
|
|
buf[-1] += f'\x03{fg},{bg}{CHAR}'
|
2023-06-26 06:41:28 +00:00
|
|
|
else:
|
2023-06-26 09:32:13 +00:00
|
|
|
buf[-1] += f'\x03{fg},{bg}{CHAR}'
|
2023-06-26 06:41:28 +00:00
|
|
|
last_fg = fg
|
|
|
|
last_bg = bg
|
2023-06-26 09:32:13 +00:00
|
|
|
if len(buf[-1].encode('utf-8', 'ignore')) > max_line_len:
|
|
|
|
if img_width - 5 < 10:
|
|
|
|
raise Exception('internal error')
|
|
|
|
return convert_image(image, max_line_len, img_width-5)
|
|
|
|
return buf
|
2023-06-26 06:41:28 +00:00
|
|
|
|
|
|
|
def hex_to_rgb(color):
|
|
|
|
r = color >> 16
|
|
|
|
g = (color >> 8) % 256
|
|
|
|
b = color % 256
|
|
|
|
return (r,g,b)
|
|
|
|
|
|
|
|
def rgb_to_hex(rgb):
|
2023-06-26 09:32:13 +00:00
|
|
|
r = rgb[0]
|
|
|
|
g = rgb[1]
|
|
|
|
b = rgb[2]
|
2023-06-26 06:41:28 +00:00
|
|
|
return (r << 16) + (g << 8) + b
|
|
|
|
|
|
|
|
def color_distance_squared(c1, c2):
|
|
|
|
dr = c1[0] - c2[0]
|
|
|
|
dg = c1[1] - c2[1]
|
|
|
|
db = c1[2] - c2[2]
|
|
|
|
return dr * dr + dg * dg + db * db
|
|
|
|
|
|
|
|
class AnsiPixel:
|
2023-06-28 18:08:37 +00:00
|
|
|
def __init__(self, pixel_u32, palette):
|
|
|
|
self.irc = self.nearest_hex_color(pixel_u32, palette)
|
2023-06-26 06:41:28 +00:00
|
|
|
|
|
|
|
def nearest_hex_color(self, pixel_u32, hex_colors):
|
|
|
|
rgb_colors = [hex_to_rgb(color) for color in hex_colors]
|
|
|
|
rgb_colors.sort(key=lambda rgb: color_distance_squared(hex_to_rgb(pixel_u32), rgb))
|
|
|
|
hex_color = rgb_to_hex(rgb_colors[0])
|
|
|
|
return hex_colors.index(hex_color)
|
|
|
|
|
|
|
|
class AnsiPixelPair:
|
|
|
|
def __init__(self, top, bottom):
|
|
|
|
self.top = top
|
|
|
|
self.bottom = bottom
|