일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | 5 | ||
6 | 7 | 8 | 9 | 10 | 11 | 12 |
13 | 14 | 15 | 16 | 17 | 18 | 19 |
20 | 21 | 22 | 23 | 24 | 25 | 26 |
27 | 28 | 29 | 30 | 31 |
- 빅데이터분석기사필기
- 금융권 부트캠프
- 빅분기
- 데이터분석자격증
- 금융권 it
- 부트캠프
- 전문가특강
- 멀티캠퍼스
- KB국민은행
- kb 취업교육
- 반별이벤트
- autohotkey
- 빅데이터분석기사
- kb취업교육
- 이차원배열
- kb네트워킹캠프
- sql
- SQLD
- 빅분기필기
- kb it's your life
- 첫알고리즘평가
- sql내장함수
- kbit교육
- SQL데이터타입
- kb it's your life 기자단
- 금융권it
- kb it's your life 6기
- kb 기자단
- prefixsum #C언어
- 데이터분석
- Today
- Total
지식보부상님의 공부 일지
Vue [08] 컴포넌트 심화 본문
[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:사용자정의 모델명
[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>
'KB IT's Your Life > Vue' 카테고리의 다른 글
Vue[10-1] vue-router를 이용한 라우팅 (1) | 2025.03.28 |
---|---|
Vue [09] Composition API (0) | 2025.03.27 |
Vue [07-2] TodoList 예제 리팩토링 (0) | 2025.03.27 |
Vue [07-1] 단일 파일 컴포넌트를 이용한 Vue 애플리케이션 개발 (1) | 2025.03.27 |
Vue [05] 이벤트 처리 (1) | 2025.03.26 |