스프링 #14 영속 비즈니스 계층의 CRUD 구현 (SB Admin 2 페이지 적용)

수박·2020년 6월 10일
0

Spring

목록 보기
13/18

스프링 #14 영속/ 비즈니스 계층의 CRUD 구현 (SB Admin 2 페이지 적용)

SB Admin 2 (디자인 템플릿) 다운로드

http://samara.computer/531_adaptive/startbootstrap-master/startbootstrap-master/sb-admin-v2.php

템플릿을 다운로드하고, 압축해제한 파일의 내용을 webapp/resources에 복사한다. 그리고 해당 템플릿을 적용하기 위해 다음 과정을 진행한다.

/board/list 페이지가 작동하도록, BoardController.java와, list.jsp를 작성한다.

@Log4j
@Controller
@RequestMapping("/board")
public class BoardController {
	
	@GetMapping("/list")
	public void list() {
		log.info("list......");
	}
}

list.jsp는 동작확인을 위해 간단하게 작성한다.

정상적으로 페이지가 동작한다면, SB Admin2의 pages 폴더의 tables.html내용을 list.jsp로 그대로 복사, 수정하고 실행한다.

이때 기존 list.jsp의 상단 지시자는 지우지 않는다.

<%@ page language="java" contentType="text/html; charset=UTF-8"
    pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html lang="en">

내용을 복사한 후, 페이지를 실행해보면 페이지가 깨지는 것을 확인할 수 있다. 이는 list.jsp에서 링크한 css파일의 경로가 아직 지정되지 않았기 때문에 발생한다. 해당 에러는 F12개발자도구 -> network -> 새로고침시 확인할 수 있다.

css,js 경로 에러

해결법은 jsp파일의 경로를 수정하는 것이다.

 <link href="/resources/css/bootstrap.min.css" rel="stylesheet">
    <link href="/resources/font-awesome/css/font-awesome.css" rel="stylesheet">

    <!-- Page-Level Plugin CSS - Tables -->
    <link href="/resources/css/plugins/dataTables/dataTables.bootstrap.css" rel="stylesheet">

    <!-- SB Admin CSS - Include with every page -->
    <link href="/resources/css/sb-admin.css" rel="stylesheet">

link의 경로의 앞에 현재 css파일이 위치한 resources경로를 추가해주면 해결된다.

그리고나서 서버를 재시작하면 css가 제대로 적용됨을 확인할 수 있다.


includes 적용

JSP를 작성할 때마다 많은 양의 HTML코드를 피하기 위해 JSP의 includes 지시자를 활용해 사전 작업을 해야한다.

views/includes폴더를 작성하고, header.jsp, footer.jsp를 작성한다.

header.jsp 적용

페이지의 핵심이 아닌, 위쪽 영역 HTML을 처리하기 위해 작성.

image-20200518215532925

page-wrapper부터가 핵심적인 내용이므로, list.jsp의

div id='page-wrapper' 까지 잘라서 header.jsp의 내용으로 처리한다.

<%@ page language="java" contentType="text/html; charset=UTF-8"
    pageEncoding="UTF-8"%>
    
    <%@include file="../includes/header.jsp" %>
     
	<div id="page-wrapper">
        <!-- 하위를 제외한 상단의 모든내용을 header.jsp에 복사 -->

서버를 재시작해 제대로 출력되는지 확인한다.

footer.jsp 적용

div id='page-wrapper'이 끝나는 태그 부터 마지막까지, 마찬가지로 footer.jsp로 처리한다.

jQuery 라이브러리 변경

footer에 jQuery라이브러리가 포함되어 있으므로, header에도 선언을 하면, 작성하는 jsp에서 자유롭게 사용할 수 있음

https://developers.google.com/speed/libraries#jquery 에서 jquery 링크를 추가한다.

header.jsp 하단

<script src="https://ajax.googleapis.com/ajax/libs/jquery/3.5.1/jquery.min.js"></script>

반응형 웹 처리

SB Admin2은 반응형으로 제작되어 있지만, jquery최신 버전 사용상태에서는 새로고침시 메뉴가 펼쳐지는 문제가 발생 현재까지의 과정을 따라했을때 해당 에러는 발생하지 않음 이전 버전의 SB Admin2에서 발생한 문제 같음


목록 화면 처리

아직 많은 HTML코드가 list.jsp에 존재하므로, 태그를 적용시킨다.

