VoucherVision / vouchervision /generate_partner_collage.py
phyloforfun's picture
July 18 update
c5e57d6
import requests
from bs4 import BeautifulSoup
from PIL import Image
from io import BytesIO
import time
# Global variables
H = 200
ROWS = 6
PADDING = 30
# Step 1: Fetch the Images from the URL Folder
def fetch_image_urls(url):
response = requests.get(url + '?t=' + str(time.time()))
soup = BeautifulSoup(response.content, 'html.parser')
images = {}
for node in soup.find_all('a'):
href = node.get('href')
if href.endswith(('.png', '.jpg', '.jpeg')):
try:
image_index = int(href.split('__')[0])
images[image_index] = url + '/' + href + '?t=' + str(time.time())
except ValueError:
print(f"Skipping invalid image: {href}")
return images
# Step 2: Resize Images to Height H
def fetch_image(url):
response = requests.get(url)
return Image.open(BytesIO(response.content))
def resize_images(images, target_height):
resized_images = {}
for index, img in images.items():
ratio = target_height / img.height
new_width = int(img.width * ratio)
resized_img = img.resize((new_width, target_height), Image.BICUBIC)
resized_images[index] = resized_img
return resized_images
# Step 3: Create a Collage with Efficient Placement Algorithm
def create_collage(image_urls, collage_path, H, ROWS, PADDING):
images = {index: fetch_image(url) for index, url in image_urls.items()}
resized_images = resize_images(images, H) # Resize to H pixels height
center_image = resized_images.pop(0)
other_images = list(resized_images.items())
# Calculate collage size based on the number of rows
collage_width = 3000 # 16:9 aspect ratio width
collage_height = (H + PADDING) * ROWS + 2 * PADDING # Adjust height based on number of rows, add padding to top and bottom
collage = Image.new('RGB', (collage_width, collage_height), (255, 255, 255))
# Sort images by width and height
sorted_images = sorted(other_images, key=lambda x: x[1].width * x[1].height, reverse=True)
# Create alternate placement list and insert the center image in the middle
alternate_images = []
i, j = 0, len(sorted_images) - 1
halfway_point = (len(sorted_images) + 1) // 2
count = 0
while i <= j:
if count == halfway_point:
alternate_images.append((0, center_image))
if i == j:
alternate_images.append(sorted_images[i])
else:
alternate_images.append(sorted_images[i])
alternate_images.append(sorted_images[j])
i += 1
j -= 1
count += 2
# Calculate number of images per row
images_per_row = len(alternate_images) // ROWS
extra_images = len(alternate_images) % ROWS
# Place images in rows with only padding space between them
def place_images_in_rows(images, collage, max_width, padding, row_height, rows, images_per_row, extra_images):
y = padding
for current_row in range(rows):
row_images_count = images_per_row + (1 if extra_images > 0 else 0)
extra_images -= 1 if extra_images > 0 else 0
row_images = images[:row_images_count]
row_width = sum(img.width for idx, img in row_images) + padding * (row_images_count - 1)
x = (max_width - row_width) // 2
for idx, img in row_images:
collage.paste(img, (x, y))
x += img.width + padding
y += row_height + padding
images = images[row_images_count:]
place_images_in_rows(alternate_images, collage, collage_width, PADDING, H, ROWS, images_per_row, extra_images)
collage.save(collage_path)
# Define the URL folder and other constants
url_folder = 'https://leafmachine.org/partners/'
collage_path = 'img/collage.jpg'
# Fetch, Create, and Update
image_urls = fetch_image_urls(url_folder)
create_collage(image_urls, collage_path, H, ROWS, PADDING)