-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathtext_on_image.py
89 lines (64 loc) · 2.31 KB
/
text_on_image.py
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
import os
import textwrap
from PIL import Image, ImageDraw, ImageFont
import requests
FONT_SIZE = 30
FONT = ImageFont.truetype('opensans.ttf', FONT_SIZE)
TEXT_COLOR = (0, 0, 0, 255) # black, 100% opacity
OVERLAY_COLOR = (255, 255, 255, 178) # white, ~70% opacity
TEXT_OFFSET = (20, 20)
TEXT_CHAR_LIMIT = 250
RESIZE_WIDTH = 500
TEXT_WIDTH = 30 # experimental
CARDS = 'cards'
IN_FILE = os.path.join(CARDS, 'input.png')
OUT_FILE = os.path.join(CARDS, 'output.png')
def download_url(url, in_file=IN_FILE, chunk_size=2000):
print('Downloading {}'.format(url))
r = requests.get(url, stream=True)
print('Saving as {}'.format(in_file))
with open(in_file, 'wb') as fd:
for chunk in r.iter_content(chunk_size):
fd.write(chunk)
return in_file
def get_input(prompt):
# https://stackoverflow.com/questions/30239092/how-to-get-multiline-input-from-user
print(prompt)
lines = []
while True:
line = input()
if line:
line = textwrap.fill(line, width=TEXT_WIDTH)
lines.append(line)
else:
break
return '\n\n'.join(lines)
def _resize(img):
basewidth = RESIZE_WIDTH
wpercent = basewidth / float(img.size[0])
hsize = int(float(img.size[1]) * float(wpercent))
return img.resize((basewidth, hsize), Image.ANTIALIAS)
def create_img_with_text(base_img, input_text, out_file=OUT_FILE):
input_text = textwrap.fill(input_text, width=TEXT_WIDTH)
if os.path.isfile(out_file):
os.remove(out_file)
base = Image.open(base_img).convert('RGBA')
base = _resize(base)
txt = Image.new('RGBA', base.size, OVERLAY_COLOR)
draw_context = ImageDraw.Draw(txt)
draw_context.text(TEXT_OFFSET, input_text, font=FONT, fill=TEXT_COLOR)
out = Image.alpha_composite(base, txt)
out.save(out_file)
return out_file
if __name__ == '__main__':
resource = input('Image file (local or url to download): ')
if os.path.isfile(resource):
base_img = resource
else:
base_img = download_url(resource)
prompt = 'Text to put on image '
prompt += '(max {} chars, double enter = end): '.format(TEXT_CHAR_LIMIT)
input_text = get_input(prompt)
input_text = input_text[:TEXT_CHAR_LIMIT]
out_file = create_img_with_text(base_img, input_text)
print('{} created'.format(out_file))