2017-02-11 5 views
0

Я использую Vuejs и dataTable для одного из моих проектов. Я делаю вызов Ajax и выталкиваю данные в массив. После этого я использую v-for для циклирования данных в теге <tr>. Большую часть времени он не работает. Таблица загружается, как только страница закончила загрузку. Для получения данных ajax требуется немного времени. Вот результат. Он говорит, что нет данных в таблице enter image description hereVuejs Ajax call and dataTable

Таким образом, этот вариант не работает должным образом. Я думал использовать функцию setTimeout (что было плохой идеей), чтобы загрузить таблицу через некоторое время. Каким будет правильный способ сделать это? Разделяя код:

new Vue({ 
       el: '#app', 
       data: { 
        entries: [], 
       }, 
       methods:{ 
        getData(){ 
         var route = '/admin/temporary-enrolled-students'; 
         this.$http.get(route).then((response)=>{ 
          for(var i = 0; i< response.data.length;i++) 
          { 
           this.entries.push({ 
            scId: response.data[i].id, 
            name: response.data[i].user.name, 
            phone: response.data[i].user.phone, 
            email: response.data[i].user.email, 
            courseId: response.data[i].course.id, 
            courseName: response.data[i].course.course_title, 
            expiryDate: response.data[i].expiry_date, 
            shares: response.data[i].number_of_shares, 
            expired: (response.data[i].expired == 1), 
            enrollDate: response.data[i].created_at 
           }) 
          } 

         }) 
        }, 
       }, 
       mounted(){ 
        this.getData(); 
       }, 
      }); 
//data table 
$(function() { 
      setTimeout(()=> {   
       $("#temp-enroll").DataTable({ 
        "paging": true, 
        "ordering": false, 
        "info": true, 
        "autoWidth": false 
       }); 
      },1000); 
     }); 

в лезвии:

ответ

1

Хорошо, я пробовал это и работал именно то, что хотел. Спасибо всем за поддержку.

new Vue({ 
       el: '#app', 
       data: { 
        entries: [], 
       }, 
       methods:{ 
        getData(){ 
         var route = '/admin/temporary-enrolled-students'; 
         this.$http.get(route).then((response)=>{ 
          for(var i = 0; i< response.data.length;i++) 
          { 
           this.entries.push({ 
            scId: response.data[i].id, 
            name: response.data[i].user.name, 
            ............................ 
            ...................... 
            enrollDate: response.data[i].created_at 
           }) 
          } 

         }).then(()=>{ 
          $("#temp-enroll").DataTable({ 
          "paging": true, 
          "ordering": false, 
          "info": true, 
          "autoWidth": false 
          }); 
          }); 
        }, 
       }, 
       mounted(){ 
        this.getData(); 
       }, 
      }); 
+1

Примечание для тех, кто интересуется, как же '$ http' прийти на месте! Используйте 'vue-resource', и вы можете обратиться сюда: https://github.com/pagekit/vue-resource – Syed

0

Вы можете попробовать посмотреть недвижимость https://vuejs.org/v2/guide/computed.html#Watchers. Из документа doc:

В то время как вычисленные свойства более подходят, в большинстве случаев - это время, когда необходим пользовательский наблюдатель. Вот почему Vue предоставляет более общий способ реагировать на изменения данных через опцию watch. Это наиболее полезно, если вы хотите выполнить асинхронные или дорогие операции в ответ на изменение данных.

0

Я думаю, что кто-то упомянул, что вычисленное свойство - это способ пойти, и это то, что я бы рекомендовал.

Причина в том, что, как только ваш шаблон ссылается на вычисленное свойство, это вызывает ваш вызов ajax. Затем страница отобразит любые другие доступные элементы, и когда ваш вызов ajax вернется, он затем отобразит полученные данные. Никаких специальных действий в коде не требуется.

Во всяком случае, вот что это будет выглядеть ...

В вашем HTML, как пример ...

<table> 
     <tr v-for="item in serverData"> 
      <td>{{item.name}}</td> 
      <td>{{item.someOtherValue}}</td> 
     </tr> 
    </table> 

