151 lines
6.7 KiB
Vue
151 lines
6.7 KiB
Vue
<template>
|
|
<AsyncLoader :loaded="ticket.id">
|
|
<div class="container-fluid px-xl-5 mt-3">
|
|
<div class="row">
|
|
<div class="col-xl-8 offset-xl-2">
|
|
<div class="card bg-dark text-light mb-2" id="filters">
|
|
<div class="card-header">
|
|
<h3>Ticket #{{ ticket.id }} - {{ ticket.name }}</h3>
|
|
</div>
|
|
<Timeline :timeline="ticket.timeline" @sendMail="handleMail" @addComment="handleComment"/>
|
|
<div class="card-footer d-flex justify-content-between">
|
|
<button class="btn btn-secondary mr-2" @click="$router.go(-1)">Back</button>
|
|
<!--button class="btn btn-danger" @click="deleteItem({type: 'tickets', id: ticket.id})">
|
|
<font-awesome-icon icon="trash"/>
|
|
Delete
|
|
</button-->
|
|
<div class="btn-group">
|
|
<select class="form-control" v-model="selected_assignee">
|
|
<option v-for="user in users" :value="user.username">{{ user.username }}</option>
|
|
</select>
|
|
<button class="form-control btn btn-success"
|
|
@click="assignTicket(ticket)"
|
|
:disabled="!selected_assignee || (selected_assignee == ticket.assigned_to)">
|
|
Assign Ticket
|
|
</button>
|
|
</div>
|
|
<div class="btn-group">
|
|
<select class="form-control" v-model="selected_state">
|
|
<option v-for="status in state_options" :value="status.value">{{
|
|
status.text
|
|
}}
|
|
</option>
|
|
</select>
|
|
<button class="form-control btn btn-success"
|
|
@click="changeTicketStatus(ticket)"
|
|
:disabled="(selected_state == ticket.state)">
|
|
Change Status
|
|
</button>
|
|
</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}).then(()=>shipping_voucher_type=null)"
|
|
:disabled="!shipping_voucher_type">
|
|
Claim Shipping Voucher
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</AsyncLoader>
|
|
</template>
|
|
|
|
<script>
|
|
import {mapActions, mapGetters, mapState} from 'vuex';
|
|
import Timeline from "@/components/Timeline.vue";
|
|
import ClipboardButton from "@/components/inputs/ClipboardButton.vue";
|
|
import AsyncLoader from "@/components/AsyncLoader.vue";
|
|
|
|
export default {
|
|
name: 'Ticket',
|
|
components: {AsyncLoader, ClipboardButton, Timeline},
|
|
data() {
|
|
return {
|
|
selected_state: null,
|
|
selected_assignee: null,
|
|
shipping_voucher_type: null,
|
|
}
|
|
},
|
|
watch: {
|
|
ticket(val) {
|
|
if (this.selected_state == null) {
|
|
this.selected_state = val.state;
|
|
}
|
|
if (this.selected_assignee == null) {
|
|
this.selected_assignee = val.assigned_to
|
|
}
|
|
}
|
|
},
|
|
computed: {
|
|
...mapState(['tickets', 'state_options', 'users']),
|
|
...mapGetters(['availableShippingVoucherTypes']),
|
|
ticket() {
|
|
const id = parseInt(this.$route.params.id)
|
|
const ret = this.tickets.find(ticket => ticket.id === id);
|
|
return ret ? ret : {};
|
|
},
|
|
shippingEmail() {
|
|
const domain = document.location.hostname;
|
|
return `ticket+${this.ticket.uuid}@${domain}`;
|
|
}
|
|
},
|
|
//this.scheduleAfterInit(() => [this.fetchTicketStates(), this.loadTickets(), this.loadUsers(),
|
|
//this.fetchShippingVouchers()]);
|
|
methods: {
|
|
...mapActions(['deleteItem', 'markItemReturned', 'sendMail', 'updateTicketPartial', 'postComment']),
|
|
...mapActions(['loadTickets', 'fetchTicketStates', 'loadUsers', 'scheduleAfterInit']),
|
|
...mapActions(['claimShippingVoucher', 'fetchShippingVouchers']),
|
|
handleMail(mail) {
|
|
this.sendMail({
|
|
id: this.ticket.id,
|
|
message: mail
|
|
})
|
|
},
|
|
handleComment(comment) {
|
|
this.postComment({
|
|
id: this.ticket.id,
|
|
message: comment
|
|
})
|
|
},
|
|
changeTicketStatus(ticket) {
|
|
ticket.state = this.selected_state;
|
|
this.updateTicketPartial({
|
|
id: ticket.id,
|
|
state: this.selected_state,
|
|
})
|
|
},
|
|
assignTicket(ticket) {
|
|
ticket.assigned_to = this.selected_assignee;
|
|
this.updateTicketPartial({
|
|
id: ticket.id,
|
|
assigned_to: this.selected_assignee
|
|
})
|
|
},
|
|
},
|
|
mounted() {
|
|
this.scheduleAfterInit(() => [Promise.all([this.fetchTicketStates(), this.loadTickets(), this.loadUsers(), this.fetchShippingVouchers()]).then(()=>{
|
|
this.selected_state = this.ticket.state;
|
|
this.selected_assignee = this.ticket.assigned_to
|
|
})]);
|
|
//this.scheduleAfterInit(() => [this.fetchTicketStates(), this.loadTickets(), this.loadUsers(),
|
|
//this.fetchShippingVouchers()]);
|
|
}
|
|
};
|
|
</script>
|
|
|
|
<style scoped>
|
|
|
|
</style>
|