evtListPending.vue 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  1. <template>
  2. <div>
  3. <div class="inner--headers">
  4. <h2>{{ pageId }}</h2>
  5. <div class="bread--crumbs--wrap">
  6. <span>홈</span>
  7. <span>이벤트 관리</span>
  8. <span>{{ pageId }}</span>
  9. </div>
  10. </div>
  11. <div class="search--modules">
  12. <div class="form--cont--filter">
  13. <v-select
  14. v-model="filter"
  15. :items="filterArr"
  16. variant="outlined"
  17. class="custom-select"
  18. >
  19. </v-select>
  20. </div>
  21. <div class="form--cont--text">
  22. <v-text-field
  23. v-model="searchModel"
  24. class="custom-input mini"
  25. style="width: 100%"
  26. placeholder="검색어를 입력하세요"
  27. ></v-text-field>
  28. </div>
  29. <v-btn
  30. class="custom-btn btn-blue mini sch--btn"
  31. @click="fnSearch(searchModel, filter)"
  32. >검색</v-btn
  33. >
  34. </div>
  35. <div class="data--list--wrap">
  36. <div class="btn--actions--wrap">
  37. <div class="left--sections">
  38. <v-btn class="custom-btn mini btn-white" @click="fnDelEvt">선택 삭제</v-btn>
  39. </div>
  40. <div class="right--sections">
  41. <v-btn class="custom-btn mini btn-reg ml--10" @click="addLocated()"
  42. ><i class="ico"></i>신규 등록</v-btn
  43. >
  44. </div>
  45. </div>
  46. <div class="tbl-wrapper">
  47. <div class="tbl-wrap">
  48. <!-- ag grid -->
  49. <ag-grid-vue
  50. style="width: 100%; height: calc(10 * 2.94rem)"
  51. class="ag-theme-quartz"
  52. :gridOptions="gridOptions"
  53. :rowData="tblItems"
  54. rowSelection="multiple"
  55. :paginationPageSize="pageObj.pageSize"
  56. :suppressPaginationPanel="true"
  57. @grid-ready="onGridReady"
  58. @rowClicked="detailLocated"
  59. >
  60. </ag-grid-vue>
  61. <!-- 페이징 -->
  62. <div class="ag-grid-custom-pagenations">
  63. <pagination @chg_page="chgPage" :pageObj="pageObj"></pagination>
  64. </div>
  65. </div>
  66. </div>
  67. </div>
  68. </div>
  69. </template>
  70. <script setup>
  71. import { AgGridVue } from "ag-grid-vue3";
  72. import pagination from "../components/common/pagination.vue";
  73. /************************************************************************
  74. | 레이아웃
  75. ************************************************************************/
  76. definePageMeta({
  77. layout: "default",
  78. });
  79. /************************************************************************
  80. | PROPS
  81. ************************************************************************/
  82. const props = defineProps({
  83. propsData: {
  84. type: Object,
  85. default: () => {},
  86. },
  87. });
  88. /************************************************************************
  89. | 스토어
  90. ************************************************************************/
  91. const useDtStore = useDetailStore();
  92. /************************************************************************
  93. | 전역
  94. ************************************************************************/
  95. const searchModel = ref("");
  96. const filter = ref("");
  97. const filterArr = ref([
  98. { title: "선택하세요", value: "" },
  99. { title: "제목", value: "title" },
  100. ]);
  101. const { $toast, $log, $dayjs, $eventBus } = useNuxtApp();
  102. const router = useRouter();
  103. const pageId = ref("대기중 이벤트");
  104. let pageObj = ref({
  105. page: 1, // 현재 페이지
  106. pageMaxNumSize: 10, // 페이지 숫자 최대 표현 개수
  107. pageSize: 10, // 테이블 조회 데이터 개수
  108. totalCnt: 0, // 전체 페이지
  109. });
  110. const tblItems = ref([]); // stat 데이터
  111. /* eslint-disable */
  112. /* prettier-ignore */
  113. pageObj.value.totalCnt = tblItems.value.length;
  114. const remToPx = () => parseFloat(getComputedStyle(document.documentElement).fontSize);
  115. const rowHeightRem = 2.65; // 원하는 rem 값
  116. const rowHeightPx = rowHeightRem * remToPx();
  117. const gridApi = shallowRef();
  118. // gridOption
  119. const gridOptions = {
  120. columnDefs: [
  121. { checkboxSelection: true, headerCheckboxSelection: true, width: 0 },
  122. {
  123. headerName: "No",
  124. valueGetter: (params) => params.api.getDisplayedRowCount() - params.node.rowIndex,
  125. sortable: false,
  126. width: 70,
  127. },
  128. { headerName: "제목", field: "TITLE", sortable: false },
  129. {
  130. headerName: "기간",
  131. field: "STARTDATE",
  132. sortable: false,
  133. cellRenderer: (params) => {
  134. const formatDate = (str) => {
  135. if (!str) return "";
  136. const d = new Date(str);
  137. const yyyy = d.getFullYear();
  138. const mm = String(d.getMonth() + 1).padStart(2, "0");
  139. const dd = String(d.getDate()).padStart(2, "0");
  140. return `${yyyy}-${mm}-${dd}`;
  141. };
  142. const start = formatDate(params.data.STARTDATE);
  143. const end = formatDate(params.data.ENDDATE);
  144. return `${start} ~ ${end}`;
  145. },
  146. },
  147. {
  148. headerName: "상태",
  149. field: "STATUS",
  150. sortable: false,
  151. width: 70,
  152. cellRenderer: (params) => {
  153. return params.value == 0 ? "대기중" : params.value == 1 ? "진행중" : "마감";
  154. },
  155. },
  156. { headerName: "등록일", field: "REGDATE", sortable: false, width: 140 },
  157. // {
  158. // headerName: "알림 메일 수신 여부",
  159. // field: "mail_recp_yn",
  160. // sortable: false,
  161. // width: 130,
  162. // },
  163. ],
  164. rowData: tblItems.value, // 테이블 데이터
  165. autoSizeStrategy: {
  166. type: "fitGridWidth", // width맞춤
  167. },
  168. suppressMovableColumns: true,
  169. headerHeight: rowHeightPx,
  170. rowHeight: rowHeightPx,
  171. pagination: true,
  172. suppressPaginationPanel: true, // 하단 default 페이징 컨트롤 숨김
  173. //rowSelection: {
  174. // checkboxes: true,
  175. // headerCheckbox: true,
  176. // enableClickSelection: false,
  177. // mode: "multiRow",
  178. //},
  179. };
  180. /************************************************************************
  181. | 함수(METHODS)
  182. ************************************************************************/
  183. const onGridReady = (__PARAMS) => {
  184. gridApi.value = __PARAMS.api;
  185. };
  186. const chgPage = (__PAGE) => {
  187. pageObj.value.page = __PAGE;
  188. gridApi.value.paginationGoToPage(__PAGE - 1);
  189. };
  190. const addLocated = () => {
  191. router.push({
  192. path: "/view/event/evtAdd",
  193. });
  194. useDtStore.boardInfo.pageType = "I";
  195. };
  196. const detailLocated = (__EVENT) => {
  197. router.push({
  198. path: "/view/event/evtAdd",
  199. });
  200. useDtStore.boardInfo.seq = __EVENT.data.SEQ;
  201. useDtStore.boardInfo.pageType = "U";
  202. useDtStore.boardInfo.status = "0";
  203. };
  204. const evtListGet = () => {
  205. let _req = {
  206. compId: useAuthStore().getCompanyId,
  207. status: "0",
  208. };
  209. useAxios()
  210. .post("/evt/list", _req)
  211. .then((res) => {
  212. _req._size = res.data.length;
  213. tblItems.value = res.data;
  214. pageObj.value.totalCnt = tblItems.value.length;
  215. });
  216. };
  217. const fnSearch = (__KEYWORD, __FILTER) => {
  218. let _req = {
  219. compId: useAuthStore().getCompanyId,
  220. filter: __FILTER,
  221. keyword: __KEYWORD,
  222. status: "0",
  223. // _size: 1000,
  224. // _index: 0,
  225. // admin_name:
  226. // __FILTER == "admin_name" ? __KEYWORD : __FILTER == "" ? __KEYWORD : null,
  227. // id: __FILTER == "id" ? __KEYWORD : __FILTER == "" ? __KEYWORD : null,
  228. };
  229. useAxios()
  230. .post("/evt/search", _req)
  231. .then((res) => {
  232. _req._size = res.data.length;
  233. tblItems.value = res.data;
  234. pageObj.value.totalCnt = tblItems.value.length;
  235. })
  236. .catch((error) => {});
  237. };
  238. const fnDelete = async (seqArr) => {
  239. if (!seqArr.length) return;
  240. for (const seq of seqArr) {
  241. await useAxios().post(`evt/delete/${seq}`);
  242. }
  243. window.location.reload();
  244. };
  245. const fnDelEvt = () => {
  246. const selected = gridApi.value.getSelectedRows();
  247. if (selected.length === 0) {
  248. let param = {
  249. id: pageId,
  250. title: "이벤트 삭제",
  251. content: "삭제할 이벤트를 선택하세요.",
  252. yes: {
  253. text: "확인",
  254. isProc: false,
  255. },
  256. };
  257. $eventBus.emit("OPEN_CONFIRM_POP_UP", param);
  258. } else {
  259. let param = {
  260. id: pageId,
  261. title: "이벤트 삭제",
  262. content: "삭제하시겠습니까?",
  263. yes: {
  264. text: "확인",
  265. isProc: true,
  266. event: "FN_DELETE",
  267. param: selected,
  268. },
  269. no: {
  270. text: "취소",
  271. isProc: false,
  272. },
  273. };
  274. $eventBus.emit("OPEN_CONFIRM_POP_UP", param);
  275. }
  276. };
  277. /************************************************************************
  278. | 팝업 이벤트버스 정의
  279. ************************************************************************/
  280. $eventBus.off("FN_DELETE");
  281. $eventBus.on("FN_DELETE", (selected) => {
  282. const seqList = selected.map((row) => row.SEQ);
  283. fnDelete(seqList);
  284. });
  285. /************************************************************************
  286. | WATCH
  287. ************************************************************************/
  288. watch(
  289. () => props,
  290. () => {
  291. searchObj.value = props.propsData;
  292. fnGetStat();
  293. },
  294. { deep: true }
  295. );
  296. onMounted(() => {
  297. evtListGet();
  298. });
  299. </script>