2018-05-20 22:53:14 +10:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Models;
|
|
|
|
|
2020-04-01 23:35:06 +11:00
|
|
|
use Carbon\Carbon;
|
2020-05-25 17:45:17 +10:00
|
|
|
use Clarkeash\Doorman\Facades\Doorman;
|
|
|
|
use Clarkeash\Doorman\Models\Invite;
|
2018-05-20 22:53:14 +10:00
|
|
|
use Illuminate\Database\Eloquent\Model;
|
2020-05-25 17:45:17 +10:00
|
|
|
use Illuminate\Support\Arr;
|
2024-07-09 20:17:30 +10:00
|
|
|
use Illuminate\Support\Collection;
|
2024-08-03 10:06:25 +10:00
|
|
|
use Illuminate\Support\Facades\Mail;
|
2024-07-07 19:10:00 +10:00
|
|
|
use Leenooks\Casts\LeenooksCarbon;
|
2021-06-29 16:36:34 +10:00
|
|
|
use Leenooks\Traits\ScopeActive;
|
|
|
|
|
2024-07-07 19:10:00 +10:00
|
|
|
use App\Casts\CollectionOrNull;
|
2021-06-29 16:36:34 +10:00
|
|
|
use App\Interfaces\IDs;
|
2024-08-03 10:06:25 +10:00
|
|
|
use App\Mail\{InvoiceEmail,InvoiceGeneratedAdmin};
|
|
|
|
use App\Traits\{PushNew,SiteID};
|
2020-04-01 23:35:06 +11:00
|
|
|
|
2021-06-29 16:36:34 +10:00
|
|
|
/**
|
|
|
|
* Class Invoice
|
|
|
|
* Invoices that belong to an Account
|
|
|
|
*
|
|
|
|
* Attributes for services:
|
2024-07-09 20:17:30 +10:00
|
|
|
* + created_at : Date the invoice was created
|
2021-06-29 16:36:34 +10:00
|
|
|
* + due : Balance due on an invoice
|
2024-07-09 20:17:30 +10:00
|
|
|
* + due_at : Date the invoice is due
|
2021-06-29 16:36:34 +10:00
|
|
|
* + lid : Local ID for invoice
|
|
|
|
* + paid : Total of payments received (excluding pending)
|
|
|
|
* + paid_date : Date the invoice was paid in full
|
|
|
|
* + paid_pending : Total of pending payments received
|
|
|
|
* + sid : System ID for invoice
|
2022-04-22 14:41:18 +10:00
|
|
|
* + sub_total : Invoice sub-total before taxes
|
2024-07-09 20:17:30 +10:00
|
|
|
* + tax_total : Invoices total of taxes
|
2021-06-29 16:36:34 +10:00
|
|
|
* + total : Invoice total
|
|
|
|
*
|
|
|
|
* @package App\Models
|
|
|
|
*/
|
|
|
|
class Invoice extends Model implements IDs
|
2018-05-20 22:53:14 +10:00
|
|
|
{
|
2024-08-03 10:06:25 +10:00
|
|
|
use PushNew,ScopeActive,SiteID;
|
2021-06-29 16:36:34 +10:00
|
|
|
|
2022-06-13 19:58:53 +10:00
|
|
|
protected $casts = [
|
2024-07-05 16:38:31 +10:00
|
|
|
'created_at' => 'datetime:Y-m-d',
|
2024-07-07 19:10:00 +10:00
|
|
|
'due_at' => LeenooksCarbon::class,
|
|
|
|
'reminders' => CollectionOrNull::class,
|
2024-07-05 16:38:31 +10:00
|
|
|
'_paid_at' => 'datetime:Y-m-d',
|
2022-04-22 14:41:18 +10:00
|
|
|
];
|
2018-06-19 22:31:49 +10:00
|
|
|
|
2022-04-22 10:36:41 +10:00
|
|
|
public const BILL_WEEKLY = 0;
|
|
|
|
public const BILL_MONTHLY = 1;
|
|
|
|
public const BILL_QUARTERLY = 2;
|
|
|
|
public const BILL_SEMI_YEARLY = 3;
|
|
|
|
public const BILL_YEARLY = 4;
|
|
|
|
public const BILL_TWOYEARS = 5;
|
|
|
|
public const BILL_THREEYEARS = 6;
|
|
|
|
public const BILL_FOURYEARS = 7;
|
|
|
|
public const BILL_FIVEYEARS = 8;
|
|
|
|
|
2021-12-24 12:14:01 +11:00
|
|
|
/* Our available billing periods */
|
|
|
|
public const billing_periods = [
|
2022-04-22 14:41:18 +10:00
|
|
|
self::BILL_WEEKLY => [
|
|
|
|
'name' => 'Weekly',
|
|
|
|
'interval' => 0.25,
|
|
|
|
],
|
|
|
|
self::BILL_MONTHLY => [
|
|
|
|
'name' => 'Monthly',
|
|
|
|
'interval' => 1,
|
|
|
|
],
|
|
|
|
self::BILL_QUARTERLY => [
|
|
|
|
'name' => 'Quarterly',
|
|
|
|
'interval' => 3,
|
|
|
|
],
|
|
|
|
self::BILL_SEMI_YEARLY => [
|
|
|
|
'name' => 'Semi-Annually',
|
|
|
|
'interval' => 6,
|
|
|
|
],
|
|
|
|
self::BILL_YEARLY => [
|
|
|
|
'name' => 'Annually',
|
|
|
|
'interval' => 12,
|
|
|
|
],
|
|
|
|
self::BILL_TWOYEARS => [
|
|
|
|
'name' => 'Two years',
|
|
|
|
'interval' => 24,
|
|
|
|
],
|
|
|
|
self::BILL_THREEYEARS => [
|
|
|
|
'name' => 'Three Years',
|
|
|
|
'interval' => 36,
|
|
|
|
],
|
|
|
|
self::BILL_FOURYEARS => [
|
|
|
|
'name' => 'Four Years',
|
|
|
|
'interval' => 48,
|
|
|
|
],
|
|
|
|
SELF::BILL_FIVEYEARS => [
|
|
|
|
'name' => 'Five Years',
|
|
|
|
'interval' => 60,
|
|
|
|
],
|
|
|
|
];
|
2021-12-24 12:14:01 +11:00
|
|
|
|
2024-08-03 10:06:25 +10:00
|
|
|
// Our related items that need to be updated when we call pushNew()
|
|
|
|
protected $pushable = ['items_active'];
|
2020-07-06 16:02:59 +10:00
|
|
|
|
2020-02-10 22:07:46 +11:00
|
|
|
protected $with = [
|
2024-07-09 20:17:30 +10:00
|
|
|
'items_active:id,start_at,stop_at,quantity,price_base,discount_amt,item_type,product_id,service_id,invoice_id',
|
|
|
|
'items_active.taxes:id,invoice_item_id,amount,tax_id',
|
|
|
|
'items_active.product:id',
|
|
|
|
'items_active.product.translate:id,product_id,name_short,name_detail',
|
|
|
|
'payment_items_active:id,amount,payment_id,invoice_id',
|
2024-08-11 00:36:10 +10:00
|
|
|
'payment_items_active.payment:id,paid_at',
|
2020-02-10 22:07:46 +11:00
|
|
|
];
|
|
|
|
|
2022-07-29 16:06:19 +10:00
|
|
|
/* STATIC METHODS */
|
2021-12-24 12:14:01 +11:00
|
|
|
|
2024-08-03 10:06:25 +10:00
|
|
|
public static function boot()
|
|
|
|
{
|
|
|
|
parent::boot();
|
|
|
|
|
|
|
|
static::created(function($model) {
|
|
|
|
// Send an email to an admin that the invoice was created
|
|
|
|
$uo = User::where('email',config('osb.admin'))->sole();
|
|
|
|
|
|
|
|
Mail::to($uo->email)
|
|
|
|
->send(new InvoiceGeneratedAdmin($model));
|
|
|
|
|
|
|
|
// @todo Queue an email to the user
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-12-24 12:14:01 +11:00
|
|
|
/**
|
|
|
|
* This works out what multiplier to use to change billing periods
|
|
|
|
*
|
|
|
|
* @param int $source
|
|
|
|
* @param int $target
|
|
|
|
* @return float
|
|
|
|
*/
|
|
|
|
public static function billing_change(int $source,int $target): float
|
|
|
|
{
|
|
|
|
return Arr::get(self::billing_periods,$target.'.interval')/Arr::get(self::billing_periods,$source.'.interval');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the name for the billing interval
|
|
|
|
*
|
|
|
|
* @param int $interval
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public static function billing_name(int $interval): string
|
|
|
|
{
|
|
|
|
$interval = collect(self::billing_periods)->get($interval);
|
|
|
|
|
|
|
|
return Arr::get($interval,'name','Unknown');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the number of months in the billing interval
|
|
|
|
*
|
|
|
|
* @param int $interval
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
public static function billing_period(int $interval): int
|
|
|
|
{
|
|
|
|
$interval = collect(self::billing_periods)->get($interval);
|
|
|
|
|
|
|
|
return Arr::get($interval,'interval',0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Given a contract in months, this will calculate the number of billing intervals required
|
|
|
|
*
|
|
|
|
* @param int $contract_term
|
|
|
|
* @param int $source
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
public static function billing_term(int $contract_term,int $source): int
|
|
|
|
{
|
|
|
|
return ceil(($contract_term ?: 1)/(Arr::get(self::billing_periods,$source.'.interval') ?: 1));
|
|
|
|
}
|
|
|
|
|
2024-07-29 23:12:53 +10:00
|
|
|
/**
|
|
|
|
* Work out the time period for a particular date and invoice period
|
|
|
|
*
|
|
|
|
* @param \Leenooks\Carbon $date
|
|
|
|
* @param int $interval
|
|
|
|
* @param bool $strict
|
|
|
|
* @return Collection
|
|
|
|
* @throws \Exception
|
|
|
|
*/
|
|
|
|
public static function invoice_period(Carbon $date,int $interval,bool $strict): Collection
|
|
|
|
{
|
|
|
|
$date_start = $date->clone();
|
|
|
|
$date_end = $date->clone();
|
|
|
|
|
|
|
|
switch ($interval) {
|
|
|
|
case self::BILL_WEEKLY:
|
|
|
|
$result = collect([
|
|
|
|
'start' => $strict
|
|
|
|
? $date_start->startOfWeek()
|
|
|
|
: $date_start,
|
|
|
|
'end'=> $strict
|
|
|
|
? $date_end->endOfWeek()
|
|
|
|
: $date_end->addWeek()->subDay()
|
|
|
|
]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case self::BILL_MONTHLY:
|
|
|
|
$result = collect([
|
|
|
|
'start' => $strict
|
|
|
|
? $date_start->startOfMonth()
|
|
|
|
: $date_start,
|
|
|
|
'end' => $strict
|
|
|
|
? $date_end->endOfMonth()
|
|
|
|
: $date_end->addMonth()->subDay()
|
|
|
|
]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case self::BILL_QUARTERLY:
|
|
|
|
$result = collect([
|
|
|
|
'start' => $strict// The service charges
|
|
|
|
? $date_start->startOfQuarter()
|
|
|
|
: $date_start,
|
|
|
|
'end' => $strict
|
|
|
|
? $date_end->endOfQuarter()
|
|
|
|
: $date_end->addQuarter()->subDay()
|
|
|
|
]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case self::BILL_SEMI_YEARLY:
|
|
|
|
$result = collect([
|
|
|
|
'start' => $strict
|
|
|
|
? $date_start->startOfHalf()
|
|
|
|
: $date_start,
|
|
|
|
'end' => $strict
|
|
|
|
? $date_end->endOfHalf()
|
|
|
|
: $date_end->addQuarters(2)->subDay()
|
|
|
|
]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case self::BILL_YEARLY:
|
|
|
|
$result = collect([
|
|
|
|
'start' => $strict
|
|
|
|
? $date_start->startOfYear()
|
|
|
|
: $date_start,
|
|
|
|
'end' => $strict
|
|
|
|
? $date_end->endOfYear()
|
|
|
|
: $date_end->addYear()->subDay()
|
|
|
|
]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case self::BILL_TWOYEARS:
|
|
|
|
if (! $strict) {
|
|
|
|
$result = collect([
|
|
|
|
'start' => $date_start,
|
|
|
|
'end' => $date_end->addYears(2)->subDay(),
|
|
|
|
]);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
$data_end = $date_end->addYears(2)->subDay()->endOfYear();
|
|
|
|
|
|
|
|
// Make sure we end on an even year
|
|
|
|
if ($data_end->clone()->addDay()->year%2)
|
|
|
|
$data_end = $data_end->subYear();
|
|
|
|
|
|
|
|
$result = collect([
|
|
|
|
'start' => $data_end->clone()->subYears(2)->addDay(),
|
|
|
|
'end' => $data_end,
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// NOTE: price_recur_strict ignored
|
|
|
|
case self::BILL_THREEYEARS:
|
|
|
|
$result = collect([
|
|
|
|
'start' => $date_start,
|
|
|
|
'end' => $date_end->addYears(3)->subDay(),
|
|
|
|
]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
// NOTE: price_recur_strict ignored
|
|
|
|
case self::BILL_FOURYEARS:
|
|
|
|
$result = collect([
|
|
|
|
'start' => $date_start,
|
|
|
|
'end' => $date_end->addYears(4)->subDay(),
|
|
|
|
]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
// NOTE: price_recur_strict ignored
|
|
|
|
case self::BILL_FIVEYEARS:
|
|
|
|
$result = collect([
|
|
|
|
'start' => $date_start,
|
|
|
|
'end' => $date_end->addYears(5)->subDay(),
|
|
|
|
]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw new \Exception('Unknown recur_schedule: '.$interval);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param \Leenooks\Carbon $start Start Date
|
|
|
|
* @param Carbon $end End Date
|
2024-08-10 10:14:47 +10:00
|
|
|
* @param Collection $period
|
2024-07-29 23:12:53 +10:00
|
|
|
* @return float
|
|
|
|
* @throws \Exception
|
|
|
|
*/
|
|
|
|
public static function invoice_quantity(Carbon $start,Carbon $end,Collection $period): float
|
|
|
|
{
|
|
|
|
if ($start->lessThan(Arr::get($period,'start')) || $end->greaterThan(Arr::get($period,'end')))
|
|
|
|
throw new \Exception('Billing Period differ');
|
|
|
|
|
|
|
|
$d = Arr::get($period,'start')->diffInDays(Arr::get($period,'end'));
|
|
|
|
if (! $d)
|
|
|
|
throw new \Exception('Start and End period dates cannot be the same');
|
|
|
|
|
|
|
|
return round(($d-Arr::get($period,'start')->diffInDays($start)-$end->diffInDays(Arr::get($period,'end')))/$d,2);
|
|
|
|
}
|
|
|
|
|
2022-04-22 14:41:18 +10:00
|
|
|
/* INTERFACES */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Invoice Local ID
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getLIDAttribute(): string
|
|
|
|
{
|
|
|
|
return sprintf('%06s',$this->id);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Invoice System ID
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getSIDAttribute(): string
|
|
|
|
{
|
|
|
|
return sprintf('%02s-%04s-%s',$this->site_id,$this->account_id,$this->getLIDAttribute());
|
|
|
|
}
|
|
|
|
|
2021-06-29 16:36:34 +10:00
|
|
|
/* RELATIONS */
|
2018-05-20 22:53:14 +10:00
|
|
|
|
2024-07-09 20:17:30 +10:00
|
|
|
/**
|
|
|
|
* Account this invoice belongs to
|
|
|
|
*/
|
2018-05-20 22:53:14 +10:00
|
|
|
public function account()
|
|
|
|
{
|
2018-07-13 14:53:44 +10:00
|
|
|
return $this->belongsTo(Account::class);
|
2018-05-20 22:53:14 +10:00
|
|
|
}
|
|
|
|
|
2024-07-09 20:17:30 +10:00
|
|
|
/**
|
|
|
|
* Items on this invoice belongs to
|
|
|
|
*/
|
2018-06-19 22:31:49 +10:00
|
|
|
public function items()
|
2018-05-20 22:53:14 +10:00
|
|
|
{
|
2021-06-29 16:36:34 +10:00
|
|
|
return $this->hasMany(InvoiceItem::class)
|
2022-04-19 17:07:39 +10:00
|
|
|
->with(['taxes','product']);
|
2021-06-29 16:36:34 +10:00
|
|
|
}
|
|
|
|
|
2024-07-09 20:17:30 +10:00
|
|
|
/**
|
|
|
|
* Active items on this invoice belongs to
|
|
|
|
*/
|
|
|
|
public function items_active()
|
|
|
|
{
|
|
|
|
return $this->items()
|
|
|
|
->where('active',TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Payments applied to this invoice
|
|
|
|
*/
|
2021-06-29 16:36:34 +10:00
|
|
|
public function payments()
|
|
|
|
{
|
2022-06-13 17:24:43 +10:00
|
|
|
return $this->hasManyThrough(Payment::class,PaymentItem::class,NULL,'id',NULL,'payment_id')
|
2024-08-10 10:14:47 +10:00
|
|
|
->where('payments.active',TRUE);
|
2018-05-20 22:53:14 +10:00
|
|
|
}
|
|
|
|
|
2024-07-09 20:17:30 +10:00
|
|
|
/**
|
|
|
|
* Payment items attached to this invoice
|
|
|
|
*/
|
|
|
|
public function payment_items()
|
2018-05-20 22:53:14 +10:00
|
|
|
{
|
|
|
|
return $this->hasMany(PaymentItem::class);
|
|
|
|
}
|
|
|
|
|
2024-07-09 20:17:30 +10:00
|
|
|
public function payment_items_active()
|
|
|
|
{
|
|
|
|
return $this->payment_items()
|
|
|
|
->where('payment_items.active',TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 3rd party provider details to this invoice (eg: accounting providers)
|
|
|
|
*/
|
2023-05-12 20:09:51 +10:00
|
|
|
public function providers()
|
|
|
|
{
|
|
|
|
return $this->belongsToMany(ProviderOauth::class,'invoice__provider')
|
|
|
|
->where('invoice__provider.site_id',$this->site_id)
|
|
|
|
->withPivot('ref','synctoken','created_at','updated_at');
|
|
|
|
}
|
|
|
|
|
2021-06-29 16:36:34 +10:00
|
|
|
/* SCOPES */
|
2020-01-12 23:42:32 +11:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Search for a record
|
|
|
|
*
|
|
|
|
* @param $query
|
|
|
|
* @param string $term
|
2021-06-29 16:36:34 +10:00
|
|
|
* @return mixed
|
2020-01-12 23:42:32 +11:00
|
|
|
*/
|
|
|
|
public function scopeSearch($query,string $term)
|
|
|
|
{
|
|
|
|
return $query->where('id','like','%'.$term.'%');
|
|
|
|
}
|
|
|
|
|
2021-06-29 16:36:34 +10:00
|
|
|
/* ATTRIBUTES */
|
2020-01-12 23:42:32 +11:00
|
|
|
|
2021-06-29 16:36:34 +10:00
|
|
|
/**
|
|
|
|
* Balance due on an invoice
|
2022-04-22 14:41:18 +10:00
|
|
|
* @return float
|
2021-06-29 16:36:34 +10:00
|
|
|
*/
|
|
|
|
public function getDueAttribute(): float
|
2018-05-20 22:53:14 +10:00
|
|
|
{
|
2021-07-23 17:25:26 +10:00
|
|
|
return sprintf('%3.2f',$this->getTotalAttribute()-$this->getPaidAttribute());
|
2018-05-20 22:53:14 +10:00
|
|
|
}
|
|
|
|
|
2021-06-29 16:36:34 +10:00
|
|
|
/**
|
|
|
|
* Total of payments received for this invoice
|
|
|
|
* excluding pending payments
|
|
|
|
*
|
|
|
|
* @return float
|
|
|
|
*/
|
|
|
|
public function getPaidAttribute(): float
|
|
|
|
{
|
2024-07-09 20:17:30 +10:00
|
|
|
return $this->payment_items_active->sum('amount');
|
2021-06-29 16:36:34 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the date that the invoice was paid in full.
|
2021-07-23 17:25:26 +10:00
|
|
|
* We assume the last payment received pays it in full, if its fully paid.
|
2021-06-29 16:36:34 +10:00
|
|
|
*
|
|
|
|
* @return Carbon|null
|
|
|
|
*/
|
|
|
|
public function getPaidDateAttribute(): ?Carbon
|
|
|
|
{
|
2024-07-09 20:17:30 +10:00
|
|
|
// If the invoice still has a due balance, its not paid
|
2021-07-23 17:25:26 +10:00
|
|
|
if ($this->getDueAttribute())
|
|
|
|
return NULL;
|
|
|
|
|
2024-07-09 20:17:30 +10:00
|
|
|
$o = $this
|
|
|
|
->payments
|
|
|
|
->filter(fn($item)=>(! $item->pending_status))
|
2021-06-29 16:36:34 +10:00
|
|
|
->last();
|
|
|
|
|
2022-06-13 15:46:38 +10:00
|
|
|
return $o?->paid_at;
|
2021-06-29 16:36:34 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Total of pending payments received for this invoice
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
|
|
|
public function getPaidPendingAttribute(): float
|
2020-07-27 14:49:59 +10:00
|
|
|
{
|
2024-07-09 20:17:30 +10:00
|
|
|
return $this->payment_items
|
|
|
|
->filter(fn($item)=>$item->payment->pending_status)
|
2022-04-22 15:23:08 +10:00
|
|
|
->sum('amount');
|
2018-05-20 22:53:14 +10:00
|
|
|
}
|
|
|
|
|
2021-06-29 16:36:34 +10:00
|
|
|
/**
|
|
|
|
* Get invoice subtotal before taxes
|
|
|
|
*
|
|
|
|
* @return float
|
|
|
|
*/
|
|
|
|
public function getSubTotalAttribute(): float
|
2018-08-01 17:09:38 +10:00
|
|
|
{
|
2024-07-09 20:17:30 +10:00
|
|
|
return $this->items_active->sum('sub_total');
|
2018-08-01 17:09:38 +10:00
|
|
|
}
|
|
|
|
|
2021-06-29 16:36:34 +10:00
|
|
|
/**
|
|
|
|
* Get the invoices taxes total
|
|
|
|
*
|
|
|
|
* @return float
|
|
|
|
*/
|
|
|
|
public function getTaxTotalAttribute(): float
|
|
|
|
{
|
2024-07-09 20:17:30 +10:00
|
|
|
return $this->items_active->sum('tax');
|
2018-08-01 17:09:38 +10:00
|
|
|
}
|
|
|
|
|
2021-06-29 16:36:34 +10:00
|
|
|
/**
|
|
|
|
* Invoice total due
|
|
|
|
*
|
2021-07-23 17:25:26 +10:00
|
|
|
* @return float
|
2021-06-29 16:36:34 +10:00
|
|
|
*/
|
|
|
|
public function getTotalAttribute(): float
|
2018-05-20 22:53:14 +10:00
|
|
|
{
|
2024-07-09 20:17:30 +10:00
|
|
|
return $this->getSubTotalAttribute()+$this->getTaxTotalAttribute();
|
2018-05-20 22:53:14 +10:00
|
|
|
}
|
|
|
|
|
2022-04-22 14:41:18 +10:00
|
|
|
/* METHODS */
|
|
|
|
|
2020-05-25 17:45:17 +10:00
|
|
|
/**
|
|
|
|
* Return a download link for non-auth downloads
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function download_link(): string
|
|
|
|
{
|
|
|
|
// Re-use an existing code
|
|
|
|
$io = Invite::where('for',$this->account->user->email)->first();
|
|
|
|
|
2022-04-22 14:41:18 +10:00
|
|
|
$tokendate = ($x=Carbon::now()->addDays(21)) > ($y=$this->due_at->addDays(21)) ? $x : $y;
|
2020-05-25 17:45:17 +10:00
|
|
|
|
|
|
|
// Extend the expire date
|
2024-07-09 20:17:30 +10:00
|
|
|
if ($io && ($tokendate > $io->valid_until)) {
|
2020-05-25 17:45:17 +10:00
|
|
|
$io->valid_until = $tokendate;
|
|
|
|
$io->save();
|
|
|
|
}
|
|
|
|
|
2024-07-09 20:17:30 +10:00
|
|
|
$code = (! $io)
|
|
|
|
? Doorman::generate()
|
|
|
|
->for($this->account->user->email)
|
|
|
|
->uses(0)
|
|
|
|
->expiresOn($tokendate)
|
|
|
|
->make()
|
|
|
|
->first()
|
|
|
|
->code
|
|
|
|
: $io->code;
|
2020-05-25 17:45:17 +10:00
|
|
|
|
|
|
|
return url('u/invoice',[$this->id,'email',$code]);
|
|
|
|
}
|
|
|
|
|
2024-07-09 20:17:30 +10:00
|
|
|
/**
|
|
|
|
* Return all the items on an invoice for a particular service and product
|
|
|
|
*
|
|
|
|
* @param Product $po
|
|
|
|
* @param Service $so
|
|
|
|
* @return Collection
|
|
|
|
*/
|
|
|
|
public function product_service_items(Product $po,Service $so): Collection
|
2018-08-01 17:09:38 +10:00
|
|
|
{
|
2024-07-09 20:17:30 +10:00
|
|
|
return $this
|
|
|
|
->items_active
|
|
|
|
->filter(fn($item)=>($item->product_id === $po->id) && ($item->service_id === $so->id))
|
|
|
|
->sortBy('item_type');
|
2018-08-01 17:09:38 +10:00
|
|
|
}
|
2020-04-01 23:35:06 +11:00
|
|
|
|
2020-05-25 17:45:17 +10:00
|
|
|
/**
|
|
|
|
* @param string $key
|
2022-06-13 19:58:53 +10:00
|
|
|
* @return array
|
2020-05-25 17:45:17 +10:00
|
|
|
* @todo Ugly hack to update reminders
|
|
|
|
*/
|
2022-06-13 19:58:53 +10:00
|
|
|
public function reminders(string $key): array
|
|
|
|
{
|
|
|
|
$r = $this->reminders;
|
2020-05-25 17:45:17 +10:00
|
|
|
if (! Arr::get($r,$key)) {
|
|
|
|
$r[$key] = time();
|
|
|
|
}
|
2022-06-13 19:58:53 +10:00
|
|
|
|
|
|
|
return $r;
|
2020-05-25 17:45:17 +10:00
|
|
|
}
|
|
|
|
|
2020-04-01 23:35:06 +11:00
|
|
|
/**
|
2022-04-22 14:41:18 +10:00
|
|
|
* Automatically set our due_at at save time.
|
2020-04-01 23:35:06 +11:00
|
|
|
*
|
|
|
|
* @param array $options
|
|
|
|
* @return bool
|
2024-07-09 20:17:30 +10:00
|
|
|
* @todo Change this to a saving event
|
2020-04-01 23:35:06 +11:00
|
|
|
*/
|
2022-06-13 19:58:53 +10:00
|
|
|
public function save(array $options = [])
|
|
|
|
{
|
2020-04-01 23:35:06 +11:00
|
|
|
// Automatically set the date_due attribute for new records.
|
2022-04-22 14:41:18 +10:00
|
|
|
if (! $this->exists AND ! $this->due_at) {
|
|
|
|
$this->due_at = $this->items->min('start_at');
|
2020-04-01 23:35:06 +11:00
|
|
|
|
|
|
|
// @todo This 7 days should be sysetm configurable
|
2022-04-22 14:41:18 +10:00
|
|
|
if (($x=Carbon::now()->addDay(7)) > $this->due_at)
|
|
|
|
$this->due_at = $x;
|
2020-04-01 23:35:06 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
return parent::save($options);
|
|
|
|
}
|
2024-07-09 20:17:30 +10:00
|
|
|
|
2024-08-03 10:06:25 +10:00
|
|
|
/**
|
|
|
|
* Record the invoice being sent
|
|
|
|
*
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
public function send(): int
|
|
|
|
{
|
|
|
|
$result = Mail::to($this->account->user->email)
|
|
|
|
->send(new InvoiceEmail($this));
|
|
|
|
|
|
|
|
$this->print_status = TRUE;
|
|
|
|
|
|
|
|
if ($this->reminders->has('sent'))
|
|
|
|
$this->reminders->put('sent',collect($this->reminders->get('sent')));
|
|
|
|
else
|
|
|
|
$this->reminders->put('sent',collect());
|
|
|
|
|
|
|
|
$this->reminders->get('sent')->push(Carbon::now());
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
2024-07-09 20:17:30 +10:00
|
|
|
/**
|
|
|
|
* Group the invoice items by product ID, returning the number of products and total
|
|
|
|
*
|
|
|
|
* @return Collection
|
|
|
|
*/
|
|
|
|
public function summary_products(): Collection
|
|
|
|
{
|
|
|
|
$return = collect();
|
|
|
|
|
2024-08-03 10:06:25 +10:00
|
|
|
foreach ($this->items_active->groupBy('product_id') as $id => $o) {
|
|
|
|
if (! $id) {
|
|
|
|
$po = new Product;
|
|
|
|
$po->translate = new ProductTranslate;
|
|
|
|
$po->translate->name_detail = 'Miscellanious';
|
|
|
|
|
|
|
|
} else {
|
|
|
|
$po = $o->first()->product;
|
|
|
|
}
|
|
|
|
|
2024-07-09 20:17:30 +10:00
|
|
|
$po->count = count($o->pluck('service_id')->unique());
|
|
|
|
|
|
|
|
$return->push([
|
2024-08-03 10:06:25 +10:00
|
|
|
'product' => $po,
|
2024-07-09 20:17:30 +10:00
|
|
|
'services' => $o->pluck('service_id')->unique(),
|
|
|
|
'sub_total' => $o->sum('sub_total'),
|
|
|
|
'tax_total' => $o->sum('tax'),
|
|
|
|
'total' => $o->sum('total'),
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $return->sortBy('product.name');
|
|
|
|
}
|
2024-08-03 10:06:25 +10:00
|
|
|
|
|
|
|
public function summary_other(): Collection
|
|
|
|
{
|
|
|
|
$result = collect();
|
|
|
|
|
|
|
|
foreach ($this->items_active->whereNull('service_id') as $o) {
|
|
|
|
dd($o);
|
|
|
|
$result->push([
|
|
|
|
'description' => 'Account Items',
|
|
|
|
'sub_total' => $o->sub_total,
|
|
|
|
'tax_total' => $o->tax,
|
|
|
|
'total' => $o->total,
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
}
|
2018-05-20 22:53:14 +10:00
|
|
|
}
|