스프링부트+jsp로 배달사이트 만들기-28 관리자페이지(가게 정보 수정, 메뉴 추가)

2021. 12. 17. 01:38스프링부트

가게 정보 수정하기

 

가게 정보 수정 메서드를 추가합니다

@ResponseBody 대신 ResponseEntity를 리턴하게 하고 이전에 만들었던 @IsMyStore어노테이션으로 파라미터의 store의 id가 계정에 등록되지 않은 id일 경우 401 에러가 발생합니다

가게 정보의 일부분을 수정하는 것 이므로 PatchMapping을 사용하였습니다

@IsMyStore
@PatchMapping("/admin/management/storeInfo")
public ResponseEntity<Store> storeInfoUpdate(Store store, MultipartFile file) throws IOException {
	if(!file.isEmpty()){
		String img = uploadFile.fildUpload(file);
		store.setStoreImg(img);
		store.setStoreThumb(img);
	}
	adminService.storeInfoUpdate(store);
	return new ResponseEntity<Store>(store,HttpStatus.OK);
}

 

service와 dao에 추가합니다

 

AdminServiceImp

@Override
public void storeInfoUpdate(Store store) {
	adminDAO.storeInfoUpdate(store);
}

 

AdminDAOImp

@Override
public void storeInfoUpdate(Store store) {
	sql.update("admin.storeInfoUpdate", store);
}

 

AdminMapper

<update id="storeInfoUpdate">
	UPDATE BM_STORE SET
		CATEGORY = #{category }
		,STORE_NAME = #{storeName }
		,STORE_ADDRESS1 = #{storeAddress1 }
		,STORE_ADDRESS2 = #{storeAddress2 }
		,STORE_ADDRESS3 = #{storeAddress3 }
		,STORE_PHONE = #{storePhone }
		,STORE_IMG = #{storeImg }
		,STORE_THUMB = #{storeThumb }
		,OPENING_TIME = #{openingTime }
		,CLOSING_TIME = #{closingTime }
		,MIN_DELEVERY = #{minDelevery }
		,DELEVERY_TIME = #{deleveryTime }
		,DELEVERY_TIP = #{deleveryTip }
		,STORE_DES = #{storeDes }
	WHERE
		ID = #{id }  
</update>

 

가게 주소 변경시에도 api를 사용해서 주소를 변경할 수 있게 해야합니다

include폴더에 기존의 modifyAddress.jsp를 복사하고 ajax부분을 삭제합니다

 

addressSearch.jsp

<%@ page language="java" contentType="text/html; charset=UTF-8"
	pageEncoding="UTF-8"%>
<!-- iOS에서는 position:fixed 버그가 있음, 적용하는 사이트에 맞게 position:absolute 등을 이용하여 top,left값 조정 필요 -->
<div id="layer"
	style="display: none; position: fixed; overflow: hidden; z-index: 2; -webkit-overflow-scrolling: touch;">
	<img src="//t1.daumcdn.net/postcode/resource/images/close.png"
		id="btnCloseLayer"
		style="cursor: pointer; position: absolute; right: -3px; top: -3px; z-index: 1"
		onclick="closeDaumPostcode()" alt="닫기 버튼">
</div>
 
<script src="//t1.daumcdn.net/mapjsapi/bundle/postcode/prod/postcode.v2.js"></script>
 
