fotolab 0.30.0__py3-none-any.whl → 0.31.1__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- fotolab/__init__.py +7 -8
- fotolab/subcommands/animate.py +13 -7
- fotolab/subcommands/auto.py +1 -0
- fotolab/subcommands/border.py +8 -8
- fotolab/subcommands/info.py +16 -22
- fotolab/subcommands/watermark.py +41 -10
- {fotolab-0.30.0.dist-info → fotolab-0.31.1.dist-info}/METADATA +7 -2
- {fotolab-0.30.0.dist-info → fotolab-0.31.1.dist-info}/RECORD +11 -11
- {fotolab-0.30.0.dist-info → fotolab-0.31.1.dist-info}/LICENSE.md +0 -0
- {fotolab-0.30.0.dist-info → fotolab-0.31.1.dist-info}/WHEEL +0 -0
- {fotolab-0.30.0.dist-info → fotolab-0.31.1.dist-info}/entry_points.txt +0 -0
fotolab/__init__.py
CHANGED
@@ -24,7 +24,7 @@ from pathlib import Path
|
|
24
24
|
|
25
25
|
from PIL import Image
|
26
26
|
|
27
|
-
__version__ = "0.
|
27
|
+
__version__ = "0.31.1"
|
28
28
|
|
29
29
|
log = logging.getLogger(__name__)
|
30
30
|
|
@@ -59,15 +59,14 @@ def save_gif_image(
|
|
59
59
|
_open_image(new_filename)
|
60
60
|
|
61
61
|
|
62
|
-
def save_image(
|
62
|
+
def save_image(
|
63
|
+
args: argparse.Namespace,
|
64
|
+
new_image: Image.Image,
|
65
|
+
output_filename: str,
|
66
|
+
subcommand: str,
|
67
|
+
) -> None:
|
63
68
|
"""Save image after image operation.
|
64
69
|
|
65
|
-
Args:
|
66
|
-
args (argparse.Namespace): Config from command line arguments
|
67
|
-
new_image(PIL.Image.Image): Modified image
|
68
|
-
output_filename(str): Save filename image
|
69
|
-
subcommand(str): Subcommand used to call this function
|
70
|
-
|
71
70
|
Returns:
|
72
71
|
None
|
73
72
|
"""
|
fotolab/subcommands/animate.py
CHANGED
@@ -73,7 +73,10 @@ def build_subparser(subparsers) -> None:
|
|
73
73
|
dest="duration",
|
74
74
|
type=_validate_duration,
|
75
75
|
default=2500,
|
76
|
-
help=
|
76
|
+
help=(
|
77
|
+
"set the duration in milliseconds "
|
78
|
+
"(must be a positive integer, default: '%(default)s')"
|
79
|
+
),
|
77
80
|
metavar="DURATION",
|
78
81
|
)
|
79
82
|
|
@@ -120,7 +123,10 @@ def build_subparser(subparsers) -> None:
|
|
120
123
|
type=int,
|
121
124
|
default=4,
|
122
125
|
choices=range(0, 7),
|
123
|
-
help=
|
126
|
+
help=(
|
127
|
+
"set WEBP encoding method "
|
128
|
+
"(0=fast, 6=slow/best, default: '%(default)s')"
|
129
|
+
),
|
124
130
|
metavar="METHOD",
|
125
131
|
)
|
126
132
|
|
@@ -170,17 +176,17 @@ def run(args: argparse.Namespace) -> None:
|
|
170
176
|
"save_all": True,
|
171
177
|
"duration": args.duration,
|
172
178
|
"loop": args.loop,
|
173
|
-
"optimize": True,
|
179
|
+
"optimize": True,
|
174
180
|
}
|
175
181
|
|
182
|
+
# Pillow's WEBP save doesn't use a general 'optimize' like GIF.
|
183
|
+
# Specific WEBP params like 'method' and 'quality' control this. We
|
184
|
+
# can remove 'optimize' if it causes issues or is ignored for WEBP.
|
185
|
+
# For now, let's keep it, Pillow might handle it or ignore it.
|
176
186
|
if args.format == "webp":
|
177
187
|
save_kwargs["quality"] = args.webp_quality
|
178
188
|
save_kwargs["lossless"] = args.webp_lossless
|
179
189
|
save_kwargs["method"] = args.webp_method
|
180
|
-
# Pillow's WEBP save doesn't use a general 'optimize' like GIF.
|
181
|
-
# Specific WEBP params like 'method' and 'quality' control this.
|
182
|
-
# We can remove 'optimize' if it causes issues or is ignored for WEBP.
|
183
|
-
# For now, let's keep it, Pillow might handle it or ignore it.
|
184
190
|
|
185
191
|
main_frame.save(new_filename, **save_kwargs)
|
186
192
|
finally:
|
fotolab/subcommands/auto.py
CHANGED
fotolab/subcommands/border.py
CHANGED
@@ -17,7 +17,7 @@
|
|
17
17
|
|
18
18
|
import argparse
|
19
19
|
import logging
|
20
|
-
from typing import Tuple
|
20
|
+
from typing import Tuple
|
21
21
|
|
22
22
|
from PIL import Image, ImageColor, ImageOps
|
23
23
|
|
@@ -26,7 +26,7 @@ from fotolab import save_image
|
|
26
26
|
log = logging.getLogger(__name__)
|
27
27
|
|
28
28
|
|
29
|
-
def build_subparser(subparsers) -> None:
|
29
|
+
def build_subparser(subparsers: argparse._SubParsersAction) -> None:
|
30
30
|
"""Build the subparser."""
|
31
31
|
border_parser = subparsers.add_parser("border", help="add border to image")
|
32
32
|
|
@@ -150,17 +150,16 @@ def run(args: argparse.Namespace) -> None:
|
|
150
150
|
|
151
151
|
def get_border(
|
152
152
|
args: argparse.Namespace,
|
153
|
-
) ->
|
153
|
+
) -> Tuple[int, int, int, int]:
|
154
154
|
"""Calculate the border dimensions.
|
155
155
|
|
156
156
|
Args:
|
157
157
|
args (argparse.Namespace): Command line arguments
|
158
158
|
|
159
159
|
Returns:
|
160
|
-
|
161
|
-
If individual widths are specified,
|
162
|
-
|
163
|
-
sides.
|
160
|
+
Tuple[int, int, int, int]: Border dimensions in pixels as (left, top,
|
161
|
+
right, bottom) widths. If individual widths are not specified,
|
162
|
+
a uniform width is returned for all sides.
|
164
163
|
"""
|
165
164
|
if any(
|
166
165
|
[
|
@@ -176,4 +175,5 @@ def get_border(
|
|
176
175
|
args.width_right,
|
177
176
|
args.width_bottom,
|
178
177
|
)
|
179
|
-
|
178
|
+
# If no individual widths are specified, use the general width for all sides
|
179
|
+
return (args.width, args.width, args.width, args.width)
|
fotolab/subcommands/info.py
CHANGED
@@ -81,31 +81,25 @@ def run(args: argparse.Namespace) -> None:
|
|
81
81
|
return
|
82
82
|
|
83
83
|
output_info = []
|
84
|
-
specific_info_requested = False
|
85
84
|
|
86
|
-
|
87
|
-
|
88
|
-
output_info.append(camera_metadata(exif_tags))
|
85
|
+
if args.camera:
|
86
|
+
output_info.append(get_formatted_camera_info(exif_tags))
|
89
87
|
|
90
|
-
|
91
|
-
|
92
|
-
output_info.append(datetime(exif_tags))
|
88
|
+
if args.datetime:
|
89
|
+
output_info.append(get_formatted_datetime_info(exif_tags))
|
93
90
|
|
94
|
-
|
95
|
-
|
96
|
-
|
97
|
-
|
98
|
-
|
99
|
-
|
100
|
-
|
91
|
+
if output_info: # Check if any specific info was added
|
92
|
+
print("\n".join(output_info))
|
93
|
+
else:
|
94
|
+
# Print all tags if no specific info was requested
|
95
|
+
tag_name_width = max(map(len, exif_tags))
|
96
|
+
for tag_name, tag_value in exif_tags.items():
|
97
|
+
print(f"{tag_name:<{tag_name_width}}: {tag_value}")
|
101
98
|
|
102
99
|
|
103
100
|
def extract_exif_tags(image: Image.Image, sort: bool = False) -> dict:
|
104
101
|
"""Extract Exif metadata from image."""
|
105
|
-
|
106
|
-
exif = image._getexif()
|
107
|
-
except AttributeError:
|
108
|
-
exif = None
|
102
|
+
exif = image.getexif()
|
109
103
|
|
110
104
|
log.debug(exif)
|
111
105
|
|
@@ -122,13 +116,13 @@ def extract_exif_tags(image: Image.Image, sort: bool = False) -> dict:
|
|
122
116
|
return filtered_info
|
123
117
|
|
124
118
|
|
125
|
-
def
|
126
|
-
"""Extract datetime metadata."""
|
119
|
+
def get_formatted_datetime_info(exif_tags: dict):
|
120
|
+
"""Extract and format datetime metadata."""
|
127
121
|
return exif_tags.get("DateTime", "Not available")
|
128
122
|
|
129
123
|
|
130
|
-
def
|
131
|
-
"""Extract camera and model metadata."""
|
124
|
+
def get_formatted_camera_info(exif_tags: dict):
|
125
|
+
"""Extract and format camera make and model metadata."""
|
132
126
|
make = exif_tags.get("Make", "")
|
133
127
|
model = exif_tags.get("Model", "")
|
134
128
|
metadata = f"{make} {model}"
|
fotolab/subcommands/watermark.py
CHANGED
@@ -24,7 +24,7 @@ from typing import Tuple
|
|
24
24
|
from PIL import Image, ImageColor, ImageDraw, ImageFont, ImageSequence
|
25
25
|
|
26
26
|
from fotolab import save_image
|
27
|
-
from fotolab.subcommands.info import
|
27
|
+
from fotolab.subcommands.info import get_formatted_camera_info
|
28
28
|
|
29
29
|
log: logging.Logger = logging.getLogger(__name__)
|
30
30
|
|
@@ -129,6 +129,21 @@ def build_subparser(subparsers: argparse._SubParsersAction) -> None:
|
|
129
129
|
metavar="OUTLINE_COLOR",
|
130
130
|
)
|
131
131
|
|
132
|
+
watermark_parser.add_argument(
|
133
|
+
"-a",
|
134
|
+
"--alpha",
|
135
|
+
dest="alpha",
|
136
|
+
type=int,
|
137
|
+
default=128,
|
138
|
+
choices=range(0, 256),
|
139
|
+
metavar="ALPHA_VALUE",
|
140
|
+
help=(
|
141
|
+
"set the transparency of the watermark text (0-255, "
|
142
|
+
"where 0 is fully transparent and 255 is fully opaque; "
|
143
|
+
"default: '%(default)s')"
|
144
|
+
),
|
145
|
+
)
|
146
|
+
|
132
147
|
watermark_parser.add_argument(
|
133
148
|
"--camera",
|
134
149
|
default=False,
|
@@ -176,7 +191,15 @@ def run(args: argparse.Namespace) -> None:
|
|
176
191
|
log.debug(args)
|
177
192
|
|
178
193
|
for image_filename in args.image_filenames:
|
179
|
-
|
194
|
+
try:
|
195
|
+
image: Image.Image = Image.open(image_filename)
|
196
|
+
except FileNotFoundError:
|
197
|
+
log.error("Image file not found: %s", image_filename)
|
198
|
+
continue
|
199
|
+
except Exception as e:
|
200
|
+
log.error("Could not open image %s: %s", image_filename, e)
|
201
|
+
continue
|
202
|
+
|
180
203
|
if image.format == "GIF":
|
181
204
|
watermark_gif_image(image, image_filename, args)
|
182
205
|
else:
|
@@ -202,7 +225,7 @@ def watermark_gif_image(
|
|
202
225
|
frames: list[Image.Image] = []
|
203
226
|
for frame in ImageSequence.Iterator(original_image):
|
204
227
|
watermarked_frame: Image.Image = watermark_image(
|
205
|
-
args, frame.convert("RGBA")
|
228
|
+
args, frame.convert("RGBA"), args.alpha
|
206
229
|
)
|
207
230
|
frames.append(watermarked_frame)
|
208
231
|
|
@@ -241,11 +264,11 @@ def watermark_non_gif_image(
|
|
241
264
|
Returns:
|
242
265
|
Image.Image: The watermarked image
|
243
266
|
"""
|
244
|
-
return watermark_image(args, original_image)
|
267
|
+
return watermark_image(args, original_image, args.alpha)
|
245
268
|
|
246
269
|
|
247
270
|
def watermark_image(
|
248
|
-
args: argparse.Namespace, original_image: Image.Image
|
271
|
+
args: argparse.Namespace, original_image: Image.Image, alpha: int
|
249
272
|
) -> Image.Image:
|
250
273
|
"""Watermark an image."""
|
251
274
|
watermarked_image: Image.Image = original_image.copy()
|
@@ -268,23 +291,31 @@ def watermark_image(
|
|
268
291
|
calc_padding(original_image, args),
|
269
292
|
)
|
270
293
|
|
294
|
+
try:
|
295
|
+
font_fill_color = ImageColor.getrgb(args.font_color)
|
296
|
+
stroke_fill_color = ImageColor.getrgb(args.outline_color)
|
297
|
+
except ValueError:
|
298
|
+
log.error("Invalid font or outline color specified. Using defaults.")
|
299
|
+
font_fill_color = ImageColor.getrgb("white")
|
300
|
+
stroke_fill_color = ImageColor.getrgb("black")
|
301
|
+
|
271
302
|
draw.text(
|
272
303
|
(position_x, position_y),
|
273
304
|
text,
|
274
305
|
font=font,
|
275
|
-
fill=(*
|
306
|
+
fill=(*font_fill_color, alpha),
|
276
307
|
stroke_width=calc_font_outline_width(original_image, args),
|
277
|
-
stroke_fill=(*
|
308
|
+
stroke_fill=(*stroke_fill_color, alpha),
|
278
309
|
)
|
279
310
|
return watermarked_image
|
280
311
|
|
281
312
|
|
282
313
|
def prepare_text(args: argparse.Namespace, image: Image.Image) -> str:
|
283
314
|
"""Prepare the watermark text."""
|
284
|
-
text
|
315
|
+
text = args.text
|
285
316
|
if args.camera:
|
286
|
-
metadata_text
|
287
|
-
if metadata_text:
|
317
|
+
metadata_text = get_formatted_camera_info(image)
|
318
|
+
if metadata_text:
|
288
319
|
text = metadata_text
|
289
320
|
else:
|
290
321
|
log.warning(
|
@@ -1,6 +1,6 @@
|
|
1
1
|
Metadata-Version: 2.1
|
2
2
|
Name: fotolab
|
3
|
-
Version: 0.
|
3
|
+
Version: 0.31.1
|
4
4
|
Summary: A console program that manipulate images.
|
5
5
|
Keywords: photography,photo
|
6
6
|
Author-email: Kian-Meng Ang <kianmeng@cpan.org>
|
@@ -371,7 +371,8 @@ fotolab watermark -h
|
|
371
371
|
usage: fotolab watermark [-h] [-t WATERMARK_TEXT]
|
372
372
|
[-p {top-left,top-right,bottom-left,bottom-right}]
|
373
373
|
[-pd PADDING] [-fs FONT_SIZE] [-fc FONT_COLOR]
|
374
|
-
[-ow OUTLINE_WIDTH] [-oc OUTLINE_COLOR]
|
374
|
+
[-ow OUTLINE_WIDTH] [-oc OUTLINE_COLOR]
|
375
|
+
[-a ALPHA_VALUE] [--camera]
|
375
376
|
[-l | --lowercase | --no-lowercase] [-op]
|
376
377
|
[-od OUTPUT_DIR]
|
377
378
|
IMAGE_FILENAMES [IMAGE_FILENAMES ...]
|
@@ -400,6 +401,10 @@ options:
|
|
400
401
|
-oc, --outline-color OUTLINE_COLOR
|
401
402
|
set the outline color of the watermark text (default:
|
402
403
|
'black')
|
404
|
+
-a, --alpha ALPHA_VALUE
|
405
|
+
set the transparency of the watermark text (0-255,
|
406
|
+
where 0 is fully transparent and 255 is fully opaque;
|
407
|
+
default: '128')
|
403
408
|
--camera use camera metadata as watermark
|
404
409
|
-l, --lowercase, --no-lowercase
|
405
410
|
lowercase the watermark text
|
@@ -1,21 +1,21 @@
|
|
1
|
-
fotolab/__init__.py,sha256=
|
1
|
+
fotolab/__init__.py,sha256=KWNA2ZPAuWOJFMr7zE2Lqnp2yOIi8e7vNdeo9uanRsQ,3087
|
2
2
|
fotolab/__main__.py,sha256=aboOURPs_snOXTEWYR0q8oq1UTY9e-NxCd1j33V0wHI,833
|
3
3
|
fotolab/cli.py,sha256=oFiQXmsu3wIsM_DpZnL4B94sAoB62L16Am-cjxGmosY,4406
|
4
4
|
fotolab/subcommands/__init__.py,sha256=l3DlIaJ3u3jGjnC1H1yV8LZ_nPqOLJ6gikD4BCaMAQ0,1129
|
5
|
-
fotolab/subcommands/animate.py,sha256=
|
6
|
-
fotolab/subcommands/auto.py,sha256=
|
7
|
-
fotolab/subcommands/border.py,sha256
|
5
|
+
fotolab/subcommands/animate.py,sha256=_tfIJOcwft-OP1PluPThWMsO6RAunj5RbhxeRi8Gf9w,5549
|
6
|
+
fotolab/subcommands/auto.py,sha256=3Hl1cCdu65GcfwvwiMRAcex6c7zA-KqIY6AFnB6nI3w,2447
|
7
|
+
fotolab/subcommands/border.py,sha256=-RNAYVcVTVrySzkmu3bIf__FTzog1BFC1jXqozF7PVM,4695
|
8
8
|
fotolab/subcommands/contrast.py,sha256=fcXmHnxDw74j5ZUDQ5cwWh0N4tpyqqvEjymnpITgrEk,3027
|
9
9
|
fotolab/subcommands/env.py,sha256=QoxRvzZKgmoHTUxDV4QYhdChCpMWs5TbXFY_qIpIQpE,1469
|
10
10
|
fotolab/subcommands/halftone.py,sha256=lt6RV0OuZkGs1LigTC1EcCCY42CocPFHWaJTDjJ5LFM,6693
|
11
|
-
fotolab/subcommands/info.py,sha256=
|
11
|
+
fotolab/subcommands/info.py,sha256=H3voMi67cKoHT2Mu4RUNQBPdb_MspetPjhOvy-YyNnE,3563
|
12
12
|
fotolab/subcommands/montage.py,sha256=d_3EcyRSFS8fKkczlHO8IRP-mrKhQUtkQndjfd0MKsc,2566
|
13
13
|
fotolab/subcommands/resize.py,sha256=UOb2rg_5ArRj0gxPTDOww_ix3tqJRC0W5b_S-Lt1G4w,5444
|
14
14
|
fotolab/subcommands/rotate.py,sha256=uBFjHyjiBSQLtrtH1p9myODIHUDr1gkL4PpU-6Y1Ofo,2575
|
15
15
|
fotolab/subcommands/sharpen.py,sha256=YNho2IPbc-lPvSy3Bsjehc2JOEy27LPqFSGRULs9MyY,3492
|
16
|
-
fotolab/subcommands/watermark.py,sha256=
|
17
|
-
fotolab-0.
|
18
|
-
fotolab-0.
|
19
|
-
fotolab-0.
|
20
|
-
fotolab-0.
|
21
|
-
fotolab-0.
|
16
|
+
fotolab/subcommands/watermark.py,sha256=wKv_8eREUZ20hNF2xKs1XZlcSNbB2kdG_kbBlaXhJ1I,11298
|
17
|
+
fotolab-0.31.1.dist-info/entry_points.txt,sha256=mvw7AY_yZkIyjAxPtHNed9X99NZeLnMxEeAfEJUbrCM,44
|
18
|
+
fotolab-0.31.1.dist-info/LICENSE.md,sha256=tGtFDwxWTjuR9syrJoSv1Hiffd2u8Tu8cYClfrXS_YU,31956
|
19
|
+
fotolab-0.31.1.dist-info/WHEEL,sha256=EZbGkh7Ie4PoZfRQ8I0ZuP9VklN_TvcZ6DSE5Uar4z4,81
|
20
|
+
fotolab-0.31.1.dist-info/METADATA,sha256=UjjTDVhNEU_PHEGJE-Cc_Z2zNugHVnxycHU3lzpBLIM,13656
|
21
|
+
fotolab-0.31.1.dist-info/RECORD,,
|
File without changes
|
File without changes
|
File without changes
|