feat(recipes): add RecipeCard component with compact/full image variants
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
66
frontend/src/lib/recipes/RecipeCard.svelte
Normal file
66
frontend/src/lib/recipes/RecipeCard.svelte
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
type RecipeSummary = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
cookTimeMin?: number;
|
||||||
|
effort?: string;
|
||||||
|
heroImageUrl?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
let { recipe, compact = false }: { recipe: RecipeSummary; compact?: boolean } = $props();
|
||||||
|
|
||||||
|
let metadata = $derived(
|
||||||
|
[
|
||||||
|
recipe.cookTimeMin != null ? `${recipe.cookTimeMin} Min` : null,
|
||||||
|
recipe.effort ?? null
|
||||||
|
]
|
||||||
|
.filter(Boolean)
|
||||||
|
.join(' · ')
|
||||||
|
);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<a
|
||||||
|
href="/recipes/{recipe.id}"
|
||||||
|
class="block rounded-[var(--radius-md)] overflow-hidden border border-[var(--color-border)] bg-[var(--color-surface)] hover:shadow-[var(--shadow-card)]"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
data-testid="image-area"
|
||||||
|
class="w-full overflow-hidden {compact ? 'h-[64px]' : 'h-[100px]'}"
|
||||||
|
>
|
||||||
|
{#if recipe.heroImageUrl}
|
||||||
|
<img src={recipe.heroImageUrl} alt={recipe.name} class="w-full h-full object-cover" />
|
||||||
|
{:else}
|
||||||
|
<div
|
||||||
|
data-testid="image-placeholder"
|
||||||
|
class="w-full h-full bg-[var(--color-border)] flex items-center justify-center"
|
||||||
|
>
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="1.5"
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
aria-hidden="true"
|
||||||
|
class="text-[var(--color-text-muted)] opacity-50"
|
||||||
|
>
|
||||||
|
<!-- plate -->
|
||||||
|
<circle cx="12" cy="13" r="6" />
|
||||||
|
<path d="M12 7V5" />
|
||||||
|
<!-- fork tines -->
|
||||||
|
<path d="M8 3v3c0 1.1.9 2 2 2h4" />
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="px-2 py-1.5">
|
||||||
|
<p class="font-medium text-[13px] text-[var(--color-text)] truncate">{recipe.name}</p>
|
||||||
|
{#if metadata}
|
||||||
|
<p class="text-[11px] text-[var(--color-text-muted)]">{metadata}</p>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
62
frontend/src/lib/recipes/RecipeCard.test.ts
Normal file
62
frontend/src/lib/recipes/RecipeCard.test.ts
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
import { describe, it, expect, vi } from 'vitest';
|
||||||
|
import { render, screen } from '@testing-library/svelte';
|
||||||
|
import { userEvent } from '@testing-library/user-event';
|
||||||
|
import RecipeCard from './RecipeCard.svelte';
|
||||||
|
|
||||||
|
const mockRecipe = {
|
||||||
|
id: 'recipe-1',
|
||||||
|
name: 'Spaghetti Bolognese',
|
||||||
|
cookTimeMin: 30,
|
||||||
|
effort: 'Easy',
|
||||||
|
heroImageUrl: undefined
|
||||||
|
};
|
||||||
|
|
||||||
|
describe('RecipeCard', () => {
|
||||||
|
it('renders the recipe name', () => {
|
||||||
|
render(RecipeCard, { props: { recipe: mockRecipe } });
|
||||||
|
expect(screen.getByText('Spaghetti Bolognese')).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('renders cook time when present', () => {
|
||||||
|
render(RecipeCard, { props: { recipe: mockRecipe } });
|
||||||
|
expect(screen.getByText(/30/)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('renders effort when present', () => {
|
||||||
|
render(RecipeCard, { props: { recipe: mockRecipe } });
|
||||||
|
expect(screen.getByText(/easy/i)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('shows placeholder when no heroImageUrl', () => {
|
||||||
|
render(RecipeCard, { props: { recipe: { ...mockRecipe, heroImageUrl: undefined } } });
|
||||||
|
expect(screen.queryByRole('img')).not.toBeInTheDocument();
|
||||||
|
expect(document.querySelector('[data-testid="image-placeholder"]')).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('shows image when heroImageUrl is provided', () => {
|
||||||
|
render(RecipeCard, {
|
||||||
|
props: { recipe: { ...mockRecipe, heroImageUrl: '/uploads/test.jpg' } }
|
||||||
|
});
|
||||||
|
const img = screen.getByRole('img');
|
||||||
|
expect(img).toHaveAttribute('src', '/uploads/test.jpg');
|
||||||
|
expect(img).toHaveAttribute('alt', 'Spaghetti Bolognese');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('wraps in a link to the recipe detail page', () => {
|
||||||
|
render(RecipeCard, { props: { recipe: mockRecipe } });
|
||||||
|
const link = screen.getByRole('link');
|
||||||
|
expect(link).toHaveAttribute('href', '/recipes/recipe-1');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('applies compact image height when compact prop is true', () => {
|
||||||
|
render(RecipeCard, { props: { recipe: mockRecipe, compact: true } });
|
||||||
|
const imageArea = document.querySelector('[data-testid="image-area"]');
|
||||||
|
expect(imageArea?.className).toContain('h-[64px]');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('applies full image height when compact prop is false', () => {
|
||||||
|
render(RecipeCard, { props: { recipe: mockRecipe, compact: false } });
|
||||||
|
const imageArea = document.querySelector('[data-testid="image-area"]');
|
||||||
|
expect(imageArea?.className).toContain('h-[100px]');
|
||||||
|
});
|
||||||
|
});
|
||||||
Reference in New Issue
Block a user