Dynamic row in el-table Vue JS - vue.js

I'm learning Vue JS. I'm using el-table to display my data as follows:
el-table data
Basically, I'm creating a function for a button's clicking event that contains parameter using properties of the row in the table (scope.row.id) in the code:
<el-table-column align="center" prop="id" label="Chức năng" width="150">
<template slot-scope="scope">
<el-button
type="primary"
icon="el-icon-switch-button"
#click="searchImportForm(scope.row.id)"
></el-button>
</template>
</el-table-column>
Normally it works fine. However, I've recently added a new feature to the table which is the "search" feature. There is a new text box for the user to input a keyword and the table returns all the information containing that keyword.
Now, when I press the button, the function does not work anymore. And my problem is that the error tells "Cannot read property "id" of undefined".
Does anyone how to handle with dynamic data table like this ??
Update: Here is the full code of my problem
<template>
<div class="app-container">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<div class="importListHeader">
<!-- <button #click="getImportData">Lấy dữ liệu</button> -->
<div class="importList-header-box-boxID">
<div>Mã thùng</div>
<el-input v-model="searchByUser.boxID" placeholder="Nhập mã" class="importListBoxID" />
</div>
<div class="importList-header-box-productList">
<div>Tên SP</div>
<div class="importListMultiselect">
<multiselect
placeholder="Chọn sản phẩm"
v-model="searchByUser.productName"
:options="productOptions"
label="name"
></multiselect>
</div>
</div>
<div class="importList-header-box-supplier">
<div>Tên xưởng</div>
<el-input v-model="searchByUser.keyword" placeholder="Nhập tên xưởng" class="importListSupplier" />
</div>
<div class="importList-header-box-fromDate">
<div>Từ ngày</div>
<el-input
id="importListFromDate"
v-model="searchByUser.firstDate"
placeholder="dd/mm/yyyy"
class="inputBox"
/>
</div>
<div class="importList-header-box-toDate">
<div>Đến ngày</div>
<el-input
id="importListToDate"
v-model="searchByUser.lastDate"
placeholder="dd/mm/yyyy"
class="inputBox"
/>
</div>
<div class="importList-header-box">
<el-button
class="buttonImportList"
type="success"
#click="submitSearchImportInfo"
icon="el-icon-search"
></el-button>
</div>
</div>
<el-table ref="singleTable" :data="filteredLists" border fit highlight-current-row>
<el-table-column align="center" label="#" width="55" prop="id"></el-table-column>
<el-table-column align="center" label="Xưởng SX" width="200" prop="supplier"></el-table-column>
<el-table-column align="center" label="Ngày nhập kho" width="150" prop="date"></el-table-column>
<el-table-column align="center" label="Trạng thái" width="150">
<template slot-scope="scope">
<h3
v-if="scope.row.status == 'Chưa hoàn thành' "
style="background-color: #e8e8e8; border-radius: 8px; font-size: 14px"
>{{ scope.row.status }}</h3>
<h3
v-if="scope.row.status == 'Hoàn thành' "
style="background-color: #67c23a; border-radius: 8px; font-size: 14px"
>{{ scope.row.status }}</h3>
</template>
</el-table-column>
<!--
<el-table-column class-name="status-col" label="Status" width="110" align="center">
<template slot-scope="scope">
<el-tag :type="scope.row.status | statusFilter">{{ scope.row.status }}</el-tag>
</template>
</el-table-column>
-->
<el-table-column align="center" prop="id" label="Chức năng" width="150">
<template slot-scope="scope">
<el-button
type="primary"
icon="el-icon-switch-button"
#click="searchImportForm(scope.row.id)"
></el-button>
</template>
</el-table-column>
<el-table-column align="center" label="Người tạo phiếu" width="180">
<template slot-scope="scope">
<el-tag id="importListCreate">{{ scope.row.create_user }}</el-tag>
</template>
</el-table-column>
<el-table-column align="center" label="Người thực hiện" width="180">\
<template slot-scope="scope">
<el-tag id="importListConfirm">{{ scope.row.confirm_user }}</el-tag>
</template>
</el-table-column>
<el-table-column align="center" label="Ghi chú" prop="description"></el-table-column>
</el-table>
</div>
</template>
<script>
import axios from 'axios'
import Multiselect from 'vue-multiselect'
export default {
filters: {
statusFilter(status) {
const statusMap = {
published: 'success',
draft: 'gray',
deleted: 'danger'
}
return statusMap[status]
}
},
data() {
return {
searchByUser: {
keyword: '',
firstDate: '',
lastDate: '',
boxID: '',
productName: ''
},
lists: [
// {
// id: '',
// supplier: '',
// date: '',
// status: ''
// }
],
productOptions: [{}],
startDate: '00-00-0000',
endDate: '31-12-3000'
}
},
beforeRouteEnter(to, from, next) {
axios
.post('http://192.168.1.93:3000/displayAllInPaper')
// axios.post('http://192.168.1.93:3000/displayAllInPaper')
.then(res => {
console.log(res)
next(vm => {
for (var i = 0; i < res.data.length; i++) {
vm.lists.push({
id: res.data[i].id,
supplier: res.data[i].supplier,
date: res.data[i].created_at,
status:
res.data[i].cur_status == 'p'
? 'Chưa hoàn thành'
: 'Hoàn thành',
description: res.data[i].paper_desc,
create_user: res.data[i].create_user,
confirm_user: res.data[i].confirm_user
})
}
axios.post('http://192.168.1.93:3000/getProductType').then(res => {
console.log(res)
for (var i = 0; i < res.data.length; i++) {
vm.productOptions.push({
name: res.data[i].cur_name
})
}
})
})
})
},
methods: {
submitSearchImportInfo() {
axios
.post('http://192.168.1.93:3000/searchInPaperWithProduct', {
productID: this.searchByUser.boxID,
productName: this.searchByUser.productName
})
.then(res => {
console.log(res)
this.lists.splice(0, this.lists.length)
for (var i = 0; i < res.data.length; i++) {
this.lists.push({
id: res.data[i].id,
supplier: res.data[i].supplier,
date: res.data[i].created_at,
status:
res.data[i].cur_status == 'p' ? 'Chưa hoàn thành' : 'Hoàn thành'
})
}
})
},
setCurrentRow(row) {
this.$refs.singleTable.setCurrentRow(row)
},
searchImportForm(id_par) {
this.$store.state.products.splice(0, this.$store.state.products.length)
this.$store.state.inPaperID = this.filteredLists[id_par - 1].id
axios
.post(
'http://192.168.1.93:3000/getDetailInPaper',
this.filteredLists[id_par - 1]
)
.then(res => {
console.log(res)
for (var i = 0; i < res.data.length; i++) {
this.$store.state.products.push({
productID: res.data[i].id,
cur_name: res.data[i].cur_name,
perbox: res.data[i].perbox,
box_amount: res.data[i].box_amount,
scan_number: res.data[i].scan_number
})
}
})
this.$router.push({ path: '/import/details' })
},
filteredByName(lists) {
return lists.filter(list =>
list.supplier.match(this.searchByUser.keyword)
)
},
localizeDate(date) {
if (!date || !date.includes('/')) return date
const [dd, mm, yyyy] = date.split('/')
return (date = `${yyyy}-${mm}-${dd}`)
},
convertDate(date) {
const [yyyy, mm, dd] = date.split('-')
var newDate = `${yyyy}${mm}${dd}`
var newDate2 = parseInt(newDate)
return (date = newDate2)
},
filteredByDateRange(lists) {
if (this.searchByUser.firstDate != '') {
this.startDate = this.searchByUser.firstDate
}
if (this.searchByUser.lastDate != '') {
this.endDate = this.searchByUser.lastDate
}
return lists.filter(list =>
this.convertDate(list.date) >=
this.convertDate(this.localizeDate(this.startDate)) &&
this.convertDate(list.date) <=
this.convertDate(this.localizeDate(this.endDate))
? list
: ''
)
}
},
computed: {
filteredLists: function() {
// return this.filteredByDateRange(this.lists)
if (
this.searchByUser.keyword == '' &&
this.searchByUser.firstDate == '' &&
this.searchByUser.lastDate == ''
) {
return this.filteredByName(this.lists)
}
if (
this.searchByUser.keyword != '' &&
this.searchByUser.firstDate == '' &&
this.searchByUser.lastDate == ''
) {
return this.filteredByName(this.lists)
}
if (
this.searchByUser.keyword == '' &&
(this.searchByUser.firstDate != '' || this.searchByUser.lastDate != '')
) {
return this.filteredByDateRange(this.lists)
}
if (
this.searchByUser.keyword != '' &&
(this.searchByUser.firstDate != '' || this.searchByUser.lastDate != '')
) {
return this.filteredByDateRange(this.filteredByName(this.lists))
}
}
},
components: {
Multiselect
}
}
</script>
<style src="vue-multiselect/dist/vue-multiselect.min.css"></style>
<style>
.importListHeader {
max-width: 96%;
display: flex;
text-align: left;
justify-content: space-between;
}
.inputBox {
width: 250px;
height: 100px;
}
.el-dropdown-menu {
overflow-y: scroll;
}
.el-dropdown-link {
cursor: pointer;
color: #ffffff;
}
.importListMultiselect .multiselect__tags {
min-height: 40px;
display: block;
padding: 8px 40px 0 8px;
border-radius: 5px;
border: 0.5px solid #e8e8e8;
background: #fff;
font-size: 14px;
width: 320px;
}
.importListMultiselect .multiselect {
-webkit-box-sizing: content-box;
box-sizing: content-box;
display: block;
position: relative;
width: 320px;
min-height: 40px;
text-align: left;
color: #35495e;
}
#importListFromDate {
width: 200px;
}
#importListToDate {
width: 200px;
}
.buttonImportList {
position: absolute;
top: 39px;
}
.el-icon-arrow-down {
font-size: 8px;
}
.el-dropdown {
vertical-align: top;
}
.importListBoxID {
max-width: 210px;
}
.importList-header-box-boxID {
max-width: 200px;
}
.importList-header-box-productList {
max-width: 320px;
}
.importList-header-box-supplier {
max-width: 250px;
}
.importList-header-box-fromDate {
max-width: 200px;
}
.importList-header-box-toDate {
max-width: 200px;
}
#importListCreate {
font-size: 15px;
}
#importListConfirm {
font-size: 15px;
color: #35495e;
background-color: transparent;
border-style: none;
}
</style>

