Sparta 내일배움캠프 Java 5기/[사전캠프] mini_mini Team Project ✓
(24.04.08) 프로젝트 1차 코드 취합
프로젝트 1차 코드 취합
팀원별 기능 구현 정리
- A
배너 무한 루프 기능구현
- B
백그라운드 이미지
등록하기 부분 백그라운드 구별
사진 별 프레임 씌우기
사진, 설명 넣기
- Genie
Wireframe 작성
등록 모달 구현
-D
등록된 카드에 삭제 버튼 구현하고, 버튼을 클릭하면 삭제 모달창을 띄우고, 모달창에 예, 아니요 둘 중 하나를 선택하고 한 후 예를 선택하면 등록된 카드를 삭제하게 만들기
더보기
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>배스킨라빈스(추후수정)</title>
<script src="https://ajax.googleapis.com/ajax/libs/jquery/3.5.1/jquery.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/js/bootstrap.bundle.min.js"></script>
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.0.2/dist/css/bootstrap.min.css" rel="stylesheet"
integrity="sha384-EVSTQN3/azprG1Anm3QDgpJLIm9Nao0Yz1ztcQTwFspd3yD65VohhpuuCOmLASjC" crossorigin="anonymous">
<style>
@import url('https://fonts.googleapis.com/css2?family=Noto+Sans+KR:wght@100..900&display=swap');
* {
font-family: "Noto Sans KR", sans-serif;
font-optical-sizing: auto;
font-weight: <weight>;
font-style: normal;
}
body {
background-size: cover;
background-position: center;
background-attachment: fixed;
}
.mytitle {
background-color: transparent;
height: 300px;
color: black;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
}
.mytitle>button {
height: 50px;
weight: 150px;
width:auto;
background-color: #F986BD;
color: white;
border: 1px solid #F986BD;
border-radius: 5px;
margin-top: 10px;
}
.mycard {
width: auto;
height: 50%;
margin: 30px auto 0px auto;
background-color: white;
}
.mypostingbox {
width: 500px;
margin: 30px auto 0px auto;
padding: 20px;
box-shadow: 0px 0px 3px 0px blue;
border-radius: 5px;
}
.mybtn {
display: flex;
flex-direction: row;
align-items: center;
justify-content: center;
}
.mybtn>button {
margin-right: 5px;
}
/*모달에 관한 CSS */
.modal-footer {
display: flex;
justify-content: center;
}
.modal-footer button {
margin: 0 5px;
width: 40%;
}
.btn-flavors {
margin-bottom: 10px;
}
.btn-flavors .btn {
margin: 5px;
}
/*등록 카드의 아이스크림 배치 등에 관한 */
.card .row .col img {
width: 120px;
}
.card .row {
margin: -30px 0;
}
.card .col {
margin-right: -150px;
}
.card {
display: flex;
justify-content: center;
padding-top: 30px;
background-color: white;
}
.buttons {
margin-right: 5px;
}
.section input[id*="slide"] {
display: none;
}
.section {
background-color: blanchedalmond;
width: auto;
height: 35%;
margin: 50px auto 0 auto;
}
.section .slidewrap {
width: auto;
height: auto;
margin: 0 auto 0 auto;
position: center;
}
.section .slidelist {
white-space: nowrap;
font-size: 0;
overflow: hidden;
position: relative;
width: auto;
height: auto;
margin: 10px 15% 10px 15%;
}
.section .slidelist>li {
display: inline-block;
vertical-align: middle;
width: 100%;
transition: all .7s;
}
.section .slidelist>li>a {
display: block;
position: relative;
}
.section .slidelist>li>a img {
width: 100%;
}
.section .slidelist label {
position: absolute;
z-index: 10;
top: 50%;
transform: translateY(-50%);
padding: 50px;
cursor: pointer;
}
.section .slidelist .textbox {
position: absolute;
z-index: 1;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
line-height: 1.6;
text-align: center;
}
/* input에 체크되면 슬라이드 효과 */
.section input[id="slide01"]:checked~.slidewrap .slidelist>li {
transform: translateX(0%);
}
.section input[id="slide02"]:checked~.slidewrap .slidelist>li {
transform: translateX(-100%);
}
.section input[id="slide03"]:checked~.slidewrap .slidelist>li {
transform: translateX(-200%);
}
/* 좌,우 슬라이드 버튼 */
.slide-control>div {
display: none;
}
.section .left {
left: 30px;
background: url('./img/left.png') center center / 100% no-repeat;
}
.section .right {
right: 30px;
background: url('./img/right.png') center center / 100% no-repeat;
}
.section input[id="slide01"]:checked~.slidewrap .slide-control>div:nth-child(1) {
display: block;
}
.section input[id="slide02"]:checked~.slidewrap .slide-control>div:nth-child(2) {
display: block;
}
.section input[id="slide03"]:checked~.slidewrap .slide-control>div:nth-child(3) {
display: block;
}
/* 페이징 */
.slide-pagelist {
text-align: center;
padding: 20px;
}
.slide-pagelist>li {
display: inline-block;
vertical-align: middle;
}
.slide-pagelist>li>label {
display: block;
padding: 8px 30px;
border-radius: 30px;
background: #ccc;
margin: 20px auto 10px auto;
cursor: pointer;
}
.section input[id="slide01"]:checked~.slidewrap .slide-pagelist>li:nth-child(1)>label {
background: #999;
}
.section input[id="slide02"]:checked~.slidewrap .slide-pagelist>li:nth-child(2)>label {
background: #999;
}
.section input[id="slide03"]:checked~.slidewrap .slide-pagelist>li:nth-child(3)>label {
background: #999;
}
</style>
<script type="module">
import { initializeApp } from "https://www.gstatic.com/firebasejs/9.22.0/firebase-app.js";
import { getFirestore } from "https://www.gstatic.com/firebasejs/9.22.0/firebase-firestore.js";
import { collection, addDoc } from "https://www.gstatic.com/firebasejs/9.22.0/firebase-firestore.js";
import { getDocs } from "https://www.gstatic.com/firebasejs/9.22.0/firebase-firestore.js";
// Firebase 구성 정보 설정
const firebaseConfig = {
apiKey: "AIzaSyCGxdvDyVsYEfoEnitKaonaXjMaxpxUzDw",
authDomain: "sparta-bee1a.firebaseapp.com",
projectId: "sparta-bee1a",
storageBucket: "sparta-bee1a.appspot.com",
messagingSenderId: "354592468831",
appId: "1:354592468831:web:86f959a0da2ff302d9d0ea",
measurementId: "G-FW22WNY4XT"
};
const app = initializeApp(firebaseConfig);
const db = getFirestore(app);
$("#postingbtn").click(async function () {
let title = $('#title').val();
let name = $('#name').val();
let recommend = $('#recommend').val();
var selectedFlavors = [];
// 선택한 맛에대한 이미지 URL을 baskin_DB=아이스크림 이미지 DB에서 불러서 저장하기
$(".btn-flavors .btn").each(function () {
if ($(this).hasClass('active')) {
selectedFlavors.push($(this).text());
}
});
let flavorsString = selectedFlavors.join(', ');
let imageURLs = [];
// baskin_DB에는 flavor아이스크림 이름, imageURL 해당 이미지URL 필드로 자료저장되어있음
for (let i = 0; i < selectedFlavors.length; i++) {
let flavor = selectedFlavors[i];
let querySnapshot = await getDocs(collection(db, "baskin_DB"));
querySnapshot.forEach((doc) => {
if (doc.data().flavor === flavor) {
imageURLs.push(doc.data().imageURL);
}
});
}
let doc = {
'title': title,
'name': name,
'recommend': recommend,
'imageURL': imageURLs, // 가져온 imageURL들 배열
'flavors': flavorsString
};
await addDoc(collection(db, "baskin_register"), doc);
alert('등록되었습니다');
window.location.reload();
});
$("#postingbtn-close").click(function () {
$('#exampleModal').modal('hide');
});
// 등록된 imageURL들은 많아 배열로 만든 후, 반복문으로 2열로 만들어 카드에 넣기
let docs = await getDocs(collection(db, "baskin_register"));
docs.forEach((doc) => {
let row = doc.data();
let images = row['imageURL'];
let title = row['title'];
let name = row['name'];
let recommend = row['recommend'];
let flavors = row['flavors'];
let temp_html = `
<div class="col">
<div class="card h-100">
<div class="row flavor-table">`;
images.forEach((image, index) => {
temp_html += `
<div class="col">
<img src="${image}" class="card-img-top" alt="...">
</div>`;
if ((index + 1) % 2 === 0 && index !== images.length - 1) {
temp_html += `</div><div class="row flavor-table">`;
}
});
temp_html += `
</div> <!-- flavor-table 닫음 -->
<div class="card-body">
<h5 class="card-title" style="margin-top: 30px;">${title}</h5>
<p class="card-text" style="color: gray;">${recommend}</p>
<br>
<p class="card-text" style="font-size: 13px;">by ${name}</p>
</div>
<div class="card-footer">
<small class="text-body-secondary" style="color: gray;"">${flavors}</small>
</div>
<div class="buttons">
<button type="button" class="btn btn-dark" id="delete1" style="margin-left: 45px;">삭제</button>
<button type="button" class="btn btn-dark" id="enlargement" style="margin-left: 10px;" >확대</button>
</div>
</div>
</div>`;
$('#card').append(temp_html);
});
$('#postingbox').hide();
$("#savebtn").click(async function () {
$('#postingbox').toggle();
})
document.addEventListener('click', async function (event) {
if (event.target && event.target.id === 'delete1') {
if (confirm('정말 삭제하시겠습니까?')) {
var card = event.target.closest('.card')
console.log(card)
if (card) {
var cardClass = card.classList.value
card.style.transition = 'transform 0.5s ease-in-out'
card.style.transform = 'translateX(500%)'
setTimeout(() => {
card.parentElement.removeChild(card)
}, 500)
}
}
} else if (event.target && event.target.id === 'enlargement') {
var card = event.target.closest('.card')
if (card) {
// 현재 확대 상태를 확인하기 위한 변수
var isEnlarged = card.style.transform === 'scale(1.5)'
// 확대/축소 토글
if (isEnlarged) {
card.style.transform = '' // 기본 값으로 변경하여 축소
card.style.zIndex = '' // 기본 값으로 변경하여 다른 카드와 같은 레벨로 설정
} else {
// 확대될 때 zIndex 값을 크게 설정하여 다른 카드보다 앞쪽에 표시
card.style.transition = 'transform 0.5s ease-in-out'
card.style.transform = 'scale(1.5)' // 확대
card.style.zIndex = '999' // 다른 카드보다 큰 값으로 설정
// 여기에 추가적인 확대 작업을 수행할 수 있습니다.
}
}
}
})
</script>
<script>
$(document).ready(function () {
$('#modalbtn').click(function () {
$('#exampleModal').modal('show');
});
});
</script>
<script>
$(document).ready(function () {
$('.btn').on('click', function () {
$(this).toggleClass('active');
});
});
</script>
</head>
<body>
<div class="section">
<input type="radio" name="slide" id="slide01" checked>
<input type="radio" name="slide" id="slide02">
<input type="radio" name="slide" id="slide03">
<div class="slidewrap">
<ul class="slidelist">
<!-- 슬라이드 영역 -->
<li class="slideitem">
<a>
<img src="https://www.baskinrobbins.co.kr/upload/main/banner/77c24fc6211c25ff06a303ef04b0a5b6.png">
</a>
</li>
<li class="slideitem">
<a>
<img src="">
</a>
</li>
<li class="slideitem">
<a>
<img src="https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcSJi0Xjx1O8xmr_I77c5pLWeHJ9gV9B4OY_Gg&s">
</a>
</li class="slideitem">
<!-- 좌,우 슬라이드 버튼 -->
<div class="slide-control">
<div>
<label for="slide03" class="left"></label>
<label for="slide02" class="right"></label>
</div>
<div>
<label for="slide01" class="left"></label>
<label for="slide03" class="right"></label>
</div>
<div>
<label for="slide02" class="left"></label>
<label for="slide01" class="right"></label>
</div>
</div>
</ul>
<!-- 페이징 -->
<ul class="slide-pagelist">
<li><label for="slide01"></label></li>
<li><label for="slide02"></label></li>
<li><label for="slide03"></label></li>
</ul>
<div class="mytitle">
<h1 style="font-size: 50px;">배스킨라빈스(추후수정)</h1>
<button id="modalbtn" type="button" class="btn btn-info" data-bs-toggle="modal" data-bs-target="#exampleModal">
등록하기
</button>
</div>
<div class="mycard">
<div id="card" class="row row-cols-1 row-cols-md-5 g-4">
</div>
</div>
<!-- 아이스크림 선택 모달 -->
<div class="modal fade" id="exampleModal" tabindex="-1" aria-labelledby="exampleModalLabel" aria-hidden="true">
<div class="modal-dialog modal-dialog-centered">
<div class="modal-content">
<div class="modal-header">
<h5 class="modal-title" id="exampleModalLabel">정보 입력하기</h5>
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
</div>
<div class="modal-body">
<!-- 아이스크림 선택 버튼들 -->
<div class="btn-flavors">
<input type="checkbox" class="btn-check" id="btn-check-1" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-1">아이스 도쿄바나나</label>
<input type="checkbox" class="btn-check" id="btn-check-2" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-2">와사비 아이스크림</label>
<input type="checkbox" class="btn-check" id="btn-check-3" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-3">팥있는 말차당</label>
<input type="checkbox" class="btn-check" id="btn-check-4" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-4">봉쥬르, 마카롱</label>
<input type="checkbox" class="btn-check" id="btn-check-5" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-5">피치 요거트</label>
<input type="checkbox" class="btn-check" id="btn-check-6" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-6">민트 초코 봉봉</label>
<input type="checkbox" class="btn-check" id="btn-check-7" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-7">망고 탱고</label>
<input type="checkbox" class="btn-check" id="btn-check-8" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-8">러브미</label>
<input type="checkbox" class="btn-check" id="btn-check-9" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-9">아빠는 딸바봉</label>
<input type="checkbox" class="btn-check" id="btn-check-10" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-10">엄마는 외계인</label>
<input type="checkbox" class="btn-check" id="btn-check-11" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-11">민트 초콜릿 칩</label>
<input type="checkbox" class="btn-check" id="btn-check-12" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-12">뉴욕 치즈케이크</label>
<input type="checkbox" class="btn-check" id="btn-check-13" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-13">레인보우 샤베트</label>
<input type="checkbox" class="btn-check" id="btn-check-14" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-14">체리쥬빌레</label>
<input type="checkbox" class="btn-check" id="btn-check-15" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-15">슈팅스타</label>
<input type="checkbox" class="btn-check" id="btn-check-16" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-16">사랑에 빠진 딸기</label>
<input type="checkbox" class="btn-check" id="btn-check-17" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-17">오레오 쿠키 앤 크림</label>
<input type="checkbox" class="btn-check" id="btn-check-18" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-18">베리베리 스트로베리</label>
<input type="checkbox" class="btn-check" id="btn-check-19" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-19">31요거트</label>
<input type="checkbox" class="btn-check" id="btn-check-20" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-20">바람과 함께 사라지다</label>
<input type="checkbox" class="btn-check" id="btn-check-21" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-21">바닐라</label>
<input type="checkbox" class="btn-check" id="btn-check-22" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-22">이상한 나라의 솜사탕</label>
<input type="checkbox" class="btn-check" id="btn-check-23" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-23">초콜릿 무스</label>
<input type="checkbox" class="btn-check" id="btn-check-24" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-24">그린티</label>
<input type="checkbox" class="btn-check" id="btn-check-25" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-25">피스타치오 아몬드</label>
<input type="checkbox" class="btn-check" id="btn-check-26" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-26">초콜릿</label>
<input type="checkbox" class="btn-check" id="btn-check-27" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-27">자모카 아몬드 훠지</label>
<input type="checkbox" class="btn-check" id="btn-check-28" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-28">월넛</label>
<input type="checkbox" class="btn-check" id="btn-check-29" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-29">아몬드 봉봉</label>
<input type="checkbox" class="btn-check" id="btn-check-30" autocomplete="off">
<label class="btn btn-outline-secondary" for="btn-check-30">애플 민트</label>
</div>
<div class="form-floating mb-3">
<input type="email" class="form-control" id="title" placeholder="name@example.com">
<label for="floatingInput">제목</label>
</div>
<div class="form-floating mb-3">
<input type="email" class="form-control" id="name" placeholder="name@example.com">
<label for="floatingInput">작성자 이름</label>
</div>
<div class="form-floating">
<textarea class="form-control" id="recommend" placeholder="Leave a comment here"
id="floatingTextarea"></textarea>
<label for="floatingTextarea">추천 이유</label>
</div>
</div>
<div class="modal-footer">
<!-- 모달 푸터 -->
<button id="postingbtn" type="button" class="btn btn-info">등록하기</button>
<button id="postingbtn-close" type="button" class="btn btn-secondary"
data-bs-dismiss="modal">닫기</button>
</div>
</div>
</div>
</div>
</body>
</html>
+
추가 기능 구상 및
코드 수정 2차 취합 계획
'Sparta 내일배움캠프 Java 5기 > [사전캠프] mini_mini Team Project ✓' 카테고리의 다른 글
(24.04.05) 기능 구현 코드 짜기 (0) | 2024.04.08 |
---|---|
(24.04.05) 미니프로젝트 선정, 와이어프레임, 기능 구현 분배 (1) | 2024.04.08 |
(24.04.02) 미니프로젝트 기획안 + 브레인스토밍 (0) | 2024.04.02 |