Add a meme generator
This commit is contained in:
parent
cfcdd908b6
commit
2622ac9c43
@ -2,6 +2,8 @@
|
||||
|
||||
This repository stores most of my common config files. It is designed to be deployable to pretty much any system. Assuming ideal conditions, any machine is one `sh ./install-<os>` away from behaving like my personal workstation.
|
||||
|
||||
*I know its called ew**config**, but at this point, its more of a monorepo of scripts*
|
||||
|
||||
## Setup
|
||||
|
||||
The scripts in this repository have the following dependencies:
|
||||
|
BIN
configs/memegen/fonts/impact.ttf
Normal file
BIN
configs/memegen/fonts/impact.ttf
Normal file
Binary file not shown.
17
configs/memegen/templates/bernie-asking/config.json
Normal file
17
configs/memegen/templates/bernie-asking/config.json
Normal file
@ -0,0 +1,17 @@
|
||||
{
|
||||
"font": "impact.ttf",
|
||||
"fill_color": [255, 255, 255],
|
||||
"stroke_color": [0, 0, 0],
|
||||
"stroke_width": 2,
|
||||
"zones": {
|
||||
"bottom": {
|
||||
"horizontal_align": "center",
|
||||
"horizontal_offset": 0,
|
||||
"vertical_align": "bottom",
|
||||
"vertical_offset": -50,
|
||||
"width": "80%",
|
||||
"max_line_height": 50,
|
||||
"line_spacing": 5
|
||||
}
|
||||
}
|
||||
}
|
BIN
configs/memegen/templates/bernie-asking/template.png
Normal file
BIN
configs/memegen/templates/bernie-asking/template.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 414 KiB |
@ -119,6 +119,9 @@ if [ -d ~/.var/app/org.prismlauncher.PrismLauncher ]; then
|
||||
flatpak override --user --filesystem=~/.config/minecraft org.prismlauncher.PrismLauncher
|
||||
fi
|
||||
|
||||
# Memegen
|
||||
ln -nsf $EWCONFIG_ROOT/configs/memegen ~/.config/memegen
|
||||
|
||||
# -- Optional Configs --
|
||||
set +x
|
||||
|
||||
|
226
scripts/memegen
Executable file
226
scripts/memegen
Executable file
@ -0,0 +1,226 @@
|
||||
#! /usr/bin/env python
|
||||
import argparse
|
||||
import sys
|
||||
import logging
|
||||
import json
|
||||
import subprocess
|
||||
from PIL import Image, ImageDraw, ImageFont
|
||||
from typing import List, Dict, Any, Optional
|
||||
from pathlib import Path
|
||||
from dataclasses import dataclass
|
||||
from enum import Enum
|
||||
from datetime import datetime
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
CONFIG_DIR = Path("~/.config/memegen").expanduser()
|
||||
DEFAULT_OUTPUT_DIR = Path("~/Pictures/memes").expanduser()
|
||||
|
||||
|
||||
@dataclass
|
||||
class MemeTemplate:
|
||||
image_path: Path
|
||||
config: Dict[str, Any]
|
||||
|
||||
|
||||
class HorizontalAlignment(Enum):
|
||||
LEFT = "left"
|
||||
CENTER = "center"
|
||||
RIGHT = "right"
|
||||
|
||||
|
||||
class VerticalAlignment(Enum):
|
||||
TOP = "top"
|
||||
CENTER = "center"
|
||||
BOTTOM = "bottom"
|
||||
|
||||
|
||||
def discover_templates() -> List[str]:
|
||||
# Find all directories in the templates directory
|
||||
return [p.name for p in (CONFIG_DIR / "templates").glob("*") if p.is_dir()]
|
||||
|
||||
|
||||
def load_template(name: str) -> MemeTemplate:
|
||||
logger.info(f"Loading template: {name}")
|
||||
|
||||
# Find the template directory
|
||||
template_dir = CONFIG_DIR / "templates" / name
|
||||
if not template_dir.exists():
|
||||
logger.error(f"Template {name} does not exist")
|
||||
sys.exit(1)
|
||||
|
||||
return MemeTemplate(
|
||||
image_path=template_dir / "template.png",
|
||||
config=json.loads((template_dir / "config.json").read_text()),
|
||||
)
|
||||
|
||||
|
||||
def calc_width_from_image(width_str: str, image_width: int) -> int:
|
||||
if width_str.endswith("%"):
|
||||
return int(image_width * int(width_str[:-1]) / 100)
|
||||
else:
|
||||
return int(width_str)
|
||||
|
||||
|
||||
def render_text_on_image(image: Image, text: str, zone: str, config: Dict[str, Any]):
|
||||
# NOTE: This must handle text with newlines
|
||||
# Get the zone config
|
||||
zone_config = config["zones"][zone]
|
||||
horizontal_alignment = HorizontalAlignment(zone_config["horizontal_align"])
|
||||
vertical_alignment = VerticalAlignment(zone_config["vertical_align"])
|
||||
text_width = calc_width_from_image(zone_config["width"], image.width)
|
||||
max_line_height = zone_config["max_line_height"]
|
||||
font_path = CONFIG_DIR / "fonts" / config["font"]
|
||||
|
||||
# Create the font
|
||||
font = None
|
||||
font_size = 1
|
||||
while True:
|
||||
font = ImageFont.truetype(str(font_path), font_size)
|
||||
|
||||
# Split the text into lines
|
||||
lines = text.splitlines()
|
||||
bounding_boxes = []
|
||||
for line in lines:
|
||||
bounding_boxes.append(font.getbbox(line))
|
||||
|
||||
# Calculate the height of the text
|
||||
line_height = max([bbox[3] for bbox in bounding_boxes])
|
||||
total_height = sum(
|
||||
[bbox[3] + zone_config["line_spacing"] for bbox in bounding_boxes]
|
||||
)
|
||||
max_width = max([bbox[2] for bbox in bounding_boxes])
|
||||
|
||||
# If we have a max line height, ensure we don't exceed it
|
||||
if max_line_height and line_height > max_line_height:
|
||||
font_size -= 1
|
||||
break
|
||||
|
||||
# Don't exceed the width
|
||||
if max_width > text_width:
|
||||
font_size -= 1
|
||||
break
|
||||
|
||||
# Increment the font size
|
||||
font_size += 1
|
||||
|
||||
# Determine the starting Y position
|
||||
y = zone_config["vertical_offset"]
|
||||
if vertical_alignment == VerticalAlignment.CENTER:
|
||||
y += (image.height - total_height) / 2
|
||||
elif vertical_alignment == VerticalAlignment.BOTTOM:
|
||||
y += image.height - total_height
|
||||
|
||||
# Render each line onto the image
|
||||
draw = ImageDraw.Draw(image)
|
||||
for line in text.splitlines():
|
||||
# Calculate the x position
|
||||
if horizontal_alignment == HorizontalAlignment.LEFT:
|
||||
x = zone_config["horizontal_offset"]
|
||||
elif horizontal_alignment == HorizontalAlignment.CENTER:
|
||||
x = ((image.width - font.getbbox(line)[2]) / 2) + zone_config[
|
||||
"horizontal_offset"
|
||||
]
|
||||
elif horizontal_alignment == HorizontalAlignment.RIGHT:
|
||||
x = (image.width - font.getbbox(line)[2]) + zone_config["horizontal_offset"]
|
||||
else:
|
||||
raise ValueError(f"Invalid horizontal alignment: {horizontal_alignment}")
|
||||
|
||||
# Render the text
|
||||
draw.text(
|
||||
(x, y),
|
||||
line,
|
||||
fill=tuple(config["fill_color"]),
|
||||
stroke_fill=tuple(config["stroke_color"]),
|
||||
stroke_width=config["stroke_width"],
|
||||
font=font,
|
||||
)
|
||||
|
||||
# Increment the y position
|
||||
y += line_height + zone_config["line_spacing"]
|
||||
|
||||
|
||||
def main() -> int:
|
||||
# Handle program arguments
|
||||
ap = argparse.ArgumentParser(prog="memegen", description="Generates memes")
|
||||
ap.add_argument(
|
||||
"template", help="The template to use", choices=discover_templates()
|
||||
)
|
||||
ap.add_argument("--top-text", help="Top text (if applicable)")
|
||||
ap.add_argument("--bottom-text", help="Bottom text (if applicable)")
|
||||
ap.add_argument(
|
||||
"--keep-case", help="Keep the case of the text", action="store_true"
|
||||
)
|
||||
ap.add_argument("--output", "-o", help="Output file path")
|
||||
ap.add_argument("--show", help="Show the image after creation", action="store_true")
|
||||
ap.add_argument(
|
||||
"-v", "--verbose", help="Enable verbose logging", action="store_true"
|
||||
)
|
||||
args = ap.parse_args()
|
||||
|
||||
# Configure logging
|
||||
logging.basicConfig(
|
||||
level=logging.DEBUG if args.verbose else logging.INFO,
|
||||
format="%(levelname)s: %(message)s",
|
||||
)
|
||||
|
||||
# Load the template
|
||||
template = load_template(args.template)
|
||||
template_supports_top_text = "top" in template.config["zones"]
|
||||
template_supports_bottom_text = "bottom" in template.config["zones"]
|
||||
|
||||
# Ensure we have text
|
||||
if args.top_text and not template_supports_top_text:
|
||||
logger.error(f"Template {args.template} does not support top text")
|
||||
sys.exit(1)
|
||||
if args.bottom_text and not template_supports_bottom_text:
|
||||
logger.error(f"Template {args.template} does not support bottom text")
|
||||
sys.exit(1)
|
||||
if not args.top_text and not args.bottom_text:
|
||||
logger.error("No text provided")
|
||||
if not all([template_supports_top_text, template_supports_bottom_text]):
|
||||
required_text = "top" if template_supports_top_text else "bottom"
|
||||
logger.error(
|
||||
f"Template {args.template} requires the --{required_text}-text argument"
|
||||
)
|
||||
sys.exit(1)
|
||||
|
||||
# Transform the text
|
||||
# fmt:off
|
||||
top_text = args.top_text.upper() if args.top_text and (not args.keep_case) else args.top_text
|
||||
bottom_text = args.bottom_text.upper() if args.bottom_text and (not args.keep_case) else args.bottom_text
|
||||
top_text = top_text.replace("\\n", "\n").replace("\\N", "\n") if top_text else None
|
||||
bottom_text = bottom_text.replace("\\n", "\n").replace("\\N", "\n") if bottom_text else None
|
||||
# fmt: on
|
||||
|
||||
# Load the image
|
||||
image = Image.open(template.image_path)
|
||||
|
||||
# Render the text
|
||||
if top_text:
|
||||
render_text_on_image(image, top_text, "top", template.config)
|
||||
if bottom_text:
|
||||
render_text_on_image(image, bottom_text, "bottom", template.config)
|
||||
|
||||
# Build the output path
|
||||
output_path = (
|
||||
Path(args.output)
|
||||
if args.output
|
||||
else (
|
||||
DEFAULT_OUTPUT_DIR
|
||||
/ f"{args.template}-{datetime.now().strftime('%Y-%m-%d-%H-%M-%S')}.png"
|
||||
)
|
||||
)
|
||||
output_path.parent.mkdir(parents=True, exist_ok=True)
|
||||
|
||||
# Save the image
|
||||
image.save(output_path)
|
||||
|
||||
# Show the image
|
||||
if args.show:
|
||||
subprocess.run(["xdg-open", str(output_path)])
|
||||
|
||||
return 0
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
sys.exit(main())
|
Loading…
x
Reference in New Issue
Block a user