Related

Can not change option state data value by other values in Vue.js

I am trying to change result value between true/false by other values. I want set result to true, if all other values are not empty.
I think it looks like good, but result property returns true all time. So i can not hide result <p> tag.
<script src="https://unpkg.com/vue#3.2.37/dist/vue.global.prod.js"></script>
<div id="app">
<label>
Name :
<input type="text" v-model="name" />
</label>
<label>
Size :
<input type="text" v-model="size" />
</label>
<label>
Quantity :
<input type="text" v-model="quantity" />
</label>
<p v-if="result">Summary: {{ name }} - {{ size }} - {{ quantity }}</p>
</div>
<script>
const { createApp } = Vue;
createApp({
data() {
return {
name: '',
size: '',
quantity: '',
result: this.name !== '' && this.size !== '' && this.quantity !== '',
};
},
}).mount('#app');
</script>
<style>
#app {
font-family: Avenir, Helvetica, Arial, sans-serif;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
text-align: left;
color: #2c3e50;
margin-top: 60px;
}
label {
display: block;
margin: 0 0 10px;
}
input {
display: block;
}
</style>
In this case you should defing result as a computed property :
createApp({
data() {
return {
name: '',
size: '',
quantity: '',
};
},
computed:{
result(){
return this.name !== '' && this.size !== '' && this.quantity !== ''
}
}
}).mount('#app');

