feat: ArticleCard component

This commit is contained in:
achmad
2026-05-28 22:29:19 +07:00
parent 6ba2c8b932
commit 50a4e63e2a
3 changed files with 103 additions and 11 deletions
+43
View File
@@ -0,0 +1,43 @@
import { describe, it, expect } from 'vitest'
import { render, screen } from '@testing-library/react'
import ArticleCard from '@/components/article/ArticleCard'
import type { Article } from '@/lib/types'
const mockArticle: Article = {
id: '1',
title: 'Frieren Season 2 Officially Announced',
slug: 'frieren-season-2-announced',
status: 'published',
content: null,
excerpt: 'The beloved series returns.',
featured_image: null,
published_at: '2026-05-28T00:00:00Z',
is_featured: false,
seo_title: null,
seo_description: null,
category: { id: '1', name: 'Anime', slug: 'anime', description: null },
tags: [],
}
describe('ArticleCard', () => {
it('renders the article title', () => {
render(<ArticleCard article={mockArticle} />)
expect(screen.getByText('Frieren Season 2 Officially Announced')).toBeInTheDocument()
})
it('renders the category name', () => {
render(<ArticleCard article={mockArticle} />)
expect(screen.getByText('Anime')).toBeInTheDocument()
})
it('links to the correct article URL', () => {
render(<ArticleCard article={mockArticle} />)
const link = screen.getByRole('link')
expect(link).toHaveAttribute('href', '/anime/frieren-season-2-announced')
})
it('renders the Read -> CTA', () => {
render(<ArticleCard article={mockArticle} />)
expect(screen.getByText('Read →')).toBeInTheDocument()
})
})
+42
View File
@@ -0,0 +1,42 @@
import Image from 'next/image'
import Link from 'next/link'
import { getAssetUrl } from '@/lib/directus'
import type { Article } from '@/lib/types'
interface Props {
article: Article
}
export default function ArticleCard({ article }: Props) {
const href = `/${article.category.slug}/${article.slug}`
return (
<Link
href={href}
className="group block bg-bg-card border border-border rounded-md overflow-hidden hover:border-[#3a4560] transition-colors"
>
<div className="relative aspect-video bg-bg-elevated overflow-hidden">
{article.featured_image ? (
<Image
src={getAssetUrl(article.featured_image, { width: 640, quality: 80 })}
alt={article.title}
fill
className="object-cover"
sizes="(max-width: 768px) 100vw, (max-width: 1200px) 50vw, 33vw"
/>
) : (
<div className="w-full h-full bg-bg-elevated" />
)}
</div>
<div className="p-3">
<p className="text-[10px] font-semibold text-text-muted uppercase tracking-widest mb-1.5">
{article.category.name}
</p>
<h3 className="text-sm font-semibold text-text-primary leading-snug mb-3 line-clamp-2 group-hover:text-accent transition-colors">
{article.title}
</h3>
<span className="text-xs font-semibold text-accent">Read </span>
</div>
</Link>
)
}
+16 -9
View File
@@ -8,12 +8,19 @@ import {
} from '@directus/sdk'
import type { Article, Category, SiteSettings } from './types'
const directus = createDirectus(process.env.DIRECTUS_URL!)
let _client: ReturnType<typeof createDirectus> | null = null
function getClient() {
if (!_client) {
_client = createDirectus(process.env.DIRECTUS_URL!)
.with(staticToken(process.env.DIRECTUS_TOKEN!))
.with(rest())
}
return _client
}
export async function getAllCategories(): Promise<Category[]> {
return directus.request(
return getClient().request(
readItems('categories', {
fields: ['id', 'name', 'slug', 'description'],
sort: ['name'],
@@ -22,7 +29,7 @@ export async function getAllCategories(): Promise<Category[]> {
}
export async function getCategoryBySlug(slug: string): Promise<Category | null> {
const results = await directus.request(
const results = await getClient().request(
readItems('categories', {
fields: ['id', 'name', 'slug', 'description'],
filter: { slug: { _eq: slug } },
@@ -42,7 +49,7 @@ export async function getArticles(options: {
if (categorySlug) {
filter['category'] = { slug: { _eq: categorySlug } }
}
return directus.request(
return getClient().request(
readItems('articles', {
fields: [
'id', 'title', 'slug', 'status', 'excerpt', 'featured_image',
@@ -58,7 +65,7 @@ export async function getArticles(options: {
}
export async function getArticleBySlug(slug: string): Promise<Article | null> {
const results = await directus.request(
const results = await getClient().request(
readItems('articles', {
fields: [
'id', 'title', 'slug', 'status', 'content', 'excerpt',
@@ -79,7 +86,7 @@ export async function getArticlePathById(id: string): Promise<{
category: { slug: string }
} | null> {
try {
return await directus.request(
return await getClient().request(
readItem('articles', id, {
fields: ['slug', 'category.slug'],
})
@@ -93,7 +100,7 @@ export async function getRelatedArticles(
categorySlug: string,
excludeSlug: string,
): Promise<Article[]> {
return directus.request(
return getClient().request(
readItems('articles', {
fields: [
'id', 'title', 'slug', 'excerpt', 'featured_image',
@@ -111,7 +118,7 @@ export async function getRelatedArticles(
}
export async function getSiteSettings(): Promise<SiteSettings> {
return directus.request(
return getClient().request(
readSingleton('site_settings', {
fields: [
'id', 'site_name',
@@ -125,7 +132,7 @@ export async function getSiteSettings(): Promise<SiteSettings> {
}
export async function searchArticles(query: string): Promise<Article[]> {
return directus.request(
return getClient().request(
readItems('articles', {
fields: ['id', 'title', 'slug', 'category.slug', 'category.name'],
search: query,