From a64a7e39cf610851fe8ca39bb9edafecf9e586ae Mon Sep 17 00:00:00 2001 From: Till JS Date: Thu, 23 Apr 2026 14:01:40 +0200 Subject: [PATCH] feat(profile): UI for me-images management at /profile/me-images (M2) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit M2 of docs/plans/me-images-and-reference-generation.md — the Settings surface that sits on top of the M1 data layer. Users can now upload a Face and a Fullbody reference into two primary slots, toss extra references into a grid, and toggle each image's "KI darf nutzen" flag individually. Route placement: /profile/me-images (not /settings/me-images as the plan originally proposed). The repo convention is per-module subroutes (/todo/settings, /invoices/settings, …) — there is no global /settings namespace to hang this off. Plan doc updated accordingly. - MeImageUploadZone: drag-and-drop + file-picker, pattern from picture/ListView but refactored into a reusable component. Fires onFiles(File[]) so the parent decides kind + slot. - MeImageSlotCard: large card for Face / Fullbody primary slots. When filled it shows the portrait + the image's AI-toggle + delete + a compact "Neues Bild setzen" replacement zone. When empty it collapses into a large drop-zone. - MeImageTile: grid tile for everything that isn't currently holding a primary slot — thumbnail, kind badge, Robot-AI-toggle, Star primary-promotion (only enabled for kinds that map to a slot), Trash delete. - MeImagesView: orchestrates queries (useImageByPrimary for each slot + useAllMeImages for the rest), upload flow (readDimensions → uploadMeImageFile → store.createMeImage → optional setPrimary in the same tick), and the three write actions (toggleAi, togglePrimary, delete). Dropping a file on a slot drop-zone both uploads and claims the slot, so the old holder automatically falls into the grid. - Client: profile/api/me-images.ts wraps the M1 endpoint with authStore.getValidToken() → Bearer header and a small readImageDimensions helper that exposes natural width/height synchronously (mana-media reports them later but we want them for the Dexie row's first write). - Discoverability: profile ListView "Konto" tab gains a "Meine Bilder" action button that navigates to the new route with a one-line hint. Still open (later commits): the hard-migration that rewrites auth.users.image → meImages(primaryFor='avatar'), the global aiUsesReferenceImages kill-switch (lives on profile singleton), and the Picture-generator's Reference picker (M4, rides on top of M3's backend endpoint). Co-Authored-By: Claude Opus 4.7 (1M context) --- .../src/lib/modules/profile/ListView.svelte | 15 +- .../lib/modules/profile/MeImagesView.svelte | 194 ++++++++++++++++++ .../src/lib/modules/profile/api/me-images.ts | 61 ++++++ .../profile/components/MeImageSlotCard.svelte | 103 ++++++++++ .../profile/components/MeImageTile.svelte | 87 ++++++++ .../components/MeImageUploadZone.svelte | 89 ++++++++ .../(app)/profile/me-images/+page.svelte | 12 ++ .../me-images-and-reference-generation.md | 8 +- 8 files changed, 564 insertions(+), 5 deletions(-) create mode 100644 apps/mana/apps/web/src/lib/modules/profile/MeImagesView.svelte create mode 100644 apps/mana/apps/web/src/lib/modules/profile/api/me-images.ts create mode 100644 apps/mana/apps/web/src/lib/modules/profile/components/MeImageSlotCard.svelte create mode 100644 apps/mana/apps/web/src/lib/modules/profile/components/MeImageTile.svelte create mode 100644 apps/mana/apps/web/src/lib/modules/profile/components/MeImageUploadZone.svelte create mode 100644 apps/mana/apps/web/src/routes/(app)/profile/me-images/+page.svelte diff --git a/apps/mana/apps/web/src/lib/modules/profile/ListView.svelte b/apps/mana/apps/web/src/lib/modules/profile/ListView.svelte index 6e5d0468e..5da06a545 100644 --- a/apps/mana/apps/web/src/lib/modules/profile/ListView.svelte +++ b/apps/mana/apps/web/src/lib/modules/profile/ListView.svelte @@ -211,6 +211,12 @@