How can I add multiple product in cart using vue js and django rest framework

I am trying to add multiple product in the cart. I am using vuejs and django rest framework. My problem is: When I add a product into the cart it added successfully But when I add another product it doesnt add. It adds the same product again and again.
For example:
I have three products named "A" and "B" and "C". I added "A". Then i try to add "C" in the cart. But it stills adds "A" in the cart. I cleared the session and tried again still add the "A" product if I try to add "C" product first. It always add "A" product.
Here is my store/index.js:
import { createStore } from 'vuex'
export default createStore({
state: {
cart: {
items: []
},
isAuthenticated: false,
token: '',
isLoading: false,
},
mutations: {
initializeStore(state){
if(localStorage.getItem('cart')){
state.cart = JSON.parse(localStorage.getItem('cart'))
}
else{
localStorage.setItem('cart', JSON.stringify(state.cart))
}
},
addToCart(state, item) {
let exists = state.cart.items.filter(i => i.product.id === item.product.id)
if (exists.length){
exists[0].quantity = parseInt(exists[0].quantity) + parseInt(item.quantity)
}
else{
state.cart.items.push(item)
}
localStorage.setItem('cart', JSON.stringify(state.cart))
}
},
actions: {
},
modules: {
}
})
Here is my add to cart page and code:
<template>
<br />
<div class="col">
<div class="col1">
<img v-bind:src="product.get_image" alt="">
</div>
<div class="col2">
<div class="product__title">
<p>PRODUCT TITLE</p>
<h2>{{ product.name }}</h2>
<small>{{ product.short_description }}</small>
</div>
<div class="product__price">
<p>Price: $ {{ product.price }}</p>
</div>
<div class="product__button">
<input type="hidden" v-model="quantity" min="1">
<button type="submit" class="button__primary" #click="addToCart">Add To Cart</button>
</div>
</div>
</div>
</template>
<script>
import axios from 'axios'
import {toast} from 'bulma-toast'
export default {
name: "ProductDetail",
data () {
return {
product: {},
quantity: 1
}
},
mounted() {
this.getProduct()
},
methods: {
getProduct() {
const categorySlug = this.$route.params.category_slug
const productSlug = this.$route.params.product_slug
axios.get(`/api/product-details/${categorySlug}/${productSlug}/`)
.then(response => {
this.product = response.data
})
.catch(error => {
console.log(error)
})
},
addToCart() {
if(isNaN(this.quantity) || this.quantity < 1){
this.quantity = 1
}
const item = {
product: this.product,
quantity: this.quantity
}
this.$store.commit("addToCart", item)
toast({
message: "Product has been added to cart" + item.product.name,
type: "is-success",
pauseOnHover: true,
duration: 2000,
position: "bottom-right",
dismissible: true,
})
}
}
}
</script>
<style scoped>
.col {
display: grid;
grid-template-columns: 1fr 1fr;
grid-gap: 5rem;
height: 50vh;
padding: 60px;
}
.col1 {
height: 520px;
width: 100%;
}
.col1 img{
object-fit: fill;
height: 520px;
width: 100%;
}
.col2 {
display: grid;
grid-auto-rows: 1fr 1fr 1fr 1fr;
background: #fff;
padding: 12px;
}
.product__button button{
height: 40px;
width: 100%;
border: none;
background: #007bc4;
color: #fff;
border-radius: 3px;
}
.product__price p {
color: #007bc4;
font-size: 18px;
}
</style>
Here is the cartitem component:
<template>
<tr class="is-fullwidth">
<td>{{ item.product.name }}</td>
<td>$ {{ item.product.price }}</td>
<td>
<button #click="increment(item)" class="plusButton">+</button>
{{ item.quantity }}
<button #click="decrement(item)" class="minusButton">-</button>
</td>
<td>{{ getTotal(item).toFixed(2) }}</td>
<td><button class="delete"></button></td>
</tr>
</template>
<script>
export default {
name: "Cartitem",
props: {
initialItem: Object
},
data() {
return {
item: this.initialItem
}
},
methods: {
getTotal(item) {
return item.quantity * item.product.price
},
increment(item){
item.quantity += 1
this.updateCart()
},
decrement(item) {
item.quantity -= 1
if(item.quantity === 0){
this.$emit('removeFromCart', item)
}
this.updateCart()
},
updateCart() {
localStorage.setItem('cart', JSON.stringify(this.$store.state.cart))
},
removeFromCart(item) {
this.$emit('removeFromCart', item)
this.updateCart()
}
}
}
</script>
<style scoped>
.plusButton{
border: none;
background: #fff;
font-size: 19px;
}
.minusButton{
border: none;
background: #fff;
font-size: 19px;
}
</style>
I am new in vue js. I am trying to build this projetc so that i can learn. But this issue is eating my brain. I tried to use find function in sotre/index.js. It solved my problem though but if i clear the cookies and try to add product it gives me error.
is there any solution for me?
Thanks in advance.

