| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479 |
- <template>
- <div>
- <div class="inner--headers">
- <h2>{{ pageId }}</h2>
- <div class="bread--crumbs--wrap">
- <span>홈</span>
- <span>{{ pageId }}</span>
- </div>
- </div>
- <div class="search--modules type2">
- <div class="search--inner">
- <div class="form--cont--filter">
- <v-select
- v-model="filter"
- :items="filderArr"
- variant="outlined"
- class="custom-select"
- >
- </v-select>
- </div>
- <div class="form--cont--text">
- <v-text-field
- v-model="searchModel"
- class="custom-input mini"
- style="width: 100%"
- placeholder="검색어를 입력하세요"
- ></v-text-field>
- </div>
- </div>
- <div class="search--inner">
- <div class="calendar-wrap ml--0">
- <div class="calendar">
- <VueDatePicker
- :format="datePickerFormat"
- v-model="searchStartDate"
- placeholder="날짜를 선택하세요"
- :auto-apply="true"
- week-start="0"
- ></VueDatePicker>
- </div>
- <span class="text">~</span>
- <div class="calendar">
- <VueDatePicker
- v-model="searchEndDate"
- :format="datePickerFormat"
- placeholder="날짜를 선택하세요"
- :auto-apply="true"
- week-start="0"
- ></VueDatePicker>
- </div>
- <div class="month--selector">
- <v-btn
- v-for="option in dateOptions"
- :key="option.value"
- :class="{ actv: selectedRange === option.value }"
- @click="setDateRange(option.value)"
- elevation="0"
- >
- {{ option.label }}
- </v-btn>
- </div>
- </div>
- </div>
- <v-btn
- class="custom-btn btn-blue mini sch--btn"
- @click="fnSearch(searchModel, filter)"
- >검색</v-btn
- >
- </div>
- <div class="data--list--wrap">
- <div class="item--list--wrap">
- <div class="cs--list--wrap">
- <div class="cs--header">
- <h3>문의 등록</h3>
- <v-btn class="custom-btn" @click="addLocated()">문의 등록</v-btn>
- </div>
- <div class="cs--header">
- <h3>나의 문의 내역</h3>
- </div>
- <div class="cs--list" v-if="csList.length > 0">
- <div v-for="(items, index) in paginatedItems" :key="index">
- <div class="list" @click="toItemDetail(items.SEQ)">
- <span class="list--seq">{{ items.SEQ }}</span>
- <span class="list--circle">{{ items.CATEGORY == 'D' ? '기능문의' : '기타문의' }}</span>
- <span class="list--writer">{{ items.NICK_NAME ? items.NICK_NAME : items.COMPANY_NAME }}</span>
- <span class="list--title">{{ items.TITLE }}</span>
- <span class="list--ml" :class="getStatusClass(items.STATUS)">{{ items.STATUS == '0' ? "답변대기" : "답변완료" }}</span>
- <span class="list--date">{{ formatDate(items.REGDATE) }}</span>
- </div>
- </div>
- </div>
- <div class="cs--list" v-else>
- <div class="no-data">
- <h4>등록된 문의가 없습니다</h4>
- <p>새로운 문의를 등록해보세요</p>
- </div>
- </div>
- </div>
- <div class="item--pagination" v-if="csList.length > 0">
- <v-pagination
- v-model="currentPage"
- :length="Math.ceil(csList.length / itemsPerPage)"
- ></v-pagination>
- </div>
- </div>
- </div>
- <!-- 문의 등록 팝업 -->
- <v-dialog v-model="showInquiryModal" max-width="600px" persistent>
- <v-card class="inquiry-modal">
- <v-card-title class="modal-header">
- <h3>문의 등록</h3>
- <v-btn icon @click="closeModal" class="close-btn">
- <v-icon>mdi-close</v-icon>
- </v-btn>
- </v-card-title>
- <v-card-text class="modal-body">
- <form @submit.prevent="submitInquiry">
- <div class="form-group">
- <label>문의 유형 <span class="required">*</span></label>
- <v-select
- v-model="inquiryForm.category"
- :items="categoryOptions"
- variant="outlined"
- placeholder="문의 유형을 선택하세요"
- class="custom-select"
- :rules="[v => !!v || '문의 유형을 선택해주세요']"
- ></v-select>
- </div>
- <div class="form-group">
- <label>제목 <span class="required">*</span></label>
- <v-text-field
- v-model="inquiryForm.title"
- variant="outlined"
- placeholder="문의 제목을 입력하세요"
- class="custom-input"
- :rules="[v => !!v || '제목을 입력해주세요']"
- ></v-text-field>
- </div>
- <div class="form-group">
- <label>내용 <span class="required">*</span></label>
- <v-textarea
- v-model="inquiryForm.content"
- variant="outlined"
- placeholder="문의 내용을 상세히 입력해주세요"
- rows="6"
- class="custom-textarea"
- :rules="[v => !!v || '내용을 입력해주세요']"
- ></v-textarea>
- </div>
- <div class="form-group">
- <label>첨부파일</label>
- <v-file-input
- v-model="inquiryForm.attachments"
- variant="outlined"
- placeholder="파일을 선택하세요"
- hide-details=""
- prepend-icon=""
- append-inner-icon="mdi-paperclip"
- class="custom-file-input mb--30"
- accept="image/*,.pdf,.doc,.docx,.hwp"
- multiple
- show-size
- ></v-file-input>
- <p class="file-info">* 이미지, PDF, 문서 파일만 업로드 가능 (최대 10MB)</p>
- </div>
- </form>
- </v-card-text>
- <v-card-actions class="modal-footer">
- <v-btn @click="closeModal" class="cancel-btn">취소</v-btn>
- <v-btn @click="submitInquiry" class="submit-btn" :loading="isSubmitting">등록하기</v-btn>
- </v-card-actions>
- </v-card>
- </v-dialog>
- </div>
- </template>
- <script setup>
- import VueDatePicker from "@vuepic/vue-datepicker";
- import "@vuepic/vue-datepicker/dist/main.css";
- import dayjs from 'dayjs';
- /************************************************************************
- | 레이아웃
- ************************************************************************/
- definePageMeta({
- layout: "default",
- });
- /************************************************************************
- | PROPS
- ************************************************************************/
- const props = defineProps({
- propsData: {
- type: Object,
- default: () => {},
- },
- });
- /************************************************************************
- | 스토어
- ************************************************************************/
- const useDtStore = useDetailStore();
- const useAtStore = useAuthStore();
- /************************************************************************
- | 전역
- ************************************************************************/
- const memberType = useAtStore.auth.memberType;
- const searchModel = ref("");
- const selectedRange = ref('all');
- const searchStartDate = ref("");
- const searchEndDate = ref("");
- const dateOptions = [
- { label: '오늘', value: 'today' },
- { label: '7일', value: '7d' },
- { label: '1개월', value: '1m' },
- { label: '3개월', value: '3m' },
- { label: '전체', value: 'all' },
- ]
- const datePickerFormat = "yyyy-MM-dd";
- const filter = ref("");
- const filderArr = ref([
- { title: "전체", value: "" },
- { title: "제목", value: "title" },
- { title: "내용", value: "content" },
- { title: "작성자", value: "writer" }
- ]);
- const { $toast, $log, $dayjs, $eventBus } = useNuxtApp();
- const router = useRouter();
- const pageId = computed(() => {
- return '고객센터';
- });
- const csList = ref([]);
- const itemsPerPage = 7;
- const currentPage = ref(1);
- // 문의 등록 팝업 관련
- const showInquiryModal = ref(false);
- const isSubmitting = ref(false);
- const inquiryForm = ref({
- category: '',
- title: '',
- content: '',
- attachments: []
- });
- const categoryOptions = ref([
- { title: '기능문의', value: 'D' },
- { title: '기타문의', value: 'E' }
- ]);
- /* eslint-disable */
- /* prettier-ignore */
- /************************************************************************
- | 함수(METHODS)
- ************************************************************************/
- const isRecentUpdate = (dateStr) => {
- const today = new Date();
- const updateDate = new Date(dateStr);
- const diffDays = (today - updateDate) / (1000 * 60 * 60 * 24);
- // 업데이트 날짜가 오늘 날짜 기준 최근 7일인지 확인
- return diffDays <= 7;
- }
- const paginatedItems = computed(() => {
- const start = (currentPage.value - 1) * itemsPerPage;
- return csList.value.slice(start, start + itemsPerPage);
- });
- const setDateRange = (range) => {
- const today = dayjs();
- switch(range) {
- case 'today' :
- searchStartDate.value = today.format('YYYY-MM-DD');
- searchEndDate.value = today.format('YYYY-MM-DD');
- selectedRange.value = 'today';
- break;
- case '7d':
- searchStartDate.value = today.subtract(7, 'day').format('YYYY-MM-DD');
- searchEndDate.value = today.format('YYYY-MM-DD');
- selectedRange.value = '7d';
- break;
- case '1m':
- searchStartDate.value = today.subtract(1, 'month').format('YYYY-MM-DD');
- searchEndDate.value = today.format('YYYY-MM-DD');
- selectedRange.value = '1m';
- break;
- case '3m':
- searchStartDate.value = today.subtract(3, 'month').format('YYYY-MM-DD');
- searchEndDate.value = today.format('YYYY-MM-DD');
- selectedRange.value = '3m';
- break;
- case 'all':
- searchStartDate.value = "";
- searchEndDate.value = today.format('YYYY-MM-DD');
- selectedRange.value = 'all';
- break
- }
- }
- const addLocated = () => {
- showInquiryModal.value = true;
- };
- const toItemDetail = (__EVENT) => {
- router.push({
- path: "/view/common/cs/detail",
- });
- useDtStore.boardInfo.seq = __EVENT;
- };
- const csListGet = async () => {
- let _req = {
- USER_SEQ : useAtStore.auth.seq,
- keyword: '',
- filter: '',
- startDate: '',
- endDate: ''
- };
- if(useAtStore.auth.seq == 2){
- _req.USER_SEQ = 0;
- }
- await useAxios()
- .post("/cs/list", _req)
- .then((res) => {
- csList.value = res.data;
- });
- };
- const fnSearch = (__KEYWORD, __FILTER) => {
- let _req = {
- USER_SEQ: useAtStore.auth.seq,
- filter: __FILTER,
- keyword: __KEYWORD,
- startDate: searchStartDate.value ? dayjs(searchStartDate.value).format('YYYY-MM-DD') : '',
- endDate: searchEndDate.value ? dayjs(searchEndDate.value).format('YYYY-MM-DD') : ''
- };
- // 관리자인 경우 모든 문의 조회
- if(useAtStore.auth.seq == 2){
- _req.USER_SEQ = 0;
- }
- useAxios()
- .post("/cs/search", _req)
- .then((res) => {
- csList.value = res.data;
- currentPage.value = 1; // 검색 후 첫 페이지로 이동
- })
- .catch((error) => {
- console.error('검색 실패:', error);
- $toast.error('검색에 실패했습니다.');
- });
- };
- const goToDeliveryDetail = (item) => {
- // 제품 정보를 스토어에 저장
- useDtStore.boardInfo.seq = item.SEQ;
- useDtStore.boardInfo.pageType = "D";
-
- // 배송 관리 페이지로 이동
- router.push({
- path: "/view/common/deli/detail",
- query: {
- itemId: item.SEQ,
- itemName: item.NAME,
- price1: item.PRICE1,
- price2: item.PRICE2 || item.PRICE1,
- thumbFile: item.THUMB_FILE || ''
- }
- });
- };
- const getStatusClass = (status) => {
- switch(status) {
- case '0':
- return 'status-waiting';
- case '1':
- return 'status-completed';
- }
- };
- const formatDate = (dateStr) => {
- if (!dateStr) return '';
- const date = new Date(dateStr);
- const year = date.getFullYear();
- const month = String(date.getMonth() + 1).padStart(2, '0');
- const day = String(date.getDate()).padStart(2, '0');
- return `${year}.${month}.${day}`;
- };
- // 팝업 관련 함수들
- const closeModal = () => {
- showInquiryModal.value = false;
- resetForm();
- };
- const resetForm = () => {
- inquiryForm.value = {
- category: '',
- title: '',
- content: '',
- attachments: []
- };
- };
- const submitInquiry = async () => {
- // 폼 유효성 검사
- if (!inquiryForm.value.category) {
- $toast.error('문의 유형을 선택해주세요.');
- return;
- }
- if (!inquiryForm.value.title) {
- $toast.error('제목을 입력해주세요.');
- return;
- }
- if (!inquiryForm.value.content) {
- $toast.error('내용을 입력해주세요.');
- return;
- }
- isSubmitting.value = true;
-
- try {
- const formData = new FormData();
- formData.append('USER_SEQ', useAtStore.auth.seq);
- formData.append('CATEGORY', inquiryForm.value.category);
- formData.append('TITLE', inquiryForm.value.title);
- formData.append('CONTENT', inquiryForm.value.content);
-
- // 첨부파일 처리
- if (inquiryForm.value.attachments && inquiryForm.value.attachments.length > 0) {
- inquiryForm.value.attachments.forEach((file, index) => {
- formData.append(`files[${index}]`, file);
- });
- }
- await useAxios()
- .post("/cs/reg", formData, {
- headers: {
- 'Content-Type': 'multipart/form-data'
- }
- })
- .then((res) => {
- if (res.data.success) {
- $toast.success('문의가 성공적으로 등록되었습니다.');
- closeModal();
- csListGet(); // 목록 새로고침
- } else {
- $toast.error('문의 등록에 실패했습니다.');
- }
- });
- } catch (error) {
- $toast.error('문의 등록 중 오류가 발생했습니다.');
- console.error('Error submitting inquiry:', error);
- } finally {
- isSubmitting.value = false;
- }
- };
- /************************************************************************
- | WATCH
- ************************************************************************/
- onMounted(() => {
- csListGet();
- // 날짜 초기화
- const today = dayjs();
- searchEndDate.value = today.format('YYYY-MM-DD');
- });
- </script>
|