diff --git a/.claude/commands/wygeneruj-projekty.md b/.claude/commands/wygeneruj-projekty.md index 0760ec1..d5ff92a 100644 --- a/.claude/commands/wygeneruj-projekty.md +++ b/.claude/commands/wygeneruj-projekty.md @@ -122,7 +122,9 @@ python tools/generowanie/email_message_fetcher.py --email "{buyer_email}" --days - **Data uroczystości** — format DD.MM.YYYY, może być opisana jako "Data: 30.04.2026" lub "Data uroczystości: 30.04.2026" lub po prostu "30.04.2026" - **Życzenia** — opcjonalny tekst (np. "Na zdrowie!", "Dziękujemy!"). Jeśli brak — zostaw domyślne z szablonu - Ignoruj pola takie jak "Kolor tekstu", "Zakrętka" — te dotyczą produkcji, nie projektu -- **Imię dziecka — zawsze dopełniacz:** Szablony używają formy "Chrzest Święty [kogo?]", "Komunia Święta [kogo?]" itp. Imię podane przez klienta w mianowniku **zawsze** przekształć na dopełniacz przed przekazaniem do skryptu. Przykłady: Ignacy → Ignacego, Teodor → Teodora, Zosia → Zosi, Maja → Mai, Jakub → Jakuba, Franciszek → Franciszka. Raportuj przekształcenie w planie (oryginał → dopełniacz). +- **Imię dziecka — forma zależna od skryptu:** + - **Pudełka, magnesy (`pudelko_*`, `magnes_*`)** — dopełniacz: "Chrzest Święty [kogo?]", "Komunia Święta [kogo?]". Przykłady: Ignacy → Ignacego, Teodor → Teodora, Zosia → Zosi, Maja → Mai, Jakub → Jakuba. Raportuj przekształcenie w planie. + - **Prośby o chrzestnego (`prosba_chrzestny_*`)** — **mianownik**: imię dziecka to podpis na karcie (dziecko „pisze" prośbę). Przekaż dosłownie: Tymon → Tymon, Borysek → Borysek. NIE stosuj dopełniacza. - **Nazwa klienta** (do nazwy pliku wyjściowego) — z `order_addresses.name` (pole `buyer_name` w zapytaniu) **WAŻNE — semantyczny podział pola "Kto składa życzenia" na `naglowek` + `od_kogo`:** @@ -304,6 +306,8 @@ Photoshop jako znak łamania linii w warstwach tekstowych rozpoznaje **`\r` (CR) Dla każdego potwierdzonego produktu: 1. **Sprawdź kolizję nazwy pliku** w katalogu wyjściowym. Jeśli plik `{klient}.psd` (lub inne rozszerzenie używane przez skrypt) już istnieje — dodaj do nazwy klienta sufiks z indeksem `01`, `02`, ... (np. `Anna Rak` → `Anna Rak 01`, a jeśli i to zajęte → `Anna Rak 02`). Inkrementuj aż znajdziesz wolną nazwę. Dopiero wtedy przekaż wynik jako `--klient`. 2. Uruchom skrypt: `python tools/generowanie/{script_name} --klient "Imię Nazwisko" ...` (parametry zależne od skryptu — dla buteleczek/pudełek zwykle `--imie/--imiona`, `--data`, opcjonalnie `--zyczenia`/`--podziekowanie`; dla produktów ze zdjęciem dorzuć `--zdjecie /sciezka/do/foto.jpg`; **dla skryptów `magnes_*` dorzuć `--liczba {order_items.quantity}`** — szablony mają N slotów na arkuszu A3, nadmiar ma być ukrywany żeby nie drukować pustych kopii) + +**WAŻNE — imię dziecka w skryptach `magnes_chrzest_*`:** Skrypty magnesów chrzestnych konstruują tytuł jako `"Chrzest Święty\r{imie}"` — słowo "Święty" **nie jest odmieniane**, odmienia się **tylko imię** (dopełniacz). Przekaż samo imię w dopełniaczu (np. `--imie "Tymka"`, `--imie "Rozalii"`, `--imie "Jakuba"`). Wynik: "Chrzest Święty↵Tymka" — poprawnie. 3. Jeśli mapowanie ma `output_dir` — skrypt powinien zapisać tam (jeśli obsługuje) ### 8. Oznacz w bazie diff --git a/.paul/governance/governance_2026-04-20.jsonl b/.paul/governance/governance_2026-04-20.jsonl new file mode 100644 index 0000000..30c4b66 --- /dev/null +++ b/.paul/governance/governance_2026-04-20.jsonl @@ -0,0 +1,13 @@ +{"ts":"2026-04-20T07:00:27Z","tool":"Bash","cmd":"tasklist //FI \"IMAGENAME eq Photoshop.exe\" 2>/dev/null | grep -i photoshop\",\"description\":\"Check if Photoshop is running\"},\"tool_response\":{\"stdout\":\"Binary file (standard input) matches\"","cwd":"/c/visual studio code/projekty/orderPRO"} +{"ts":"2026-04-20T07:11:23Z","tool":"Edit","file":"C:\\\\visual studio code\\\\projekty\\\\orderPRO\\\\tools\\\\generowanie\\\\magnes_chrzest_stopki_serce.py","cwd":"/c/visual studio code/projekty/orderPRO"} +{"ts":"2026-04-20T07:11:28Z","tool":"Edit","file":"C:\\\\visual studio code\\\\projekty\\\\orderPRO\\\\tools\\\\generowanie\\\\magnes_chrzest_stopki_serce.py","cwd":"/c/visual studio code/projekty/orderPRO"} +{"ts":"2026-04-20T07:15:46Z","tool":"Edit","file":"C:\\\\visual studio code\\\\projekty\\\\orderPRO\\\\.claude\\\\commands\\\\wygeneruj-projekty.md","cwd":"/c/visual studio code/projekty/orderPRO"} +{"ts":"2026-04-20T07:21:19Z","tool":"Edit","file":"C:\\\\visual studio code\\\\projekty\\\\orderPRO\\\\.claude\\\\commands\\\\wygeneruj-projekty.md","cwd":"/c/visual studio code/projekty/orderPRO"} +{"ts":"2026-04-20T11:01:40Z","tool":"Write","file":"C:\\\\tmp\\\\explore_psd.py","cwd":"/c/visual studio code/projekty/orderPRO"} +{"ts":"2026-04-20T11:05:05Z","tool":"Write","file":"C:\\\\tmp\\\\rename_layers.py","cwd":"/c/visual studio code/projekty/orderPRO"} +{"ts":"2026-04-20T11:05:30Z","tool":"Write","file":"C:\\\\visual studio code\\\\projekty\\\\orderPRO\\\\tools\\\\generowanie\\\\magnes_slub_kwiaty_wzor5.py","cwd":"/c/visual studio code/projekty/orderPRO"} +{"ts":"2026-04-20T11:11:00Z","tool":"Write","file":"C:\\\\tmp\\\\insert_mapping.sql","cwd":"/c/visual studio code/projekty/orderPRO"} +{"ts":"2026-04-20T13:26:19Z","tool":"Write","file":"C:\\\\tmp\\\\check_komunia_psd.py","cwd":"/c/visual studio code/projekty/orderPRO"} +{"ts":"2026-04-20T13:27:15Z","tool":"Edit","file":"C:\\\\visual studio code\\\\projekty\\\\orderPRO\\\\tools\\\\generowanie\\\\_pudelko_komunia_core.py","cwd":"/c/visual studio code/projekty/orderPRO"} +{"ts":"2026-04-20T13:27:21Z","tool":"Edit","file":"C:\\\\visual studio code\\\\projekty\\\\orderPRO\\\\tools\\\\generowanie\\\\_pudelko_komunia_core.py","cwd":"/c/visual studio code/projekty/orderPRO"} +{"ts":"2026-04-20T13:27:27Z","tool":"Edit","file":"C:\\\\visual studio code\\\\projekty\\\\orderPRO\\\\tools\\\\generowanie\\\\_pudelko_komunia_core.py","cwd":"/c/visual studio code/projekty/orderPRO"} diff --git a/.vscode/ftp-kr.sync.cache.json b/.vscode/ftp-kr.sync.cache.json index 5429d1b..81dfbb1 100644 --- a/.vscode/ftp-kr.sync.cache.json +++ b/.vscode/ftp-kr.sync.cache.json @@ -5681,6 +5681,12 @@ } } }, + "tmp_stats_debug.php": { + "type": "-", + "size": 578, + "lmtime": 0, + "modified": false + }, "tools": { "generowanie": { "akrylowa_statuetka_podziekowanie_rodzice_wzor1.py": { diff --git a/tools/generowanie/_pudelko_komunia_core.py b/tools/generowanie/_pudelko_komunia_core.py index ba1178f..8c4cb78 100644 --- a/tools/generowanie/_pudelko_komunia_core.py +++ b/tools/generowanie/_pudelko_komunia_core.py @@ -1,10 +1,13 @@ """ Wspólny rdzeń dla generatorów PSD pudełek komunijnych (chłopiec/dziewczynka). -Makieta A3 zawiera 4 grupy warstw (GL/GP/DL/DP) — 1 PSD = 1 zamówienie, +Makieta A3 zawiera do 4 grup warstw (GL/GP/DL/DP) — 1 PSD = 1 zamówienie, te same dane lecą do górnej i dolnej części makiety. Brak Smart Objectów, teksty są bezpośrednio w grupach. Po normalizacji nazw (rename w PSD) wszystkie warianty mają identyczną strukturę: od_kogo / naglowek / zyczenia / imie / data / tytul. + +Jeśli szablon nie zawiera niektórych grup (np. tylko DL/DP bez GL/GP) — +pomijamy brakujące bez błędu. """ import argparse @@ -41,9 +44,15 @@ def set_text(layer, new_text): def get_container(doc, set_name, inner_name): - container = doc.layerSets[set_name] + try: + container = doc.layerSets[set_name] + except Exception: + return None if inner_name: - container = container.layerSets[inner_name] + try: + container = container.layerSets[inner_name] + except Exception: + return None return container @@ -71,12 +80,20 @@ def generate(szablon_path, gotowe_dir, imie, data, od_kogo, klient, imie_map = {"imie": imie, "data": data, "tytul": tytul} for set_name, inner in ZYCZENIA_GROUPS: + container = get_container(doc, set_name, inner) + if container is None: + print(f" [{set_name}] życzenia: BRAK GRUPY — pomijam") + continue print(f" [{set_name}] życzenia:") - apply_replacements(get_container(doc, set_name, inner), zyczenia_map) + apply_replacements(container, zyczenia_map) for set_name, inner in IMIE_GROUPS: + container = get_container(doc, set_name, inner) + if container is None: + print(f" [{set_name}] imię/data: BRAK GRUPY — pomijam") + continue print(f" [{set_name}] imię/data:") - apply_replacements(get_container(doc, set_name, inner), imie_map) + apply_replacements(container, imie_map) psd_opts = ps.PhotoshopSaveOptions() doc.saveAs(output_path, psd_opts, True) diff --git a/tools/generowanie/magnes_chrzest_stopki_serce.py b/tools/generowanie/magnes_chrzest_stopki_serce.py index 16f0616..6479d2a 100644 --- a/tools/generowanie/magnes_chrzest_stopki_serce.py +++ b/tools/generowanie/magnes_chrzest_stopki_serce.py @@ -7,7 +7,7 @@ Edycja warstw wewnatrz SO "projekt" propaguje zmiany na wszystkie kopie. Struktura wewnatrz Smart Object "projekt" -> "Warstwa 2": - podziekowanie (text, domyslnie "Dziękuję za przybycie.") - data (DD.MM.RRRR) -- tytul (2 linie: "Chrzest Świętego\r{imie}") +- tytul (2 linie: "Chrzest Święty\r{imie}") Wymaga: uruchomiony Adobe Photoshop, photoshop-python-api. """ @@ -98,7 +98,7 @@ def generate(klient, imie, data, liczba, podziekowanie=None): print(f" Wewnatrz SO: {inner.name}") w2 = inner.layerSets[INNER_SET_NAME] - set_layer_text(w2, "tytul", f"Chrzest Świętego\r{imie}") + set_layer_text(w2, "tytul", f"Chrzest Święty\r{imie}") set_layer_text(w2, "data", data) set_layer_text(w2, "podziekowanie", podziekowanie) diff --git a/tools/generowanie/magnes_slub_kwiaty_wzor5.py b/tools/generowanie/magnes_slub_kwiaty_wzor5.py new file mode 100644 index 0000000..1f5503f --- /dev/null +++ b/tools/generowanie/magnes_slub_kwiaty_wzor5.py @@ -0,0 +1,99 @@ +""" +Generator PSD - Magnes podziekowanie dla gosci weselnych, plaster brzozy (Kwiaty Wzor 5). + +Otwiera szablon PSD, podmienia teksty w grupie "Tekst" z zachowaniem pozycji warstw, +zapisuje jako nowy PSD w folderze _gotowe. + +Parametry: +- imiona (format "{zenskie} i {meskie}") +- data (DD.MM.RRRR) + +Wymaga: uruchomiony Adobe Photoshop, pakiet photoshop-python-api. + +Uzycie: + python magnes_slub_kwiaty_wzor5.py --imie_zenskie "Marta" --imie_meskie "Mateusz" --data "26.04.2025" --klient "Marta Kowalska" +""" + +import argparse +import os +import photoshop.api as ps + + +PROJEKT_DIR = os.path.join( + r"d:\pomysloweprezenty.pl\projekty\ślub - magnesy", + "Magnes podziękowanie dla gości weselnych na plastrze brzozy - Kwiaty Wzór 5", +) +SZABLON_PATH = os.path.join( + PROJEKT_DIR, + "Magnes podziękowanie dla gości weselnych na plastrze brzozy - Kwiaty Wzór 5.psd", +) +GOTOWE_DIR = os.path.join(PROJEKT_DIR, "_gotowe") + +TEKST_GROUP = "Tekst" + + +def set_text(layer, new_text): + """Zmienia tekst zachowujac srodek bounding boxa (centrowane warstwy).""" + b = [float(x) for x in layer.bounds] + cx, cy = (b[0] + b[2]) / 2, (b[1] + b[3]) / 2 + layer.textItem.contents = new_text + b2 = [float(x) for x in layer.bounds] + cx2, cy2 = (b2[0] + b2[2]) / 2, (b2[1] + b2[3]) / 2 + dx, dy = cx - cx2, cy - cy2 + if dx or dy: + layer.translate(dx, dy) + + +def generate(imie_zenskie, imie_meskie, data, klient): + """Generuje PSD z podmienionymi danymi.""" + os.makedirs(GOTOWE_DIR, exist_ok=True) + output_path = os.path.join(GOTOWE_DIR, f"{klient}.psd") + + app = ps.Application() + doc = app.open(SZABLON_PATH) + print(f"Otwarto szablon: {doc.name}") + + try: + tekst_group = doc.layerSets[TEKST_GROUP] + + replacements = { + "imiona": f"{imie_zenskie} i {imie_meskie}", + "data": data, + } + + for layer_name, new_text in replacements.items(): + layer = tekst_group.artLayers[layer_name] + old_text = layer.textItem.contents + set_text(layer, new_text) + print(f" {layer_name}: \"{old_text}\" -> \"{new_text}\"") + + psd_opts = ps.PhotoshopSaveOptions() + doc.saveAs(output_path, psd_opts, True) + print(f"Zapisano: {output_path}") + finally: + app.activeDocument.close(ps.SaveOptions.DoNotSaveChanges) + print("Gotowe!") + + return output_path + + +def main(): + parser = argparse.ArgumentParser( + description="Generator PSD - Magnes slub plaster brzozy (Kwiaty Wzor 5)" + ) + parser.add_argument("--imie_zenskie", required=True, help="Imie zenskie (np. Marta)") + parser.add_argument("--imie_meskie", required=True, help="Imie meskie (np. Mateusz)") + parser.add_argument("--data", required=True, help="Data uroczystosci (np. 26.04.2025)") + parser.add_argument("--klient", required=True, help="Nazwa pliku wyjsciowego") + + args = parser.parse_args() + generate( + imie_zenskie=args.imie_zenskie, + imie_meskie=args.imie_meskie, + data=args.data, + klient=args.klient, + ) + + +if __name__ == "__main__": + main()