mirror of
				https://github.com/invoiceninja/invoiceninja.git
				synced 2025-10-24 19:39:22 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			473 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			473 lines
		
	
	
		
			14 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\Pdf;
 | |
| 
 | |
| use App\DataMapper\CompanySettings;
 | |
| use App\Libraries\MultiDB;
 | |
| use App\Models\Client;
 | |
| use App\Models\ClientContact;
 | |
| use App\Models\Country;
 | |
| use App\Models\Credit;
 | |
| use App\Models\CreditInvitation;
 | |
| use App\Models\Currency;
 | |
| use App\Models\Design;
 | |
| use App\Models\Invoice;
 | |
| use App\Models\InvoiceInvitation;
 | |
| use App\Models\PurchaseOrder;
 | |
| use App\Models\PurchaseOrderInvitation;
 | |
| use App\Models\Quote;
 | |
| use App\Models\QuoteInvitation;
 | |
| use App\Models\RecurringInvoice;
 | |
| use App\Models\RecurringInvoiceInvitation;
 | |
| use App\Models\Vendor;
 | |
| use App\Models\VendorContact;
 | |
| use App\Utils\Ninja;
 | |
| use App\Utils\Traits\AppSetup;
 | |
| use App\Utils\Traits\MakesHash;
 | |
| use Illuminate\Support\Collection;
 | |
| use Illuminate\Support\Facades\App;
 | |
| use Illuminate\Support\Facades\Cache;
 | |
| 
 | |
| class PdfConfiguration
 | |
