๋ณธ๋ฌธ์œผ๋กœ ๊ฑด๋„ˆ๋›ฐ๊ธฐ

25-03-06

๐Ÿ“Œ Daily Reportโ€‹

https://github.com/ssginc-be/DOKI/issues/38


๐Ÿ“Œ ํ”„๋กœ์ ํŠธ ์ƒํ™ฉโ€‹

์šด์˜์ž -> ์ด์šฉ์ž ์ธก์˜ SSE ํ† ์ŠคํŠธ UI๊ฐ€ ์–ผ์ถ” ๋๋‚ฌ๋‹ค.

noti_toast_ui

๊ทธ๋Ÿฐ๋ฐ ๋ฌธ์ œ๊ฐ€ ์ƒ๊ฒผ๋‹ค.

์ƒ๊ธด๊ฒŒ ์˜ ๋งˆ์Œ์— ๋“ค์ง€ ์•Š๋Š”๋‹ค ๐Ÿคจ

์• ๋‹ˆ๋ฉ”์ด์…˜์€ ๋ง›์žˆ์ง€๋งŒ ๋„๋ฉ”์ธ ์ƒ๊ฐํ•˜๋ฉด ๋„ˆ๋ฌด ํ†ตํ†ต ํŠ€๋Š”๊ฒƒ ๊ฐ™์Œ.


๐Ÿ“Œ ์Šคํฌ๋ฆฝํŠธ ๋กœ๋”ฉ ์ˆœ์„œ ๊ณ ๋ คํ•˜๊ธฐโ€‹

StoreController.java
// ๋น„ํšŒ์›์ด๊ฑฐ๋‚˜ ๋กœ๊ทธ์ธํ•œ ์ด์šฉ์ž (null ์ฒดํฌ๊ฐ€ ๋งจ ์œ„์— ์žˆ์–ด์•ผ ํ•จ)
if (role == null || role.equals("MEMBER")) {
...
model.addAttribute("memberCode", code);

return "index"; // ํŒ์—…์Šคํ† ์–ด ๋ชฉ๋ก ํŽ˜์ด์ง€๋กœ ์ด๋™
}

thymeleaf_diff

index.html์€ layout-member.html์„ ๋ ˆ์ด์•„์›ƒ์œผ๋กœ ์‚ฌ์šฉํ•˜๊ณ  ์žˆ๋Š”๋ฐ,

layout-member.js์—์„œ ์ ‘๊ทผํ•˜๋Š” memberCode๊ฐ€ undefined์šด๋ฐํ”ผ๋„ค๋กœ ์ฐํžˆ์ง€ ์•Š์œผ๋ ค๋ฉด, ํ›„์ž์™€ ๊ฐ™์ด ์Šคํฌ๋ฆฝํŠธ ๋กœ๋”ฉ ์ˆœ์„œ๋ฅผ ์กฐ์ •ํ•ด์•ผ ํ•œ๋‹ค.


๐Ÿ“Œ ์˜์™ธ๋กœ ๊ทผ๋ณธ ์žˆ๋Š” replaceChildโ€‹

๋†€๋ž๊ฒŒ๋„ ๋ ˆํผ๋Ÿฐ์Šค์— ์˜๊ฑฐํ•˜๋ฉด, replaceChild๋Š” (dest, src) ํ˜•์‹์˜ ๊ทผ๋ณธ์„ ๋”ฐ๋ฅด๊ณ  ์žˆ์—ˆ๋‹ค.

๋‹ค์‹œ ๋งํ•ด newNode๊ฐ€ 2๋ฒˆ์งธ ์ธ์ž๋กœ ๋“ค์–ด๊ฐ„๋‹ค๋Š” ๋œป์ด๋‹ค.

// ํ† ์ŠคํŠธ ๋ทฐ ์ปจํŠธ๋กค
function showAlarmToast(message, dateTime) {
// parent div (toast box)
const notiToastBoxDiv = document.getElementById('noti-toast-box');

// old div
const notiToastDataDiv = document.getElementById('noti-toast-data');
const notiToastDatetimeDiv = document.getElementById('noti-toast-datetime');

// new div
const newDataDiv = document.createElement("div");
newDataDiv.classList.add('noti-toast-data');
newDataDiv.appendChild(document.createTextNode(message));

const newDatetimeDiv = document.createElement("div");
newDatetimeDiv.classList.add('noti-toast-datetime');
newDatetimeDiv.appendChild(document.createTextNode(dateTime));

// div ๊ต์ฒด
notiToastBoxDiv.replaceChild(notiToastDataDiv, newDataDiv); // error
notiToastBoxDiv.replaceChild(notiToastDatetimeDiv, newDatetimeDiv); // error

// ํ† ์ŠคํŠธ ๋ฐ•์Šค ๋ณด์—ฌ์ฃผ๊ธฐ
notiToastBoxDiv.classList.add("active");

// 5์ดˆ ํ›„ ํ† ์ŠคํŠธ ๋ฐ•์Šค ์ˆจ๊ธฐ๊ธฐ
setTimeout(() =>{
notiToastBoxDiv.classList.remove("active");
}, 5000)
}

