Write.cshtml 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. @page
  2. @model Admin.Pages.Channel.List.WriteModel
  3. @{
  4. ViewData["Title"] = "채널 등록";
  5. }
  6. <div class="container">
  7. <h3>@ViewData["Title"]</h3>
  8. <hr />
  9. <partial name="_StatusMessage" />
  10. <div class="alert alert-success" role="alert">
  11. 채널 등록 시 YouTube API를 통해 자동 수집된 정보입니다.<br />
  12. 관리자가 직접 등록 시 잘못된 결과가 발생할 수 있습니다.
  13. </div>
  14. <form id="fAdminWrite" method="post" accept-charset="utf-8" autocomplete="off">
  15. <div class="row mb-2">
  16. <label class="col-sm-2 col-form-label"><span class="text-danger">*</span> 회원(소유자)</label>
  17. <div class="col-sm-10">
  18. <input type="hidden" asp-for="Input.MemberID" id="memberIdHidden" />
  19. <div id="memberSelected" class="mb-1" style="display:none;">
  20. <span class="badge bg-primary fs-6 fw-normal" id="memberBadge">
  21. <span id="memberBadgeText"></span>
  22. <button type="button" class="btn-close btn-close-white ms-2" aria-label="제거" id="memberRemoveBtn" style="font-size:0.6em;vertical-align:middle;"></button>
  23. </span>
  24. </div>
  25. <div class="position-relative" id="memberSearchWrap">
  26. <div class="input-group">
  27. <span class="input-group-text"><i class="bi bi-search"></i></span>
  28. <input type="text" class="form-control" id="memberSearchInput" placeholder="회원 ID, SID, 이메일로 검색" autocomplete="off" />
  29. </div>
  30. <div class="list-group position-absolute w-100 shadow" id="memberSearchResults" style="z-index:1050;display:none;max-height:300px;overflow-y:auto;"></div>
  31. </div>
  32. </div>
  33. </div>
  34. <div class="row mb-2">
  35. <label asp-for="Input.SID" class="col-sm-2 col-form-label"><span class="text-danger">*</span> SID</label>
  36. <div class="col-sm-10">
  37. <input type="text" asp-for="Input.SID" class="form-control" required minlength="24" maxlength="24" placeholder="중복 시 등록이 불가합니다. 24자" />
  38. <div class="text-muted form-text">
  39. YouTube 채널 고유 ID (예: UCxxxxxxxxxxxxxxxxxxxxxx)
  40. </div>
  41. </div>
  42. </div>
  43. <div class="row mb-2">
  44. <label asp-for="Input.Name" class="col-sm-2 col-form-label"><span class="text-danger">*</span> 이름</label>
  45. <div class="col-sm-10">
  46. <input type="text" asp-for="Input.Name" class="form-control" required maxlength="200" />
  47. </div>
  48. </div>
  49. <div class="row mb-2">
  50. <label asp-for="Input.Handle" class="col-sm-2 col-form-label">핸들</label>
  51. <div class="col-sm-10">
  52. <div class="input-group">
  53. <span class="input-group-text">@@</span>
  54. <input type="text" asp-for="Input.Handle" class="form-control" maxlength="30" />
  55. </div>
  56. </div>
  57. </div>
  58. <div class="row mb-2">
  59. <label asp-for="Input.YouTubeUrl" class="col-sm-2 col-form-label"><span class="text-danger">*</span> YouTube 주소</label>
  60. <div class="col-sm-10">
  61. <input type="url" asp-for="Input.YouTubeUrl" class="form-control" required maxlength="255" />
  62. <div class="text-muted form-text">
  63. YouTube 채널 주소 (예: https://www.youtube.com/channel/UCxxxxxxxxxxxxxxxxxxxxxx)
  64. </div>
  65. </div>
  66. </div>
  67. <div class="row mb-2">
  68. <label asp-for="Input.PlatformFeeRate" class="col-sm-2 col-form-label"><span class="text-danger">*</span> 수수료(%)</label>
  69. <div class="col-sm-10">
  70. <div class="row">
  71. <div class="col col-md-auto">
  72. <div class="input-group">
  73. <input type="number" asp-for="Input.PlatformFeeRate" class="form-control" required min="0" max="100" step="0.1" />
  74. <span class="input-group-text">%</span>
  75. </div>
  76. </div>
  77. </div>
  78. <span asp-validation-for="Input.PlatformFeeRate" class="text-danger"></span>
  79. </div>
  80. </div>
  81. <div class="row mb-2">
  82. <label asp-for="Input.IsVerified" class="col-sm-2 col-form-label">인증 여부</label>
  83. <div class="col-sm-10 align-content-center">
  84. <div class="form-check form-check-inline">
  85. <input type="checkbox" asp-for="Input.IsVerified" class="form-check-input" />
  86. <label class="form-check-label" asp-for="Input.IsVerified">인증합니다.</label>
  87. </div>
  88. </div>
  89. </div>
  90. <div class="row mb-2">
  91. <label asp-for="Input.IsActive" class="col-sm-2 col-form-label">사용 여부</label>
  92. <div class="col-sm-10 align-content-center">
  93. <div class="form-check form-check-inline">
  94. <input type="checkbox" asp-for="Input.IsActive" class="form-check-input" />
  95. <label class="form-check-label" asp-for="Input.IsActive">사용합니다.</label>
  96. </div>
  97. </div>
  98. </div>
  99. <hr />
  100. <div class="d-grid gap-2 text-center d-md-block">
  101. <button type="submit" class="btn btn-success">저장</button>
  102. <a href="/Channel/List" class="btn btn-secondary">취소</a>
  103. </div>
  104. <br />
  105. </form>
  106. </div>
  107. @section Scripts {
  108. <script>
  109. $(function () {
  110. let timer = null;
  111. const $input = $('#memberSearchInput');
  112. const $results = $('#memberSearchResults');
  113. const $hidden = $('#memberIdHidden');
  114. const $selected = $('#memberSelected');
  115. const $badgeText = $('#memberBadgeText');
  116. const $searchWrap = $('#memberSearchWrap');
  117. function showSelected(id, email, name, sid) {
  118. const display = id + ' (' + email + ')' + (name ? ' ' + name : '') + (sid ? ' ' + sid : '');
  119. $badgeText.text(display);
  120. $hidden.val(id);
  121. $selected.show();
  122. $searchWrap.hide();
  123. $results.hide();
  124. }
  125. function clearSelected() {
  126. $hidden.val('');
  127. $selected.hide();
  128. $searchWrap.show();
  129. $input.val('').focus();
  130. }
  131. $('#memberRemoveBtn').on('click', function () {
  132. clearSelected();
  133. });
  134. $input.on('input', function () {
  135. clearTimeout(timer);
  136. const val = $(this).val().trim();
  137. if (val.length < 1) {
  138. $results.hide().empty();
  139. return;
  140. }
  141. timer = setTimeout(function () {
  142. $.getJSON('/Channel/List/Write?handler=SearchMember&keyword=' + encodeURIComponent(val), function (data) {
  143. $results.empty();
  144. if (data.length === 0) {
  145. $results.append('<div class="list-group-item text-muted">검색 결과가 없습니다.</div>');
  146. } else {
  147. $.each(data, function (i, m) {
  148. const text = m.id + ' (' + m.email + ')' + (m.name ? ' ' + m.name : '') + (m.sid ? ' ' + m.sid : '');
  149. $results.append(
  150. $('<a href="#" class="list-group-item list-group-item-action"></a>')
  151. .text(text)
  152. .on('click', function (e) {
  153. e.preventDefault();
  154. showSelected(m.id, m.email, m.name, m.sid);
  155. })
  156. );
  157. });
  158. }
  159. $results.show();
  160. });
  161. }, 300);
  162. });
  163. $(document).on('click', function (e) {
  164. if (!$(e.target).closest('#memberSearchWrap').length) {
  165. $results.hide();
  166. }
  167. });
  168. $input.on('focus', function () {
  169. if ($results.children().length > 0) {
  170. $results.show();
  171. }
  172. });
  173. // 폼 제출 시 회원 선택 여부 확인
  174. $('#fAdminWrite').on('submit', function (e) {
  175. if (!$hidden.val() || $hidden.val() === '0') {
  176. e.preventDefault();
  177. alert('회원(소유자)을 선택해주세요.');
  178. $input.focus();
  179. }
  180. });
  181. });
  182. </script>
  183. }