| |
| |
| |
| |
|
|
| import os |
| import random |
| from PIL import Image, ImageDraw, ImageFont, ImageOps, ImageEnhance |
| from ..config import color_mapping |
|
|
| font_dir = os.path.join(os.path.dirname(os.path.dirname(os.path.realpath(__file__))), "fonts") |
| file_list = [f for f in os.listdir(font_dir) if os.path.isfile(os.path.join(font_dir, f)) and f.lower().endswith(".ttf")] |
|
|
|
|
| def align_text(align, img_height, text_height, text_pos_y, margins): |
| if align == "center": |
| text_plot_y = img_height / 2 - text_height / 2 + text_pos_y |
| elif align == "top": |
| text_plot_y = text_pos_y + margins |
| elif align == "bottom": |
| text_plot_y = img_height - text_height + text_pos_y - margins |
| return text_plot_y |
|
|
|
|
| def justify_text(justify, img_width, line_width, margins): |
| if justify == "left": |
| text_plot_x = 0 + margins |
| elif justify == "right": |
| text_plot_x = img_width - line_width - margins |
| elif justify == "center": |
| text_plot_x = img_width/2 - line_width/2 |
| return text_plot_x |
|
|
|
|
| def get_text_size(draw, text, font): |
| bbox = draw.textbbox((0, 0), text, font=font) |
|
|
| |
| text_width = bbox[2] - bbox[0] |
| text_height = bbox[3] - bbox[1] |
| return text_width, text_height |
|
|
|
|
| def draw_masked_text(text_mask, text, |
| font_name, font_size, |
| margins, line_spacing, |
| position_x, position_y, |
| align, justify, |
| rotation_angle, rotation_options): |
| |
| |
| draw = ImageDraw.Draw(text_mask) |
|
|
| |
| font_folder = "fonts" |
| font_file = os.path.join(font_folder, font_name) |
| resolved_font_path = os.path.join(os.path.dirname(os.path.dirname(os.path.realpath(__file__))), font_file) |
| font = ImageFont.truetype(str(resolved_font_path), size=font_size) |
|
|
| |
| text_lines = text.split('\n') |
|
|
| |
| max_text_width = 0 |
| max_text_height = 0 |
|
|
| for line in text_lines: |
| |
| line_width, line_height = get_text_size(draw, line, font) |
| |
| line_height = line_height + line_spacing |
| max_text_width = max(max_text_width, line_width) |
| max_text_height = max(max_text_height, line_height) |
| |
| |
| image_width, image_height = text_mask.size |
| image_center_x = image_width / 2 |
| image_center_y = image_height / 2 |
|
|
| text_pos_y = position_y |
| sum_text_plot_y = 0 |
| text_height = max_text_height * len(text_lines) |
|
|
| for line in text_lines: |
| |
| line_width, _ = get_text_size(draw, line, font) |
| |
| |
| text_plot_x = position_x + justify_text(justify, image_width, line_width, margins) |
| text_plot_y = align_text(align, image_height, text_height, text_pos_y, margins) |
| |
| |
| draw.text((text_plot_x, text_plot_y), line, fill=255, font=font) |
| |
| text_pos_y += max_text_height |
| sum_text_plot_y += text_plot_y |
|
|
| |
| text_center_x = text_plot_x + max_text_width / 2 |
| text_center_y = sum_text_plot_y / len(text_lines) |
|
|
| if rotation_options == "text center": |
| rotated_text_mask = text_mask.rotate(rotation_angle, center=(text_center_x, text_center_y)) |
| elif rotation_options == "image center": |
| rotated_text_mask = text_mask.rotate(rotation_angle, center=(image_center_x, image_center_y)) |
| |
| return rotated_text_mask |
|
|
| def draw_text_on_image(draw, y_position, bar_width, bar_height, text, font, text_color, font_outline): |
|
|
| |
| text_width, text_height = get_text_size(draw, text, font) |
| |
| if font_outline == "thin": |
| outline_thickness = text_height // 40 |
| elif font_outline == "thick": |
| outline_thickness = text_height // 20 |
| elif font_outline == "extra thick": |
| outline_thickness = text_height // 10 |
| |
| outline_color = (0, 0, 0) |
| |
| text_lines = text.split('\n') |
| |
| if len(text_lines) == 1: |
| x = (bar_width - text_width) // 2 |
| y = y_position + (bar_height - text_height) // 2 - (bar_height * 0.10) |
| if font_outline == "none": |
| draw.text((x, y), text, fill=text_color, font=font) |
| else: |
| draw.text((x, y), text, fill=text_color, font=font, stroke_width=outline_thickness, stroke_fill='black') |
| elif len(text_lines) > 1: |
| |
| text_width, text_height = get_text_size(draw, text_lines[0], font) |
| |
| x = (bar_width - text_width) // 2 |
| y = y_position + (bar_height - text_height * 2) // 2 - (bar_height * 0.15) |
| if font_outline == "none": |
| draw.text((x, y), text_lines[0], fill=text_color, font=font) |
| else: |
| draw.text((x, y), text_lines[0], fill=text_color, font=font, stroke_width=outline_thickness, stroke_fill='black') |
|
|
| |
| text_width, text_height = get_text_size(draw, text_lines[1], font) |
| |
| x = (bar_width - text_width) // 2 |
| y = y_position + (bar_height - text_height * 2) // 2 + text_height - (bar_height * 0.00) |
| if font_outline == "none": |
| draw.text((x, y), text_lines[1], fill=text_color, font=font) |
| else: |
| draw.text((x, y), text_lines[1], fill=text_color, font=font, stroke_width=outline_thickness, stroke_fill='black') |
|
|
|
|
| def get_font_size(draw, text, max_width, max_height, font_path, max_font_size): |
|
|
| |
| max_width = max_width * 0.9 |
| |
| |
| font_size = max_font_size |
| font = ImageFont.truetype(str(font_path), size=font_size) |
|
|
| |
| text_lines = text.split('\n')[:2] |
| |
| if len(text_lines) == 2: |
| font_size = min(max_height//2, max_font_size) |
| font = ImageFont.truetype(str(font_path), size=font_size) |
| |
| |
| max_text_width = 0 |
| longest_line = text_lines[0] |
| for line in text_lines: |
| |
| line_width, line_height = get_text_size(draw, line, font) |
| |
| if line_width > max_text_width: |
| longest_line = line |
| max_text_width = max(max_text_width, line_width) |
| |
| |
| text_width, text_height = get_text_size(draw, text, font) |
| |
| |
| while max_text_width > max_width or text_height > 0.88 * max_height / len(text_lines): |
| font_size -= 1 |
| font = ImageFont.truetype(str(font_path), size=font_size) |
| max_text_width, text_height = get_text_size(draw, longest_line, font) |
| |
| return font |
|
|
|
|
| def hex_to_rgb(hex_color): |
| hex_color = hex_color.lstrip('#') |
| r = int(hex_color[0:2], 16) |
| g = int(hex_color[2:4], 16) |
| b = int(hex_color[4:6], 16) |
| return (r, g, b) |
|
|
|
|
| def text_panel(image_width, image_height, text, |
| font_name, font_size, font_color, |
| font_outline_thickness, font_outline_color, |
| background_color, |
| margins, line_spacing, |
| position_x, position_y, |
| align, justify, |
| rotation_angle, rotation_options): |
|
|
| """ |
| Create an image with text overlaid on a background. |
| |
| Returns: |
| PIL.Image.Image: Image with text overlaid on the background. |
| """ |
| |
| |
| size = (image_width, image_height) |
| panel = Image.new('RGB', size, background_color) |
|
|
| |
| image_out = draw_text(panel, text, |
| font_name, font_size, font_color, |
| font_outline_thickness, font_outline_color, |
| background_color, |
| margins, line_spacing, |
| position_x, position_y, |
| align, justify, |
| rotation_angle, rotation_options) |
| |
| return image_out |
|
|
|
|
| def draw_text(panel, text, |
| font_name, font_size, font_color, |
| font_outline_thickness, font_outline_color, |
| bg_color, |
| margins, line_spacing, |
| position_x, position_y, |
| align, justify, |
| rotation_angle, rotation_options): |
| |
| |
| draw = ImageDraw.Draw(panel) |
|
|
| |
| font_folder = "fonts" |
| font_file = os.path.join(font_folder, font_name) |
| resolved_font_path = os.path.join(os.path.dirname(os.path.dirname(os.path.realpath(__file__))), font_file) |
| font = ImageFont.truetype(str(resolved_font_path), size=font_size) |
|
|
| |
| text_lines = text.split('\n') |
|
|
| |
| max_text_width = 0 |
| max_text_height = 0 |
|
|
| for line in text_lines: |
| |
| line_width, line_height = get_text_size(draw, line, font) |
| |
| line_height = line_height + line_spacing |
| max_text_width = max(max_text_width, line_width) |
| max_text_height = max(max_text_height, line_height) |
| |
| |
| image_center_x = panel.width / 2 |
| image_center_y = panel.height / 2 |
|
|
| text_pos_y = position_y |
| sum_text_plot_y = 0 |
| text_height = max_text_height * len(text_lines) |
|
|
| for line in text_lines: |
| |
| line_width, line_height = get_text_size(draw, line, font) |
| |
| |
| text_plot_x = position_x + justify_text(justify, panel.width, line_width, margins) |
| text_plot_y = align_text(align, panel.height, text_height, text_pos_y, margins) |
| |
| |
| draw.text((text_plot_x, text_plot_y), line, fill=font_color, font=font, stroke_width=font_outline_thickness, stroke_fill=font_outline_color) |
|
|
| text_pos_y += max_text_height |
| sum_text_plot_y += text_plot_y |
|
|
| text_center_x = text_plot_x + max_text_width / 2 |
| text_center_y = sum_text_plot_y / len(text_lines) |
|
|
| if rotation_options == "text center": |
| rotated_panel = panel.rotate(rotation_angle, center=(text_center_x, text_center_y), resample=Image.BILINEAR) |
| elif rotation_options == "image center": |
| rotated_panel = panel.rotate(rotation_angle, center=(image_center_x, image_center_y), resample=Image.BILINEAR) |
| |
| return rotated_panel |
|
|
|
|
| def combine_images(images, layout_direction='horizontal'): |
| """ |
| Combine a list of PIL Image objects either horizontally or vertically. |
| |
| Args: |
| images (list of PIL.Image.Image): List of PIL Image objects to combine. |
| layout_direction (str): 'horizontal' for horizontal layout, 'vertical' for vertical layout. |
| |
| Returns: |
| PIL.Image.Image: Combined image. |
| """ |
|
|
| if layout_direction == 'horizontal': |
| combined_width = sum(image.width for image in images) |
| combined_height = max(image.height for image in images) |
| else: |
| combined_width = max(image.width for image in images) |
| combined_height = sum(image.height for image in images) |
|
|
| combined_image = Image.new('RGB', (combined_width, combined_height)) |
|
|
| x_offset = 0 |
| y_offset = 0 |
| for image in images: |
| combined_image.paste(image, (x_offset, y_offset)) |
| if layout_direction == 'horizontal': |
| x_offset += image.width |
| else: |
| y_offset += image.height |
|
|
| return combined_image |
|
|
|
|
| def apply_outline_and_border(images, outline_thickness, outline_color, border_thickness, border_color): |
| for i, image in enumerate(images): |
| |
| if outline_thickness > 0: |
| image = ImageOps.expand(image, outline_thickness, fill=outline_color) |
| |
| |
| if border_thickness > 0: |
| image = ImageOps.expand(image, border_thickness, fill=border_color) |
|
|
| images[i] = image |
| |
| return images |
|
|
|
|
| def get_color_values(color, color_hex, color_mapping): |
| |
| |
|
|
| if color == "custom": |
| color_rgb = hex_to_rgb(color_hex) |
| else: |
| color_rgb = color_mapping.get(color, (0, 0, 0)) |
|
|
| return color_rgb |
|
|
|
|
| def hex_to_rgb(hex_color): |
| hex_color = hex_color.lstrip('#') |
| r = int(hex_color[0:2], 16) |
| g = int(hex_color[2:4], 16) |
| b = int(hex_color[4:6], 16) |
| return (r, g, b) |
|
|
|
|
| def crop_and_resize_image(image, target_width, target_height): |
| width, height = image.size |
| aspect_ratio = width / height |
| target_aspect_ratio = target_width / target_height |
|
|
| if aspect_ratio > target_aspect_ratio: |
| |
| crop_width = int(height * target_aspect_ratio) |
| crop_height = height |
| left = (width - crop_width) // 2 |
| top = 0 |
| else: |
| |
| crop_height = int(width / target_aspect_ratio) |
| crop_width = width |
| left = 0 |
| top = (height - crop_height) // 2 |
| |
| |
| cropped_image = image.crop((left, top, left + crop_width, top + crop_height)) |
| |
| return cropped_image |
|
|
|
|
| def create_and_paste_panel(page, border_thickness, outline_thickness, |
| panel_width, panel_height, page_width, |
| panel_color, bg_color, outline_color, |
| images, i, j, k, len_images, reading_direction): |
| panel = Image.new("RGB", (panel_width, panel_height), panel_color) |
| if k < len_images: |
| img = images[k] |
| image = crop_and_resize_image(img, panel_width, panel_height) |
| image.thumbnail((panel_width, panel_height), Image.Resampling.LANCZOS) |
| panel.paste(image, (0, 0)) |
| panel = ImageOps.expand(panel, border=outline_thickness, fill=outline_color) |
| panel = ImageOps.expand(panel, border=border_thickness, fill=bg_color) |
| new_panel_width, new_panel_height = panel.size |
| if reading_direction == "right to left": |
| page.paste(panel, (page_width - (j + 1) * new_panel_width, i * new_panel_height)) |
| else: |
| page.paste(panel, (j * new_panel_width, i * new_panel_height)) |
|
|
|
|
| def reduce_opacity(img, opacity): |
| """Returns an image with reduced opacity.""" |
| assert opacity >= 0 and opacity <= 1 |
| if img.mode != 'RGBA': |
| img = img.convert('RGBA') |
| else: |
| img = img.copy() |
| alpha = img.split()[3] |
| alpha = ImageEnhance.Brightness(alpha).enhance(opacity) |
| img.putalpha(alpha) |
| return img |
| |
|
|
| def random_hex_color(): |
| |
| r = random.randint(0, 255) |
| g = random.randint(0, 255) |
| b = random.randint(0, 255) |
|
|
| |
| hex_color = "#{:02x}{:02x}{:02x}".format(r, g, b) |
|
|
| return hex_color |
|
|
|
|
| def random_rgb(): |
| |
| r = random.randint(0, 255) |
| g = random.randint(0, 255) |
| b = random.randint(0, 255) |
|
|
| |
| rgb_string = "{},{},{}".format(r, g, b) |
|
|
| return rgb_string |