kr.order.action => UserOrderFormAction.java
package kr.order.action;
import java.util.List;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import kr.cart.dao.CartDAO;
import kr.cart.vo.CartVO;
import kr.controller.Action;
import kr.item.dao.ItemDAO;
import kr.item.vo.ItemVO;
public class UserOrderFormAction implements Action{
@Override
public String execute(HttpServletRequest request, HttpServletResponse response) throws Exception {
HttpSession session = request.getSession();
Integer user_num =
(Integer)session.getAttribute("user_num");
if(user_num==null) {//로그인이 되지 않은 경우
return "redirect:/member/loginForm.do";
}
//로그인 된 경우
//POST방식의 접근만 허용
if(request.getMethod().toUpperCase().equals("GET")) {
return "/WEB-INF/views/common/notice.jsp";
}
CartDAO dao = CartDAO.getInstance();
int all_total = dao.getTotalByMem_num(user_num);
if(all_total<=0) {
request.setAttribute("notice_msg",
"정상적인 주문이 아니거나 상품의 수량이 부족합니다.");
request.setAttribute("notice_url",
request.getContextPath()+"/item/itemList.do");
return "/WEB-INF/views/common/alert_view.jsp";
}
//장바구니에 담겨있는 상품 정보 호출
List<CartVO> cartList = dao.getListCart(user_num);
ItemDAO itemDAO = ItemDAO.getInstance();
for(CartVO cart : cartList) {
ItemVO item = itemDAO.getItem(cart.getItem_num());
if(item.getStatus()==1) {
//상품미표시
request.setAttribute("notice_msg","["+item.getName()+"]상품 판매 중지" );
request.setAttribute("notice_url", request.getContextPath()+"/cart/list.do");
return "/WEB-INF/views/common/alert_view.jsp";
}
if(item.getQuantity()< cart.getOrder_quantity()) {
//상품 재고 수량 부족
request.setAttribute("notice_msg", "["+item.getName()+"]재고수량 부족으로 주문불가");
request.setAttribute("notice_url", request.getContextPath()+"/cart/list.do");
return "/WEB-INF/views/common/alert_view.jsp";
}
}
request.setAttribute("list", cartList);
request.setAttribute("all_total", all_total);
return "/WEB-INF/views/order/user_orderForm.jsp";
}
}
views/order=>user_orderForm.jsp
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %>
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>상품구매</title>
<link rel="stylesheet"
href="${pageContext.request.contextPath}/css/style.css" type="text/css">
<script type="text/javascript"
src="${pageContext.request.contextPath}/js/jquery-3.7.1.min.js"></script>
<script type="text/javascript">
window.onload=function(){
//상품 구매 유효성 체크
const myForm = document.getElementById('order_form');
myForm.onsubmit=function(){
const items = document.querySelectorAll('input[type="text"]');
for(let i=0; i<items.length; i++){
if(items[i].value.trim()==''){
const label = document.querySelector('label[for="'+items[i].id+'"]');
alert(label.textContent+' 항목은 필수 입력');
items[i].value = '';
items[i].focus();
return false;
}
if(items[i].id == 'zipcode' && !/^[0-9]{5}$/.test(items[i].value)){
alert('우편번호를 입력하세요(숫자5자리)')
item[i].value='';
item[i].focus();
return false;
}
}//end of for
//결제수단 유효성체크
const radio = document.querySelectorAll('input[type="radio"]:checked');
if(radio.length<1){
alert('결제수단을 선택하세요');
return false;
}
};
};
</script>
</head>
<body>
<div class="page-main">
<jsp:include page="/WEB-INF/views/common/header.jsp" />
<div class="content-main">
<h2>상품구매</h2>
<table>
<tr>
<th>상품명</th>
<th>수량</th>
<th>상품가격</th>
<th>합계</th>
</tr>
<c:forEach var="cart" items="${list}">
<tr>
<td><a
href="${pageContext.request.contextPath}/item/detail.do?item_num=${cart.item_num}">
<img
src="${pageContext.request.contextPath}/upload/${cart.itemVO.photo1}"
width="80"> ${cart.itemVO.name}
</a></td>
<td class="align-center"><fmt:formatNumber
value="${cart.order_quantity}" /></td>
<td class="align-center"><fmt:formatNumber
value="${cart.itemVO.price}" />원</td>
<td class="align-center"><fmt:formatNumber
value="${cart.sub_total}" />원 <br> <input type="button"
value="삭제" class="cart-del" data-cartnum="${cart.cart_num}">
</td>
</tr>
</c:forEach>
<tr>
<td colspan="3" class="align-right"><b>총구매금액</b></td>
<td class="align-center"><fmt:formatNumber
value="${all_total}" />원</td>
</tr>
</table>
<form action="order.do" method="post" id="order_form">
<ul>
<li>
<label for="receive_name">받는 사람</label>
<input type="text" name="receive_name" id="receive_name" maxlength="10">
</li>
<li>
<label for="zipcode">우편번호</label>
<input type="text" name="receive_post" id="zipcode" maxlength="5">
<input type="button" onclick="execDaumPostcode()" value="우편번호 찾기">
</li>
<li>
<label for="address1">주소</label>
<input
type="text" name="receive_address1" id="address1" maxlength="30">
</li>
<li>
<label for="address2">상세주소</label>
<input type="text" name="receive_address2" id="address2" maxlength="30">
</li>
<li>
<label for="receive_phone">전화번호</label>
<input type="text" name="receive_phone" id="receive_phone" maxlength="15"></li>
<li>
<label>결제수단</label>
<input type="radio" name="payment" id="payment1" value="1">통장입금
<input type="radio" name="payment" id="payment2" value="2">카드결제
</li>
<li>
<label for="notice">남기실 말씀</label>
<textarea rows="5" cols="30" name="notice" id="notice" maxlength="1300"></textarea>
</li>
</ul>
<div class="align-center">
<input type="submit" value="주문"> <input type="button"
value="홈으로"
onclick="location.href='${pageContext.request.contextPath}/main/main.do'">
</div>
</form>
<!-- 다음우편번호 API시작 -->
<!-- iOS에서는 position:fixed 버그가 있음, 적용하는 사이트에 맞게 position:absolute 등을 이용하여 top,left값 조정 필요 -->
<div id="layer"
style="display: none; position: fixed; overflow: hidden; z-index: 1; -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 execDaumPostcode() {
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 + ')';
}
//(주의)address1에 참고항목이 보여지도록 수정
// 조합된 참고항목을 해당 필드에 넣는다.
//(수정) document.getElementById("address2").value = extraAddr;
}
//(수정) else {
//(수정) document.getElementById("address2").value = '';
//(수정) }
// 우편번호와 주소 정보를 해당 필드에 넣는다.
document.getElementById('zipcode').value = data.zonecode;
//(수정) + extraAddr를 추가해서 address1에 참고항목이 보여지도록 수정
document.getElementById("address1").value = addr + extraAddr;
// 커서를 상세주소 필드로 이동한다.
document.getElementById("address2").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>
</div>
<!-- 다음우편번호 API 끝 -->
</div>
</body>
</html>
OrderDAO 추가
//주문 등록
public void insertOrder(OrderVO order, List<OrderDetailVO> orderDetailList)throws Exception{
Connection conn = null;
PreparedStatement pstmt = null;
PreparedStatement pstmt2 = null;
PreparedStatement pstmt3 = null;
PreparedStatement pstmt4 = null;
PreparedStatement pstmt5 = null;
ResultSet rs = null;
String sql =null;
int order_num = 0;
try {
//커넥션풀로부터 커넥션 할당
conn = DBUtil.getConnection();
//오토커밋 해제
conn.setAutoCommit(false);
//order_num 구하기 (테이블이 2개여서 시퀀스가 겹칠 수 있기 때문에 따로 만들어준다.)
sql="SELECT zorder_seq.nextval FROM dual";
//PreparedStatment 객체 생성
pstmt = conn.prepareStatement(sql);
rs = pstmt.executeQuery();
if(rs.next()) {
order_num = rs.getInt(1);
}
//주문정보 처리
sql="INSERT INTO zorder (order_num,order_total,payment,receive_name,receive_post,"
+ "receive_address1,receive_address2,receive_phone,notice,mem_num) "
+ "VALUES (?,?,?,?,?,?,?,?,?,?)";
pstmt2 = conn.prepareStatement(sql);
pstmt2.setInt(1, order_num);
pstmt2.setInt(2, order.getOrder_total());
pstmt2.setInt(3, order.getPayment());
pstmt2.setString(4, order.getReceive_name());
pstmt2.setString(5, order.getReceive_post());
pstmt2.setString(6, order.getReceive_address1());
pstmt2.setString(7, order.getReceive_address2());
pstmt2.setString(8, order.getReceive_phone());
pstmt2.setString(9, order.getNotice());
pstmt2.setInt(10, order.getMem_num());
pstmt2.executeUpdate();
//주문상세정보 처리
sql = "INSERT INTO zoder_detail(detail_num,item_num,"
+ "item_name,item_price,item_total,order_quantity,"
+ "order_num) VALUES (zorder_detail_seq.nextval,"
+ "?,?,?,?,?,?)";
pstmt3 = conn.prepareStatement(sql);
for(int i=0; i<orderDetailList.size(); i++) {
OrderDetailVO orderDetail = orderDetailList.get(i);
pstmt3.setInt(1, orderDetail.getItem_num());
pstmt3.setString(2, orderDetail.getItem_name());
pstmt3.setInt(3, orderDetail.getItem_price());
pstmt3.setInt(4, orderDetail.getItem_total());
pstmt3.setInt(5, orderDetail.getOrder_quantity());
pstmt3.setInt(6, order_num);
pstmt3.addBatch();//쿼리를 메모리에 올림
//계속 추가하면 outOfMemory 발생, 1000개 단위로 executeBatch() : 메모리에있는걸 오라클로 보낸다(?)
if(i % 1000 ==0) {
pstmt3.executeBatch();
}
}
pstmt3.executeBatch(); //쿼리를 전송
//상품의 재고수 차감
sql = "UPDATE zitem SET quantity=quantity-? WHERE item_num=?";
pstmt4 = conn.prepareStatement(sql);
for(int i=0; i<orderDetailList.size(); i++) {
OrderDetailVO orderDetail = orderDetailList.get(i);
pstmt4.setInt(1, orderDetail.getOrder_quantity());
pstmt4.setInt(2, orderDetail.getItem_num());
pstmt4.addBatch();//쿼리를 메모리에 올림
if(i % 1000 ==0) {
pstmt4.executeBatch();
}
}
pstmt4.executeBatch();//쿼리 전송
//장바구니에서 주문상품 삭제
sql = "DELETE FROM zcart WHERE mem_num=?";
pstmt5 = conn.prepareStatement(sql);
pstmt5.setInt(1, order.getMem_num());
pstmt5.executeUpdate();
//모든 SQL문이 정상 수행
conn.commit();
}catch(Exception e) {
//SQL문이 하나라도 실패하면 롤백
conn.rollback();
throw new Exception(e);
}finally {
DBUtil.executeClose(null, pstmt5, null);
DBUtil.executeClose(null, pstmt4, null);
DBUtil.executeClose(null, pstmt3, null);
DBUtil.executeClose(null, pstmt2, null);
DBUtil.executeClose(rs, pstmt, conn);
}
}
UserOrderAction
package kr.order.action;
import java.util.ArrayList;
import java.util.List;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import kr.cart.dao.CartDAO;
import kr.cart.vo.CartVO;
import kr.controller.Action;
import kr.item.dao.ItemDAO;
import kr.item.vo.ItemVO;
import kr.order.dao.OrderDAO;
import kr.order.vo.OrderDetailVO;
import kr.order.vo.OrderVO;
public class UserOrderAction implements Action {
@Override
public String execute(HttpServletRequest request, HttpServletResponse response) throws Exception {
HttpSession session = request.getSession();
Integer user_num =
(Integer)session.getAttribute("user_num");
if(user_num==null) {//로그인이 되지 않은 경우
return "redirect:/member/loginForm.do";
}
//로그인 된 경우
//POST방식의 접근만 허용
if(request.getMethod().toUpperCase().equals("GET")) {
return "/WEB-INF/views/common/notice.jsp";
}
//전송된 데이터 인코딩 타입 지정
request.setCharacterEncoding("utf-8");
CartDAO dao = CartDAO.getInstance();
int all_total = dao.getTotalByMem_num(user_num);
if(all_total<=0) {
request.setAttribute("notice_msg",
"정상적인 주문이 아니거나 상품의 수량이 부족합니다.");
request.setAttribute("notice_url",
request.getContextPath()+"/item/itemList.do");
return "/WEB-INF/views/common/alert_view.jsp";
}
//장바구니에 담겨있는 상품 정보 호출 (사려는사람의 제품 총구매액을 보여줌)
List<CartVO> cartList = dao.getListCart(user_num);
//개별상품 정보 담기
List<OrderDetailVO> orderDetailList = new ArrayList<OrderDetailVO>();
ItemDAO itemDAO = ItemDAO.getInstance();
for(CartVO cart : cartList) {
ItemVO item = itemDAO.getItem(cart.getItem_num());
if(item.getStatus()==1) {
//상품미표시
request.setAttribute("notice_msg","["+item.getName()+"]상품 판매 중지" );
request.setAttribute("notice_url", request.getContextPath()+"/cart/list.do");
return "/WEB-INF/views/common/alert_view.jsp";
}
if(item.getQuantity()< cart.getOrder_quantity()) {
//상품 재고 수량 부족
request.setAttribute("notice_msg", "["+item.getName()+"]재고수량 부족으로 주문불가");
request.setAttribute("notice_url", request.getContextPath()+"/cart/list.do");
return "/WEB-INF/views/common/alert_view.jsp";
}
//자바빈에 개별상품 정보 저장
OrderDetailVO orderDetail = new OrderDetailVO();
orderDetail.setItem_num(cart.getItem_num());
orderDetail.setItem_name(cart.getItemVO().getName());
orderDetail.setItem_price(cart.getItemVO().getPrice());
orderDetail.setOrder_quantity(cart.getOrder_quantity());
orderDetail.setItem_total(cart.getSub_total());
orderDetailList.add(orderDetail);
}
//구매정보 담기
OrderVO order = new OrderVO();
order.setOrder_total(all_total);
order.setPayment(Integer.parseInt(request.getParameter("payment")));
order.setReceive_name(request.getParameter("receive_name"));
order.setReceive_post(request.getParameter("receive_post"));
order.setReceive_address1(request.getParameter("receive_address1"));
order.setReceive_address2(request.getParameter("receive_address2"));
order.setReceive_phone(request.getParameter("receive_phone"));
order.setNotice(request.getParameter("notice"));
order.setMem_num(user_num);
OrderDAO orderDAO = OrderDAO.getInstance();
orderDAO.insertOrder(order, orderDetailList);
//Refresh 정보를 응답 헤더에 추가
String url = request.getContextPath()+"/main/main.do";
response.addHeader("Refresh", "2;url="+url);
request.setAttribute("result_title", "상품 주문 완료");
request.setAttribute("result_msg", "주문이 완료되었습니다.");
request.setAttribute("result_url", url);
return "/WEB-INF/views/common/result_view.jsp";
}
}