---
phase: 18-global-items-public-profiles
plan: 04
type: execute
wave: 3
depends_on: ["18-02"]
files_modified:
- src/client/hooks/useGlobalItems.ts
- src/client/routes/global-items/index.tsx
- src/client/routes/global-items/$globalItemId.tsx
- src/client/components/GlobalItemCard.tsx
- src/client/components/LinkToGlobalItem.tsx
- src/client/lib/api.ts
autonomous: false
requirements: [GLOB-03, GLOB-04, GLOB-05]
must_haves:
truths:
- "User can browse a global catalog page listing all global items with brand, model, category"
- "User can search the global catalog by typing a query and results filter in real-time"
- "User can click a global item to see its detail page with specs, image, description, and owner count"
- "User can link a personal collection item to a global item via a UI control"
artifacts:
- path: "src/client/hooks/useGlobalItems.ts"
provides: "useGlobalItems, useGlobalItem, useLinkItem, useUnlinkItem hooks"
exports: ["useGlobalItems", "useGlobalItem"]
- path: "src/client/routes/global-items/index.tsx"
provides: "Global catalog browse/search page"
min_lines: 40
- path: "src/client/routes/global-items/$globalItemId.tsx"
provides: "Global item detail page with owner count"
min_lines: 30
- path: "src/client/components/GlobalItemCard.tsx"
provides: "Card component for global item in list"
min_lines: 20
key_links:
- from: "src/client/routes/global-items/index.tsx"
to: "src/client/hooks/useGlobalItems.ts"
via: "useGlobalItems hook"
pattern: "useGlobalItems"
- from: "src/client/hooks/useGlobalItems.ts"
to: "/api/global-items"
via: "apiGet fetch"
pattern: "apiGet.*global-items"
---
Build the global item catalog client: search/browse page, detail page with owner count, item cards, and link-to-global-item UI. Users can discover gear and connect their personal items to the shared catalog.
Purpose: Delivers the client-side experience for GLOB-03 (search), GLOB-04 (linking), and GLOB-05 (detail with owner count).
Output: useGlobalItems hook, catalog browse page, detail page, GlobalItemCard, LinkToGlobalItem component
@$HOME/.claude/get-shit-done/workflows/execute-plan.md
@$HOME/.claude/get-shit-done/templates/summary.md
@.planning/PROJECT.md
@.planning/ROADMAP.md
@.planning/phases/18-global-items-public-profiles/18-CONTEXT.md
@.planning/phases/18-global-items-public-profiles/18-RESEARCH.md
@.planning/phases/18-global-items-public-profiles/18-02-SUMMARY.md
@src/client/hooks/useItems.ts
@src/client/lib/api.ts
@src/client/routes/collection/index.tsx
GET /api/global-items?q=string -> GlobalItem[]
GET /api/global-items/:id -> { ...GlobalItem, ownerCount: number }
POST /api/items/:id/link { globalItemId: number } -> ItemGlobalLink (201)
DELETE /api/items/:id/link -> 200
type GlobalItem = { id, brand, model, category, weightGrams, priceCents, imageUrl, description, createdAt }
type GlobalItemWithOwnerCount = GlobalItem & { ownerCount: number }
Task 1: Global item hooks and catalog pages
src/client/hooks/useGlobalItems.ts, src/client/routes/global-items/index.tsx, src/client/routes/global-items/$globalItemId.tsx, src/client/components/GlobalItemCard.tsx
src/client/hooks/useItems.ts, src/client/lib/api.ts, src/client/routes/collection/index.tsx
**useGlobalItems.ts** hook: Create at `src/client/hooks/useGlobalItems.ts`. Follow existing hook pattern from useItems.ts.
1. `useGlobalItems(query?: string)` — `useQuery` with key `["global-items", query]`, fetches `apiGet("/api/global-items" + (query ? "?q=" + encodeURIComponent(query) : ""))`. Use 300ms debounced query value for search (or accept debounce at the component level).
2. `useGlobalItem(id: number | null)` — `useQuery` with key `["global-items", id]`, fetches `apiGet("/api/global-items/${id}")`, `enabled: id != null`.
3. `useLinkItem()` — `useMutation` calling `apiPost("/api/items/${itemId}/link", { globalItemId })`. On success, invalidate `["items"]` and `["global-items"]` query keys.
4. `useUnlinkItem()` — `useMutation` calling `apiDelete("/api/items/${itemId}/link")`. On success, invalidate same keys.
**GlobalItemCard.tsx**: Create at `src/client/components/GlobalItemCard.tsx`. Card displaying brand, model, category badge, weight (formatted as g/kg), price (formatted from cents). Links to `/global-items/${item.id}` detail page. Show image thumbnail if imageUrl exists. Light/airy Tailwind styling matching existing collection cards.
**global-items/index.tsx**: Catalog browse/search page.
- Search input at top with placeholder "Search gear by brand or model..."
- Debounce input by 300ms before passing to `useGlobalItems(debouncedQuery)`
- Grid of GlobalItemCard components (responsive: 1 col mobile, 2 cols md, 3 cols lg)
- Loading skeleton while fetching
- Empty state: "No items found" or "Search the global gear catalog"
- TanStack Router: `createFileRoute("/global-items/")` with component export
**global-items/$globalItemId.tsx**: Detail page.
- TanStack Router: `createFileRoute("/global-items/$globalItemId")` with params loader
- Fetch single item with `useGlobalItem(Number(globalItemId))`
- Display: brand, model, category, weight, price, description, image (full size)
- Show owner count badge: "{N} users own this" or "Be the first to add this"
- Back link to catalog
bun run lint 2>&1 | tail -5
- grep -q "useGlobalItems" src/client/hooks/useGlobalItems.ts
- grep -q "useGlobalItem" src/client/hooks/useGlobalItems.ts
- grep -q "useLinkItem" src/client/hooks/useGlobalItems.ts
- test -f src/client/routes/global-items/index.tsx
- test -f "src/client/routes/global-items/\$globalItemId.tsx"
- test -f src/client/components/GlobalItemCard.tsx
Global catalog page shows searchable grid of items. Detail page shows specs, image, and owner count. Hooks handle all data fetching and mutations. Lint passes.
Task 2: Link-to-global-item UI in collection
src/client/components/LinkToGlobalItem.tsx
src/client/routes/collection/index.tsx, src/client/hooks/useGlobalItems.ts
**LinkToGlobalItem.tsx**: Create at `src/client/components/LinkToGlobalItem.tsx`. Per D-04 and D-18.
A component that allows linking a user's personal item to a global catalog entry. Design as a small modal/popover or inline search:
1. Trigger: A "Link to catalog" button shown on item detail or edit view. If already linked, show "Linked to {brand} {model}" with an unlink option.
2. When triggered, show a search input that calls `useGlobalItems(query)` with debounce.
3. Display matching global items as clickable options.
4. On select, call `useLinkItem()` mutation with itemId and globalItemId.
5. Show success state: linked item name with a link to the global item detail page.
6. Unlink: If already linked, show the linked global item with an "Unlink" button that calls `useUnlinkItem()`.
Keep it simple — a dropdown/combobox pattern works well. Use Tailwind for styling. Match the light/airy aesthetic of existing components.
Wire this component into the item edit form or item detail view at the appropriate place (after the existing form fields, or as a separate section below item details).
bun run lint 2>&1 | tail -5
- grep -q "LinkToGlobalItem" src/client/components/LinkToGlobalItem.tsx
- grep -q "useLinkItem\|linkItem" src/client/components/LinkToGlobalItem.tsx
- grep -q "useUnlinkItem\|unlinkItem" src/client/components/LinkToGlobalItem.tsx
Users can search the global catalog from within their item view, link/unlink their item, and see the current link status. Lint passes.
Task 3: Verify global catalog UI
none
Human verification of the global item catalog UI. Review what was built: browse page with search, detail page with owner count, and link/unlink from collection items.
Steps to verify:
1. Start dev server: `bun run dev`
2. Navigate to `/global-items` — should see catalog with seed items in a grid
3. Type "revelate" in search — should filter to matching items
4. Click a global item — detail page shows brand, model, specs, "0 users own this"
5. Go to your collection, open an item, find "Link to catalog" control
6. Search for a global item and link it — should show linked status
7. Return to global item detail — owner count should now show "1 user owns this"
8. Unlink the item — owner count returns to 0
bun run build 2>&1 | tail -3
User approves global catalog UI: search works, detail page shows owner count, link/unlink flow is functional.
- `bun run lint` passes
- `bun run build` succeeds (client builds with new routes)
- Visual verification of catalog page, search, detail page, and link/unlink flow
Global catalog is browsable and searchable. Item detail shows owner count. Users can link/unlink personal items to global entries. All pages render correctly with Tailwind styling.