File size: 2,188 Bytes
d36e4bb
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
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
import textwrap
from typing import List, Tuple, Optional

import numpy as np
from PIL import Image, ImageDraw, ImageFont

LINE_WIDTH = 20


def add_text_to_image(image: np.ndarray, text: str, text_color: Tuple[int, int, int] = (0, 0, 0),
                      min_lines: Optional[int] = None, add_below: bool = True):
    import textwrap
    lines = textwrap.wrap(text, width=LINE_WIDTH)
    if min_lines is not None and len(lines) < min_lines:
        if add_below:
            lines += [''] * (min_lines - len(lines))
        else:
            lines = [''] * (min_lines - len(lines)) + lines
    h, w, c = image.shape
    offset = int(h * .12)
    img = np.ones((h + offset * len(lines), w, c), dtype=np.uint8) * 255
    font_size = int(offset * .8)

    try:
        font = ImageFont.truetype("assets/OpenSans-Regular.ttf", font_size)
        textsize = font.getbbox(text)
        y_offset = (offset - textsize[3]) // 2
    except:
        font = ImageFont.load_default()
        y_offset = offset // 2

    if add_below:
        img[:h] = image
    else:
        img[-h:] = image
    img = Image.fromarray(img)
    draw = ImageDraw.Draw(img)
    for i, line in enumerate(lines):
        line_size = font.getbbox(line)
        text_x = (w - line_size[2]) // 2
        if add_below:
            draw.text((text_x, h + y_offset + offset * i), line, font=font, fill=text_color)
        else:
            draw.text((text_x, 0 + y_offset + offset * i), line, font=font, fill=text_color)
    return np.array(img)


def create_table_plot(titles: List[str], images: List[Image.Image], captions: List[str]) -> Image.Image:
    title_max_lines = np.max([len(textwrap.wrap(text, width=LINE_WIDTH)) for text in titles])
    caption_max_lines = np.max([len(textwrap.wrap(text, width=LINE_WIDTH)) for text in captions])
    out_images = []
    for i in range(len(images)):
        im = np.array(images[i])
        im = add_text_to_image(im, titles[i], add_below=False, min_lines=title_max_lines)
        im = add_text_to_image(im, captions[i], add_below=True, min_lines=caption_max_lines)
        out_images.append(im)
    image = Image.fromarray(np.concatenate(out_images, axis=1))
    return image