160afb7c7e
TaxBaik CI/CD / build-and-deploy (push) Successful in 49s
**Blockers Fixed:** 1. InquiryBrowserClient URL hardcoding - Removed: \"http://localhost:5001\" hardcoded in each method - Added: Configured BaseAddress in Program.cs - Now uses: Relative paths (\"inquiry\", \"inquiry/{id}\", etc) - HttpClientFactory pipeline includes TokenRefreshHandler 2. Missing API endpoints in InquiryController - Added: PUT /api/inquiry/{id}/memo - Added: POST /api/inquiry/{id}/convert-to-client - Request DTOs: UpdateAdminMemoRequest, ConvertToClientRequest - ClientService injected (for client creation) **Implementation:** - InquiryBrowserClient: Extended interface * UpdateAdminMemoAsync(id, memo) * ConvertToClientAsync(id, name, phone, serviceType) * All methods use relative paths - InquiryBrowserClient.ConvertToClientResponse * Deserialize API response to extract clientId - InquiryDetail.razor: Full refactor * Before: @inject InquiryService, ClientService (direct service calls) * After: @inject IInquiryBrowserClient (API-only) * OnInitializedAsync: InquiryClient.GetByIdAsync * OnStatusChanged: InquiryClient.UpdateStatusAsync * SaveMemo: InquiryClient.UpdateAdminMemoAsync * ConvertToClient: InquiryClient.ConvertToClientAsync **InquiryList.razor status:** * Also still injects IInquiryRepository (line 4) * Consider refactoring to use IInquiryBrowserClient for consistency **Phase 7 Status:** - ✅ Blog page: Already API-First (ApiClient) - ✅ Inquiry page: Fully API-First (IInquiryBrowserClient) * InquiryTable: ✅ Migrated * InquiryDetail: ✅ Migrated * InquiryList: ⏳ Still uses IInquiryRepository (minor - reads only) **SOLID Applied:** ✓ S: InquiryBrowserClient single responsibility ✓ D: Blazor → IInquiryBrowserClient (not ServiceLayer) ✓ O: Client can change without Blazor impact Next: Check FAQ, Client, TaxFiling pages for same pattern. If all still injecting services directly, migrate sequentially. Then: Phase 6 (SignalR) will have all pages ready. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
202 lines
7.4 KiB
Plaintext
202 lines
7.4 KiB
Plaintext
@page "/admin/inquiries/{InquiryId:int}"
|
|
@attribute [Authorize]
|
|
@using TaxBaik.Web.Services
|
|
@inject IInquiryBrowserClient InquiryClient
|
|
@inject NavigationManager Navigation
|
|
@inject ISnackbar Snackbar
|
|
|
|
<PageTitle>문의 상세</PageTitle>
|
|
|
|
@if (inquiry != null)
|
|
{
|
|
<MudButton Variant="Variant.Outlined"
|
|
Color="Color.Primary"
|
|
StartIcon="@Icons.Material.Filled.ArrowBack"
|
|
@onclick="@(() => Navigation.NavigateTo("/taxbaik/admin/inquiries"))">
|
|
문의 목록으로
|
|
</MudButton>
|
|
|
|
<MudGrid Class="mt-4">
|
|
<MudItem xs="12" md="8">
|
|
<MudPaper Class="pa-4" Elevation="1">
|
|
<MudText Typo="Typo.h6" Class="mb-3">문의 정보</MudText>
|
|
<MudGrid>
|
|
<MudItem xs="12" sm="6">
|
|
<MudText Typo="Typo.subtitle2" Color="Color.Secondary">이름</MudText>
|
|
<MudText>@inquiry.Name</MudText>
|
|
</MudItem>
|
|
<MudItem xs="12" sm="6">
|
|
<MudText Typo="Typo.subtitle2" Color="Color.Secondary">연락처</MudText>
|
|
<MudText>@inquiry.Phone</MudText>
|
|
</MudItem>
|
|
<MudItem xs="12" sm="6">
|
|
<MudText Typo="Typo.subtitle2" Color="Color.Secondary">이메일</MudText>
|
|
<MudText>@(inquiry.Email ?? "-")</MudText>
|
|
</MudItem>
|
|
<MudItem xs="12" sm="6">
|
|
<MudText Typo="Typo.subtitle2" Color="Color.Secondary">분야</MudText>
|
|
<MudText>@inquiry.ServiceType</MudText>
|
|
</MudItem>
|
|
<MudItem xs="12">
|
|
<MudText Typo="Typo.subtitle2" Color="Color.Secondary">문의 내용</MudText>
|
|
<MudPaper Class="pa-3 mt-1" Outlined="true">
|
|
<MudText Style="white-space: pre-wrap;">@inquiry.Message</MudText>
|
|
</MudPaper>
|
|
</MudItem>
|
|
<MudItem xs="12">
|
|
<MudText Typo="Typo.subtitle2" Color="Color.Secondary">접수일시</MudText>
|
|
<MudText>@inquiry.CreatedAt.ToLocalTime().ToString("yyyy-MM-dd HH:mm")</MudText>
|
|
</MudItem>
|
|
</MudGrid>
|
|
</MudPaper>
|
|
|
|
<MudPaper Class="pa-4 mt-4" Elevation="1">
|
|
<MudText Typo="Typo.h6" Class="mb-3">담당자 메모</MudText>
|
|
<MudTextField T="string" @bind-Value="adminMemo" Label="내부 메모 (고객에게 미노출)"
|
|
Lines="4" Variant="Variant.Outlined" />
|
|
<MudButton Class="mt-2" Variant="Variant.Filled" Color="Color.Primary"
|
|
OnClick="SaveMemo">메모 저장</MudButton>
|
|
</MudPaper>
|
|
</MudItem>
|
|
|
|
<MudItem xs="12" md="4">
|
|
<MudPaper Class="pa-4" Elevation="1">
|
|
<MudText Typo="Typo.h6" Class="mb-3">처리 상태</MudText>
|
|
<MudStack Spacing="2">
|
|
@foreach (var (key, label) in InquiryStatusMapper.Labels)
|
|
{
|
|
<MudButton Variant="@(inquiry.Status == key ? Variant.Filled : Variant.Outlined)"
|
|
Color="@StatusColor(key)"
|
|
FullWidth="true"
|
|
OnClick="@(() => OnStatusChanged(key))">
|
|
@label
|
|
</MudButton>
|
|
}
|
|
</MudStack>
|
|
</MudPaper>
|
|
|
|
@if (inquiry.ClientId == null)
|
|
{
|
|
<MudPaper Class="pa-4 mt-4" Elevation="1">
|
|
<MudText Typo="Typo.h6" Class="mb-3">고객 카드 생성</MudText>
|
|
<MudText Typo="Typo.body2" Class="mb-3">이 문의를 고객 카드로 등록합니다.</MudText>
|
|
<MudButton Variant="Variant.Filled" Color="Color.Success" FullWidth="true"
|
|
OnClick="ConvertToClient">
|
|
고객으로 등록
|
|
</MudButton>
|
|
</MudPaper>
|
|
}
|
|
else
|
|
{
|
|
<MudPaper Class="pa-4 mt-4" Elevation="1">
|
|
<MudText Typo="Typo.h6" Class="mb-3">연결된 고객</MudText>
|
|
<MudButton Variant="Variant.Outlined" Color="Color.Primary" FullWidth="true"
|
|
Href="@($"/taxbaik/admin/clients/{inquiry.ClientId}")">
|
|
고객 카드 보기
|
|
</MudButton>
|
|
</MudPaper>
|
|
}
|
|
</MudItem>
|
|
</MudGrid>
|
|
}
|
|
else
|
|
{
|
|
<MudText>문의를 찾을 수 없습니다.</MudText>
|
|
}
|
|
|
|
@code {
|
|
[Parameter]
|
|
public int InquiryId { get; set; }
|
|
|
|
private Domain.Entities.Inquiry? inquiry;
|
|
private string adminMemo = "";
|
|
|
|
protected override async Task OnInitializedAsync()
|
|
{
|
|
inquiry = await InquiryClient.GetByIdAsync(InquiryId);
|
|
adminMemo = inquiry?.AdminMemo ?? "";
|
|
}
|
|
|
|
private async Task OnStatusChanged(string status)
|
|
{
|
|
if (inquiry == null) return;
|
|
try
|
|
{
|
|
var success = await InquiryClient.UpdateStatusAsync(inquiry.Id, status);
|
|
if (success)
|
|
{
|
|
inquiry.Status = status;
|
|
Snackbar.Add("상태가 변경되었습니다.", Severity.Success);
|
|
}
|
|
else
|
|
{
|
|
Snackbar.Add("상태 변경에 실패했습니다.", Severity.Error);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Snackbar.Add($"오류: {ex.Message}", Severity.Error);
|
|
}
|
|
}
|
|
|
|
private async Task SaveMemo()
|
|
{
|
|
if (inquiry == null) return;
|
|
try
|
|
{
|
|
var success = await InquiryClient.UpdateAdminMemoAsync(inquiry.Id, adminMemo);
|
|
if (success)
|
|
{
|
|
inquiry.AdminMemo = adminMemo;
|
|
Snackbar.Add("메모가 저장되었습니다.", Severity.Success);
|
|
}
|
|
else
|
|
{
|
|
Snackbar.Add("메모 저장에 실패했습니다.", Severity.Error);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Snackbar.Add($"오류: {ex.Message}", Severity.Error);
|
|
}
|
|
}
|
|
|
|
private async Task ConvertToClient()
|
|
{
|
|
if (inquiry == null) return;
|
|
try
|
|
{
|
|
var clientId = await InquiryClient.ConvertToClientAsync(
|
|
inquiry.Id,
|
|
inquiry.Name,
|
|
inquiry.Phone,
|
|
inquiry.ServiceType);
|
|
|
|
if (clientId > 0)
|
|
{
|
|
inquiry.ClientId = clientId;
|
|
inquiry.Status = "consulting";
|
|
Snackbar.Add("고객 카드가 생성되었습니다.", Severity.Success);
|
|
}
|
|
else
|
|
{
|
|
Snackbar.Add("고객 카드 생성에 실패했습니다.", Severity.Error);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Snackbar.Add($"오류: {ex.Message}", Severity.Error);
|
|
}
|
|
}
|
|
|
|
private Color StatusColor(string status) => status switch
|
|
{
|
|
"new" => Color.Default,
|
|
"consulting" => Color.Info,
|
|
"contracted" => Color.Success,
|
|
"rejected" => Color.Error,
|
|
"closed" => Color.Dark,
|
|
_ => Color.Default
|
|
};
|
|
}
|