1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
|
#!/usr/bin/env python
import xtermcolors
reset_sequence = '\033[39;49m'
def termify_pixels(img):
sx, sy = img.size
out = ''
fg,bg = None,None
fgd,bgd = {},{}
def bgescape(color):
nonlocal bg, bgd
if bg == color:
return ''
bg=color
if color == (0,0,0,0):
return '\033[49m'
if color in bgd:
return bgd[color]
r,g,b,_ = color
bgd[color] = '\033[48;5;'+str(xtermcolors.closest_color(r,g,b))+'m'
return bgd[color]
def fgescape(color):
nonlocal fg, fgd
if fg == color:
return ''
fg=color
r,g,b,_ = color
fgd[color] = '\033[38;5;'+str(xtermcolors.closest_color(r,g,b))+'m'
return fgd[color]
def balloon(x,y):
if x+1 == img.size[0] or img.im.getpixel((x+1, y)) != (0,255,0,127):
w = 1
while x-w >= 0 and img.im.getpixel((x-w, y)) == (0,255,0,127):
w += 1
return '$balloon{}$'.format(w)
return ''
for y in range(0, sy, 2):
for x in range(sx):
coltop = img.im.getpixel((x, y))
colbot = img.im.getpixel((x, y+1)) if y+1 < img.size[1] else (0,0,0,0)
if coltop[3] == 127: #Control colors
out += reset_sequence
out += {(255, 0, 0, 127): lambda x,y:'$\\$',
(0, 0, 255, 127): lambda x,y:'$/$',
(0, 255, 0, 127): balloon
}.get(coltop, lambda x,y:' ')(x,y)
continue
if coltop[3] != 255:
coltop = (0,0,0,0)
if colbot[3] != 255:
colbot = (0,0,0,0)
#Da magicks: ▀█▄
c,cf = '▀','█'
te,be = fgescape,bgescape
if coltop == (0,0,0,0) or ((coltop == bg or colbot == fg) and not colbot == (0,0,0,0)):
c,cf,te,be = '▄',' ',be,te
if colbot == coltop:
c,te,be = cf,te,te
out += te(coltop) + be(colbot) + c
out = (out.rstrip() if bg == (0,0,0,0) else out) + '\n'
return out[:-1] + reset_sequence + '\n'
if __name__ == '__main__':
import os, sys, argparse, os.path, json
from multiprocessing import Pool
from PIL import Image, PngImagePlugin
parser = argparse.ArgumentParser(description='Render pixel images on 256-color ANSI terminals')
parser.add_argument('image', type=str, nargs='*')
parser.add_argument('-d', '--output-dir', type=str, help='Output directory (if not given, output to stdout)')
args = parser.parse_args()
def convert(f):
img = Image.open(f).convert("RGBA")
if args.output_dir:
print(f)
foo, _, _ = f.rpartition('.png')
output = os.path.join(args.output_dir, os.path.basename(foo)+'.pony')
metadata = json.loads(img.info.get('pixelterm-metadata'))
comment = metadata.get('_comment')
if comment is not None:
del metadata['_comment']
comment = '\n'+comment
else:
comment = ''
metadataarea = '$$$\n' +\
'\n'.join([ '\n'.join([ k.upper() + ': ' + v for v in metadata[k] ]) for k in sorted(metadata.keys()) ]) +\
comment + '\n$$$\n'
with open(output, 'w') as of:
of.write(metadataarea)
of.write(termify_pixels(img))
else:
print(termify_pixels(img))
p = Pool()
p.map(convert, args.image)
|