Write.cshtml 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. @page
  2. @model Admin.Pages.Forum.Posts.List.WriteModel
  3. @using Microsoft.AspNetCore.Mvc.Rendering
  4. @{
  5. ViewData["Title"] = "게시글 등록";
  6. }
  7. <partial name="_Editor" />
  8. <div class="container">
  9. <h3 class="mb-3">@ViewData["Title"]</h3>
  10. <hr />
  11. <partial name="_StatusMessage" />
  12. <form id="fPostWrite" method="post" accept-charset="utf-8" autocomplete="off" enctype="multipart/form-data">
  13. <!-- 게시판 -->
  14. <div class="row mb-2">
  15. <label class="col-sm-2 col-form-label"><span class="text-danger">*</span> 게시판</label>
  16. <div class="col-sm-10">
  17. @if ((Model.BoardList?.Count ?? 0) > 0)
  18. {
  19. <select asp-for="Input.BoardID" class="form-select w-auto" required asp-items="@Model.BoardList" id="Input_BoardID">
  20. <option value="">- 선택 -</option>
  21. </select>
  22. }
  23. else
  24. {
  25. <input type="number" asp-for="Input.BoardID" class="form-control w-auto" required />
  26. }
  27. </div>
  28. </div>
  29. <!-- 말머리 -->
  30. <div class="row mb-2" id="prefixRow" hidden>
  31. <label class="col-sm-2 col-form-label">말머리</label>
  32. <div class="col-sm-10">
  33. <select asp-for="Input.BoardPrefixID" class="form-select w-auto" id="boardPrefixSelect">
  34. <option value="">- 선택 -</option>
  35. </select>
  36. </div>
  37. </div>
  38. <!-- 제목 -->
  39. <div class="row mb-2">
  40. <label class="col-sm-2 col-form-label"><span class="text-danger">*</span> 제목</label>
  41. <div class="col-sm-10">
  42. <input type="text" asp-for="Input.Subject" class="form-control"
  43. required maxlength="255" placeholder="제목을 입력하세요 (최대 255자)" />
  44. </div>
  45. </div>
  46. <!-- 내용 (CKEditor) -->
  47. <div class="row mb-2">
  48. <label class="col-sm-2 col-form-label">내용</label>
  49. <div class="col-sm-10">
  50. <textarea asp-for="Input.Content" class="ck-editor" id="Input_Content" rows="12"
  51. placeholder="내용을 입력하세요"></textarea>
  52. </div>
  53. </div>
  54. <!-- 썸네일 -->
  55. <div class="row mb-2">
  56. <label class="col-sm-2 col-form-label">대표 이미지</label>
  57. <div class="col-sm-10">
  58. <input type="file" asp-for="Input.ThumbnailFile" class="form-control" accept="image/*" />
  59. </div>
  60. </div>
  61. <!-- 첨부파일 -->
  62. <div class="row mb-2">
  63. <label class="col-sm-2 col-form-label">첨부파일</label>
  64. <div class="col-sm-10">
  65. <input type="file" asp-for="Input.Files" class="form-control" multiple accept=".jpg,.jpeg,.png,.gif,.webp,.bmp,.pdf,.doc,.docx,.xls,.xlsx,.ppt,.pptx,.txt,.zip,.rar,.7z,.hwp,.hwpx,.csv" />
  66. <span class="form-text text-muted">여러 파일을 선택할 수 있습니다.</span>
  67. </div>
  68. </div>
  69. <!-- 태그 -->
  70. <div class="row mb-2">
  71. <label class="col-sm-2 col-form-label">태그</label>
  72. <div class="col-sm-10">
  73. <input type="text" asp-for="Input.TagsInput" class="form-control" placeholder="태그를 쉼표(,)로 구분하여 입력하세요" />
  74. <span class="form-text text-muted">예: 비트코인, 이더리움, 블록체인</span>
  75. </div>
  76. </div>
  77. <!-- 상태 -->
  78. <div class="row mb-2">
  79. <label class="col-sm-2 col-form-label">상태</label>
  80. <div class="col-sm-10 align-content-center">
  81. <div class="form-check form-check-inline">
  82. <input class="form-check-input" type="checkbox" asp-for="Input.IsNotice" />
  83. <label class="form-check-label" for="Input_IsNotice">공지</label>
  84. </div>
  85. <div class="form-check form-check-inline">
  86. <input class="form-check-input" type="checkbox" asp-for="Input.IsSecret" />
  87. <label class="form-check-label" for="Input_IsSecret">비밀</label>
  88. </div>
  89. <div class="form-check form-check-inline">
  90. <input class="form-check-input" type="checkbox" asp-for="Input.IsAnonymous" />
  91. <label class="form-check-label" for="Input_IsAnonymous">익명</label>
  92. </div>
  93. </div>
  94. </div>
  95. <hr />
  96. <div class="d-grid gap-2 text-center d-md-block">
  97. <button type="submit" class="btn btn-success">저장</button>
  98. <a class="btn btn-secondary btn-cancel" href="/Forum/Posts/List">취소</a>
  99. </div>
  100. <br />
  101. </form>
  102. </div>
  103. @section Scripts {
  104. <script>
  105. // 게시판 변경 시 말머리 동적 로드
  106. document.getElementById("Input_BoardID")?.addEventListener("change", async function () {
  107. const boardID = this.value;
  108. const prefixRow = document.getElementById("prefixRow");
  109. const prefixSelect = document.getElementById("boardPrefixSelect");
  110. prefixSelect.innerHTML = '<option value="">- 선택 -</option>';
  111. if (!boardID) {
  112. prefixRow.hidden = true;
  113. return;
  114. }
  115. try {
  116. const res = await fetch(`/Forum/Posts/List/Write?handler=Prefixes&boardID=${boardID}`);
  117. const items = await res.json();
  118. if (items.length > 0) {
  119. items.forEach(item => {
  120. const opt = document.createElement("option");
  121. opt.value = item.id;
  122. opt.textContent = item.name;
  123. prefixSelect.appendChild(opt);
  124. });
  125. prefixRow.hidden = false;
  126. } else {
  127. prefixRow.hidden = true;
  128. }
  129. } catch {
  130. prefixRow.hidden = true;
  131. }
  132. });
  133. // 취소 버튼 확인
  134. $(function () {
  135. $(".btn-cancel").on("click", function (e) {
  136. const s = $("input[name='Input.Subject']").val()?.trim();
  137. if (s) {
  138. e.preventDefault();
  139. if (confirm("내용이나 제목이 남아 있습니다. 글 작성을 취소하시겠습니까?")) {
  140. location.href = $(this).attr("href");
  141. }
  142. }
  143. });
  144. });
  145. </script>
  146. }