From 4669e9a7de9750b0242ba8b2f2b749489383b4bb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Pablo=20Mu=C3=B1oz?= Date: Thu, 16 Nov 2023 22:57:03 +0100 Subject: [PATCH 1/3] Add basic version of multiple back images --- cartuli/definition.py | 48 +++++++++++++++++++++++++++++++++---------- 1 file changed, 37 insertions(+), 11 deletions(-) diff --git a/cartuli/definition.py b/cartuli/definition.py index 43bd76b..35b99a2 100644 --- a/cartuli/definition.py +++ b/cartuli/definition.py @@ -91,25 +91,51 @@ def _create_deck(self, definition: dict, name: str) -> Deck: name=Path(path).stem ) for path in front_image_files if self.__cards_filter(path)) ) + # TODO: Add warning if no images are found if len(front_image_files) != len(front_images): logger.debug(f"Front images filterd from {len(front_image_files)} to " f" {len(front_images)} for '{name}' deck") back_image = None if 'back' in definition: - back_image_file = definition['back']['image'] - if self.__cards_filter(back_image_file): + if 'image' in definition['back']: + back_image_file = definition['back']['image'] + if self.__cards_filter(back_image_file): + back_filter = definition['back'].get('filter', '') + back_image = self.filters[back_filter].apply( + CardImage( + definition['back']['image'], + size=size, + bleed=from_str(definition['back'].get('bleed', str(CardImage.DEFAULT_BLEED))), + name=Path(back_image_file).stem + ) + ) + return Deck((Card(image) for image in front_images), default_back=back_image, size=size, name=name) + else: + logger.debug(f"Back image '{back_image_file}' filtered for '{name}' deck") + + if 'images' in definition['back']: back_filter = definition['back'].get('filter', '') - back_image = self.filters[back_filter].apply( - CardImage( - definition['back']['image'], - size=size, - bleed=from_str(definition['back'].get('bleed', str(CardImage.DEFAULT_BLEED))), - name=Path(back_image_file).stem + back_image_files = sorted(glob(definition['back']['images'])) + logger.debug(f"Found {len(back_image_files)} back images for '{name}' deck") + with Pool(processes=cpu_count() - 1) as pool: + back_images = pool.map( + self.filters[back_filter].apply, + (CardImage( + path, size=size, + bleed=from_str(definition['back'].get('bleed', str(CardImage.DEFAULT_BLEED))), + name=Path(path).stem + ) for path in back_image_files if self.__cards_filter(path)) ) - ) + # TODO: Add warning if no images are found + return Deck( + (Card(front_image, back_image) for front_image, back_image in zip(front_images, back_images)), + size=size, name=name + ) + if len(back_image_files) != len(back_images): + logger.debug(f"Back images filterd from {len(back_image_files)} to " + f" {len(back_images)} for '{name}' deck") else: - logger.debug(f"Back image '{back_image_file}' filtered for '{name}' deck") - return Deck((Card(image) for image in front_images), default_back=back_image, size=size, name=name) + return Deck((Card(image) for image in front_images), size=size, name=name) @property def sheets(self) -> dict[tuple[str], Sheet]: From 04abc1d04d08cfb4d5ca309129a084e023ac351d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Pablo=20Mu=C3=B1oz?= Date: Thu, 16 Nov 2023 22:57:03 +0100 Subject: [PATCH 2/3] Solve issues in first implementation --- cartuli/definition.py | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/cartuli/definition.py b/cartuli/definition.py index 35b99a2..50813ab 100644 --- a/cartuli/definition.py +++ b/cartuli/definition.py @@ -91,10 +91,12 @@ def _create_deck(self, definition: dict, name: str) -> Deck: name=Path(path).stem ) for path in front_image_files if self.__cards_filter(path)) ) - # TODO: Add warning if no images are found + # TODO: Add warning if no images are found and manage it properly and do not create + # deck if len(front_image_files) != len(front_images): logger.debug(f"Front images filterd from {len(front_image_files)} to " f" {len(front_images)} for '{name}' deck") + back_image = None if 'back' in definition: if 'image' in definition['back']: @@ -127,6 +129,7 @@ def _create_deck(self, definition: dict, name: str) -> Deck: ) for path in back_image_files if self.__cards_filter(path)) ) # TODO: Add warning if no images are found + # TODO: Add error if front and back sizes do not match return Deck( (Card(front_image, back_image) for front_image, back_image in zip(front_images, back_images)), size=size, name=name @@ -134,8 +137,8 @@ def _create_deck(self, definition: dict, name: str) -> Deck: if len(back_image_files) != len(back_images): logger.debug(f"Back images filterd from {len(back_image_files)} to " f" {len(back_images)} for '{name}' deck") - else: - return Deck((Card(image) for image in front_images), size=size, name=name) + + return Deck((Card(image) for image in front_images), size=size, name=name) @property def sheets(self) -> dict[tuple[str], Sheet]: From 870a44cef462f80d9742b4d0d6a8f1a6cf3bb9f0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Pablo=20Mu=C3=B1oz?= Date: Sun, 19 Nov 2023 23:29:51 +0100 Subject: [PATCH 3/3] Simplify how card images are loaded --- cartuli/definition.py | 117 ++++++++++++++--------------- tests/files/simple-cartulifile.yml | 2 +- tests/test_definition.py | 4 +- 3 files changed, 58 insertions(+), 65 deletions(-) diff --git a/cartuli/definition.py b/cartuli/definition.py index 50813ab..11a62e5 100644 --- a/cartuli/definition.py +++ b/cartuli/definition.py @@ -11,7 +11,7 @@ from multiprocessing import Pool, cpu_count from pathlib import Path -from .card import Card, CardImage +from .card import CardImage, Card from .deck import Deck from .filters import Filter, NullFilter, from_dict as filter_from_dict from .measure import Size, from_str @@ -21,6 +21,10 @@ CardsFilter = Callable[[Path], bool] +class DefinitionError(Exception): + pass + + class Definition: """Definition.""" @@ -67,78 +71,67 @@ def decks(self) -> list[Deck]: self.__decks = [] for name, deck_definition in self.__values.get('decks', {}).items(): logger.debug(f"Deck '{name}' definition {deck_definition}") - self.__decks.append(self._create_deck(deck_definition, name)) + self.__decks.append(self._load_deck(deck_definition, name)) if not self.__decks: logger.warning('No decks loaded in definition') return self.__decks - def _create_deck(self, definition: dict, name: str) -> Deck: + def _load_images(self, images_definition: dict, size: Size, deck_name: str, side: str = 'front') -> list[CardImage]: + logger = logging.getLogger('cartuli.definition.Definition.decks') + + image_filter = images_definition.get('filter', '') + image_files = sorted(glob(images_definition['images'])) + logger.debug(f"Found {len(image_files)} {side} images for '{deck_name}' deck") + with Pool(processes=cpu_count() - 1) as pool: + images = pool.map( + self.filters[image_filter].apply, + (CardImage( + path, size=size, + bleed=from_str(images_definition.get('bleed', str(CardImage.DEFAULT_BLEED))), + name=Path(path).stem + ) for path in image_files if self.__cards_filter(path)) + ) + if len(image_files) != len(images): + logger.debug(f"{side.capitalize()} images filterd from {len(image_files)} to " + f" {len(images)} for '{deck_name}' deck") + + return images + + def _load_deck(self, definition: dict, name: str) -> Deck: logger = logging.getLogger('cartuli.definition.Definition.decks') size = Size.from_str(definition['size']) - front_images = [] + cards = [] + if 'front' in definition: - front_filter = definition['front'].get('filter', '') - front_image_files = sorted(glob(definition['front']['images'])) - logger.debug(f"Found {len(front_image_files)} front images for '{name}' deck") - with Pool(processes=cpu_count() - 1) as pool: - front_images = pool.map( - self.filters[front_filter].apply, - (CardImage( - path, size=size, - bleed=from_str(definition['front'].get('bleed', str(CardImage.DEFAULT_BLEED))), - name=Path(path).stem - ) for path in front_image_files if self.__cards_filter(path)) + front_images = self._load_images(definition['front'], size, name, 'front') + if 'back' in definition: + back_images = self._load_images(definition['back'], size, name, 'back') + if len(front_images) != len(back_images): + raise DefinitionError(f"The number of front ({len(front_images)}) and " + f"back ({len(back_images)}) images must be the same") + cards = [Card(front_image, back_image) for front_image, back_image in zip(front_images, back_images)] + else: + cards = [Card(image) for image in front_images] + + if not cards: + logger.warning(f"No cards found for deck {name} with specified fiters") + + default_back = None + if 'default_back' in definition: + default_back_file = definition['default_back']['image'] + default_back_filter = definition['default_back'].get('filter', '') + default_back = self.filters[default_back_filter].apply( + CardImage( + default_back_file, + size=size, + bleed=from_str(definition['default_back'].get('bleed', str(CardImage.DEFAULT_BLEED))), + name=Path(default_back_file).stem ) - # TODO: Add warning if no images are found and manage it properly and do not create - # deck - if len(front_image_files) != len(front_images): - logger.debug(f"Front images filterd from {len(front_image_files)} to " - f" {len(front_images)} for '{name}' deck") - - back_image = None - if 'back' in definition: - if 'image' in definition['back']: - back_image_file = definition['back']['image'] - if self.__cards_filter(back_image_file): - back_filter = definition['back'].get('filter', '') - back_image = self.filters[back_filter].apply( - CardImage( - definition['back']['image'], - size=size, - bleed=from_str(definition['back'].get('bleed', str(CardImage.DEFAULT_BLEED))), - name=Path(back_image_file).stem - ) - ) - return Deck((Card(image) for image in front_images), default_back=back_image, size=size, name=name) - else: - logger.debug(f"Back image '{back_image_file}' filtered for '{name}' deck") - - if 'images' in definition['back']: - back_filter = definition['back'].get('filter', '') - back_image_files = sorted(glob(definition['back']['images'])) - logger.debug(f"Found {len(back_image_files)} back images for '{name}' deck") - with Pool(processes=cpu_count() - 1) as pool: - back_images = pool.map( - self.filters[back_filter].apply, - (CardImage( - path, size=size, - bleed=from_str(definition['back'].get('bleed', str(CardImage.DEFAULT_BLEED))), - name=Path(path).stem - ) for path in back_image_files if self.__cards_filter(path)) - ) - # TODO: Add warning if no images are found - # TODO: Add error if front and back sizes do not match - return Deck( - (Card(front_image, back_image) for front_image, back_image in zip(front_images, back_images)), - size=size, name=name - ) - if len(back_image_files) != len(back_images): - logger.debug(f"Back images filterd from {len(back_image_files)} to " - f" {len(back_images)} for '{name}' deck") + ) - return Deck((Card(image) for image in front_images), size=size, name=name) + return Deck(cards, name=name, default_back=default_back, size=size) @property def sheets(self) -> dict[tuple[str], Sheet]: diff --git a/tests/files/simple-cartulifile.yml b/tests/files/simple-cartulifile.yml index 7db251a..d2d7d07 100644 --- a/tests/files/simple-cartulifile.yml +++ b/tests/files/simple-cartulifile.yml @@ -3,7 +3,7 @@ decks: size: "STANDARD" front: images: "cards/*.png" - back: + default_back: image: "card-back.png" outputs: sheet: diff --git a/tests/test_definition.py b/tests/test_definition.py index f2a45d2..4187b0a 100644 --- a/tests/test_definition.py +++ b/tests/test_definition.py @@ -20,7 +20,7 @@ def test_file(fixture_file): 'front': { 'images': "cards/*.png" }, - 'back': { + 'default_back': { 'image': "card-back.png" } } @@ -76,7 +76,7 @@ def test_definition(random_image_file): 'images': str(random_image_dir / "*.png"), 'bleed': '2*mm' }, - 'back': { + 'default_back': { 'image': str(random_image_file("back")) } }