contentdb/app/blueprints/thumbnails/__init__.py

86 lines
2.5 KiB
Python
Raw Permalink Normal View History

2020-07-12 17:34:25 +02:00
# ContentDB
2021-01-30 17:59:42 +01:00
# Copyright (C) 2018-21 rubenwardy
2018-05-29 17:19:17 +02:00
#
# This program is free software: you can redistribute it and/or modify
2021-01-30 17:59:42 +01:00
# it under the terms of the GNU Affero General Public License as published by
2018-05-29 17:19:17 +02:00
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
2021-01-30 17:59:42 +01:00
# GNU Affero General Public License for more details.
2018-05-29 17:19:17 +02:00
#
2021-01-30 17:59:42 +01:00
# You should have received a copy of the GNU Affero General Public License
2018-05-29 17:19:17 +02:00
# along with this program. If not, see <https://www.gnu.org/licenses/>.
from flask import abort, send_file, Blueprint, current_app
bp = Blueprint("thumbnails", __name__)
2018-05-29 17:19:17 +02:00
2018-07-28 17:03:48 +02:00
import os
2018-05-29 17:19:17 +02:00
from PIL import Image
2018-12-21 22:22:15 +01:00
ALLOWED_RESOLUTIONS=[(100,67), (270,180), (350,233)]
2018-05-29 17:19:17 +02:00
def mkdir(path):
assert path != "" and path is not None
2020-07-10 21:50:25 +02:00
try:
if not os.path.isdir(path):
os.mkdir(path)
except FileExistsError:
pass
2018-05-29 17:19:17 +02:00
2018-05-29 17:56:35 +02:00
2018-07-28 17:03:48 +02:00
def resize_and_crop(img_path, modified_path, size):
try:
img = Image.open(img_path)
except FileNotFoundError:
abort(404)
2018-07-28 17:03:48 +02:00
# Get current and desired ratio for the images
img_ratio = img.size[0] / float(img.size[1])
ratio = size[0] / float(size[1])
# Is more portrait than target, scale and crop
if ratio > img_ratio:
img = img.resize((int(size[0]), int(size[0] * img.size[1] / img.size[0])),
Image.BICUBIC)
box = (0, (img.size[1] - size[1]) / 2, img.size[0], (img.size[1] + size[1]) / 2)
img = img.crop(box)
# Is more landscape than target, scale and crop
elif ratio < img_ratio:
img = img.resize((int(size[1] * img.size[0] / img.size[1]), int(size[1])),
Image.BICUBIC)
box = ((img.size[0] - size[0]) / 2, 0, (img.size[0] + size[0]) / 2, img.size[1])
img = img.crop(box)
# Is exactly the same ratio as target
else:
img = img.resize(size, Image.BICUBIC)
img.save(modified_path)
2018-12-21 22:22:15 +01:00
@bp.route("/thumbnails/<int:level>/<img>")
2018-12-21 22:22:15 +01:00
def make_thumbnail(img, level):
if level > len(ALLOWED_RESOLUTIONS) or level <= 0:
2018-05-29 17:19:17 +02:00
abort(403)
2018-12-21 22:22:15 +01:00
w, h = ALLOWED_RESOLUTIONS[level - 1]
2020-01-18 02:20:32 +01:00
upload_dir = current_app.config["UPLOAD_DIR"]
thumbnail_dir = current_app.config["THUMBNAIL_DIR"]
mkdir(thumbnail_dir)
2018-05-29 17:19:17 +02:00
2020-01-18 02:20:32 +01:00
output_dir = os.path.join(thumbnail_dir, str(level))
mkdir(output_dir)
2018-05-29 17:19:17 +02:00
2020-01-18 02:20:32 +01:00
cache_filepath = os.path.join(output_dir, img)
source_filepath = os.path.join(upload_dir, img)
resize_and_crop(source_filepath, cache_filepath, (w, h))
2018-05-29 17:19:17 +02:00
return send_file(cache_filepath)