브라우저 구조에 대해 간략히 공부해보자.

 

이미지 출처:  http://taligarsiel.com/Projects/howbrowserswork1.htm

 

1. User Interface (사용자 인터페이스)

사용자가 접근할 수 있는 모든 영역을 말한다.

즉, 페이지를 보여주는 창(사이트)을 제외한 나머지 모든 부분을 말한다.

ex: 주소표시줄, 이전이나 다음 혹은 새로고침 버튼등. 

user interface 예시 (네모박스를 제외한 모든 영역)

 

 

2. Browser Engine (브라우저 엔진)

User Interface와 Rendering Engine 두 사이의 동작을 제어한다. (연결 부분이 된다.)

자료 저장소(Data Storage)를 참조하고 있어서 로컬에 데이터 저장 및 읽기 등의 작업을 할 수 있다.

 

 

3. Rendering Engine (렌더링 엔진)

웹 서버로 부터 받은 자원을 브라우저 상에 표시하기 위한 작업을 한다.

즉, *HTML과 CSS을 파싱하는 동작을 한다. (중요!)

 

- 크롬은 각 탭마다 별도의 렌더링 엔진 인스턴스를 유지한다고 한다. (= 독립된 프로세스)

(크롬은 컴퓨터에서 사용하는 모든 탭, 플러그인 및 확장 프로그램에 대해 개별 프로세스를 생성하도록 설계되었다.)

별도의 프로세스

 

 

4. Networking (통신)

HTTP 요청과 같은 각종 네트워크 요청을 수행하는 부분이다.

 

 

5. Javascript Interpreter (or Javascript Engine ?) (자바스크립트 해석기) 

JS를 읽고 해석하고 실행하는 역할을 한다. (Chrome 같은 경우 V8엔진을 사용한다.)

 

 

6. UI Backend (or Display Backend) (UI 백엔드)

브라우저의 기본적인 위젯을 그리는 인터페이스이다.

ex: input, select, 

ui backend 예시

 

 

7. Data Persistance (자료 저장소)

브라우저 메모리(보조 기억 장치)를 사용하여 데이터를 저장하는 부분이다.

ex: LocalStorage, SessionStorage, Cookie...

 

 

 


더 깊은 내용에 대해 알고 싶거나, 참고한 글을 확인하고 싶다면!

https://d2.naver.com/helloworld/59361

http://taligarsiel.com/Projects/howbrowserswork1.htm

기존 작업자가 작업해 둔 코드가 있었고, 요구 조건에 따라 이슈가 전달되어왔다.

 

기존 코드

function addComma(value) {
	return value.toString().replace(/\B(?=(\d{3})+(?!\d))/g, ',');
}

 

addComma 결과 값

 

요구 조건은 아래와 같았고.. 

1. 숫자 천 단위마다 콤마 추가

2. 소수점에서는 콤마 추가하지 않아야 함

 

기존 코드에서는 소수점에서도 콤마가 추가되고 있었다.

function addComma(value) {
	return value.toString().replace(/\B(?<!\.\d*)(?=(\d{3})+(?!\d))/g, ',');
}

수정한 addComma 결과 값

 

잘되는 듯 했지만, 크로스브라우징 이슈가 있었다. 

safari에서 흰페이지로 뜨는 것이었다. 두둥.. 

그리고 SyntaxError: Invalid regular expression: invalid group specifier name 에러 메시지가 출력되고 있었다.

 

왜???

검토해보니, 정규식에 사용된 lookbehind 패턴 때문이었다.

 

정규식에는 두가지 알고리즘이 존재한다고 한다.

  • Deterministic Finite Automaton (DFA): 문자열의 문자를 한번만 확인한다.
  • Nondeterministic Finite Automaton (NFA): 최적의 일치를 찾을 때 까지 여러번 확인한다.

여기에서 자바스크립트는 NFA 알고리즘을 사용하고 있고 이 알고리즘의 동작으로 인해 Catastrophic Backtracking 가 일어날 수 있다고 한다.

 

regexp-catastrophic-backtracking 란?

더보기

오랜 시간 실행됨에 있어서 js 엔진이 중단되는 현상이라고 한다.

일반적인 증상으로 특정 문자열의 경우 CPU를 100% 사용하여 중단된다고 한다. 

그리고 중단되기 때문에 화면에 아무것또 뜨지 않았던 것이다.(;;)

해결법으로는..

1. 조합의 수를 줄여서 순서대로 찾아 나가는 방식

2. 역 추적을 방지하는 lookahead 패턴으로 개발

