天天看點

Vue學習記錄--父子元件之間傳值以及案例評論清單的實作

父元件向子元件傳值

  1. 元件執行個體定義方式,注意:一定要使用​

    ​props​

    ​屬性來定義父元件傳遞過來的資料
<script>
    // 建立 Vue 執行個體,得到 ViewModel
    var vm = new Vue({
      el: '#app',
      data: {
        msg: '這是父元件中的消息'
      },
      components: {
        son: {
          template: '<h1>這是子元件 --- {{finfo}}</h1>',
          props: ['finfo']
        }
      }
    });
  </script>      
  1. 使用​

    ​v-bind​

    ​或簡化指令,将資料傳遞到子元件中:
<div id="app">
    <son :finfo="msg"></son>
  </div>      

子元件向父元件傳值

  1. 原理:父元件将方法的引用,傳遞到子元件内部,子元件在内部調用父元件傳遞過來的方法,同時把要發送給父元件的資料,在調用方法的時候當作參數傳遞進去;
  2. 父元件将方法的引用傳遞給子元件,其中,​

    ​getMsg​

    ​​是父元件中​

    ​methods​

    ​​中定義的方法名稱,​

    ​func​

    ​是子元件調用傳遞過來方法時候的方法名稱
<son @func="getMsg"></son>      
  1. 子元件内部通過​

    ​this.$emit('方法名', 要傳遞的資料)​

    ​方式,來調用父元件中的方法,同時把資料傳遞給父元件使用
<div id="app">
    <!-- 引用父元件 -->
    <son @func="getMsg"></son>

    <!-- 元件模闆定義 -->
    <script type="x-template" id="son">
      <div>
        <input type="button" value="向父元件傳值" @click="sendMsg" />
      </div>
    </script>
  </div>

  <script>
    // 子元件的定義方式
    Vue.component('son', {
      template: '#son', // 元件模闆Id
      methods: {
        sendMsg() { // 按鈕的點選事件
          this.$emit('func', 'OK'); // 調用父元件傳遞過來的方法,同時把資料傳遞出去
        }
      }
    });

    // 建立 Vue 執行個體,得到 ViewModel
    var vm = new Vue({
      el: '#app',
      data: {},
      methods: {
        getMsg(val){ // 子元件中,通過 this.$emit() 實際調用的方法,在此進行定義
          alert(val);
        }
      }
    });
  </script>      
<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <meta http-equiv="X-UA-Compatible" content="ie=edge">
  <title>Document</title>
  <script src="./lib/vue-2.4.0.js"></script>
  <link rel="stylesheet" href="./lib/bootstrap-3.3.7.css">
</head>

<body>
  <div id="app">


    <cmt-box @func="loadComments"></cmt-box>


    <ul class="list-group">
      <li class="list-group-item" v-for="item in list" :key="item.id">
        <span class="badge">評論人: {{ item.user }}</span>
        {{ item.content }}
      </li>
    </ul>


  </div>


  <template id="tmpl">
    <div>

      <div class="form-group">
        <label>評論人:</label>
        <input type="text" class="form-control" v-model="user">
      </div>

      <div class="form-group">
        <label>評論内容:</label>
        <textarea class="form-control" v-model="content"></textarea>
      </div>

      <div class="form-group">
        <input type="button" value="發表評論" class="btn btn-primary" @click="postComment">
      </div>

    </div>
  </template>

  <script>

    var commentBox = {
      data() {
        return {
          user: '',
          content: ''
        }
      },
      template: '#tmpl',
      methods: {
        postComment() { // 發表評論的方法
          // 分析:發表評論的業務邏輯
          // 1. 評論資料存到哪裡去???   存放到了 localStorage 中  localStorage.setItem('cmts', '')
          // 2. 先組織出一個最新的評論資料對象
          // 3. 想辦法,把 第二步中,得到的評論對象,儲存到 localStorage 中:
          //  3.1 localStorage 隻支援存放字元串資料, 要先調用 JSON.stringify 
          //  3.2 在儲存 最新的 評論資料之前,要先從 localStorage 擷取到之前的評論資料(string), 轉換為 一個  數組對象, 然後,把最新的評論, push 到這個數組
          //  3.3 如果擷取到的 localStorage 中的 評論字元串,為空不存在, 則  可以 傳回一個 '[]'  讓 JSON.parse 去轉換
          //  3.4  把 最新的  評論清單數組,再次調用 JSON.stringify 轉為  數組字元串,然後調用 localStorage.setItem()

          var comment = { id: Date.now(), user: this.user, content: this.content }

          // 從 localStorage 中擷取所有的評論
          var list = JSON.parse(localStorage.getItem('cmts') || '[]')
          list.unshift(comment)
          // 重新儲存最新的 評論資料
          localStorage.setItem('cmts', JSON.stringify(list))

          this.user = this.content = ''

          // this.loadComments() // ?????
          this.$emit('func')
        }
      }
    }

    // 建立 Vue 執行個體,得到 ViewModel
    var vm = new Vue({
      el: '#app',
      data: {
        list: [
          { id: Date.now(), user: '李白', content: '天生我材必有用' },
          { id: Date.now(), user: '江小白', content: '勸君更盡一杯酒' },
          { id: Date.now(), user: '小馬', content: '我姓馬, 風吹草低見牛羊的馬' }
        ]
      },
      beforeCreate(){ // 注意:這裡不能調用 loadComments 方法,因為在執行這個鈎子函數的時候,data 和 methods 都還沒有被初始化好

      },
      created(){
        this.loadComments()
      },
      methods: {
        loadComments() { // 從本地的 localStorage 中,加載評論清單
          var list = JSON.parse(localStorage.getItem('cmts') || '[]')
          this.list = list
        }
      },
      components: {
        'cmt-box': commentBox
      }
    });
  </script>
</body>

</html>