未経験からフロントエンドエンジニアに転職し、Vue.jsを勉強中なので、勉強したことの備忘録。
初投稿です。誤りやアドバイス等あればご指摘ください。
Vue.js とは
JavaScriptのフレームワーク。
コンポーネントとして各プログラムを部品化し、再利用することができる。
Vue.js はDOM操作を自動で行ってくれるため、コードが複雑になりにくいのが利点。
コンポーネントはHTMLとJavaScriptをセットにできる(単一ファイルコンポーネントならCSSもセットにできる)。
Vue.js でHello, World を表示させる
<div id="app">
<p>{{ message }}<p>
</div>
var app = new Vue({
el: "#app",
data: {
message: "Hello, World"
}
Vue.js のディレクティブ一覧
v- から始まるVue.js 独特の属性。
v-bind
要素の属性値に、Vueインスタンスのデータを結びつける
:
で省略表記できる。
<div id="app">
<a v-bind:href="url">google</a>
</div>
var app = new Vue({
el: "#app",
data: {
url: "https://www.google.com/"
}
v-on
クリックした時などのDOMイベントの受け取りを行う。
@
で省略表記できる。
<div id="app">
<button v-on:click="greet">イベント発生</button>
</div>
var app = new Vue({
el: "#app",
methods: {
greet: function() {
alert('Hello!');
}
v-if, v-show
条件分岐を記述する。
・v-if
<div id="app">
<div v-if="type === 'A'">A</div>
<div v-else-if="type === 'B'">B</div>
<div v-else>C</div>
</div>
new Vue({
el: "#app",
data: {
type: "B" // Bが表示される
}
})
・v-show
<div id="app">
<div v-show="type === 'A'">A</div>
<div v-show="type === 'B'">B</div>
<div v-show="type === 'C'">C</div>
</div>
new Vue({
el: "#app",
data: {
type: "C" //Cが表示される
}
})
v-ifによる条件を満たさなかった場合は要素はコメント化され、削除される。
v-showによる条件を満たさなかった場合は、要素にdisplay: none;
が付与される。
v-for
dataオプションに登録された配列やオブジェクトから要素を繰り返して表示する。
下記ではanimal
という変数に代入しているが、何でも良い。
<div id="app">
<ul>
<li v-for="animal in animals" :key="animal.id">
{{ animal }}
</li>
</ul>
</div>
new Vue({
el: "#app",
data: {
animals: ["cat", "dog", "bird", "monkey"]
}
})
v-model
双方向データバインディングを行う。フォームの入力値と同期させたいデータを同期させる。
<div id="app">
<input type="text" v-model="message">
<p>{{ message }}</p>
</div>
new Vue({
el: "#app",
data: {
message: 'hogehoge'
}
})
オプションの構成
computed(算出プロパティ)
methods
と似ているが違いは下記の通り。
methods
・キャッシュされない
・呼び出すときに()
が必要
・再描画が起きるたびに更新される
computed
・キャッシュされる
・呼び出すときに()
が不要
・依存するVueインスタンスの値が更新された時のみ更新される
<div id="app">
<div>
縦:<input type="text" v-model="height">
横:<input type="text" v-model="width">
</div>
<p>面積: {{ squareArea }}</p>
</div>
new Vue({
el: "#app",
data: {
height: "",
width: ""
},
computed: {
squareArea: function () {
return this.height * this.width
}
}
})
watch
特定のデータまたは算出プロパティの状態を監視し、変化があったときに登録した処理を自動で実行する。
<div id="app">
<div class="form-group">
名前:<input type="text" id="name" v-model="name">
<p class="error" v-if="error.name">{{ error.name }}</p>
</div>
</div>
new Vue({
el: '#app',
data: {
name: '',
error: {}
},
watch: {
name: function (value) {
if (value === '') {
this.error.name = '名前は入力してね'
} else {
this.error.name = ''
}
}
}
})
ライフサイクルフック
公式ドキュメントがわかりやすかった。
ライフサイクルダイアグラム
#コンポーネント
機能を持つ部品ごとに切り分けて、管理する仕組み。
部品ごとに管理することにより、再利用やカスタマイズが容易になる。
methods
やcomputed
などのオプションを持たせられる。
グローバル登録
自動的に全てのコンポーネントから使用できる
<div id="app">
<button-hoge></button-hoge>
</div>
Vue.component("button-hoge", {
data: function () {
return { count: 0 };
},
methods: {
countUp: function () {
this.count += 1;
}
},
template: `
<button @click="countUp()">
{{ count }} 回押した
</button>
`
});
new Vue({
el: '#app'
})
ローカル登録
特定のコンポーネントに登録する。ローカルとして登録され、特定のコンポーネントのスコープ内でのみ使用できる。
<div id="app">
<btn-hoge></btn-hoge>
</div>
var btnHoge = {
template: `
<button>hoge</button>
`
};
new Vue({
el: '#app',
components: {
'btn-hoge': btnHoge
}
})
コンポーネントは、new Vue
する前に定義する。
ルート要素は単一にすること。複数の要素があるときは<div>
などで囲んであげる。
親子コンポーネント間のデータのやり取り
親→子
単方向なので、子から親へデータを渡すことはできない。
親が属性で渡して、子がprops
で受け取る
<!-- 親コンポーネント -->
<div id="app">
<child hoge="hoge"></child>
<child hoge="fuga"></child>
</div>
// 子コンポーネント
Vue.component('child', {
template: '<span>{{ hoge }}</span>',
props: ['hoge']
})
new Vue({
el: '#app'
})
・props
は受け取りデータ型を指定することができる。
以下、公式ドキュメントより引用
Vue.component('my-component', {
props: {
// 基本的な型の検査 (`null` と `undefined` は全てのバリデーションにパスします)
propA: Number,
// 複数の型の許容
propB: [String, Number],
// 文字列型を必須で要求する
propC: {
type: String,
required: true
},
// デフォルト値つきの数値型
propD: {
type: Number,
default: 100
},
// デフォルト値つきのオブジェクト型
propE: {
type: Object,
// オブジェクトもしくは配列のデフォルト値は
// 必ずそれを生み出すための関数を返す必要があります。
default: function () {
return { message: 'hello' }
}
},
// カスタマイズしたバリデーション関数
propF: {
validator: function (value) {
// プロパティの値は、必ずいずれかの文字列でなければならない
return ['success', 'warning', 'danger'].indexOf(value) !== -1
}
}
}
})
子→親
子が$emit
で渡して、親がon
で受け取る。
<body>
<div id="app">
<child @childs-event="parentMethod"></child>
</div>
// 子がイベントを起こす
Vue.component('child', {
template: '<button @click="clickEvent">event!</button>',
methods: {
clickEvent: function () {
this.$emit('childs-event')
}
}
})
// 親が受け取る
new Vue({
el: '#app',
methods: {
parentMethod: function () {
console.log("クリックされました");
}
}
})
終わりに
Vuex、Vue Router については勉強中。随時、Qiitaなどでアウトプットしながら、身につけていく。
参考資料
この記事は以下の情報を参考にして執筆しました。
https://jp.vuejs.org/
[基礎から学ぶVue.js](https://www.amazon.co.jp/dp/4863542453/ref=cm_sw_r_tw_dp_U_x_PbWqDb6FDT5S5 via @amazonJP)