두번째 방식을 권장한다고 한다.

 

정규식의 전방 탐색(Lookahead)과 후방 탐색(Lookbehind) 패턴에 대해 간단히 알아보면...

전방 탐색은 작성한 패턴에 일치하는 영역이 존재하여도 그 값이 제외되어서 나오는 패턴이고

후방탐색은 전방 탐색이 앞에 있는 문자열을 탐색하는 거라면 후방탐색은 뒤에 있는 문자열을 탐색하는 것이라고 한다.

이해가 잘 안되면 이 블로그 글을 읽어보자! (저는 이해에 도움이 많이되었습니다 -> https://blog.hexabrain.net/205)

 

즉, 처음에 수정한 정규식에서 (?<!\.\d*) 이 부분이 lookbehind에 해당하는 것이었다.

(1234.1234) <- 뒤의 문자열을 탐색

해결법을 찾기 위해 검색에 검색을 해봤고~!

정리가 잘되어있는 블로그 글을 발견했다!!! ㄳㄳ -> Javascript에서 천 단위 구분 기호로 쉼표가 있는 숫자를 인쇄하는 방법

 

그리고 선택한 방법들을 작성해보았다.!

1. split, join

function addComma(value) {
	let splitVal = value.toString().split('.');   
	splitVal[0] = splitVal[0].replace(/\B(?=(\d{3})+(?!\d))/g, ","); 
    return splitVal.join("."); 
}

split-join 방식

 

2. replace function

function addComma(value){ 
	return value
    	.toString()
        .replace(/(\..*)$|(\d)(?=(\d{3})+(?!\d))/g, (digit, fract) => fract || digit + ',');
}

replace function을 방식

 

3. toLocaleString('ko-KR')

이때 maximumFractionDigits를 따로 수정하지 않으면 소수점 자릿수는 3으로 기본 값으로 출력된다.

(예시로 5로 바꾸어봤다)

function addComma(value) {
	return value.toLocaleString('ko-KR', { maximumFractionDigits: 5 });
}

toLocaleString 방식

 

여러 방식을 알아보면서 정규식에 대해서 정말정말 공부하면 도움이 많이 되겠다고 생각한 하루였다.

인프런에 정규식 강의가 있던데,.ㅎㅎ..

 

 


참고!! (감사감사! 합니다.)

toLocaleString()

lookbehind 문법은 사파리와 익스플로러에서 쓸 수 없다.

Javascript에서 천 단위 구분 기호로 쉼표가 있는 숫자를 인쇄하는 방법

내장 메소드를 사용하여 숫자 천단위마다 콤마 찍기

   할일 목록을 만들면서 Vue3 배워보자 - 01

   할일 목록을 만들면서 Vue3 배워보자 - 02

> 할일 목록을 만들면서 Vue3 배워보자 - 03

 


 

이번에는 Store를 통해 작업을 진행해보겠습니다.

기존에는 vuex를 사용해서 작업을 했었는데, vue3에서는 공식적으로 pinia 라이브러리를 사용하기를 권고하고 있습니다.

Pinia - Vuex를 대체할 새로운 Store!

 

사실 Pinia 는 Vue의 코어팀에서 활동하는 Eduardo 가
Vuex 5 의 RFC 에 따라 Vuex 5 의 구현체로 Pinia 를 작성한 것이다.
공식문서에 따르면 Vue 의 철학을 충실하게 따르면서 Pinia 를 개발중이며, 추후에 Pinia 와 Vuex 5 두 프로젝트를 하나로 합치거나 아주 쉽게 이동할 수 있도록 지원할 것이라 한다
- 출처: 사이트이동

 

 

Store 작업

pinia 설치

npm i pinia

 

 

store/index.ts 

파일 생성하고 아래와 같이 작성합니다.

import { createPinia } from "pinia";
const pinia = createPinia();

export default pinia;

 

 

main.ts

pinia를 등록해줍니다.

import { createApp } from "vue";
import App from "./App.vue";
import router from "./router";

import BootstrapVue3 from "bootstrap-vue-3";
import "bootstrap/dist/css/bootstrap.css";
import "bootstrap-vue-3/dist/bootstrap-vue-3.css";

import store from "@/store/index";

createApp(App).use(router).use(store).use(BootstrapVue3).mount("#app");

 

store/modules/todo.ts

파일 생성 후 뼈대를 작성합니다.

pinia는 vuex 처럼 따로 어딘가에 등록 할 필요가 없으며 mutatiaons이 없습니다.

actions에서 하거나 컴포넌트 내부에서 작업해도 간단합니다.

import { defineStore } from "pinia";

export const storeTodo = defineStore("todo", {
  state: () => ({}),
  getters: {},
  actions: {},
});

 

02편에서 작업했던 것들을 수정하면서 Store를 사용해보도록 하겠습니다.

 

 

App.vue 

todoList를 Store의 state로 옮겨줍니다.

그리고 싹 지워줍니다.

(interface, props, handler, data등... )

<template>
  <div class="container">
    <div class="row">
      <base-header />
    </div>
    <div class="row">
      <todo-input />
    </div>
    <div class="row">
      <router-view />
    </div>
  </div>
</template>
<script lang="ts" setup>
import BaseHeader from "@/components/header.vue";
import TodoInput from "@/components/todo-input.vue";
</script>

 

 

store/modules/todo.ts

App.vue에서 지웠던 todoList를 state에 등록하고

add, remove에 해당하는 이벤트 로직들을  actions에 등록합니다.

import { defineStore } from "pinia";
import { TodoItem } from "../index.interface";

export const useStoreTodo = defineStore("todo", {
  state: () => ({
    todoList: [
      {
        id: 0,
        title: "청소하기",
        status: "active",
      },
      {
        id: 1,
        title: "공과금 내기",
        status: "active",
      },
      {
        id: 2,
        title: "운동 30분하기",
        status: "clear",
      },
    ] as TodoItem[],
  }),
  getters: {},
  actions: {
    addTodoItem(item: TodoItem) {
      this.todoList.push(item);
    },
    removeTodoItem(id: number) {
      this.todoList.splice(id, 1);
    },
  },
});

 

 

store/index.interface.ts

여기저기 중복되고 있던 interface TodoItem을 모아줍니다.

다른 컴포넌트에서 사용하길 원한다면 import해서 사용하면됩니다.

export interface TodoItem {
  id: number;
  title: string;
  status: "active" | "clear";
}

 

 

views/item-list.vue

이제 interface파일에서 import해서 사용할 수 있게된 TodoItem interface는 지우고 store/index.interface에서 가져옵니다.

renderTodoList를 선언하여 ref로 감싸주어 해당 데이터의 변화를 감지하도록 합니다.

이때 Store에서 export하고 있는 인스턴스를 import해서 호출 한 뒤 바로 사용하면 됩니다.

<template>
  <todo-item
    v-for="item in renderTodoList"
    :key="item.id"
    :id="item.id"
    :title="item.title"
    :status="item.status"
  />
</template>
<script lang="ts">
export default {
  name: "ItemList",
};
</script>
<script lang="ts" setup>
import { ref } from "vue";
import { useStoreTodo } from "@/store/modules/todo";
import TodoItem from "@/components/todo-item.vue";

const store = useStoreTodo();
const renderTodoList = ref(store.todoList);
</script>
<style lang="scss"></style>

 

 

components/todo-item.vue

(props의 interfece도 TodoItem을 바로 사용하고 싶으나, 저는 문제가 발생하여 못했습니다.이 부분은 할 수 있으면 개선해야겠습니다.)actions의 함수도 바로 사용할 수 있습니다.

<template>
  <div class="input-group">
    <div class="input-group-text">
      <input
        class="form-check-input mt-0"
        type="checkbox"
        :checked="props.status === 'clear'"
      />
    </div>
    <input type="text" class="form-control" :value="props.title" disabled />
    <button
      class="btn btn-outline-secondary"
      type="button"
      @click="handleRemoveItem"
    >
      X
    </button>
  </div>
</template>
<script lang="ts">
export default {
  name: "TodoItem",
};
</script>
<script lang="ts" setup>
import { defineProps } from "vue";
import { useStoreTodo } from "@/store/modules/todo";
import { TodoItem } from "@/store/index.interface";

interface Props {
  id: TodoItem["id"];
  title: TodoItem["title"];
  status: TodoItem["status"];
}

const store = useStoreTodo();
const props = defineProps<Props>();

const handleRemoveItem = () => {
  store.removeTodoItem(props.id);
};
</script>

 

 

components/todo-input.vue

input도 마찬가지로 store를 사용하도록 합니다.

<template>
  <input
    type="text"
    class="form-control"
    placeholder="할일을 입력해주세요."
    :value="inputValue"
    @keyup.enter="handleAddItem"
  />
</template>
<script lang="ts">
export default {
  name: "TodoInput",
};
</script>
<script lang="ts" setup>
import { ref } from "vue";
import { useStoreTodo } from "@/store/modules/todo";

const store = useStoreTodo();
const inputValue = ref("");

const handleAddItem = (event: Event) => {
  const value = (event.target as HTMLInputElement).value;
  if (!value) return;
  inputValue.value = value;
  store.addTodoItem({
    id: 1,
    title: value,
    status: "active",
  });
  inputValue.value = "";
};
</script>

 

 

store/modules/todo.ts

이전에 구현하지 않았던 checked에 대한 로직도 추가하겠습니다.

Store에 actions를 추가해줍니다.

import { defineStore } from "pinia";
import { TodoItem } from "../index.interface";

export const useStoreTodo = defineStore("todo", {
  state: () => ({
    todoList: [
      {
        id: 0,
        title: "청소하기",
        status: "active",
      },
      {
        id: 1,
        title: "공과금 내기",
        status: "active",
      },
      {
        id: 2,
        title: "운동 30분하기",
        status: "clear",
      },
    ] as TodoItem[],
  }),
  getters: {},
  actions: {
    addTodoItem(item: TodoItem) {
      this.todoList.push(item);
    },
    removeTodoItem(id: number) {
      this.todoList.splice(id, 1);
    },
    changedStatus({
      id,
      status,
    }: {
      id: TodoItem["id"];
      status: TodoItem["status"];
    }) {
      this.todoList[id].status = status;
    },
  },
});

 

 

components/todo-item.vue

item의 체크를 할 경우, Store의 change 함수가 실행되어야 하므로 checkbox의 change이벤트에 핸들러를 바인딩해줍니다.

<template>
  <div class="input-group">
    <div class="input-group-text">
      <input
        class="form-check-input mt-0"
        type="checkbox"
        :checked="props.status === 'clear'"
        @change="handleChangeStatus"
      />
    </div>
    <input type="text" class="form-control" :value="props.title" disabled />
    <button
      class="btn btn-outline-secondary"
      type="button"
      @click="handleRemoveItem"
    >
      X
    </button>
  </div>
</template>
<script lang="ts">
export default {
  name: "TodoItem",
};
</script>
<script lang="ts" setup>
import { defineProps } from "vue";
import { useStoreTodo } from "@/store/modules/todo";
import { TodoItem } from "@/store/index.interface";

interface Props {
  id: TodoItem["id"];
  title: TodoItem["title"];
  status: TodoItem["status"];
}

const store = useStoreTodo();
const props = defineProps<Props>();

const handleRemoveItem = () => {
  store.removeTodoItem(props.id);
};
const handleChangeStatus = () => {
  const changeStatus = props.status === "active" ? "clear" : "active";
  store.changedStatus({
    id: props.id,
    status: changeStatus,
  });
};
</script>

 

 

(별도) Pinia에서 작업이 호출될 때, 콘솔로 찍어서 확인해보기

pinia 참고

 

 

store/index.ts

import { createPinia } from "pinia";
const pinia = createPinia();

pinia.use(({ store }) => {
  if (store) {
    store.$onAction(({ name, args, after, onError }) => {
      console.log(
        `%c 🍍 Event Name: ${name}`,
        "background: #222; color: #bada55"
      );
      const startTime = Date.now();
      console.log(
        `%c 🍍 Start: '${name}' with params [${args.join(", ")}].`,
        "background: #222; color: #bada55"
      );
      after((result) => {
        console.log(
          `%c 🍍 Finished: '${name}' after ${
            Date.now() - startTime
          }ms.\nResult: ${result}.`,
          "background: #222; color: #bada55"
        );
      });
      onError((error) => {
        console.log(
          `%c 🍍 Failed "${name}" after ${
            Date.now() - startTime
          }ms.\nError: ${error}.`,
          "background: red; color: white"
        );
      });
    });
  }
});

export default pinia;

귀여운 파인애플을 넣어서 작업해줍니다. 

store 이벤트가 발생할때 콘솔에 로그를 남깁니다.

 

삭제했을때 console.log

 

자 이렇게 하고나서 하나 더 해야겠죠???? 새로고침해도 데이터가 남아있을 수 있도록 localStorage에 데이터를 저장하는 작업을 하겠습니다.

 

 

localStorage에 데이터 저장하기

@vueuse/core 라이브러리 사용하여 어마어마하게 간단히 작업해보겠습니다.

 

설치

npm i @vueuse/core

 

 

store/modules/todo.ts

useStorage를 import해 온 뒤 저장하려는 state를 감싸줍니다. 

useStorage('이름', 데이터); 

크롬 개발자도구를 통해 localStorage에 저장된 것을 확인합니다.

import { defineStore } from "pinia";
import { useStorage } from "@vueuse/core";
import { TodoItem } from "../index.interface";

export const useStoreTodo = defineStore("todo", {
  state: () => ({
    todoList: useStorage("todoList", [
      {
        id: 0,
        title: "청소하기",
        status: "active",
      },
      {
        id: 1,
        title: "공과금 내기",
        status: "active",
      },
      {
        id: 2,
        title: "운동 30분하기",
        status: "clear",
      },
    ] as TodoItem[]),
  }),
  getters: {},
  actions: {
    addTodoItem(item: TodoItem) {
      this.todoList.push(item);
    },
    removeTodoItem(id: number) {
      this.todoList.splice(id, 1);
    },
    changedStatus({
      id,
      status,
    }: {
      id: TodoItem["id"];
      status: TodoItem["status"];
    }) {
      this.todoList[id].status = status;
    },
  },
});

localStorage

 

이제 더미로 넣었던 데이터를 비우고 정리 작업을 해줍니다.

그리고 remove에 splice시에 id를 index개념으로 하다보니 발생하는 버그도 해결하겠습니다.

 

 

store/modules/todo.ts

최초 데이터를 빈 배열로 바꿔줍니다.

그리고 removeTodoItem의 이벤트 내부 로직을 변경했습니다.

저는 for로 했으나 데이터가 크지 않기도 하고 직관적인 배열 메서드를 사용해도 좋습니다.

(이때, 기존 배열에서 수정하여 작업하고 있는데, 새로운 데이터를 담게 되면 오류가 발생하여서 이렇게 작업을 하였습니다. 이건 더 찾아봐야할 것 같습니다.)

import { defineStore } from "pinia";
import { useStorage } from "@vueuse/core";
import { TodoItem } from "../index.interface";

export const useStoreTodo = defineStore("todo", {
  state: () => ({
    todoList: useStorage("todoList", [] as TodoItem[]),
  }),
  getters: {},
  actions: {
    addTodoItem(item: TodoItem) {
      this.todoList.push(item);
    },
    removeTodoItem(id: number) {
      let index = null;
      for (let i = 0; this.todoList.length > i; i++) {
        const item = this.todoList[i];
        if (item.id === id) {
          console.log(item);
          index = i;
          break;
        }
      }

      if (index !== null) {
        this.todoList.splice(index, 1);
      }
    },
    changedStatus({
      id,
      status,
    }: {
      id: TodoItem["id"];
      status: TodoItem["status"];
    }) {
      this.todoList[id].status = status;
    },
  },
});

 

 

components/todo-input.vue

등록 시 id값도 마지막 데이터의 id를 찾아서 +1해줍니다.

<template>
  <input
    type="text"
    class="form-control"
    placeholder="할일을 입력해주세요."
    :value="inputValue"
    @keyup.enter="handleAddItem"
  />
</template>
<script lang="ts">
export default {
  name: "TodoInput",
};
</script>
<script lang="ts" setup>
import { ref } from "vue";
import { useStoreTodo } from "@/store/modules/todo";

const store = useStoreTodo();
const inputValue = ref("");

const handleAddItem = (event: Event) => {
  const value = (event.target as HTMLInputElement).value;
  if (!value) return;
  const length = store.todoList.length;
  const createdId = length === 0 ? 0 : store.todoList[length - 1].id + 1;

  inputValue.value = value;
  store.addTodoItem({
    id: createdId,
    title: value,
    status: "active",
  });
  inputValue.value = "";
};
</script>

 

id 저장 동작 확인

 

 

라우터를 통해 리스트 구분해서 보기( All / Active / Clear ) 

views/item-list.vue

useRoute를 통해서 다이나믹라우터의 status로 넘겨준 값을 사용하고,

watch를 통해 값이 변경되면 (header의 all, active, clear를 클릭 시) renderList의 값을 변경해 줍니다.

<template>
  <todo-item
    v-for="item in renderTodoList"
    :key="item.id"
    :id="item.id"
    :title="item.title"
    :status="item.status"
  />
</template>
<script lang="ts">
export default {
  name: "ItemList",
};
</script>
<script lang="ts" setup>
import { ref, watch } from "vue";
import { useRoute } from "vue-router";
import { useStoreTodo } from "@/store/modules/todo";
import TodoItem from "@/components/todo-item.vue";

const route = useRoute();
const store = useStoreTodo();
let renderTodoList = ref(store.todoList);

watch(
  () => route.params.status,
  (newVal) => {
    if (!newVal) {
      renderTodoList.value = store.todoList;
    } else if (newVal === "active" || newVal === "clear") {
      renderTodoList.value = [...store.todoList].filter((item: any) => {
        return item.status === newVal;
      });
    }
  },
  { deep: true }
);
</script>

 

이제 아이템을 등록해보고, 체크박스를 체크한 후 header의 링크를 눌러서 변화를 확인합니다.

 

All일 경우

all

Active일 경우

active

Clear일 경우

clear

 

 

 

 

최종적으로 todo-list 만들기 작업을 완료하였습니다.

저는 vue3의 compositionAPI 방식보다 script-setup 방식이 문법이 더 간결해서 마음에 듭니다.

바뀐 부분이 많이 있으니 공식문서를 확인해보시길 추천합니다.

 

 

pacakge.json 

{
  "name": "todo-list",
  "version": "0.1.0",
  "private": true,
  "scripts": {
    "serve": "vue-cli-service serve",
    "build": "vue-cli-service build",
    "lint": "vue-cli-service lint"
  },
  "dependencies": {
    "@popperjs/core": "^2.11.2",
    "@vueuse/core": "^7.7.1",
    "bootstrap": "^5.1.3",
    "bootstrap-vue-3": "^0.1.7",
    "core-js": "^3.6.5",
    "pinia": "^2.0.11",
    "vue": "^3.0.0",
    "vue-router": "^4.0.0-0"
  },
  "devDependencies": {
    "@typescript-eslint/eslint-plugin": "^4.18.0",
    "@typescript-eslint/parser": "^4.18.0",
    "@vue/cli-plugin-babel": "~4.5.15",
    "@vue/cli-plugin-eslint": "~4.5.15",
    "@vue/cli-plugin-router": "~4.5.15",
    "@vue/cli-plugin-typescript": "~4.5.15",
    "@vue/cli-service": "~4.5.15",
    "@vue/compiler-sfc": "^3.0.0",
    "@vue/eslint-config-prettier": "^6.0.0",
    "@vue/eslint-config-typescript": "^7.0.0",
    "eslint": "^6.7.2",
    "eslint-plugin-prettier": "^3.3.1",
    "eslint-plugin-vue": "^7.0.0",
    "node-sass": "^4.12.0",
    "prettier": "^2.2.1",
    "sass-loader": "^8.0.2",
    "typescript": "~4.1.5"
  }
}

   할일 목록을 만들면서 Vue3 배워보자 - 01

> 할일 목록을 만들면서 Vue3 배워보자 - 02

   할일 목록을 만들면서 Vue3 배워보자 - 03

 


 

먼저 Store 없이, emits을 통해서 데이터 통신하는 방식으로 개발해보겠습니다.

Style 영역은 01과 달라진 부분이 없으므로 코드에 나타내지 않겠습니다.

 

 

기능 개발 - 01 등록하기

 

views/App.vue

<template>
  <div class="container">
    <div class="row">
      <base-header />
    </div>
    <div class="row">
      <todo-input @update:todo="handleUpdateTodoList" />
    </div>
    <div class="row">
      <router-view :todoList="todoList" />
    </div>
  </div>
</template>
<script lang="ts" setup>
import { ref } from "vue";
import BaseHeader from "@/components/header.vue";
import TodoInput from "@/components/todo-input.vue";

interface TodoItem {
  id: number;
  title: string;
  status: "active" | "clear";
}

const todoList = ref([
  {
    id: 0,
    title: "청소하기",
    status: "active",
  },
  {
    id: 1,
    title: "공과금 내기",
    status: "active",
  },
  {
    id: 2,
    title: "운동 30분하기",
    status: "clear",
  },
] as TodoItem[]);

const handleUpdateTodoList = (item: TodoItem) => {
  todoList.value.push(item);
};
</script>

데이터 정의는 ref를 사용합니다.

1. App.vue가 가장 최상위 파일이므로 todoList 데이터를 만들어줍니다. 

TodoItem에 대한 interface도 작성해줍니다. 

이때, status는 string으로 해도되지만 'active'와 'clear'만 사용할 것이므로 문자열로 적어줍니다. (vue에서 props받을 때, validator 생각하면 될 듯 합니다.) 이 외의 문자열이 오거나 타입이 다르면 에러가 납니다.

2. todo-input에서 emit을 통해 이벤트(update:todo)가 발생하면 handleUpdateTodoList 함수를 호출하여 todoList에 데이터를 추가해줄 것입니다. 

 

 

components/todo-input.vue

<template>
  <input
    type="text"
    class="form-control"
    placeholder="할일을 입력해주세요."
    :value="inputValue"
    @keyup.enter="handleAddItem"
  />
</template>
<script lang="ts">
export default {
  name: "TodoInput",
};
</script>
<script lang="ts" setup>
import { ref, defineEmits } from "vue";

const inputValue = ref("");
const emit = defineEmits(["update:todo"]);

const handleAddItem = (event: Event) => {
  const value = (event.target as HTMLInputElement).value;
  if (!value) return;
  inputValue.value = value;
  emit("update:todo", {
    id: 1,
    title: value,
    status: "active",
  });
  inputValue.value = "";
};
</script>

이제 App.vue까지 통신하기위해서 데이터를 입력하는 가장작은 단위의 컴포넌트인, todo-input 컴포넌트 작업을 하겠습니다.

데이터 inputValue를 ref를 통해 선언합니다.

1. input에 :value로 inputValue를 등록하고, enter를 했을 경우에 대한 keyup 이벤트를 등록합니다.

2. defineEmits을 사용해 emit을 등록해줍니다. 상위 컴포넌트에서 이벤트를 받을때와 동일한 이름이어야합니다.

위의 App.vue에서 등록한 이름과 동일하게 (update:todo) 작성합니다. 

3. handleAddItem 이벤트가 발생하게 되면 실행될 로직을 작성합니다.

emit에 params로 TodoItem에서 받는 값 id, title, status를 전달합니다. 그 후 inputValue를 비워줍니다.

 

 

components/todo-item.vue

<template>
  <div class="input-group">
    <div class="input-group-text">
      <input
        class="form-check-input mt-0"
        type="checkbox"
        :checked="props.status === 'clear'"
      />
    </div>
    <input type="text" class="form-control" :value="props.title" disabled />
    <button class="btn btn-outline-secondary" type="button">X</button>
  </div>
</template>
<script lang="ts">
export default {
  name: "TodoItem",
};
</script>
<script lang="ts" setup>
import { defineProps } from "vue";

interface TodoItem {
  id: number;
  title: string;
  status: "active" | "clear";
}

const props = defineProps<TodoItem>();
</script>

1. checkbox의 checked가 'clear' 일 경우에 체크될 수 있도록 적용해줍니다.

2. views/item-list에서 todo-item에 필요한 props를 내려줄 것입니다. 입력할때 id, title, status를 보내고 있으니 내려오는 값도 동일합니다. defineProps를 통해 props 선언해줍니다.

이때 interface에 TodoItem을 또 등록하고, props에 타입을 선언해주면 내부에 따로 등록을 하지 않아도 됩니다.

만얀 interface를 사용하지 않을 것이라면, 아래와 같이 다른 방식으로 작업해도 됩니다.

defineProps 내부에서 선언

const props = defineProps({
  id: {
    type: Number,
    required: true,
  },
  title: {
    type: String,
    required: true,
  },
  status: {
    type: String,
    required: true,
    validator: (val: string) => ["active", "clear"].includes(val),
  },
});

 

 

views/item-list.vue

v-for를 통해 todo-item 컴포넌트들을 여러개 생성해줍니다.

<template>
  <todo-item
    v-for="item in props.todoList"
    :key="item.id"
    :id="item.id"
    :title="item.title"
    :status="item.status"
  />
</template>
<script lang="ts">
export default {
  name: "ItemList",
};
</script>
<script lang="ts" setup>
import { defineProps } from "vue";
import TodoItem from "@/components/todo-item.vue";

interface ITodoItem {
  id: number;
  title: string;
  status: "active" | "clear";
}

interface Props {
  todoList: ITodoItem[];
}

const props = defineProps<Props>();
</script>

 

렌더링

할일을 입력한 후, 엔터키를 누르면 아이템이 추가되는 것을 확인할 수 있습니다.

할일 등록된 모습

 

 

기능 개발 - 02 삭제하기

작은 단위 컴포넌트의 기능부터 큰 범위로 올라가면서 개발해보도록 하겠습니다.

 

component/todo-item.vue

<template>
  <div class="input-group">
    <div class="input-group-text">
      <input
        class="form-check-input mt-0"
        type="checkbox"
        :checked="props.status === 'clear'"
      />
    </div>
    <input type="text" class="form-control" :value="props.title" disabled />
    <button
      class="btn btn-outline-secondary"
      type="button"
      @click="handleRemoveItem"
    >
      X
    </button>
  </div>
</template>
<script lang="ts">
export default {
  name: "TodoItem",
};
</script>
<script lang="ts" setup>
import { defineProps, defineEmits } from "vue";

interface TodoItem {
  id: number;
  title: string;
  status: "active" | "clear";
}

const props = defineProps<TodoItem>();
const emit = defineEmits(["remove:todo"]);
const handleRemoveItem = () => {
  emit("remove:todo", props.id);
};
</script>

1. x 버튼(삭제) 클릭시에 handleRemoveItem 이벤트 바인딩 해줍니다.

해당 이벤트가 실행되면 emit을 통해 remove:todo가 실행됩니다. 상위 컴포넌트(item-list)에서 이벤트를 캐치할 수 있습니다.

이때, 기억해야할 것은 TodoList는 최상위 컴포넌트(App.vue)에 있으므로 App.vue까지 이벤트가 닿아야할 것입니다. (props로 전달받은 데이터는 수정을 하면 안됩니다.)

2. TodoList에서 지워야하는 아이템을 찾아야하므로 유니크한 값(id)를 param로 전달합니다.

 

 

views/item-list.vue

todo-item의 바로 부모 컴포넌트입니다. (App.vue는 item-list를 호출하고 있습니다.)

그렇기 때문에 todo-item에서 받은 remove:todo 이벤트에서 다시한번 emit 해줍니다. (네이밍을 동일하게 해줍니다만, 다르게 하고 App.vue에서 동일한 이름으로 받으면됩니다.)

간단히 말하면, 부모-자식 간의 emit의 이벤트 이름이 동일하면됩니다.

<template>
  <todo-item
    v-for="item in props.todoList"
    :key="item.id"
    :id="item.id"
    :title="item.title"
    :status="item.status"
    @remove:todo="(id) => emit('remove:todo', id)"
  />
</template>
<script lang="ts">
export default {
  name: "ItemList",
};
</script>
<script lang="ts" setup>
import { defineProps, defineEmits } from "vue";
import TodoItem from "@/components/todo-item.vue";

interface ITodoItem {
  id: number;
  title: string;
  status: "active" | "clear";
}

interface Props {
  todoList: ITodoItem[];
}

const props = defineProps<Props>();
const emit = defineEmits(["remove:todo"]);
</script>

 

 

App.vue

<template>
  <div class="container">
    <div class="row">
      <base-header />
    </div>
    <div class="row">
      <todo-input @update:todo="handleUpdateTodoList" />
    </div>
    <div class="row">
      <router-view :todoList="todoList" @remove:todo="handleRemoveTodoItem" />
    </div>
  </div>
</template>
<script lang="ts" setup>
import { ref } from "vue";
import BaseHeader from "@/components/header.vue";
import TodoInput from "@/components/todo-input.vue";

interface TodoItem {
  id: number;
  title: string;
  status: "active" | "clear";
}

const todoList = ref([
  {
    id: 0,
    title: "청소하기",
    status: "active",
  },
  {
    id: 1,
    title: "공과금 내기",
    status: "active",
  },
  {
    id: 2,
    title: "운동 30분하기",
    status: "clear",
  },
] as TodoItem[]);

const handleUpdateTodoList = (item: TodoItem) => {
  todoList.value.push(item);
};
const handleRemoveTodoItem = (id: number) => {
  todoList.value = todoList.value.filter((item: TodoItem) => item.id !== id);
};
</script>
<style lang="scss">
#app {
  padding: 50px 0;

  .row + .row {
    margin-top: 15px;
  }
}
</style>

최상위 컴포넌트에서 자식 컴포넌트(item-list.vue)에서 emit한 이벤트를 캐치해줍니다.

handleRemoveTodoItem을 바인딩합니다. 해당 이벤트는 todoList에서 id를 찾아서 제거한 값을 새로운 값으로 넣어줍니다.

x 버튼 클릭 시 해당 item 삭제

 

자, 지금까지 작업한 것들은 몇가지 문제가 있습니다.

- changed 개발을 까먹었다.(머쓱) 

1. input 등록 시 id를 정적으로 넣고있어서 등록 시 id값이 다 동일하게 저장되고 있다. 이것은 삭제 시 문제가 된다.

2. 새로고침 시 수정한 데이터가 사라진다.

3. interface가 여러군데 중복 선언 되었다.

4. remove:todo 두번에 걸쳐서 emit하고 있다. (즉, 실수해서 버그나기 딱 좋다)

 

 

다음편 보기

 

 

 

+ Recent posts