Merge branch 'master'

This commit is contained in:
Mohit Panjwani
2022-04-01 12:04:06 +05:30
16 changed files with 876 additions and 1297 deletions
@@ -103,6 +103,7 @@ class CustomerStatsController extends Controller
) )
->whereCompany() ->whereCompany()
->whereCustomer($customer->id) ->whereCustomer($customer->id)
->where('status', '<>', Invoice::STATUS_DRAFT)
->sum('total'); ->sum('total');
$totalReceipts = Payment::whereBetween( $totalReceipts = Payment::whereBetween(
'payment_date', 'payment_date',
@@ -104,6 +104,7 @@ class DashboardController extends Controller
'invoice_date', 'invoice_date',
[$startDate->format('Y-m-d'), $start->format('Y-m-d')] [$startDate->format('Y-m-d'), $start->format('Y-m-d')]
) )
->where('status', '<>', Invoice::STATUS_DRAFT)
->whereCompany() ->whereCompany()
->sum('base_total'); ->sum('base_total');
@@ -141,6 +142,7 @@ class DashboardController extends Controller
$recent_due_invoices = Invoice::with('customer') $recent_due_invoices = Invoice::with('customer')
->whereCompany() ->whereCompany()
->where('base_due_amount', '>', 0) ->where('base_due_amount', '>', 0)
->where('status', '<>', Invoice::STATUS_DRAFT)
->take(5) ->take(5)
->latest() ->latest()
->get(); ->get();
@@ -24,6 +24,7 @@ class InvoicesController extends Controller
$limit = $request->has('limit') ? $request->limit : 10; $limit = $request->has('limit') ? $request->limit : 10;
$invoices = Invoice::whereCompany() $invoices = Invoice::whereCompany()
->whereTabFilters($request->tab_status)
->join('customers', 'customers.id', '=', 'invoices.customer_id') ->join('customers', 'customers.id', '=', 'invoices.customer_id')
->applyFilters($request->all()) ->applyFilters($request->all())
->select('invoices.*', 'customers.name') ->select('invoices.*', 'customers.name')
+1 -1
View File
@@ -23,7 +23,7 @@ class EstimateResource extends JsonResource
'reference_number' => $this->reference_number, 'reference_number' => $this->reference_number,
'tax_per_item' => $this->tax_per_item, 'tax_per_item' => $this->tax_per_item,
'discount_per_item' => $this->discount_per_item, 'discount_per_item' => $this->discount_per_item,
'notes' => $this->getNotes(), 'notes' => $this->notes,
'discount' => $this->discount, 'discount' => $this->discount,
'discount_type' => $this->discount_type, 'discount_type' => $this->discount_type,
'discount_val' => $this->discount_val, 'discount_val' => $this->discount_val,
+1 -1
View File
@@ -18,7 +18,7 @@ class PaymentResource extends JsonResource
'id' => $this->id, 'id' => $this->id,
'payment_number' => $this->payment_number, 'payment_number' => $this->payment_number,
'payment_date' => $this->payment_date, 'payment_date' => $this->payment_date,
'notes' => $this->getNotes(), 'notes' => $this->notes,
'amount' => $this->amount, 'amount' => $this->amount,
'unique_hash' => $this->unique_hash, 'unique_hash' => $this->unique_hash,
'invoice_id' => $this->invoice_id, 'invoice_id' => $this->invoice_id,
+2 -1
View File
@@ -483,7 +483,8 @@ class Estimate extends Model implements HasMedia
'{ESTIMATE_DATE}' => $this->formattedEstimateDate, '{ESTIMATE_DATE}' => $this->formattedEstimateDate,
'{ESTIMATE_EXPIRY_DATE}' => $this->formattedExpiryDate, '{ESTIMATE_EXPIRY_DATE}' => $this->formattedExpiryDate,
'{ESTIMATE_NUMBER}' => $this->estimate_number, '{ESTIMATE_NUMBER}' => $this->estimate_number,
'{ESTIMATE_REF_NUMBER}' => $this->reference_number, '{PDF_LINK}' => $this->estimatePdfUrl,
'{TOTAL_AMOUNT}' => format_money_pdf($this->total, $this->customer->currency)
]; ];
} }
+2 -2
View File
@@ -240,7 +240,7 @@ class Expense extends Model implements HasMedia
} }
if ($request->hasFile('attachment_receipt')) { if ($request->hasFile('attachment_receipt')) {
$expense->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts'); $expense->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts', 'local');
} }
if ($request->customFields) { if ($request->customFields) {
@@ -267,7 +267,7 @@ class Expense extends Model implements HasMedia
} }
if ($request->hasFile('attachment_receipt')) { if ($request->hasFile('attachment_receipt')) {
$this->clearMediaCollection('receipts'); $this->clearMediaCollection('receipts');
$this->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts'); $this->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts', 'local');
} }
if ($request->customFields) { if ($request->customFields) {
+37 -17
View File
@@ -187,16 +187,6 @@ class Invoice extends Model implements HasMedia
return Carbon::parse($this->invoice_date)->format($dateFormat); return Carbon::parse($this->invoice_date)->format($dateFormat);
} }
public function scopeWhereStatus($query, $status)
{
return $query->where('invoices.status', $status);
}
public function scopeWherePaidStatus($query, $status)
{
return $query->where('invoices.paid_status', $status);
}
public function scopeWhereDueStatus($query, $status) public function scopeWhereDueStatus($query, $status)
{ {
return $query->whereIn('invoices.paid_status', [ return $query->whereIn('invoices.paid_status', [
@@ -234,6 +224,40 @@ class Invoice extends Model implements HasMedia
$query->orderBy($orderByField, $orderBy); $query->orderBy($orderByField, $orderBy);
} }
public function scopeWhereStatus($query, $status)
{
return $query->where('invoices.status', $status);
}
public function scopeWherePaidStatus($query, $status)
{
return $query->where('invoices.paid_status', $status);
}
public function scopeWhereTabFilters($query, $status)
{
if ($status == "DRAFT") {
return $query->where('invoices.status', $status);
}
if ($status == "SENT") {
return $query->whereIn('invoices.status', [
self::STATUS_SENT,
self::STATUS_VIEWED,
self::STATUS_COMPLETED
]);
}
if ($status == 'DUE') {
return $query->whereIn('invoices.paid_status', [
self::STATUS_UNPAID,
self::STATUS_PARTIALLY_PAID,
]);
}
return ;
}
public function scopeApplyFilters($query, array $filters) public function scopeApplyFilters($query, array $filters)
{ {
$filters = collect($filters); $filters = collect($filters);
@@ -249,17 +273,11 @@ class Invoice extends Model implements HasMedia
$filters->get('status') == self::STATUS_PAID $filters->get('status') == self::STATUS_PAID
) { ) {
$query->wherePaidStatus($filters->get('status')); $query->wherePaidStatus($filters->get('status'));
} elseif ($filters->get('status') == 'DUE') {
$query->whereDueStatus($filters->get('status'));
} else { } else {
$query->whereStatus($filters->get('status')); $query->whereStatus($filters->get('status'));
} }
} }
if ($filters->get('paid_status')) {
$query->wherePaidStatus($filters->get('status'));
}
if ($filters->get('invoice_id')) { if ($filters->get('invoice_id')) {
$query->whereInvoice($filters->get('invoice_id')); $query->whereInvoice($filters->get('invoice_id'));
} }
@@ -650,7 +668,9 @@ class Invoice extends Model implements HasMedia
'{INVOICE_DATE}' => $this->formattedInvoiceDate, '{INVOICE_DATE}' => $this->formattedInvoiceDate,
'{INVOICE_DUE_DATE}' => $this->formattedDueDate, '{INVOICE_DUE_DATE}' => $this->formattedDueDate,
'{INVOICE_NUMBER}' => $this->invoice_number, '{INVOICE_NUMBER}' => $this->invoice_number,
'{INVOICE_REF_NUMBER}' => $this->reference_number, '{PDF_LINK}' => $this->invoicePdfUrl,
'{DUE_AMOUNT}' => format_money_pdf($this->due_amount, $this->customer->currency),
'{TOTAL_AMOUNT}' => format_money_pdf($this->total, $this->customer->currency)
]; ];
} }
+2 -1
View File
@@ -435,7 +435,8 @@ class Payment extends Model implements HasMedia
'{PAYMENT_DATE}' => $this->formattedPaymentDate, '{PAYMENT_DATE}' => $this->formattedPaymentDate,
'{PAYMENT_MODE}' => $this->paymentMethod ? $this->paymentMethod->name : null, '{PAYMENT_MODE}' => $this->paymentMethod ? $this->paymentMethod->name : null,
'{PAYMENT_NUMBER}' => $this->payment_number, '{PAYMENT_NUMBER}' => $this->payment_number,
'{PAYMENT_AMOUNT}' => $this->reference_number, '{PDF_LINK}' => $this->paymentPdfUrl,
'{PAYMENT_AMOUNT}' => format_money_pdf($this->amount, $this->customer->currency)
]; ];
} }
+2 -2
View File
@@ -27,7 +27,7 @@
"vite": "^2.6.1" "vite": "^2.6.1"
}, },
"dependencies": { "dependencies": {
"@headlessui/vue": "^1.4.0", "@headlessui/vue": "^1.5.0",
"@heroicons/vue": "^1.0.1", "@heroicons/vue": "^1.0.1",
"@popperjs/core": "^2.9.2", "@popperjs/core": "^2.9.2",
"@stripe/stripe-js": "^1.21.2", "@stripe/stripe-js": "^1.21.2",
@@ -48,7 +48,7 @@
"mini-svg-data-uri": "^1.3.3", "mini-svg-data-uri": "^1.3.3",
"moment": "^2.29.1", "moment": "^2.29.1",
"pinia": "^2.0.4", "pinia": "^2.0.4",
"v-money3": "^3.13.5", "v-money3": "3.16.1",
"v-tooltip": "^4.0.0-alpha.1", "v-tooltip": "^4.0.0-alpha.1",
"vue": "^3.2.0-beta.5", "vue": "^3.2.0-beta.5",
"vue-flatpickr-component": "^9.0.3", "vue-flatpickr-component": "^9.0.3",
@@ -56,7 +56,7 @@
<BaseMultiselect <BaseMultiselect
v-model="filters.status" v-model="filters.status"
:groups="true" :groups="true"
:options="status" :options="invoiceStatus"
searchable searchable
:placeholder="$t('general.select_a_status')" :placeholder="$t('general.select_a_status')"
@update:modelValue="setActiveTab" @update:modelValue="setActiveTab"
@@ -130,11 +130,27 @@
" "
> >
<!-- Tabs --> <!-- Tabs -->
<BaseTabGroup class="-mb-5" @change="setStatusFilter"> <BaseTabGroup
<BaseTab :title="$t('general.all')" filter="" /> class="-mb-5"
<BaseTab :title="$t('general.draft')" filter="DRAFT" /> :selected-index="selectedIndex"
<BaseTab :title="$t('general.sent')" filter="SENT" /> @change="changeTabStatus"
<BaseTab :title="$t('general.due')" filter="DUE" /> >
<BaseTab
:title="invoiceTabStatus[0].title"
:tab-status="invoiceTabStatus[0].value"
/>
<BaseTab
:title="invoiceTabStatus[1].title"
:tab-status="invoiceTabStatus[1].value"
/>
<BaseTab
:title="invoiceTabStatus[2].title"
:tab-status="invoiceTabStatus[2].value"
/>
<BaseTab
:title="invoiceTabStatus[3].title"
:tab-status="invoiceTabStatus[3].value"
/>
</BaseTabGroup> </BaseTabGroup>
<BaseDropdown <BaseDropdown
@@ -289,10 +305,10 @@ const utils = inject('$utils')
const table = ref(null) const table = ref(null)
const showFilters = ref(false) const showFilters = ref(false)
const status = ref([ const invoiceStatus = ref([
{ {
label: 'Status', label: 'Status',
options: ['DRAFT', 'DUE', 'SENT', 'VIEWED', 'COMPLETED'], options: ['DRAFT', 'SENT', 'VIEWED', 'COMPLETED'],
}, },
{ {
label: 'Paid Status', label: 'Paid Status',
@@ -300,10 +316,29 @@ const status = ref([
}, },
, ,
]) ])
const invoiceTabStatus = {
0: {
title: t('general.all'),
value: '',
},
1: {
title: t('general.draft'),
value: 'DRAFT',
},
2: {
title: t('general.sent'),
value: 'SENT',
},
3: {
title: t('general.due'),
value: 'DUE',
},
}
const isRequestOngoing = ref(true) const isRequestOngoing = ref(true)
const activeTab = ref('general.draft')
const router = useRouter() const router = useRouter()
const userStore = useUserStore() const userStore = useUserStore()
const selectedIndex = ref(0)
let filters = reactive({ let filters = reactive({
customer_id: '', customer_id: '',
@@ -311,6 +346,7 @@ let filters = reactive({
from_date: '', from_date: '',
to_date: '', to_date: '',
invoice_number: '', invoice_number: '',
tab_status: '',
}) })
const showEmptyScreen = computed( const showEmptyScreen = computed(
@@ -401,6 +437,7 @@ async function fetchData({ page, filter, sort }) {
from_date: filters.from_date, from_date: filters.from_date,
to_date: filters.to_date, to_date: filters.to_date,
invoice_number: filters.invoice_number, invoice_number: filters.invoice_number,
tab_status: filters.tab_status,
orderByField: sort.fieldName || 'created_at', orderByField: sort.fieldName || 'created_at',
orderBy: sort.order || 'desc', orderBy: sort.order || 'desc',
page, page,
@@ -423,29 +460,9 @@ async function fetchData({ page, filter, sort }) {
} }
} }
function setStatusFilter(val) { function changeTabStatus(val, index) {
if (activeTab.value == val.title) { filters.tab_status = val['tab-status']
return true selectedIndex.value = index
}
activeTab.value = val.title
switch (val.title) {
case t('general.draft'):
filters.status = 'DRAFT'
break
case t('general.sent'):
filters.status = 'SENT'
break
case t('general.due'):
filters.status = 'DUE'
break
default:
filters.status = ''
break
}
} }
function setFilters() { function setFilters() {
@@ -463,8 +480,6 @@ function clearFilter() {
filters.from_date = '' filters.from_date = ''
filters.to_date = '' filters.to_date = ''
filters.invoice_number = '' filters.invoice_number = ''
activeTab.value = t('general.all')
} }
async function removeMultipleInvoices() { async function removeMultipleInvoices() {
@@ -505,39 +520,21 @@ function toggleFilter() {
function setActiveTab(val) { function setActiveTab(val) {
switch (val) { switch (val) {
case 'DRAFT': case 'DRAFT':
activeTab.value = t('general.draft') selectedIndex.value = 1
break break
case 'SENT': case 'SENT':
activeTab.value = t('general.sent') case 'VIEWED':
break
case 'DUE':
activeTab.value = t('general.due')
break
case 'COMPLETED': case 'COMPLETED':
activeTab.value = t('invoices.completed')
break
case 'PAID': case 'PAID':
activeTab.value = t('invoices.paid') selectedIndex.value = 2
break break
case 'UNPAID': case 'UNPAID':
activeTab.value = t('invoices.unpaid')
break
case 'PARTIALLY_PAID': case 'PARTIALLY_PAID':
activeTab.value = t('invoices.partially_paid') selectedIndex.value = 3
break
case 'VIEWED':
activeTab.value = t('invoices.viewed')
break
default:
activeTab.value = t('general.all')
break break
} }
filters.tab_status = invoiceTabStatus[selectedIndex.value].value
} }
</script> </script>
@@ -82,9 +82,9 @@
required required
> >
<BaseCustomerSelectInput <BaseCustomerSelectInput
v-if="!isLoadingContent"
v-model="paymentStore.currentPayment.customer_id" v-model="paymentStore.currentPayment.customer_id"
:content-loading="isLoadingContent" :content-loading="isLoadingContent"
v-if="!isLoadingContent"
:invalid="v$.currentPayment.customer_id.$error" :invalid="v$.currentPayment.customer_id.$error"
:placeholder="$t('customers.select_a_customer')" :placeholder="$t('customers.select_a_customer')"
show-action show-action
@@ -423,7 +423,7 @@ function onCustomerChange(customer_id) {
if (customer_id) { if (customer_id) {
let data = { let data = {
customer_id: customer_id, customer_id: customer_id,
status: 'DUE', tab_status: 'DUE',
limit: 'all', limit: 'all',
} }
@@ -446,7 +446,11 @@ function onCustomerChange(customer_id) {
paymentStore.currentPayment.selectedCustomer = res2.data.data paymentStore.currentPayment.selectedCustomer = res2.data.data
paymentStore.currentPayment.customer = res2.data.data paymentStore.currentPayment.customer = res2.data.data
paymentStore.currentPayment.currency = res2.data.data.currency paymentStore.currentPayment.currency = res2.data.data.currency
if(isEdit.value && !customerStore.editCustomer && paymentStore.currentPayment.customer_id) { if (
isEdit.value &&
!customerStore.editCustomer &&
paymentStore.currentPayment.customer_id
) {
customerStore.editCustomer = res2.data.data customerStore.editCustomer = res2.data.data
} }
} }
@@ -126,7 +126,7 @@ onMounted(() => {
}) })
const value = computed({ const value = computed({
get: () => props.modelValue, get: () => (props.modelValue ? props.modelValue : ''),
set: (value) => { set: (value) => {
emit('update:modelValue', value) emit('update:modelValue', value)
}, },
@@ -195,7 +195,9 @@ async function getFields() {
{ label: 'Date', value: 'INVOICE_DATE' }, { label: 'Date', value: 'INVOICE_DATE' },
{ label: 'Due Date', value: 'INVOICE_DUE_DATE' }, { label: 'Due Date', value: 'INVOICE_DUE_DATE' },
{ label: 'Number', value: 'INVOICE_NUMBER' }, { label: 'Number', value: 'INVOICE_NUMBER' },
{ label: 'Ref Number', value: 'INVOICE_REF_NUMBER' }, { label: 'PDF Link', value: 'PDF_LINK' },
{ label: 'Due Amount', value: 'DUE_AMOUNT' },
{ label: 'Total Amount', value: 'TOTAL_AMOUNT' },
...invoiceFields.value.map((i) => ({ ...invoiceFields.value.map((i) => ({
label: i.label, label: i.label,
value: i.slug, value: i.slug,
@@ -211,7 +213,8 @@ async function getFields() {
{ label: 'Date', value: 'ESTIMATE_DATE' }, { label: 'Date', value: 'ESTIMATE_DATE' },
{ label: 'Expiry Date', value: 'ESTIMATE_EXPIRY_DATE' }, { label: 'Expiry Date', value: 'ESTIMATE_EXPIRY_DATE' },
{ label: 'Number', value: 'ESTIMATE_NUMBER' }, { label: 'Number', value: 'ESTIMATE_NUMBER' },
{ label: 'Ref Number', value: 'ESTIMATE_REF_NUMBER' }, { label: 'PDF Link', value: 'PDF_LINK' },
{ label: 'Total Amount', value: 'TOTAL_AMOUNT' },
...estimateFields.value.map((i) => ({ ...estimateFields.value.map((i) => ({
label: i.label, label: i.label,
value: i.slug, value: i.slug,
@@ -228,6 +231,7 @@ async function getFields() {
{ label: 'Number', value: 'PAYMENT_NUMBER' }, { label: 'Number', value: 'PAYMENT_NUMBER' },
{ label: 'Mode', value: 'PAYMENT_MODE' }, { label: 'Mode', value: 'PAYMENT_MODE' },
{ label: 'Amount', value: 'PAYMENT_AMOUNT' }, { label: 'Amount', value: 'PAYMENT_AMOUNT' },
{ label: 'PDF Link', value: 'PDF_LINK' },
...paymentFields.value.map((i) => ({ ...paymentFields.value.map((i) => ({
label: i.label, label: i.label,
value: i.slug, value: i.slug,
@@ -1,6 +1,10 @@
<template> <template>
<div> <div>
<TabGroup :default-index="defaultIndex" @change="onChange"> <TabGroup
:selected-index="selectedIndex"
:default-index="defaultIndex"
@change="onChange"
>
<TabList <TabList
:class="[ :class="[
'flex border-b border-grey-light', 'flex border-b border-grey-light',
@@ -54,6 +58,10 @@ const props = defineProps({
type: Number, type: Number,
default: 0, default: 0,
}, },
selectedIndex: {
type: Number,
default: 0,
},
filter: { filter: {
type: String, type: String,
default: null, default: null,
@@ -67,6 +75,6 @@ const slots = useSlots()
const tabs = computed(() => slots.default().map((tab) => tab.props)) const tabs = computed(() => slots.default().map((tab) => tab.props))
function onChange(d) { function onChange(d) {
emit('change', tabs.value[d]) emit('change', tabs.value[d], d)
} }
</script> </script>
+2 -2
View File
@@ -7,12 +7,12 @@ export function usePopper(options) {
let popper = ref(null) let popper = ref(null)
onMounted(() => { onMounted(() => {
watchEffect(onInvalidate => { watchEffect((onInvalidate) => {
if (!container.value) return if (!container.value) return
if (!activator.value) return if (!activator.value) return
let containerEl = container.value.el || container.value let containerEl = container.value.el || container.value
let activatorEl = activator.value.el || activator.value let activatorEl = activator.value.$el || activator.value
if (!(activatorEl instanceof HTMLElement)) return if (!(activatorEl instanceof HTMLElement)) return
if (!(containerEl instanceof HTMLElement)) return if (!(containerEl instanceof HTMLElement)) return
+744 -1204
View File
File diff suppressed because it is too large Load Diff