JSTL의 출력, 포맷을 적용하는 태그 라이브러리를 추가한다.

<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
<%@ taglib uri="http://java.sun.com/jsp/jstl/fmt" prefix="fmt" %>

JSTL

https://mkil.tistory.com/249

  • JSP 표준 태그 라이브러리

  • 자신만의 태그를 추가할 수 있음

  • c:if, c:forEach

  • 변수 출력, 저장, 반복문, 조건문 등을 JSP에서 사용할 수 있는 라이브러리

Model에 담긴 데이터 출력

에러

No primary or default constructor found for interface java.util.List
널포인터 에러

기존 코드

@Autowired
	private BoardService service;
@GetMapping("/list")
	public String list(Model model) {
		list = service.getList();
		System.out.println("**Controller getList**");
		log.info("list......");
		model.addAttribute("board", list);
		return "/board/list";
	}

기존 코드에 추가

	@Autowired
	private BoardService service;
	//private List<BoardVO> list;
  • List 생성자에러 https://seongtak-yoon.tistory.com/53
    • 기본 생성자가 없는 모델 객체를 바로 사용해서 발생한 에러이다.
    • modelAttribute를 사용하기 위해서, JavaBeans 규칙에 맞는 객체여야한다.(getter/setter 등이 있어야함)

bean 등록

기존 코드처럼, List를 주입 받지 않으면 저 빨간 네모표시(좀 더 찾아봐야함)가 나오지 않고, 에러가 있었는데 수정코드를 작성하니 사진과 같이 list가 생기고 에러가 해결되었다.. https://stackoverflow.com/questions/61872809/what-means-red-square-in-spring

=> Serivce를 주입받지 않아서 발생한 에러.

  • BoardController에서 DB에 있는 값을 불러올 때 널포인터 에러발생했다. Controller단에서 Service를 주입받아 사용해야할 것 같아서 @Setter Serivce를 작성했는데 마찬가지로 같은 오류가 발생했다.
    해당 어노테이션을 @Inject, @Autowired로 변경하니 해결됨.

JSP에서 데이터를 계속 불러오지 못한 이슈가 발생했었는데 DB에서 데이터를 추가한 후 COMMIT을 하지 않아서 list가 비어있는 상태였다. 테이블을 삭제한 후 다시 데이터를 추가시키고 난 다음 COMMIT 과정을 수행하니 해결할 수 있었다.

게시글 화면처리

최종 결과물

해당 템플릿은 페이징처리를 지원한다. 따라서 이를 제거하고 다시 구현할 예정이다.


등록 입력 페이지, 등록처리

데이터를 입력할 register 페이지를 작성한다.

Controller

먼저 입력 form을 보여주기 위한 GET방식의 메소드를 작성한다. 리턴타입이 void이면 명시된 url의 페이지를 출력해준다.

@GetMapping("/register")
	public void register() {
	}
	

JSP에서 작성한 데이터를 controller에 전달하고, 연동된 DB까지 전달하여 저장하는 로직을 구현한다.

그렇기 위해선 GET과 POST방식에 대해서 이해하는 것이 필요하다.

http패킷은 헤더와 바디로 구성되어있다. 헤더에는 클라이언트, 브라우저 정보 그리고 URL등의 정보가 있다.

바디는 보통 비어있는데 특정 데이터를 담아 서버에게 요청보내는 용도로 사용한다. GET, POST는 HTTP메소드로 서버에 데이터를 요청하기 위해 사용한다.

GET, POST

  • 서버에 요청하는 메소드
  • 서버에 데이터를 보내는 것이라고 생각한다.

GET

  • 클라이언트가 요청한 데이터를 URL뒤에 붙여 보낸다.
  • 헤더에 데이터를 포함하여 전송한다.
  • 전송길이에 제한이 있다.
  • ? 마크는 URL의 끝을 알리며, 데이터 표현의 시작점이다.
  • 데이터는 Key, value로 이루어져있고 2개이상을 보낼 때, &로 구분지어진다
  • Body는 빈 상태이다.
  • 다른 사람이 확인할 수 있다.

예시

www.example.com?id=mommoo&pass=1234

출처: https://mommoo.tistory.com/60 [개발자로 홀로 서기]

POST

  • 데이터 전송을 기반으로 한 요청 메소드
  • URL에 붙이지 않고 BODY에다 데이터를 넣어 보낸다.

