React Custom Components 과제를 통해 Modal
, Tab
, Tag
, Toggle
을 만들어 보았다.
결과는 storybook을 통해 확인하면서 작업했다.
import { useState, useRef } from "react";
import styled from "styled-components";
export const ModalContainer = styled.div`
// TODO : Modal을 구현하는데 전체적으로 필요한 CSS를 구현합니다.
width: 100vw;
height: 100vh;
display: flex;
justify-content: center;
align-items: center;
border: 1px solid black;
position: relative;
`;
export const ModalBackdrop = styled.div`
// TODO : Modal이 떴을 때의 배경을 깔아주는 CSS를 구현합니다.
position: fixed;
width: 100vw;
height: 100vh;
background-color: #d9d9d9;
display: flex;
justify-content: center;
align-items: center;
`;
export const ModalBtn = styled.button`
background-color: var(--coz-purple-600);
text-decoration: none;
border: none;
padding: 20px;
color: white;
border-radius: 30px;
cursor: grab;
`;
export const ModalView = styled.div.attrs((props) => ({
// attrs 메소드를 이용해서 아래와 같이 div 엘리먼트에 속성을 추가할 수 있습니다.
role: "dialog",
}))`
// TODO : Modal창 CSS를 구현합니다.
display: flex;
justify-content: center;
align-items: center;
width: 400px;
height: 200px;
background-color: white;
position: relative;
> button {
position: absolute;
top: 2px;
right: 7px;
cursor: pointer;
}
`;
export const Modal = () => {
const [isOpen, setIsOpen] = useState(false);
const modalRef = useRef(null);
const openModalHandler = () => {
// TODO : isOpen의 상태를 변경하는 메소드를 구현합니다.
setIsOpen(!isOpen);
};
const modalSideClick = (e) => {
if (modalRef.current === e.target) {
setIsOpen(!isOpen);
}
};
return (
<>
<ModalContainer>
{/* TODO : 조건부 렌더링을 활용해서 Modal이 열린 상태(isOpen이 true인 상태)일 때만 모달창과 배경이 뜰 수 있게 구현해야 합니다. */}
<ModalBtn onClick={openModalHandler}>
{isOpen ? "Opened!" : "Open Modal"}{" "}
</ModalBtn>
{isOpen ? (
<ModalBackdrop ref={modalRef} onClick={modalSideClick}>
<ModalView>
<button onClick={openModalHandler}>×</button>
<div>HELLO CODESTATES!</div>
</ModalView>
</ModalBackdrop>
) : null}
</ModalContainer>
</>
);
};
기본적인 css와 로직 코드가 작성되어 있었다.
- Modal 창을 띄워 줄 Open Modal 버튼이 있어야 합니다.
- 버튼을 클릭하면 Modal 컴포넌트 내부에 Modal배경, Modal창 div 엘리먼트가 렌더링되어야 합니다.
- Modal창이 렌더링 된 상태에서 버튼을 다시 클릭하면 Modal배경, Modal창 div 엘리먼트가 사라집니다.
- Modal 창 밖을 클릭하면, Modal배경, Modal창 div 엘리먼트가 사라집니다.
이 4개의 테스트를 통과해야했다.
다른 부분은 완성을 했지만, 3번을 계속 통과하지 못했다.
처음 작성한 코드이다.
{isOpen ? (
<ModalBackdrop ref={modalRef} onClick={modalSideClick}>
<ModalView>
<button onClick={openModalHandler}>×</button>
<div>HELLO CODESTATES!</div>
</ModalView>
</ModalBackdrop>
) : (
<ModalBtn onClick={openModalHandler}>
Opened!
</ModalBtn>
)}
처음에는 useRef
를 잘못 사용했나 했지만, 그것은 아니였다.
import { useState } from "react";
import styled from "styled-components";
// TODO: Styled-Component 라이브러리를 활용해 TabMenu 와 Desc 컴포넌트의 CSS를 구현합니다.
const TabMenu = styled.ul`
background-color: #dcdcdc;
color: rgba(73, 73, 73, 0.5);
font-weight: bold;
display: flex;
flex-direction: row;
justify-items: center;
align-items: center;
list-style: none;
margin-bottom: 7rem;
cursor: pointer;
.submenu {
${"" /* 기본 Tabmenu 에 대한 CSS를 구현합니다. */}
width: 33vw;
}
.focused {
${"" /* 선택된 Tabmenu 에만 적용되는 CSS를 구현합니다. */}
background-color: rgba(52,12,196,0.91);
color: #ffffff;
}
& div.desc {
text-align: center;
}
`;
const Desc = styled.div`
text-align: center;
`;
export const Tab = () => {
// TIP: Tab Menu 중 현재 어떤 Tab이 선택되어 있는지 확인하기 위한
// currentTab 상태와 currentTab을 갱신하는 함수가 존재해야 하고, 초기값은 0 입니다.
const [currentTab, setCurrentTab] = useState(0);
const menuArr = [
{ name: "Tab1", content: "Tab menu ONE" },
{ name: "Tab2", content: "Tab menu TWO" },
{ name: "Tab3", content: "Tab menu THREE" },
];
const selectMenuHandler = (index) => {
// TIP: parameter로 현재 선택한 인덱스 값을 전달해야 하며, 이벤트 객체(event)는 쓰지 않습니다
// TODO : 해당 함수가 실행되면 현재 선택된 Tab Menu 가 갱신되도록 함수를 완성하세요.
setCurrentTab(index);
return index;
};
return (
<>
<div>
<TabMenu>
{/*TODO: 아래 하드코딩된 내용 대신에, map을 이용한 반복으로 코드를 수정합니다.*/}
{/*TIP: li 엘리먼트의 class명의 경우 선택된 tab 은 'submenu focused' 가 되며,
나머지 2개의 tab은 'submenu' 가 됩니다.*/}
{menuArr.map((val, index) => (
<li
className={`${
index === currentTab ? "submenu focused" : "submenu"
}`}
onClick={() => selectMenuHandler(index)}
>
{val.name}
</li>
))}
</TabMenu>
<Desc>
{/*TODO: 아래 하드코딩된 내용 대신에, 현재 선택된 메뉴 따른 content를 표시하세요*/}
<p>{menuArr[currentTab].content}</p>
</Desc>
</div>
</>
);
};
map
함수와 삼항연산자를 이용해서 마우스로 선택한 index
값과 같은 메뉴에는 submenu focused
라는 class명을, 아닌 것은 submenu
가 된다.
또한 menuArr[0].content
로 하드코딩된 내용을 menuArr[currentTab].content
로 바꿔주었다.
import { useState } from "react";
import styled from "styled-components";
// TODO: Styled-Component 라이브러리를 활용해 여러분만의 tag 를 자유롭게 꾸며 보세요!
export const TagsInput = styled.div`
margin: 8rem auto;
display: flex;
align-items: flex-start;
flex-wrap: wrap;
min-height: 48px;
width: 480px;
padding: 0 8px;
border: 1px solid rgb(214, 216, 218);
border-radius: 6px;
> ul {
display: flex;
flex-wrap: wrap;
padding: 0;
margin: 8px 0 0 0;
> .tag {
width: auto;
height: 32px;
display: flex;
align-items: center;
justify-content: center;
color: #fff;
padding: 0 8px;
font-size: 14px;
list-style: none;
border-radius: 6px;
margin: 0 8px 8px 0;
background: var(--coz-purple-600);
> .tag-close-icon {
display: block;
width: 16px;
height: 16px;
line-height: 16px;
text-align: center;
font-size: 14px;
margin-left: 8px;
color: var(--coz-purple-600);
border-radius: 50%;
background: #fff;
cursor: pointer;
}
}
}
> input {
flex: 1;
border: none;
height: 46px;
font-size: 14px;
padding: 4px 0 0 0;
:focus {
outline: transparent;
}
}
&:focus-within {
border: 1px solid var(--coz-purple-600);
}
`;
export const Tag = () => {
const initialTags = ["CodeStates", "kimcoding"];
const [tags, setTags] = useState(initialTags);
const removeTags = (indexToRemove) => {
// TODO : 태그를 삭제하는 메소드를 완성하세요.
setTags(tags.filter((val, index) => index !== indexToRemove));
};
const addTags = (e) => {
// TODO : tags 배열에 새로운 태그를 추가하는 메소드를 완성하세요.
// 이 메소드는 태그 추가 외에도 아래 3 가지 기능을 수행할 수 있어야 합니다.
// - 이미 입력되어 있는 태그인지 검사하여 이미 있는 태그라면 추가하지 말기
// - 아무것도 입력하지 않은 채 Enter 키 입력시 메소드 실행하지 말기
// - 태그가 추가되면 input 창 비우기
if (e.target.value !== "") {
const filter = tags.includes(e.target.value);
if (!filter) {
setTags([...tags, e.target.value]);
console.log(tags);
}
}
};
return (
<>
<TagsInput>
<ul id="tags">
{tags.map((tag, index) => (
<li key={index} className="tag">
<span className="tag-title">{tag}</span>
<span
className="tag-close-icon"
onClick={() => removeTags(index)}
>
{/* TODO : tag-close-icon이 tag-title 오른쪽에 x 로 표시되도록 하고,
삭제 아이콘을 click 했을 때 removeTags 메소드가 실행되어야 합니다. */}
X
</span>
</li>
))}
</ul>
<input
className="tag-input"
type="text"
onKeyUp={(e) => {
/* 키보드의 Enter 키에 의해 addTags 메소드가 실행되어야 합니다. */
if (e.key === "Enter") {
addTags(e);
e.target.value = "";
}
}}
placeholder="Press enter to add tags"
/>
</TagsInput>
</>
);
};
태그를 삭제하는 remove
함수는 filter
를 통해 배열 tags
에 있는 index
값과 삭제하고 하는 항목의 index
값을 비교하여 태그를 삭제한다.
태그를 추가하는 addTags
함수도 filter
로 배열 tags
에 같은 값이 있는지 e.target.value
와 비교하고 같은값이 없다면 태그를 추가해준다.
import { useState } from "react";
import styled from "styled-components";
const ToggleContainer = styled.div`
position: relative;
margin-top: 8rem;
left: 47%;
cursor: pointer;
> .toggle-container {
width: 50px;
height: 24px;
border-radius: 30px;
background-color: #8b8b8b;
// TODO : .toggle--checked 클래스가 활성화 되었을 경우의 CSS를 구현합니다.
&.toggle--checked {
background-color: #5c305a;
}
}
> .toggle-circle {
position: absolute;
top: 1px;
left: 1px;
width: 22px;
height: 22px;
border-radius: 50%;
background-color: #ffffff;
transition-duration: 1s;
// TODO : .toggle--checked 클래스가 활성화 되었을 경우의 CSS를 구현합니다.
&.toggle--checked {
left: 27px;
background-color: #ffffff;
transition-duration: 1s;
}
}
`;
const Desc = styled.div`
// TODO : 설명 부분의 CSS를 구현합니다.
`;
export const Toggle = () => {
const [isOn, setisOn] = useState(false);
const toggleHandler = () => {
// TODO : isOn의 상태를 변경하는 메소드를 구현합니다.
setisOn(!isOn);
};
return (
<>
<ToggleContainer
// TODO : 클릭하면 토글이 켜진 상태(isOn)를 boolean 타입으로 변경하는 메소드가 실행되어야 합니다.
onClick={toggleHandler}
>
{/* TODO : 아래에 div 엘리먼트 2개가 있습니다. 각각의 클래스를 'toggle-container', 'toggle-circle' 로 지정하세요. */}
{/* TIP : Toggle Switch가 ON인 상태일 경우에만 toggle--checked 클래스를 div 엘리먼트 2개에 모두 추가합니다. 조건부 스타일링을 활용하세요. */}
<div className={`toggle-container ${isOn ? `toggle--checked` : ""}`} />
<div className={`toggle-circle ${isOn ? `toggle--checked` : ""}`} />
</ToggleContainer>
{/* TODO : Desc 컴포넌트를 활용해야 합니다. */}
{/* TIP: Toggle Switch가 ON인 상태일 경우에 Desc 컴포넌트 내부의 텍스트를 'Toggle Switch ON'으로, 그렇지 않은 경우 'Toggle Switch OFF'가 됩니다. 조건부 렌더링을 활용하세요. */}
{isOn ? "Toggle Switch On" : "Toggle Switch Off"}
</>
);
};
div
엘리먼트에서 toggle의 상태를 확인하고 켜진 상태이면 toggle--checked
라는 class명을 추가해준다.
styled.component
에서 toggle-checked
부분을 따로 작성해서, toggle-checked
가 있으면 toggle-circle
의 위치를 조정해준다.