feat(youtube-tab): RevenueTab 수익 추적 서브탭

This commit is contained in:
2026-05-01 14:48:47 +09:00
parent 16b8cc59ae
commit 3e54b2c98d

View File

@@ -1,3 +1,276 @@
// src/pages/music/components/RevenueTab.jsx
import { useState, useEffect } from 'react';
import {
getRevenueDashboard, getRevenueRecords,
addRevenueRecord, updateRevenueRecord, deleteRevenueRecord,
} from '../../../api';
const COUNTRIES = ['BR', 'US', 'ID', 'MX', 'KR'];
const currentMonth = () => new Date().toISOString().slice(0, 7);
export default function RevenueTab() {
return null;
const [dashboard, setDashboard] = useState(null);
const [records, setRecords] = useState([]);
const [form, setForm] = useState({
yt_video_id: '', record_month: currentMonth(),
revenue_usd: '', views: '', country: 'BR',
});
const [saving, setSaving] = useState(false);
const [editingId, setEditingId] = useState(null);
const [editForm, setEditForm] = useState({});
const loadAll = async () => {
const [dash, recs] = await Promise.all([
getRevenueDashboard().catch(() => null),
getRevenueRecords().catch(() => []),
]);
setDashboard(dash);
setRecords(Array.isArray(recs) ? recs : recs.records ?? []);
};
useEffect(() => { loadAll(); }, []);
const handleAdd = async () => {
if (!form.yt_video_id || !form.revenue_usd || !form.views) return;
setSaving(true);
try {
await addRevenueRecord({
yt_video_id: form.yt_video_id,
record_month: form.record_month,
revenue_usd: parseFloat(form.revenue_usd),
views: parseInt(form.views, 10),
country: form.country,
});
setForm({ yt_video_id: '', record_month: currentMonth(), revenue_usd: '', views: '', country: 'BR' });
await loadAll();
} catch (e) {
console.error('addRevenueRecord:', e);
} finally {
setSaving(false);
}
};
const handleEditSave = async () => {
try {
await updateRevenueRecord(editingId, {
revenue_usd: parseFloat(editForm.revenue_usd),
views: parseInt(editForm.views, 10),
});
setEditingId(null);
await loadAll();
} catch (e) {
console.error('updateRevenueRecord:', e);
}
};
const handleDelete = async (id) => {
if (!window.confirm('이 기록을 삭제할까요?')) return;
try {
await deleteRevenueRecord(id);
await loadAll();
} catch (e) {
console.error('deleteRevenueRecord:', e);
}
};
// 영상별 RPM 상위 5개 (bar chart 용)
const chartData = records
.filter(r => r.views > 0)
.map(r => ({
label: r.yt_video_id,
rpm: (r.revenue_usd / r.views) * 1000,
}))
.sort((a, b) => b.rpm - a.rpm)
.slice(0, 5);
const maxRpm = chartData.length > 0 ? Math.max(...chartData.map(d => d.rpm)) : 1;
return (
<div className="yt-content">
{/* 대시보드 카드 3개 */}
<div className="yt-dash-cards">
<div className="yt-dash-card">
<div className="yt-dash-card__label"> 수익</div>
<div className="yt-dash-card__value yt-dash-card__value--green">
${dashboard?.total_revenue_usd?.toFixed(2) ?? '—'}
</div>
<div className="yt-dash-card__sub">누적</div>
</div>
<div className="yt-dash-card">
<div className="yt-dash-card__label"> 조회수</div>
<div className="yt-dash-card__value yt-dash-card__value--blue">
{dashboard?.total_views != null
? (dashboard.total_views >= 1000
? `${(dashboard.total_views / 1000).toFixed(1)}K`
: String(dashboard.total_views))
: '—'}
</div>
<div className="yt-dash-card__sub">누적</div>
</div>
<div className="yt-dash-card">
<div className="yt-dash-card__label">평균 RPM</div>
<div className="yt-dash-card__value yt-dash-card__value--amber">
${dashboard?.avg_rpm?.toFixed(2) ?? '—'}
</div>
<div className="yt-dash-card__sub">가중평균</div>
</div>
</div>
{/* 영상별 RPM 바 차트 */}
{chartData.length > 0 && (
<div className="yt-card">
<h3 className="yt-card__title">영상별 RPM 비교</h3>
<div className="yt-bar-chart">
{chartData.map((d, i) => (
<div key={i} className="yt-bar-row">
<div className="yt-bar-row__label" title={d.label}>
{d.label.slice(0, 11)}
</div>
<div className="yt-bar-row__track">
<div
className="yt-bar-row__fill"
style={{ width: `${(d.rpm / maxRpm) * 100}%` }}
/>
</div>
<div className="yt-bar-row__value">${d.rpm.toFixed(2)}</div>
</div>
))}
</div>
</div>
)}
{/* 수익 기록 추가 폼 */}
<div className="yt-card yt-card--create">
<h3 className="yt-card__title">+ 수익 기록 추가</h3>
<div className="yt-form-grid">
<div className="yt-field">
<label className="yt-field__label">YouTube 영상 ID</label>
<input
className="yt-input"
value={form.yt_video_id}
onChange={e => setForm(f => ({ ...f, yt_video_id: e.target.value }))}
placeholder="dQw4w9WgXcQ"
/>
</div>
<div className="yt-field">
<label className="yt-field__label">기록 </label>
<input
className="yt-input"
type="month"
value={form.record_month}
onChange={e => setForm(f => ({ ...f, record_month: e.target.value }))}
/>
</div>
<div className="yt-field">
<label className="yt-field__label">수익 (USD)</label>
<input
className="yt-input"
type="number"
step="0.01"
value={form.revenue_usd}
onChange={e => setForm(f => ({ ...f, revenue_usd: e.target.value }))}
placeholder="3.45"
/>
</div>
<div className="yt-field">
<label className="yt-field__label">조회수</label>
<input
className="yt-input"
type="number"
value={form.views}
onChange={e => setForm(f => ({ ...f, views: e.target.value }))}
placeholder="1200"
/>
</div>
</div>
<div className="yt-row yt-row--bottom">
<select
className="yt-select"
value={form.country}
onChange={e => setForm(f => ({ ...f, country: e.target.value }))}
>
{COUNTRIES.map(c => <option key={c} value={c}>{c}</option>)}
</select>
<button
type="button"
className="ms-btn ms-btn--primary"
onClick={handleAdd}
disabled={saving}
>
{saving ? '저장 중...' : '저장'}
</button>
</div>
</div>
{/* 수익 기록 테이블 */}
<div className="yt-card">
<h3 className="yt-card__title">수익 기록</h3>
{records.length === 0 ? (
<p className="yt-empty">수익 기록이 없습니다. 폼으로 추가해보세요.</p>
) : (
<div className="yt-table">
<div className="yt-table__header">
<span>영상 ID</span>
<span></span>
<span>수익</span>
<span>조회수</span>
<span>RPM</span>
<span />
</div>
{records.map(rec => (
editingId === rec.id ? (
<div key={rec.id} className="yt-table__row yt-table__row--editing">
<span className="yt-table__cell">{rec.yt_video_id.slice(0, 11)}</span>
<span className="yt-table__cell">{rec.record_month}</span>
<input
className="yt-input yt-input--sm"
type="number"
step="0.01"
value={editForm.revenue_usd}
onChange={e => setEditForm(f => ({ ...f, revenue_usd: e.target.value }))}
/>
<input
className="yt-input yt-input--sm"
type="number"
value={editForm.views}
onChange={e => setEditForm(f => ({ ...f, views: e.target.value }))}
/>
<span className="yt-table__cell"></span>
<div className="yt-table__actions">
<button type="button" className="ms-btn ms-btn--primary ms-btn--sm" onClick={handleEditSave}>저장</button>
<button type="button" className="ms-btn ms-btn--ghost ms-btn--sm" onClick={() => setEditingId(null)}>취소</button>
</div>
</div>
) : (
<div
key={rec.id}
className="yt-table__row"
onClick={() => {
setEditingId(rec.id);
setEditForm({ revenue_usd: rec.revenue_usd, views: rec.views });
}}
style={{ cursor: 'pointer' }}
>
<span className="yt-table__cell yt-table__cell--mono">{rec.yt_video_id.slice(0, 11)}</span>
<span className="yt-table__cell">{rec.record_month}</span>
<span className="yt-table__cell yt-table__cell--green">${rec.revenue_usd?.toFixed(2)}</span>
<span className="yt-table__cell">{rec.views?.toLocaleString()}</span>
<span className="yt-table__cell yt-table__cell--amber">
{rec.views > 0
? `$${((rec.revenue_usd / rec.views) * 1000).toFixed(2)}`
: '—'}
</span>
<button
type="button"
className="ms-btn--icon ms-btn--danger"
onClick={e => { e.stopPropagation(); handleDelete(rec.id); }}
aria-label="삭제"
></button>
</div>
)
))}
</div>
)}
</div>
</div>
);
}