Vue.js を1から学んでみた。にて書いてきましたが、
長くなり編集しにくくなってきたので、分割しました(こんなやりかたしないのかな?知りたい。)
7.Vueコンポーネント
基本
前提:コンポーネントを使わない場合
Vueインスタンスを一つ作成した場合、DOMとして展開されるのは一つだけ。
<script src="https://cdn.jsdelivr.net/npm/vue"></script>
<div id="app7-1"></div>
<!-- これは二つ目なので無視される --> 
<div id="app7-1"></div>
<p>これは表示される</p>
var vm = new Vue({
  el: '#app7-1',
  data: {
    message: '1つしか表示されないはず'
  },
  // template→renderに
  render: function(createElement) {
  	return createElement('h1', this.message)
  }
})
上記のように、vmインスタンスをhtmlで使い回すことができない。。
命名規則
ケバブケースかパスカルケースならOK。
- ケバブケース: first-name
- パスカルケース: FirstName
どっちかに統一させることが大事。
パスカルケースでおすすめ。理由は以下の通り。
- JavaScriptでケバブケースは使わないため。
- HTMLタグと見分けつきやすい。パスカルケースは、Vueコンポーネントとわかりやすい。
コンポーネント使用例
同じインスタンスを複数使う場合、Vueコンポーネントを使用する
<script src="https://cdn.jsdelivr.net/npm/vue"></script>
<div id="app7-2">
  <Component-name></component-name>
  <component-name></component-name>
  <component-name></component-name>
  
  <!-- これはvmインスタンスができてから宣言されるため、使えない -->
  <component-name2></component-name2>
</div>
<p>最下層</p>
<!-- 以下はマウントされていないので使えない -->
<component-name></component-name>
// 第1引数:コンポーネント名
Vue.component('component-name', {
  // 第2引数:Vueコンポーネント用の書き方色々
  // 今回は簡単にtemplateだけ
  template: '<p>2つ以上表示される</p>'
});
var vm = new Vue({
  el: '#app7-2',
  data: {
    message: '1つしか表示されないはず'
  }
});
// 上から呼ばれるので、#app7-2では使えない
Vue.component('component-name2', {
  // 第2引数:Vueコンポーネント用の書き方色々
  // 今回は簡単にtemplateだけ
  template: '<p>これは表示されないはず</p>'
});
dataは関数にする
Vue.jsの仕様として、コンポーネントのdataは関数にする必要がある。
なぜ関数にする必要があるかというと、仮にdataがオブジェクトだったとすると全てのインスタンスでdataが共有されてしまうため。
コンポーネント化することで複数のインスタンスを作成することができるようになったが、
そのインスタンスのフィールドの値が全て同じだったら意味がない。
よって、関数にする必要があります。
<script src="https://cdn.jsdelivr.net/npm/vue"></script>
<div id="app7-3">
  <component-name></component-name>
</div>
Vue.component('component-name', {
  // コンポーネントではdataを関数にする
  data: function() {
    return {
      message: 'コンポーネントdata'
    }
  },
  template: '<p>{{message}}</p>'
});
var vm = new Vue({
  el: '#app7-3',
});
グローバル登録・ローカル登録
sample7-3.jsのように、Vueコンポーネントを定義したより後ろのVueインスタンス(インスタンス名:vm)で使えるようにする登録方法をグローバル登録という。
それに対して、指定したVueインスタンスに直接Vueコンポーネントを定義できる。この登録方法をローカル登録という。
<script src="https://cdn.jsdelivr.net/npm/vue"></script>
<div id="app7-4-1">
  <!-- ローカル登録のコンポーネント名が勝つ -->
  <component-name></component-name>
</div>
<hr>
<div id="app7-4-2">
  <!-- グローバル登録のコンポーネントが表示される -->
  <component-name></component-name>
</div>
// コンポーネントのグローバル登録
Vue.component('component-name', {
  // コンポーネントではdataを関数にする
  data: function() {
    return {
      message: 'グローバル登録'
    }
  },
  template: '<p>{{message}}</p>'
});
// ローカル登録では、コンポーネントを変数で定義する
var component = {
  // コンポーネントではdataを関数にする
  data: function() {
    return {
      message: 'ローカル登録'
    }
  },
  template: '<p>{{message}}</p>'
}
var vm1 = new Vue({
  el: '#app7-4-1',
  // コンポーネントのローカル登録
  components: {
    'component-name' : component
  }
});
var vm2 = new Vue({
  el: '#app7-4-2',
});