<script>
	// 우편번호 찾기 화면을 넣을 element
	var element_layer = document.getElementById('layer');
 
	function closeDaumPostcode() {
		// iframe을 넣은 element를 안보이게 한다.
		element_layer.style.display = 'none';
 
	}
 
	function addressSearch() {
		new daum.Postcode(
				{
					oncomplete : function(data) {
						// 검색결과 항목을 클릭했을때 실행할 코드를 작성하는 부분.
 
						// 각 주소의 노출 규칙에 따라 주소를 조합한다.
						// 내려오는 변수가 값이 없는 경우엔 공백('')값을 가지므로, 이를 참고하여 분기 한다.
						var addr = ''; // 주소 변수
						var extraAddr = ''; // 참고항목 변수
 
						//사용자가 선택한 주소 타입에 따라 해당 주소 값을 가져온다.
						if (data.userSelectedType === 'R') { // 사용자가 도로명 주소를 선택했을 경우
							addr = data.roadAddress;
						} else { // 사용자가 지번 주소를 선택했을 경우(J)
							addr = data.jibunAddress;
						}
 
						// 사용자가 선택한 주소가 도로명 타입일때 참고항목을 조합한다.
						if (data.userSelectedType === 'R') {
							// 법정동명이 있을 경우 추가한다. (법정리는 제외)
							// 법정동의 경우 마지막 문자가 "동/로/가"로 끝난다.
							if (data.bname !== ''
									&& /[동|로|가]$/g.test(data.bname)) {
								extraAddr += data.bname;
							}
							// 건물명이 있고, 공동주택일 경우 추가한다.
							if (data.buildingName !== ''
									&& data.apartment === 'Y') {
								extraAddr += (extraAddr !== '' ? ', '
										+ data.buildingName : data.buildingName);
							}
							// 표시할 참고항목이 있을 경우, 괄호까지 추가한 최종 문자열을 만든다.
							if (extraAddr !== '') {
								extraAddr = ' (' + extraAddr + ')';
							}
							// 조합된 참고항목을 해당 필드에 넣는다.
						} else {
						}
 
						// 우편번호와 주소 정보를 해당 필드에 넣는다.
						document.getElementById('address1').value = data.zonecode;
						document.getElementById("address2").value = addr;
 
 
						// 커서를 상세주소 필드로 이동한다.
						document.getElementById("address3").focus();
 
						// iframe을 넣은 element를 안보이게 한다.
						// (autoClose:false 기능을 이용한다면, 아래 코드를 제거해야 화면에서 사라지지 않는다.)
						element_layer.style.display = 'none';
 
					},
					width : '100%',
					height : '100%',
					maxSuggestItems : 5
				}).embed(element_layer);
 
		// iframe을 넣은 element를 보이게 한다.
		element_layer.style.display = 'block';
 
		// iframe을 넣은 element의 위치를 화면의 가운데로 이동시킨다.
		initLayerPosition();
	}
 
	// 브라우저의 크기 변경에 따라 레이어를 가운데로 이동시키고자 하실때에는
	// resize이벤트나, orientationchange이벤트를 이용하여 값이 변경될때마다 아래 함수를 실행 시켜 주시거나,
	// 직접 element_layer의 top,left값을 수정해 주시면 됩니다.
	function initLayerPosition() {
		var width = 300; //우편번호서비스가 들어갈 element의 width
		var height = 400; //우편번호서비스가 들어갈 element의 height
		var borderWidth = 5; //샘플에서 사용하는 border의 두께
 
		// 위에서 선언한 값들을 실제 element에 넣는다.
		element_layer.style.width = width + 'px';
		element_layer.style.height = height + 'px';
		element_layer.style.border = borderWidth + 'px solid';
		// 실행되는 순간의 화면 너비와 높이 값을 가져와서 중앙에 뜰 수 있도록 위치를 계다.
		element_layer.style.left = (((window.innerWidth || document.documentElement.clientWidth) - width) / 2 - borderWidth)
				+ 'px';
		element_layer.style.top = (((window.innerHeight || document.documentElement.clientHeight) - height) / 2 - borderWidth)
				+ 'px';
	}
</script>

 

 

카테고리 수정 시 새로고침 전에 화면에서 바뀌지 않는걸 확인했습니다

adminDetail.js에 다음 부분을 추가합니다

 

 


메뉴 추가하기

 

메뉴 추가 메서드를 추가합니다

@IsMyStore
@PostMapping("/admin/management/menu")
public ResponseEntity<Food> addMenu(Food food, String[] foodOption, Integer[] foodOptionPrice, MultipartFile file) throws IOException {
	if(file.isEmpty()) {
		String img = File.separator + "img" + File.separator + "none.gif";
		food.setFoodImg(img);
		food.setFoodThumb(img);
	} else {
		String img = uploadFile.fildUpload(file);
		food.setFoodImg(img);
		food.setFoodThumb(img);
	}
	
	adminService.addMenu(food, foodOption, foodOptionPrice);
	return new ResponseEntity<Food>(food,HttpStatus.OK);
}

메뉴 정보를 담은 Food클래스와 옵션 추가하기 버튼을 눌러 생성한 박스 수 만큼 메뉴 옵션정보를 배열로 받습니다

 

service와 dao에 추가합니다

