July 2020 (#2)
* wip * Forked lektor-thumbnail-generator * Post Co-authored-by: Felipe M <fmartingr@Going-Merry.local>
This commit is contained in:
parent
b7799aac92
commit
9709127df1
8 changed files with 258 additions and 1 deletions
|
@ -0,0 +1,120 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
import shutil
|
||||
|
||||
from lektor.build_programs import AttachmentBuildProgram, buildprogram
|
||||
from lektor.context import get_ctx
|
||||
from lektor.db import Image
|
||||
from lektor.imagetools import (computed_height, find_imagemagick,
|
||||
get_image_info, get_quality)
|
||||
from lektor.pluginsystem import Plugin
|
||||
from lektor.reporter import reporter
|
||||
from lektor.utils import portable_popen
|
||||
from werkzeug.utils import cached_property
|
||||
|
||||
|
||||
# We override process_image here because Lektor does not support adding extra
|
||||
# parameters yet, but it will soon, and this can be removed when it does.
|
||||
def process_image(
|
||||
ctx,
|
||||
source_image,
|
||||
dst_filename,
|
||||
width=None,
|
||||
height=None,
|
||||
mode=None,
|
||||
quality=None,
|
||||
extra_params=None,
|
||||
):
|
||||
"""Build image from source image, optionally compressing and resizing.
|
||||
"source_image" is the absolute path of the source in the content directory,
|
||||
"dst_filename" is the absolute path of the target in the output directory.
|
||||
"""
|
||||
if width is None and height is None:
|
||||
raise ValueError("Must specify at least one of width or height.")
|
||||
|
||||
im = find_imagemagick(ctx.build_state.config["IMAGEMAGICK_EXECUTABLE"])
|
||||
|
||||
if quality is None:
|
||||
quality = get_quality(source_image)
|
||||
|
||||
resize_key = ""
|
||||
if width is not None:
|
||||
resize_key += str(width)
|
||||
if height is not None:
|
||||
resize_key += "x" + str(height)
|
||||
|
||||
cmdline = [im, source_image, "-auto-orient"]
|
||||
cmdline += ["-resize", resize_key]
|
||||
|
||||
if extra_params:
|
||||
cmdline.extend(extra_params)
|
||||
|
||||
cmdline += ["-quality", str(quality), dst_filename]
|
||||
|
||||
reporter.report_debug_info("imagemagick cmd line", cmdline)
|
||||
portable_popen(cmdline).wait()
|
||||
|
||||
|
||||
@buildprogram(Image)
|
||||
class ResizedImageBuildProgram(AttachmentBuildProgram):
|
||||
def build_artifact(self, artifact):
|
||||
ctx = get_ctx()
|
||||
plugin = ctx.env.plugins["thumbnail-generator"]
|
||||
config = plugin.config
|
||||
|
||||
artifact.ensure_dir()
|
||||
AttachmentBuildProgram.build_artifact(self, artifact)
|
||||
|
||||
if not config:
|
||||
return
|
||||
|
||||
source_img = artifact.source_obj.attachment_filename
|
||||
|
||||
with open(source_img, "rb") as f:
|
||||
_, w, h = get_image_info(f)
|
||||
|
||||
# For every section in the config, we need to generate one image.
|
||||
for item, conf in config.items():
|
||||
width = int(conf["max_width"])
|
||||
height = int(conf.get("max_height", "0"))
|
||||
quality = int(conf.get("quality", "100"))
|
||||
|
||||
if not height:
|
||||
height = computed_height(source_img, width, w, h)
|
||||
|
||||
df = artifact.source_obj.url_path
|
||||
ext_pos = df.rfind(".")
|
||||
dst_filename = "%s-%s.%s" % (df[:ext_pos], item, df[ext_pos + 1 :])
|
||||
|
||||
def closure(dst_filename, source_img, width, height, resize_image=True):
|
||||
# We need this closure, otherwise variables get updated and this
|
||||
# doesn't work at all.
|
||||
@ctx.sub_artifact(artifact_name=dst_filename, sources=[source_img])
|
||||
def build_thumbnail_artifact(artifact):
|
||||
artifact.ensure_dir()
|
||||
if not resize_image:
|
||||
shutil.copy2(source_img, artifact.dst_filename)
|
||||
else:
|
||||
process_image(
|
||||
ctx,
|
||||
source_img,
|
||||
artifact.dst_filename,
|
||||
width,
|
||||
height,
|
||||
quality=100,
|
||||
)
|
||||
|
||||
# If the image is larger than the max_width, resize it, otherwise
|
||||
# just copy it.
|
||||
resize_image = w > width or h > height
|
||||
closure(dst_filename, source_img, width, height, resize_image)
|
||||
|
||||
|
||||
class ThumbnailGeneratorPlugin(Plugin):
|
||||
name = u"thumbnail-generator"
|
||||
description = u"A configurable way to generate thumbnails."
|
||||
image_exts = ["png", "jpg", "jpeg", "gif"]
|
||||
|
||||
@cached_property
|
||||
def config(self):
|
||||
conf = self.get_config()
|
||||
return {section: conf.section_as_dict(section) for section in conf.sections()}
|
17
packages/lektor-thumbnail-generator/setup.py
Normal file
17
packages/lektor-thumbnail-generator/setup.py
Normal file
|
@ -0,0 +1,17 @@
|
|||
from setuptools import setup
|
||||
|
||||
setup(
|
||||
name='lektor-thumbnail-generator',
|
||||
version='0.4.0',
|
||||
author=u'Stavros Korokithakis,,,',
|
||||
author_email='hi@stavros.io',
|
||||
url='https://github.com/skorokithakis/lektor-thumbnail-generator/',
|
||||
description="This plugin automatically generates thumbnails for any images in your Lektor content. The difference between this plugin and the `thumbnail` filter is that this is geared towards content, i.e. you don't need to have any references to the images in your templates at all.",
|
||||
license='MIT',
|
||||
py_modules=['lektor_thumbnail_generator'],
|
||||
entry_points={
|
||||
'lektor.plugins': [
|
||||
'thumbnail-generator = lektor_thumbnail_generator:ThumbnailGeneratorPlugin',
|
||||
]
|
||||
}
|
||||
)
|
Loading…
Add table
Add a link
Reference in a new issue