Vue.js 中級編
おさらい
算出プロパティ
new Vue({
el: '#app',
data: {
text: 'Vue.js is awesome'
},
computed: {
upperText() {
return this.text.toUpperCase()
}
}
})
<div id="#app">
{{ upperText }}
</div>
プロパティとして
アクセスする
算出プロパティ | メソッド | |
---|---|---|
オプション名 | computed | methods |
呼び出し方 | プロパティ {{ someProp }} |
メソッド {{ someMethod() }} |
キャッシュ | される | されない |
引数 | なし | あり |
注意点
Date.now() など Vue.js 以外で値が変わるものを
算出プロパティの中で使うとうまく更新が反映されない
https://codepen.io/ktsn/pen/OoywbP?editors=1111
ウォッチャー
new Vue({
el: '#app',
data: {
text: 'hello'
},
watch: {
text(newValue, oldValue) {
// this.text が変わったときに実行される
}
}
})
new Vue({
// ...
watch: {
text: {
handler(newValue, oldValue) {
// this.text が変わったときに実行される
},
immediate: true, // 初期値設定時にも実行されるようにする
deep: true // オブジェクトの子孫も監視する
}
}
})
watch のオプション
// 文字列
vm.$watch('a.b.c', function (newVal, oldVal) {
// vm.a.b.c が変更されたら実行される
})
// 関数
vm.$watch(
function () {
return this.a + this.b
},
function (newVal, oldVal) {
// this.a + this.b の値が変わったら実行される
}
)
ライフサイクル
例えば……
- created でスライダーのページめくりのタイマーを起動
- mounted で外部の UI ライブラリをマウント
- beforeDestroy で↑のような処理を無効化
コンポーネント
Vue.component('v-button', {
template: `...`,
props: { /* ... */ },
// ... インスタンスオプションと同様 ...
}
グローバルに登録
すべてのテンプレートで使えるようにする
const VButton = {
template: `...`,
props: { /* ... */ },
// ... インスタンスオプションと同様 ...
}
// 他のコンポーネント
const AnotherComp = {
// VButton を登録
components: { VButton }
}
ローカルに登録
あるコンポーネントのテンプレートでのみ使えるようにする
Vue.component('v-button', {
props: {
// theme プロパティを定義
// `this.theme` で参照できる
theme: {
type: String,
default: null
}
},
// ... インスタンスオプションと同様 ...
})
props
コンポーネントにデータを渡す
<!-- theme に 'blue' を渡す -->
<v-button theme="blue" />
Vue.component('v-button', {
methods: {
onClick () {
// `click` イベントを送出
this.$emit('click')
}
}
// ... インスタンスオプションと同様 ...
})
カスタムイベント
コンポーネントの外側にイベントを通知
<!-- クリックを監視 -->
<v-button
v-on:click="onClickButton"
/>
Vue.component('v-button', {
// 一部省略
template: `<button>
<span>
<slot />
</span>
</button>`
// ... インスタンスオプションと同様 ...
})
スロット
コンポーネントの小要素を指定の場所に配置
<!-- 小要素が slot と差し替わる -->
<v-button>
ボタンテキスト
</v-button>
Vue.component('v-button', {
// 初期データを返す関数
data() {
return {
localData: 'hoge'
}
}
})
data の書き方の違い
初期データを返す関数にする必要がある
単一ファイル
コンポーネント
<template>
<!-- コンポーネントのテンプレート -->
</template>
<script>
// コンポーネントオプション
export default {
data () { /* ... */ }
}
</script>
<style>
/* コンポーネントのスタイル */
</style>
<template lang="pug">
//- Pug
button.button(type="button")
slot
</template>
<script lang="ts">
// TypeScript
import Vue from 'vue'
export default Vue.extend({
data () { /* ... */ }
})
</script>
<style lang="scss">
/* SCSS */
.button {
&_text {
}
}
</style>
Scoped CSS
scoped とつけるとスタイルがコンポーネント外に漏れなくなる
<template>
<button class="button">
<slot />
</button>
</template>
<style scoped>
/* このコンポーネントにのみ適用 */
.button {
background-color: blue;
}
</style>
応用
既存ライブラリをコンポーネントにする
Further Reading
- Vue.js 公式ガイド
https://jp.vuejs.org/v2/guide/
- Vue.js 公式クックブック
https://jp.vuejs.org/v2/cookbook/
- やわらかVue.js
https://scrapbox.io/vue-yawaraka/
- Vue.js入門 (技評の Web 連載)
https://gihyo.jp/dev/serial/01/vuejs
- Vue.js入門 基礎から実践アプリケーション開発まで
9月22日発売
Vue.js 中級編
By katashin
Vue.js 中級編
- 2,702