327 lines
12 KiB
Vue
327 lines
12 KiB
Vue
<template>
|
|
<UDashboardNavbar :badge="filteredRows.length" title="Ausgangsbelege">
|
|
<template #right>
|
|
<UInput
|
|
id="searchinput"
|
|
v-model="searchString"
|
|
autocomplete="off"
|
|
class="hidden lg:block"
|
|
icon="i-heroicons-funnel"
|
|
placeholder="Suche..."
|
|
@keydown.esc="$event.target.blur()"
|
|
>
|
|
<template #trailing>
|
|
<UKbd value="/"/>
|
|
</template>
|
|
</UInput>
|
|
<UButton
|
|
v-if="searchString.length > 0"
|
|
color="rose"
|
|
icon="i-heroicons-x-mark"
|
|
variant="outline"
|
|
@click="clearSearchString()"
|
|
/>
|
|
<UButton
|
|
@click="router.push(`/createDocument/edit`)"
|
|
>
|
|
+ Ausgangsbeleg
|
|
</UButton>
|
|
</template>
|
|
</UDashboardNavbar>
|
|
<UDashboardToolbar>
|
|
<template #right>
|
|
<USelectMenu
|
|
v-if="selectableFilters.length > 0"
|
|
v-model="selectedFilters"
|
|
:color="selectedFilters.length > 0 ? 'primary' : 'white'"
|
|
:options="selectableFilters"
|
|
:ui-menu="{ width: 'min-w-max' }"
|
|
icon="i-heroicons-adjustments-horizontal-solid"
|
|
multiple
|
|
>
|
|
<template #label>
|
|
Filter
|
|
</template>
|
|
</USelectMenu>
|
|
</template>
|
|
</UDashboardToolbar>
|
|
<UTabs :items="selectedTypes" class="m-3">
|
|
<template #default="{item}">
|
|
{{ item.label }}
|
|
<UBadge
|
|
class="ml-2"
|
|
variant="outline"
|
|
>
|
|
{{ getRowsForTab(item.key).length }}
|
|
</UBadge>
|
|
</template>
|
|
<template #item="{item}">
|
|
<div style="height: 80vh; overflow-y: scroll">
|
|
<UTable
|
|
:columns="getColumnsForTab(item.key)"
|
|
:empty-state="{ icon: 'i-heroicons-circle-stack-20-solid', label: 'Keine Belege anzuzeigen' }"
|
|
:rows="getRowsForTab(item.key)"
|
|
:ui="{ divide: 'divide-gray-200 dark:divide-gray-800' }"
|
|
class="w-full"
|
|
@select="selectItem"
|
|
>
|
|
<template #type-data="{row}">
|
|
<span v-if="row.type === 'cancellationInvoices'" class="text-cyan-500">{{
|
|
dataStore.documentTypesForCreation[row.type].labelSingle
|
|
}} für {{ filteredRows.find(i => row.createddocument?.id === i.id)?.documentNumber }}</span>
|
|
<span v-else>{{ dataStore.documentTypesForCreation[row.type].labelSingle }}</span>
|
|
</template>
|
|
|
|
<template #state-data="{row}">
|
|
<span v-if="row.state === 'Entwurf'" class="text-rose-500">{{ row.state }}</span>
|
|
<span
|
|
v-if="row.state === 'Gebucht' && !items.find(i => i.createddocument && i.createddocument.id === row.id)"
|
|
class="text-primary-500"
|
|
>
|
|
{{ row.state }}
|
|
</span>
|
|
<span
|
|
v-else-if="row.state === 'Gebucht' && items.find(i => i.createddocument && i.createddocument.id === row.id && i.type === 'cancellationInvoices') && ['invoices','advanceInvoices'].includes(row.type)"
|
|
class="text-cyan-500"
|
|
>
|
|
Storniert mit {{ items.find(i => i.createddocument && i.createddocument.id === row.id).documentNumber }}
|
|
</span>
|
|
<span v-else-if="row.state === 'Gebucht'" class="text-primary-500">{{ row.state }}</span>
|
|
</template>
|
|
|
|
<template #partner-data="{row}">
|
|
<span v-if="row.customer && row.customer.name.length < 21">{{ row.customer ? row.customer.name : "" }}</span>
|
|
<UTooltip v-else-if="row.customer && row.customer.name.length > 20" :text="row.customer.name">
|
|
{{ row.customer.name.substring(0, 20) }}...
|
|
</UTooltip>
|
|
</template>
|
|
|
|
<template #reference-data="{row}">
|
|
<span v-if="row === filteredRows[selectedItem]" class="text-primary-500 font-bold">{{ row.documentNumber }}</span>
|
|
<span v-else>{{ row.documentNumber }}</span>
|
|
</template>
|
|
|
|
<template #date-data="{row}">
|
|
<span v-if="row.date">{{ row.date ? dayjs(row.date).format("DD.MM.YY") : '' }}</span>
|
|
<span v-if="row.documentDate">{{ row.documentDate ? dayjs(row.documentDate).format("DD.MM.YY") : '' }}</span>
|
|
</template>
|
|
|
|
<template #dueDate-data="{row}">
|
|
<span
|
|
v-if="row.state === 'Gebucht' && row.paymentDays && ['invoices','advanceInvoices'].includes(row.type) && !items.find(i => i.linkedDocument && i.linkedDocument.id === row.id)"
|
|
:class="dayjs(row.documentDate).add(row.paymentDays,'day').diff(dayjs()) <= 0 && !isPaid(row) ? ['text-rose-500'] : '' "
|
|
>
|
|
{{ row.documentDate ? dayjs(row.documentDate).add(row.paymentDays, 'day').format("DD.MM.YY") : '' }}
|
|
</span>
|
|
</template>
|
|
|
|
<template #paid-data="{row}">
|
|
<div
|
|
v-if="(row.type === 'invoices' ||row.type === 'advanceInvoices') && row.state === 'Gebucht' && !items.find(i => i.linkedDocument && i.linkedDocument.id === row.id)">
|
|
<span v-if="useSum().getIsPaid(row,items)" class="text-primary-500">Bezahlt</span>
|
|
<span v-else class="text-rose-600">Offen</span>
|
|
</div>
|
|
</template>
|
|
|
|
<template #amount-data="{row}">
|
|
<span v-if="row.type !== 'deliveryNotes'">{{ displayCurrency(useSum().getCreatedDocumentSum(row, items)) }}</span>
|
|
</template>
|
|
|
|
<template #amountOpen-data="{row}">
|
|
<span
|
|
v-if="!['deliveryNotes','cancellationInvoices','quotes','confirmationOrders'].includes(row.type) && row.state !== 'Entwurf' && !useSum().getIsPaid(row,items) && !items.find(i => i.linkedDocument && i.linkedDocument.id === row.id) ">
|
|
{{ displayCurrency(useSum().getCreatedDocumentSum(row, items) - row.statementallocations.reduce((n, {amount}) => n + amount, 0)) }}
|
|
</span>
|
|
</template>
|
|
</UTable>
|
|
</div>
|
|
</template>
|
|
</UTabs>
|
|
</template>
|
|
|
|
<script setup>
|
|
import dayjs from "dayjs";
|
|
import { ref, computed, watch } from 'vue';
|
|
|
|
const dataStore = useDataStore()
|
|
const tempStore = useTempStore()
|
|
const router = useRouter()
|
|
|
|
const type = "createddocuments"
|
|
const dataType = dataStore.dataTypes[type]
|
|
|
|
const items = ref([])
|
|
const selectedItem = ref(0)
|
|
const activeTabIndex = ref(0)
|
|
|
|
// Debounce-Logik für die Suche
|
|
const searchString = ref(tempStore.searchStrings['createddocuments'] || '')
|
|
const debouncedSearchString = ref(searchString.value)
|
|
let debounceTimeout = null
|
|
|
|
watch(searchString, (newVal) => {
|
|
clearTimeout(debounceTimeout)
|
|
debounceTimeout = setTimeout(() => {
|
|
debouncedSearchString.value = newVal
|
|
tempStore.modifySearchString('createddocuments', newVal)
|
|
}, 300) // 300ms warten nach dem letzten Tastendruck
|
|
})
|
|
|
|
defineShortcuts({
|
|
'/': () => {
|
|
document.getElementById("searchinput").focus()
|
|
},
|
|
'+': () => {
|
|
router.push('/createDocument/edit')
|
|
},
|
|
'Enter': {
|
|
usingInput: true,
|
|
handler: () => {
|
|
if (filteredRows.value[selectedItem.value]) {
|
|
router.push(`/createDocument/show/${filteredRows.value[selectedItem.value].id}`)
|
|
}
|
|
}
|
|
},
|
|
'arrowdown': () => {
|
|
if (selectedItem.value < filteredRows.value.length - 1) {
|
|
selectedItem.value += 1
|
|
} else {
|
|
selectedItem.value = 0
|
|
}
|
|
},
|
|
'arrowup': () => {
|
|
if (selectedItem.value === 0) {
|
|
selectedItem.value = filteredRows.value.length - 1
|
|
} else {
|
|
selectedItem.value -= 1
|
|
}
|
|
}
|
|
})
|
|
|
|
const setupPage = async () => {
|
|
items.value = (await useEntities("createddocuments").select("*, customer(id,name), statementallocations(id,amount),linkedDocument(*)", "documentNumber", true, true))
|
|
}
|
|
|
|
setupPage()
|
|
|
|
const templateColumns = [
|
|
{key: "reference", label: "Referenz"},
|
|
{key: 'type', label: "Typ"},
|
|
{key: 'state', label: "Status"},
|
|
{key: "amount", label: "Betrag"},
|
|
{key: 'partner', label: "Kunde"},
|
|
{key: "date", label: "Datum"},
|
|
{key: "amountOpen", label: "Offener Betrag"},
|
|
{key: "paid", label: "Bezahlt"},
|
|
{key: "dueDate", label: "Fällig"}
|
|
]
|
|
|
|
const draftColumns = [
|
|
{key: 'type', label: "Typ"},
|
|
{key: 'state', label: "Status"},
|
|
{key: 'partner', label: "Kunde"},
|
|
{key: "date", label: "Erstellt am"},
|
|
{key: "amount", label: "Betrag"}
|
|
]
|
|
|
|
const selectedColumns = ref(tempStore.columns["createddocuments"] ? tempStore.columns["createddocuments"] : templateColumns)
|
|
const columns = computed(() => templateColumns.filter((column) => selectedColumns.value.find(i => i.key === column.key)))
|
|
|
|
const getColumnsForTab = (tabKey) => {
|
|
if (tabKey === 'drafts') {
|
|
return draftColumns
|
|
}
|
|
return columns.value
|
|
}
|
|
|
|
const templateTypes = [
|
|
{ key: "drafts", label: "Entwürfe" },
|
|
{ key: "invoices", label: "Rechnungen" },
|
|
{ key: "quotes", label: "Angebote" },
|
|
{ key: "deliveryNotes", label: "Lieferscheine" },
|
|
{ key: "confirmationOrders", label: "Auftragsbestätigungen" }
|
|
]
|
|
const selectedTypes = ref(tempStore.filters["createddocuments"] ? tempStore.filters["createddocuments"] : templateTypes)
|
|
const types = computed(() => {
|
|
return templateTypes.filter((type) => selectedTypes.value.find(i => i.key === type.key))
|
|
})
|
|
|
|
const selectItem = (item) => {
|
|
if (item.state === "Entwurf") {
|
|
router.push(`/createDocument/edit/${item.id}`)
|
|
} else {
|
|
router.push(`/createDocument/show/${item.id}`)
|
|
}
|
|
}
|
|
|
|
const displayCurrency = (value, currency = "€") => {
|
|
return `${Number(value).toFixed(2).replace(".", ",")} ${currency}`
|
|
}
|
|
|
|
const clearSearchString = () => {
|
|
tempStore.clearSearchString('createddocuments')
|
|
searchString.value = ''
|
|
debouncedSearchString.value = ''
|
|
}
|
|
|
|
const openUnpaidInvoicesFilter = {
|
|
name: 'Nur offene Rechnungen',
|
|
filterFunction: (row) => {
|
|
return ['invoices', 'advanceInvoices'].includes(row.type)
|
|
&& row.state === 'Gebucht'
|
|
&& !useSum().getIsPaid(row, items.value)
|
|
&& !items.value.find(i => i.linkedDocument && i.linkedDocument.id === row.id)
|
|
}
|
|
}
|
|
|
|
const availableFilters = computed(() => [...dataType.filters, openUnpaidInvoicesFilter])
|
|
const selectableFilters = computed(() => availableFilters.value.map(i => i.name))
|
|
const selectedFilters = ref(dataType.filters.filter(i => i.default).map(i => i.name) || [])
|
|
|
|
const filteredRows = computed(() => {
|
|
let tempItems = items.value.filter(i => types.value.find(x => {
|
|
if (x.key === 'drafts') return i.state === 'Entwurf'
|
|
if (i.state === 'Entwurf' && x.key !== 'drafts') return false
|
|
if (x.key === 'invoices') return ['invoices', 'advanceInvoices', 'cancellationInvoices'].includes(i.type)
|
|
return x.key === i.type
|
|
}))
|
|
|
|
tempItems = tempItems.filter(i => i.type !== "serialInvoices")
|
|
|
|
tempItems = tempItems.map(i => {
|
|
return {
|
|
...i,
|
|
class: i.archived ? 'bg-red-500/50 dark:bg-red-400/50' : null
|
|
}
|
|
})
|
|
|
|
if (selectedFilters.value.length > 0) {
|
|
selectedFilters.value.forEach(filterName => {
|
|
const filter = availableFilters.value.find(i => i.name === filterName)
|
|
if (filter?.filterFunction) {
|
|
tempItems = tempItems.filter(filter.filterFunction)
|
|
}
|
|
})
|
|
}
|
|
|
|
// Hier nutzen wir nun den debounced Wert für die lokale Suche
|
|
const results = useSearch(debouncedSearchString.value, tempItems.slice().reverse())
|
|
return results
|
|
})
|
|
|
|
const getRowsForTab = (tabKey) => {
|
|
return filteredRows.value.filter(row => {
|
|
if (tabKey === 'drafts') return row.state === 'Entwurf'
|
|
if (row.state === 'Entwurf') return false
|
|
if (tabKey === 'invoices') return ['invoices', 'advanceInvoices', 'cancellationInvoices'].includes(row.type)
|
|
return row.type === tabKey
|
|
})
|
|
}
|
|
|
|
const isPaid = (item) => {
|
|
let amountPaid = 0
|
|
item.statementallocations.forEach(allocation => amountPaid += allocation.amount)
|
|
return Number(amountPaid.toFixed(2)) === useSum().getCreatedDocumentSum(item, items.value)
|
|
}
|
|
</script>
|