BasicItemController

package hello.itemservice.web.basic;

import hello.itemservice.domain.item.Item;
import hello.itemservice.domain.item.ItemRepository;
import lombok.RequiredArgsConstructor;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;

import javax.annotation.PostConstruct;
import java.util.List;

@Controller
@RequestMapping("/basic/items")
@RequiredArgsConstructor //...1
public class BasicItemController {

    private final ItemRepository itemRepository;

		//...2
    @GetMapping
    public String items(Model model) {
        List<Item> items = itemRepository.findAll();
        model.addAttribute("items", items);
        return "basic/item";
    }

    /**
     * 테스트용 데이터 추가
     */
		//...3
    @PostConstruct
    public void init() {
        itemRepository.save(new Item("itemA", 10000, 10));
        itemRepository.save(new Item("itemB", 20000, 20));
    }

}
  1. 롬복을 활용한 코드 최적화

    참고

  2. items

  3. 테스트용 데이터 추가

items.html 정적 HTML을 뷰 템플릿(templates) 영역으로 복사하고 다음과 같이 수정하자

/resources/static/items.html → 복사 → /resources/templates/basic/items.html

<!DOCTYPE HTML>
<html xmlns:th="<http://www.thymeleaf.org>"> <!--...1-->
<head>
    <meta charset="utf-8">
		<!--...2,3-->
    <link th:href="@{/css/bootstrap.min.css}"
            href="../css/bootstrap.min.css" rel="stylesheet">
</head>
<body>

<div class="container" style="max-width: 600px">
    <div class="py-5 text-center">
        <h2>상품 목록</h2>
    </div>

    <div class="row">
        <div class="col">
						<!--...2,3-->
            <button class="btn btn-primary float-end"
                    onclick="location.href='addForm.html'"
                    th:onclick="|location.href='@{/basic/items/add}'|"
                    type="button">상품 등록
            </button>
        </div>
    </div>

    <hr class="my-4">

    <div>
        <table class="table">
            <thead>
            <tr>
                <th>ID</th>
                <th>상품명</th>
                <th>가격</th>
                <th>수량</th>
            </tr>
            </thead>
            <tbody>
						<!--...4-->
            <tr th:each="item : ${items}">
								<!--...5-->
                <td><a href="item.html" th:href="@{/basic/items/{itemId}(itemId=${item.id})}" th:text="${item.id}">회원id</a></td>
                <td><a href="item.html" th:href="@{|/basic/items/${item.id}|}" th:text="${item.itemName}">상품명</a></td>
                <td th:text="${item.price}">10000</td>
                <td th:text="${item.quantity}">10</td>
            </tr>
            </tbody>
        </table>
    </div>

</div> <!-- /container -->

</body>
</html>
  1. 타임리프 사용 선언

  2. 속성 변경 - th:href

<aside> ❗ 타임리프 핵심

핵심은 th:xxx 가 붙은 부분은 서버사이드에서 렌더링 되고, 기존 것을 대체합니다. th:xxx 이 없으면 기존 html의 xxx 속성이 그대로 사용됩니다.

위의 예시에서 기본은 ../css/bootstrap.min.css 경로로 열립니다.

하지만 서버를 실행하고 뷰 템플릿을 거치면 /css/bootstrap.min.css 경로로 열립니다.

HTML을 파일로 직접 열었을 때, th:xxx 가 있어도 웹 브라우저는 th: 속성을 알지 못하므로 무시합니다.

따라서 HTML을 파일 보기를 유지하면서 템플릿 기능도 할 수 있습니다.

  1. 상품 등록 폼으로 이동
  2. 반복해서 출력
  3. 상품 ID를 선택하는 링크