export default { 
    name: 'pets', 
    data() { 
    return { 
    } 
    }, 
    computed: { 
    serverData: function() { 
     // ajax call using axios - returns a json array [] 
     axios.get("http://localhost:8080/getSomeData") 
     .success(function(data) { 
      return data; 
     }; 
    } 
} 

}

Обратите внимание на возвращение в обещание успеха возвращается из вызова ajax - нет необходимости хранить данные в другой переменной (и вам определенно не нужно будет прокручивать ее, как показывает приведенный выше пример кода noob. Сделайте это на сервере, верните массив, готовый к рендерингу - если он это не ваш веб-сервис, а затем напишите свой собственный, который вызывает оригинал и делать манипуляции с данными там).

Детали вашего шаблона, ссылающиеся на вычисленное значение, будут отображаться после поступления данных - нет необходимости смотреть его.

Надеюсь, это поможет!

Vue - это бомба!

0
<template> 
    <div class="panel panel-default tablecontainer"> 
     <div class="panel-heading">ChartBookin Import</div> 
     <br> 
     <div class='col-md-12'> 
      <div class='col-md-3'></div> 
      <div class='col-md-3'> 
       <div class="panel panel-default"> 

        <div class="panel-body"> 
         <commitChart :width="150" 
            :height="150"></commitChart> 
        </div> 

       </div> 
      </div> 

      <div class='col-md-3'> 
       <div class="panel panel-default"> 

        <div class="panel-body"> 
         <InboxMessage :width="150" 
            :height="150"></InboxMessage> 
        </div> 

       </div> 
      </div> 
     </div> 
     <div class="panel-body"> 
      <div class='col-md-3'> 
       <label> by Account </label> 
       <select v-model="teamId" class="form-control" rows="3"> 
        <option VALUE="" disabled> CHOISIR UN TEAM</option> 
        <option v-for="option in options" v-bind:value="option.id">{{option.name}}</option> 
       </select> 
      </div> 
      <div class='col-md-3'> 
       <label> by Date</label> 
       <div class="form-group"> 

        <input type="text" class="form-control" name="daterange" 
          value="01/01/2017-01/31/2017"/> 
       </div> 
      </div> 
      <div class='col-md-5'></div> 
      <div class='col-md-1'> 
       <label>Records</label> 
       <div class="form-group"> 

        <select v-model="elementsPerPage" class="form-control"> 

         <option v-for="option in filtre" v-bind:value="option"> 
          {{ option }} 
         </option> 
        </select> 
       </div> 
      </div> 

      <div id="sixthTable"> 

       <table class="table-bordered table-striped table-bordered table-hover"> 
        <thead> 
        <tr> 

         <th v-for=" (key,value) in rows[0]" v-on:click="sortTable(value)">{{value}} 
          <div class="arrow" v-if="value == sortColumn" 
           v-bind:class="[ascending ? 'arrow_up' : 'arrow_down']"></div> 
         </th> 
         <th>Actions</th> 
        </tr> 
        </thead> 
        <tbody> 

        <tr v-if=" rows.length > 0" v-for="row in rows"> 
         <td v-for="(key, value) in row">{{ key }}</td> 
         <td> 
          <a :href="'/stat_booking_import/' + row.Id"> 
           <span class="glyphicon glyphicon-eye-open"></span> 
          </a> 

         </td> 
        </tr> 
        <tr v-else> No Result Founded</tr> 
        </tbody> 

       </table> 
       <div class="pagination"> 


        <div id="paginatebutton"> 
         <a href="#">&laquo;</a> 
         <a class="" v-for="i in num_pages()" 
          v-bind:class="[i == currentPage ? 'active' : '']" 
          v-on:click="change_page(i)">{{i}} 
         </a> 
         <a href="#">&raquo;</a> 
        </div> 

        <div class="col-md-12" id="paginatetexte"> 
         <p v-if="pages > (elementsPerPage*currentPage) "> 
          Showing 1 to {{elementsPerPage * currentPage }} of {{ pages }} records 
         </p> 
         <p v-else> 
          Showing 1 to {{pages}} of {{ pages }} records 
         </p> 
        </div> 

       </div> 
      </div> 

     </div> 
    </div> 


</template> 


<script> 
    import Vue from 'vue'; 
    import axios from 'axios'; 
    import CommitChart from './Mail'; 
    import InboxMessage from './Inbox'; 


    export default { 
     components: { 
      CommitChart, 
      InboxMessage 

     }, 
     data() { 
      return { 
       filtre: [10, 25, 50, 100], 
       option: 0, 
       options: [], 
       currentPage: 1, 
       elementsPerPage: 10, 
       pages: 0, 
       ascending: false, 
       sortColumn: '', 
       startdate: null, 
       enddate: null, 
       options: [], 
       teamId: null, 
       columns: [], 
       messages: [], 
       date: 0, 
       rows: {}, 
      } 
     }, 

     created() { 
      this.getData(); 
      this.getTeams(); 
      this.getMailInbox(); 

     }, 
     mounted() { 
      let vm = this; 
      $(document).ready(function() { 
       $('input[name="daterange"]').daterangepicker({ 
        ranges: { 
         'Today': [moment(), moment()], 
         'Yesterday': [moment().subtract(1, 'days'), moment().subtract(1, 'days')], 
         'Last 7 Days': [moment().subtract(6, 'days'), moment()], 
         'This Month': [moment().startOf('month'), moment().endOf('month')] 
        }, 
        locale: { 
         format: 'YYYY-MM-DD' 
        }, 
       }); 

       $('.applyBtn').click(function() { 
        vm.startdate = $("input[name=daterangepicker_start]").val(); 
        vm.enddate = $("input[name=daterangepicker_end]").val(); 
        vm.getData(); 


       }); 
       $('input[name="daterange"]').on('apply.daterangepicker', function (ev, picker) { 
        vm.startdate = $("input[name=daterangepicker_start]").val(); 
        vm.enddate = $("input[name=daterangepicker_end]").val(); 
        vm.getData(); 
       }); 

      }); 
     }, 
     watch: { 
      date: function() { 
       this.getData(); 
      }, 
      teamId: function() { 
       this.getData(); 
      }, 
      elementsPerPage: function() { 
       this.getData(); 
      } 


     }, 
     methods: { 

      getData() { 

       axios.get(`/admin/stat_booking_import.json/` + this.startdate + `/` + this.teamId + `/` + this.enddate + `/` + this.elementsPerPage + `?page=` + this.currentPage) 
        .then(response => { 
         this.rows = response.data.elements.data; 
         this.columns = Object.keys(this.rows[0]); 
         this.pages = response.data.elements.total_element; 
        }) 
        .catch(e => { 
         this.errors.push(e) 
        }) 

      }, 
      getTeams() { 
       axios.get('/admin/team.json') 
        .then(response => { 
         this.options = response.data.data; 
         this.errors = []; 

        }) 
        .catch(e => { 
         e.message = "aucun resultat trouvé essayer de choisir une autre date"; 
         this.errors.push(e) 
        }) 
      }, 
      getMailInbox() { 
       axios.get(`/mailstorage.json`) 
        .then(response => { 
         this.messages = response.data.data; 
         console.log(this.messages); 

        }) 
        .catch(e => { 
         this.errors.push(e) 
        }); 
      }, 

      sortTable(col) { 
       if (this.sortColumn === col) { 
        this.ascending = !this.ascending; 
       } else { 
        this.ascending = true; 
        this.sortColumn = col; 
       } 

       var ascending = this.ascending; 

       this.rows.sort(function (a, b) { 
        if (a[col] > b[col]) { 
         return ascending ? 1 : -1 
        } else if (a[col] < b[col]) { 
         return ascending ? -1 : 1 
        } 
        return 0; 
       }) 
      }, 
      num_pages() { 

       return Math.ceil(this.pages/this.elementsPerPage); 
      }, 
      get_rows() { 
       var start = (this.currentPage - 1) * this.elementsPerPage; 
       var end = start + this.elementsPerPage; 
       return this.rows.slice(start, end); 
      }, 
      change_page(page) { 
       this.currentPage = page; 
       this.getData(); 
      } 
     }, 

    } 


</script> 
<style type="text/css"> 
    table { 
     width: 100%; 
    } 

    table td { 
     text-align: center; 
    } 

    table td { 
     text-align: center; 
     padding: 8px; 

    } 

    table td:last-child { 
     border-right: none; 
    } 

    .pagination { 
     display: inline-block; 
    } 

    .pagination a { 
     color: #3097D1; 
     float: left; 
     padding: 8px 16px; 
     text-decoration: none; 
     transition: background-color .3s; 
     border: 1px solid #ddd; 
    } 

    .pagination a.active { 
     background-color: #3097D1; 
     color: white; 
     border: 1px solid #3097D1; 
    } 

    .arrow_down { 
     background-image: url('') 
    } 

    .arrow_up { 
     background-image: url('') 
    } 

    .arrow { 
     float: right; 
     width: 12px; 
     height: 15px; 
     background-repeat: no-repeat; 
     background-size: contain; 
     background-position-y: bottom; 
    } 

    .number { 
     display: inline-block; 
     padding: 4px 10px; 
     margin: 0px 5px; 
     cursor: pointer; 

    } 

    .number:hover, 
    .number.active { 
     background-color: #3097D1; 
     border-color: #3097D1; 
    } 

    #paginatetexte { 
     padding-top: 6%; 

import Vue from 'vue' 
 

 

 
new Vue({ 
 
    el: '#statistique', 
 
    render: h => h(require('./StatBookingImport.vue')) 
 
});
<style type="text/css"> 
 
    table { 
 
     width: 100%; 
 
    } 
 

 
    table td { 
 
     text-align: center; 
 
    } 
 

 
    table td { 
 
     text-align: center; 
 
     padding: 8px; 
 

 
    } 
 

 
    table td:last-child { 
 
     border-right: none; 
 
    } 
 

 
    .pagination { 
 
     display: inline-block; 
 
    } 
 

 
    .pagination a { 
 
     color: #3097D1; 
 
     float: left; 
 
     padding: 8px 16px; 
 
     text-decoration: none; 
 
     transition: background-color .3s; 
 
     border: 1px solid #ddd; 
 
    } 
 

 
    .pagination a.active { 
 
     background-color: #3097D1; 
 
     color: white; 
 
     border: 1px solid #3097D1; 
 
    } 
 

 
    .arrow_down { 
 
     background-image: url('') 
 
    } 
 

 
    .arrow_up { 
 
     background-image: url('') 
 
    } 
 

 
    .arrow { 
 
     float: right; 
 
     width: 12px; 
 
     height: 15px; 
 
     background-repeat: no-repeat; 
 
     background-size: contain; 
 
     background-position-y: bottom; 
 
    } 
 

 
    .number { 
 
     display: inline-block; 
 
     padding: 4px 10px; 
 
     margin: 0px 5px; 
 
     cursor: pointer; 
 

 
    } 
 

 
    .number:hover, 
 
    .number.active { 
 
     background-color: #3097D1; 
 
     border-color: #3097D1; 
 
    } 
 

 
    #paginatetexte { 
 
     padding-top: 6%; 
 
    } 
 

 
    #paginatebutton { 
 
     border-radius: 1px; 
 
    } 
 

 
    .tablecontainer { 
 

 
     margin-right: 2%; 
 
     margin-left: 2%; 
 
    } 
 

 
    .mailinbox { 
 
     margin-left: 1%; 
 
    } 
 

 
