4358b189c8
TaxBaik CI/CD / build-and-deploy (push) Successful in 3m2s
**Blazor Pages Refactored (9 pages):** ✅ ClientList.razor (Service → IClientBrowserClient) ✅ ClientEdit.razor (Service → IClientBrowserClient) ✅ TaxFilingList.razor (Service → ITaxFilingBrowserClient) ✅ FilingTable.razor (Service → ITaxFilingBrowserClient) ✅ FaqList.razor (Service → IFaqBrowserClient) ✅ FaqEdit.razor (Service → IFaqBrowserClient) ✅ AnnouncementList.razor (Service → IAnnouncementBrowserClient) ✅ AnnouncementEdit.razor (Service → IAnnouncementBrowserClient) ✅ Previously: Dashboard, InquiryTable, InquiryDetail **Pattern Applied Consistently:** - Removed all direct service injections (Service Layer) - Injected specialized Browser Clients (API Layer) - Error handling with Snackbar notifications - Try-catch for all API calls - Graceful fallbacks (empty lists on error) **Phase 7 Complete: 100% API-First Refactoring** All admin pages now use: ClientBrowserClient → /api/client (Clients) TaxFilingBrowserClient → /api/tax-filing (Tax Filings) FaqBrowserClient → /api/faq (FAQs) AnnouncementBrowserClient → /api/announcement (Announcements) InquiryBrowserClient → /api/inquiry (Inquiries) AdminDashboardClient → /api/admin-dashboard (Dashboard) **SOLID + Maintainability Achieved:** ✓ Single Responsibility: Each client = one domain ✓ Open/Closed: Extensible without modifying Blazor ✓ Dependency Inversion: Blazor → Abstractions, not services ✓ Interface Segregation: Fine-grained client interfaces ✓ Liskov Substitution: Interchangeable implementations Build: ✅ Success (0 errors) Status: Ready for Phase 6 (SignalR Integration) Next: NotificationHub for real-time dashboard updates Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
110 lines
3.4 KiB
Plaintext
110 lines
3.4 KiB
Plaintext
@using TaxBaik.Web.Services
|
|
@using TaxBaik.Domain.Entities
|
|
@inject ITaxFilingBrowserClient FilingClient
|
|
@inject ISnackbar Snackbar
|
|
|
|
@if (Filings == null || Filings.Count == 0)
|
|
{
|
|
<MudText Class="pa-4" Color="Color.Secondary">항목이 없습니다.</MudText>
|
|
}
|
|
else
|
|
{
|
|
<MudTable Items="Filings" Hover="true" Dense="true" Class="mt-2">
|
|
<HeaderContent>
|
|
<MudTh>고객</MudTh>
|
|
<MudTh>신고 유형</MudTh>
|
|
<MudTh>기한</MudTh>
|
|
<MudTh>D-day</MudTh>
|
|
<MudTh>메모</MudTh>
|
|
<MudTh>처리</MudTh>
|
|
</HeaderContent>
|
|
<RowTemplate>
|
|
<MudTd>@context.ClientName</MudTd>
|
|
<MudTd>@context.FilingType</MudTd>
|
|
<MudTd>@context.DueDate.ToString("yyyy-MM-dd")</MudTd>
|
|
<MudTd>
|
|
@{
|
|
var dday = (context.DueDate.Date - DateTime.Today).Days;
|
|
}
|
|
@if (dday < 0)
|
|
{
|
|
<MudChip T="string" Size="Size.Small" Color="Color.Error">D+@(-dday)</MudChip>
|
|
}
|
|
else if (dday <= 7)
|
|
{
|
|
<MudChip T="string" Size="Size.Small" Color="Color.Warning">D-@dday</MudChip>
|
|
}
|
|
else
|
|
{
|
|
<MudText Typo="Typo.body2">D-@dday</MudText>
|
|
}
|
|
</MudTd>
|
|
<MudTd>@(context.Memo ?? "")</MudTd>
|
|
<MudTd>
|
|
@if (context.Status == "pending")
|
|
{
|
|
<MudButton Size="Size.Small" Variant="Variant.Filled" Color="Color.Success"
|
|
OnClick="@(() => MarkFiled(context))">완료</MudButton>
|
|
}
|
|
else if (context.Status == "filed")
|
|
{
|
|
<MudChip T="string" Size="Size.Small" Color="Color.Success">완료</MudChip>
|
|
}
|
|
<MudIconButton Icon="@Icons.Material.Filled.Delete" Size="Size.Small" Color="Color.Error"
|
|
OnClick="@(() => DeleteFiling(context.Id))" />
|
|
</MudTd>
|
|
</RowTemplate>
|
|
</MudTable>
|
|
}
|
|
|
|
@code {
|
|
[Parameter]
|
|
public List<TaxFiling>? Filings { get; set; }
|
|
|
|
[Parameter]
|
|
public EventCallback OnStatusChange { get; set; }
|
|
|
|
private async Task MarkFiled(TaxFiling filing)
|
|
{
|
|
try
|
|
{
|
|
filing.Status = "filed";
|
|
var result = await FilingClient.UpdateAsync(filing.Id, filing);
|
|
if (result != null)
|
|
{
|
|
Snackbar.Add("신고 완료 처리되었습니다.", Severity.Success);
|
|
await OnStatusChange.InvokeAsync();
|
|
}
|
|
else
|
|
{
|
|
Snackbar.Add("처리 실패", Severity.Error);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Snackbar.Add($"오류: {ex.Message}", Severity.Error);
|
|
}
|
|
}
|
|
|
|
private async Task DeleteFiling(int id)
|
|
{
|
|
try
|
|
{
|
|
var success = await FilingClient.DeleteAsync(id);
|
|
if (success)
|
|
{
|
|
Snackbar.Add("삭제되었습니다.", Severity.Info);
|
|
await OnStatusChange.InvokeAsync();
|
|
}
|
|
else
|
|
{
|
|
Snackbar.Add("삭제 실패", Severity.Error);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Snackbar.Add($"오류: {ex.Message}", Severity.Error);
|
|
}
|
|
}
|
|
}
|