<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width">
<title>repl.it</title>
<link href="style.css" rel="stylesheet" type="text/css" />
</head>
<body>
<!-- 상단의 네비게이션 바 부분 -->
<div class="nav">
<div class ="nav-item">메뉴1</div>
<div class ="nav-item">메뉴2</div>
<div class ="nav-item">메뉴3</div>
<div class ="nav-item">메뉴4</div>
</div>
<!-- 중간 테이블 -->
<div class = "main">
<div class ="title">
타이틀 부분
</div>
<div class="subtitle">
타이틀 아래의 내용을 기입 할 수 있습니다.
</div>
</div>
<div class = "leftside">
<div class="leftside-item">
왼쪽입니다.
</div>
<div class="leftside-item">
왼쪽2입니다.
</div>
<div class="leftside-item">
왼쪽3입니다.
</div>
</div>
<script src="script.js"></script>
</body>
</html>
CSS (스타일링) 구현하기
.box{
width : 100px;
height : 100px;
border : 1px solid black;
}
1. css는 HTML 엘레먼트의 스타일링을 하기 위해 사용
2. box는 class="box" 속성을 가진 엘레먼트를 스타일하고 싶다고 명시하는 것
3. width,height,border 같은 규칙을 써서 엘레먼트의 모양 및 위치를 바꿈
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width">
<title>repl.it</title>
<link href="style.css" rel="stylesheet" type="text/css" />
<!-- 네비게이션 바 부분 css
height : 높이를 지정
border-bottom : 1px solid black : 선을 하나 그어준다.
display : flex; : 한 줄로 정렬한다.
align-items : center; : 글자를 중간으로 옮긴다.
-->
<style>
.nav {
height : 70px;
border-bottom : 1px solid black;
display : flex;
align-items : center;
}
</style>
</head>
<body>
<!-- 상단의 네비게이션 바 부분 -->
<div class="nav">
<div class ="nav-item">메뉴1</div>
<div class ="nav-item">메뉴2</div>
<div class ="nav-item">메뉴3</div>
<div class ="nav-item">메뉴4</div>
</div>
<!-- 중간 테이블 -->
<div class = "main">
<div class ="title">
타이틀 부분
</div>
<div class="subtitle">
타이틀 아래의 내용을 기입 할 수 있습니다.
</div>
</div>
<div class = "leftside">
<div class="leftside-item">
왼쪽입니다.
</div>
<div class="leftside-item">
왼쪽2입니다.
</div>
<div class="leftside-item">
왼쪽3입니다.
</div>
</div>
<script src="script.js"></script>
</body>
</html>
스타일링의 결과 값
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width">
<title>repl.it</title>
<link href="style.css" rel="stylesheet" type="text/css" />
<!-- 네비게이션 바 부분 css -->
<style>
.nav {
height : 70px;
border-bottom : 1px solid black;
display : flex;
align-items : center;
}
.nav-right-items {
display : flex;
margin-left : auto;
}
.nav-item {
margin-left : 10px;
}
.company-name {
margin-left : 20px;
}
.title {
font-size : 60px;
font-weight : bold;
}
.subtitle {
font-size : 40px;
font-weight : 300;
}
</style>
</head>
<body>
<!-- 상단의 네비게이션 바 부분 -->
<div class="nav">
<div class="company-name">
GO치삼의 개발일지
</div>
<!-- 감싼 부분을 nav-right-item로 정의한다. -->
<div class="nav-right-items">
<div class ="nav-item">메뉴1</div>
<div class ="nav-item">메뉴2</div>
<div class ="nav-item">메뉴3</div>
<div class ="nav-item">메뉴4</div>
</div>
</div>
<!-- 중간 테이블 -->
<div class = "main">
<div class ="title">
타이틀 부분
</div>
<div class="subtitle">
타이틀 아래의 내용을 기입 할 수 있습니다.
</div>
</div>
<div class = "leftside">
<div class="leftside-item">
왼쪽입니다.
</div>
<div class="leftside-item">
왼쪽2입니다.
</div>
<div class="leftside-item">
왼쪽3입니다.
</div>
</div>
<script src="script.js"></script>
</body>
</html>
스타일링의 결과값
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width">
<title>repl.it</title>
<link href="style.css" rel="stylesheet" type="text/css" />
<!-- 네비게이션 바 부분 css -->
<style>
.nav {
height : 70px;
border-bottom : 1px solid black;
display : flex;
align-items : center;
}
.nav-right-items {
display : flex;
margin-left : auto;
}
.nav-item {
margin-left : 10px;
}
.company-name {
margin-left : 20px;
}
.title {
text-align : center;
font-size : 60px;
font-weight : bold;
}
.subtitle {
text-align : center;
font-size : 40px;
font-weight : 300;
}
</style>
</head>
<body>
<!-- 상단의 네비게이션 바 부분 -->
<div class="nav">
<div class="company-name">
GO치삼의 개발일지
</div>
<!-- 감싼 부분을 nav-right-item로 정의한다. -->
<div class="nav-right-items">
<div class ="nav-item">메뉴1</div>
<div class ="nav-item">메뉴2</div>
<div class ="nav-item">메뉴3</div>
<div class ="nav-item">메뉴4</div>
</div>
</div>
<!-- 중간 테이블 -->
<div class = "main">
<div class ="title">
타이틀 부분
</div>
<div class="subtitle">
타이틀 아래의 내용을 기입 할 수 있습니다.
</div>
</div>
<div class = "leftside">
<div class="leftside-item">
왼쪽입니다.
</div>
<div class="leftside-item">
왼쪽2입니다.
</div>
<div class="leftside-item">
왼쪽3입니다.
</div>
</div>
<script src="script.js"></script>
</body>
</html>
글의 내용을 중앙으로 정렬한다.
정렬의 결과값
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width">
<title>repl.it</title>
<link href="style.css" rel="stylesheet" type="text/css" />
<!-- 네비게이션 바 부분 css -->
<style>
.nav {
height : 70px;
border-bottom : 1px solid black;
display : flex;
align-items : center;
}
.nav-right-items {
display : flex;
margin-left : auto;
}
.nav-item {
margin-left : 10px;
}
.company-name {
margin-left : 20px;
}
.title {
text-align : center;
font-size : 3.5rem;
font-weight : bold;
}
.subtitle {
text-align : center;
font-size : 1.25rem;
font-weight : 300;
}
.main {
width : 800px;
margin : 0 auto;
margin-top : 60px;
}
</style>
</head>
<body>
<!-- 상단의 네비게이션 바 부분 -->
<div class="nav">
<div class="company-name">
GO치삼의 개발일지
</div>
<!-- 감싼 부분을 nav-right-item로 정의한다. -->
<div class="nav-right-items">
<div class ="nav-item">메뉴1</div>
<div class ="nav-item">메뉴2</div>
<div class ="nav-item">메뉴3</div>
<div class ="nav-item">메뉴4</div>
</div>
</div>
<!-- 중간 테이블 -->
<div class = "main">
<div class ="title">
타이틀 부분
</div>
<div class="subtitle">
타이틀 아래의 내용을 기입 할 수 있습니다.
</div>
</div>
<div class = "leftside">
<div class="leftside-item">
왼쪽입니다.
</div>
<div class="leftside-item">
왼쪽2입니다.
</div>
<div class="leftside-item">
왼쪽3입니다.
</div>
</div>
<script src="script.js"></script>
</body>
</html>
margin-top으로 윗 부붙에 여백을 지정해줍니다.
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width">
<title>repl.it</title>
<link href="style.css" rel="stylesheet" type="text/css" />
<!-- 네비게이션 바 부분 css -->
<style>
.nav {
height : 70px;
border-bottom : 1px solid black;
display : flex;
align-items : center;
}
.nav-right-items {
display : flex;
margin-left : auto;
}
.nav-item {
margin-left : 10px;
}
.company-name {
margin-left : 20px;
}
.title {
text-align : center;
font-size : 3.5rem;
font-weight : bold;
}
.subtitle {
text-align : center;
font-size : 1.25rem;
font-weight : 300;
}
.main {
width : 1000px;
margin : 0 auto;
margin-top : 60px;
}
.leftside-item {
width : 300px;
height : 350px;
border : 1px solid black;
margin : 20px;
border-radius : 4px;
}
.leftside {
display : flex;
}
.leftside-item-title {
font-size: 1.5rem;
background : rgba(0,0,0,.03);
text-align : center;
height : 53px;
line-height : 53px;
border-bottom: 1px solid black;
}
.leftside-item-price {
font-size: 1.5rem;
font-weight : bold;
padding : 20px;
text-align : center;
}
.leftside-item-detail {
font-size: 1.0rem;
font-weight : bold;
padding : 20px;
text-align : center;
}
.leftside-item-button {
padding: .5rem 1rem;
font-size: 1.25rem;
line-height : 1.5;
border-radius : .3rem;
color : #007bff;
background-color : transparent;
background-image : none;
border-color: #007bff;
margin-top : 20px;
margin-left : 85px;
}
.leftside-item-button-active {
background-color : #007bff;
color : white;
}
</style>
</head>
<body>
<!-- 상단의 네비게이션 바 부분 -->
<div class="nav">
<div class="company-name">
GO치삼의 개발일지
</div>
<!-- 감싼 부분을 nav-right-item로 정의한다. -->
<div class="nav-right-items">
<div class ="nav-item">메뉴1</div>
<div class ="nav-item">메뉴2</div>
<div class ="nav-item">메뉴3</div>
<div class ="nav-item">메뉴4</div>
</div>
</div>
<!-- 중간 테이블 -->
<div class = "main">
<div class ="title">
타이틀 부분
</div>
<div class="subtitle">
타이틀 아래의 내용을 기입 할 수 있습니다.타이틀 아래의 내용을 기입 할 수 있습니다.타이틀 아래의 내용을 기입 할 수 있습니다.타이틀 아래의 내용을 기입 할 수 있습니다.타이틀 아래의 내용을 기입 할 수 있습니다.타이틀 아래의 내용을 기입 할 수 있습니다.타이틀 아래의 내용을 기입 할 수 있습니다.타이틀 아래의 내용을 기입 할 수 있습니다.타이틀 아래의 내용을 기입 할 수 있습니다.타이틀 아래의 내용을 기입 할 수 있습니다.타이틀 아래의 내용을 기입 할 수 있습니다.타이틀 아래의 내용을 기입 할 수 있습니다.
</div>
<div class = "leftside">
<div class="leftside-item">
<div class = "leftside-item-title">
왼쪽의 윗칸
</div>
<div class = "leftside-item-price">
왼쪽의 중간칸(가격)
</div>
<div class = "leftside-item-detail">
왼쪽의 아랫칸
</div>
<button class = "leftside-item-button">
클릭하세요
</button>
</div>
<div class="leftside-item">
<div class = "leftside-item-title">
왼쪽1의 윗칸
</div>
<div class = "leftside-item-price">
왼쪽1의 중간칸
</div>
<div class = "leftside-item-detail">
왼쪽1의 아랫칸
</div>
<button class = "leftside-item-button leftside-item-button-active">
클릭하세요
</button>
</div>
<div class="leftside-item">
<div class = "leftside-item-title">
왼쪽2의 윗칸
</div>
<div class = "leftside-item-price">
왼쪽2의 중간칸
</div>
<div class = "leftside-item-detail">
왼쪽2의 아랫칸
</div>
<button class = "leftside-item-button leftside-item-button-active" >
클릭하세요
</button>
</div>
</div>
</div>
<script src="script.js"></script>
</body>
</html>
같은 방식으로 신청 해준 후 , 파일질러로 닷홈 접속과 같이 들어간 후 같이 작업한 파일을 업로드 해준다.
컨트롤+F키로 email 주소를 찾아 준 뒤 수정 해주도록 한다.보여지는 결과값들jqBootstrapValidation.js를 열어주어 메일 기능 오류시 나오는 오류를 한글로 수정해준다.
if (settings.options.sniffHtml) {
var message = "";
// ---------------------------------------------------------
// PATTERN
// ---------------------------------------------------------
if ($this.attr("pattern") !== undefined) {
message = "예상치 못한 형식입니다.<!-- data-validation-pattern-message to override -->";
if ($this.data("validationPatternMessage")) {
message = $this.data("validationPatternMessage");
}
$this.data("validationPatternMessage", message);
$this.data("validationPatternRegex", $this.attr("pattern"));
}
// ---------------------------------------------------------
// MAX
// ---------------------------------------------------------
if ($this.attr("max") !== undefined || $this.attr("aria-valuemax") !== undefined) {
var max = ($this.attr("max") !== undefined ? $this.attr("max") : $this.attr("aria-valuemax"));
message = "값이 너무 높습니다. 최댓값 : '" + max + "'<!-- data-validation-max-message to override -->";
if ($this.data("validationMaxMessage")) {
message = $this.data("validationMaxMessage");
}
$this.data("validationMaxMessage", message);
$this.data("validationMaxMax", max);
}
// ---------------------------------------------------------
// MIN
// ---------------------------------------------------------
if ($this.attr("min") !== undefined || $this.attr("aria-valuemin") !== undefined) {
var min = ($this.attr("min") !== undefined ? $this.attr("min") : $this.attr("aria-valuemin"));
message = "값이 너무 낮습니다. 최솟값 :'" + min + "'<!-- data-validation-min-message to override -->";
if ($this.data("validationMinMessage")) {
message = $this.data("validationMinMessage");
}
$this.data("validationMinMessage", message);
$this.data("validationMinMin", min);
}
// ---------------------------------------------------------
// MAXLENGTH
// ---------------------------------------------------------
if ($this.attr("maxlength") !== undefined) {
message = "값이 너무 깁니다. 최댓값 : '" + $this.attr("maxlength") + "' characters<!-- data-validation-maxlength-message to override -->";
if ($this.data("validationMaxlengthMessage")) {
message = $this.data("validationMaxlengthMessage");
}
$this.data("validationMaxlengthMessage", message);
$this.data("validationMaxlengthMaxlength", $this.attr("maxlength"));
}
// ---------------------------------------------------------
// MINLENGTH
// ---------------------------------------------------------
if ($this.attr("minlength") !== undefined) {
message = "값이 너무 짧습니다. 최소값 : '" + $this.attr("minlength") + "' characters<!-- data-validation-minlength-message to override -->";
if ($this.data("validationMinlengthMessage")) {
message = $this.data("validationMinlengthMessage");
}
$this.data("validationMinlengthMessage", message);
$this.data("validationMinlengthMinlength", $this.attr("minlength"));
}
// ---------------------------------------------------------
// REQUIRED
// ---------------------------------------------------------
if ($this.attr("required") !== undefined || $this.attr("aria-required") !== undefined) {
message = settings.builtInValidators.required.message;
if ($this.data("validationRequiredMessage")) {
message = $this.data("validationRequiredMessage");
}
$this.data("validationRequiredMessage", message);
}
// ---------------------------------------------------------
// NUMBER
// ---------------------------------------------------------
if ($this.attr("type") !== undefined && $this.attr("type").toLowerCase() === "number") {
message = settings.builtInValidators.number.message;
if ($this.data("validationNumberMessage")) {
message = $this.data("validationNumberMessage");
}
$this.data("validationNumberMessage", message);
}
// ---------------------------------------------------------
// EMAIL
// ---------------------------------------------------------
if ($this.attr("type") !== undefined && $this.attr("type").toLowerCase() === "email") {
message = "이메일 형식이 잘못 입력 되었습니다. <!-- data-validator-validemail-message to override -->";
if ($this.data("validationValidemailMessage")) {
message = $this.data("validationValidemailMessage");
} else if ($this.data("validationEmailMessage")) {
message = $this.data("validationEmailMessage");
}
$this.data("validationValidemailMessage", message);
}
컨텍트미 파일 수정
$(function () {
$(
"#contactForm input,#contactForm textarea,#contactForm button"
).jqBootstrapValidation({
preventSubmit: true,
submitError: function ($form, event, errors) {
// additional error messages or events
},
submitSuccess: function ($form, event) {
event.preventDefault(); // prevent default submit behaviour
// get values from FORM
var name = $("input#name").val();
var email = $("input#email").val();
var phone = $("input#phone").val();
var message = $("textarea#message").val();
var firstName = name; // For Success/Failure Message
// Check for white space in name for Success/Fail message
if (firstName.indexOf(" ") >= 0) {
firstName = name.split(" ").slice(0, -1).join(" ");
}
$this = $("#sendMessageButton");
$this.prop("disabled", true); // Disable submit button until AJAX call is complete to prevent duplicate messages
$.ajax({
url: "/assets/mail/contact_me.php",
type: "POST",
data: {
name: name,
phone: phone,
email: email,
message: message,
},
cache: false,
success: function () {
// Success message
$("#success").html("<div class='alert alert-success'>");
$("#success > .alert-success")
.html(
"<button type='button' class='close' data-dismiss='alert' aria-hidden='true'>×"
)
.append("</button>");
$("#success > .alert-success").append(
"<strong>메일이 성공적으로 보내졌습니다. </strong>"
);
$("#success > .alert-success").append("</div>");
//clear all fields
$("#contactForm").trigger("reset");
},
error: function () {
// Fail message
$("#success").html("<div class='alert alert-danger'>");
$("#success > .alert-danger")
.html(
"<button type='button' class='close' data-dismiss='alert' aria-hidden='true'>×"
)
.append("</button>");
$("#success > .alert-danger").append(
$("<strong>").text(
"죄송합니다. " +
firstName +
"님, 현재 메일 서버가 응답하지 않습니다. 나중에 다시 시도해주세요. "
)
);
$("#success > .alert-danger").append("</div>");
//clear all fields
$("#contactForm").trigger("reset");
},
complete: function () {
setTimeout(function () {
$this.prop("disabled", false); // Re-enable submit button when AJAX call is complete
}, 1000);
},
});
},
filter: function () {
return $(this).is(":visible");
},
});
$('a[data-toggle="tab"]').click(function (e) {
e.preventDefault();
$(this).tab("show");
});
});
/*When clicking on Full hide fail/success boxes */
$("#name").focus(function () {
$("#success").html("");
});
contact_me.php 파일을 열어 준 후, 수정한다.
<?php
// Check for empty fields
if(empty($_POST['name']) || empty($_POST['email']) || empty($_POST['phone']) || empty($_POST['message']) || !filter_var($_POST['email'], FILTER_VALIDATE_EMAIL)) {
http_response_code(500);
exit();
}
$name = strip_tags(htmlspecialchars($_POST['name']));
$email = strip_tags(htmlspecialchars($_POST['email']));
$phone = strip_tags(htmlspecialchars($_POST['phone']));
$message = strip_tags(htmlspecialchars($_POST['message']));
// Create the email and send the message
$to = "helloqu@naver.com"; // Add your email address in between the "" replacing yourname@yourdomain.com - This is where the form will send a message to.
$subject = "메일이 도착하였습니다.: $name";
$body = "당신의 개인 홈페이지에서 메일이 도착하였습니다.\n\n"."이곳의 자세한 내용을 확인 하세요.:\n\nName: $name\n\nEmail: $email\n\nPhone: $phone\n\nMessage:\n$message";
$header = "From: http://helloqu.dothome.co.kr/\n"; // This is the email address the generated message will be from. We recommend using something like noreply@yourdomain.com.
$header .= "Reply-To: $email";
if(!mail($to, $subject, $body, $header))
http_response_code(500);
?>
package com.ohj.controller;
import java.util.List;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model; // 스프링은 모델이다.(모델 앤 뷰 객체)
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.servlet.mvc.support.RedirectAttributes;
import com.ohj.domain.BoardVO;
import com.ohj.service.BoardService;
import lombok.AllArgsConstructor;
import lombok.extern.log4j.Log4j;
@Controller
@Log4j
@RequestMapping("/board/*")
@AllArgsConstructor
public class BoardController {
private BoardService service;
@GetMapping("/list")
public void list(Model model) {
log.info("list");
model.addAttribute("list", service.getList()); // 반환타입 List<BoardVO> --> list.jsp
}// list() END
// 이 리스트 메소드가 호출되려면 /board/list 로 되어야 한다.
// ModelAndView 에 값 실어주려면 setAttribute 인데 여기선 addAttribute 라는 메소드로 리스트란 명칭을 실어주는 것이다.
@GetMapping("/register")
public void register() {
}//register() END
@PostMapping("/register")
public String register(BoardVO board, RedirectAttributes rttr) {
log.info("register:" + board);
service.register(board);
rttr.addFlashAttribute("result", board.getBno()); // 딱 1회만 전송할 수 있도록 만들어주는 메서드(addFlashAttribute)
return "redirect:/board/list";
}//register() END
@GetMapping({"/get","/modify"}) //매개변수에서 url패턴명이 바로 model객체 내부로 들어가서 return 되는 것(모델 객체가 리턴)
public void get(@RequestParam("bno") Long bno, Model model) { // @RequestParam("bno") 이 부분은 board/get?bno=n 과 같이 데이터 전달을 알려준다.(자동주입)
//request.getParameter("bno")를 대신해주는 것이 @RequestParam("bno")
log.info("/get or modify");
model.addAttribute("board",service.get(bno)); //BoardVO 반환타입
}//get() END
@PostMapping("/modify")
public String modify(BoardVO board, RedirectAttributes rttr) {
log.info("modify : " + board);
if (service.modify(board)) {
rttr.addFlashAttribute("result", "success");
}
return "redirect:/board/list";
}// modify() END
//★삭제는 반드시 POST 방식으로 처리
@PostMapping("/remove")
public String remove(@RequestParam("bno") Long bno, RedirectAttributes rttr) {
log.info("remove...." + bno);
if(service.remove(bno)) {
rttr.addFlashAttribute("result","success");
}
return "redirect:/board/list";
}//remove() END
}//BoardController CLASS END