Merge branch 'main' into login

This commit is contained in:
yoon 2025-03-13 21:18:18 +09:00
commit 946e3441e3
6 changed files with 98 additions and 43 deletions

View File

@ -16,6 +16,7 @@
height: 8px !important; height: 8px !important;
border-radius: 2px !important; border-radius: 2px !important;
font-size: 0px !important; font-size: 0px !important;
margin-left: -0.5% !important;
} }
/* 오후 반차 그래프 (오른쪽 절반) */ /* 오후 반차 그래프 (오른쪽 절반) */
.fc-daygrid-event.half-day-pm { .fc-daygrid-event.half-day-pm {
@ -24,6 +25,7 @@
margin-left: auto !important; margin-left: auto !important;
border-radius: 2px !important; border-radius: 2px !important;
font-size: 0px !important; font-size: 0px !important;
margin-right: -0.5% !important;
} }
/* 연차 그래프 (풀) */ /* 연차 그래프 (풀) */
.fc-daygrid-event.full-day { .fc-daygrid-event.full-day {
@ -69,7 +71,7 @@ background-color: rgba(0, 0, 0, 0.05); /* 연한 배경 효과 */
.fc-day-sat-sun { .fc-day-sat-sun {
cursor: not-allowed !important; cursor: not-allowed !important;
} }
/* 과거 날짜 (오늘 이전) */ /* 과거 날짜 (오늘 -7일일) */
.fc-daygrid-day.past { .fc-daygrid-day.past {
cursor: not-allowed !important; cursor: not-allowed !important;
} }

View File

@ -124,12 +124,21 @@ const common = {
* @param { String } profileImg * @param { String } profileImg
* @returns * @returns
*/ */
getProfileImage(profileImg) { getProfileImage(profileImg, isAnonymous = false) {
let profileImgUrl = '/img/icons/icon.png'; // 기본 프로필 이미지 경로 const defaultProfileImg = '/img/icons/icon.png'; // 기본 프로필 이미지 경로
const anonymousImg = '/img/avatars/default-Profile.jpg'; // 익명 이미지
let profileImgUrl = isAnonymous ? anonymousImg : defaultProfileImg;
const UserProfile = `${import.meta.env.VITE_SERVER}upload/img/profile/${profileImg}`; const UserProfile = `${import.meta.env.VITE_SERVER}upload/img/profile/${profileImg}`;
return !profileImg || profileImg === '' ? profileImgUrl : UserProfile; return !profileImg || profileImg === '' ? profileImgUrl : UserProfile;
}, },
setDefaultImage(event, deafultImg = '/img/icons/icon.png') {
return (event.target.src = deafultImg);
},
showImage(event) {
return (event.target.style.visibility = 'visible');
},
}; };
export default { export default {

View File

@ -2,7 +2,13 @@
<div class="d-flex align-items-center flex-wrap"> <div class="d-flex align-items-center flex-wrap">
<div class="d-flex align-items-center"> <div class="d-flex align-items-center">
<div class="avatar me-2"> <div class="avatar me-2">
<img :src="getProfileImage(profileImg)" alt="Avatar" class="rounded-circle" /> <img
:src="getProfileImage(profileImg)"
alt="user"
class="rounded-circle"
@error="setDefaultImage($event)"
@load="showImage($event)"
/>
</div> </div>
<div class="me-2"> <div class="me-2">
@ -120,6 +126,14 @@
// //
const getProfileImage = profileImg => { const getProfileImage = profileImg => {
return $common.getProfileImage(profileImg); return $common.getProfileImage(profileImg, true);
};
const setDefaultImage = e => {
return $common.setDefaultImage(e);
};
const showImage = e => {
return $common.showImage(e);
}; };
</script> </script>

View File

@ -5,7 +5,7 @@
for="profilePic" for="profilePic"
class="rounded-circle m-auto ui-bg-cover position-relative cursor-pointer" class="rounded-circle m-auto ui-bg-cover position-relative cursor-pointer"
id="profileLabel" id="profileLabel"
style="width: 100px; height: 100px; background-image: url(public/img/avatars/default-Profile.jpg); background-repeat: no-repeat;" style="width: 100px; height: 100px; background-image: url(img/avatars/default-Profile.jpg); background-repeat: no-repeat"
> >
</label> </label>
@ -53,14 +53,14 @@
<span v-if="passwordcheckError" class="invalid-feedback d-block">{{ passwordcheckError }}</span> <span v-if="passwordcheckError" class="invalid-feedback d-block">{{ passwordcheckError }}</span>
<FormSelect <FormSelect
title="비밀번호 힌트" title="비밀번호 힌트"
name="pwhint" name="pwhint"
:is-essential="true" :is-essential="true"
:is-row="false" :is-row="false"
:is-label="true" :is-label="true"
:is-common="true" :is-common="true"
:data="pwhintList" :data="pwhintList"
@update:data="pwhint = $event" @update:data="pwhint = $event"
/> />
<UserFormInput <UserFormInput
@ -164,7 +164,7 @@
<script setup> <script setup>
import { ref, watch } from 'vue'; import { ref, watch } from 'vue';
import $api from '@api'; import $api from '@api';
import commonApi from '@/common/commonApi' import commonApi from '@/common/commonApi';
import UserFormInput from '@c/input/UserFormInput.vue'; import UserFormInput from '@c/input/UserFormInput.vue';
import FormSelect from '@c/input/FormSelect.vue'; import FormSelect from '@c/input/FormSelect.vue';
import ArrInput from '@c/input/ArrInput.vue'; import ArrInput from '@c/input/ArrInput.vue';
@ -187,13 +187,13 @@
const birth = ref(''); const birth = ref('');
const address = ref(''); const address = ref('');
const detailAddress = ref(''); const detailAddress = ref('');
const postcode = ref(''); // const postcode = ref(''); //
const phone = ref(''); const phone = ref('');
const phoneError = ref(''); const phoneError = ref('');
const color = ref(''); // color const color = ref(''); // color
const colorError = ref(''); const colorError = ref('');
const mbti = ref(''); // MBTI const mbti = ref(''); // MBTI
const pwhint = ref(''); // pwhint const pwhint = ref(''); // pwhint
const profilAlert = ref(false); const profilAlert = ref(false);
const idAlert = ref(false); const idAlert = ref(false);
@ -211,7 +211,6 @@
const toastStore = useToastStore(); const toastStore = useToastStore();
// //
const profileValid = (size, type) => { const profileValid = (size, type) => {
const maxSize = 5 * 1024 * 1024; const maxSize = 5 * 1024 * 1024;
@ -240,7 +239,7 @@
// , // ,
if (!profileValid(file.size, file.type)) { if (!profileValid(file.size, file.type)) {
e.target.value = ''; e.target.value = '';
profileLabel.style.backgroundImage = 'url("public/img/avatars/default-Profile.jpg")'; profileLabel.style.backgroundImage = 'url("img/avatars/default-Profile.jpg")';
return false; return false;
} }
@ -277,16 +276,17 @@
// , mbti, // , mbti,
const { colorList, mbtiList, pwhintList } = commonApi({ const { colorList, mbtiList, pwhintList } = commonApi({
loadColor: true, colorType: 'YON', loadColor: true,
colorType: 'YON',
loadMbti: true, loadMbti: true,
loadPwhint: true, loadPwhint: true,
}); });
// //
const handleAddressUpdate = (addressData) => { const handleAddressUpdate = addressData => {
address.value = addressData.address; address.value = addressData.address;
detailAddress.value = addressData.detailAddress; detailAddress.value = addressData.detailAddress;
postcode.value = addressData.postcode; // postcode.value = addressData.postcode; //
}; };
// //
@ -313,7 +313,7 @@
} }
}; };
const handleColorUpdate = async (newColor) => { const handleColorUpdate = async newColor => {
color.value = newColor; color.value = newColor;
colorError.value = ''; colorError.value = '';
colorErrorAlert.value = false; colorErrorAlert.value = false;
@ -357,9 +357,21 @@
profilAlert.value = false; profilAlert.value = false;
} }
if (profilAlert.value || idAlert.value || idErrorAlert.value || passwordAlert.value || passwordcheckAlert.value || if (
passwordcheckErrorAlert.value || pwhintResAlert.value || nameAlert.value || birthAlert.value || profilAlert.value ||
addressAlert.value || phoneAlert.value || phoneErrorAlert.value || colorErrorAlert.value) { idAlert.value ||
idErrorAlert.value ||
passwordAlert.value ||
passwordcheckAlert.value ||
passwordcheckErrorAlert.value ||
pwhintResAlert.value ||
nameAlert.value ||
birthAlert.value ||
addressAlert.value ||
phoneAlert.value ||
phoneErrorAlert.value ||
colorErrorAlert.value
) {
return; return;
} }
@ -378,7 +390,7 @@
formData.append('memberMbt', mbti.value); formData.append('memberMbt', mbti.value);
formData.append('memberPrf', profile.value); formData.append('memberPrf', profile.value);
const response = await $api.post('/user/join', formData, { isFormData : true }); const response = await $api.post('/user/join', formData, { isFormData: true });
if (response.status === 200) { if (response.status === 200) {
toastStore.onToast('등록신청이 완료되었습니다. 관리자 승인 후 이용가능합니다.', 's'); toastStore.onToast('등록신청이 완료되었습니다. 관리자 승인 후 이용가능합니다.', 's');
@ -386,4 +398,3 @@
} }
}; };
</script> </script>

