프리온보딩 권장 기술인 Javascript와 React를 중심으로 개발하였습니다. ES6+ 문법을 최대한 활용하여 개발하고자 노력했습니다. Class Component가 아닌 Functional Component를 이용하였고 Hooks 을 활용하여 개발을 하였습니다.
web Accessability 표준에 맞도록 시멘틱 엘레멘트를 지킬려 노력하였습니다. Cross Browser Check(Safari, Microsoft Edge, Firefox, Chrome)를 하였으며, 모바일 환경에서도 이용할 수 있는 반응형 웹페이지를 만들기 위해 노력했습니다.
구현과제의 양이 많지 않고 컴포넌트 구조가 복잡하지 않아 Tailwind CSS를 활용하였습니다. CSS 속성은 Mozilla에서 제안한 CSS 속성 기술 순서를 참고하여 정리하였습니다.
읽기 쉬운 코드를 위해 code Formatter 역할을 해주는 Prettier와 ESLint를 이용하여 표준에 맞는 코드컨벤션을 지키려 노력하였습니다.
구현과제 Demo 확인을 위한 Headless CMS인 Vercel을 이용하여 배포하였습니다.
-
구현한 방법과 이유
input 태그의 Type을 Checkbox로 하여 개발하였습니다. screen reader에도 잘 동작할 수 있도록 시멘틱 엘레먼트를 준수하였습니다. label 태그 내 input를 넣어줌으로서 label을 클릭하였을 떄도 동일한 기능이 동작하도록 하였습니다.
Toggle은 외부의 Toggle Box와 내부의 Toggle Slider로 구분했습니다. Tailwind의 JIT(just in time) mode active를 활용하여 Toggle Box가 클릭되었을때 움직이고 Toggle Slider의 배경색이 바꾸도록 하였습니다.
이때 Toggle Switch 가 "ON", "OFF" 인지 확인하기 위해 State Hook과 Conditional(Ternary) Operator를 사용하였습니다.
-
구현하면서 어려웠던점과 해결
라이브러리로 사용하던 Toggle 버튼을 Pure CSS로 구현하는게 어려웠습니다. useRef 등을 활용하여 toggle의 clickEvent를 감지하여 DOM 객체의 CSS element를 바꾸는 방법보단 Pure CSS로 개발하는 것이 더욱 직관적이라고 생각하여 Tailwind의 peer's class를 사용했습니다.
label 태그가 toggle 버튼을 누를때마다 width 값이 계속 미세하게 변하여 좌우로 왔다갔다 합니다. label 태그가 차지하는 width를 고정하는 방법을 찾아보았고, 자식 요소의 width를 rem이 아닌 px값으로 두고 flex를 활용한 레이아웃으 해결하였습니다.
-
구현한 방법과 이유
State Hook을 활용하여 Open Modal Button의 클릭유무를 확인하였고 Conditional(Ternary) Operator를 통해 아래와 같은 로직으로 코드가 작동하도록 구현했습니다.
Open Modal button이 클릭됨 ? modal창 열림 : null
-
구현하면서 어려웠던점과 해결
modal의 위치를 컴포넌트 가운데가 아닌 화면 가운데 두는 것이 어려웠습니다. 이를 해결하고자 CSS property 중 inset을 활용하여 부모 요소의 가운데 오도록 레이아웃을 만들었습니다.
반응형으로 구현시 modal 밖으로 content가 튀어나오는걸 막는게 어려웠습니다. 이를 CSS property 중 overflow를 통해 해결하였습니다.
-
구현한 방법과 이유
Tab 목록을 객체로 매핑해주었습니다. 코드의 반복되는 list 를 Array.prototype.map()을 통해 표현해주었습니다. 각 list에는 ClickEventHandler()를 두어 어떤 탭이 눌렸는지 구분하였습니다. 이때 id값은 State Hook을 통해 관리하습니다. 컴포넌트가 바뀌어야 할 부분에 객체의 key값으로 객체의 value를 가져오는 형태로 만들었습니다.
-
구현하면서 어려웠던점과 해결
반복되는 li 태그를 어떻게 리팩토링 할지 고민하였고 Array.prototype.map()을 사용하였습니다. 탭 클릭 시 탭 배경의 색을 변경해주기 위해 :active를 사용해주었습니다. 문제는 클릭한 순간만 색이 바뀌어서 State Hook을 이용하여 클릭된 탭의 상태를 관리하였습니다.
-
구현한 방법과 이유
State Hook을 통해 사용자가 입력한 내역을 관리합니다. 이때 사용자에게 보여지는 tag와 저장한 상태를 매핑해주는 방식으로 구현하였습니다.
태그를 저장할때는 Spread syntax(...) 통해 항상 마지막 위치에 저장되도록 하였습니다. 태그 최대 저장 갯수는 5개이며 중복을 허용하지 않도록 구현 하였습니다.
태그를 삭제할때는 Array.prototype.filter()를 사용하여 사용자가 제거하려는 태그의 인덱스와 일치하는 태그를 찾도록 했습니다. 발견되면 필터링되고 나머지 태그는 Spread syntax(...)를 이용하여 다시 태그 상태를 업데이트합니다.
-
구현하면서 어려웠던점과 해결
input focus 외부 div 태그의 border color가 바뀌지 않아 어려움을 겪었습니다. 그래서 input에 outline을 없애고 input과 태그를 모두 감싸고 있는 div 태그에 :focus-within을 활용하여 border가 변경되도록 구현하였습니다.
태그 작성 시 입력값의 길이가 길어질 경우 사용자 경험을 해치는 문제가 있어 말줄임을 활용하여 입력값이 길 경우 생략하도록 구현하였습니다. 또한 태그가 너무 많아지는 것을 방지하기 위해 태그 갯수의 제한을 두었습니다.
- 구현한 방법과 이유
자동완성이 될 내역의 Data는 배열에 담아 놓았습니다. Effect Hook을 이용하여 input값을 입력하였을때 입력된 값을 포함하는 내용만 Array.prototype.filter()통해 필터링 된 상태로 변경한다. 그럼 해당 상태에 대해 Array.prototype.map()을 통해 자동 완성 리스트를 화면에 출력해주었습니다.
이때 웹 접근성을 높이기 위해 키보드 'ArrowUp','ArrowDown' 버튼으로 내역의 리스트를 고를 수 있게 하였습니다. 이를 위해 State Hook으로 현재 자동완성 리스트의 index의 상태를 관리하여 자동완성 리스트의 key값과 index 값이 같은 경우 input에 value에 저장되도록 하였습니다.
- 구현하면서 어려웠던점과 해결
해당 문제의 구현이 가장 오래걸렸다. 실력이 부족하여 인터넷에 많이 구글링하였습니다. Array.prototype.filter(), Array.prototype.map()을 잘 이용해야 했고, Effect Hook를 적절하게 이용해 input value에 대한 상태를 관찰하며 필터링을 해주는 것이 중요했습니다.
또한 modal과 비슷하게 input value가 있을때와 없을 때, 자동완성 리스트에 값이 있을때와 없을 때를 비교해주며 창을 띄워줘야 어려웠습니다.
input을 클릭하였을 때 하단만 border-radius가 사라지고 자동완성 리스트가 붙으면서 리스트의 하단에 border-radius가 나타나도록 구현하는 부분이 어려웠습니다. 태그의 상속을 보면 ul > li
와 같이 되어 있는데 ul
에 라운드 처리를 해줘도 li
에는 라운드가 지지 않는 문제점이 있었고 :last-child를 활용하여 해당 문제를 해결해주었습니다.
- 구현한 방법과 이유
변경하고자 하는 요소에 DoubleClickEvent가 발생하면 State Hook을 통해 해당 요소의 수정가능 상태를 관리합니다. 그리고 Conditional(Ternary) Operator를 통해 값에 따라 display되는 요소를 달리하였습니다.
나이가 0살 이하로 입력되는 것은 불가능하기 때문에 음수제한과 이름 입력 최대글자수를 10글자로 제한하였습니다.
요소가 수정가능 상태값이 true라면 ? input 요소가 보인다 : 요소의 값이 보인다.
- 구현하면서 어려웠던점과 해결
- DoubleClickEvent가 존재 하는지 처음 알았습니다. 그래서 이를 해결하기 위해 구글링을 하여 해당 문제를 해결하였습니다.
$ git clone https://github.com/hhhminmewanted_pre_onboarding.git
$ cd custom-component
$ npm install
$ npm start
├── README.md
└── custom-component
├── README.md
├── package-lock.json
├── package.json
├── postcss.config.js
├── public
│ ├── favicon.ico
│ ├── index.html
│ ├── logo192.png
│ ├── logo512.png
│ ├── manifest.json
│ └── robots.txt
├── src
│ ├── App.js
│ ├── App.test.js
│ ├── component
│ ├── index.css
│ ├── index.js
│ ├── reportWebVitals.js
│ └── setupTests.js
└── tailwind.config.js
<타입> 리스트
- feat : 기능 (새로운 기능)
- fix : 버그 (버그 수정)
- refactor : 리팩토링
- style : 스타일 (코드 형식. 세미콜론 추가 : 비즈니스 로직에 변경 없음)
- docs : 문서 (문서 추가, 수정, 삭제)
- test : 테스트 (테스트 코드 추가, 수정, 삭제 : 비즈니스 로직에 변경 없)
- chore : 기타 변경사항, 셋팅 ㅗ간련 (빌드 스크립트 수정 등)
커밋 시 아래의 규칙을 지켰습니다.
- 제목은 명령문으로 한다.
- 제목 끝에 마침표(.)를 쓰지 않는다.
- 제목과 본문을 한 줄 띄워 분리한다.
- 본문은 “어떻게" 보다 “무엇을", “왜" 설명한다.
- 본문에 여러줄의 메시지를 작성할 땐 “-”로 구분한다.
Main branch
- 배포 가능한 상태만을 관리하는 브랜치입니다. Vercel과 배포 파이프라인이 연결되어 바로 배포된 도메인으로 확인할 수 있게 하였습니다.
- 새로운 Feature가 추가되었을때 1.0 → 2.0 과 같이 정수 버전을 릴리즈 했습니다.
- 새로운 기능이 추가되지 않고 버그 및 디자인 수정의 경우 1.0 → 1.1 과 같이 소수점 버전을 릴리즈 했습니다.
Develop branch
- 다음에 배포할 것을 개발하는 브랜치입니다.
- develop 브랜치는 통합 브랜치의 역할을 하며, 평소에는 이 브랜치를 기반으로 개발을 진행합니다.
Feature branch
- 기능을 개발하는 브랜치로, develop 브랜치로부터 분기합니다.
- feature 브랜치는 그 기능을 다 완성할때까지 유지하고, 다 완성되면 Develop 브랜치로 merge합니다.
- merge 후 배포가 완료된 브랜치는 바로 삭제 하고 있습니다.