fotolab 0.30.0__tar.gz → 0.31.0__tar.gz

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.
Files changed (56) hide show
  1. {fotolab-0.30.0 → fotolab-0.31.0}/.gitignore +1 -0
  2. {fotolab-0.30.0 → fotolab-0.31.0}/CHANGELOG.md +21 -12
  3. {fotolab-0.30.0 → fotolab-0.31.0}/PKG-INFO +1 -1
  4. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/__init__.py +7 -8
  5. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/subcommands/border.py +7 -7
  6. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/subcommands/watermark.py +37 -6
  7. {fotolab-0.30.0 → fotolab-0.31.0}/tests/conftest.py +11 -0
  8. fotolab-0.31.0/tests/test_animate_subcommand.py +7 -0
  9. fotolab-0.31.0/tests/test_auto_subcommand.py +7 -0
  10. fotolab-0.31.0/tests/test_border_subcommand.py +6 -0
  11. fotolab-0.31.0/tests/test_contrast_subcommand.py +6 -0
  12. fotolab-0.31.0/tests/test_env_subcommand.py +23 -0
  13. fotolab-0.31.0/tests/test_halftone_subcommand.py +6 -0
  14. fotolab-0.31.0/tests/test_info_subcommand.py +6 -0
  15. fotolab-0.31.0/tests/test_montage_subcommand.py +6 -0
  16. fotolab-0.31.0/tests/test_resize_subcommand.py +6 -0
  17. fotolab-0.31.0/tests/test_rotate_subcommand.py +6 -0
  18. fotolab-0.31.0/tests/test_sharpen_subcommand.py +6 -0
  19. fotolab-0.31.0/tests/test_watermark_subcommand.py +6 -0
  20. fotolab-0.30.0/tests/test_env_subcommand.py +0 -10
  21. {fotolab-0.30.0 → fotolab-0.31.0}/.coveragerc +0 -0
  22. {fotolab-0.30.0 → fotolab-0.31.0}/.pre-commit-config.yaml +0 -0
  23. {fotolab-0.30.0 → fotolab-0.31.0}/.python-version +0 -0
  24. {fotolab-0.30.0 → fotolab-0.31.0}/CONTRIBUTING.md +0 -0
  25. {fotolab-0.30.0 → fotolab-0.31.0}/LICENSE.md +0 -0
  26. {fotolab-0.30.0 → fotolab-0.31.0}/Pipfile +0 -0
  27. {fotolab-0.30.0 → fotolab-0.31.0}/Pipfile.lock +0 -0
  28. {fotolab-0.30.0 → fotolab-0.31.0}/README.md +0 -0
  29. {fotolab-0.30.0 → fotolab-0.31.0}/docs/Makefile +0 -0
  30. {fotolab-0.30.0 → fotolab-0.31.0}/docs/make.bat +0 -0
  31. {fotolab-0.30.0 → fotolab-0.31.0}/docs/source/CHANGELOG.md +0 -0
  32. {fotolab-0.30.0 → fotolab-0.31.0}/docs/source/CONTRIBUTING.md +0 -0
  33. {fotolab-0.30.0 → fotolab-0.31.0}/docs/source/LICENSE.md +0 -0
  34. {fotolab-0.30.0 → fotolab-0.31.0}/docs/source/README.md +0 -0
  35. {fotolab-0.30.0 → fotolab-0.31.0}/docs/source/_static/logo.jpg +0 -0
  36. {fotolab-0.30.0 → fotolab-0.31.0}/docs/source/conf.py +0 -0
  37. {fotolab-0.30.0 → fotolab-0.31.0}/docs/source/index.rst +0 -0
  38. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/__main__.py +0 -0
  39. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/cli.py +0 -0
  40. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/subcommands/__init__.py +0 -0
  41. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/subcommands/animate.py +0 -0
  42. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/subcommands/auto.py +0 -0
  43. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/subcommands/contrast.py +0 -0
  44. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/subcommands/env.py +0 -0
  45. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/subcommands/halftone.py +0 -0
  46. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/subcommands/info.py +0 -0
  47. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/subcommands/montage.py +0 -0
  48. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/subcommands/resize.py +0 -0
  49. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/subcommands/rotate.py +0 -0
  50. {fotolab-0.30.0 → fotolab-0.31.0}/fotolab/subcommands/sharpen.py +0 -0
  51. {fotolab-0.30.0 → fotolab-0.31.0}/generate +0 -0
  52. {fotolab-0.30.0 → fotolab-0.31.0}/noxfile.py +0 -0
  53. {fotolab-0.30.0 → fotolab-0.31.0}/pyproject.toml +0 -0
  54. {fotolab-0.30.0 → fotolab-0.31.0}/tests/__init__.py +0 -0
  55. {fotolab-0.30.0 → fotolab-0.31.0}/tests/test_help_flag.py +0 -0
  56. {fotolab-0.30.0 → fotolab-0.31.0}/tests/test_quiet_flag.py +0 -0
