mirror of
https://github.com/invoiceninja/invoiceninja.git
synced 2024-11-10 21:22:58 +01:00
215 lines
6.2 KiB
PHP
215 lines
6.2 KiB
PHP
<?php
|
|
/**
|
|
* Invoice Ninja (https://invoiceninja.com).
|
|
*
|
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
|
*
|
|
* @copyright Copyright (c) 2022. Invoice Ninja LLC (https://invoiceninja.com)
|
|
*
|
|
* @license https://www.elastic.co/licensing/elastic-license
|
|
*/
|
|
|
|
namespace App\Services\Client;
|
|
|
|
use App\Models\Client;
|
|
use App\Models\Credit;
|
|
use App\Services\Client\Merge;
|
|
use App\Services\Client\PaymentMethod;
|
|
use App\Services\Email\EmailObject;
|
|
use App\Services\Email\EmailService;
|
|
use App\Utils\Number;
|
|
use App\Utils\Traits\MakesDates;
|
|
use Illuminate\Mail\Mailables\Address;
|
|
use Illuminate\Support\Facades\DB;
|
|
|
|
class ClientService
|
|
{
|
|
use MakesDates;
|
|
|
|
private string $client_start_date;
|
|
|
|
private string $client_end_date;
|
|
|
|
public function __construct(private Client $client){}
|
|
|
|
public function updateBalance(float $amount)
|
|
{
|
|
|
|
try {
|
|
DB::connection(config('database.default'))->transaction(function () use($amount) {
|
|
|
|
$this->client = Client::withTrashed()->where('id', $this->client->id)->lockForUpdate()->first();
|
|
$this->client->balance += $amount;
|
|
$this->client->save();
|
|
|
|
}, 1);
|
|
}
|
|
catch (\Throwable $throwable) {
|
|
nlog("DB ERROR " . $throwable->getMessage());
|
|
}
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
public function updateBalanceAndPaidToDate(float $balance, float $paid_to_date)
|
|
{
|
|
|
|
try {
|
|
DB::connection(config('database.default'))->transaction(function () use($balance, $paid_to_date) {
|
|
|
|
$this->client = Client::withTrashed()->where('id', $this->client->id)->lockForUpdate()->first();
|
|
$this->client->balance += $balance;
|
|
$this->client->paid_to_date += $paid_to_date;
|
|
$this->client->save();
|
|
|
|
}, 1);
|
|
}
|
|
catch (\Throwable $throwable) {
|
|
nlog("DB ERROR " . $throwable->getMessage());
|
|
}
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
public function updatePaidToDate(float $amount)
|
|
{
|
|
|
|
DB::connection(config('database.default'))->transaction(function () use($amount) {
|
|
|
|
$this->client = Client::withTrashed()->where('id', $this->client->id)->lockForUpdate()->first();
|
|
$this->client->paid_to_date += $amount;
|
|
$this->client->save();
|
|
|
|
}, 1);
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
public function adjustCreditBalance(float $amount)
|
|
{
|
|
|
|
$this->client->credit_balance += $amount;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
public function getCreditBalance() :float
|
|
{
|
|
|
|
$credits = Credit::withTrashed()->where('client_id', $this->client->id)
|
|
->where('is_deleted', false)
|
|
->where(function ($query) {
|
|
$query->whereDate('due_date', '<=', now()->format('Y-m-d'))
|
|
->orWhereNull('due_date');
|
|
})
|
|
->orderBy('created_at', 'ASC');
|
|
|
|
return Number::roundValue($credits->sum('balance'), $this->client->currency()->precision);
|
|
|
|
}
|
|
|
|
public function getCredits()
|
|
{
|
|
return Credit::where('client_id', $this->client->id)
|
|
->where('is_deleted', false)
|
|
->where('balance', '>', 0)
|
|
->where(function ($query) {
|
|
$query->whereDate('due_date', '<=', now()->format('Y-m-d'))
|
|
->orWhereNull('due_date');
|
|
})
|
|
->orderBy('created_at', 'ASC')->get();
|
|
}
|
|
|
|
public function getPaymentMethods(float $amount)
|
|
{
|
|
return (new PaymentMethod($this->client, $amount))->run();
|
|
}
|
|
|
|
public function merge(Client $mergable_client)
|
|
{
|
|
$this->client = (new Merge($this->client, $mergable_client))->run();
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Generate the client statement.
|
|
*
|
|
* @param array $options
|
|
* @param bool $send_email determines if we should send this statement direct to the client
|
|
*/
|
|
public function statement(array $options = [], bool $send_email = false)
|
|
{
|
|
$statement = (new Statement($this->client, $options));
|
|
|
|
$pdf = $statement->run();
|
|
|
|
if($send_email)
|
|
return $this->emailStatement($pdf, $statement->options);
|
|
|
|
return $pdf;
|
|
}
|
|
|
|
/**
|
|
* Emails the statement to the client
|
|
*
|
|
* @param mixed $pdf The pdf blob
|
|
* @param array $options The statement options array
|
|
* @return void
|
|
*/
|
|
private function emailStatement($pdf, array $options): void
|
|
{
|
|
|
|
$this->client_start_date = $this->translateDate($options['start_date'], $this->client->date_format(), $this->client->locale());
|
|
$this->client_end_date = $this->translateDate($options['end_date'], $this->client->date_format(), $this->client->locale());
|
|
|
|
$email_service = new EmailService($this->buildStatementMailableData($pdf), $this->client->company);
|
|
|
|
$email_service->send();
|
|
|
|
}
|
|
|
|
/**
|
|
* Builds and returns an EmailObject for Client Statements
|
|
*
|
|
* @param mixed $pdf The PDF to send
|
|
* @return EmailObject The EmailObject to send
|
|
*/
|
|
public function buildStatementMailableData($pdf) :EmailObject
|
|
{
|
|
|
|
$email_object = new EmailObject;
|
|
$email_object->to = [new Address($this->client->present()->email(), $this->client->present()->name())];
|
|
$email_object->attachments = [['file' => base64_encode($pdf), 'name' => ctrans('texts.statement') . ".pdf"]];
|
|
$email_object->settings = $this->client->getMergedSettings();
|
|
$email_object->company = $this->client->company;
|
|
$email_object->client = $this->client;
|
|
$email_object->email_template_subject = 'email_subject_statement';
|
|
$email_object->email_template_body = 'email_template_statement';
|
|
$email_object->variables = [
|
|
'$client' => $this->client->present()->name(),
|
|
'$start_date' => $this->client_start_date,
|
|
'$end_date' => $this->client_end_date,
|
|
];
|
|
|
|
return $email_object;
|
|
|
|
}
|
|
|
|
/**
|
|
* Saves the client instance
|
|
*
|
|
* @return Client The Client Model
|
|
*/
|
|
public function save() :Client
|
|
{
|
|
$this->client->save();
|
|
|
|
return $this->client->fresh();
|
|
}
|
|
}
|