지식보부상님의 공부 일지

Vue [08] 컴포넌트 심화 본문

KB IT's Your Life/Vue

Vue [08] 컴포넌트 심화

지식보부상님 2025. 3. 27. 11:37

[1] 단일 파일 컴포넌트에서의 스타일

◈ 전역 CSS

- src/main.js 에서 임포트한 CSS 스타일

- 페이지 전체에 적용됨

- assets/main.css 에 기술

- 다른 컴포넌트들과의 스타일과 충돌 피하는 것 고려해야

 

◈ 범위 CSS

- 컴포넌트가 렌더링하는 요소에 특성 기반 추가적인 식별자 부여하여 => 충돌 회피

- <style> 태그에 <style scoped> 이용하면 각각의 태그에 대한 스타일 지정이 가능 i.e. 식별자가 부여됨

더보기
더보기
더보기

Child1.vue

<template>
  <div class="child">
    <h2>Child1</h2>
  </div>
</template>

<sciprt>
    export default{
        name: 'Child1',
    };
</sciprt>

<style scoped>
.child {
  background-color: yellow;
  border: solid 1px black;
  margin: 1.5em;
  padding: 1em;
}
</style>

 

Child2.vue

<template>
  <div class="child">
    <h2>Child2</h2>
  </div>
</template>

<sciprt>
      export default{
          name: 'Child2',
      };
  </sciprt>

<style scoped>
.child {
  background-color: skyblue;
  border: solid 1px black;
  margin: 1.5em;
  padding: 1em;
}
</style>

Child3.vue

<template>
  <div class="child">
    <h2>Child3</h2>
  </div>
</template>

<sciprt>
      export default{
          name: 'Child3',
      };
  </sciprt>

<style>
.child {
  background-color: pink;
  border: solid 1px black;
  margin: 1.5em;
  padding: 1em;
}
</style>

App.vue

<template>
  <div>
    <Child1 />
    <Child2 />
    <Child3 />
  </div>
</template>

<script>
import Child1 from './components/Child1.vue';
import Child2 from './components/Child2.vue';
import Child3 from './components/Child3.vue';

export default {
  name: 'App',
  components: { Child1, Child2, Child3 },
};
</script>

◈ CSS 모듈

- <style module> 이용하여 CSS 스타일을 객체처럼 처리

 

[2] 슬롯

◈ 슬롯(Slot)

- 부모~자식 컴포넌트 사이에 템플릿 정보를 전달하는 방법

cf) 부모~자식 컴포넌트 사이에 정보 교환 방법: props, 이벤트 (Vue [07] 참고)

- 부모 컴포넌트는 템플릿을 결정, 자식 컴포넌트는 위치를 결정하여 부모→자식으로 템플릿 전달

- 자식 컴포넌트 원하는 위치에 <slot></slot>

- 전달되는 템플릿 없는 경우 보여줄 fallback UI는 <slot>여기!</slot> 사이에 작성

 

 명명된 슬롯(Named Slot)

- 자식 컴포넌트에 slot 이 여러개 있으면 구분이 필요 => slot name 이용

- 자식 컴포넌트: <slot name="슬롯명"></slot>

- 부모 컴포넌트: <... v-slot: 슬롯명></....>   *슬롯 명에 " "(큰따옴표) 없음 주의!

- 화면 레이아웃 관리 목적으로 주로 사용

 

◈ 범위 슬롯(Scoped Slot)

- 부모 컴포넌트에서 템플릿에 자식 컴포넌트의 데이터를 바인딩할 때 사용

- 전달된 데이터는 슬롯 템플릿 내부범위에서만 사용 가능

더보기
더보기
더보기

src/components/FancyPhotoBox.vue (자식 컴포넌트)

<template lang="">
  <div class="card">
    <slot name="title"></slot>
    <slot name="photo"></slot>
  </div>
</template>
<script>
export default {
  name: 'FancyPhotoBox',
};
</script>
<style scoped>
.card {
  padding: 20px;
  background-color: skyblue;
  width: 300px;
  height: 400px;
  border-radius: 10%;
  border-color: gray;
  border-style: solid;
  border-width: 10px;
  text-align: center;
}
</style>

 

