79492184d0
- WBS-CRM-02: 상담 이력 (consultations 테이블 V008, ClientDetail.razor) - WBS-CRM-03: 문의→고객 전환 (V009 client_id FK, InquiryDetail 고객등록 버튼) - WBS-CRM-04: 신고 일정 캘린더 (tax_filings 테이블 V010, TaxFilingList.razor) - WBS-CRM-05: 문의 상태 5단계 확장 (V011, InquiryStatus enum, InquiryList 탭) - WBS-MKT-04: 시즌 시뮬레이터 어드민 페이지 (SeasonSimulator.razor) - WBS-UX-04: 개인정보처리방침 /taxbaik/privacy, 이용약관 /taxbaik/terms - Dashboard.razor 마감 임박 신고 위젯 추가 Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
172 lines
6.7 KiB
Plaintext
172 lines
6.7 KiB
Plaintext
@page "/admin/inquiries/{InquiryId:int}"
|
|
@attribute [Authorize]
|
|
@using TaxBaik.Application.Services
|
|
@inject InquiryService InquiryService
|
|
@inject ClientService ClientService
|
|
@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 InquiryService.GetByIdAsync(InquiryId);
|
|
adminMemo = inquiry?.AdminMemo ?? "";
|
|
}
|
|
|
|
private async Task OnStatusChanged(string status)
|
|
{
|
|
if (inquiry == null) return;
|
|
try
|
|
{
|
|
await InquiryService.UpdateStatusAsync(inquiry.Id, status, "관리자");
|
|
inquiry.Status = status;
|
|
Snackbar.Add("상태가 변경되었습니다.", Severity.Success);
|
|
}
|
|
catch (ValidationException ex)
|
|
{
|
|
Snackbar.Add(ex.Message, Severity.Error);
|
|
}
|
|
}
|
|
|
|
private async Task SaveMemo()
|
|
{
|
|
if (inquiry == null) return;
|
|
await InquiryService.UpdateAdminMemoAsync(inquiry.Id, adminMemo);
|
|
inquiry.AdminMemo = adminMemo;
|
|
Snackbar.Add("메모가 저장되었습니다.", Severity.Success);
|
|
}
|
|
|
|
private async Task ConvertToClient()
|
|
{
|
|
if (inquiry == null) return;
|
|
try
|
|
{
|
|
var clientId = await ClientService.CreateFromInquiryAsync(inquiry.Name, inquiry.Phone, inquiry.ServiceType);
|
|
await InquiryService.LinkClientAsync(inquiry.Id, clientId);
|
|
await InquiryService.UpdateStatusAsync(inquiry.Id, "consulting", "관리자");
|
|
inquiry.ClientId = clientId;
|
|
inquiry.Status = "consulting";
|
|
Snackbar.Add("고객 카드가 생성되었습니다.", Severity.Success);
|
|
}
|
|
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
|
|
};
|
|
}
|