Dev
[DEV] react/typescript로 노노그램 만들어보기
쭘봉
2024. 3. 22. 20:16
노노그램 토이프로젝트는 기획이나 정리 없이 머릿 속 순서대로 하고 있다!
정신 없이 흘러가기도 한다.
뭔가 퍼즐을 풀고 만드는 것을 좋아하다보니
평소에 노노그램을 폰으로 즐겨한다.
매번 취업을 위한 과제나 알고리즘을 풀다가 지쳐버린 나
내마음대로 만드는 것을 하고 싶어져서 노노그램 게임을 만들어보기로함!
1. 설치
npx create-react-app nonogram --template typescript
2. 절대경로 설정하기
{
"compilerOptions": {
"baseUrl": "src",
},
"include": ["src"]
}
3. sass 설치
npm install sass
4. reset.css 받기
/* box-sizing 규칙을 명시합니다. */
*,
*::before,
*::after {
box-sizing: border-box;
}
/* 폰트 크기의 팽창을 방지합니다. */
html {
-moz-text-size-adjust: none;
-webkit-text-size-adjust: none;
text-size-adjust: none;
}
/* 기본 여백을 제거하여 작성된 CSS를 더 잘 제어할 수 있습니다. */
body,
h1,
h2,
h3,
h4,
p,
figure,
blockquote,
dl,
dd {
margin-block-end: 0;
}
/* list를 role값으로 갖는 ul, ol 요소의 기본 목록 스타일을 제거합니다. */
ul[role='list'],
ol[role='list'] {
list-style: none;
}
/* 핵심 body의 기본값을 설정합니다. */
body {
min-height: 100vh;
line-height: 1.5;
}
/* 제목 요소와 상호작용하는 요소에 대해 line-height를 더 짧게 설정합니다. */
h1,
h2,
h3,
h4,
button,
input,
label {
line-height: 1.1;
}
/* 제목에 대한 text-wrap을 balance로 설정합니다. */
h1,
h2,
h3,
h4 {
text-wrap: balance;
}
/* 클래스가 없는 기본 a 태그 요소는 기본 스타일을 가져옵니다. */
a:not([class]) {
text-decoration-skip-ink: auto;
color: currentColor;
}
/* 이미지 관련 작업을 더 쉽게 합니다. */
img,
picture {
max-width: 100%;
display: block;
}
/* input 및 button 항목들이 글꼴을 상속하도록 합니다. */
input,
button,
textarea,
select {
font: inherit;
}
/* 행 속성이 없는 textarea가 너무 작지 않도록 합니다. */
textarea:not([rows]) {
min-height: 10em;
}
/* 고정된 모든 항목에는 여분의 스크롤 여백이 있어야 합니다. */
:target {
scroll-margin-block: 5ex;
}
5. 폴더 정리 및 코드 정리
6. 본격전인 개발전에 10x10 기반에 노노그램 데이터를 만들어보자.
검색해보니까 완성본이 없어서 직접 풀어본다.. 예시 데이터로 좋을뜻!
데이터를 10x10 100개의 요소를 가진 2차원 배열로 만드는게 직관적이고 편해보인다.
const nonogram = [
[1, 1, 1, 1, 0, 0, 0, 0, 1, 1],
[1, 1, 1, 0, 0, 0, 0, 1, 1, 1],
[1, 0, 0, 0, 1, 1, 1, 1, 1, 1],
[0, 1, 0, 0, 1, 1, 1, 1, 1, 1],
[0, 0, 0, 0, 0, 0, 0, 1, 1, 1],
[1, 1, 0, 0, 0, 0, 0, 0, 1, 1],
[1, 1, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 1, 0, 1, 0, 0, 0, 0, 0, 0],
[1, 1, 1, 1, 1, 0, 0, 0, 0, 1],
[1, 1, 1, 0, 0, 0, 0, 0, 1, 1],
];
얼핏보면 토끼가 보...인다
7. 반복해서 표현해보자
import './nonogram.scss';
const nonogram = [
[1, 1, 1, 1, 0, 0, 0, 0, 1, 1],
[1, 1, 1, 0, 0, 0, 0, 1, 1, 1],
[1, 0, 0, 0, 1, 1, 1, 1, 1, 1],
[0, 1, 0, 0, 1, 1, 1, 1, 1, 1],
[0, 0, 0, 0, 0, 0, 0, 1, 1, 1],
[1, 1, 0, 0, 0, 0, 0, 0, 1, 1],
[1, 1, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 1, 0, 1, 0, 0, 0, 0, 0, 0],
[1, 1, 1, 1, 1, 0, 0, 0, 0, 1],
[1, 1, 1, 0, 0, 0, 0, 0, 1, 1],
];
function Nonogram() {
return (
<div>
{nonogram.map((y, indexY) => {
return (
<div key={indexY} className="row">
{y.map((x, indexX) => {
return (
<div
key={indexX}
className={`column ${x === 1 ? 'checked' : ''}`}></div>
);
})}
</div>
);
})}
</div>
);
}
export default Nonogram;
.row {
display: flex;
}
.column {
display: flex;
border: 1px solid gray;
width: 25px;
height: 25px;
&.checked {
background: black;
}
}
그럴싸하게 나오는군!
갑자기 사용해본적 없는 styled-components를 사용해보고 싶어졌다. 해보자
부모에서 nonogram data를 props로 넘겨주고 styled-components 사용한다면 이렇게 된다.
npm install styled-components
//App.tsx
<Nonogram nonogram={nonogram} />
//Nonogram.tsx
import React from 'react';
import styled from 'styled-components';
interface ColumnProps {
checked: boolean;
}
const Row = styled.div`
display: flex;
`;
const Column = styled.div<ColumnProps>`
display: flex;
border: 1px solid gray;
width: 25px;
height: 25px;
background: ${(props) => (props.checked ? 'black' : 'none')};
`;
type NonogramArray = number[][];
interface NonogramProps {
nonogram: NonogramArray;
}
const Nonogram: React.FC<NonogramProps> = ({ nonogram }) => {
return (
<div>
{nonogram.map((y, indexY) => (
<Row key={indexY}>
{y.map((x, indexX) => (
<Column key={indexX} checked={x === 1} />
))}
</Row>
))}
</div>
);
};
export default Nonogram;
다음 회차에는 게임 기본 기능을 만들어본다~