</style>
<template> 
 
    <div class="panel panel-default tablecontainer"> 
 
     <div class="panel-heading">ChartBookin Import</div> 
 
     <br> 
 
     <div class='col-md-12'> 
 
      <div class='col-md-3'></div> 
 
      <div class='col-md-3'> 
 
       <div class="panel panel-default"> 
 

 
        <div class="panel-body"> 
 
         <commitChart :width="150" 
 
            :height="150"></commitChart> 
 
        </div> 
 

 
       </div> 
 
      </div> 
 

 
      <div class='col-md-3'> 
 
       <div class="panel panel-default"> 
 

 
        <div class="panel-body"> 
 
         <InboxMessage :width="150" 
 
            :height="150"></InboxMessage> 
 
        </div> 
 

 
       </div> 
 
      </div> 
 
     </div> 
 
     <div class="panel-body"> 
 
      <div class='col-md-3'> 
 
       <label> by Account </label> 
 
       <select v-model="teamId" class="form-control" rows="3"> 
 
        <option VALUE="" disabled> CHOISIR UN TEAM</option> 
 
        <option v-for="option in options" v-bind:value="option.id">{{option.name}}</option> 
 
       </select> 
 
      </div> 
 
      <div class='col-md-3'> 
 
       <label> by Date</label> 
 
       <div class="form-group"> 
 

 
        <input type="text" class="form-control" name="daterange" 
 
          value="01/01/2017-01/31/2017"/> 
 
       </div> 
 
      </div> 
 
      <div class='col-md-5'></div> 
 
      <div class='col-md-1'> 
 
       <label>Records</label> 
 
       <div class="form-group"> 
 

 
        <select v-model="elementsPerPage" class="form-control"> 
 

 
         <option v-for="option in filtre" v-bind:value="option"> 
 
          {{ option }} 
 
         </option> 
 
        </select> 
 
       </div> 
 
      </div> 
 

 
      <div id="sixthTable"> 
 

 
       <table class="table-bordered table-striped table-bordered table-hover"> 
 
        <thead> 
 
        <tr> 
 

 
         <th v-for=" (key,value) in rows[0]" v-on:click="sortTable(value)">{{value}} 
 
          <div class="arrow" v-if="value == sortColumn" 
 
           v-bind:class="[ascending ? 'arrow_up' : 'arrow_down']"></div> 
 
         </th> 
 
         <th>Actions</th> 
 
        </tr> 
 
        </thead> 
 
        <tbody> 
 

 
        <tr v-if=" rows.length > 0" v-for="row in rows"> 
 
         <td v-for="(key, value) in row">{{ key }}</td> 
 
         <td> 
 
          <a :href="'/stat_booking_import/' + row.Id"> 
 
           <span class="glyphicon glyphicon-eye-open"></span> 
 
          </a> 
 

 
         </td> 
 
        </tr> 
 
        <tr v-else> No Result Founded</tr> 
 
        </tbody> 
 

 
       </table> 
 
       <div class="pagination"> 
 

 

 
        <div id="paginatebutton"> 
 
         <a href="#">&laquo;</a> 
 
         <a class="" v-for="i in num_pages()" 
 
          v-bind:class="[i == currentPage ? 'active' : '']" 
 
          v-on:click="change_page(i)">{{i}} 
 
         </a> 
 
         <a href="#">&raquo;</a> 
 
        </div> 
 

 
        <div class="col-md-12" id="paginatetexte"> 
 
         <p v-if="pages > (elementsPerPage*currentPage) "> 
 
          Showing 1 to {{elementsPerPage * currentPage }} of {{ pages }} records 
 
         </p> 
 
         <p v-else> 
 
          Showing 1 to {{pages}} of {{ pages }} records 
 
         </p> 
 
        </div> 
 

 
       </div> 
 
      </div> 
 

 
     </div> 
 
    </div> 
 

 

 