client_console


๊ทผ๋ฐ ๊ฒฐ๊ตญ์—” replaceWith๋ผ๋Š” ๋” ๋ชจ๋˜ํ•œ ๋ฐฉ์‹์ด ์žˆ์–ด์„œ ์ด๋ฅผ ์ฑ„ํƒํ–ˆ๋‹ค.

๊ทธ๋ฆฌ๊ณ  ์ƒˆ ๋…ธ๋“œ์— id ์ง€์ •ํ•˜๋Š”๊ฑฐ ๊นŒ๋จน์–ด์„œ (-> ์ตœ์ดˆ replace ์ดํ›„ getElementById ํ˜ธ์ถœ์‹œ ํ„ฐ์ง)

ํ•ด๋‹น ์ฝ”๋“œ๋„ ์ถ”๊ฐ€ํ–ˆ๋‹ค.

// ํ† ์ŠคํŠธ ๋ทฐ ์ปจํŠธ๋กค
function showAlarmToast(message, dateTime) {
console.log('show toast'); // logging
// parent div (toast box)
const notiToastBoxDiv = document.getElementById('noti-toast-box');

// old div
const notiToastDataDiv = document.getElementById('noti-toast-data');
const notiToastDatetimeDiv = document.getElementById('noti-toast-datetime');

// new div
const newDataDiv = document.createElement("div");
newDataDiv.classList.add('noti-toast-data');
newDataDiv.id = 'noti-toast-data';
newDataDiv.appendChild(document.createTextNode(message));

const newDatetimeDiv = document.createElement("div");
newDatetimeDiv.classList.add('noti-toast-datetime');
newDatetimeDiv.id = 'noti-toast-datetime';
newDatetimeDiv.appendChild(document.createTextNode(dateTime));

// div ๊ต์ฒด
notiToastDataDiv.replaceWith(newDataDiv);
notiToastDatetimeDiv.replaceWith(newDatetimeDiv);

// ํ† ์ŠคํŠธ ๋ฐ•์Šค ๋ณด์—ฌ์ฃผ๊ธฐ
notiToastBoxDiv.classList.add("active");

// 5์ดˆ ํ›„ ํ† ์ŠคํŠธ ๋ฐ•์Šค ์ˆจ๊ธฐ๊ธฐ
setTimeout(() =>{
console.log('hide toast'); // logging
notiToastBoxDiv.classList.remove("active");
}, 5000)
}

client_console


๐Ÿ“Œ ํƒ€์ž„๋ฆฌํ”„ ๋™์  class ์ถ”๊ฐ€โ€‹

ํ•ด๋‹น ๋ฐฉ๋ฒ•์œผ๋กœ ๊ฐ€๋Šฅํ•˜๋‹ค๊ณ  ํ•œ๋‹ค.


๐Ÿ“Œ ๋ˆˆ๋ฌผ๋‚˜๋Š” th border-radius ์ ์šฉํ•˜๊ธฐโ€‹

table์— ์ง€์ •ํ•œ border-collapse: collapse; ์†์„ฑ๊ณผ radius ์†์„ฑ์ด ์ถฉ๋Œํ•œ๋‹ค๋”๋ผ.

box-shadow์™€ ์„ ํƒ์ž ๋…ธ๊ฐ€๋‹ค๋กœ ๊ฒจ์šฐ ํ•ด๊ฒฐํ–ˆ๋Š”๋ฐ

๊ฒจ์šฐ ํ•ด๊ฒฐํ•œ๊ฒŒ ์Šคํƒ€์ผ๋ง์ด์–ด์„œ ํ˜„ํƒ€๊ฐ€ ์™”๋‹ค. CSS ํ‘œ์ค€์ด ๋งŽ์ด ๋ฐœ์ „ํ•˜๊ธธ ๋ฐ”๋žŒ...

โŒ ์‘ ์•ˆ๋จนํžˆ๋Š” ์ฝ”๋“œ
table {
width: 100%;
border-collapse: collapse;
border-top-left-radius: 6px;
border-top-right-radius: 6px;
margin-bottom: 48px;
}
โœ”๏ธ ๋จนํžˆ๋Š” ์ฝ”๋“œ
th {
font-family: "Pretendard", sans-serif;
color: #6C6C71;
background-color: #FCFCFC;
font-weight: 600;
font-size: 13px;
padding: 13px 16px;
text-align: left;
}
th:first-child {
border-top-left-radius: 6px;
box-shadow: 0 0 0 1px #E2E2E9;
}
th:last-child {
border-top-right-radius: 6px;
box-shadow: 0 0 0 1px #E2E2E9;
}
th:not(:last-child) {
border-right: 1px solid #E2E2E9;
}
th:not(:first-child):not(:last-child) {
box-shadow: 0 -1px 0 #E2E2E9, 0 1px 0 #E2E2E9;
}

table_result

ํ™•๋Œ€์ƒท