-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathcontent.py
410 lines (326 loc) · 14.5 KB
/
content.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
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
import unittest
import log
import sys
import os
import json
import textwrap
import sysfont
import util
from PIL import Image, ImageDraw, ImageFont
def wrap_pixel_width(text, maxwidth, font, linesep='\n'):
""" Split into a list of text lines, such that none of them exceeds the pixel width """
ret = []
# Start by respecting any \n newlines in the text
paragraphs = text.split(linesep)
# Each of those sections is wrapped individually
for paragraph in paragraphs:
if (paragraph == ""): # respect empty lines...
ret += [" "]
continue
for chars in range(len(paragraph), 1, -1):
lines = textwrap.wrap(paragraph, chars)
too_wide = False
for l in lines:
lw, _ = font.getsize(l)
if (lw > maxwidth): too_wide = True
if (not too_wide):
ret += lines
break
if (len(ret) == 0): ret = None
return ret
# Generate a label-image of a size that fits the text and render it
def render_lines(lines, font, color="#000000", justify="left", spacing=4):
width = 0
height = 0
# The fixed height of one line of text. Let's say M is about right.
_, lineheight = font.getsize("M")
# Measure the text to figure out our bounds
for l in lines:
w,h = font.getsize(l)
width = max(w, width)
height = height + lineheight + spacing
# Some fonts render outside that calculated size, for some reason.
# Give them some margins and crop it off later.
image = Image.new("RGBA", (width*2, height*2), (0,0,0,0))
draw = ImageDraw.Draw(image)
# Render the text onto our label
y = 0
for l in lines:
w,_ = font.getsize(l)
# Alignment affects each line of text differently.
x = 0
if (justify == "center"): x = (width - w) / 2
if (justify == "right"): x = width - w
draw.text((x, y), l, font=font, fill=color)
y += lineheight + spacing
# Crop the image down to the exact bounds of the text
return image.crop(image.getbbox())
class TextLabel:
""" Parsed version of a single text-label object """
def __init__(self, json):
""" Parse out the various settings of a text label and clean them up """
self.name = util.get_default(json, "name", "text") # Only used in complex layout
self.source = util.get_default(json, "source", "text.txt") # Optional, used by simple layout
self.static = util.get_default(json, "static", None) # Optional, used by layout.
self.x = util.get_default(json, "x", 10, int)
self.y = util.get_default(json, "y", 10, int)
self.width = util.get_default(json, "width", 40000, int)
self.height = util.get_default(json, "height", 40000, int)
self.color = util.get_default(json, "color", "#000000")
self.fontface = util.get_default(json, "font-face", "Palatino Linotype")
self.fontsize = util.get_default(json, "font-size", 10, int)
self.spacing = util.get_default(json, "line-spacing", 4, int)
self.justify = util.get_default(json, "justify", "left")
self.x_align = util.get_default(json, "x-align", "left")
self.y_align = util.get_default(json, "y-align", "top")
self.wordwrap = util.get_default(json, "wordwrap", True, bool)
self.rotation = util.get_default(json, "rotation", 0, int)
weight = util.get_default(json, "font-weight", "regular")
self.fontweight = sysfont.STYLE_NORMAL
if (weight == "bold"): self.fontweight = sysfont.STYLE_BOLD
if (weight == "italic"): self.fontweight = sysfont.STYLE_ITALIC
fallback_fonts = [ "Arial", "liberation sans", "dejavu sans" ]
# Try to auto-select a font based on the user's string
candidate_font = sysfont.get_font(self.fontface, self.fontweight)
for font_name in fallback_fonts:
if (candidate_font is None):
# Fallback to arial
candidate_font = sysfont.get_font(font_name, self.fontweight)
if (candidate_font is not None):
log.log.write("Unable to locate font %s. Falling back to %s\n" % (self.fontface, candidate_font))
if (candidate_font is None):
log.log.write("Unable to locate font %s or any fallback. Unicode support will not be available.\n" % self.fontface)
self.font = ImageFont.load_default()
if (candidate_font is not None):
self.font = ImageFont.truetype(candidate_font, self.fontsize)
#print(candidate_font)
def render(self, card_dims, text):
""" Generate a transparent PIL card layer with the text on it """
# If the user has set a max width, respect that.
# If not, we use the edge of the card.
if (self.rotation == 0):
maxwidth, maxheight = util.aligned_maxdims((self.x, self.y),
(self.width, self.height),
card_dims,
self.x_align,
self.y_align)
else:
# Due to rotation of the text, we don't fully take the card's edges into account.
maxdim = max(card_dims)
maxwidth = min(self.width, maxdim)
maxheight = min(self.height, maxdim)
# Split the text into lines, in a way that fits our width
lines = [text]
if (self.wordwrap):
lines = wrap_pixel_width(text, maxwidth, self.font, linesep='\\n')
if (lines is None):
log.log.write("Warning: Unable to wrap text label \"%s\"\n" % text)
return None
# Render the text, one line at a time
label = render_lines(lines,
font=self.font,
color=self.color,
justify=self.justify,
spacing=self.spacing)
if (label.width > maxwidth):
log.log.write("Warning: Text label overflows max width (%d > %d): \"%s\"\n" % (label.width, maxwidth, text))
return None
if (label.height > maxheight):
log.log.write("Warning: Text label overflows max height (%d > %d): \"%s\"\n" % (label.height, maxheight, text))
return None
if (self.rotation != 0):
label = util.rotate_image(label, self.rotation)
# Figure out where to place the top-left corner of the label
x,y = util.alignment_to_absolute((self.x, self.y), label.size, self.x_align, self.y_align)
if (x < 0 or y < 0 or
x + label.width > card_dims[0] or
y + label.height > card_dims[1]):
log.log.write("Warning: Text label overflows card boundary: \"%s\"\n" % text)
return None
image = Image.new("RGBA", card_dims, (0,0,0,0))
image.paste(label, (x,y), mask=label)
return image
class ImageLabel:
""" Parsed version of a single image-label object """
def __init__(self, json):
""" Parse out the various settings of a text label and clean them up """
self.name = util.get_default(json, "name", "image") # Only used in complex layout
self.source = util.get_default(json, "source", "images.txt") # Only used in simple layout
self.static = util.get_default(json, "static", None) # Optional, used by layout.
self.x = util.get_default(json, "x", 10, int)
self.y = util.get_default(json, "y", 10, int)
self.width = util.get_default(json, "width", 0, int)
self.height = util.get_default(json, "height", 0, int)
self.x_align = util.get_default(json, "x-align", "left")
self.y_align = util.get_default(json, "y-align", "top")
self.rotation = util.get_default(json, "rotation", 0, int)
def render(self, card_dims, image):
""" Generate a transparent PIL card layer with the image on it """
# Since images aren't wrapped, we simply accept the user's scaling
# settings or (if they are 0), use the image's own dimensions.
# If the image falls outside the card boundaries, we warn but allow it.
w, h = image.size
aspect = w/h
if (self.width == 0 and self.height == 0):
# No scaling, use image as-is
pass
else:
scalew,scaleh = self.width, self.height
if (scalew == 0):
# Proportional scaling to given height
scalew = round(scaleh * aspect)
if (scaleh == 0):
scaleh = round(scalew / aspect)
image = image.resize((scalew, scaleh), Image.ANTIALIAS)
if (self.rotation != 0):
image = util.rotate_image(image, self.rotation)
# Figure out where to place the top-left corner of the label
x,y = util.alignment_to_absolute((self.x, self.y), image.size, self.x_align, self.y_align)
if (x < 0 or y < 0 or
x + image.width > card_dims[0] or
y + image.height > card_dims[1]):
log.log.write("Warning: Image label overflows card boundary")
card = Image.new("RGBA", card_dims, (0,0,0,0))
card.paste(image, (x,y), mask=image)
return card
class ContentGenerator:
""" Loads and caches files (text, JSON and images) from the deck directory """
def __init__(self, directory):
self.directory = directory
self.loaded_texts = {}
self.loaded_json = {}
self.loaded_images = {}
def gen_text_simple(self, filename):
""" Fetch one line from a given text file in the deck directory """
if (filename not in self.loaded_texts):
path = os.path.join(self.directory, filename)
handle = open(path, "r", encoding="utf-8-sig")
if (handle is None):
log.log.write("Unable to open text file %s\n" % path)
return None
self.loaded_texts[filename] = handle
line = self.loaded_texts[filename].readline().rstrip()
if (line == ""):
# End of file
return None
return line
def gen_text_complex(self, filename):
""" Fetch an entire card (named fields) from a JSON file in the deck directory """
if (filename not in self.loaded_json):
path = os.path.join(self.directory, filename)
handle = open(path, "r", encoding="utf-8-sig")
if (handle is None):
log.log.write("Unable to open json file %s\n" % path)
return None
try:
self.loaded_json[filename] = json.load(handle)
except ValueError as e:
sys.stderr.write("JSON error in %s: %s\n" % (path, e))
return None
if (len(self.loaded_json[filename]) == 0):
#End of file
return None
texts = self.loaded_json[filename].pop(0)
return texts
def load_image(self, filename):
if (filename not in self.loaded_images):
path = os.path.join(self.directory, filename)
try:
image = Image.open(path)
image.load()
except:
log.log.write("Unable to load image %s\n" % path)
return None
self.loaded_images[filename] = image
return self.loaded_images[filename].copy()
def gen_image_simple(self, source):
filename = self.gen_text_simple(source)
if (filename is None):
# Ran out of image names in that text file.
# This is analogous to running out of text for a label - no warning.
return None
return self.load_image(filename)
#
# Unit tests
#
class TestTextStuff(unittest.TestCase):
def test_text_init(self):
# This one makes use of all the default values
lab_default = TextLabel({})
self.assertEqual(lab_default.x, 10)
self.assertEqual(lab_default.y, 10)
self.assertEqual(lab_default.width, 40000)
self.assertEqual(lab_default.height, 40000)
self.assertEqual(lab_default.color, "#000000")
self.assertEqual(lab_default.source, "text.txt")
self.assertEqual(lab_default.fontface, "Palatino Linotype")
self.assertEqual(lab_default.fontsize, 10)
self.assertEqual(lab_default.fontweight, sysfont.STYLE_NORMAL)
self.assertEqual(lab_default.spacing, 4)
self.assertEqual(lab_default.justify, "left")
self.assertEqual(lab_default.rotation, 0)
self.assertEqual(lab_default.wordwrap, True)
self.assertEqual(lab_default.y_align, "top")
# This one overrides every possible setting
dic = {
"source": "phrases.txt",
"x": 20,
"y": 30,
"width": 300,
"height": 200,
"color": "#DEADBE",
"font-face": "courier new",
"font-size": 32,
"font-weight": "bold",
"line-spacing": 12,
"justify": "center",
"rotation": 90,
"wordwrap": False,
"x-align": "right",
"y-align": "bottom"
}
lab_override = TextLabel(dic)
self.assertEqual(lab_override.source, dic["source"])
self.assertEqual(lab_override.x, dic["x"])
self.assertEqual(lab_override.y, dic["y"])
self.assertEqual(lab_override.width, dic["width"])
self.assertEqual(lab_override.height, dic["height"])
self.assertEqual(lab_override.color, dic["color"])
self.assertEqual(lab_override.fontface, dic["font-face"])
self.assertEqual(lab_override.fontsize, dic["font-size"])
self.assertEqual(lab_override.fontweight, sysfont.STYLE_BOLD)
self.assertEqual(lab_override.spacing, dic["line-spacing"])
self.assertEqual(lab_override.justify, dic["justify"])
self.assertEqual(lab_override.rotation, dic["rotation"])
self.assertEqual(lab_override.wordwrap, dic["wordwrap"])
self.assertEqual(lab_override.x_align, dic["x-align"])
self.assertEqual(lab_override.y_align, dic["y-align"])
def test_render_text(self):
lab_default = TextLabel({ "color": "#AABBCC" })
layer = lab_default.render((100, 30), "Hello")
self.assertEqual(layer.width, 100)
self.assertEqual(layer.height, 30)
# There are some transparent pixels and some of the text color. Good enough.
self.assertEqual(layer.getextrema(), ((0, 0xAA), (0, 0xBB), (0, 0xCC), (0, 255)))
def test_render_unicode(self):
lab_default = TextLabel({ "color": "#AABBCC" })
layer = lab_default.render((100, 30), u"καλημέρα")
self.assertEqual(layer.width, 100)
self.assertEqual(layer.height, 30)
# There are some transparent pixels and some of the text color. Good enough.
self.assertEqual(layer.getextrema(), ((0, 0xAA), (0, 0xBB), (0, 0xCC), (0, 255)))
def test_antialiasing(self):
lab_small = TextLabel({ "color": "#FFFFFF", "font-size": 16, "font-face": "Liberation Serif" })
lab_large = TextLabel({ "color": "#FFFFFF", "font-size": 32, "font-face": "Liberation Serif" })
text = "This is running text, rendered both natively at 32px and then at twice the size and downscaled."
w,h = (150, 500)
img_small = lab_small.render((w, h), text)
img_large = lab_large.render((2*w, 2*h), text)
img_large = img_large.resize((w, h), Image.ANTIALIAS)
img_compare = Image.new("RGBA", (2*w, h), (0,0,0,0))
img_compare.paste(img_small, (0,0), mask=img_small)
img_compare.paste(img_large, (w,0), mask=img_large)
img_compare.show()
# TODO: Test text wrapping
if __name__ == '__main__':
unittest.main()