tax_collection = collect([]); $this->invoice = $invoice; if ($this->invoice->client) { $this->currency = $this->invoice->client->currency(); $this->client = $this->invoice->client; $this->shouldCalculateTax(); } else { $this->currency = $this->invoice->vendor->currency(); } $this->line_items = []; } public function process(): self { if (!$this->invoice->line_items || !is_array($this->invoice->line_items)) { $this->items = []; return $this; } $this->calcLineItems(); return $this; } private function calcLineItems(): self { foreach ($this->invoice->line_items as $this->item) { $this->cleanLineItem() ->sumLineItem() ->setDiscount() ->calcTaxes() ->push(); } return $this; } private function shouldCalculateTax(): self { if (!$this->invoice->company?->calculate_taxes) { $this->calc_tax = false; return $this; } //should we be filtering by client country here? do we need to reflect at the company <=> client level? if (in_array($this->client->country->iso_3166_2, $this->tax_jurisdictions)) { //only calculate for supported tax jurisdictions $class = "App\DataMapper\Tax\\".$this->client->company->country()->iso_3166_2."\\Rule"; $this->rule = new $class(); $this->rule ->setEntity($this->invoice) ->init(); $this->calc_tax = true; return $this; } return $this; } private function push(): self { $this->sub_total += $this->getLineTotal(); $this->gross_sub_total += $this->getGrossLineTotal(); $this->line_items[] = $this->item; return $this; } private function sumLineItem() { $this->setLineTotal($this->item->cost * $this->item->quantity); return $this; } private function setDiscount() { if ($this->invoice->is_amount_discount) { $this->setLineTotal($this->getLineTotal() - $this->formatValue($this->item->discount, $this->currency->precision)); } else { $discount = ($this->item->line_total * ($this->item->discount / 100)); $this->setLineTotal($this->formatValue(($this->getLineTotal() - $discount), $this->currency->precision)); } $this->item->is_amount_discount = $this->invoice->is_amount_discount; return $this; } /** * Attempts to calculate taxes based on the clients location * * @return self */ private function calcTaxesAutomatically(): self { $this->rule->tax($this->item); $this->item->tax_name1 = $this->rule->tax_name1; $this->item->tax_rate1 = $this->rule->tax_rate1; $this->item->tax_name2 = $this->rule->tax_name2; $this->item->tax_rate2 = $this->rule->tax_rate2; $this->item->tax_name3 = $this->rule->tax_name3; $this->item->tax_rate3 = $this->rule->tax_rate3; return $this; } private function calcTaxes() { if ($this->calc_tax) { $this->calcTaxesAutomatically(); } $item_tax = 0; $amount = $this->item->line_total - ($this->item->line_total * ($this->invoice->discount / 100)); $item_tax_rate1_total = $this->calcAmountLineTax($this->item->tax_rate1, $amount); $item_tax += $item_tax_rate1_total; if (strlen($this->item->tax_name1) > 1) { $this->groupTax($this->item->tax_name1, $this->item->tax_rate1, $item_tax_rate1_total); } $item_tax_rate2_total = $this->calcAmountLineTax($this->item->tax_rate2, $amount); $item_tax += $item_tax_rate2_total; if (strlen($this->item->tax_name2) > 1) { $this->groupTax($this->item->tax_name2, $this->item->tax_rate2, $item_tax_rate2_total); } $item_tax_rate3_total = $this->calcAmountLineTax($this->item->tax_rate3, $amount); $item_tax += $item_tax_rate3_total; if (strlen($this->item->tax_name3) > 1) { $this->groupTax($this->item->tax_name3, $this->item->tax_rate3, $item_tax_rate3_total); } $this->setTotalTaxes($this->formatValue($item_tax, $this->currency->precision)); $this->item->gross_line_total = $this->getLineTotal() + $item_tax; $this->item->tax_amount = $item_tax; return $this; } private function groupTax($tax_name, $tax_rate, $tax_total) { $group_tax = []; $key = str_replace(' ', '', $tax_name.$tax_rate); $group_tax = ['key' => $key, 'total' => $tax_total, 'tax_name' => $tax_name.' '.floatval($tax_rate).'%']; $this->tax_collection->push(collect($group_tax)); } public function getTotalTaxes() { return $this->total_taxes; } public function setTotalTaxes($total) { $this->total_taxes = $total; return $this; } public function setLineTotal($total) { $this->item->line_total = $total; return $this; } public function getLineTotal() { return $this->item->line_total; } public function getGrossLineTotal() { return $this->item->gross_line_total; } public function getLineItems() { return $this->line_items; } public function getGroupedTaxes() { return $this->tax_collection; } public function setGroupedTaxes($group_taxes) { $this->tax_collection = $group_taxes; return $this; } public function getSubTotal() { return $this->sub_total; } public function getGrossSubTotal() { return $this->gross_sub_total; } public function setSubTotal($value) { $this->sub_total = $value; return $this; } /** * Invoice Amount Discount. * * The problem, when calculating invoice level discounts, * the tax collected changes. * * We need to synthetically reduce the line_total amounts * and recalculate the taxes and then pass back * the updated map */ public function calcTaxesWithAmountDiscount() { $this->setGroupedTaxes(collect([])); $item_tax = 0; foreach ($this->line_items as $this->item) { if ($this->item->line_total == 0) { continue; } //$amount = $this->item->line_total - ($this->item->line_total * ($this->invoice->discount / $this->sub_total)); $amount = ($this->sub_total > 0) ? $this->item->line_total - ($this->item->line_total * ($this->invoice->discount / $this->sub_total)) : 0; $item_tax_rate1_total = $this->calcAmountLineTax($this->item->tax_rate1, $amount); $item_tax += $item_tax_rate1_total; if ($item_tax_rate1_total != 0) { $this->groupTax($this->item->tax_name1, $this->item->tax_rate1, $item_tax_rate1_total); } $item_tax_rate2_total = $this->calcAmountLineTax($this->item->tax_rate2, $amount); $item_tax += $item_tax_rate2_total; if ($item_tax_rate2_total != 0) { $this->groupTax($this->item->tax_name2, $this->item->tax_rate2, $item_tax_rate2_total); } $item_tax_rate3_total = $this->calcAmountLineTax($this->item->tax_rate3, $amount); $item_tax += $item_tax_rate3_total; if ($item_tax_rate3_total != 0) { $this->groupTax($this->item->tax_name3, $this->item->tax_rate3, $item_tax_rate3_total); } } $this->setTotalTaxes($item_tax); } /** * Sets default casts for the values in the line_items. * * @return $this */ private function cleanLineItem() { $invoice_item = (object) get_class_vars(InvoiceItem::class); unset($invoice_item->casts); foreach ($invoice_item as $key => $value) { if (! property_exists($this->item, $key) || ! isset($this->item->{$key})) { $this->item->{$key} = $value; $this->item->{$key} = BaseSettings::castAttribute(InvoiceItem::$casts[$key], $value); } } return $this; } }