따라서 View를 출력하기 위해 GET방식의 메소드를 사용해 화면을 뿌려주고, 데이터 전달을 위해서 POST방식의 메소드를 구현한다. 현재 제목, 내용 그리고 작성자의 데이터를 View에서 서버에 전달한다.

전달 후 페이지는 board/register에서 board/list로 이동할 것이고 이때 redirect을 이용한다.

그러나 Redirect 방식은 GET방식이라 데이터 전송에 적절하지 않다.

해당 방법으로 데이터를 전달할 수 있을까?

스프링이 제공하는 RedirectAttributes클래스를 사용하면 된다.

해당 클래스는 리다이렉트가 발생하기 이전에 모든 플래시속성을 세션에 복사한다.

리다이렉트 이후 저장된 플래시 속성을 모델로 이동시킨다.

addFlashAttirbutes()메소드는 위의 역할을 하는 메소드이며, 리다이렉트 이후 소멸한다.

그렇다면 POST방식으로 Controller에서 데이터를 받아와서(form태그에서 데이터를 받을 때 각 속성의 name은 전달받을 객체의 멤버변수의 이름과 일치해야한다. BoardVO의 title, content, writer 등)

service의 register메소드로 등록(insert)을 처리한 다음, 해당 객체(Board)의 PK키(Bno)에 일치하는 객체를 Redirect하기전에 addFlashAttribute메소드를 통해 View에 전달하여, Redirect해서 호출된 list메소드에서 getList()를 통해 DB에 저장된 게시글 목록을 출력한다.

Redirect를 사용하는 이유

등록, 수정, 삭제작업을 처리하고 동일 내용을 할 수 없더록 하기 위해서 URL이동하는 방식

해당 수행의 결과를 바로알 수 있도록

를 이용하는 모달창을 사용한다.(jsp에서 alert와 비슷한 것 같음)

<script type="text/javascript">
$(document).ready(function)(){
	var result='<c:out value="${result}"/>';
});
</script>

여기서 bno를 받기 위해서 insertSelectKey를 사용해야한다 -> 입력하기전 특정 키 값을 가져온 다음, 그 값을 이용해서 처리하는 방식.

==> Bno가 2씩 증가하는 이유가 여기있는 것 같음. => 특정 키 값을 불러올 때, nextval을 사용하고, values에도 nextval을 쓰기때문에 2번호출하므로 2씩 증가 => 수정

