e5981769b9
TaxBaik CI/CD / build-and-deploy (push) Successful in 2m11s
- Admin: replace the global @rendermode on <Routes>/<Router> with per-page render mode. Login.razor now prerenders (form visible before WASM loads); every other [Authorize] page stays prerender: false to avoid the AuthorizeRouteView blank-render regression from earlier attempts. Adds a "준비 중" -> "로그인" splash tied to WASM boot completion, and lets the authenticated-shell loading overlay stay up until AdminShell actually renders. - Contact.cshtml: fix the "Agree" checkbox missing value="true" - a checked box sent the browser-default "on", which bool model binding can't parse, so ModelState.IsValid silently went false and OnPostAsync returned a blank form with no visible error on every submission. Validation summary widened from ModelOnly to All so this class of failure isn't silent again. - TelegramInquiryNotificationService: read Telegram:InquiryChatId (falling back to ChatId) instead of only ChatId, matching the channel routing CLAUDE.md documents and deploy.yml already provisions as separate secrets. - Reconcile CLAUDE.md's self-contradicting Phase 8 prerender notes (Phase 9), rewrite validate_admin_render.sh for the per-page design, and add a SmartAdmin 5.5 design reference section to DOUZONE_UX_GUIDE.md for future admin screens (existing screens unchanged, tracked as WBS P4-03). Co-Authored-By: Claude Sonnet 5 <noreply@anthropic.com>
156 lines
4.6 KiB
Plaintext
156 lines
4.6 KiB
Plaintext
@page "/admin/blog/{id:int}/edit"
|
|
@rendermode @(new InteractiveWebAssemblyRenderMode(prerender: false))
|
|
@attribute [Authorize]
|
|
@using TaxBaik.Application.DTOs
|
|
@using TaxBaik.WasmClient.Components.Admin.Pages.Blog
|
|
@inject IBlogBrowserClient BlogClient
|
|
@inject ICategoryBrowserClient CategoryClient
|
|
@inject NavigationManager Navigation
|
|
@inject ISnackbar Snackbar
|
|
@inject IDialogService DialogService
|
|
|
|
<PageTitle>포스트 수정</PageTitle>
|
|
|
|
<AdminCrudPageShell Title="포스트 수정"
|
|
Eyebrow="Content"
|
|
Subtitle="블로그 포스트를 수정합니다."
|
|
Loading="@isLoading"
|
|
SkeletonContent="@EditorSkeleton"
|
|
OnCancel="@GoBack">
|
|
@if (post == null)
|
|
{
|
|
<MudAlert Severity="Severity.Error" Class="mt-4">포스트를 찾을 수 없습니다.</MudAlert>
|
|
}
|
|
else
|
|
{
|
|
<MudPaper Class="pa-4 mt-4" Elevation="1">
|
|
<BlogForm Model="model" Categories="categories" SubmitText="저장" OnSubmit="SavePost" />
|
|
<div class="mt-4">
|
|
<MudButton Variant="Variant.Outlined" Color="Color.Error" @onclick="DeletePost">삭제</MudButton>
|
|
</div>
|
|
</MudPaper>
|
|
}
|
|
</AdminCrudPageShell>
|
|
|
|
@code {
|
|
[Parameter]
|
|
public int Id { get; set; }
|
|
|
|
private TaxBaik.Application.DTOs.BlogPostResponseDto? post;
|
|
private IReadOnlyList<Domain.Entities.Category> categories = [];
|
|
private BlogForm.BlogFormModel model = new();
|
|
private bool isLoading = true;
|
|
|
|
private RenderFragment EditorSkeleton => builder =>
|
|
{
|
|
builder.OpenComponent<AdminSkeletonRows>(0);
|
|
builder.AddAttribute(1, "Rows", 5);
|
|
builder.AddAttribute(2, "Columns", 3);
|
|
builder.CloseComponent();
|
|
};
|
|
|
|
protected override async Task OnInitializedAsync()
|
|
{
|
|
try
|
|
{
|
|
post = await BlogClient.GetByIdAsync(Id);
|
|
if (post != null)
|
|
{
|
|
categories = await CategoryClient.GetAllAsync();
|
|
MapPostToModel(post);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Snackbar.Add($"포스트 로드 실패: {ex.Message}", Severity.Error);
|
|
}
|
|
finally
|
|
{
|
|
isLoading = false;
|
|
}
|
|
}
|
|
|
|
private void MapPostToModel(TaxBaik.Application.DTOs.BlogPostResponseDto post)
|
|
{
|
|
model.Title = post.Title;
|
|
model.Content = post.Content;
|
|
model.CategoryId = post.CategoryId;
|
|
model.Tags = post.Tags;
|
|
model.SeoTitle = post.SeoTitle;
|
|
model.SeoDescription = post.SeoDescription;
|
|
model.IsPublished = post.IsPublished;
|
|
}
|
|
|
|
private void GoBack()
|
|
{
|
|
Navigation.NavigateTo("/taxbaik/admin/blog");
|
|
}
|
|
|
|
private async Task SavePost()
|
|
{
|
|
if (post == null)
|
|
return;
|
|
|
|
try
|
|
{
|
|
var result = await BlogClient.UpdateAsync(post.Id, new CreateBlogPostDto
|
|
{
|
|
Title = model.Title,
|
|
Content = model.Content,
|
|
CategoryId = model.CategoryId,
|
|
Tags = model.Tags,
|
|
SeoTitle = model.SeoTitle,
|
|
SeoDescription = model.SeoDescription,
|
|
IsPublished = model.IsPublished
|
|
});
|
|
|
|
if (result == null)
|
|
{
|
|
Snackbar.Add("저장 실패: 포스트를 저장하지 못했습니다.", Severity.Error);
|
|
return;
|
|
}
|
|
|
|
Snackbar.Add("포스트가 저장되었습니다.", Severity.Success);
|
|
Navigation.NavigateTo("/taxbaik/admin/blog");
|
|
}
|
|
catch (ValidationException ex)
|
|
{
|
|
Snackbar.Add(ex.Message, Severity.Error);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Snackbar.Add($"저장 실패: {ex.Message}", Severity.Error);
|
|
}
|
|
}
|
|
|
|
private async Task DeletePost()
|
|
{
|
|
if (post == null)
|
|
return;
|
|
|
|
var result = await DialogService.ShowMessageBox(
|
|
"포스트 삭제",
|
|
"정말 삭제하시겠습니까? 이 작업은 취소할 수 없습니다.",
|
|
"삭제", "취소");
|
|
|
|
if (result != true)
|
|
return;
|
|
|
|
try
|
|
{
|
|
var deleted = await BlogClient.DeleteAsync(post.Id);
|
|
if (!deleted)
|
|
{
|
|
Snackbar.Add("삭제 실패: 포스트를 삭제하지 못했습니다.", Severity.Error);
|
|
return;
|
|
}
|
|
Snackbar.Add("포스트가 삭제되었습니다.", Severity.Success);
|
|
Navigation.NavigateTo("/taxbaik/admin/blog");
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Snackbar.Add($"삭제 실패: {ex.Message}", Severity.Error);
|
|
}
|
|
}
|
|
}
|