Vue.js
프론트 엔드
- HTML, CSS , JavaScript
- Vue.js, React(facebook), Angular(google)
Vue.js
- 사용자 인터페이스를 만들기 위한 프로그레시브 프레임워크
- 현대적인 tool과 다양한 라비러리를 통해 SPA(Single page Aplication)를 완벽하게 지원
Evan You (2014)
- Angular 개발자 출신
- 학사 미술, 미술사 전공/ 석사 디자인 & 테크놀로지 전공
- 구글 Angular 보다 더 가볍고, 간편하게 사용할 수 있는 프레임워크를 만들기 위해 개발
SPA
단일 페이지 애플리케이션
현재 페이지를 동적으로 작성, 사용자와 소통하는 웹 애플리케이션
단일 페이지 구성, 처음 페이지만 받아오고 (HTML) 동적으로 DOM을 구성
- 새로고침 X
- 사용자 경험(UX)을 향상
- 트래픽 감소, 속도, 사용성, 반응성
동작원리 일부가 CSR을 따름
등장 배경
- 모바일 최적화에 대한 필요성
CSR (Client side Rendering)
- 최초 요청 시 서버에서 빈 문서를 응답 후 클라이언트에서 데이터를 요청해서 데이터를 받아 DOM을 렌더링하는 방식
- SSR보다 초기 전송되는 페이지 속도는 빠르지만, 서비스에서 필요한 데이터 클라이언트(브라우저)에서 추가로 요청하여 재구성해야기때문에 페이지 완료시점은 SSR보다 느림
- SPA가 사용하는 렌더링 방식
CSR단점
Server > HTML > JS
Vue가 JS를 다 실행해야하고 이 과정이 다끝나야 페이지를 띄움
// 빈페이지를 받아온다. JS에서 전부 다운받아 실행시켜서 완료해야지 사용자가 사용가능하다.
장점
서버와 클라이언트 간의 트래픽 감소
- 웹 애플리케이션에 필요한 모든 정적 리소스를 최초에 한번 다운로드
사용자 경험 향상
- 전체 페이지를 다시 렌더링하지 않고 변경되는 부분만을 갱신
단점
SEO(검색엔진 최적화) 문제가 발생할 수 있음
- 구글 엔진 , naver 등 검색결과 최상단에 올리기가 힘들다. (단, 강제하는 방법이있음)
SSR(반대되는 개념 Server side)
- 서버에서 사용자에게 보여줄 page를 미리 모두 구성해서 사용자에게 페이지를 보여줌
Server > 모두완료시킴 HTML > 보여줌 ( 매번 HTML을 받아야해서 느림)
장점
- 초기 로딩 속도가 빠르기 때문에 사용자가 컨텐츠를 빨리 볼 수 있음
- SEO(검색엔진 최적화)가 가능
단점
모든 요청에 새로고침이 되기 때문에 사용자 경험이 떨어짐
- 상대적으로 요청 횟수가 많아져 서버 부담이 커짐
SEO
search engine Optimization(검색 엔진 최적화)
웹 페이지 검색엔진이 자료를 수집하고 순위를 메기는 방식에 맞게
웹페이지를 구성해서 검색 결과의 상위에 노출될 수 있도록 하는 작업
인터넷 마케팅 방법중 하나
구글 등장이후 검색 엔진들이 컨텐츠의 신뢰도를 파악하는 기초 지표로 사용됨
- 다른 웹사이트에서 얼마나 이용됐나
- 얼마나 제목이 ?
SEO 문제 대응
Vue.js 또는 React 등의 SPA 프레임워크는 SSR을 지원하는 SEO 대응 기술이 존제
Vue.js 는 왜 사용해야 할까
- 페이지 규모 계속 커져, 데이터 늘어나고, 사용자랑 상 호작용 많이 이뤄짐
- Vanilla JS만으로 관리하기가 어렵다.
Django & Vue.js 코드 작성 순서
Django
- url > view > temlplate
Vue.js
- Data 로직 작성 > DOM작성
Vue.js
(https://kr.vuejs.org/v2/guide/index.html)
여기에 다있습니다.
# 개발버전
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
<!DOCTYPE html>
<html lang="ko">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Vue Quick Start</title>
</head>
<body>
<!-- 2. 선언적 렌더링 -->
<h2>선언적 렌더링</h2>
<div id="app">
<p>{{ message }}</p>
<p>{{ message }}</p>
<p>{{ message }}</p>
</div>
<!-- 3. 엘리멘트 속성 바인딩 , 데이터 바인딩 -->
<h2>Element 속성 바인딩</h2>
<div id="app-2">
<span v-bind:title="message">
내 위에 잠시 마우스를 올리면 동적으로 바인딩 된 title을 볼 수 있습니다!
</span>
</div>
<!-- 4. 조건 v-if="들어갈조건"-->
<h2>조건</h2>
<div id="app-3">
<p v-if="isVisible">이제 나를 볼 수 있어요</p>
</div>
<!-- 5. 반복 -->
<h2>반복</h2>
<div id="app-4">
<ol>
<li v-for="todo in todos">
{{ todo.text }}
</li>
</ol>
</div>
<!-- 6. 사용자 입력 핸들링 -->
<h2>사용자 입력 핸들링</h2>
<div id="app-5">
<p>{{ message }}</p>
<button v-on:click="reverseMessage">메시지 뒤집기</button>
</div>
<!-- 1. Vue CDN -->
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
<script>
// const pList = document.querySelectorAll('p')
// pList.forEach(p => {
// p.innerText = '새로운 메세지'
// })
// 2. 선언적 렌더링
var app = new Vue({
el: '#app',
data: {
message: '안녕하세요 Vue!'
}
})
// 3. 엘리먼트 속성 바인딩 , 데이터 바인딩
var app2 = new Vue({
el: '#app-2',
data: {
message: '너모재밌다'
}
})
// 4. 조건
var app3 = new Vue({
el: '#app-3',
data: {
isVisible: true
}
})
// 5. 반복
var app4 = new Vue({
el: '#app-4',
data: {
todos: [
{ text: 'JavaScript 배우기' },
{ text: 'Vue 배우기' },
{ text: '무언가 멋진 것을 만들기' }
]
}
})
// 6. 사용자 입력 핸들링
var app5 = new Vue({
el: '#app-5',
data: {
message: '안녕하세요! Vue.js!'
},
methods: {
reverseMessage: function () {
this.message = this.message.split('').reverse().join('')
}
}
})
</script>
</body>
</html>
Vue instance
하나의 Vue 인스탠스는 Vue 컴포넌트이다.
- Vue Instance === Vue Component
Options/DOM – ‘el’
- Vue 인스턴스의 데이터 객체
- Vue 앱의 데이터를 정의하는 곳
- v-bind, v-on과 같은 디렉티브에서 사용가능
Options/Data – ‘data’
- Vue 인스턴스에 추가할 메서드
- Vue template에서 interpolation을 통해 접근 가능
- v-on과 같은 디렉티브에서도 사용 가능 •
- Vue 객체 내 다른 함수에서 this 키워드를 통해 접근 가
this keyword in vue.js(중요)
- vue 인스턴스 자체를 가리킴(자기자신 객체를 가르침)
Template Syntax
렌더링 된 DOM을 기본 Vue 인스턴스의 데이터에 선언적으로 바인딩 할 수 있는 HTML 기반 템플릿 구문을 사용
- Interpolation
- Directiv
v-text
- 엘리먼트의 textContent를 업데이트
- 내부적으로 interpolation 문법이 v-text로 컴파일 됨
<div id = "app">
<p v-text="message"></p>
<p>{{ message }}</P>
<div>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
<script>
const app = new Vue({
el: '#app',
data: {
message:'Hello',
}
})
</script>
v-html
- 안녕!
- v-html 사용 절대 금지 Xss 공격을 받을수있다.
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
<div id="app">
<p v-html="coolMessage"></p>
</div>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
<script>
const app = new Vue({
el: '#app',
data:{
coolMessage: '<h1>안녕!</h1>'
},
})
</script>
</body>
</html>
v-show
isLoggedIn 이 참이면 보이고 거짓이면 안보이고~
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
<div id="app">
<p v-show="isLoggedIn">보이나요</p>
</div>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
<script>
const app = new Vue({
el: '#app',
data:{
isLoggedIn: true,
},
})
</script>
</body>
</html>
v-if, v-else-if, v-else
조건문으로 나타내깅~
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
<div id="app">
<p v-if="myType==='AB'">
AB입니다.
</p>
<p v-else-if="myType==='O'"> O입니다.</p>
<p v-else>사람이 아닙니다.</p>
</div>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
<script>
const app = new Vue({
el: '#app',
data: {
seen: true,
myType: 'O',
},
})
</script>
</body>
</html>
v-for
key 속성
v-if와 v-for 와 동시에 사용하지 말 것!
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
<div id="app">
<h2>String</h2>
<p v-for="(char, idx) in message" v-bind:key="idx">
{{ idx }} {{ char }}
</p>
<h2>Array</h2>
<p v-for="(todo, idx) in todos" v-bind:key="idx">
{{ todo.title }}
</p>
<h2>Object</h2>
<p v-for="(value, idx) in myObj" v-bind:key="idx">
{{ value }}
</p>
</div>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
<script>
const app = new Vue({
el: '#app',
data: {
message:' Hello, Vue!',
todos: [
{ title:'점심 먹기'},
{ title: 'JS복습하기'},
],
myObj: {
name: 'Lee',
age: 100,
},
},
})
</script>
</body>
</html>
v-on(중요 많이 씌임)
v-on:clikc => @click 너무많이쓰여서
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
<div id="app">
<!-- 메서드 핸들러 -->
<button v-on:click="onClick">Show Message</button>
<button @click="onClick">Show Message</button>
<!-- 기본 동작 방지 (이벤트가 발동했을때)-->
<form @submit.prevent>
<button>submit</button>
</form>
<!-- 키 별칭을 이용한 키 입력 수식어 (때질때 발동)-->
<input type="text" @keyup.enter="onInput">
<input type="text" @keyup.enter="onInput2($event, '값')">
</div>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
<script>
const app = new Vue({
el:'#app',
data:{
message: 'Hello, Vue!',
},
methods: {
onClick: function() {
alert(this.message)
},
onInput: function(e) {
console.log(e.target.value)
},
onInput2: function(e, value) {
console.log(e, value)
},
},
})
</script>
</body>
</html>
v-bind
https://kr.vuejs.org/v2/guide/index.html
class와 속성의 bind
동적으로 클래스 바인딩을 주고싶다
HTML요소의 속성에 Vue의 상태 데이터를 값으로 할당
Object 형태로 사용하면 value가 true인 key가 class 바인딩 값으로 할당
약어
- : (콜론)
- v-bind:href => :href
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<style>
.active {
color: red;
}
.my-background-color {
background-color: yellow;
}
</style>
</head>
<body>
<div id="app">
<!-- 속성 바인딩 -->
<img v-bind:src="imgSrc" alt="">
<img :src="imgSrc" alt="">
<hr>
<!-- 클래스 바인딩 -->
<div :class="{ active:isRed }">클래스 바인딩</div>
<button @Click="toggleIsRed">toggle</button>
<div :class="[activeRed, myBackground]">클래스 바인딩2</div>
<hr>
<!-- 스타일 바인딩 -->
<ul>
<li
:style="{ fontSize: fontSize + 'px' }"
:class="{ active: todo.isActive }"
>
{{ todo.title }}
</li>
</ul>
</div>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
<script>
const app = new Vue({
el: '#app',
data : {
imgSrc: '',
isRed: false,
activeRed: 'active',
myBackground: 'my-background-color',
todoList: [
{ id:1, title: '여진님 퀴즈에 답하기', isActive: false },
{ id:1, title: '보충 수업하기', isActive: true },
],
fontSize: '48px',
},
methods: {
toggleIsRed: function() {
this.isRed = !this.isRed
},
},
})
</script>
</body>
</html>
v-model
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
<div id="app">
<!-- 1. input = > data -->
<h1> input => data</h1>
<p>{{ message }}</p>
<input @input="onInput" type="text">
<!-- 2. input <=> data -->
<h1> input <=> data</h1>
<p>{{ message2 }}</p>
<input v-model="message2" type="text">
<!-- 3. CheckBox -->
<!-- input id와 label의 for 는 같아야한다. -->
<h1>CheckBox</h1>
<input type="checkbox" id="checkbox" v-model="checked">
<label for="checkbox">{{ checked }}</label>
</div>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
<script>
const app = new Vue({
el: '#app',
data: {
message:'',
message2:'',
checked:false,
},
methods: {
onInput: function(event){
//console.log(event)
this.message = event.target.value
},
},
})
</script>
</body>
</html>
computed 와 methods 의 차이
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
<div id="app">
<p>{{ message }}</p>
<p>{{ reversedMessageComputed }}</p>
<p>{{ reversedMessageMethods() }}</p>
<input type="text" v-model="message">
<p>{{ count }}</p>
<button @click="onClick">증가!</button>
</div>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
<script>
const app = new Vue({
el: '#app',
data: {
count: 0,
message: '안녕하세요!'
},
computed: {
reversedMessageComputed: function() {
// 1. 캐싱됨
// 2. 반드시 리턴 필요
// 3. 종속 대상이 바뀔때만 호출
return this.message.split('').reverse().join('')
},
},
methods: {
reversedMessageMethods: function() {
this.count += 1
// ...
console.log('methods실행됨.')
return this.message.split('').reverse().join('')
},
onClick: function() {
this.count += 1
},
},
})
</script>
</body>
</html>
watch
filters
말그대로 필터
Lifecycle Hooks
태어났을떄 자동으로 호출되게하기
반드시 함수호출 ( 메소드랑 같은 라인에 적어주기)
'SSAFY > Django' 카테고리의 다른 글
[Django] Dog api(댕댕이 랜덤 추출기) html 만들기! (0) | 2021.05.03 |
---|---|
[Django] 게시판, 좋아요, 댓글, 계정 기능 종합 (0) | 2021.05.02 |
Dom 실무? (0) | 2021.04.29 |
[Django] Dom (0) | 2021.04.28 |
[Django] swagger (0) | 2021.04.27 |