<insert id="insertSelectKey">
		<selectKey keyProperty="bno" order="BEFORE" resultType="long">
			select seq_board.nextval from dual			
		</selectKey>
	
		insert into tbl_board(bno, title, content, writer)
		values(seq_board.nextval, #{title},#{content} ,#{writer})<!-- values의 nextval 부분을 #{bno}로 수정하여 해결-->
	</insert>

목록에서 버튼으로 이동하기

register화면으로 이동하는 버튼을 생성한다.

list.jsp에서 버튼을 추가하고 하단에 버튼의 기능을하는 스크립트를 추가한다.

 <div class="panel-heading">
	 Board List Page
<Button id='regBtn' type="button" class="btn btn-xs pull-right">Register
 New Board</Button>                     
</div>
<!-- ...in script -->
	$("#regBtn").on("click", function(){
				self.location="/board/register";
			});

조회페이지와 이동

  • 페이지링크를 통해 "GET"방식으로 특정 번호 게시물을 조회할 수 있는 기능을 작성한다.

mapper의 get메소드는 mapper.xml에 작성되어있다. = select * , 즉 모든 데이터를 DB에서 가져와 BoardVO객체로 반환한다.


	@GetMapping("/get")
	public void get(@RequestParam("bno") Long bno, Model model) {
		log.info("/get......");
		model.addAttribute("board", service.get(bno));
	}

파라미터로 전달한 'bno'값을 가진 BoardVO객체를 controller에서 받아와, model객체에 전달한다. 전달한 BoardVO객체를 get.jsp에서 받아서 이에 해당하는 값들을 조회할 수 있도록 한다.

JSP

<button data-oper='modify' class="btn btn-default" onclick="location.href='/board/modify/?bno=<c:out value="${board.bno }"/>'">
								Modify
							</button>
							<button data-oper='list' class="btn btn-default" onclick="location.href='/board/list'">
							List</button>
  • data - : 데이터속성. 태그에 데이터를 저장할 수 있는 속성이다. 에는 원하는 name을 붙여주면 된다.

목록(list)페이지에서 글 조회

a 태그를 통해 위 파트에서 진행한 get의 URL로 링크시킨다.

	<td><a target='_blank' href='/board/get?bno=<c:out value="${board.bno }"/>'>
                                   			<c:out value="${board.title }"/></a></td>
                                

같은 페이지로의 이동이 아닌 새창을 통한다면 태그의 속성인 target='_blank'를 지정하면 된다.

뒤로가기의 문제 (스크립트에 대한 학습필요. 정리만 해둠)

  1. 게시글 등록
  2. 목록페이지로 redirect되면서 모달창 생성
  3. 게시글 조회
  4. 조회페이지에서 뒤로가기시 모달창 재 생성

window history는 스택구조로 동작한다.

여기서 O표시는 모달창이 필요없다는 표시로 한다.

  1. board/list(O) 목록페이지 조회
  2. board/list(O) , board/register 등록페이지 이동
  3. board/list(O), board/register , board/list 등록 후 list로 이동 (스택 상단에서 모달창 생성 후 모달창에 표시를 해두어야한다.)
  4. board/list(O), board/register, board/list(O) board/register - 다른 글 등록

3번에서, 스택의 상단인 list페이지에 모달창이 필요없다는 표시를 해두어야 모달창이 다시나오는 오류를 막을 수 있다.

window의 history객체를 이용해 해당 문제를 해결한다.

list.jsp

	
var result='<c:out value="${resultBno}"/>';
//resultBno 데이터를 controller의 register메소드에서 받아와 result에 매핑시킨다.
			
checkModal(result);
//매핑시킨 result를 가지고 chekModal메소드를 호출한다.
			
history.replaceState({}, null, null);
			
			
			
function checkModal(result){
//result가 null이면, 그대로 종료하고 그렇지 않으면 modal창을 생성한다.
//함수호출순서상 checkModal이 먼저 수행되게 되는데, 첫번째 모달창을 출력하고  history.replaceState()를 호출하게 된다.
			//그렇다면 그 다음부터는 뒤로가기를 눌러도 이미 현재 창에서는 모달이 출력되었다는 표시(history.state)가 되었으므로 모달이 나오지 않는다.
				if(result === '' || history.state)
				{
					return ;
				}
				

게시글 수정 / 삭제 처리

수정/삭제 페이지로 이동

조회(get)페이지랑 같은 방법을 사용한다.

GetMapping으로 modify.jsp를 보여줄 수 있는 부분을 추가하고 PostMapping으로 수정된 데이터들을 form으로 묶어 전달하는 부분을 구현한다.

	@PostMapping("/modify")
	public String modify(BoardVO board, RedirectAttributes rttr)
	{
		boolean flag;
		
		log.info("modified..........");
		flag = service.modify(board);
		
		if(!flag) 
		{
			return "redirect:/board/modify";
		}
		rttr.addFlashAttribute("resultBno", board.getBno());
		rttr.addFlashAttribute("separator", "modify");
		return "redirect:/board/list";
	}

separator의 문자열을 따로 추가하여 수정/삭제/등록될때마다 출력되는 문구를 구분짓게 구현했다.

삭제부분

@PostMapping("/remove")
	public String remove(@RequestParam ("bno") Long bno, RedirectAttributes rttr)
	{
		log.info("remove........ bno : " + bno);
		service.remove(bno);
		rttr.addFlashAttribute("resultBno", bno);
		rttr.addFlashAttribute("separator", "remove");
		return "redirect:/board/list";
	}

추가해야할 부분

게시글 작성시 어느 하나라도 데이터가 null로 submit될때 alert수행하고, null인 input박스로 포커싱하도록 추가하기

writer는 세션으로 자동으로 아이디 받아오도록 수정하기

??

register메소드에서 addFlashAttribute를 작성하는 이유는 뭘까?

이미 View에서 받아온 데이터를 register하여 DB에 저장되어있는 상태인데 redirect 호출전 View쪽으로 list라는 이름의 객체로 전달해도 사용하지 않는데 말이다.

redirect하기전에 register화면에서 list객체를 사용할 일이 있다면 작성해주어도 상관없을텐데 말이다. ?!?

=>바로 뒷장에 내용이 나온다 -ㅅ-

Modal창에 데이터를 이용하기 위해서

해당 코드에서 작성된 글의 PK(bno)를 list객체로 전달하는데, modal창에서 이를 사용하기 때문이다.

0개의 댓글