src/components/FancyPhotoBox (부모 컴포넌트)

<template lang="">
  <div>
    <FancyPhotoBox>
      <template v-slot:title>
        <h1>침교동을 아시나요?</h1>
      </template>
      <template v-slot:photo>
        <img
          src=""
          alt="침교동"
          width="300px"
        />
      </template>
    </FancyPhotoBox>
  </div>
</template>

<script>
import FancyPhotoBox from './FancyPhotoBox.vue';
export default {
  name: 'FancyPhotoBoxParent',
  components: { FancyPhotoBox },
};
</script>
<style lang=""></style>

 

src/App.vue

<script setup>
import FancyPhotoBoxParent from './components/FancyPhotoBoxParent.vue';
</script>

<template>
  <FancyPhotoBoxParent />
</template>

 

[3] 동적 컴포넌트

◈ 동적 컴포넌트(Dynamic Component)

- 화면 동일한 위치에 여러 컴포넌트 표현해야 하는 경우 用 (v-if 적용하는 것과 동일 효과)

- <component> 요소 템플릿에 작성

 

[4] 컴포넌트에서의 v-model 디렉티브

◈ 사용자 정의 v-model 만들기

- 부모 컴포넌트: <child-component v-model:message="parentMessage" />

- 자식 컴포넌트:  update:사용자정의 모델명

{
  name: "ChildComponent",
  props: {message:String},
  template:`
  <input type="text" :value="message"
  @input="$emit('update:message', $event.target.value)"/>
  `
}

 

[5] provide, inject를 이용한 공용 데이터 사용

◈ provide, inject

- 기존 props는 계층 구조 따라 연속적으로 속성 전달해야한다는 문제점이 있었음

=> 공용 데이터를 부모 컴포넌트(App)에서 provide 하고, 하위 컴포넌트 어디든 데이터 inject 하면 用 가능하도록 함

- 공통 조상끼리만 사용 가능하다는 단점도 있음 

 

[6] 텔레포트

◈ 텔레포트(Teleport)

- 컴포넌트 트리 계층구조와 관계없이 별도의 요소에 렌더링해야하는 경우 用

- 애플리케이션에서 모달, 튤립과 같이 메이니 화면에 독립적이게 공유 UI 제공하는 경우 用

더보기
더보기
더보기

src/components/Modal.vue (자식 컴포넌트)

<template lang="">
  <div class="modal-overlay" v-if="visible">
    <div class="modal">
      <button class="close" @click="sendClose">닫기</button>
      <div class="modal-content">
        <slot></slot>
      </div>
    </div>
  </div>
</template>
<script>
export default {
  name: 'Modal',
  props: {
    visible: {
      type: Boolean,
      required: true,
    },
  },
  methods: {
    sendClose() {
      this.$emit('close');
    },
  },
};
</script>
<style scoped>
.modal-overlay {
  position: fixed;
  top: 0;
  left: 0;
  right: 0;
  bottom: 0;
  background-color: rgba(0, 0, 0, 0.5);
  display: flex;
  justify-content: center;
  align-items: center;
}

.modal {
  background: white;
  padding: 20px;
  border-radius: 5px;
  position: relative;
}

.close-button {
  position: absolute;
  top: 10px;
  right: 10px;
  background: transparent;
  border: none;
  font-size: 20px;
  font-weight: 700;
  cursor: pointer;
}

.modal-content {
  margin-top: 40px;
}
</style>

 

src/components/ModalParent.vue (부모 컴포넌트)

<template lang="">
  <div>
    <button @click="showModal">모달 열기</button>
    <Modal :visible="isModalVisible" @close="hideModal">
      <h2>모달 내용</h2>
      <p>안녕하세요</p>
    </Modal>
  </div>
</template>

<script>
import Modal from './Modal.vue';
export default {
  name: 'ModalParent',
  components: { Modal },
  data() {
    return {
      isModalVisible: false,
    };
  },
  methods: {
    showModal() {
      this.isModalVisible = true;
    },
    hideModal() {
      this.isModalVisible = false;
    },
  },
};
</script>
<style lang=""></style>

 

src/App.vue

<script setup>
import ModalParent from './components/ModalParent.vue';
</script>

<template>
  <ModalParent />
</template>