stash
This commit is contained in:
parent
373bc7ad40
commit
a2633bff7a
4 changed files with 51 additions and 13 deletions
|
@ -1,3 +1,4 @@
|
||||||
|
from django.utils import timezone
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django_softdelete.models import SoftDeleteModel
|
from django_softdelete.models import SoftDeleteModel
|
||||||
|
|
||||||
|
@ -148,3 +149,10 @@ class ShippingVoucher(models.Model):
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.voucher + ' (' + self.type + ')'
|
return self.voucher + ' (' + self.type + ')'
|
||||||
|
|
||||||
|
def save(self, *args, **kwargs):
|
||||||
|
if self.used_at is None and self.issue_thread is not None:
|
||||||
|
self.used_at = timezone.now()
|
||||||
|
super().save(*args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -88,6 +88,12 @@ const store = createStore({
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
availableShippingVoucherTypes: state => {
|
||||||
|
return Object.keys(state.shippingVoucherTypes).map(key => {
|
||||||
|
var count = state.shippingVouchers.filter(voucher => voucher.type === key && voucher.issue_thread === null).length;
|
||||||
|
return {id: key, count: count, name: state.shippingVoucherTypes[key]};
|
||||||
|
});
|
||||||
|
},
|
||||||
layout: (state, getters) => {
|
layout: (state, getters) => {
|
||||||
if (router.currentRoute.value.query.layout)
|
if (router.currentRoute.value.query.layout)
|
||||||
return router.currentRoute.value.query.layout;
|
return router.currentRoute.value.query.layout;
|
||||||
|
@ -476,6 +482,16 @@ const store = createStore({
|
||||||
if (data && success) {
|
if (data && success) {
|
||||||
dispatch('fetchShippingVouchers');
|
dispatch('fetchShippingVouchers');
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
async claimShippingVoucher({dispatch, state}, {ticket, shipping_voucher_type}) {
|
||||||
|
const id = state.shippingVouchers.filter(voucher => voucher.type === shipping_voucher_type && voucher.issue_thread === null)[0].id;
|
||||||
|
const {
|
||||||
|
data,
|
||||||
|
success
|
||||||
|
} = await http.patch(`/2/shipping_vouchers/${id}/`, {issue_thread: ticket}, state.user.token);
|
||||||
|
if (data && success) {
|
||||||
|
dispatch('fetchShippingVouchers');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
plugins: [
|
plugins: [
|
||||||
|
|
|
@ -13,10 +13,6 @@
|
||||||
<font-awesome-icon icon="trash"/>
|
<font-awesome-icon icon="trash"/>
|
||||||
Delete
|
Delete
|
||||||
</button-->
|
</button-->
|
||||||
<ClipboardButton :payload="shippingEmail" class="btn btn-primary">
|
|
||||||
<font-awesome-icon icon="clipboard"/>
|
|
||||||
Copy DHL contact to clipboard
|
|
||||||
</ClipboardButton>
|
|
||||||
<div class="btn-group">
|
<div class="btn-group">
|
||||||
<select class="form-control" v-model="ticket.assigned_to">
|
<select class="form-control" v-model="ticket.assigned_to">
|
||||||
<option v-for="user in users" :value="user.username">{{ user.username }}</option>
|
<option v-for="user in users" :value="user.username">{{ user.username }}</option>
|
||||||
|
@ -34,6 +30,23 @@
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="card-footer d-flex justify-content-between">
|
||||||
|
<ClipboardButton :payload="shippingEmail" class="btn btn-primary">
|
||||||
|
<font-awesome-icon icon="clipboard"/>
|
||||||
|
Copy DHL contact to clipboard
|
||||||
|
</ClipboardButton>
|
||||||
|
<div class="btn-group">
|
||||||
|
<select class="form-control" v-model="shipping_voucher_type">
|
||||||
|
<option v-for="type in availableShippingVoucherTypes.filter(t=>t.count>0)"
|
||||||
|
:value="type.id">{{ type.name }}
|
||||||
|
</option>
|
||||||
|
</select>
|
||||||
|
<button class="form-control btn btn-success"
|
||||||
|
@click="claimShippingVoucher({ticket: ticket.id, shipping_voucher_type})">
|
||||||
|
Claim Shipping Voucher
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -41,15 +54,21 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import {mapActions, mapState} from 'vuex';
|
import {mapActions, mapGetters, mapState} from 'vuex';
|
||||||
import Timeline from "@/components/Timeline.vue";
|
import Timeline from "@/components/Timeline.vue";
|
||||||
import ClipboardButton from "@/components/inputs/ClipboardButton.vue";
|
import ClipboardButton from "@/components/inputs/ClipboardButton.vue";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'Ticket',
|
name: 'Ticket',
|
||||||
components: {ClipboardButton, Timeline},
|
components: {ClipboardButton, Timeline},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
shipping_voucher_type: null
|
||||||
|
}
|
||||||
|
},
|
||||||
computed: {
|
computed: {
|
||||||
...mapState(['tickets', 'state_options', 'users']),
|
...mapState(['tickets', 'state_options', 'users']),
|
||||||
|
...mapGetters(['availableShippingVoucherTypes']),
|
||||||
ticket() {
|
ticket() {
|
||||||
const id = parseInt(this.$route.params.id)
|
const id = parseInt(this.$route.params.id)
|
||||||
const ret = this.tickets.find(ticket => ticket.id === id);
|
const ret = this.tickets.find(ticket => ticket.id === id);
|
||||||
|
@ -63,6 +82,7 @@ export default {
|
||||||
methods: {
|
methods: {
|
||||||
...mapActions(['deleteItem', 'markItemReturned', 'sendMail', 'updateTicketPartial', 'postComment']),
|
...mapActions(['deleteItem', 'markItemReturned', 'sendMail', 'updateTicketPartial', 'postComment']),
|
||||||
...mapActions(['loadTickets', 'fetchTicketStates', 'loadUsers', 'scheduleAfterInit']),
|
...mapActions(['loadTickets', 'fetchTicketStates', 'loadUsers', 'scheduleAfterInit']),
|
||||||
|
...mapActions(['claimShippingVoucher']),
|
||||||
handleMail(mail) {
|
handleMail(mail) {
|
||||||
this.sendMail({
|
this.sendMail({
|
||||||
id: this.ticket.id,
|
id: this.ticket.id,
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
<h3>Shipping Vouchers</h3>
|
<h3>Shipping Vouchers</h3>
|
||||||
<div class="mt-3">
|
<div class="mt-3">
|
||||||
<h5>Shipping Voucher Types</h5>
|
<h5>Shipping Voucher Types</h5>
|
||||||
<span v-for="(type, key) in availableShippingVoucherTypes()" :key="key" class="mr-2">
|
<span v-for="(type, key) in availableShippingVoucherTypes" :key="key" class="mr-2">
|
||||||
<span v-if="type.count > 2" class="badge badge-success">{{ type.name }} - {{ type.count }}</span>
|
<span v-if="type.count > 2" class="badge badge-success">{{ type.name }} - {{ type.count }}</span>
|
||||||
<span v-else-if="type.count > 0" class="badge badge-warning" v-if="type.count > 0">
|
<span v-else-if="type.count > 0" class="badge badge-warning" v-if="type.count > 0">
|
||||||
{{ type.name }} - {{ type.count }}
|
{{ type.name }} - {{ type.count }}
|
||||||
|
@ -64,7 +64,7 @@ export default {
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
...mapState(['shippingVouchers', 'shippingVoucherTypes']),
|
...mapState(['shippingVouchers', 'shippingVoucherTypes']),
|
||||||
...mapGetters(['getEventSlug']),
|
...mapGetters(['getEventSlug', 'availableShippingVoucherTypes']),
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
...mapActions(['fetchShippingVouchers', 'createShippingVoucher']),
|
...mapActions(['fetchShippingVouchers', 'createShippingVoucher']),
|
||||||
|
@ -85,12 +85,6 @@ export default {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
availableShippingVoucherTypes() {
|
|
||||||
return Object.keys(this.shippingVoucherTypes).map(key => {
|
|
||||||
var count = this.shippingVouchers.filter(voucher => voucher.type === key && voucher.issue_thread === null).length;
|
|
||||||
return {id: key, count: count, name: this.shippingVoucherTypes[key]};
|
|
||||||
});
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
this.fetchShippingVouchers();
|
this.fetchShippingVouchers();
|
||||||
|
|
Loading…
Reference in a new issue