feat: add new portfolio creation page
This commit is contained in:
parent
cbf30c6bb4
commit
c1f175f9bd
103
frontend/src/app/portfolio/new/page.tsx
Normal file
103
frontend/src/app/portfolio/new/page.tsx
Normal file
@ -0,0 +1,103 @@
|
||||
'use client';
|
||||
|
||||
import { useState } from 'react';
|
||||
import { useRouter } from 'next/navigation';
|
||||
import Sidebar from '@/components/layout/Sidebar';
|
||||
import Header from '@/components/layout/Header';
|
||||
import { api } from '@/lib/api';
|
||||
|
||||
export default function NewPortfolioPage() {
|
||||
const router = useRouter();
|
||||
const [name, setName] = useState('');
|
||||
const [portfolioType, setPortfolioType] = useState('general');
|
||||
const [loading, setLoading] = useState(false);
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
|
||||
const handleSubmit = async (e: React.FormEvent) => {
|
||||
e.preventDefault();
|
||||
setLoading(true);
|
||||
setError(null);
|
||||
|
||||
try {
|
||||
const portfolio = await api.post('/api/portfolios', {
|
||||
name,
|
||||
portfolio_type: portfolioType,
|
||||
});
|
||||
router.push(`/portfolio/${(portfolio as { id: number }).id}`);
|
||||
} catch (err) {
|
||||
const message = err instanceof Error ? err.message : 'Failed to create portfolio';
|
||||
setError(message);
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="flex min-h-screen">
|
||||
<Sidebar />
|
||||
<div className="flex-1">
|
||||
<Header />
|
||||
<main className="p-6">
|
||||
<h1 className="text-2xl font-bold text-gray-800 mb-6">새 포트폴리오</h1>
|
||||
|
||||
{error && (
|
||||
<div className="bg-red-100 border border-red-400 text-red-700 px-4 py-3 rounded mb-4">
|
||||
{error}
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div className="max-w-md">
|
||||
<form onSubmit={handleSubmit} className="bg-white rounded-lg shadow p-6">
|
||||
<div className="mb-4">
|
||||
<label htmlFor="portfolio-name" className="block text-sm font-medium text-gray-700 mb-2">
|
||||
포트폴리오 이름
|
||||
</label>
|
||||
<input
|
||||
id="portfolio-name"
|
||||
type="text"
|
||||
value={name}
|
||||
onChange={(e) => setName(e.target.value)}
|
||||
className="w-full px-3 py-2 border rounded focus:outline-none focus:ring-2 focus:ring-blue-500"
|
||||
placeholder="예: 퇴직연금 포트폴리오"
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div className="mb-6">
|
||||
<label htmlFor="portfolio-type" className="block text-sm font-medium text-gray-700 mb-2">
|
||||
유형
|
||||
</label>
|
||||
<select
|
||||
id="portfolio-type"
|
||||
value={portfolioType}
|
||||
onChange={(e) => setPortfolioType(e.target.value)}
|
||||
className="w-full px-3 py-2 border rounded focus:outline-none focus:ring-2 focus:ring-blue-500"
|
||||
>
|
||||
<option value="general">일반</option>
|
||||
<option value="pension">퇴직연금</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div className="flex gap-2">
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => router.back()}
|
||||
className="flex-1 px-4 py-2 border rounded hover:bg-gray-50 transition-colors"
|
||||
>
|
||||
취소
|
||||
</button>
|
||||
<button
|
||||
type="submit"
|
||||
disabled={loading || !name}
|
||||
className="flex-1 px-4 py-2 bg-blue-600 text-white rounded hover:bg-blue-700 disabled:bg-blue-400 transition-colors"
|
||||
>
|
||||
{loading ? '생성 중...' : '생성'}
|
||||
</button>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</main>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
Loading…
x
Reference in New Issue
Block a user