import { css, html } from 'lit'; import { customElement, property } from 'lit/decorators.js'; import type { ListCardsResponse } from '../types/index.js'; import { RoxyDataElement } from '../utils/base-element.js'; import { baseStyles } from '../utils/base-styles.js'; import { capitalize } from '../utils/string.js'; /** A single card row from the catalog response. Kept spec-derived so the tile never reads a field the API does not return. */ type CatalogCard = ListCardsResponse['cards'][number]; /** * Tarot catalog. Renders GET /tarot/cards as a responsive gallery of the deck: each tile carries the Rider-Waite-Smith artwork, the card name, and an arcana/suit caption. Filter the deck server-side (arcana, suit, number, paging) and pass the page response; the component renders whatever cards it carries. Pairs with `` for a single-card detail view and `` for readings. */ @customElement('roxy-tarot-catalog') export class RoxyTarotCatalog extends RoxyDataElement { static styles = [ baseStyles, css` .wrap { display: grid; gap: var(--roxy-space-md, 1rem); } .head { display: flex; align-items: baseline; justify-content: space-between; gap: var(--roxy-space-sm, 0.5rem); flex-wrap: wrap; } .title { margin: 0; font-size: var(--roxy-text-lg, 1.125rem); font-weight: var(--roxy-weight-bold, 600); color: var(--roxy-fg, #0a0a0a); } .count { color: var(--roxy-muted, #71717a); font-size: var(--roxy-text-sm, 0.875rem); } .grid { display: grid; grid-template-columns: repeat(auto-fill, minmax(7rem, 1fr)); gap: var(--roxy-space-md, 1rem); margin: 0; padding: 0; list-style: none; } .tile { display: grid; gap: var(--roxy-space-xs, 0.25rem); background: var(--roxy-surface, #fff); border: 1px solid var(--roxy-border, #e4e4e7); border-radius: var(--roxy-radius-md, 8px); padding: var(--roxy-space-sm, 0.5rem); box-shadow: var(--roxy-shadow-sm); } .art { aspect-ratio: 2 / 3; width: 100%; border-radius: var(--roxy-radius-sm, 4px); object-fit: cover; background: color-mix(in srgb, var(--roxy-border, #e4e4e7) 35%, transparent); } .name { margin: 0; font-size: var(--roxy-text-sm, 0.875rem); font-weight: var(--roxy-weight-bold, 600); color: var(--roxy-fg, #0a0a0a); } .meta { margin: 0; font-size: var(--roxy-text-xs, 0.75rem); color: var(--roxy-muted, #71717a); } `, ]; /** * Override the auto-derived gallery heading. Empty by default, in which case the heading is "Tarot deck". */ @property({ type: String, reflect: true }) heading = ''; protected renderEmpty() { return html`
No cards
`; } protected renderData(d: ListCardsResponse) { const cards = d.cards ?? []; if (cards.length === 0) return this.renderEmpty(); const title = this.heading || 'Tarot deck'; const total = typeof d.total === 'number' ? d.total : cards.length; return html`

${title}

${total} ${total === 1 ? 'card' : 'cards'}
    ${cards.map( (c) => html`
  • ${ c.imageUrl ? html`${c.name` : html`` }

    ${c.name}

    ${cardMeta(c)}

  • `, )}
`; } } /** * Caption line for a catalog tile. Minor Arcana cards name their suit (`Minor · Cups`); Major Arcana cards read `Major Arcana`. Both derive only from the spec `arcana` and `suit` fields. */ function cardMeta(c: CatalogCard): string { if (c.suit) return `${capitalize(c.arcana)} · ${capitalize(c.suit)}`; return `${capitalize(c.arcana)} Arcana`; } declare global { interface HTMLElementTagNameMap { 'roxy-tarot-catalog': RoxyTarotCatalog; } }