@Transactional
@Override
public void addMenu(Food food, String[] foodOption, Integer[] foodOptionPrice) {
	long foodId = adminDAO.addMenu(food);
	
	if(foodOption != null) {
		List<Map<String, Object>> optionList = new ArrayList<>();
		
		for(int i=0;i<foodOption.length;i++) {
			Map<String, Object> optionMap = new HashMap<>();
			optionMap.put("optionName", foodOption[i]);
			optionMap.put("optionPrice", foodOptionPrice[i]);
			optionMap.put("foodId", foodId);
			optionList.add(optionMap);
		}
		
		adminDAO.addMenuOption(optionList);
	}
}

PL/SQL로 메뉴와 옵션을 한번에 등록하고 싶었지만 insert 후 음식의 id를 가져오는데 실패해 메뉴insert와 옵션insert를 따로 실행하였습니다

 

먼저 addMenu로 음식을 등록을 한 후 mybatis의 selectKey를 이용해 등록한 foodId를 리턴받습니다

이 foodId로 다시 옵션들을 등록하는데 이 과정에서 에러가 날 경우 전부 rollback하기 위해 @Transactional을 붙였습니다

 

AdminDAOImp

@Override
public long addMenu(Food food) {
	sql.insert("admin.addMenu", food);
	return food.getId();
}
 
 
@Override
public void addMenuOption(List<Map<String, Object>> optionList) {
	sql.insert("admin.addMenuOption", optionList);
}

 

adminMapper

<insert id="addMenu">
	<selectKey keyProperty="id" resultType="long" order="BEFORE" >
    SELECT FOOD_ID_SEQ.NEXTVAL FROM DUAL
   </selectKey>
   INSERT INTO BM_FOOD (
		   ID
		   ,STORE_ID
		   ,FOOD_NAME
		   ,FOOD_PRICE
		   ,FOOD_DEC
		   ,FOOD_IMG
		   ,FOOD_THUMB
	   ) VALUES (
		   #{id }
		   ,#{storeId }
		   ,#{foodName }
		   ,#{foodPrice }
		   ,#{foodDec }
		   ,#{foodImg }
		   ,#{foodThumb }
	   )
</insert>

SELECT FOOD_ID_SEQ.NEXTVAL FROM DUAL을 실행해 foodId를 생성하고 이 foodId를 keyProperty에 설정한 id라는 변수로 사용합니다 selectKey의 order를 before로 하게 되면 insert가 실행되기 전에 실행 됩니다

insert문을 실행하고 난 뒤 AdminDAOImp의 addMenu메서드의 파라미터로 전달한 Food클래스 id에 세팅됩니다

리턴받은 id로 addMenuOption을 실행합니다

 

<insert id="addMenuOption">
	INSERT INTO BM_FOOD_OPTION 
	<foreach collection="list" item="item"  separator="UNION ALL" >
			SELECT	GET_OPTION_SEQ()
					,#{item.foodId }
					,#{item.optionName }
					,#{item.optionPrice } 
			FROM	DUAL
	</foreach>
</insert>

foreach문 안에 GET_OPTION_SEQ()라는 함수를 사용했는데 OPTION_ID_SEQ.NEXTVAL로 실행했더니 에러가 나서 이 시퀀스를 함수로 감싸서 사용했습니다

 

DB에서 아래를 실행해 함수를 생성합니다

CREATE OR REPLACE FUNCTION GET_OPTION_SEQ
    RETURN NUMBER
IS
    SEQ NUMBER;
BEGIN
  SELECT OPTION_ID_SEQ.NEXTVAL INTO SEQ FROM DUAL;
RETURN SEQ;
END;
  • GET_OPTION_SEQ : 함수 이름
  • RETURN NUMBER : BEGIN 아래 코드를 실행 한 결과의 타입
  • SEQ NUMBER; : BEGIN 아래 코드를 실행 한 결과를 대입 할 변수
  • SELECT OPTION_ID_SEQ.NEXTVAL INTO SEQ FROM DUAL;  OPTION_ID_SEQ.NEXTVAL을 SELECT 후 SEQ에 대입
  • RETURN SEQ; 이 함수를 실행했을 때 SEQ를 반환

 

실제 실행 SQL

INSERT INTO BM_FOOD_OPTION 
	SELECT	GET_OPTION_SEQ(), 1, '옵션이름1', 500 
    UNION ALL
    SELECT	GET_OPTION_SEQ(), 2, '옵션이름2', 600
    UNION ALL
    SELECT	GET_OPTION_SEQ(), 1, '옵션이름3', 700 ...