| {
 | |
|     use MakesHash;
 | |
|     use AppSetup;
 | |
| 
 | |
|     public ?Client $client;
 | |
| 
 | |
|     public ?ClientContact $contact;
 | |
| 
 | |
|     public Country $country;
 | |
| 
 | |
|     public Currency $currency;
 | |
| 
 | |
|     public Client | Vendor $currency_entity;
 | |
| 
 | |
|     public Design $design;
 | |
| 
 | |
|     public Invoice | Credit | Quote | PurchaseOrder | RecurringInvoice $entity;
 | |
| 
 | |
|     public string $entity_design_id;
 | |
| 
 | |
|     public string $entity_string;
 | |
| 
 | |
|     public ?string $path;
 | |
| 
 | |
|     public array $pdf_variables;
 | |
| 
 | |
|     public object $settings;
 | |
| 
 | |
|     public $settings_object;
 | |
| 
 | |
|     public ?Vendor $vendor;
 | |
| 
 | |
|     public ?VendorContact $vendor_contact;
 | |
| 
 | |
|     public string $date_format;
 | |
| 
 | |
|     public string $locale;
 | |
| 
 | |
|     public Collection $tax_map;
 | |
| 
 | |
|     public ?array $total_tax_map;
 | |
|     /**
 | |
|      * __construct
 | |
|      *
 | |
|      * @param  PdfService $service
 | |
|      * @return void
 | |
|      */
 | |
|     public function __construct(public PdfService $service)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * init
 | |
|      *
 | |
|      * @return self
 | |
|      */
 | |
|     public function init(): self
 | |
|     {
 | |
|         MultiDB::setDb($this->service->company->db);
 | |
| 
 | |
|         $this->setEntityType()
 | |
|              ->setDateFormat()
 | |
|              ->setPdfVariables()
 | |
|              ->setDesign()
 | |
|              ->setCurrencyForPdf()
 | |
|              ->setLocale();
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * setLocale
 | |
|      *
 | |
|      * @return self
 | |
|      */
 | |
|     private function setLocale(): self
 | |
|     {
 | |
|         App::forgetInstance('translator');
 | |
| 
 | |
|         $t = app('translator');
 | |
| 
 | |
|         App::setLocale($this->settings_object->locale());
 | |
| 
 | |
|         $t->replace(Ninja::transformTranslations($this->settings));
 | |
| 
 | |
|         $this->locale = $this->settings_object->locale();
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * setCurrency
 | |
|      *
 | |
|      * @return self
 | |
|      */
 | |
|     private function setCurrencyForPdf(): self
 | |
|     {
 | |
|         $this->currency = $this->client ? $this->client->currency() : $this->vendor->currency();
 | |
| 
 | |
|         $this->currency_entity = $this->client ? $this->client : $this->vendor;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * setPdfVariables
 | |
|      *
 | |
|      * @return self
 | |
|      */
 | |
|     public function setPdfVariables(): self
 | |
|     {
 | |
|         $default = (array) CompanySettings::getEntityVariableDefaults();
 | |
| 
 | |
|         // $variables = (array)$this->service->company->settings->pdf_variables;
 | |
|         $variables = (array)$this->settings->pdf_variables;
 | |
| 
 | |
|         foreach ($default as $property => $value) {
 | |
|             if (array_key_exists($property, $variables)) {
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             $variables[$property] = $value;
 | |
|         }
 | |
| 
 | |
|         $this->pdf_variables = $variables;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * setEntityType
 | |
|      *
 | |
|      * @return self
 | |
|      */
 | |
|     private function setEntityType(): self
 | |
|     {
 | |
|         $entity_design_id = '';
 | |
| 
 | |
|         if ($this->service->invitation instanceof InvoiceInvitation) {
 | |
|             $this->entity = $this->service->invitation->invoice;
 | |
|             $this->entity_string = 'invoice';
 | |
|             $this->client = $this->entity->client;
 | |
|             $this->contact = $this->service->invitation->contact;
 | |
|             $this->path = $this->client->invoice_filepath($this->service->invitation);
 | |
|             $this->entity_design_id = 'invoice_design_id';
 | |
|             $this->settings = $this->client->getMergedSettings();
 | |
|             $this->settings_object = $this->client;
 | |
|             $this->country = $this->client->country ?? $this->client->company->country();
 | |
|         } elseif ($this->service->invitation instanceof QuoteInvitation) {
 | |
|             $this->entity = $this->service->invitation->quote;
 | |
|             $this->entity_string = 'quote';
 | |
|             $this->client = $this->entity->client;
 | |
|             $this->contact = $this->service->invitation->contact;
 | |
|             $this->path = $this->client->quote_filepath($this->service->invitation);
 | |
|             $this->entity_design_id = 'quote_design_id';
 | |
|             $this->settings = $this->client->getMergedSettings();
 | |
|             $this->settings_object = $this->client;
 | |
|             $this->country = $this->client->country ?? $this->client->company->country();
 | |
|         } elseif ($this->service->invitation instanceof CreditInvitation) {
 | |
|             $this->entity = $this->service->invitation->credit;
 | |
|             $this->entity_string = 'credit';
 | |
|             $this->client = $this->entity->client;
 | |
|             $this->contact = $this->service->invitation->contact;
 | |
|             $this->path = $this->client->credit_filepath($this->service->invitation);
 | |
|             $this->entity_design_id = 'credit_design_id';
 | |
|             $this->settings = $this->client->getMergedSettings();
 | |
|             $this->settings_object = $this->client;
 | |
|             $this->country = $this->client->country ?? $this->client->company->country();
 | |
|         } elseif ($this->service->invitation instanceof RecurringInvoiceInvitation) {
 | |
|             $this->entity = $this->service->invitation->recurring_invoice;
 | |
|             $this->entity_string = 'recurring_invoice';
 | |
|             $this->client = $this->entity->client;
 | |
|             $this->contact = $this->service->invitation->contact;
 | |
|             $this->path = $this->client->recurring_invoice_filepath($this->service->invitation);
 | |
|             $this->entity_design_id = 'invoice_design_id';
 | |
|             $this->settings = $this->client->getMergedSettings();
 | |
|             $this->settings_object = $this->client;
 | |
|             $this->country = $this->client->country ?? $this->client->company->country();
 | |
|         } elseif ($this->service->invitation instanceof PurchaseOrderInvitation) {
 | |
|             $this->entity = $this->service->invitation->purchase_order;
 | |
|             $this->entity_string = 'purchase_order';
 | |
|             $this->vendor = $this->entity->vendor;
 | |
|             $this->vendor_contact = $this->service->invitation->contact;
 | |
|             $this->path = $this->vendor->purchase_order_filepath($this->service->invitation);
 | |
|             $this->entity_design_id = 'purchase_order_design_id';
 | |
|             $this->settings = $this->vendor->company->settings;
 | |
|             $this->settings_object = $this->vendor;
 | |
|             $this->client = null;
 | |
|             $this->country = $this->vendor->country ?? $this->vendor->company->country();
 | |
|         } else {
 | |
|             throw new \Exception('Unable to resolve entity', 500);
 | |
|         }
 | |
| 
 | |
|         $this->setTaxMap($this->entity->calc()->getTaxMap());
 | |
|         $this->setTotalTaxMap($this->entity->calc()->getTotalTaxMap());
 | |
| 
 | |
|         $this->path = $this->path.$this->entity->numberFormatter().'.pdf';
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     public function setTaxMap($map): self
 | |
|     {
 | |
|         $this->tax_map = $map;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     public function setTotalTaxMap($map): self
 | |
|     {
 | |
|         $this->total_tax_map = $map;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     public function setCurrency(Currency $currency): self
 | |
|     {
 | |
|         $this->currency = $currency;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     public function setCountry(Country $country): self
 | |
|     {
 | |
|         $this->country = $country;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * setDesign
 | |
|      *
 | |
|      * @return self
 | |
|      */
 | |
|     private function setDesign(): self
 | |
|     {
 | |
| 
 | |
|         $design_id = $this->entity->design_id ?: $this->decodePrimaryKey($this->settings_object->getSetting($this->entity_design_id));
 | |
| 
 | |
|         $this->design = Design::withTrashed()->find($design_id) ?? Design::withTrashed()->find(2);
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * formatMoney
 | |
|      *
 | |
|      * @param  float $value
 | |
|      * @return string
 | |
|      */
 | |
|     public function formatMoney($value): string
 | |
|     {
 | |
|         $value = floatval($value);
 | |
| 
 | |
|         $thousand = $this->currency->thousand_separator;
 | |
|         $decimal = $this->currency->decimal_separator;
 | |
|         $precision = $this->currency->precision;
 | |
|         $code = $this->currency->code;
 | |
|         $swapSymbol = $this->currency->swap_currency_symbol;
 | |
| 
 | |
|         if (isset($this->country->thousand_separator) && strlen($this->country->thousand_separator) >= 1) {
 | |
|             $thousand = $this->country->thousand_separator;
 | |
|         }
 | |
| 
 | |
|         if (isset($this->country->decimal_separator) && strlen($this->country->decimal_separator) >= 1) {
 | |
|             $decimal = $this->country->decimal_separator;
 | |
|         }
 | |
| 
 | |
|         if (isset($this->country->swap_currency_symbol) && strlen($this->country->swap_currency_symbol) >= 1) {
 | |
|             $swapSymbol = $this->country->swap_currency_symbol;
 | |
|         }
 | |
| 
 | |
|         $value = number_format($value, $precision, $decimal, $thousand);
 | |
|         $symbol = $this->currency->symbol;
 | |
| 
 | |
|         if ($this->settings->show_currency_code === true && $this->currency->code == 'CHF') {
 | |
|             return "{$code} {$value}";
 | |
|         } elseif ($this->settings->show_currency_code === true) {
 | |
|             return "{$value} {$code}";
 | |
|         } elseif ($swapSymbol) {
 | |
|             return "{$value} ".trim($symbol);
 | |
|         } elseif ($this->settings->show_currency_code === false) {
 | |
|             return "{$symbol}{$value}";
 | |
|         } else {
 | |
|             $value = floatval($value);
 | |
|             $thousand = $this->currency->thousand_separator;
 | |
|             $decimal = $this->currency->decimal_separator;
 | |
|             $precision = $this->currency->precision;
 | |
| 
 | |
|             return number_format($value, $precision, $decimal, $thousand);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Formats a given value based on the clients currency.
 | |
|      *
 | |
|      * @param  float  $value    The number to be formatted
 | |
|      *
 | |
|      * @return string           The formatted value
 | |
|      */
 | |
|     public function formatValueNoTrailingZeroes($value): string
 | |
|     {
 | |
|         $value = floatval($value);
 | |
| 
 | |
|         $thousand = $this->currency->thousand_separator;
 | |
|         $decimal = $this->currency->decimal_separator;
 | |
| 
 | |
|         /* Country settings override client settings */
 | |
|         if (isset($this->country->thousand_separator) && strlen($this->country->thousand_separator) >= 1) {
 | |
|             $thousand = $this->country->thousand_separator;
 | |
|         }
 | |
| 
 | |
|         if (isset($this->country->decimal_separator) && strlen($this->country->decimal_separator) >= 1) {
 | |
|             $decimal = $this->country->decimal_separator;
 | |
|         }
 | |
| 
 | |
|         $precision = 10;
 | |
| 
 | |
|         return rtrim(rtrim(number_format($value, $precision, $decimal, $thousand), '0'), $decimal);
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /**
 | |
|      * Formats a given value based on the clients currency AND country.
 | |
|      *
 | |
|      * @param float $value The number to be formatted
 | |
|      * @return string           The formatted value
 | |
|      */
 | |
|     public function formatMoneyNoRounding($value): string
 | |
|     {
 | |
| 
 | |
|         $_value = $value;
 | |
| 
 | |
|         $thousand = $this->currency->thousand_separator;
 | |
|         $decimal = $this->currency->decimal_separator;
 | |
|         $precision = $this->currency->precision;
 | |
|         $code = $this->currency->code;
 | |
|         $swapSymbol = $this->currency->swap_currency_symbol;
 | |
| 
 | |
|         /* Country settings override client settings */
 | |
|         if (isset($this->country->thousand_separator) && strlen($this->country->thousand_separator) >= 1) {
 | |
|             $thousand = $this->country->thousand_separator;
 | |
|         }
 | |
| 
 | |
|         if (isset($this->country->decimal_separator) && strlen($this->country->decimal_separator) >= 1) {
 | |
|             $decimal = $this->country->decimal_separator;
 | |
|         }
 | |
| 
 | |
|         if (isset($this->country->swap_currency_symbol) && strlen($this->country->swap_currency_symbol) >= 1) {
 | |
|             $swapSymbol = $this->country->swap_currency_symbol;
 | |
|         }
 | |
| 
 | |
|         /* 08-01-2022 allow increased precision for unit price*/
 | |
|         $v = rtrim(sprintf('%f', $value), '0');
 | |
|         $parts = explode('.', $v);
 | |
| 
 | |
|         /* 08-02-2023 special if block to render $0.5 to $0.50*/
 | |
|         if ($v < 1 && strlen($v) == 3) {
 | |
|             $precision = 2;
 | |
|         } elseif ($v < 1) {
 | |
|             $precision = strlen($v) - strrpos($v, '.') - 1;
 | |
|         }
 | |
| 
 | |
|         if (is_array($parts) && $parts[0] != 0) {
 | |
|             $precision = 2;
 | |
|         }
 | |
| 
 | |
|         //04-04-2023 if currency = JPY override precision to 0
 | |
|         if($this->currency->code == 'JPY') {
 | |
|             $precision = 0;
 | |
|         }
 | |
| 
 | |
|         $value = number_format($v, $precision, $decimal, $thousand);
 | |
|         $symbol = $this->currency->symbol;
 | |
| 
 | |
|         if ($this->settings->show_currency_code === true && $this->currency->code == 'CHF') {
 | |
|             return "{$code} {$value}";
 | |
|         } elseif ($this->settings->show_currency_code === true) {
 | |
|             return "{$value} {$code}";
 | |
|         } elseif ($swapSymbol) {
 | |
|             return "{$value} ".trim($symbol);
 | |
|         } elseif ($this->settings->show_currency_code === false) {
 | |
|             if ($_value < 0) {
 | |
|                 $value = substr($value, 1);
 | |
|                 $symbol = "-{$symbol}";
 | |
|             }
 | |
| 
 | |
|             return "{$symbol}{$value}";
 | |
|         } else {
 | |
|             return $this->formatValue($value);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Formats a given value based on the clients currency.
 | |
|      *
 | |
|      * @param  float  $value    The number to be formatted
 | |
|      *
 | |
|      * @return string           The formatted value
 | |
|      */
 | |
|     public function formatValue($value): string
 | |
|     {
 | |
|         $value = floatval($value);
 | |
| 
 | |
|         $thousand = $this->currency->thousand_separator;
 | |
|         $decimal = $this->currency->decimal_separator;
 | |
|         $precision = $this->currency->precision;
 | |
| 
 | |
|         return number_format($value, $precision, $decimal, $thousand);
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /**
 | |
|      * date_format
 | |
|      *
 | |
|      * @return self
 | |
|      */
 | |
|     public function setDateFormat(): self
 | |
|     {
 | |
|         $date_formats = Cache::get('date_formats');
 | |
| 
 | |
|         if (! $date_formats) {
 | |
|             $this->buildCache(true);
 | |
|         }
 | |
| 
 | |
|         $this->date_format = $date_formats->filter(function ($item) {
 | |
|             return $item->id == $this->settings->date_format_id;
 | |
|         })->first()->format;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| }
 |