테스트 인프라: - Jest 30 + jest-environment-jsdom + RTL + jest-dom matchers - next/jest로 SWC/Next.js 자동 통합 - jest.config.ts (setupFilesAfterEnv) + jest.setup.ts - npm scripts: test, test:watch - 샘플 테스트 3개, 13/13 통과: - i18n/config: isLocale + detectBrowserLocale (5 케이스) - Stars 컴포넌트: 별점/aria/clamp/showNumber (5 케이스) - admin-utils: getAdminToken + authHeaders (4 케이스) ARIA Tabs (MyReviewsList): - role=tablist + tab + aria-selected + aria-controls + tabIndex - panel에 role=tabpanel + aria-labelledby next/image: - next.config.ts remotePatterns: lh3.googleusercontent.com / i.ytimg.com / yt3.ggpht.com - ReviewSection의 user_avatar_url에 명시적 eslint-disable + 사유 후속(별도): 전체 컴포넌트 테스트 점진 추가, 백엔드 JUnit 인프라, E2E (Playwright), CI 통합 설계서: docs/design/343-frontend-test-infra/README.md Refs: #343
160 lines
5.6 KiB
TypeScript
160 lines
5.6 KiB
TypeScript
"use client";
|
|
|
|
import { useState } from "react";
|
|
import type { Review, Memo } from "@/lib/api";
|
|
import Icon from "@/components/Icon";
|
|
import Stars from "@/components/Stars";
|
|
|
|
interface MyReview extends Review {
|
|
restaurant_id: string;
|
|
restaurant_name: string | null;
|
|
}
|
|
|
|
interface MyMemo extends Memo {
|
|
restaurant_name: string | null;
|
|
}
|
|
|
|
interface MyReviewsListProps {
|
|
reviews: MyReview[];
|
|
memos: MyMemo[];
|
|
onClose: () => void;
|
|
onSelectRestaurant: (restaurantId: string) => void;
|
|
}
|
|
|
|
export default function MyReviewsList({
|
|
reviews,
|
|
memos,
|
|
onClose,
|
|
onSelectRestaurant,
|
|
}: MyReviewsListProps) {
|
|
const [tab, setTab] = useState<"reviews" | "memos">("reviews");
|
|
|
|
return (
|
|
<div className="p-4 space-y-3">
|
|
<div className="flex justify-between items-center">
|
|
<h2 className="font-bold text-lg">내 기록</h2>
|
|
<button
|
|
onClick={onClose}
|
|
className="text-gray-400 hover:text-gray-600"
|
|
>
|
|
<Icon name="close" size={18} />
|
|
</button>
|
|
</div>
|
|
|
|
{/* #343 — WAI-ARIA Tabs 패턴 */}
|
|
<div role="tablist" aria-label="내 활동" className="flex gap-1 border-b">
|
|
<button
|
|
role="tab"
|
|
id="tab-reviews"
|
|
aria-selected={tab === "reviews"}
|
|
aria-controls="panel-reviews"
|
|
tabIndex={tab === "reviews" ? 0 : -1}
|
|
onClick={() => setTab("reviews")}
|
|
className={`px-3 py-1.5 text-sm font-medium border-b-2 transition-colors ${
|
|
tab === "reviews"
|
|
? "border-brand-500 text-brand-600"
|
|
: "border-transparent text-gray-500 hover:text-gray-700"
|
|
}`}
|
|
>
|
|
<Icon name="rate_review" size={14} className="mr-1" />
|
|
리뷰 ({reviews.length})
|
|
</button>
|
|
<button
|
|
role="tab"
|
|
id="tab-memos"
|
|
aria-selected={tab === "memos"}
|
|
aria-controls="panel-memos"
|
|
tabIndex={tab === "memos" ? 0 : -1}
|
|
onClick={() => setTab("memos")}
|
|
className={`px-3 py-1.5 text-sm font-medium border-b-2 transition-colors ${
|
|
tab === "memos"
|
|
? "border-brand-500 text-brand-600"
|
|
: "border-transparent text-gray-500 hover:text-gray-700"
|
|
}`}
|
|
>
|
|
<Icon name="edit_note" size={14} className="mr-1" />
|
|
메모 ({memos.length})
|
|
</button>
|
|
</div>
|
|
|
|
{tab === "reviews" ? (
|
|
<div role="tabpanel" id="panel-reviews" aria-labelledby="tab-reviews">
|
|
{reviews.length === 0 ? (
|
|
<p className="text-sm text-gray-500 py-8 text-center">
|
|
아직 작성한 리뷰가 없습니다.
|
|
</p>
|
|
) : (
|
|
<div className="space-y-2">
|
|
{reviews.map((r) => (
|
|
<button
|
|
key={r.id}
|
|
onClick={() => onSelectRestaurant(r.restaurant_id)}
|
|
className="w-full text-left border rounded-lg p-3 hover:bg-gray-50 transition-colors"
|
|
>
|
|
<div className="flex items-center justify-between mb-1">
|
|
<span className="font-semibold text-sm truncate">
|
|
{r.restaurant_name || "알 수 없는 식당"}
|
|
</span>
|
|
<span className="text-sm shrink-0 ml-2 flex items-center gap-1">
|
|
<Stars rating={r.rating} />
|
|
<span className="text-gray-500">{r.rating}</span>
|
|
</span>
|
|
</div>
|
|
{r.review_text && (
|
|
<p className="text-xs text-gray-600 line-clamp-2">
|
|
{r.review_text}
|
|
</p>
|
|
)}
|
|
<div className="flex items-center gap-2 mt-1.5 text-[10px] text-gray-400">
|
|
{r.visited_at && <span>방문: {r.visited_at}</span>}
|
|
{r.created_at && <span>{r.created_at.slice(0, 10)}</span>}
|
|
</div>
|
|
</button>
|
|
))}
|
|
</div>
|
|
)}
|
|
</div>
|
|
) : (
|
|
<div role="tabpanel" id="panel-memos" aria-labelledby="tab-memos">
|
|
{memos.length === 0 ? (
|
|
<p className="text-sm text-gray-500 py-8 text-center">
|
|
아직 작성한 메모가 없습니다.
|
|
</p>
|
|
) : (
|
|
<div className="space-y-2">
|
|
{memos.map((m) => (
|
|
<button
|
|
key={m.id}
|
|
onClick={() => onSelectRestaurant(m.restaurant_id)}
|
|
className="w-full text-left border border-brand-200 rounded-lg p-3 bg-brand-50/30 hover:bg-brand-50 transition-colors"
|
|
>
|
|
<div className="flex items-center justify-between mb-1">
|
|
<span className="font-semibold text-sm truncate">
|
|
{m.restaurant_name || "알 수 없는 식당"}
|
|
</span>
|
|
{m.rating && (
|
|
<span className="text-sm shrink-0 ml-2 flex items-center gap-1">
|
|
<Stars rating={m.rating} />
|
|
<span className="text-gray-500">{m.rating}</span>
|
|
</span>
|
|
)}
|
|
</div>
|
|
{m.memo_text && (
|
|
<p className="text-xs text-gray-600 line-clamp-2">
|
|
{m.memo_text}
|
|
</p>
|
|
)}
|
|
<div className="flex items-center gap-2 mt-1.5 text-[10px] text-gray-400">
|
|
{m.visited_at && <span>방문: {m.visited_at}</span>}
|
|
<span className="text-brand-400">비공개</span>
|
|
</div>
|
|
</button>
|
|
))}
|
|
</div>
|
|
)}
|
|
</div>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|