</template> 
 

 

 
<script> 
 
    import Vue from 'vue'; 
 
    import axios from 'axios'; 
 
    import CommitChart from './Mail'; 
 
    import InboxMessage from './Inbox'; 
 

 

 
    export default { 
 
     components: { 
 
      CommitChart, 
 
      InboxMessage 
 

 
     }, 
 
     data() { 
 
      return { 
 
       filtre: [10, 25, 50, 100], 
 
       option: 0, 
 
       options: [], 
 
       currentPage: 1, 
 
       elementsPerPage: 10, 
 
       pages: 0, 
 
       ascending: false, 
 
       sortColumn: '', 
 
       startdate: null, 
 
       enddate: null, 
 
       options: [], 
 
       teamId: null, 
 
       columns: [], 
 
       messages: [], 
 
       date: 0, 
 
       rows: {}, 
 
      } 
 
     }, 
 

 
     created() { 
 
      this.getData(); 
 
      this.getTeams(); 
 
      this.getMailInbox(); 
 

 
     }, 
 
     mounted() { 
 
      let vm = this; 
 
      $(document).ready(function() { 
 
       $('input[name="daterange"]').daterangepicker({ 
 
        ranges: { 
 
         'Today': [moment(), moment()], 
 
         'Yesterday': [moment().subtract(1, 'days'), moment().subtract(1, 'days')], 
 
         'Last 7 Days': [moment().subtract(6, 'days'), moment()], 
 
         'This Month': [moment().startOf('month'), moment().endOf('month')] 
 
        }, 
 
        locale: { 
 
         format: 'YYYY-MM-DD' 
 
        }, 
 
       }); 
 

 
       $('.applyBtn').click(function() { 
 
        vm.startdate = $("input[name=daterangepicker_start]").val(); 
 
        vm.enddate = $("input[name=daterangepicker_end]").val(); 
 
        vm.getData(); 
 

 

 
       }); 
 
       $('input[name="daterange"]').on('apply.daterangepicker', function (ev, picker) { 
 
        vm.startdate = $("input[name=daterangepicker_start]").val(); 
 
        vm.enddate = $("input[name=daterangepicker_end]").val(); 
 
        vm.getData(); 
 
       }); 
 

 
      }); 
 
     }, 
 
     watch: { 
 
      date: function() { 
 
       this.getData(); 
 
      }, 
 
      teamId: function() { 
 
       this.getData(); 
 
      }, 
 
      elementsPerPage: function() { 
 
       this.getData(); 
 
      } 
 

 

 
     }, 
 
     methods: { 
 

 
      getData() { 
 

 
       axios.get(`/admin/stat_booking_import.json/` + this.startdate + `/` + this.teamId + `/` + this.enddate + `/` + this.elementsPerPage + `?page=` + this.currentPage) 
 
        .then(response => { 
 
         this.rows = response.data.elements.data; 
 
         this.columns = Object.keys(this.rows[0]); 
 
         this.pages = response.data.elements.total_element; 
 
        }) 
 
        .catch(e => { 
 
         this.errors.push(e) 
 
        }) 
 

 
      }, 
 
      getTeams() { 
 
       axios.get('/admin/team.json') 
 
        .then(response => { 
 
         this.options = response.data.data; 
 
         this.errors = []; 
 

 
        }) 
 
        .catch(e => { 
 
         e.message = "aucun resultat trouvé essayer de choisir une autre date"; 
 
         this.errors.push(e) 
 
        }) 
 
      }, 
 
      getMailInbox() { 
 
       axios.get(`/mailstorage.json`) 
 
        .then(response => { 
 
         this.messages = response.data.data; 
 
         console.log(this.messages); 
 

 
        }) 
 
        .catch(e => { 
 
         this.errors.push(e) 
 
        }); 
 
      }, 
 

 
      sortTable(col) { 
 
       if (this.sortColumn === col) { 
 
        this.ascending = !this.ascending; 
 
       } else { 
 
        this.ascending = true; 
 
        this.sortColumn = col; 
 
       } 
 

 
       var ascending = this.ascending; 
 

 
       this.rows.sort(function (a, b) { 
 
        if (a[col] > b[col]) { 
 
         return ascending ? 1 : -1 
 
        } else if (a[col] < b[col]) { 
 
         return ascending ? -1 : 1 
 
        } 
 
        return 0; 
 
       }) 
 
      }, 
 
      num_pages() { 
 

 
       return Math.ceil(this.pages/this.elementsPerPage); 
 
      }, 
 
      get_rows() { 
 
       var start = (this.currentPage - 1) * this.elementsPerPage; 
 
       var end = start + this.elementsPerPage; 
 
       return this.rows.slice(start, end); 
 
      }, 
 
      change_page(page) { 
 
       this.currentPage = page; 
 
       this.getData(); 
 
      } 
 
     }, 
 

 
    } 
 

 

 
</script>

} 

    #paginatebutton { 
     border-radius: 1px; 
    } 

    .tablecontainer { 

     margin-right: 2%; 
     margin-left: 2%; 
    } 

    .mailinbox { 
     margin-left: 1%; 
    } 

</style>