@@ -162,5 +162,6 @@ cython_debug/
162
162
  # heatmaps
163
163
  *.png
164
164
  .aider*
165
+ .opencode
165
166
  TODO
166
167
  output
@@ -7,6 +7,15 @@ and this project adheres to [0-based versioning](https://0ver.org/).
7
7
 
8
8
  ## [Unreleased]
9
9
 
10
+ ## v0.31.0 (2025-05-25)
11
+
12
+ - Add basic tests for several CLI subcommands
13
+ - Add test for auto subcommand and image fixture
14
+ - Add watermark transparency option and improve error handling
15
+ - Fix markdown in changelog
16
+ - Refine `env` subcommand test assertions
17
+ - Standardize border dimension return as a tuple
18
+
10
19
  ## v0.30.0 (2025-05-18)
11
20
 
12
21
  - Add WEBP `quality`, `lossless`, and `method` args
@@ -14,14 +23,14 @@ and this project adheres to [0-based versioning](https://0ver.org/).
14
23
  - Close image files after creating animation
15
24
  - Update help message generated by latest Python to readme
16
25
  - Update help message in readme
17
- - Validate duration for animate command
26
+ - Validate duration for `animate` subcommand
18
27
 
19
28
  ## v0.29.2 (2025-05-11)
20
29
 
21
30
  - Bump deps
22
31
  - Code format
23
32
  - Improve info subcommand by adding error handling for missing keys
24
- - Remove default value for image filename argument in info subcommand
33
+ - Remove default value for image filename argument in `info` subcommand
25
34
  - Use context manager to open image and remove redundant image close.
26
35
 
27
36
  ## v0.29.1 (2025-05-04)
@@ -34,8 +43,8 @@ and this project adheres to [0-based versioning](https://0ver.org/).
34
43
 
35
44
  ## v0.29.0 (2025-04-27)
36
45
 
37
- - Add `cells` argument to halftone subcommand and function
38
- - Add grayscale option to halftone subcommand for grayscale conversion
46
+ - Add `cells` argument to `halftone` subcommand and function
47
+ - Add `grayscale` option to `halftone` subcommand for grayscale conversion
39
48
  - Bump deps
40
49
  - Calculate dot radius relative to cell size based on brightness.
41
50
  - Refactor CLI to handle missing subcommand execution function gracefully
@@ -51,22 +60,22 @@ and this project adheres to [0-based versioning](https://0ver.org/).
51
60
  ## v0.28.5 (2025-04-13)
52
61
 
53
62
  - Group build and publish package commands together
54
- - Improve contrast subcommand and add pylint disable
55
- - Prompt to publish package in release nox job
56
- - Resolve pylint raised issue
57
- - Validate cutoff value is between 0 and 50 in contrast subcommand
63
+ - Improve `contrast` subcommand and add `pylint` disable
64
+ - Prompt to publish package in release `nox` job
65
+ - Resolve `pylint` raised issue
66
+ - Validate `cutoff` value is between 0 and 50 in `contrast` subcommand
58
67
 
59
68
  ## v0.28.4 (2025-04-06)
60
69
 
61
70
  - Build the release after release `nox` job
62
71
  - Bump `pre-commit` for `flake8`
63
72
  - Fix incorrect git options
64
- - Install flit as deps for dev environment
73
+ - Install `flit` as deps for dev environment
65
74
  - Update help message in readme
66
75
 
67
76
  ## v0.28.3 (2025-03-30)
68
77
 
69
- - Bump `pre-commit` hook for validate-project
78
+ - Bump `pre-commit` hook for `validate-project`
70
79
  - Commit changes after bump release
71
80
  - Remove exception handling as it's handled in parent calling function
72
81
  - Resolve W0718: Catching too general exception Exception
@@ -76,7 +85,7 @@ and this project adheres to [0-based versioning](https://0ver.org/).
76
85
  ## v0.28.2 (2025-03-23)
77
86
 
78
87
  - Accept width in integer instead of string
79
- - Bump pre-commit hook for validate-project
88
+ - Bump `pre-commit` hook for `validate-project`
80
89
  - Handle file not found and other exceptions when opening/processing images
81
90
  - Remove all unnecessary exceptions handling
82
91
  - Use Union for border width return type to allow int or tuple
@@ -97,7 +106,7 @@ and this project adheres to [0-based versioning](https://0ver.org/).
97
106
  - Refactor getting output filename when saving image
98
107
  - Remove `--output-dir` and `--open` args from main cli
99
108
  - Update help message in readme
100
- - Use single `save_gif_image` function in sharpen and halftone subcommand
109
+ - Use single `save_gif_image` function in sharpen and `halftone` subcommand
101
110
 
102
111
  ## v0.27.2 (2025-03-02)
103
112
 
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: fotolab
3
- Version: 0.30.0
3
+ Version: 0.31.0
4
4
  Summary: A console program that manipulate images.
5
5
  Keywords: photography,photo
6
6
  Author-email: Kian-Meng Ang <kianmeng@cpan.org>
@@ -24,7 +24,7 @@ from pathlib import Path
24
24
 
25
25
  from PIL import Image
26
26
 
27
- __version__ = "0.30.0"
27
+ __version__ = "0.31.0"
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(args, new_image, output_filename, subcommand):
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
  """
@@ -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
- ) -> Union[Tuple[int, int, int, int], int]:
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
- Union[Tuple[int, int, int, int], int]: Border dimensions in pixels.
161
- If individual widths are specified, returns a tuple of (left, top,
162
- right, bottom) widths. Otherwise, returns a uniform width for all
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
- return args.width
178
+ # If no individual widths are specified, use the general width for all sides
179
+ return (args.width, args.width, args.width, args.width)
@@ -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
- image: Image.Image = Image.open(image_filename)
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,13 +291,21 @@ 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=(*ImageColor.getrgb(args.font_color), 128),
306
+ fill=(*font_fill_color, alpha),
276
307
  stroke_width=calc_font_outline_width(original_image, args),
277
- stroke_fill=(*ImageColor.getrgb(args.outline_color), 128),
308
+ stroke_fill=(*stroke_fill_color, alpha),
278
309
  )
279
310
  return watermarked_image
280
311
 
@@ -27,6 +27,17 @@ def fixture_csv_file(tmpdir):
27
27
  return csv_file
28
28
 
29
29
 
30
+ @pytest.fixture(autouse=True, name="image_file")
31
+ def fixture_image_file(tmpdir):
32
+ def image_file(filename):
33
+ src = Path(FIXTURE_PATH, filename)
34
+ des = Path(tmpdir, src.name)
35
+ copyfile(src, des)
36
+ return des
37
+
38
+ return image_file
39
+
40
+
30
41
  @pytest.fixture(autouse=True, name="cli_runner")
31
42
  def fixture_cli_runner(tmpdir):
32
43
  def cli_runner(*args, **kwargs):
@@ -0,0 +1,7 @@
1
+ # pylint: disable=C0114,C0116
2
+
3
+ def test_animate_subcommand(cli_runner, image_file):
4
+ img_path1 = image_file("sample.png")
5
+ img_path2 = image_file("sample.png")
6
+ ret = cli_runner('animate', str(img_path1), str(img_path2))
7
+ assert ret.returncode == 0
@@ -0,0 +1,7 @@
1
+ # pylint: disable=C0114,C0116
2
+
3
+ def test_auto_subcommand(cli_runner, image_file):
4
+ """Test auto subcommand."""
5
+ img_path = image_file("sample.png")
6
+ ret = cli_runner("auto", str(img_path))
7
+ assert ret.returncode == 0
@@ -0,0 +1,6 @@
1
+ # pylint: disable=C0114,C0116
2
+
3
+ def test_border_subcommand(cli_runner, image_file):
4
+ img_path = image_file("sample.png")
5
+ ret = cli_runner('border', str(img_path), '--width', '10')
6
+ assert ret.returncode == 0
@@ -0,0 +1,6 @@
1
+ # pylint: disable=C0114,C0116
2
+
3
+ def test_contrast_subcommand(cli_runner, image_file):
4
+ img_path = image_file("sample.png")
5
+ ret = cli_runner('contrast', str(img_path))
6
+ assert ret.returncode == 0
@@ -0,0 +1,23 @@
1
+ # pylint: disable=C0114,C0116
2
+
3
+ import platform
4
+ import sys
5
+
6
+ from fotolab import __version__
7
+
8
+
9
+ def test_env_output(cli_runner):
10
+ ret = cli_runner("env")
11
+
12
+ actual_sys_version = sys.version.replace("\n", "")
13
+ actual_platform = platform.platform()
14
+
15
+ expected_output = (
16
+ f"fotolab: {__version__}\n"
17
+ f"python: {actual_sys_version}\n"
18
+ f"platform: {actual_platform}\n"
19
+ )
20
+
21
+ assert ret.stdout == expected_output
22
+ assert ret.stderr == ""
23
+ assert ret.returncode == 0
@@ -0,0 +1,6 @@
1
+ # pylint: disable=C0114,C0116
2
+
3
+ def test_halftone_subcommand(cli_runner, image_file):
4
+ img_path = image_file("sample.png")
5
+ ret = cli_runner('halftone', str(img_path))
6
+ assert ret.returncode == 0
@@ -0,0 +1,6 @@
1
+ # pylint: disable=C0114,C0116
2
+
3
+ def test_info_subcommand(cli_runner, image_file):
4
+ img_path = image_file("sample.png")
5
+ ret = cli_runner('info', str(img_path))
6
+ assert ret.returncode == 0
@@ -0,0 +1,6 @@
1
+ # pylint: disable=C0114,C0116
2
+
3
+ def test_montage_subcommand(cli_runner, image_file):
4
+ img_path = image_file("sample.png")
5
+ ret = cli_runner('montage', str(img_path), str(img_path))
6
+ assert ret.returncode == 0
@@ -0,0 +1,6 @@
1
+ # pylint: disable=C0114,C0116
2
+
3
+ def test_resize_subcommand(cli_runner, image_file):
4
+ img_path = image_file("sample.png")
5
+ ret = cli_runner('resize', str(img_path), '--width', '200')
6
+ assert ret.returncode == 0
@@ -0,0 +1,6 @@
1
+ # pylint: disable=C0114,C0116
2
+
3
+ def test_rotate_subcommand(cli_runner, image_file):
4
+ img_path = image_file("sample.png")
5
+ ret = cli_runner('rotate', str(img_path))
6
+ assert ret.returncode == 0
@@ -0,0 +1,6 @@
1
+ # pylint: disable=C0114,C0116
2
+
3
+ def test_sharpen_subcommand(cli_runner, image_file):
4
+ img_path = image_file("sample.png")
5
+ ret = cli_runner('sharpen', str(img_path))
6
+ assert ret.returncode == 0
@@ -0,0 +1,6 @@
1
+ # pylint: disable=C0114,C0116
2
+
3
+ def test_watermark_subcommand(cli_runner, image_file):
4
+ img_path = image_file("sample.png")
5
+ ret = cli_runner('watermark', str(img_path), '--text', 'Test')
6
+ assert ret.returncode == 0
@@ -1,10 +0,0 @@
1
- # pylint: disable=C0114,C0116
2
-
3
- from fotolab import __version__
4
-
5
-
6
- def test_env_output(cli_runner):
7
- ret = cli_runner("env")
8
- assert f"fotolab: {__version__}" in ret.stdout
9
- assert "python: " in ret.stdout
10
- assert "platform: " in ret.stdout
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes