<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<!-- jstl core 쓸때 태그에 c 로 표시. -->
<%@ taglib uri="http://java.sun.com/jsp/jstl/fmt" prefix="fmt"%>
<!-- jstl fmt 쓸때 위와 같음. fmt : formatter 형식 맞춰서 표시 -->
<%@ include file="../includes/header.jsp"%>
<div class="row">
<div class="col-lg-12">
<h1 class="page-header">Board Register</h1>
</div>
</div>
<div class="row">
<div class="col-lg-12">
<div class="panel panel-default">
<div class="panel-heading">글쓰기</div>
<div class="panel-body">
<form role="form" action="/board/register" method="post">
<div class="form-group">
<label>Title</label> <input class="form-control" name='title'>
</div>
<div class="form-group">
<label>Text area</label>
<textarea class="form-control" rows="3" name='content'></textarea>
</div>
<div class="form-group">
<label>Writer</label> <input class="form-control" name="writer">
</div>
<button type="submit" class="btn btndefault">Submit Button</button>
<button type="reset" class="btn btn-default">Reset Button</button>
</form>
</div>
</div>
</div>
</div>
<%@ include file="../includes/footer.jsp"%>
BoardController.java에 폼을 보여주는 코드를 추가해준다
@GetMapping("/register")
public void register() {
// 이동할 주소를 리턴하지 않는다면, 요청한 이름으로의 jsp 파일을 찾음.
}
<a href="/board/get?bno=${board.bno }"><c:out value="${board.title }" /></a>
리스트의 제목을 표시해주는 부분에 이 메소드를 입력해준다.
get.jsp를 만들고 아래의 내용을 입력해 준다
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/fmt" prefix="fmt"%>
<%@ include file="../includes/header.jsp"%>
<div class="row">
<div class="col-lg-12">
<h1 class="page-header">글 읽기</h1>
</div>
</div>
<div class="row">
<div class="col-lg-12">
<div class="panel panel-default">
<div class="panel-heading"></div>
<div class="panel-body">
<div class="form-group">
게시물 번호<input class="form-control" name="bno"
value='<c:out value="${board.bno }"/>' readonly="readonly">
</div>
<div class="form-group">
제목<input class="form-control" name="title"
value='<c:out
value="${board.title }"/>' readonly="readonly">
</div>
<div class="form-group">
내용
<textarea rows="3" class="form-control" name="content"
readonly="readonly"><c:out value="${board.content }" /></textarea>
</div>
<div class="form-group">
작성자<input class="form-control" name="writer"
value='<c:out
value="${board.writer }"/>' readonly="readonly">
</div>
<button data-oper="modify" class="btn btn-warning">
<a href="/board/modify?bno=${board.bno }"> 수정</a>
</button>
<button data-oper="list" class="btn btn-info">
<a href="/board/list"> 목록</a>
</button>
</div>
</div>
</div>
</div>
BoardController.java에 get 처리에 대한 컨트롤러에 modify를 추가해준다
// 제목 링크를 클릭하여 글 상세보기 - get 방식.
@GetMapping({"/get","/modify"})
public void get(@RequestParam("bno") Long bno, Model model) {
// @RequestParam : 요청 전달값으로 글번호 이용.
log.info("/get");
model.addAttribute("board", service.get(bno));
// jsp에서 request.setAttribute 하던 것과 비슷.
// 전달값으로 명시만 하면 스프링이 자동 처리.
// 사용하는 부분만 추가 구현.
}
modify.jsp를 생성한다
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<!-- jstl core 쓸때 태그에 c 로 표시. -->
<%@ taglib uri="http://java.sun.com/jsp/jstl/fmt" prefix="fmt"%>
<!-- jstl fmt 쓸때 위와 같음.fmt : formatter 형식 맞춰서 표시 -->
<%@ include file="../includes/header.jsp"%>
<div class="row">
<div class="col-lg-12">
<h1 class="page-header">글 수정</h1>
</div>
<!-- /.col-lg-12 -->
</div>
<!-- /.row -->
<div class="row">
<div class="col-lg-12">
<div class="panel panel-default">
<div class="panel-body">
<form role="form" action="/board/modify" method="post">
<input type="hidden" name="bno" value="${board.bno}" />
<div class="form-group">
<label>Title</label> <input class="form-control" name="title"
value="${board.title }">
</div>
<div class="form-group">
<label>Text area</label>
<textarea class="form-control" rows="3" name='content'><c:out
value="${board.content}" /></textarea>
</textarea>
</div>
<div class="form-group">
<label>Writer</label> <input class="form-control" name="writer"
value='<c:out
value="${board.writer }"/>'>
</div>
<button type="submit" data-oper='modify' class="btn btn-success">Modify</button>
<button type="submit" data-oper='remove' class="btn btn-danger">Remove</button>
<button type="submit" data-oper='list' class="btn btn-info">List</button>
</form>
</div>
</div>
</div>
</div>
<script>
$(document).ready(function() {
/* 문서가 준비 됐다면, 아래 함수 수행. */
var formObj = $("form");/* 문서중 form 요소를 찾아서 변수에 할당. */
$('button').on("click", function(e) {
/* 버튼이 클릭된다면 아래 함수 수행, e라는 이벤트 객체를
전달하면서 */
e.preventDefault();/* 기본 이벤트 동작 막기. */
var operation = $(this).data("oper");
/* 버튼에서 oper 속성 읽어서 변수에 할당. */
console.log(operation);
/* 브라우저 로그로 oper값 출력. */
if (operation === 'remove') {
formObj.attr("action", "/board/remove");
/* form에 액션 속성을 변경. */
} else if (operation === 'list') {
self.location = "/board/list";
return;
}
formObj.submit();
/* 위의 조건이 아니라면 수정 처리. */
});
});
</script>
<%@ include file="../includes/footer.jsp"%>
게시판 글 수정을 하게되면 modal 창이 나오게 되는데 안내 메세지가 나오게 수정해보자
function checkModal(result) {
if (result === '') {
// == 는 값만 비교, === 은 값과 형식도 비교.
return;
}
if ($.isNumeric(result)) {// 전달된 값이 숫자인가?
if (parseInt(result) > 0) {
$(".modal-body").html("게시글 " + parseInt(result) + "번 등록");
}
} else {
$(".modal-body").html(result);
// 수정과 삭제시에는 success 라는 문자열이 전달 되므로,
// 성공 메세지만 표시하기 원함.
}
$("#myModal").modal("show");// 모달창 표시
}
글 수정이 완료되면 success라는 모달창이 나오게 된다.
먼저 kr.board.domain 패키지에 Criteria를 구현한다
package kr.board.domain;
import lombok.Getter;
import lombok.Setter;
import lombok.ToString;
@Getter
@Setter
@ToString
public class Criteria {
private int pageNum; // 현재 페이지 번호.
private int amount; // 페이지당 게시물수
public Criteria() {
this(1, 10); // 아래쪽 전달값 2개 생성자 호출.
}
public Criteria(int pageNum, int amount) {
this.pageNum = pageNum;
this.amount = amount;
}
}
BoardMapper.java에도 메소드를 추가해준다
public List<BoardVO> getListWithPaging(Criteria cri);
BoardMapper.xml에도 쿼리 메소드를 추가해준다
<select id="getListWithPaging"
resultType="kr.board.domain.BoardVO">
<![CDATA[
select bno, title, content, writer, regdate, updatedate
from
(
select /*+INDEX_DESC(tbl_board pk_board) */
rownum rn, bno, title, content, writer, regdate,
updatedate
from
tbl_board
where rownum <= #{pageNum} * #{amount}
)
where rn > (#{pageNum}-1) * #{amount}
]]>
</select>
BoardMapperTests에 가서 작동하는지 확인한다
@Test
public void testPaging() {
Criteria cri = new Criteria();
cri.setPageNum(1);
cri.setAmount(10);
List<BoardVO> list = mapper.getListWithPaging(cri);
list.forEach(board -> log.info(board.getBno()));
}
BoardService에서 메소드를 추가한다
public List<BoardVO> getList(Criteria cri);
BoardServiceImp에 가서 메소드를 추가한다
@Override
public List<BoardVO> getList(Criteria cri) {
log.info("getListWithPaging....." + cri);
return mapper.getListWithPaging(cri);
}
BoardServiceTest를 수정한다
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("file:src/main/webapp/WEB-INF/spring/root-context.xml")
@Log4j
public class BoardServiceTest {
@Setter(onMethod_ = @Autowired)
private BoardService service;
@Test
public void testGetList2() {
service.getList(new Criteria(2, 10)).forEach(board -> log.info(board));
}
}
BoardController.java에서 list를 수정한다
@GetMapping("/list")
public void list(Model model, Criteria cri) {
// log.info("list");
// model.addAttribute("list", service.getList());
// 과거 jsp에서는 request.setAttribute로 ArrayList를 전달했지만
// , 같은 역할을 model이 대신.
// 컨트롤러 >> 서비스 >> 매퍼 >> mybatis
log.info("list: " + cri);
model.addAttribute("list", service.getList(cri));
}
페이지 번호 처리를 위한 PageDTO를 구현한다
@Getter
@ToString
public class PageDTO {
private int startPage; // 페이징 시작
private int endPage; // 페이징 끝
private boolean prev, next;
private int total; // 총 게시물수
private Criteria cri; // 현재페이지와 페이지당 게시물수
public PageDTO(Criteria cri, int total) {
this.cri = cri;
this.total = total;
this.endPage = (int) (Math.ceil(cri.getPageNum() / 10.0)) * 10;
// 1페이지라고 가정하면 endPage는 10
this.startPage = this.endPage - 9;// 1
int realEnd = (int) (Math.ceil((total * 1.0) / cri.getAmount()));
// 총게시물이 20개라고 가정하면 realEnd=2
// 페이지당 보여줄 게시물 수는 10개로 가정.
if (realEnd < this.endPage) {
this.endPage = realEnd;
}
this.prev = this.startPage > 1;
this.next = this.endPage < realEnd;
}
}
BoardController 에서 list 부분을 수정한다
@GetMapping("/list")
public void list(Model model, Criteria cri) {
// log.info("list");
// model.addAttribute("list", service.getList());
// 과거 jsp에서는 request.setAttribute로 ArrayList를 전달했지만,
// 같은 역할을 model이 대신한다.
log.info("list: " + cri);
model.addAttribute("list", service.getList(cri));
model.addAttribute("pageMaker", new PageDTO(cri, 190));
}
list.jsp 에 페이지 표시구분 구현
<br>
<div>
<ul class="pagination justify-content-center">
<c:if test="${pageMaker.prev }">
<li class="page-item previous"><a
href="${pageMaker.startPage-1 }" class="page-link">Prev</a></li>
</c:if>
<c:forEach var="num" begin="${pageMaker.startPage }"
end="${pageMaker.endPage }">
<!-- 게시물 목록에서 배열을 이용한 것과 같이 begin은 배열의 시작값, end= 배열의 끝값으로 순환 처리. -->
<li class='page-item ${pageMaker.cri.pageNum==num?" active":""}'>
<!-- pageMaker객체의 cri객체의 pageNum이 현재 num과 같다면, active 출력, 아니라면 공백 출력 active는 스타일 강조처리가 부트스트랩에 이미 설정되어 있음 -->
<a href="${num }" class="page-link">${num }</a>
</li>
</c:forEach>
<c:if test="${pageMaker.next }">
<li class="page-item next"><a href="${pageMaker.endPage+1 }"
class="page-link"> Next</a></li>
</c:if>
</ul>
</div>
페이징을 구현 했지만 페이지 이동이 되지 않는다
list.jsp에 폼을 구성한다
<form id="actionForm" action="/board/list" method="get">
<input type="hidden" name="pageNum" value="${pageMaker.cri.pageNum }">
<input type="hidden" name="amount" value="${pageMaker.cri.amount }">
</form>
그리고 아래의 스크립트를 추가한다
var actionForm = $("#actionForm");
// 클래스 page-item 에 a 링크가 클릭 된다면,
$(".page-item a").on("click", function(e) {
e.preventDefault();
// 기본 이벤트 동작을 막고,
console.log("click");
// 웹 브라우저 검사 창에 클릭을 표시
actionForm.find("input[name='pageNum']").val($(this).attr("href"));
//액션폼 인풋태그에 페이지넘 값을 찾아서,
//href로 받은 값으로 대체함
actionForm.submit();
});
게시글을 누르고 들어간뒤 목록을 선택했을때 원래의 링크로 이동하는 메소드를 추가해보자
list.jsp에서
<td><a href="/board/get?bno=${board.bno }"><c:out
value="${board.title }" /></a></td>
에서
<td><a href="${board.bno }" class="move"> <c:out
value="${board.title }" /></a></td>
로 바꿔 준다
list.jsp에 메소드를 추가해준다
$(".move").on(
"click",
function(e) {
e.preventDefault();
actionForm
.append("<input type='hidden' name='bno' "
+ "value='" + $(this).attr("href")
+ "'>");
actionForm.attr("action", "/board/get");
actionForm.submit();
});
BoardController에서 수정해준다
@GetMapping({ "/get", "/modify" })
public void get(@RequestParam("bno") Long bno, @ModelAttribute("cri") Criteria cri, Model model) {
// @ModelAttribute("cri") 는 자동으로 객체 할당 저장.
// 자동으로 생성된 Criteria cri를 모델값으로 저장하는데,
// 저장명이 cri
// @RequestParam : 요청 전달값으로 글번호 이용.
log.info("/get");
model.addAttribute("board", service.get(bno));
// jsp에서 request.setAttribute 하던 것과 비슷.
// 전달값으로 명시만 하면 스프링이 자동 처리.
// 사용하는 부분만 추가 구현.
}
get.jsp에 hidden 값을 추가해준다
<form id="operForm" action="/bound/modify" method="get">
<input type="hidden" id="bno" name="bno" value="${board.bno }" />
<input type="hidden" name="pageNum" value="${cri.pageNum }" /> <input
type="hidden" name="amount" value="${cri.amount }" />
</form>
get.jsp에서 목록 버튼 수정해준다
<button data-oper="list" class="btn btn-info">
<a href="/board/list?pageNum=${cri.pageNum }&amount=${cri.amount }">
목록</a>
</button>