From 1fec91fd24625747dfa9aed87f003ac46e5de935 Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 13:50:04 +0900 Subject: [PATCH 01/20] =?UTF-8?q?chore(Sprint9):=20=EC=82=AC=EC=9A=A9?= =?UTF-8?q?=ED=95=98=EC=A7=80=20=EC=95=8A=EB=8A=94=20=EC=B4=88=EA=B8=B0=20?= =?UTF-8?q?=ED=8C=8C=EC=9D=BC=20=EB=B0=8F=20=EC=BD=94=EB=93=9C=20=EC=82=AD?= =?UTF-8?q?=EC=A0=9C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/pages/_document.js | 2 +- my-app/pages/index.js | 111 ++-------------------- my-app/public/favicon.ico | Bin 25931 -> 0 bytes my-app/public/file.svg | 1 - my-app/public/globe.svg | 1 - my-app/public/next.svg | 1 - my-app/public/vercel.svg | 1 - my-app/public/window.svg | 1 - my-app/styles/Home.module.css | 168 ---------------------------------- my-app/styles/globals.css | 38 -------- 10 files changed, 9 insertions(+), 315 deletions(-) delete mode 100644 my-app/public/favicon.ico delete mode 100644 my-app/public/file.svg delete mode 100644 my-app/public/globe.svg delete mode 100644 my-app/public/next.svg delete mode 100644 my-app/public/vercel.svg delete mode 100644 my-app/public/window.svg delete mode 100644 my-app/styles/Home.module.css diff --git a/my-app/pages/_document.js b/my-app/pages/_document.js index b2fff8b4..ffc3f3cc 100644 --- a/my-app/pages/_document.js +++ b/my-app/pages/_document.js @@ -1,4 +1,4 @@ -import { Html, Head, Main, NextScript } from "next/document"; +import { Head, Html, Main, NextScript } from "next/document"; export default function Document() { return ( diff --git a/my-app/pages/index.js b/my-app/pages/index.js index d3f2c809..78e87d45 100644 --- a/my-app/pages/index.js +++ b/my-app/pages/index.js @@ -1,116 +1,21 @@ import Head from "next/head"; -import Image from "next/image"; -import { Geist, Geist_Mono } from "next/font/google"; -import styles from "@/styles/Home.module.css"; - -const geistSans = Geist({ - variable: "--font-geist-sans", - subsets: ["latin"], -}); - -const geistMono = Geist_Mono({ - variable: "--font-geist-mono", - subsets: ["latin"], -}); export default function Home() { return ( <> - Create Next App - + Do It + -
-
- Next.js logo -
    -
  1. - Get started by editing pages/index.js. -
  2. -
  3. Save and see your changes instantly.
  4. -
- -
- - Vercel logomark - Deploy now - - - Read our docs - -
+
+
+

Welcome to Do It

-
); diff --git a/my-app/public/favicon.ico b/my-app/public/favicon.ico deleted file mode 100644 index 718d6fea4835ec2d246af9800eddb7ffb276240c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 25931 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 LrJugUA?W`A8`#=m diff --git a/my-app/public/file.svg b/my-app/public/file.svg deleted file mode 100644 index 004145cd..00000000 --- a/my-app/public/file.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/my-app/public/globe.svg b/my-app/public/globe.svg deleted file mode 100644 index 567f17b0..00000000 --- a/my-app/public/globe.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/my-app/public/next.svg b/my-app/public/next.svg deleted file mode 100644 index 5174b28c..00000000 --- a/my-app/public/next.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/my-app/public/vercel.svg b/my-app/public/vercel.svg deleted file mode 100644 index 77053960..00000000 --- a/my-app/public/vercel.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/my-app/public/window.svg b/my-app/public/window.svg deleted file mode 100644 index b2b2a44f..00000000 --- a/my-app/public/window.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/my-app/styles/Home.module.css b/my-app/styles/Home.module.css deleted file mode 100644 index a11c8f31..00000000 --- a/my-app/styles/Home.module.css +++ /dev/null @@ -1,168 +0,0 @@ -.page { - --gray-rgb: 0, 0, 0; - --gray-alpha-200: rgba(var(--gray-rgb), 0.08); - --gray-alpha-100: rgba(var(--gray-rgb), 0.05); - - --button-primary-hover: #383838; - --button-secondary-hover: #f2f2f2; - - display: grid; - grid-template-rows: 20px 1fr 20px; - align-items: center; - justify-items: center; - min-height: 100svh; - padding: 80px; - gap: 64px; - font-family: var(--font-geist-sans); -} - -@media (prefers-color-scheme: dark) { - .page { - --gray-rgb: 255, 255, 255; - --gray-alpha-200: rgba(var(--gray-rgb), 0.145); - --gray-alpha-100: rgba(var(--gray-rgb), 0.06); - - --button-primary-hover: #ccc; - --button-secondary-hover: #1a1a1a; - } -} - -.main { - display: flex; - flex-direction: column; - gap: 32px; - grid-row-start: 2; -} - -.main ol { - font-family: var(--font-geist-mono); - padding-left: 0; - margin: 0; - font-size: 14px; - line-height: 24px; - letter-spacing: -0.01em; - list-style-position: inside; -} - -.main li:not(:last-of-type) { - margin-bottom: 8px; -} - -.main code { - font-family: inherit; - background: var(--gray-alpha-100); - padding: 2px 4px; - border-radius: 4px; - font-weight: 600; -} - -.ctas { - display: flex; - gap: 16px; -} - -.ctas a { - appearance: none; - border-radius: 128px; - height: 48px; - padding: 0 20px; - border: none; - border: 1px solid transparent; - transition: - background 0.2s, - color 0.2s, - border-color 0.2s; - cursor: pointer; - display: flex; - align-items: center; - justify-content: center; - font-size: 16px; - line-height: 20px; - font-weight: 500; -} - -a.primary { - background: var(--foreground); - color: var(--background); - gap: 8px; -} - -a.secondary { - border-color: var(--gray-alpha-200); - min-width: 158px; -} - -.footer { - grid-row-start: 3; - display: flex; - gap: 24px; -} - -.footer a { - display: flex; - align-items: center; - gap: 8px; -} - -.footer img { - flex-shrink: 0; -} - -/* Enable hover only on non-touch devices */ -@media (hover: hover) and (pointer: fine) { - a.primary:hover { - background: var(--button-primary-hover); - border-color: transparent; - } - - a.secondary:hover { - background: var(--button-secondary-hover); - border-color: transparent; - } - - .footer a:hover { - text-decoration: underline; - text-underline-offset: 4px; - } -} - -@media (max-width: 600px) { - .page { - padding: 32px; - padding-bottom: 80px; - } - - .main { - align-items: center; - } - - .main ol { - text-align: center; - } - - .ctas { - flex-direction: column; - } - - .ctas a { - font-size: 14px; - height: 40px; - padding: 0 16px; - } - - a.secondary { - min-width: auto; - } - - .footer { - flex-wrap: wrap; - align-items: center; - justify-content: center; - } -} - -@media (prefers-color-scheme: dark) { - .logo { - filter: invert(); - } -} diff --git a/my-app/styles/globals.css b/my-app/styles/globals.css index e3734be1..32704170 100644 --- a/my-app/styles/globals.css +++ b/my-app/styles/globals.css @@ -2,41 +2,3 @@ --background: #ffffff; --foreground: #171717; } - -@media (prefers-color-scheme: dark) { - :root { - --background: #0a0a0a; - --foreground: #ededed; - } -} - -html, -body { - max-width: 100vw; - overflow-x: hidden; -} - -body { - color: var(--foreground); - background: var(--background); - font-family: Arial, Helvetica, sans-serif; - -webkit-font-smoothing: antialiased; - -moz-osx-font-smoothing: grayscale; -} - -* { - box-sizing: border-box; - padding: 0; - margin: 0; -} - -a { - color: inherit; - text-decoration: none; -} - -@media (prefers-color-scheme: dark) { - html { - color-scheme: dark; - } -} From c50f6f767b3b8eca4654354d51705e1ac9a79ffc Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 15:21:08 +0900 Subject: [PATCH 02/20] =?UTF-8?q?feat(Sprint9):=20Color=20component=20?= =?UTF-8?q?=EA=B5=AC=ED=98=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/components/color/color.js | 30 ++++++++++++++++++++++++++++++ my-app/styles/globals.css | 14 ++++++++++++-- 2 files changed, 42 insertions(+), 2 deletions(-) create mode 100644 my-app/components/color/color.js diff --git a/my-app/components/color/color.js b/my-app/components/color/color.js new file mode 100644 index 00000000..b198fb49 --- /dev/null +++ b/my-app/components/color/color.js @@ -0,0 +1,30 @@ +function colorVariable(name, shade) { + return `var(--color-${name}-${Math.max(100, Math.min(900, shade))})`; +} + +const Colors = { + state: { + 100: colorVariable("state", 100), + 200: colorVariable("state", 200), + 300: colorVariable("state", 300), + 400: colorVariable("state", 400), + 500: colorVariable("state", 500), + 800: colorVariable("state", 800), + 900: colorVariable("state", 900), + }, + violet: { + 100: colorVariable("violet", 100), + 600: colorVariable("violet", 600), + }, + rose: { + 500: colorVariable("rose", 500), + }, + lime: { + 300: colorVariable("lime", 300), + }, + amber: { + 800: colorVariable("amber", 800), + }, +}; + +export default Colors; diff --git a/my-app/styles/globals.css b/my-app/styles/globals.css index 32704170..b4d32ea3 100644 --- a/my-app/styles/globals.css +++ b/my-app/styles/globals.css @@ -1,4 +1,14 @@ :root { - --background: #ffffff; - --foreground: #171717; + --color-state-100: #f1f5f9; + --color-state-200: #e2e8f0; + --color-state-300: #cbd5e1; + --color-state-400: #94a3b8; + --color-state-500: #64748b; + --color-state-800: #1e293b; + --color-state-900: #0f172a; + --color-violet-100: #ede9fe; + --color-violet-600: #7c3aed; + --color-rose-500: #f43f5e; + --color-lime-300: #bef264; + --color-amber-800: #92400e; } From fba3252db627081cca983003b75964753536f331 Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 16:01:14 +0900 Subject: [PATCH 03/20] =?UTF-8?q?feat(Sprint9):=20SearchInput=20component?= =?UTF-8?q?=20=EA=B5=AC=ED=98=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/components/input/search-input.jsx | 11 ++++++++++ .../components/input/search-input.module.css | 21 +++++++++++++++++++ my-app/styles/globals.css | 4 ++++ 3 files changed, 36 insertions(+) create mode 100644 my-app/components/input/search-input.jsx create mode 100644 my-app/components/input/search-input.module.css diff --git a/my-app/components/input/search-input.jsx b/my-app/components/input/search-input.jsx new file mode 100644 index 00000000..49c7eae9 --- /dev/null +++ b/my-app/components/input/search-input.jsx @@ -0,0 +1,11 @@ +import styles from "./search-input.module.css"; + +function SearchInput({ ...props }) { + return ( +
+ +
+ ); +} + +export default SearchInput; diff --git a/my-app/components/input/search-input.module.css b/my-app/components/input/search-input.module.css new file mode 100644 index 00000000..0aacd1bc --- /dev/null +++ b/my-app/components/input/search-input.module.css @@ -0,0 +1,21 @@ +.searchInput { + background-color: var(--color-state-100); + padding: 16px 24px; + border: 2px solid var(--color-state-900); + border-radius: 30px; + box-shadow: 4px 4px 0px var(--color-state-900); +} +.searchInput input { + background: none; + border: none; + outline: none; + width: 100%; + font-size: 16px; + font-weight: 400; + color: var(--color-state-900); +} +.searchInput input::placeholder { + font-size: 16px; + font-weight: 400; + color: var(--color-state-500); +} diff --git a/my-app/styles/globals.css b/my-app/styles/globals.css index b4d32ea3..61f32570 100644 --- a/my-app/styles/globals.css +++ b/my-app/styles/globals.css @@ -12,3 +12,7 @@ --color-lime-300: #bef264; --color-amber-800: #92400e; } + +* { + box-sizing: border-box; +} From 5b8c8e417dd9aad52f58a66a8841ae6edb3f6319 Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 16:37:58 +0900 Subject: [PATCH 04/20] =?UTF-8?q?feat(Sprint9):=20Button=20component=20?= =?UTF-8?q?=EA=B5=AC=ED=98=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/components/button/button-type.js | 7 ++++ my-app/components/button/button.jsx | 27 ++++++++++++++ my-app/components/button/button.module.css | 41 ++++++++++++++++++++++ my-app/public/icons/ic-check.svg | 3 ++ my-app/public/icons/ic-plus-white.svg | 4 +++ my-app/public/icons/ic-plus.svg | 4 +++ my-app/public/icons/ic-xmark-white.svg | 4 +++ 7 files changed, 90 insertions(+) create mode 100644 my-app/components/button/button-type.js create mode 100644 my-app/components/button/button.jsx create mode 100644 my-app/components/button/button.module.css create mode 100644 my-app/public/icons/ic-check.svg create mode 100644 my-app/public/icons/ic-plus-white.svg create mode 100644 my-app/public/icons/ic-plus.svg create mode 100644 my-app/public/icons/ic-xmark-white.svg diff --git a/my-app/components/button/button-type.js b/my-app/components/button/button-type.js new file mode 100644 index 00000000..1ee54688 --- /dev/null +++ b/my-app/components/button/button-type.js @@ -0,0 +1,7 @@ +const BUTTON_TYPE = Object.freeze({ + add: "add", + edit: "edit", + delete: "delete", +}); + +export { BUTTON_TYPE }; diff --git a/my-app/components/button/button.jsx b/my-app/components/button/button.jsx new file mode 100644 index 00000000..83236c94 --- /dev/null +++ b/my-app/components/button/button.jsx @@ -0,0 +1,27 @@ +import { BUTTON_TYPE } from "./button-type"; +import styles from "./button.module.css"; + +const className = { + [BUTTON_TYPE.add]: `${styles.button} ${styles.add}`, + [BUTTON_TYPE.edit]: `${styles.button} ${styles.edit}`, + [BUTTON_TYPE.delete]: `${styles.button} ${styles.delete}`, +}; + +const leadingIcon = { + [BUTTON_TYPE.add]: "/icons/ic-plus.svg", + [BUTTON_TYPE.edit]: "/icons/ic-check.svg", + [BUTTON_TYPE.delete]: "/icons/ic-xmark-white.svg", +}; + +function Button({ children, type = BUTTON_TYPE.add, ...props }) { + return ( + + ); +} + +export default Button; diff --git a/my-app/components/button/button.module.css b/my-app/components/button/button.module.css new file mode 100644 index 00000000..07ee1f58 --- /dev/null +++ b/my-app/components/button/button.module.css @@ -0,0 +1,41 @@ +.button { + color: var(--color-state-900); + background-color: var(--color-state-200); + padding: 18px 40px; + border: 2px solid var(--color-state-900); + border-radius: 24px; + box-shadow: 4px 4px 0px var(--color-state-900); + font-size: 16px; + font-weight: 700; + line-height: 100%; + cursor: pointer; + display: flex; + align-items: center; + gap: 4px; +} +.button .trailingIcon { + width: 16px; + height: 16px; +} +.button .trailingIcon img { + width: 100%; + height: 100%; +} + +.button.add:active { + color: white; + background-color: var(--color-violet-600); +} +.button.add:active img { + filter: invert(100%) sepia(14%) saturate(1801%) hue-rotate(190deg) + brightness(116%) contrast(101%); +} + +.button.edit:active { + background-color: var(--color-lime-300); +} + +.button.delete { + color: white; + background-color: var(--color-rose-500); +} diff --git a/my-app/public/icons/ic-check.svg b/my-app/public/icons/ic-check.svg new file mode 100644 index 00000000..c1f426d3 --- /dev/null +++ b/my-app/public/icons/ic-check.svg @@ -0,0 +1,3 @@ + + + diff --git a/my-app/public/icons/ic-plus-white.svg b/my-app/public/icons/ic-plus-white.svg new file mode 100644 index 00000000..2ec289e9 --- /dev/null +++ b/my-app/public/icons/ic-plus-white.svg @@ -0,0 +1,4 @@ + + + + diff --git a/my-app/public/icons/ic-plus.svg b/my-app/public/icons/ic-plus.svg new file mode 100644 index 00000000..935663bd --- /dev/null +++ b/my-app/public/icons/ic-plus.svg @@ -0,0 +1,4 @@ + + + + diff --git a/my-app/public/icons/ic-xmark-white.svg b/my-app/public/icons/ic-xmark-white.svg new file mode 100644 index 00000000..c054bbcb --- /dev/null +++ b/my-app/public/icons/ic-xmark-white.svg @@ -0,0 +1,4 @@ + + + + From fc5eb4eaeea14437dbe9773be4a6b2cf7ee9b5c9 Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 16:38:54 +0900 Subject: [PATCH 05/20] =?UTF-8?q?feat(Sprint9):=20SearchInput=20style=20?= =?UTF-8?q?=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/components/input/search-input.module.css | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/my-app/components/input/search-input.module.css b/my-app/components/input/search-input.module.css index 0aacd1bc..d581666f 100644 --- a/my-app/components/input/search-input.module.css +++ b/my-app/components/input/search-input.module.css @@ -2,7 +2,7 @@ background-color: var(--color-state-100); padding: 16px 24px; border: 2px solid var(--color-state-900); - border-radius: 30px; + border-radius: 24px; box-shadow: 4px 4px 0px var(--color-state-900); } .searchInput input { @@ -12,10 +12,12 @@ width: 100%; font-size: 16px; font-weight: 400; + line-height: 100%; color: var(--color-state-900); } .searchInput input::placeholder { font-size: 16px; font-weight: 400; + line-height: 100%; color: var(--color-state-500); } From 33b547d52766371672ca905dda51f9261e292f11 Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 16:53:47 +0900 Subject: [PATCH 06/20] =?UTF-8?q?feat(Sprint9):=20GlobalNavBar=20component?= =?UTF-8?q?=20=EA=B5=AC=ED=98=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/components/nav-bar/global-nav-bar.jsx | 16 ++++++++ .../nav-bar/global-nav-bar.module.css | 39 +++++++++++++++++++ my-app/public/images/logo-large.svg | 17 ++++++++ my-app/public/images/logo-small.svg | 7 ++++ 4 files changed, 79 insertions(+) create mode 100644 my-app/components/nav-bar/global-nav-bar.jsx create mode 100644 my-app/components/nav-bar/global-nav-bar.module.css create mode 100644 my-app/public/images/logo-large.svg create mode 100644 my-app/public/images/logo-small.svg diff --git a/my-app/components/nav-bar/global-nav-bar.jsx b/my-app/components/nav-bar/global-nav-bar.jsx new file mode 100644 index 00000000..7281246e --- /dev/null +++ b/my-app/components/nav-bar/global-nav-bar.jsx @@ -0,0 +1,16 @@ +import styles from "./global-nav-bar.module.css"; + +function GlobalNavBar() { + return ( + + ); +} + +export default GlobalNavBar; diff --git a/my-app/components/nav-bar/global-nav-bar.module.css b/my-app/components/nav-bar/global-nav-bar.module.css new file mode 100644 index 00000000..a3f10537 --- /dev/null +++ b/my-app/components/nav-bar/global-nav-bar.module.css @@ -0,0 +1,39 @@ +.globalNavBar { + height: 60px; + background-color: white; + padding: 0 200px; + border-bottom: 1px solid var(--color-state-200); +} + +.globalNavBar .content { + max-width: 1200px; + margin: 0 auto; + display: flex; + align-items: center; +} + +@media (max-width: 1199px) { + .globalNavBar { + padding: 0 24px; + } + + .globalNavBar .content { + max-width: none; + margin: 0 auto; + display: flex; + align-items: center; + } +} + +@media (max-width: 743px) { + .globalNavBar { + padding: 0 16px; + } + + .globalNavBar .content { + max-width: none; + margin: 0 auto; + display: flex; + align-items: center; + } +} diff --git a/my-app/public/images/logo-large.svg b/my-app/public/images/logo-large.svg new file mode 100644 index 00000000..79e3a93a --- /dev/null +++ b/my-app/public/images/logo-large.svg @@ -0,0 +1,17 @@ + + + + + + + + + + + + + + + + + diff --git a/my-app/public/images/logo-small.svg b/my-app/public/images/logo-small.svg new file mode 100644 index 00000000..4fb80423 --- /dev/null +++ b/my-app/public/images/logo-small.svg @@ -0,0 +1,7 @@ + + + + + + + From 67eac7bbe2fc841c4faf8f7e868b5ae46f3143f6 Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 17:01:33 +0900 Subject: [PATCH 07/20] =?UTF-8?q?feat(Sprint9):=20Home=20page=20=EA=B5=AC?= =?UTF-8?q?=ED=98=84=20-=20nav=20bar,=20search=20form?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/components/button/button.module.css | 1 + .../components/input/search-input.module.css | 1 + my-app/pages/index.js | 21 +++++++++++++++++-- my-app/styles/home.module.css | 16 ++++++++++++++ 4 files changed, 37 insertions(+), 2 deletions(-) create mode 100644 my-app/styles/home.module.css diff --git a/my-app/components/button/button.module.css b/my-app/components/button/button.module.css index 07ee1f58..df652dcb 100644 --- a/my-app/components/button/button.module.css +++ b/my-app/components/button/button.module.css @@ -12,6 +12,7 @@ display: flex; align-items: center; gap: 4px; + white-space: nowrap; } .button .trailingIcon { width: 16px; diff --git a/my-app/components/input/search-input.module.css b/my-app/components/input/search-input.module.css index d581666f..61d3ba10 100644 --- a/my-app/components/input/search-input.module.css +++ b/my-app/components/input/search-input.module.css @@ -1,4 +1,5 @@ .searchInput { + width: 100%; background-color: var(--color-state-100); padding: 16px 24px; border: 2px solid var(--color-state-900); diff --git a/my-app/pages/index.js b/my-app/pages/index.js index 78e87d45..7c1c19a7 100644 --- a/my-app/pages/index.js +++ b/my-app/pages/index.js @@ -1,6 +1,15 @@ +import Button from "@/components/button/button"; +import { BUTTON_TYPE } from "@/components/button/button-type"; +import SearchInput from "@/components/input/search-input"; +import GlobalNavBar from "@/components/nav-bar/global-nav-bar"; +import styles from "@/styles/home.module.css"; import Head from "next/head"; export default function Home() { + const handleAddClick = (event) => { + event.preventDefault(); + }; + return ( <> @@ -13,8 +22,16 @@ export default function Home() {
-
-

Welcome to Do It

+ +
+
+
+ + + +
diff --git a/my-app/styles/home.module.css b/my-app/styles/home.module.css new file mode 100644 index 00000000..9460084c --- /dev/null +++ b/my-app/styles/home.module.css @@ -0,0 +1,16 @@ +.main { + padding: 0 200px; +} + +.content { + width: 100%; + max-width: 1200px; + margin: 0 auto; + padding-top: 24px; +} + +.searchForm { + display: flex; + align-items: center; + gap: 16px; +} From 80f89aef99edeaf605596e03bcf322baf11f79a4 Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 17:22:10 +0900 Subject: [PATCH 08/20] =?UTF-8?q?feat(Sprint9):=20Todo=20component=20?= =?UTF-8?q?=EA=B5=AC=ED=98=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/components/todo/todo.jsx | 23 ++++++++++++++++ my-app/components/todo/todo.module.css | 33 +++++++++++++++++++++++ my-app/public/images/checkbox-checked.svg | 4 +++ my-app/public/images/checkbox.svg | 3 +++ 4 files changed, 63 insertions(+) create mode 100644 my-app/components/todo/todo.jsx create mode 100644 my-app/components/todo/todo.module.css create mode 100644 my-app/public/images/checkbox-checked.svg create mode 100644 my-app/public/images/checkbox.svg diff --git a/my-app/components/todo/todo.jsx b/my-app/components/todo/todo.jsx new file mode 100644 index 00000000..b0409c63 --- /dev/null +++ b/my-app/components/todo/todo.jsx @@ -0,0 +1,23 @@ +import styles from "./todo.module.css"; + +function Todo({ children, checked = false, ...props }) { + let className = styles.todo; + if (checked) { + className += ` ${styles.checked}`; + } + + const checkImage = checked + ? "/images/checkbox-checked.svg" + : "/images/checkbox.svg"; + + return ( +
+
+ check +
+

{children}

+
+ ); +} + +export default Todo; diff --git a/my-app/components/todo/todo.module.css b/my-app/components/todo/todo.module.css new file mode 100644 index 00000000..c0b5dd49 --- /dev/null +++ b/my-app/components/todo/todo.module.css @@ -0,0 +1,33 @@ +.todo { + background-color: white; + border: 2px solid var(--color-state-900); + border-radius: 27px; + font-size: 16px; + font-weight: 400; + line-height: 100%; + color: var(--color-state-800); + display: flex; + align-items: center; + gap: 16px; + height: 50px; + padding: 9px 12px; + cursor: pointer; +} +.todo.checked { + background-color: var(--color-violet-100); + text-decoration: line-through; +} + +.todo .checkImage { + width: 32px; + height: 32px; +} + +.todo .checkImage img { + width: 100%; + height: 100%; +} + +.todo .title { + flex-grow: 1; +} diff --git a/my-app/public/images/checkbox-checked.svg b/my-app/public/images/checkbox-checked.svg new file mode 100644 index 00000000..62bc5549 --- /dev/null +++ b/my-app/public/images/checkbox-checked.svg @@ -0,0 +1,4 @@ + + + + diff --git a/my-app/public/images/checkbox.svg b/my-app/public/images/checkbox.svg new file mode 100644 index 00000000..0e06cb40 --- /dev/null +++ b/my-app/public/images/checkbox.svg @@ -0,0 +1,3 @@ + + + From fba7503cd576882ca356f815cbdfc3a8ff1edaeb Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 17:24:25 +0900 Subject: [PATCH 09/20] =?UTF-8?q?refactor(Sprint9):=20Home=20content?= =?UTF-8?q?=EC=97=90=20=EB=B0=98=EC=9D=91=ED=98=95=20=EC=A0=81=EC=9A=A9,?= =?UTF-8?q?=20GlobalNavBar=20=EB=B6=88=ED=95=84=EC=9A=94=ED=95=9C=20style?= =?UTF-8?q?=20=EC=A0=9C=EA=B1=B0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../nav-bar/global-nav-bar.module.css | 11 +---------- my-app/styles/home.module.css | 17 +++++++++++++++++ 2 files changed, 18 insertions(+), 10 deletions(-) diff --git a/my-app/components/nav-bar/global-nav-bar.module.css b/my-app/components/nav-bar/global-nav-bar.module.css index a3f10537..cceed556 100644 --- a/my-app/components/nav-bar/global-nav-bar.module.css +++ b/my-app/components/nav-bar/global-nav-bar.module.css @@ -19,9 +19,7 @@ .globalNavBar .content { max-width: none; - margin: 0 auto; - display: flex; - align-items: center; + margin: 0; } } @@ -29,11 +27,4 @@ .globalNavBar { padding: 0 16px; } - - .globalNavBar .content { - max-width: none; - margin: 0 auto; - display: flex; - align-items: center; - } } diff --git a/my-app/styles/home.module.css b/my-app/styles/home.module.css index 9460084c..404e3d83 100644 --- a/my-app/styles/home.module.css +++ b/my-app/styles/home.module.css @@ -14,3 +14,20 @@ align-items: center; gap: 16px; } + +@media (max-width: 1199px) { + .main { + padding: 0 24px; + } + + .content { + max-width: none; + margin: 0; + } +} + +@media (max-width: 743px) { + .main { + padding: 0 16px; + } +} From ad556b02d3e59156e1fda5c6686ece90c045e40e Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 18:21:32 +0900 Subject: [PATCH 10/20] =?UTF-8?q?feat(Sprint9):=20Todo=20list=20=EA=B5=AC?= =?UTF-8?q?=ED=98=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/components/todo/todo-label.jsx | 12 ++++++++++++ my-app/components/todo/todo-label.module.css | 14 ++++++++++++++ my-app/components/todo/todo-list.jsx | 18 ++++++++++++++++++ my-app/components/todo/todo-list.module.css | 14 ++++++++++++++ my-app/pages/index.js | 12 ++++++++++++ my-app/styles/globals.css | 2 ++ my-app/styles/home.module.css | 6 ++++++ 7 files changed, 78 insertions(+) create mode 100644 my-app/components/todo/todo-label.jsx create mode 100644 my-app/components/todo/todo-label.module.css create mode 100644 my-app/components/todo/todo-list.jsx create mode 100644 my-app/components/todo/todo-list.module.css diff --git a/my-app/components/todo/todo-label.jsx b/my-app/components/todo/todo-label.jsx new file mode 100644 index 00000000..137f8091 --- /dev/null +++ b/my-app/components/todo/todo-label.jsx @@ -0,0 +1,12 @@ +import styles from "./todo-label.module.css"; + +function TodoLabel({ done }) { + let className = styles.todoLabel; + if (done) { + className += ` ${styles.done}`; + } + + return {done ? "DONE" : "TO DO"}; +} + +export default TodoLabel; diff --git a/my-app/components/todo/todo-label.module.css b/my-app/components/todo/todo-label.module.css new file mode 100644 index 00000000..55cd700a --- /dev/null +++ b/my-app/components/todo/todo-label.module.css @@ -0,0 +1,14 @@ +.todoLabel { + background-color: var(--color-lime-300); + color: var(--color-green-700); + font-size: 18px; + font-weight: 400; + line-height: 100%; + padding: 6px 27px; + border-radius: 24px; +} + +.todoLabel.done { + background-color: var(--color-green-700); + color: var(--color-amber-300); +} diff --git a/my-app/components/todo/todo-list.jsx b/my-app/components/todo/todo-list.jsx new file mode 100644 index 00000000..3d429e7b --- /dev/null +++ b/my-app/components/todo/todo-list.jsx @@ -0,0 +1,18 @@ +import Todo from "./todo"; +import TodoLabel from "./todo-label"; +import styles from "./todo-list.module.css"; + +function TodoList({ done = false }) { + return ( +
+ +
+ 비타민 챙겨 먹기 + 맥주 마시기 + 운동하기 +
+
+ ); +} + +export default TodoList; diff --git a/my-app/components/todo/todo-list.module.css b/my-app/components/todo/todo-list.module.css new file mode 100644 index 00000000..9d972c35 --- /dev/null +++ b/my-app/components/todo/todo-list.module.css @@ -0,0 +1,14 @@ +.todoList { + width: 100%; + display: flex; + flex-direction: column; + align-items: flex-start; + gap: 16px; +} + +.todoListContent { + width: 100%; + display: flex; + flex-direction: column; + gap: 16px; +} diff --git a/my-app/pages/index.js b/my-app/pages/index.js index 7c1c19a7..0fa0c76c 100644 --- a/my-app/pages/index.js +++ b/my-app/pages/index.js @@ -2,14 +2,22 @@ import Button from "@/components/button/button"; import { BUTTON_TYPE } from "@/components/button/button-type"; import SearchInput from "@/components/input/search-input"; import GlobalNavBar from "@/components/nav-bar/global-nav-bar"; +import TodoList from "@/components/todo/todo-list"; import styles from "@/styles/home.module.css"; import Head from "next/head"; +import { useState } from "react"; export default function Home() { + const [checked, setChecked] = useState(false); + const handleAddClick = (event) => { event.preventDefault(); }; + const handleToDoClick = (event) => { + setChecked((prev) => !prev); + }; + return ( <> @@ -31,6 +39,10 @@ export default function Home() { 추가하기 +
+ + +
diff --git a/my-app/styles/globals.css b/my-app/styles/globals.css index 61f32570..d996a5d0 100644 --- a/my-app/styles/globals.css +++ b/my-app/styles/globals.css @@ -10,7 +10,9 @@ --color-violet-600: #7c3aed; --color-rose-500: #f43f5e; --color-lime-300: #bef264; + --color-amber-300: #fcd34d; --color-amber-800: #92400e; + --color-green-700: #15803d; } * { diff --git a/my-app/styles/home.module.css b/my-app/styles/home.module.css index 404e3d83..7b3a5ce2 100644 --- a/my-app/styles/home.module.css +++ b/my-app/styles/home.module.css @@ -15,6 +15,12 @@ gap: 16px; } +.todoListContainer { + margin-top: 40px; + display: flex; + gap: 24px; +} + @media (max-width: 1199px) { .main { padding: 0 24px; From 96d583dd1d4d557316eeca17b3195cc10e590d95 Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 18:29:27 +0900 Subject: [PATCH 11/20] =?UTF-8?q?feat(Sprint9):=20Font=20=EC=84=A4?= =?UTF-8?q?=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/components/todo/todo-label.module.css | 1 + my-app/pages/index.js | 5 +++++ my-app/styles/globals.css | 14 ++++++++++++++ 3 files changed, 20 insertions(+) diff --git a/my-app/components/todo/todo-label.module.css b/my-app/components/todo/todo-label.module.css index 55cd700a..ca617b9a 100644 --- a/my-app/components/todo/todo-label.module.css +++ b/my-app/components/todo/todo-label.module.css @@ -1,6 +1,7 @@ .todoLabel { background-color: var(--color-lime-300); color: var(--color-green-700); + font-family: "HsSantoki20"; font-size: 18px; font-weight: 400; line-height: 100%; diff --git a/my-app/pages/index.js b/my-app/pages/index.js index 0fa0c76c..d0eb1739 100644 --- a/my-app/pages/index.js +++ b/my-app/pages/index.js @@ -27,6 +27,11 @@ export default function Home() { content="A productivity app to help you get things done" /> +
diff --git a/my-app/styles/globals.css b/my-app/styles/globals.css index d996a5d0..5e1ffe40 100644 --- a/my-app/styles/globals.css +++ b/my-app/styles/globals.css @@ -15,6 +15,20 @@ --color-green-700: #15803d; } +@font-face { + font-family: "HsSantoki20"; + src: url("https://cdn.jsdelivr.net/gh/projectnoonnu/2405@1.0/HSSanTokki20-Regular.woff2") + format("woff2"); + font-weight: normal; + font-display: swap; +} + * { box-sizing: border-box; } + +body { + font-family: NanumSquare, system-ui, -apple-system, BlinkMacSystemFont, + "Segoe UI", Roboto, Oxygen, Ubuntu, Cantarell, "Open Sans", "Helvetica Neue", + sans-serif; +} From 74ba3970df5dae0749bbb1c45bc132a62df2c10f Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 18:30:07 +0900 Subject: [PATCH 12/20] =?UTF-8?q?fix(Sprint9):=20Nav=20bar=20=EC=84=B8?= =?UTF-8?q?=EB=A1=9C=20=EC=A0=95=EB=A0=AC=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/components/nav-bar/global-nav-bar.module.css | 1 + 1 file changed, 1 insertion(+) diff --git a/my-app/components/nav-bar/global-nav-bar.module.css b/my-app/components/nav-bar/global-nav-bar.module.css index cceed556..d4dd5264 100644 --- a/my-app/components/nav-bar/global-nav-bar.module.css +++ b/my-app/components/nav-bar/global-nav-bar.module.css @@ -7,6 +7,7 @@ .globalNavBar .content { max-width: 1200px; + height: 100%; margin: 0 auto; display: flex; align-items: center; From 3438aebc52d0e686aa1613f7b0f1a3665ab70398 Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 18:32:45 +0900 Subject: [PATCH 13/20] =?UTF-8?q?feat(Sprint9):=20=EB=A1=9C=EA=B3=A0=20?= =?UTF-8?q?=ED=81=B4=EB=A6=AD=20=EC=8B=9C=20home=20=EC=9D=B4=EB=8F=99,=20?= =?UTF-8?q?=EC=9E=98=EB=AA=BB=20=EC=82=AC=EC=9A=A9=ED=95=98=EA=B3=A0=20?= =?UTF-8?q?=EC=9E=88=EB=8A=94=20=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/components/nav-bar/global-nav-bar.jsx | 14 ++++++++++---- 1 file changed, 10 insertions(+), 4 deletions(-) diff --git a/my-app/components/nav-bar/global-nav-bar.jsx b/my-app/components/nav-bar/global-nav-bar.jsx index 7281246e..81b2bcad 100644 --- a/my-app/components/nav-bar/global-nav-bar.jsx +++ b/my-app/components/nav-bar/global-nav-bar.jsx @@ -1,13 +1,19 @@ +import Link from "next/link"; import styles from "./global-nav-bar.module.css"; function GlobalNavBar() { return ( ); From e56342f4a5b351b734a7e6470e8d9a51ca6c5ad7 Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 18:59:06 +0900 Subject: [PATCH 14/20] =?UTF-8?q?feat(Sprint9):=20TODO/DONE=20=EB=AA=A9?= =?UTF-8?q?=EB=A1=9D=20=EB=B9=84=EC=97=88=EC=9D=84=20=EB=95=8C=20empty=20m?= =?UTF-8?q?essage=20=ED=91=9C=EC=8B=9C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/components/todo/todo-list.jsx | 34 ++++++++++++---- my-app/components/todo/todo-list.module.css | 18 +++++++++ my-app/pages/index.js | 21 +++++++++- my-app/public/images/done-list-empty.svg | 17 ++++++++ my-app/public/images/todo-list-empty.svg | 44 +++++++++++++++++++++ 5 files changed, 125 insertions(+), 9 deletions(-) create mode 100644 my-app/public/images/done-list-empty.svg create mode 100644 my-app/public/images/todo-list-empty.svg diff --git a/my-app/components/todo/todo-list.jsx b/my-app/components/todo/todo-list.jsx index 3d429e7b..b3d241cc 100644 --- a/my-app/components/todo/todo-list.jsx +++ b/my-app/components/todo/todo-list.jsx @@ -1,16 +1,36 @@ -import Todo from "./todo"; import TodoLabel from "./todo-label"; import styles from "./todo-list.module.css"; -function TodoList({ done = false }) { +function EmptyMessage({ children }) { + const chunks = children.split("\n"); + let messageChunks = []; + for (const chunk of chunks) { + messageChunks.push({chunk}); + messageChunks.push(
); + } + messageChunks.pop(); + return

{messageChunks}

; +} + +function TodoList({ done = false, children }) { + const emptyMessage = done + ? "아직 다 한 일이 없어요.\n해야 할 일을 체크해보세요!" + : "할 일이 없어요.\nTODO를 새롭게 추가해주세요!"; + return (
-
- 비타민 챙겨 먹기 - 맥주 마시기 - 운동하기 -
+ {children.length > 0 ? ( +
{children}
+ ) : ( +
+ empty + {emptyMessage} +
+ )}
); } diff --git a/my-app/components/todo/todo-list.module.css b/my-app/components/todo/todo-list.module.css index 9d972c35..4eb1e906 100644 --- a/my-app/components/todo/todo-list.module.css +++ b/my-app/components/todo/todo-list.module.css @@ -12,3 +12,21 @@ flex-direction: column; gap: 16px; } + +.todoListEmptyContainer { + width: 100%; + display: flex; + flex-direction: column; + align-items: center; +} +.todoListEmptyContainer img { + aspect-ratio: 1 / 1; +} +.todoListEmptyContainer p { + margin: 0; + font-size: 16px; + font-weight: 700; + line-height: 100%; + color: var(--color-state-400); + text-align: center; +} diff --git a/my-app/pages/index.js b/my-app/pages/index.js index d0eb1739..66135d6b 100644 --- a/my-app/pages/index.js +++ b/my-app/pages/index.js @@ -2,6 +2,7 @@ import Button from "@/components/button/button"; import { BUTTON_TYPE } from "@/components/button/button-type"; import SearchInput from "@/components/input/search-input"; import GlobalNavBar from "@/components/nav-bar/global-nav-bar"; +import Todo from "@/components/todo/todo"; import TodoList from "@/components/todo/todo-list"; import styles from "@/styles/home.module.css"; import Head from "next/head"; @@ -9,6 +10,10 @@ import { useState } from "react"; export default function Home() { const [checked, setChecked] = useState(false); + const [todos, setTodos] = useState([]); + + const inProgressTodos = todos.filter((todo) => !todo.isCompleted); + const doneTodos = todos.filter((todo) => todo.isCompleted); const handleAddClick = (event) => { event.preventDefault(); @@ -45,8 +50,20 @@ export default function Home() {
- - + + {inProgressTodos.map((todo) => ( + + {todo.name} + + ))} + + + {doneTodos.map((todo) => ( + + {todo.name} + + ))} +
diff --git a/my-app/public/images/done-list-empty.svg b/my-app/public/images/done-list-empty.svg new file mode 100644 index 00000000..9e7b5c61 --- /dev/null +++ b/my-app/public/images/done-list-empty.svg @@ -0,0 +1,17 @@ + + + + + + + + + + + + + + + + + diff --git a/my-app/public/images/todo-list-empty.svg b/my-app/public/images/todo-list-empty.svg new file mode 100644 index 00000000..c30e0514 --- /dev/null +++ b/my-app/public/images/todo-list-empty.svg @@ -0,0 +1,44 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + From 7198c5340deaa230f78c637d44a1a81c2244cf3c Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 19:19:17 +0900 Subject: [PATCH 15/20] =?UTF-8?q?feat(Sprint9):=20Todo=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80,=20TODO/DONE=20=EC=83=81=ED=83=9C=20=EB=B3=80?= =?UTF-8?q?=EA=B2=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/components/input/search-input.jsx | 2 +- my-app/components/nav-bar/global-nav-bar.jsx | 2 +- my-app/components/todo/todo-list.jsx | 4 +- my-app/pages/index.js | 52 ++++++++++++++++---- 4 files changed, 46 insertions(+), 14 deletions(-) diff --git a/my-app/components/input/search-input.jsx b/my-app/components/input/search-input.jsx index 49c7eae9..4c51fffd 100644 --- a/my-app/components/input/search-input.jsx +++ b/my-app/components/input/search-input.jsx @@ -3,7 +3,7 @@ import styles from "./search-input.module.css"; function SearchInput({ ...props }) { return (
- +
); } diff --git a/my-app/components/nav-bar/global-nav-bar.jsx b/my-app/components/nav-bar/global-nav-bar.jsx index 81b2bcad..bc5bdae0 100644 --- a/my-app/components/nav-bar/global-nav-bar.jsx +++ b/my-app/components/nav-bar/global-nav-bar.jsx @@ -8,7 +8,7 @@ function GlobalNavBar() { logo diff --git a/my-app/components/todo/todo-list.jsx b/my-app/components/todo/todo-list.jsx index b3d241cc..79b6968d 100644 --- a/my-app/components/todo/todo-list.jsx +++ b/my-app/components/todo/todo-list.jsx @@ -5,8 +5,8 @@ function EmptyMessage({ children }) { const chunks = children.split("\n"); let messageChunks = []; for (const chunk of chunks) { - messageChunks.push({chunk}); - messageChunks.push(
); + messageChunks.push({chunk}); + messageChunks.push(
); } messageChunks.pop(); return

{messageChunks}

; diff --git a/my-app/pages/index.js b/my-app/pages/index.js index 66135d6b..da79debb 100644 --- a/my-app/pages/index.js +++ b/my-app/pages/index.js @@ -6,21 +6,41 @@ import Todo from "@/components/todo/todo"; import TodoList from "@/components/todo/todo-list"; import styles from "@/styles/home.module.css"; import Head from "next/head"; -import { useState } from "react"; +import { useMemo, useState } from "react"; export default function Home() { - const [checked, setChecked] = useState(false); + const [inputValue, setInputValue] = useState(""); const [todos, setTodos] = useState([]); - const inProgressTodos = todos.filter((todo) => !todo.isCompleted); - const doneTodos = todos.filter((todo) => todo.isCompleted); + const canAdd = useMemo(() => inputValue.trim().length > 0, [inputValue]); + + const inProgressTodos = todos + .filter((todo) => !todo.isCompleted) + .sort((a, b) => a.id - b.id); + const doneTodos = todos + .filter((todo) => todo.isCompleted) + .sort((a, b) => a.id - b.id); + + const handleInputChange = (event) => { + setInputValue(event.target.value); + }; const handleAddClick = (event) => { event.preventDefault(); + setTodos((prev) => [ + ...prev, + { id: prev.length + 1, name: inputValue, isCompleted: false }, + ]); + setInputValue(""); }; - const handleToDoClick = (event) => { - setChecked((prev) => !prev); + const handleToDoClick = (todo) => { + setTodos((prevTodos) => { + const index = prevTodos.findIndex((prevTodo) => prevTodo.id === todo.id); + let newTodos = [...prevTodos]; + newTodos[index] = { ...todo, isCompleted: !todo.isCompleted }; + return newTodos; + }); }; return ( @@ -44,22 +64,34 @@ export default function Home() {
- -
{inProgressTodos.map((todo) => ( - + handleToDoClick(todo)}> {todo.name} ))} {doneTodos.map((todo) => ( - + handleToDoClick(todo)} + > {todo.name} ))} From 55ca1a7ebdaa87b8a78a90ba719ed7621676fea5 Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 19:20:31 +0900 Subject: [PATCH 16/20] =?UTF-8?q?refactor(Sprint9):=20stylesheet=20link?= =?UTF-8?q?=EB=A5=BC=20document=EB=A1=9C=20=EC=9D=B4=EB=8F=99?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/pages/_document.js | 8 +++++++- my-app/pages/index.js | 5 ----- 2 files changed, 7 insertions(+), 6 deletions(-) diff --git a/my-app/pages/_document.js b/my-app/pages/_document.js index ffc3f3cc..ef6313e1 100644 --- a/my-app/pages/_document.js +++ b/my-app/pages/_document.js @@ -3,7 +3,13 @@ import { Head, Html, Main, NextScript } from "next/document"; export default function Document() { return ( - + + +
diff --git a/my-app/pages/index.js b/my-app/pages/index.js index da79debb..7b153816 100644 --- a/my-app/pages/index.js +++ b/my-app/pages/index.js @@ -52,11 +52,6 @@ export default function Home() { content="A productivity app to help you get things done" /> -
From 4e9e5cce1a80fe39e1ef3a93a81a050c56473d6b Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Mon, 8 Sep 2025 19:29:19 +0900 Subject: [PATCH 17/20] =?UTF-8?q?feat(Sprint9):=20Head=20=EC=A0=95?= =?UTF-8?q?=EB=B3=B4=EB=A5=BC=20document=EB=A1=9C=20=EC=9D=B4=EB=8F=99?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/pages/_document.js | 8 +++++++- my-app/pages/index.js | 10 ---------- 2 files changed, 7 insertions(+), 11 deletions(-) diff --git a/my-app/pages/_document.js b/my-app/pages/_document.js index ef6313e1..e1696de9 100644 --- a/my-app/pages/_document.js +++ b/my-app/pages/_document.js @@ -2,8 +2,14 @@ import { Head, Html, Main, NextScript } from "next/document"; export default function Document() { return ( - + + Do It + + - - Do It - - - -
From 6416546671eb52468e1352f4afe9629aad3f978b Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Thu, 11 Sep 2025 16:26:58 +0900 Subject: [PATCH 18/20] =?UTF-8?q?feat(Sprint9):=20API=20=EC=97=B0=EB=8F=99?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/libs/apis/client.js | 47 ++++++++++++++++++++++++++++++++++++++ my-app/libs/apis/todo.js | 33 ++++++++++++++++++++++++++ my-app/pages/index.js | 20 +++++++++------- 3 files changed, 92 insertions(+), 8 deletions(-) create mode 100644 my-app/libs/apis/client.js create mode 100644 my-app/libs/apis/todo.js diff --git a/my-app/libs/apis/client.js b/my-app/libs/apis/client.js new file mode 100644 index 00000000..8dbe0121 --- /dev/null +++ b/my-app/libs/apis/client.js @@ -0,0 +1,47 @@ +class Client { + constructor(baseUrl = process.env.NEXT_PUBLIC_BASE_URL) { + this.baseUrl = baseUrl; + } + + createFetchInput(endpoint) { + return `${this.baseUrl}/${endpoint.replace()}`; + } + + async get(endpoint) { + const response = await fetch(this.createFetchInput(endpoint)); + if (!response.ok) { + throw new Error(`Error fetching ${endpoint}: ${response.statusText}`); + } + return response.json(); + } + + async post(endpoint, data) { + const response = await fetch(this.createFetchInput(endpoint), { + method: "POST", + headers: { + "Content-Type": "application/json", + }, + body: JSON.stringify(data), + }); + if (!response.ok) { + throw new Error(`Error posting ${endpoint}: ${response.statusText}`); + } + return response.json(); + } + + async patch(endpoint, data) { + const response = await fetch(this.createFetchInput(endpoint), { + method: "PATCH", + headers: { + "Content-Type": "application/json", + }, + body: JSON.stringify(data), + }); + if (!response.ok) { + throw new Error(`Error patching ${endpoint}: ${response.statusText}`); + } + return response.json(); + } +} + +export default Client; diff --git a/my-app/libs/apis/todo.js b/my-app/libs/apis/todo.js new file mode 100644 index 00000000..a7182597 --- /dev/null +++ b/my-app/libs/apis/todo.js @@ -0,0 +1,33 @@ +import Client from "./client"; + +export async function getTodos() { + try { + const client = new Client(); + const items = await client.get("items"); + return items; + } catch (error) { + return []; + } +} + +export async function addTodo(name) { + try { + const client = new Client(); + const newItem = await client.post(`items`, { name }); + return newItem; + } catch (error) { + return null; + } +} + +export async function toggleTodo(todo) { + try { + const client = new Client(); + const updatedItem = await client.patch(`items/${todo.id}`, { + isCompleted: !todo.isCompleted, + }); + return updatedItem; + } catch (error) { + return null; + } +} diff --git a/my-app/pages/index.js b/my-app/pages/index.js index 06cdff49..52ca00d2 100644 --- a/my-app/pages/index.js +++ b/my-app/pages/index.js @@ -4,8 +4,9 @@ import SearchInput from "@/components/input/search-input"; import GlobalNavBar from "@/components/nav-bar/global-nav-bar"; import Todo from "@/components/todo/todo"; import TodoList from "@/components/todo/todo-list"; +import { addTodo, getTodos, toggleTodo } from "@/libs/apis/todo"; import styles from "@/styles/home.module.css"; -import { useMemo, useState } from "react"; +import { useEffect, useMemo, useState } from "react"; export default function Home() { const [inputValue, setInputValue] = useState(""); @@ -24,24 +25,27 @@ export default function Home() { setInputValue(event.target.value); }; - const handleAddClick = (event) => { + const handleAddClick = async (event) => { event.preventDefault(); - setTodos((prev) => [ - ...prev, - { id: prev.length + 1, name: inputValue, isCompleted: false }, - ]); + const newTodo = await addTodo(inputValue); + setTodos((prev) => [...prev, newTodo]); setInputValue(""); }; - const handleToDoClick = (todo) => { + const handleToDoClick = async (todo) => { + const updatedTodo = await toggleTodo(todo); setTodos((prevTodos) => { const index = prevTodos.findIndex((prevTodo) => prevTodo.id === todo.id); let newTodos = [...prevTodos]; - newTodos[index] = { ...todo, isCompleted: !todo.isCompleted }; + newTodos[index] = updatedTodo; return newTodos; }); }; + useEffect(() => { + getTodos().then((fetchedTodos) => setTodos(fetchedTodos)); + }, []); + return ( <>
From 59e35c3782b8b248968247c08d111227f852214d Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Thu, 11 Sep 2025 17:51:47 +0900 Subject: [PATCH 19/20] =?UTF-8?q?feat(Sprint9):=20=EB=A1=9C=EB=94=A9=20?= =?UTF-8?q?=EC=A4=91=20dimmed=20view=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/components/portal/portal.jsx | 15 ++++ my-app/hooks/use-async-call.js | 17 +++++ my-app/pages/index.js | 111 +++++++++++++++------------- my-app/styles/home.module.css | 7 ++ 4 files changed, 100 insertions(+), 50 deletions(-) create mode 100644 my-app/components/portal/portal.jsx create mode 100644 my-app/hooks/use-async-call.js diff --git a/my-app/components/portal/portal.jsx b/my-app/components/portal/portal.jsx new file mode 100644 index 00000000..c80ef1ab --- /dev/null +++ b/my-app/components/portal/portal.jsx @@ -0,0 +1,15 @@ +import { useEffect, useState } from "react"; +import { createPortal } from "react-dom"; + +function Portal({ children }) { + const [container, setContainer] = useState(null); + + useEffect(() => { + // setContainer(document.getElementById("portal")); + setContainer(document.body); + }, []); + + return container && createPortal(children, container); +} + +export default Portal; diff --git a/my-app/hooks/use-async-call.js b/my-app/hooks/use-async-call.js new file mode 100644 index 00000000..2253ba92 --- /dev/null +++ b/my-app/hooks/use-async-call.js @@ -0,0 +1,17 @@ +const { useState } = require("react"); + +export function useAsyncCall() { + const [isLoading, setLoading] = useState(false); + + const execute = async (callback) => { + setLoading(true); + try { + const result = await callback(); + return result; + } finally { + setLoading(false); + } + }; + + return [isLoading, execute]; +} diff --git a/my-app/pages/index.js b/my-app/pages/index.js index 52ca00d2..0e2115a1 100644 --- a/my-app/pages/index.js +++ b/my-app/pages/index.js @@ -2,8 +2,10 @@ import Button from "@/components/button/button"; import { BUTTON_TYPE } from "@/components/button/button-type"; import SearchInput from "@/components/input/search-input"; import GlobalNavBar from "@/components/nav-bar/global-nav-bar"; +import Portal from "@/components/portal/portal"; import Todo from "@/components/todo/todo"; import TodoList from "@/components/todo/todo-list"; +import { useAsyncCall } from "@/hooks/use-async-call"; import { addTodo, getTodos, toggleTodo } from "@/libs/apis/todo"; import styles from "@/styles/home.module.css"; import { useEffect, useMemo, useState } from "react"; @@ -11,6 +13,7 @@ import { useEffect, useMemo, useState } from "react"; export default function Home() { const [inputValue, setInputValue] = useState(""); const [todos, setTodos] = useState([]); + const [isLoading, execute] = useAsyncCall(); const canAdd = useMemo(() => inputValue.trim().length > 0, [inputValue]); @@ -27,68 +30,76 @@ export default function Home() { const handleAddClick = async (event) => { event.preventDefault(); - const newTodo = await addTodo(inputValue); - setTodos((prev) => [...prev, newTodo]); - setInputValue(""); + execute(async () => { + const newTodo = await addTodo(inputValue); + setTodos((prev) => [...prev, newTodo]); + setInputValue(""); + }); }; const handleToDoClick = async (todo) => { - const updatedTodo = await toggleTodo(todo); - setTodos((prevTodos) => { - const index = prevTodos.findIndex((prevTodo) => prevTodo.id === todo.id); - let newTodos = [...prevTodos]; - newTodos[index] = updatedTodo; - return newTodos; + execute(async () => { + const updatedTodo = await toggleTodo(todo); + setTodos((prevTodos) => { + const index = prevTodos.findIndex( + (prevTodo) => prevTodo.id === todo.id + ); + let newTodos = [...prevTodos]; + newTodos[index] = updatedTodo; + return newTodos; + }); }); }; useEffect(() => { - getTodos().then((fetchedTodos) => setTodos(fetchedTodos)); + execute(async () => { + const fetchedTodos = await getTodos(); + setTodos(fetchedTodos); + }); }, []); return ( <> -
- -
-
-
- - - -
- - {inProgressTodos.map((todo) => ( - handleToDoClick(todo)}> - {todo.name} - - ))} - - - {doneTodos.map((todo) => ( - handleToDoClick(todo)} - > - {todo.name} - - ))} - -
+ +
+
+
+ + + +
+ + {inProgressTodos.map((todo) => ( + handleToDoClick(todo)}> + {todo.name} + + ))} + + + {doneTodos.map((todo) => ( + handleToDoClick(todo)} + > + {todo.name} + + ))} +
-
-
+
+
+ {isLoading &&
}
); } diff --git a/my-app/styles/home.module.css b/my-app/styles/home.module.css index 7b3a5ce2..6a2f899c 100644 --- a/my-app/styles/home.module.css +++ b/my-app/styles/home.module.css @@ -21,6 +21,13 @@ gap: 24px; } +.loading { + position: fixed; + inset: 0; + background-color: rgba(0, 0, 0, 0.2); + z-index: 1000; +} + @media (max-width: 1199px) { .main { padding: 0 24px; From 644b89983ee7b60f76ead26ee055103dc191566a Mon Sep 17 00:00:00 2001 From: Chamsol Kim Date: Thu, 11 Sep 2025 18:07:18 +0900 Subject: [PATCH 20/20] =?UTF-8?q?refactor(Sprint9):=20SSR=20=EB=B0=A9?= =?UTF-8?q?=EC=8B=9D=EC=9C=BC=EB=A1=9C=20=EC=B4=88=EA=B8=B0=20=ED=99=94?= =?UTF-8?q?=EB=A9=B4=20=EB=A1=9C=EB=94=A9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- my-app/pages/index.js | 18 ++++++++---------- 1 file changed, 8 insertions(+), 10 deletions(-) diff --git a/my-app/pages/index.js b/my-app/pages/index.js index 0e2115a1..b2ae02c6 100644 --- a/my-app/pages/index.js +++ b/my-app/pages/index.js @@ -8,11 +8,16 @@ import TodoList from "@/components/todo/todo-list"; import { useAsyncCall } from "@/hooks/use-async-call"; import { addTodo, getTodos, toggleTodo } from "@/libs/apis/todo"; import styles from "@/styles/home.module.css"; -import { useEffect, useMemo, useState } from "react"; +import { useMemo, useState } from "react"; -export default function Home() { +export async function getServerSideProps() { + const todos = await getTodos(); + return { props: { todos } }; +} + +export default function Home({ todos: initialTodos }) { const [inputValue, setInputValue] = useState(""); - const [todos, setTodos] = useState([]); + const [todos, setTodos] = useState(initialTodos ?? []); const [isLoading, execute] = useAsyncCall(); const canAdd = useMemo(() => inputValue.trim().length > 0, [inputValue]); @@ -51,13 +56,6 @@ export default function Home() { }); }; - useEffect(() => { - execute(async () => { - const fetchedTodos = await getTodos(); - setTodos(fetchedTodos); - }); - }, []); - return ( <>