Handler.cs 3.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788
  1. using Application.Abstractions.Messaging;
  2. using Application.Abstractions.Data;
  3. using Domain.Entities.Wallets.ValueObject;
  4. using Microsoft.EntityFrameworkCore;
  5. namespace Application.Features.Admin.Member.Wallet.Transactions.Search;
  6. public sealed class Handler(IAppDbContext db) : IQueryHandler<Query, Response>
  7. {
  8. public async Task<Response> Handle(Query request, CancellationToken ct)
  9. {
  10. var query = db.WalletTransaction.AsNoTracking().Include(x => x.Wallet).ThenInclude(w => w.Member).AsQueryable();
  11. // 거래 유형 필터
  12. if (request.Type.HasValue)
  13. {
  14. query = query.Where(x => x.TxType == request.Type.Value);
  15. }
  16. // 키워드 검색
  17. if (!string.IsNullOrWhiteSpace(request.Keyword))
  18. {
  19. query = request.Search switch
  20. {
  21. 1 => query.Where(x => x.Wallet.Member.Email.Contains(request.Keyword)),
  22. 2 => query.Where(x => x.ID.ToString().Contains(request.Keyword)),
  23. 3 => query.Where(x => x.Wallet.ID.ToString().Contains(request.Keyword)),
  24. 4 => query.Where(x => x.Wallet.MemberID.ToString().Contains(request.Keyword)),
  25. 5 => query.Where(x => x.Wallet.Member.Name != null && x.Wallet.Member.Name.Contains(request.Keyword)),
  26. _ => query
  27. };
  28. }
  29. // 날짜 필터
  30. if (!string.IsNullOrWhiteSpace(request.StartAt) && DateTime.TryParse(request.StartAt, out var startAt))
  31. {
  32. query = query.Where(x => x.CreatedAt >= startAt);
  33. }
  34. if (!string.IsNullOrWhiteSpace(request.EndAt) && DateTime.TryParse(request.EndAt, out var endAt))
  35. {
  36. query = query.Where(x => x.CreatedAt <= endAt);
  37. }
  38. var total = await query.CountAsync(ct);
  39. // 유형별 카운트 (GroupBy 1회 쿼리)
  40. var typeCounts = await db.WalletTransaction.AsNoTracking().GroupBy(x => x.TxType).Select(g => new { Type = g.Key, Count = g.Count() }).ToListAsync(ct);
  41. int CountOf(WalletTransactionType t) => typeCounts.FirstOrDefault(x => x.Type == t)?.Count ?? 0;
  42. var skip = (request.PageNum - 1) * request.PerPage;
  43. var list = await query.OrderByDescending(x => x.ID).Skip(skip).Take(request.PerPage).Select(x => new { x.ID, x.WalletKey, MemberID = x.Wallet.MemberID, MemberEmail = x.Wallet.Member.Email, MemberName = x.Wallet.Member.Name, x.TxType, x.BalanceType, Amount = x.Amount.Value, BalanceAfter = x.BalanceAfter.Value, x.Reason, x.RefID, x.Memo, x.CreatedAt }).ToListAsync(ct);
  44. var rows = list.Select((x, idx) => new Response.Row
  45. {
  46. Num = total - skip - idx,
  47. ID = x.ID,
  48. WalletKey = x.WalletKey,
  49. MemberID = x.MemberID,
  50. MemberEmail = x.MemberEmail,
  51. MemberName = x.MemberName,
  52. TxType = x.TxType,
  53. BalanceType = x.BalanceType,
  54. Amount = (long)x.Amount,
  55. BalanceAfter = (long)x.BalanceAfter,
  56. Reason = x.Reason,
  57. RefID = x.RefID,
  58. Memo = x.Memo,
  59. CreatedAt = x.CreatedAt
  60. }).ToList();
  61. return new Response
  62. {
  63. Total = total,
  64. TotalCharge = CountOf(WalletTransactionType.Charge),
  65. TotalDonationIn = CountOf(WalletTransactionType.DonationIn),
  66. TotalDonationOut = CountOf(WalletTransactionType.DonationOut),
  67. TotalRewardEarned = CountOf(WalletTransactionType.RewardEarned),
  68. TotalSpend = CountOf(WalletTransactionType.Spend),
  69. TotalRefund = CountOf(WalletTransactionType.Refund),
  70. TotalLock = CountOf(WalletTransactionType.Lock),
  71. TotalUnlock = CountOf(WalletTransactionType.Unlock),
  72. TotalAdjusted = CountOf(WalletTransactionType.Adjusted),
  73. List = rows
  74. };
  75. }
  76. }