My disabled button does not work, it does not take into account the style Vue.js

I had set up a disable system if my form fields are not filled in and I want to move it to a custom component but the problem is that it doesn't work anymore, I think I have targeted the problem but I don't see how to solve it. I guess because it doesn't find my this.email, this.password etc. If you can help me please
<template>
<button
class="btn"
:class="{ 'btn--disabled': !validatedFields }"
:disabled="!validatedFields"
>
<slot></slot>
</button>
</template>
<script>
export default {
props: ["disabled"],
computed: {
validatedFields() {
if (this.signup) {
if (this.email != "" && this.password != "") {
return true;
} else {
return false;
}
} else {
if (this.email != "" && this.password != "") {
return true;
} else {
return false;
}
}
},
},
};
</script>
<style lang="scss" scoped>
#import "../scss/main.scss";
.btn {
width: 100%;
margin: 1rem 0;
background: $secondary;
color: #ffff;
border: none;
padding: 1rem 2rem;
}
.btn--disabled {
color: red;
}
</style>
<template>
<section>
<div class="form">
<transition name="test" mode="out-in">
<form key="login" v-if="login" #submit.prevent="login">
<h1>Connexion</h1>
<custom-input
icon="fa-at"
name="email"
type="email"
placeholder="email#gmail.com"
#updated="updatedEmail"
/>
<custom-input
icon="fa-lock"
name="password"
type="password"
placeholder="Mot de passe"
#updated="updatedPassword"
/>
<custom-btn>Connexion</custom-btn>
<p #click="login = !login">Déjà un compte ?</p>
</form>
<form
action=""
key="signup"
v-else
novalidate
#submit.prevent="createAccount"
>
<h1>Inscription</h1>
<custom-input
icon="fa-at"
name="email"
type="email"
placeholder="email#gmail.com"
#updated="updatedEmail"
/>
<custom-input
icon="fa-lock"
name="password"
type="password"
placeholder="Mot de passe"
#updated="updatedPassword"
/>
<custom-input
icon="fa-lock"
name="RepeatPassword"
type="password"
placeholder="Répéter votre mot de passe"
#updated="updatedPassword"
/>
<custom-input
icon="fa-user"
name="name"
type="text"
placeholder="John"
#updated="updatedName"
/>
<custom-file />
<custom-btn>Inscription</custom-btn>
<p #click="login = !login">Pas de compte?</p>
</form>
</transition>
</div>
</section>
</template>
<script>
// import signup from "../assets/signup.jpg";
// import connex from "../assets/connex.jpg";
import CustomInput from "./Input.vue";
import CustomBtn from "./Button.vue";
import CustomFile from "./FileInput.vue";
export default {
components: {
CustomInput,
CustomBtn,
CustomFile,
},
data() {
return {
login: true,
file: null,
};
},
methods: {
onFileChanges(event) {
let files = event.target.files || event.dataTransfer.files;
if (!files.length) return;
this.file = files[0];
},
updatedEmail(email) {
this.email = email;
},
updatedPassword(password) {
this.password = password;
},
updatedName(name) {
this.name = name;
},
},
};
</script>
<style lang="scss" scoped>
#import "../scss/main.scss";
section {
padding: 2rem;
.form {
background: #ffff;
box-shadow: 0px 15px 16.83px 0.17px rgb(0 0 0 / 5%);
border-radius: 10px;
padding: 2rem;
h1 {
text-align: center;
font-size: 1.8rem;
margin-bottom: 1rem;
color: $secondary;
}
p {
text-align: center;
margin-top: 1rem;
}
}
.test-enter-active,
.test-leave-active {
transition: opacity 0.5s;
}
.test-enter,
.test-leave-active {
opacity: 0;
}
}
</style>

