Index.cshtml 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. @page
  2. @model Admin.Pages.Forum.Reports.Comment.IndexModel
  3. @using Microsoft.AspNetCore.Mvc.Rendering
  4. @using Domain.Entities.Forum.ValueObject
  5. @{
  6. ViewData["Title"] = "댓글 신고 관리";
  7. }
  8. <div class="container-fluid">
  9. <h3>@ViewData["Title"]</h3>
  10. <hr />
  11. <partial name="_StatusMessage" />
  12. <div class="row g-2 align-items-end">
  13. <div class="col-6 col-sm-auto">
  14. <select name="boardID" id="boardID" class="form-select" form="fAdminSearch">
  15. <option value="">- 전체 -</option>
  16. @foreach (var g in (Model?.BoardList ?? Enumerable.Empty<SelectListItem>()))
  17. {
  18. <option value="@g.Value" selected="@((Model?.Query?.BoardID?.ToString() ?? "") == (g.Value ?? ""))">@g.Text</option>
  19. }
  20. </select>
  21. </div>
  22. <div class="col-6 col-sm-auto">
  23. <select name="search" id="search" class="form-select" form="fAdminSearch">
  24. <option value="">- 전체 -</option>
  25. <option value="1" selected="@(Model?.Query.Search == 1)">게시글 ID</option>
  26. <option value="2" selected="@(Model?.Query.Search == 2)">게시글 제목</option>
  27. <option value="3" selected="@(Model?.Query.Search == 3)">작성자</option>
  28. </select>
  29. </div>
  30. <div class="col-12 col-sm col-md col-lg-auto">
  31. <input type="text" name="keyword" id="keyword" class="form-control" value="@(Model?.Query.Keyword ?? "")" placeholder="검색어" form="fAdminSearch" />
  32. </div>
  33. <div class="col-12 col-md-12 col-lg-auto">
  34. <div class="input-group">
  35. <input type="date" name="startAt" id="startAt" class="form-control" value="@(Model?.Query.StartAt ?? "")" form="fAdminSearch" />
  36. <span class="input-group-text">~</span>
  37. <input type="date" name="endAt" id="endAt" class="form-control" value="@(Model?.Query.EndAt ?? "")" form="fAdminSearch" />
  38. </div>
  39. </div>
  40. <div class="col-6 col-sm col-lg-auto">
  41. <select name="status" id="status" class="form-select" form="fAdminSearch">
  42. <option value="">- 상태 -</option>
  43. <option value="0" selected="@(Model?.Query.Status == 0)">접수</option>
  44. <option value="1" selected="@(Model?.Query.Status == 1)">처리중</option>
  45. <option value="2" selected="@(Model?.Query.Status == 2)">완료</option>
  46. </select>
  47. </div>
  48. <div class="col-6 col-sm col-lg-auto">
  49. <select name="type" id="type" class="form-select" form="fAdminSearch">
  50. <option value="">- 유형 -</option>
  51. <option value="1" selected="@(Model?.Query.Type == 1)">욕설</option>
  52. <option value="2" selected="@(Model?.Query.Type == 2)">음란</option>
  53. <option value="3" selected="@(Model?.Query.Type == 3)">불법</option>
  54. <option value="4" selected="@(Model?.Query.Type == 4)">사칭</option>
  55. <option value="5" selected="@(Model?.Query.Type == 5)">현금거래유도</option>
  56. <option value="6" selected="@(Model?.Query.Type == 6)">스팸/광고</option>
  57. <option value="7" selected="@(Model?.Query.Type == 7)">도배</option>
  58. <option value="8" selected="@(Model?.Query.Type == 8)">개인정보노출</option>
  59. <option value="9" selected="@(Model?.Query.Type == 9)">기타</option>
  60. </select>
  61. </div>
  62. <div class="col-12 col-md-auto text-center">
  63. <button type="submit" id="btnSearch" class="btn btn-primary" form="fAdminSearch">검색</button>
  64. </div>
  65. </div>
  66. <hr />
  67. <div class="row g-2 align-items-center mt-2">
  68. <div class="col">
  69. Total : @Model?.Total.ToString("N0")
  70. </div>
  71. <div class="col-auto">
  72. <select name="perPage" id="perPage" class="form-select" form="fAdminSearch">
  73. <option value="10" selected="@(Model.Query.PerPage == 10)">10</option>
  74. <option value="20" selected="@(Model.Query.PerPage == 20)">20</option>
  75. <option value="50" selected="@(Model.Query.PerPage == 50)">50</option>
  76. <option value="100" selected="@(Model.Query.PerPage == 100)">100</option>
  77. </select>
  78. </div>
  79. <div class="col-auto">
  80. <button type="button" id="btnListDelete" class="btn btn-danger" disabled>삭제</button>
  81. </div>
  82. </div>
  83. <div class="table-responsive">
  84. <table class="table table-bordered mt-3">
  85. <colgroup>
  86. <col style="width: 5%;" /> <!-- ID -->
  87. <col style="width: 8%;" /> <!-- 게시판 -->
  88. <col style="width: 8%;" /> <!-- 유형 -->
  89. <col style="width: 7%;" /> <!-- 상태 -->
  90. <col /> <!-- 댓글 내용 -->
  91. <col style="width: 7%;" /> <!-- 댓글ID -->
  92. <col style="width: 10%;" /> <!-- 신고자 -->
  93. <col style="width: 10%;" /> <!-- 사유 -->
  94. <col style="width: 10%;" /> <!-- 등록일 -->
  95. </colgroup>
  96. <thead>
  97. <tr>
  98. <th rowspan="2">
  99. <div class="form-check form-check-inline">
  100. <input type="checkbox" id="checkedAll" class="form-check-input" value="1" form="fAdminList" />
  101. <label for="checkedAll">ID</label>
  102. </div>
  103. </th>
  104. <th rowspan="2">게시판</th>
  105. <th colspan="3">댓글</th>
  106. <th rowspan="2">게시글 ID</th>
  107. <th rowspan="2">신고자</th>
  108. <th rowspan="2">사유</th>
  109. <th rowspan="2">등록일</th>
  110. </tr>
  111. <tr>
  112. <th>유형</th>
  113. <th>상태</th>
  114. <th>메모</th>
  115. </tr>
  116. </thead>
  117. <tbody>
  118. @if (Model.List == null || Model.Total <= 0)
  119. {
  120. <tbody>
  121. <tr>
  122. <td colspan="9">No Data.</td>
  123. </tr>
  124. </tbody>
  125. }
  126. else
  127. {
  128. var typeLabels = new Dictionary<ReportType, string> {
  129. { ReportType.None, "-" },
  130. { ReportType.Abuse, "욕설" },
  131. { ReportType.Obscene, "음란" },
  132. { ReportType.Illegal, "불법" },
  133. { ReportType.Impersonation, "사칭" },
  134. { ReportType.CashTrade, "현금거래" },
  135. { ReportType.SpamAd, "스팸/광고" },
  136. { ReportType.Flood, "도배" },
  137. { ReportType.PersonalLeak, "개인정보" },
  138. { ReportType.Other, "기타" }
  139. };
  140. @foreach (var row in Model.List)
  141. {
  142. <tbody class="striped">
  143. <tr>
  144. <td rowspan="2">
  145. <div class="form-check form-check-inline">
  146. <input type="checkbox" name="ids[]" id="ids_@row.ID" class="form-check-input list-check-box" value="@row.ID" form="fAdminList" />
  147. <label for="ids_@row.ID">@row.ID</label>
  148. </div>
  149. </td>
  150. <td rowspan="2">@row.BoardName</td>
  151. <td colspan="3" class="text-start">
  152. [@row.CommentID] @row.Comment
  153. </td>
  154. <td rowspan="2">
  155. <a href="/Forum/Posts/List/Edit/@row.PostID">
  156. @row.PostID
  157. </a>
  158. </td>
  159. <td rowspan="2">@(row.MemberName ?? $"ID:{row.MemberID}")</td>
  160. <td rowspan="2">@row.Reason</td>
  161. <td rowspan="2">@row.CreatedAt</td>
  162. </tr>
  163. <tr>
  164. <td><span class="badge text-bg-warning">@(typeLabels.GetValueOrDefault(row.Type, "-"))</span></td>
  165. <td>
  166. @if (row.Status == ReportStatus.Received)
  167. {
  168. <span class="badge text-bg-danger">접수</span>
  169. }
  170. else if (row.Status == ReportStatus.Processing)
  171. {
  172. <span class="badge text-bg-warning">처리중</span>
  173. }
  174. else if (row.Status == ReportStatus.Completed)
  175. {
  176. <span class="badge text-bg-success">완료</span>
  177. }
  178. </td>
  179. <td>@row.Memo</td>
  180. </tr>
  181. </tbody>
  182. }
  183. }
  184. </tbody>
  185. </table>
  186. <partial name="_Pagination" model="@Model.Pagination" />
  187. </div>
  188. <div class="row g-2 align-items-center">
  189. <div class="col-auto">
  190. <select name="status" class="form-select w-auto d-inline-block" form="fAdminList">
  191. <option value="0">접수</option>
  192. <option value="1">처리중</option>
  193. <option value="2">완료</option>
  194. </select>
  195. </div>
  196. <div class="col-auto">
  197. <input type="text" name="memo" class="form-control" style="width:250px" placeholder="메모 (선택)" form="fAdminList" />
  198. </div>
  199. <div class="col-auto">
  200. <button type="button" id="btnUpdateStatus" class="btn btn-primary">상태 변경</button>
  201. </div>
  202. </div>
  203. </div>
  204. <form id="fAdminSearch" method="get" accept-charset="utf-8">
  205. <input type="hidden" name="pageNum" value="@Model.Query.PageNum" />
  206. </form>
  207. <form id="fAdminList" method="post" accept-charset="utf-8">
  208. @Html.AntiForgeryToken()
  209. <input type="hidden" name="pageNum" value="@Model.Query.PageNum" />
  210. <input type="hidden" name="perPage" value="@Model.Query.PerPage" />
  211. <input type="hidden" name="boardID" value="@Model.Query.BoardID" />
  212. <input type="hidden" name="search" value="@Model.Query.Search" />
  213. <input type="hidden" name="keyword" value="@Model.Query.Keyword" />
  214. <input type="hidden" name="startAt" value="@Model.Query.StartAt" />
  215. <input type="hidden" name="endAt" value="@Model.Query.EndAt" />
  216. <input type="hidden" name="status" value="@Model.Query.Status" />
  217. <input type="hidden" name="type" value="@Model.Query.Type" />
  218. </form>
  219. @section Scripts {
  220. <script>
  221. let searchForm = document.getElementById("fAdminSearch");
  222. $(document).on("change", "#perPage", function () {
  223. searchForm.elements["pageNum"].value = "1";
  224. searchForm.submit();
  225. });
  226. document.getElementById("btnUpdateStatus")?.addEventListener("click", function () {
  227. let form = document.getElementById("fAdminList");
  228. form.action = "?handler=UpdateStatus";
  229. form.submit();
  230. });
  231. </script>
  232. }