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.
|
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
|
## Setup
|
||||||
|
|
||||||
The scripts in this repository have the following dependencies:
|
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
|
flatpak override --user --filesystem=~/.config/minecraft org.prismlauncher.PrismLauncher
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# Memegen
|
||||||
|
ln -nsf $EWCONFIG_ROOT/configs/memegen ~/.config/memegen
|
||||||
|
|
||||||
# -- Optional Configs --
|
# -- Optional Configs --
|
||||||
set +x
|
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