Assing uploaded file value to input field multiple times using vue.js

Assign filename to input field on upload and then show success message working fine,but when i try to delete file and upload again it's not working!
new Vue({
el: "#app",
data() {
return {
form: {
message: '',
fileurl: ''
},
loading: false
}
},
methods: {
uploadImage(event) {
this.form.fileurl = 'uploaded!'
},
deleteFile(furl) {
this.form.fileurl = ''
}
}
})
body {
background: #20262E;
padding: 20px;
font-family: Helvetica;
}
#app {
background: #fff;
border-radius: 4px;
padding: 20px;
transition: all 0.2s;
}
span {
color: red;
}
<script src="https://cdn.jsdelivr.net/npm/vue#2.5.16/dist/vue.js"></script>
<div id="app">
<h3>
File Upload / remove Demo
</h3>
<hr />
<div class="custom-file attach_file" v-show="!form.fileurl">
<input type="file" id="file" name="file" #change="uploadImage($event)">
<input type="text" v-model="form.fileurl">
</div>
<p v-if="form.fileurl"> {{ form.fileurl }} <span #click="deleteFile(form.fileurl)">Delete</span></p>
</div>
I am not getting any console error as well.
This is what i have tried so far.
Can you guys please have a look at this!
The problem is that #change is not trigger when you choose the same file. The simplest solution is reset value of input when you click delete
this.$refs.fileToUpload.value = '';
new Vue({
el: "#app",
data() {
return {
form: {
message: '',
fileurl: ''
},
loading: false
}
},
methods: {
uploadImage(event) {
this.form.fileurl = 'uploaded!'
},
deleteFile(furl) {
this.form.fileurl = ''
this.$refs.fileToUpload.value = '';
}
}
})
body {
background: #20262E;
padding: 20px;
font-family: Helvetica;
}
#app {
background: #fff;
border-radius: 4px;
padding: 20px;
transition: all 0.2s;
}
span {
color: red;
}
<script src="https://cdn.jsdelivr.net/npm/vue#2.5.16/dist/vue.js"></script>
<div id="app">
<h3>
File Upload / remove Demo
</h3>
<hr />
<div class="custom-file attach_file" v-show="!form.fileurl">
<input type="file" id="file" name="file" class="custom-file-input" #change="uploadImage($event)" ref="fileToUpload">
<input type="text" v-model="form.fileurl">
</div>
<p v-if="form.fileurl"> {{ form.fileurl }} <span #click="deleteFile(form.fileurl)">Delete</span></p>
</div>
You need to clear you input field first.
Update your upload method with this.
uploadImage(event) {
this.form.fileurl = 'uploaded!'
this.success = false
this.$nextTick(() => {
this.success = true
})
Have a look at this working demo
https://jsfiddle.net/asimshahzad/8c9pcfys/

Vuejs Ajax call and dataTable

I'm using Vuejs and dataTable for one of my project. I make an Ajax call and push data into an array. After that I use v-for to loop the data in the <tr> tag. Most of the time it doesn't work. The table loads as soon as the page has finished loading.. it takes a bit of time to receive the ajax data. Here is the output. It says no data available in the table
So the serch option doesn't work properly. I thought to use a setTimeout function (which was a bad idea) to load the table after a bit of time. What would be the proper way to do it? Sharing the code :
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);
});
in blade:
Ok I tried this and working exactly what I have wanted. Thanks everyone for supporting.
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();
},
});
I think someone mentioned that a computed property is the way to go, and that is what I would recommend as well.
The reason why is that as soon as your template makes a reference to the computed property, this triggers your ajax call. The page will then render any other available elements, and when your ajax call returns, it will then render the data it received. No special action in code is required.
Anyway, here is what it would look like...
In your html, as an example...
<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;
};
}
}
}
Notice the return in the success promise returned from the ajax call - there is no need to store the data to another variable (and you should definitely not have to loop through it like the noob coder example above shows. Do that on the server, return an array ready to render - if it isn't your web service, then write your own that calls the original and do the data manipulation there).
The parts of your template that reference the computed value will render after the data arrives - there is no need to watch it.
Hope this helps!
Vue is the bomb!
<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 class="" v-for="i in num_pages()"
v-bind:class="[i == currentPage ? 'active' : '']"
v-on:click="change_page(i)">{{i}}
</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 class="" v-for="i in num_pages()"
v-bind:class="[i == currentPage ? 'active' : '']"
v-on:click="change_page(i)">{{i}}
</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>
You could try watch property https://v2.vuejs.org/v2/guide/computed.html#Watchers. From the doc:
While computed properties are more appropriate in most cases, there
are times when a custom watcher is necessary. That’s why Vue provides
a more generic way to react to data changes through the watch option.
This is most useful when you want to perform asynchronous or expensive
operations in response to changing data.