From 35af067dd9d42301e60eb383ca080d6f95ed62bd Mon Sep 17 00:00:00 2001 From: psyeon1120 Date: Sun, 26 Nov 2023 21:40:58 +0900 Subject: [PATCH] =?UTF-8?q?[PDW-94]=20feat:=20=EC=B0=A8=EB=9F=89=20?= =?UTF-8?q?=EC=B6=94=EA=B0=80/=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/App.js | 3 + src/pages/admin/car/CarManageAdd.js | 438 ++++++++++++++++++++++++++++ 2 files changed, 441 insertions(+) create mode 100644 src/pages/admin/car/CarManageAdd.js diff --git a/src/App.js b/src/App.js index 4938896..95274be 100644 --- a/src/App.js +++ b/src/App.js @@ -22,6 +22,7 @@ import CarBooking from "./pages/basic/booking/car/CarBooking"; import CarBookingCheck from "./pages/basic/booking/car/CarBookingCheck"; import CarManage from "./pages/admin/car/CarManage"; import CarBookingManage from "./pages/admin/carBookings/CarBookingManage"; +import CarManageAdd from "./pages/admin/car/CarManageAdd"; function App() { @@ -60,6 +61,8 @@ function App() { } /> } /> } /> + } /> + } /> } /> } /> } /> diff --git a/src/pages/admin/car/CarManageAdd.js b/src/pages/admin/car/CarManageAdd.js new file mode 100644 index 0000000..9c4db63 --- /dev/null +++ b/src/pages/admin/car/CarManageAdd.js @@ -0,0 +1,438 @@ +import React, {useEffect, useRef, useState} from "react"; +import styled from "styled-components"; +import {RightContainer, TitleText, WhiteContainer} from "components/rightContainer/RightContainer"; +import {ManageAddButton, ManageAddButtonImage} from "components/searchBar/ManageSearchBar"; +import {getToken} from "utils/IsLoginUtil"; +import {basicError} from "utils/ErrorHandlerUtil"; +import {AdminCarsAxios, CarsAxios, ImageUrlAxios, UsersAxios} from "api/AxiosApi"; +import SearchButtonImage from "assets/images/SearchPlus.svg" +import EmptyImg from "assets/images/EmptyImg.svg" +import {ExitBtn} from "components/modal/BigModal"; +import axios from "axios"; +import {useParams} from "react-router-dom"; + +const MarginWhiteContainer = styled(WhiteContainer)` + padding: 40px; + box-sizing: border-box; + display: flex; + flex-direction: column; + justify-content: space-between; +` + +const ColumnContainer = styled.div` + height: 40px; + display: flex; + justify-content: flex-start; + align-items: center; + margin-bottom: 40px; +` + +const ShortColumnContainer = styled(ColumnContainer)` + width: 50%; +` + +const ImgColumnContainer = styled(ShortColumnContainer)` + margin-bottom: 0; + height: 150px; + width: 300px; +` + +const TitleLabel = styled.label` + color: #8741CB; + font-size: 20px; + width: 100px; + min-width: 100px; + text-align: left; +` + +const InfoInput = styled.input.attrs({type: 'text'})` + flex: 1; + height: 20px; + border-radius: 8px; + border: 2px solid #E6E6E6; + font-size: 20px; + padding: 10px; +` + +const StaffInputContainer = styled.div` + width: 50%; + height: 100%; + display: block; + z-index: 11; + background: white; +` + +const StaffInfoInput = styled(InfoInput)` + width: 100%; +` + +const StaffSelectUl = styled.div` + width: 100%; + max-height: 120px; + overflow: scroll; + border-radius: 0 0 8px 8px; + border: 2px solid #E6E6E6; + padding: 10px; + z-index: 3; + background: white; + margin-top: -5px; +` + +const StaffNameLabel = styled.p` + color: #4C4C4C; + font-size: 20px; + text-align: left; + z-index: 4; + cursor: pointer; + margin: 10px 0; + width: fit-content; +` + +const DescriptionContainer = styled(ColumnContainer)` + height: 60px; +` + +const DescriptionInput = styled.textarea` + flex: 1; + height: 100%; + border-radius: 8px; + border: 2px solid #E6E6E6; + font-size: 20px; + padding: 10px; +` + +const AddButtonContainer = styled.div` + width: 100%; + height: fit-content; + display: flex; + justify-content: flex-end; +` + +const ImageAddContainer = styled.div` + width: 100px; + height: 40px; + margin-left: 10px; +` + +const ImageAddButton = styled.button` + min-width: 100px; + height: 100%; + border-radius: 8px; + border: none; + filter: drop-shadow(0px 2px 2px rgba(0, 0, 0, 0.25)); + font-size: 16px; +` + +const ImageInfoContainer = styled.div` + width: 320px; + min-width: 320px; + height: 180px; + border-radius: 8px; + border: 2px solid #E6E6E6; + overflow: clip; + position: relative; +` + +const AbExitBtn = styled(ExitBtn)` + position: absolute; + top: 0; + right: 0; + color: #8741CB; +` + +const PreviewImage = styled.img` + width: 320px; + height: 180px; + object-fit: contain; +` + +function CarManageAdd(props) { + let {carId} = useParams(); + + const initialValue = { + name: null, + manufacturer: null, + place: null, + description: null, + }; + const [inputValues, setInputValues] = useState(initialValue); + const {name, manufacturer, place, description} = inputValues; //비구조화 할당 + const [staff, setStaff] = useState({ + userId: null, + name: "" + }); + const [staffList, setStaffList] = useState([]); + const [imageSrc, setImgSrc] = useState(""); + const [imageFile, setImageFile] = useState(null); + const [imageUrl, setImgUrl] = useState(null); + const [isUpload, setIsUpload] = useState(false); + const [isFocusStaffInput, setIsFocusStaffInput] = useState(false); + const [carInfo, setCarInfo] = useState(null) + const imageInput = useRef(null); + + const onChangeInput = event => { + const {value, name: inputName} = event.target; + setInputValues({...inputValues, [inputName]: value}); + } + + const changeStaff = (e) => { + setStaff({...staff, name: e.target.value}); + }; + + const deleteStaff = (e) => { + if (e.key === 'Backspace') + setStaff({ + ...staff, + name: "", + userId: null + }); + }; + + const changeImageFile = () => { + imageInput.current.click(); + }; + + const handleChange = (e) => { + setImageFile(e.target.files[0]) + const reader = new FileReader(); + reader.readAsDataURL(e.target.files[0]); + reader.onloadend = () => { + setImgSrc(reader.result); + }; + }; + + const deleteImageFile = () => { + setImageFile(null); + setIsUpload(false); + imageInput.current.value = ""; + setImgSrc("") + }; + + // 이미지 람다 호출 + const getImageUrl = () => { + // todo: 검사 다 하기 + if (staff.userId === null) { + alert("책임자를 선택해주세요."); + return; + } + if (imageFile !== null) { // 이미지 파일 선택했을 때 + ImageUrlAxios.get(`?ext=${imageFile.type.split("/", 2)[1]}&dir=car`) + .then((Response) => { + setImgUrl(Response.data); + }) + .catch((error) => { + console.log(error) + }); + } else { + if (carInfo !== null) { // 수정일 때 + editCar(); + } else { + addCar(); + } + } + } + + // s3에 이미지 업로드 + const uploadImage = () => { + axios.put(imageUrl.presignedUrl, imageFile, { + headers: { + 'Content-Type': 'multipart/form-data' // Content-Type 헤더 설정 + } + }) + .then(function (rep) { + setIsUpload(true); + }) + .catch(function (err) { + console.log(err) + alert("이미지 등록에 실패했습니다. 다시 시도해주세요."); + }); + } + + useEffect(() => { + if (imageUrl !== null && imageFile !== null) + uploadImage(); + }, [imageUrl]); + + useEffect(() => { + if (isUpload) { + if (carInfo !== null) editCar(); + else addCar(); + } + }, [isUpload]); + + const addCar = () => { + AdminCarsAxios.post(``, { + responsibility: staff.userId, + description: description, + manufacturer: manufacturer, + location: place, + name: name, + imgKey: imageFile === null ? null : `car/${imageUrl.imageKey}`, + }, + { + headers: { + Authorization: getToken() + }, + }) + .then((Response) => { + alert("차량 등록이 완료되었습니다."); + window.location.href = `/admin/cars` + }) + .catch((error) => { + basicError(error) + }); + }; + + const editCar = () => { + AdminCarsAxios.patch(`/${carId}`, { + responsibility: staff.userId, + description: description, + manufacturer: manufacturer, + location: place, + name: name, + imgKey: imageFile === null ? imageUrl : `car/${imageUrl.imageKey}`, + }, + { + headers: { + Authorization: getToken() + }, + }) + .then((Response) => { + alert("차량 수정이 완료되었습니다."); + window.location.href = `/admin/cars/${carId}` + }) + .catch((error) => { + basicError(error) + }); + } + + // 차량 기존 정보 호출 + const getCarInfo = () => { + CarsAxios.get(`/${carId}`, { + headers: { + Authorization: getToken() + } + }) + .then((Response) => { + setCarInfo(Response.data.data); + setStaff({ + ...staff, + name: Response.data.data.responsibilityName, + userId: Response.data.data.responsibilityId + }); + setInputValues({ + ...inputValues, + name: Response.data.data.name, + manufacturer: Response.data.data.manufacturer, + place: Response.data.data.location, + description: Response.data.data.description + }); + setImgUrl(Response.data.data.imgUrl); + }) + .catch((Error) => { + basicError(Error) + window.alert("차량 정보를 불러올 수 없습니댜.") + window.history.back() + }) + } + useEffect(() => { + if (carId !== undefined) { + getCarInfo(); + } + }, []) + + // 책임자 리스트 검색 + const getStaffList = () => { + UsersAxios.get(`managers?name=${staff.name}`, { + headers: { + Authorization: getToken() + } + }) + .then((Response) => { + setStaffList(Response.data.data.responsibilityList); + }) + .catch((error) => { + basicError(error) + }); + }; + useEffect(() => { + getStaffList(); + }, [staff.name]); + + return ( + + {carInfo === null ? "차량 추가" : "차량 수정"} + + + + 차량명 + + + + + 제조사 + + + + + 현재위치 + + + + + 책임자 + + setIsFocusStaffInput(true)} + onBlur={() => setIsFocusStaffInput(false)}/> + {isFocusStaffInput && + + {staffList.map((staff, index) => + setStaff(staff)}>{staff.name} + )} + + } + + + + + 설명 + + + + + 첨부사진 + + + {imageFile !== null && ×} + + + 파일선택 + + + + + + + + {carInfo === null ? "차량 추가" : "차량 수정"} + + + + + + ); +} + +export default CarManageAdd;