feat: 프로젝트 진행 현황 추적 시스템 구축 + 마케팅 카피 강화

[DB]
- supabase/migrations/002_project_milestones.sql 추가
  quotes.user_id 컬럼 + project_milestones 테이블 생성 SQL

[API]
- GET  /api/projects            — 로그인 사용자의 프로젝트+마일스톤 조회
- POST /api/projects/link       — 견적서 토큰으로 계정에 프로젝트 연결
- GET/POST /api/admin/milestones — 관리자 마일스톤 목록/기본 7단계 초기화
- PATCH/DELETE /api/admin/milestones/[id] — 관리자 단계별 상태·메모 업데이트

[UI — 마이페이지]
- '프로젝트 현황' 탭 신규 추가 (Tab type 확장)
- 진행률 바, 단계별 타임라인, 개발자 메모 표시
- 견적서 코드 입력 → 계정 연결 폼

[UI — 관리자 견적서 편집]
- '진행 단계' 탭 추가: 기본 7단계 초기화, 단계별 status/메모 편집

[마케팅 카피]
- page.tsx PROMISES 4번째 추가: "진행 현황 마이페이지 실시간 확인"
- freelance 보증 카드 5번째 추가: 실시간 진행 현황 (그리드 2×5)
- services/website trust badge 5번째 추가

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-04-02 02:49:40 +09:00
parent 4b712048db
commit 19b09e3b90
11 changed files with 627 additions and 5 deletions

50
app/api/projects/route.ts Normal file
View File

@@ -0,0 +1,50 @@
import { NextResponse } from 'next/server';
import { createAdminClient } from '@/lib/supabase/admin';
import { createClient } from '@/lib/supabase/server';
export const runtime = 'nodejs';
export async function GET() {
const supabase = await createClient();
const { data: { user } } = await supabase.auth.getUser();
if (!user) return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
const admin = createAdminClient();
const { data: quotes, error: qErr } = await admin
.from('quotes')
.select('id, title, status, items, created_at')
.eq('user_id', user.id)
.in('status', ['sent', 'accepted', 'in_progress', 'completed', 'delivered'])
.order('created_at', { ascending: false });
if (qErr) return NextResponse.json({ error: qErr.message }, { status: 500 });
if (!quotes?.length) return NextResponse.json({ projects: [] });
const quoteIds = quotes.map((q) => q.id);
const { data: milestones } = await admin
.from('project_milestones')
.select('*')
.in('quote_id', quoteIds)
.order('step_number', { ascending: true });
const projects = quotes.map((q) => ({
id: q.id,
title: q.title,
status: q.status,
total: Array.isArray(q.items)
? q.items.reduce(
(s: number, i: { unitPrice?: number; quantity?: number }) =>
s + ((i.unitPrice ?? 0) * (i.quantity ?? 1)),
0
)
: 0,
created_at: q.created_at,
milestones: (milestones ?? [])
.filter((m) => m.quote_id === q.id)
.sort((a, b) => a.step_number - b.step_number),
}));
return NextResponse.json({ projects });
}