Handler.cs 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276
  1. using Application.Abstractions.Data;
  2. using Application.Abstractions.Forum;
  3. using Application.Abstractions.Messaging;
  4. using Domain.Entities.Forum.ValueObject;
  5. using Microsoft.EntityFrameworkCore;
  6. namespace Application.Features.Api.Forum.Post.Search;
  7. public sealed class Handler(IAppDbContext db, IBoardPermissionService permissionService) : IQueryHandler<Query, Response>
  8. {
  9. public async Task<Response> Handle(Query request, CancellationToken ct)
  10. {
  11. // 공통 기본 쿼리
  12. var baseQuery = db.Post.AsNoTracking().Include(c => c.Board).Include(c => c.BoardPrefix).Where(c => !c.IsDeleted).AsQueryable();
  13. if (request.BoardID.HasValue)
  14. {
  15. baseQuery = baseQuery.Where(c => c.BoardID == request.BoardID.Value);
  16. }
  17. if (!string.IsNullOrWhiteSpace(request.Keyword))
  18. {
  19. var kw = request.Keyword.Trim();
  20. baseQuery = request.Search switch
  21. {
  22. 0 => baseQuery.Where(c => c.Subject.Contains(kw)),
  23. 1 => baseQuery.Where(c => c.Content.Contains(kw)),
  24. 2 => baseQuery.Where(c => (c.Name != null && c.Name.Contains(kw)) || (c.SID != null && c.SID.Contains(kw))),
  25. _ => baseQuery.Where(c => c.Subject.Contains(kw) || c.Content.Contains(kw))
  26. };
  27. }
  28. if (!string.IsNullOrWhiteSpace(request.StartAt) && DateTime.TryParse(request.StartAt, out var startDate))
  29. {
  30. baseQuery = baseQuery.Where(c => c.CreatedAt >= startDate);
  31. }
  32. if (!string.IsNullOrWhiteSpace(request.EndAt) && DateTime.TryParse(request.EndAt, out var endDate))
  33. {
  34. baseQuery = baseQuery.Where(c => c.CreatedAt <= endDate.AddDays(1));
  35. }
  36. if (request.IsSecret == true)
  37. {
  38. baseQuery = baseQuery.Where(c => c.IsSecret);
  39. }
  40. if (request.IsReply == true)
  41. {
  42. baseQuery = baseQuery.Where(c => c.IsReply);
  43. }
  44. if (request.IsDeleted == true)
  45. {
  46. baseQuery = baseQuery.Where(c => c.IsDeleted);
  47. }
  48. var exceptSpeaker = false;
  49. var exceptNotice = false;
  50. if (request.BoardID.HasValue)
  51. {
  52. var boardMeta = await db.BoardMeta.AsNoTracking().FirstOrDefaultAsync(x => x.BoardID == request.BoardID.Value, ct);
  53. if (boardMeta is not null)
  54. {
  55. exceptSpeaker = boardMeta.List.ExceptSpeaker;
  56. exceptNotice = boardMeta.List.ExceptNotice;
  57. // QnA(1:1 문의) 게시판: 본인 글만 표시 (관리자·매니저 제외)
  58. if (boardMeta.List.Layout == BoardLayout.QnA && request.MemberID.HasValue)
  59. {
  60. var member = await db.Member.AsNoTracking().FirstOrDefaultAsync(x => x.ID == request.MemberID.Value, ct);
  61. if (member is not null && !member.IsAdmin)
  62. {
  63. var mgr = await permissionService.GetBoardManagerAsync(request.BoardID.Value, member.ID, ct);
  64. if (mgr is null)
  65. {
  66. baseQuery = baseQuery.Where(c => c.MemberID == request.MemberID.Value);
  67. }
  68. }
  69. }
  70. }
  71. }
  72. // ── 1) Speaker (전체공지) 조회: 1페이지 && ExceptSpeaker가 아닐 때 ──
  73. var speakerList = new List<Response.Row>();
  74. if (request.Page == 1 && !exceptSpeaker)
  75. {
  76. var speakers = await baseQuery
  77. .Where(c => c.IsSpeaker)
  78. .OrderByDescending(c => c.ID)
  79. .Select(c => new
  80. {
  81. c.ID,
  82. c.BoardID,
  83. BoardName = c.Board.Name,
  84. c.BoardPrefixID,
  85. BoardPrefix = c.BoardPrefix != null ? new { c.BoardPrefix.ID, c.BoardPrefix.BoardID, c.BoardPrefix.Name, c.BoardPrefix.Color, c.BoardPrefix.Posts } : null,
  86. c.Subject,
  87. c.Thumbnail,
  88. c.Name,
  89. c.SID,
  90. c.IsNotice,
  91. c.IsSecret,
  92. c.IsReply,
  93. c.IsSpeaker,
  94. c.IsAnonymous,
  95. c.IsDeleted,
  96. c.Views,
  97. c.Likes,
  98. c.Dislikes,
  99. c.Comments,
  100. c.Images,
  101. c.Medias,
  102. c.Files,
  103. c.Tags,
  104. c.UpdatedAt,
  105. c.CreatedAt
  106. })
  107. .ToListAsync(ct);
  108. speakerList = [..speakers.Select(c => new Response.Row(
  109. Num: 0,
  110. c.ID, c.BoardID, c.BoardName,
  111. c.BoardPrefixID,
  112. c.BoardPrefix != null ? new Response.BoardPrefixData(c.BoardPrefix.ID, c.BoardPrefix.BoardID, c.BoardPrefix.Name, c.BoardPrefix.Color, c.BoardPrefix.Posts) : null,
  113. c.Subject, c.Thumbnail,
  114. c.Name, c.SID, c.IsNotice, c.IsSecret, c.IsReply, c.IsSpeaker,
  115. c.IsAnonymous, c.IsDeleted, c.Views, c.Likes, c.Dislikes, c.Comments,
  116. c.Images, c.Medias, c.Files, c.Tags, c.UpdatedAt, c.CreatedAt
  117. ))];
  118. }
  119. // ── 2) Notice (게시판 공지) 조회: 1페이지 && ExceptNotice가 아닐 때 ──
  120. var noticeList = new List<Response.Row>();
  121. if (request.Page == 1 && !exceptNotice)
  122. {
  123. var notices = await baseQuery
  124. .Where(c => c.IsNotice && !c.IsSpeaker)
  125. .OrderByDescending(c => c.ID)
  126. .Select(c => new
  127. {
  128. c.ID,
  129. c.BoardID,
  130. BoardName = c.Board.Name,
  131. c.BoardPrefixID,
  132. BoardPrefix = c.BoardPrefix != null ? new { c.BoardPrefix.ID, c.BoardPrefix.BoardID, c.BoardPrefix.Name, c.BoardPrefix.Color, c.BoardPrefix.Posts } : null,
  133. c.Subject,
  134. c.Thumbnail,
  135. c.Name,
  136. c.SID,
  137. c.IsNotice,
  138. c.IsSecret,
  139. c.IsReply,
  140. c.IsSpeaker,
  141. c.IsAnonymous,
  142. c.IsDeleted,
  143. c.Views,
  144. c.Likes,
  145. c.Dislikes,
  146. c.Comments,
  147. c.Images,
  148. c.Medias,
  149. c.Files,
  150. c.Tags,
  151. c.UpdatedAt,
  152. c.CreatedAt
  153. })
  154. .ToListAsync(ct);
  155. noticeList = [..notices.Select(c => new Response.Row(
  156. Num: 0,
  157. c.ID, c.BoardID, c.BoardName,
  158. c.BoardPrefixID,
  159. c.BoardPrefix != null ? new Response.BoardPrefixData(c.BoardPrefix.ID, c.BoardPrefix.BoardID, c.BoardPrefix.Name, c.BoardPrefix.Color, c.BoardPrefix.Posts) : null,
  160. c.Subject, c.Thumbnail,
  161. c.Name, c.SID, c.IsNotice, c.IsSecret, c.IsReply, c.IsSpeaker,
  162. c.IsAnonymous, c.IsDeleted, c.Views, c.Likes, c.Dislikes, c.Comments,
  163. c.Images, c.Medias, c.Files, c.Tags, c.UpdatedAt, c.CreatedAt
  164. ))];
  165. }
  166. // ── 3) 일반 글 조회: 공지 제외 + 페이지네이션 ──
  167. var listQuery = baseQuery.Where(c => !c.IsNotice && !c.IsSpeaker);
  168. // IsNotice 필터가 명시적으로 true인 경우 (관리 목적 등) 공지만 조회
  169. if (request.IsNotice == true)
  170. {
  171. listQuery = baseQuery.Where(c => c.IsNotice);
  172. }
  173. if (request.BoardPrefixID.HasValue)
  174. {
  175. listQuery = listQuery.Where(c => c.BoardPrefixID == request.BoardPrefixID.Value);
  176. }
  177. listQuery = request.Sort switch
  178. {
  179. 1 => listQuery.OrderByDescending(c => c.Views),
  180. 2 => listQuery.OrderByDescending(c => c.Comments),
  181. 3 => listQuery.OrderByDescending(c => c.Likes),
  182. _ => listQuery.OrderByDescending(c => c.ID)
  183. };
  184. var total = await listQuery.CountAsync(ct);
  185. var list = await listQuery
  186. .Skip((request.Page - 1) * request.PerPage)
  187. .Take(request.PerPage)
  188. .Select(c => new
  189. {
  190. c.ID,
  191. c.BoardID,
  192. BoardName = c.Board.Name,
  193. c.BoardPrefixID,
  194. BoardPrefix = c.BoardPrefix != null ? new { c.BoardPrefix.ID, c.BoardPrefix.BoardID, c.BoardPrefix.Name, c.BoardPrefix.Color, c.BoardPrefix.Posts } : null,
  195. c.Subject,
  196. c.Thumbnail,
  197. c.Name,
  198. c.SID,
  199. c.IsNotice,
  200. c.IsSecret,
  201. c.IsReply,
  202. c.IsSpeaker,
  203. c.IsAnonymous,
  204. c.IsDeleted,
  205. c.Views,
  206. c.Likes,
  207. c.Dislikes,
  208. c.Comments,
  209. c.Images,
  210. c.Medias,
  211. c.Files,
  212. c.Tags,
  213. c.UpdatedAt,
  214. c.CreatedAt
  215. })
  216. .ToListAsync(ct);
  217. var startNum = total - ((request.Page - 1) * request.PerPage);
  218. return new Response(
  219. total,
  220. speakerList,
  221. noticeList,
  222. [..list.Select((c, i) => new Response.Row(
  223. Num: startNum - i,
  224. c.ID,
  225. c.BoardID,
  226. c.BoardName,
  227. c.BoardPrefixID,
  228. c.BoardPrefix != null ? new Response.BoardPrefixData(c.BoardPrefix.ID, c.BoardPrefix.BoardID, c.BoardPrefix.Name, c.BoardPrefix.Color, c.BoardPrefix.Posts) : null,
  229. c.Subject,
  230. c.Thumbnail,
  231. c.Name,
  232. c.SID,
  233. c.IsNotice,
  234. c.IsSecret,
  235. c.IsReply,
  236. c.IsSpeaker,
  237. c.IsAnonymous,
  238. c.IsDeleted,
  239. c.Views,
  240. c.Likes,
  241. c.Dislikes,
  242. c.Comments,
  243. c.Images,
  244. c.Medias,
  245. c.Files,
  246. c.Tags,
  247. c.UpdatedAt,
  248. c.CreatedAt
  249. ))]
  250. );
  251. }
  252. }