Spaces:
Running
on
Zero
Running
on
Zero
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 |