page.tsx 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. 'use client';
  2. import { useState, useEffect, useMemo } from 'react';
  3. import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
  4. import { faCircleInfo } from '@fortawesome/free-solid-svg-icons';
  5. import { fetchApi } from '@/lib/utils/client';
  6. import type { WithholdingTaxSummaryResponse } from '@/types/response/settlement/tax';
  7. import { MONTH_NAMES, TAX_GUIDE_ITEMS } from '../constants';
  8. import Loading from '@/app/component/Loading';
  9. export default function SettlementTaxPage() {
  10. const currentYear = new Date().getFullYear();
  11. const [loading, setLoading] = useState(true);
  12. const [year, setYear] = useState(currentYear);
  13. const [data, setData] = useState<WithholdingTaxSummaryResponse|null>(null);
  14. const yearOptions = useMemo(() => {
  15. const years: number[] = [];
  16. for (let y = currentYear; y >= currentYear - 4; y--) {
  17. years.push(y);
  18. }
  19. return years;
  20. }, [currentYear]);
  21. useEffect(() => {
  22. setLoading(true);
  23. fetchApi<WithholdingTaxSummaryResponse>(`/api/studio/settlement/tax?year=${year}`)
  24. .then(res => {
  25. if (res.data) {
  26. setData(res.data);
  27. }
  28. })
  29. .catch(() => {})
  30. .finally(() => setLoading(false));
  31. }, [year]);
  32. return (
  33. <div className="studio-page settlement">
  34. <div className="studio-page__header">
  35. <h1 className="studio-page__title">원천징수 내역</h1>
  36. </div>
  37. {/* 연도 선택 */}
  38. <div className="settlement__year-select">
  39. <label htmlFor="year-select">연도</label>
  40. <select
  41. id="year-select"
  42. value={year}
  43. onChange={e => setYear(Number(e.target.value))}
  44. >
  45. {yearOptions.map(y => (
  46. <option key={y} value={y}>{y}년</option>
  47. ))}
  48. </select>
  49. </div>
  50. {loading && <Loading />}
  51. {!loading && data && (
  52. <>
  53. {/* 연간 요약 카드 */}
  54. <div className="settlement__cards">
  55. <div className="settlement__card">
  56. <span className="settlement__card-label">총 지급액</span>
  57. <div className="settlement__card-value">
  58. {data.annualSummary.totalGrossAmount.toLocaleString()}원
  59. </div>
  60. </div>
  61. <div className="settlement__card">
  62. <span className="settlement__card-label">소득세 (3%)</span>
  63. <div className="settlement__card-value settlement__card-value--danger">
  64. -{data.annualSummary.totalIncomeTax.toLocaleString()}원
  65. </div>
  66. </div>
  67. <div className="settlement__card">
  68. <span className="settlement__card-label">지방소득세 (0.3%)</span>
  69. <div className="settlement__card-value settlement__card-value--danger">
  70. -{data.annualSummary.totalLocalTax.toLocaleString()}원
  71. </div>
  72. </div>
  73. <div className="settlement__card">
  74. <span className="settlement__card-label">실수령액</span>
  75. <div className="settlement__card-value settlement__card-value--primary">
  76. {data.annualSummary.totalNetAmount.toLocaleString()}원
  77. </div>
  78. </div>
  79. </div>
  80. {/* 월별 상세 */}
  81. <h2 className="settlement__section-title">월별 상세</h2>
  82. <div className="settlement__table-wrap">
  83. <table className="settlement__table">
  84. <thead>
  85. <tr>
  86. <th>월</th>
  87. <th style={{ textAlign: 'right' }}>지급액</th>
  88. <th style={{ textAlign: 'right' }}>소득세</th>
  89. <th style={{ textAlign: 'right' }}>지방소득세</th>
  90. <th style={{ textAlign: 'right' }}>실수령액</th>
  91. <th style={{ textAlign: 'right' }}>건수</th>
  92. </tr>
  93. </thead>
  94. <tbody>
  95. {data.monthlyList.length > 0 ? (
  96. data.monthlyList.map(row => (
  97. <tr key={row.month}>
  98. <td>{MONTH_NAMES[row.month - 1]}</td>
  99. <td style={{ textAlign: 'right' }}>
  100. {row.grossAmount.toLocaleString()}원
  101. </td>
  102. <td style={{ textAlign: 'right' }}>
  103. <span className="settlement__amount--minus">
  104. -{row.incomeTax.toLocaleString()}원
  105. </span>
  106. </td>
  107. <td style={{ textAlign: 'right' }}>
  108. <span className="settlement__amount--minus">
  109. -{row.localTax.toLocaleString()}원
  110. </span>
  111. </td>
  112. <td style={{ textAlign: 'right' }}>
  113. <span className="settlement__amount--plus">
  114. {row.netAmount.toLocaleString()}원
  115. </span>
  116. </td>
  117. <td style={{ textAlign: 'right' }}>{row.paymentCount}건</td>
  118. </tr>
  119. ))
  120. ) : (
  121. <tr>
  122. <td colSpan={6} className="settlement__empty">
  123. 해당 연도의 원천징수 내역이 없습니다.
  124. </td>
  125. </tr>
  126. )}
  127. </tbody>
  128. </table>
  129. </div>
  130. {/* 종합소득세 신고 안내 */}
  131. <div className="settlement__guide">
  132. <div className="settlement__guide-title">
  133. <FontAwesomeIcon icon={faCircleInfo} />
  134. 종합소득세 신고 안내
  135. </div>
  136. <ul className="settlement__guide-list">
  137. {TAX_GUIDE_ITEMS.map((item, i) => (
  138. <li key={i}>{item}</li>
  139. ))}
  140. </ul>
  141. </div>
  142. </>
  143. )}
  144. </div>
  145. );
  146. }