View File

@ -291,5 +291,7 @@ onMounted(() => {
color: #ff5733; color: #ff5733;
border-radius: 4px; border-radius: 4px;
padding: 2px 6px; padding: 2px 6px;
position: relative;
top: -1px;
} }
</style> </style>

View File

@ -152,23 +152,30 @@ function handleMonthChange(viewInfo) {
loadCalendarData(year, month); loadCalendarData(year, month);
} }
// //
//
function handleDateClick(info) { function handleDateClick(info) {
const clickedDateStr = info.dateStr; const clickedDateStr = info.dateStr;
const clickedDate = info.date; const clickedDate = info.date;
const todayStr = new Date().toISOString().split("T")[0]; const todayStr = new Date().toISOString().split("T")[0];
const todayObj = new Date(todayStr);
const oneWeekAgoObj = new Date(todayObj);
oneWeekAgoObj.setDate(todayObj.getDate() - 8); // 7
// (, ) -7
if ( if (
clickedDate.getDay() === 0 || clickedDate.getDay() === 0 || //
clickedDate.getDay() === 6 || clickedDate.getDay() === 6 || //
holidayDates.value.has(clickedDateStr) || holidayDates.value.has(clickedDateStr) || //
clickedDateStr < todayStr clickedDateStr <= oneWeekAgoObj.toISOString().split("T")[0] // -7
) { ) {
return; return;
} }
const isMyVacation = myVacations.value.some(vac => { const isMyVacation = myVacations.value.some(vac => {
const vacDate = vac.date ? vac.date.substring(0, 10) : ""; const vacDate = vac.date ? vac.date.substring(0, 10) : "";
return vacDate === clickedDateStr && !vac.receiverId; return vacDate === clickedDateStr && !vac.receiverId;
}); });
if (isMyVacation) { if (isMyVacation) {
if (selectedDates.value.get(clickedDateStr) === "delete") { if (selectedDates.value.get(clickedDateStr) === "delete") {
selectedDates.value.delete(clickedDateStr); selectedDates.value.delete(clickedDateStr);
@ -178,45 +185,55 @@ function handleDateClick(info) {
updateCalendarEvents(); updateCalendarEvents();
return; return;
} }
if (selectedDates.value.has(clickedDateStr)) { if (selectedDates.value.has(clickedDateStr)) {
selectedDates.value.delete(clickedDateStr); selectedDates.value.delete(clickedDateStr);
updateCalendarEvents(); updateCalendarEvents();
return; return;
} }
const type = halfDayType.value const type = halfDayType.value
? (halfDayType.value === "AM" ? "700101" : "700102") ? (halfDayType.value === "AM" ? "700101" : "700102")
: "700103"; : "700103";
selectedDates.value.set(clickedDateStr, type); selectedDates.value.set(clickedDateStr, type);
halfDayType.value = null; halfDayType.value = null;
updateCalendarEvents(); updateCalendarEvents();
if (halfDayButtonsRef.value) { if (halfDayButtonsRef.value) {
halfDayButtonsRef.value.resetHalfDay(); halfDayButtonsRef.value.resetHalfDay();
} }
} }
//
function markClickableDates() { function markClickableDates() {
nextTick(() => { nextTick(() => {
const todayStr = new Date().toISOString().split("T")[0]; // YYYY-MM-DD const todayStr = new Date().toISOString().split("T")[0]; // (YYYY-MM-DD)
const todayObj = new Date(todayStr); const todayObj = new Date(todayStr);
const oneWeekAgoObj = new Date(todayObj);
oneWeekAgoObj.setDate(todayObj.getDate() - 8); // 7
document.querySelectorAll(".fc-daygrid-day").forEach((cell) => { document.querySelectorAll(".fc-daygrid-day").forEach((cell) => {
const dateStr = cell.getAttribute("data-date"); const dateStr = cell.getAttribute("data-date");
if (!dateStr) return; // if (!dateStr) return; //
const dateObj = new Date(dateStr); const dateObj = new Date(dateStr);
// (, )
if (dateObj.getDay() === 0 || dateObj.getDay() === 6 || holidayDates.value.has(dateStr)) { // (, ) -7
if (
dateObj.getDay() === 0 || //
dateObj.getDay() === 6 || //
holidayDates.value.has(dateStr) || //
dateObj.getTime() === oneWeekAgoObj.getTime() // -7
) {
cell.classList.remove("clickable"); cell.classList.remove("clickable");
cell.classList.add("fc-day-sat-sun"); cell.classList.add("fc-day-sat-sun");
cell.removeEventListener("click", handleDateClick); //
} }
// ( ) // -6
else if (dateObj < todayObj) {
cell.classList.remove("clickable");
cell.classList.add("past"); //
}
// & ( )
else { else {
cell.classList.add("clickable"); cell.classList.add("clickable");
cell.classList.remove("past", "fc-day-sat-sun"); cell.classList.remove("past", "fc-day-sat-sun");
cell.addEventListener("click", handleDateClick); //
} }
}); });
}); });