import ngModule from '../module'; import './style.scss'; class Controller { constructor($scope, $state, $http, vnApp, $translate) { this.$scope = $scope; this.vnApp = vnApp; this.$translate = $translate; this.$state = $state; this.$stateParams = $state.params; this.$http = $http; this.edit = {}; this.moreOptions = [ {callback: this.markAsReserved, name: 'Mark as reserved'}, {callback: this.unmarkAsReserved, name: 'Unmark as reserved'}, {callback: this.showEditDialog, name: 'Update discount', show: () => !this.hasInvoice()}, {callback: this.createClaim, name: 'Add claim'}, {callback: this.showSMSDialog, name: 'Send SMS'} ]; this._sales = []; this.imagesPath = '//verdnatura.es/vn-image-data/catalog'; } get ticket() { return this._ticket; } set ticket(value) { this._ticket = value; this.isEditable(); } get sales() { return this._sales; } set sales(value) { this._sales = value; this.refreshTotal(); } get editedPrice() { return this._editedPrice; } set editedPrice(value) { this._editedPrice = value; this.updateNewPrice(); } refreshTotal() { this.loadSubTotal(); this.loadVAT(); } loadSubTotal() { if (!this.$stateParams.id || !this.sales) return; this.$http.get(`/api/Tickets/${this.$stateParams.id}/subtotal`).then(res => { this.subtotal = res.data || 0.0; }); } getSaleTotal(sale) { if (!sale.quantity || !sale.price) return; return sale.quantity * sale.price * ((100 - sale.discount) / 100); } loadVAT() { this.VAT = 0.0; if (!this.$stateParams.id || !this.sales) return; this.$http.get(`/api/Tickets/${this.$stateParams.id}/getVAT`).then(res => { this.VAT = res.data || 0.0; }); } get total() { return this.subtotal + this.VAT; } onMoreOpen() { let options = this.moreOptions.filter(option => { const hasShowProperty = Object.hasOwnProperty.call(option, 'show'); const shouldShow = !hasShowProperty || option.show === true || typeof option.show === 'function' && option.show(); return (shouldShow && (option.always || this.isChecked)); }); this.$scope.moreButton.data = options; } onMoreChange(callback) { callback.call(this); } isEditable() { this.$http.get(`/api/Tickets/${this.$state.params.id}/isEditable`).then(res => { this.isEditable = res.data; }); } get isChecked() { if (this.sales) { for (let instance of this.sales) if (instance.checked) return true; } return false; } /** * Returns checked instances * * @return {Array} Checked instances */ getCheckedLines() { if (!this.sales) return; return this.sales.filter(sale => { return sale.checked; }); } /** * Returns an array of indexes * from checked instances * * @return {Array} Indexes of checked instances */ getCheckedLinesIndex() { if (!this.sales) return; let indexes = []; this.sales.forEach((sale, index) => { if (sale.checked) indexes.push(index); }); return indexes; } removeCheckedLines() { const sales = this.getCheckedLines(); sales.forEach(sale => { const index = this.sales.indexOf(sale); this.sales.splice(index, 1); }); this.refreshTotal(); } onStateOkClick() { let filter = {where: {code: 'OK'}, fields: ['id']}; let json = encodeURIComponent(JSON.stringify(filter)); return this.$http.get(`/api/States?filter=${json}`).then(res => { this.onStateChange(res.data[0].id); }); } onStateChange(value) { let params = {ticketFk: this.$state.params.id, stateFk: value}; this.$http.post(`/api/TicketTrackings/changeState`, params).then(() => { this.card.reload(); this.vnApp.showSuccess(this.$translate.instant('Data saved!')); }); } onRemoveLinesClick(response) { if (response === 'ACCEPT') { let sales = this.getCheckedLines(); // Remove unsaved instances sales.forEach((sale, index) => { if (!sale.id) sales.splice(index); }); let params = {sales: sales, actualTicketFk: this.ticket.id}; let query = `/api/Sales/removes`; this.$http.post(query, params).then(() => { this.removeCheckedLines(); this.vnApp.showSuccess(this.$translate.instant('Data saved!')); }); } } showRemoveLinesDialog() { this.$scope.deleteLines.show(); } showTransferPopover(event) { let filter = {clientFk: this.ticket.clientFk, ticketFk: this.ticket.id}; let json = encodeURIComponent(JSON.stringify(filter)); let query = `/api/Tickets/threeLastActive?filter=${json}`; this.$http.get(query).then(res => { this.lastThreeTickets = res.data; }); this.$scope.transfer.parent = event.target; this.$scope.transfer.show(); } checkEmptiness(receiverTicketId) { let sales = this.getCheckedLines(); let areAllSalesSelected = sales.length === this.$scope.model.data.length; this.receiverTicketId = receiverTicketId; if (areAllSalesSelected) { let query = `/api/Tickets/${this.ticket.id}/checkEmptiness`; this.$http.get(query).then(res => { if (res.data) this.$scope.deleteTicket.show(); if (!res.data) this.moveLines(false); }); } if (!areAllSalesSelected) this.moveLines(false); } moveLines(removeEmptyTicket) { let sales = this.getCheckedLines(); let currentTicketData = { currentTicketId: this.ticket.id, clientFk: this.ticket.clientFk, addressFk: this.ticket.addressFk, agencyModeFk: this.ticket.agencyModeFk, warehouseFk: this.ticket.warehouseFk }; let params = { currentTicket: currentTicketData, receiverTicket: this.receiverTicketId ? {id: this.receiverTicketId} : currentTicketData, sales: sales, removeEmptyTicket: removeEmptyTicket }; this.$http.post(`/api/Sales/moveToTicket`, params).then(res => { if (res.data) { this.receiverTicketId = null; this.goToTicket(res.data.id); } }); } createClaim() { const claim = { ticketFk: this.ticket.id, clientFk: this.ticket.clientFk, ticketCreated: this.ticket.shipped }; const sales = this.getCheckedLines(); this.$http.post(`/api/Claims/createFromSales`, {claim: claim, sales: sales}).then(res => { this.$state.go('claim.card.basicData', {id: res.data.id}); }); } goToTicket(ticketID) { this.$state.go('ticket.card.sale', {id: ticketID}); } // Slesperson Mana getManaSalespersonMana() { this.$http.get(`/api/Tickets/${this.$state.params.id}/getSalesPersonMana`).then(res => { this.mana = res.data; }); } // Item Descriptor showDescriptor(event, itemFk) { this.quicklinks = { btnThree: { icon: 'icon-transaction', state: `item.card.diary({ id: ${itemFk}, warehouseFk: ${this.ticket.warehouseFk}, ticketFk: ${this.ticket.id} })`, tooltip: 'Item diary' } }; this.$scope.descriptor.itemFk = itemFk; this.$scope.descriptor.parent = event.target; this.$scope.descriptor.show(); } onDescriptorLoad() { this.$scope.popover.relocate(); } showEditPricePopover(event, sale) { if (!this.isEditable) return; this.sale = sale; this.editedPrice = this.sale.price; this.edit = { ticketFk: this.ticket.id, id: sale.id, quantity: sale.quantity }; this.$scope.editPricePopover.parent = event.target; this.$scope.editPricePopover.show(); } updatePrice() { if (this.editedPrice != this.sale.price) { this.$http.post(`/api/Sales/${this.edit.id}/updatePrice`, {newPrice: this.editedPrice}).then(res => { if (res.data) this.sale.price = res.data.price; this.vnApp.showSuccess(this.$translate.instant('Data saved!')); }); } this.$scope.editPricePopover.hide(); } updateNewPrice() { this.newPrice = this.sale.quantity * this.editedPrice - ((this.sale.discount * (this.sale.quantity * this.editedPrice)) / 100); } showEditDiscountPopover(event, sale) { if (!this.isEditable) return; this.sale = sale; this.edit = [{ ticketFk: this.ticket.id, id: sale.id, quantity: sale.quantity, price: sale.price, discount: sale.discount }]; this.$scope.editPopover.parent = event.target; this.$scope.editPopover.show(); } showEditDialog() { this.edit = this.getCheckedLines(); this.$scope.editDialog.show(); } hideEditDialog() { this.$scope.model.refresh(); this.$scope.editDialog.hide(); } hideEditPopover() { this.$scope.model.refresh(); this.$scope.editPopover.hide(); } /* * Unmark sale as reserved */ unmarkAsReserved() { this.setReserved(false); } /* * Mark sale as reserved */ markAsReserved() { this.setReserved(true); } setReserved(reserved) { let selectedSales = this.getCheckedLines(); let params = {sales: selectedSales, ticketFk: this.ticket.id, reserved: reserved}; let reservedSales = new Map(); this.$http.post(`/api/Sales/reserve`, params).then(res => { let isReserved = res.config.data.reserved; res.config.data.sales.forEach(sale => { reservedSales.set(sale.id, {reserved: isReserved}); }); this.sales.forEach(sale => { const reservedSale = reservedSales.get(sale.id); if (reservedSale) sale.reserved = reservedSale.reserved; }); }); } newOrderFromTicket() { this.$http.post(`/api/Orders/newFromTicket`, {ticketFk: this.ticket.id}).then(res => { this.$state.go('order.card.catalog', {id: res.data}); this.vnApp.showSuccess(this.$translate.instant('Order created')); }); } showSMSDialog() { const address = this.ticket.address; const sales = this.getCheckedLines(); const items = sales.map(sale => { return `${sale.quantity} ${sale.concept}`; }); const notAvailables = items.join(', '); const params = { ticketFk: this.ticket.id, created: this.ticket.created, notAvailables }; this.newSMS = { destinationFk: this.ticket.clientFk, destination: address.mobile || null, message: this.$translate.instant('SMSAvailability', params) }; this.$scope.sms.open(); } /** * Returns if the current ticket * is already invoiced * @return {Boolean} - True if invoiced */ hasInvoice() { return this.ticket.refFk !== null; } /** * Inserts a new instance */ add() { this.$scope.model.insert({}); } /* * Creates a new sale if it's a new instance * Updates the sale quantity for existing instance */ onChangeQuantity(sale) { if (!sale.id) this.addSale(sale); else this.updateQuantity(sale); } /* * Updates a sale quantity */ updateQuantity(sale) { const data = {quantity: parseInt(sale.quantity)}; const query = `/api/Sales/${sale.id}/updateQuantity`; this.$http.post(query, data).then(() => { this.vnApp.showSuccess(this.$translate.instant('Data saved!')); }).catch(e => { this.$scope.model.refresh(); throw e; }); } /* * Updates a sale concept */ updateConcept(sale) { const data = {concept: sale.concept}; const query = `/api/Sales/${sale.id}`; this.$http.patch(query, data).then(() => { this.vnApp.showSuccess(this.$translate.instant('Data saved!')); }).catch(e => { this.$scope.model.refresh(); throw e; }); } /* * Adds a new sale */ addSale(sale) { const data = { itemId: sale.itemFk, quantity: sale.quantity }; const query = `/api/tickets/${this.ticket.id}/addSale`; this.$http.post(query, data).then(res => { if (!res.data) return; const newSale = res.data; sale.id = newSale.id; sale.image = newSale.item.image; sale.subName = newSale.item.subName; sale.concept = newSale.concept; sale.quantity = newSale.quantity; sale.discount = newSale.discount; sale.price = newSale.price; sale.item = newSale.item; this.vnApp.showSuccess(this.$translate.instant('Data saved!')); }); } } Controller.$inject = ['$scope', '$state', '$http', 'vnApp', '$translate']; ngModule.component('vnTicketSale', { template: require('./index.html'), controller: Controller, bindings: { ticket: '<' }, require: { card: '?^vnTicketCard' } });