diff --git a/assets/css/admin/components.css b/assets/css/admin/components.css index e741568..b2fbd4c 100644 --- a/assets/css/admin/components.css +++ b/assets/css/admin/components.css @@ -1662,6 +1662,139 @@ font-size: 0.8125rem; } +.image-field-actions { + display: flex; + gap: 0.375rem; +} + +.image-field-remove-btn { + color: var(--t-accent); +} + +.image-field-choose-btn { + width: 100%; + justify-content: center; +} + +/* ── Image picker modal ─────────────────────────────────────────── */ + +.image-picker-overlay { + position: fixed; + inset: 0; + background: rgba(0, 0, 0, 0.3); + z-index: 50; + display: flex; + align-items: flex-end; + justify-content: center; + + @media (min-width: 40em) { + align-items: center; + } +} + +.image-picker { + background: var(--t-surface-base); + border-radius: 0.75rem 0.75rem 0 0; + width: 100%; + max-height: 80vh; + overflow-y: auto; + padding: 1rem; + display: flex; + flex-direction: column; + + @media (min-width: 40em) { + border-radius: 0.75rem; + max-width: 32rem; + } +} + +.image-picker-header { + display: flex; + align-items: center; + justify-content: space-between; + margin-bottom: 0.75rem; + + & h3 { + font-size: 1rem; + font-weight: 600; + } +} + +.image-picker-search { + width: 100%; + margin-bottom: 0.75rem; +} + +.image-picker-grid { + display: grid; + grid-template-columns: repeat(2, 1fr); + gap: 0.5rem; + overflow-y: auto; + + @media (min-width: 40em) { + grid-template-columns: repeat(3, 1fr); + } +} + +.image-picker-item { + display: flex; + flex-direction: column; + align-items: center; + gap: 0.375rem; + padding: 0.375rem; + border: 1px solid var(--t-border-default); + border-radius: 0.375rem; + background: none; + cursor: pointer; + text-align: center; + color: inherit; + transition: background 100ms; + overflow: clip; + min-width: 0; + + @media (hover: hover) { + &:hover { + background: var(--t-surface-sunken); + } + } +} + +.image-picker-item-thumb, +.image-picker-item-svg { + width: 100%; + aspect-ratio: 1; + border-radius: 0.25rem; + background: var(--t-surface-sunken); +} + +.image-picker-item-thumb { + object-fit: cover; +} + +.image-picker-item-svg { + display: flex; + align-items: center; + justify-content: center; + color: var(--t-text-primary); + opacity: 0.5; +} + +.image-picker-item-name { + font-size: 0.6875rem; + overflow: hidden; + text-overflow: ellipsis; + white-space: nowrap; + max-width: 100%; +} + +.image-picker-empty { + grid-column: 1 / -1; + text-align: center; + padding: 1.5rem; + color: color-mix(in oklch, var(--t-text-primary) 50%, transparent); + font-size: 0.8125rem; +} + /* ═══════════════════════════════════════════════════════════════════ Media library ═══════════════════════════════════════════════════════════════════ */ @@ -1913,13 +2046,27 @@ padding-top: 0.75rem; } +.media-detail-scrim { + display: none; +} + @media (max-width: 47.99em) { .media-grid { grid-template-columns: repeat(auto-fill, minmax(120px, 1fr)); } + .media-detail-scrim { + display: block; + position: fixed; + inset: 0; + background: rgba(0, 0, 0, 0.5); + z-index: 39; + animation: media-scrim-fade 0.2s ease-out; + } + .media-detail { position: fixed; + top: auto; bottom: 0; left: 0; right: 0; @@ -1927,9 +2074,32 @@ min-width: unset; max-height: 70vh; border-radius: 0.75rem 0.75rem 0 0; - box-shadow: 0 -4px 24px rgba(0, 0, 0, 0.15); + box-shadow: 0 -4px 24px rgba(0, 0, 0, 0.25); z-index: 40; animation: media-sheet-up 0.25s ease-out; + padding: 0.75rem 1rem; + gap: 0.5rem; + } + + .media-detail-preview { + display: none; + } + + .media-detail-meta { + font-size: 0.75rem; + gap: 0.125rem 0.5rem; + } + + .media-detail-form { + gap: 0.25rem; + } + + .media-detail-form .admin-fieldset { + margin-bottom: 0; + } + + .media-detail-form .admin-label { + font-size: 0.75rem; } } @@ -1938,4 +2108,9 @@ to { transform: translateY(0); } } +@keyframes media-scrim-fade { + from { opacity: 0; } + to { opacity: 1; } +} + } /* @layer admin */ diff --git a/lib/berrypod_web/components/block_editor_components.ex b/lib/berrypod_web/components/block_editor_components.ex index 5ac60c3..c634eb6 100644 --- a/lib/berrypod_web/components/block_editor_components.ex +++ b/lib/berrypod_web/components/block_editor_components.ex @@ -233,21 +233,37 @@ defmodule BerrypodWeb.BlockEditorComponents do {@image.alt} - <% else %> -
- <.icon name="hero-photo" class="size-6" /> - No image selected +
+ +
+ <% else %> + <% end %> -
""" @@ -449,4 +465,81 @@ defmodule BerrypodWeb.BlockEditorComponents do """ end + + # ── Image picker ────────────────────────────────────────────── + + attr :images, :list, required: true + attr :search, :string, required: true + attr :event_prefix, :string, default: "" + + def image_picker(assigns) do + search = String.downcase(assigns.search) + + filtered = + if search == "" do + assigns.images + else + Enum.filter(assigns.images, fn img -> + String.contains?(String.downcase(img.filename || ""), search) or + String.contains?(String.downcase(img.alt || ""), search) + end) + end + + assigns = assign(assigns, :filtered_images, filtered) + + ~H""" +
+
+
+

Choose image

+ +
+ + + +
+ + +

+ No images found. +

+
+
+
+ """ + end end diff --git a/lib/berrypod_web/live/admin/media.ex b/lib/berrypod_web/live/admin/media.ex index 5759321..5ad902f 100644 --- a/lib/berrypod_web/live/admin/media.ex +++ b/lib/berrypod_web/live/admin/media.ex @@ -330,6 +330,7 @@ defmodule BerrypodWeb.Admin.Media do <%!-- detail panel --%> +
<%!-- Backdrop: tapping the page dismisses the sidebar --%> @@ -1029,7 +1037,13 @@ defmodule BerrypodWeb.PageRenderer do if image.is_svg do "/image_cache/#{image.id}.webp" else - "/image_cache/#{image.id}-800.webp" + # Pick the largest variant that was actually generated + width = + image.source_width + |> Berrypod.Images.Optimizer.applicable_widths() + |> List.last() + + "/image_cache/#{image.id}-#{width || 400}.webp" end {url, image.alt || image.filename}