From e94cc28a6153e954ea3848c77c7bc9d62bbe8282 Mon Sep 17 00:00:00 2001 From: Erawat Chamanont Date: Fri, 22 Sep 2023 12:27:20 +0100 Subject: [PATCH] BTHAB-186: Update case opportunity details Update case opportunity statuses, and total amounts when: * Create or update sales order contribution * Create or update sales order * Create a payment --- .../Hook/Post/CaseSalesOrderPayment.php | 76 ++++++-- .../Post/CreateSalesOrderContribution.php | 19 +- .../AbstractBaseSalesOrderCalculator.php | 94 ++++++++++ .../CaseSalesOrderContributionCalculator.php | 80 +++----- .../CaseSalesOrderOpportunityCalculator.php | 171 ++++++++++++++++++ .../CaseSalesOrder/SalesOrderSaveAction.php | 26 +++ 6 files changed, 388 insertions(+), 78 deletions(-) create mode 100644 CRM/Civicase/Service/AbstractBaseSalesOrderCalculator.php create mode 100644 CRM/Civicase/Service/CaseSalesOrderOpportunityCalculator.php diff --git a/CRM/Civicase/Hook/Post/CaseSalesOrderPayment.php b/CRM/Civicase/Hook/Post/CaseSalesOrderPayment.php index ead823b27..894a0d81a 100644 --- a/CRM/Civicase/Hook/Post/CaseSalesOrderPayment.php +++ b/CRM/Civicase/Hook/Post/CaseSalesOrderPayment.php @@ -21,28 +21,37 @@ class CRM_Civicase_Hook_Post_CaseSalesOrderPayment { * Object reference. */ public function run($op, $objectName, $objectId, &$objectRef) { - if (!$this->shouldRun($op, $objectName)) { + if (!$this->shouldRun($op, $objectName, $objectRef)) { return; } - $entityFinancialTrxn = civicrm_api3('EntityFinancialTrxn', 'get', [ - 'sequential' => 1, - 'entity_table' => 'civicrm_contribution', - 'financial_trxn_id' => $objectRef->financial_trxn_id, - ]); - - if (empty($entityFinancialTrxn['values'][0])) { + $financialTrnxId = $objectRef->financial_trxn_id; + if (empty($financialTrnxId)) { return; } - $contributionId = $entityFinancialTrxn['values'][0]['entity_id']; + $contributionId = $this->getContributionId($financialTrnxId); + if (empty($contributionId)) { + return; + } - $salesOrderID = Contribution::get() - ->addSelect('Opportunity_Details.Quotation') + $contribution = Contribution::get() + ->addSelect('Opportunity_Details.Case_Opportunity', 'Opportunity_Details.Quotation') ->addWhere('id', '=', $contributionId) ->execute() - ->first()['Opportunity_Details.Quotation']; + ->first(); + + $this->updateQuotationFinancialStatuses($contribution['Opportunity_Details.Quotation']); + $this->updateCaseOpportunityFinancialDetails($contribution['Opportunity_Details.Case_Opportunity']); + } + /** + * Updates CaseSalesOrder financial statuses. + * + * @param int $salesOrderID + * CaseSalesOrder ID. + */ + private function updateQuotationFinancialStatuses(int $salesOrderID): void { if (empty($salesOrderID)) { return; } @@ -68,6 +77,43 @@ public function run($op, $objectName, $objectId, &$objectRef) { } } + /** + * Updates Case financial statuses. + * + * @param int? $caseId + * CaseSalesOrder ID. + */ + private function updateCaseOpportunityFinancialDetails(?int $caseId) { + if (empty($caseId)) { + return; + } + + try { + $calculator = new CRM_Civicase_Service_CaseSalesOrderOpportunityCalculator($caseId); + $calculator->updateOpportunityFinancialDetails(); + } + catch (\Throwable $th) { + CRM_Core_Error::statusBounce(ts('Error updating opportunity details')); + } + } + + /** + * Gets Contribution ID by Financial Transaction ID. + */ + private function getContributionId($financialTrxnId) { + $entityFinancialTrxn = civicrm_api3('EntityFinancialTrxn', 'get', [ + 'sequential' => 1, + 'entity_table' => 'civicrm_contribution', + 'financial_trxn_id' => $financialTrxnId, + ]); + + if (empty($entityFinancialTrxn['values'][0])) { + return NULL; + } + + return $entityFinancialTrxn['values'][0]['entity_id']; + } + /** * Determines if the hook should run or not. * @@ -75,12 +121,14 @@ public function run($op, $objectName, $objectId, &$objectRef) { * The operation being performed. * @param string $objectName * Object name. + * @param string $objectRef + * The hook object reference. * * @return bool * returns a boolean to determine if hook will run or not. */ - private function shouldRun($op, $objectName) { - return $objectName == 'EntityFinancialTrxn' && $op == 'create'; + private function shouldRun($op, $objectName, $objectRef) { + return $objectName == 'EntityFinancialTrxn' && $op == 'create' && property_exists($objectRef, 'financial_trxn_id'); } } diff --git a/CRM/Civicase/Hook/Post/CreateSalesOrderContribution.php b/CRM/Civicase/Hook/Post/CreateSalesOrderContribution.php index 86285be05..e93bf6078 100644 --- a/CRM/Civicase/Hook/Post/CreateSalesOrderContribution.php +++ b/CRM/Civicase/Hook/Post/CreateSalesOrderContribution.php @@ -34,13 +34,15 @@ public function run($op, $objectName, $objectId, &$objectRef) { return; } + $salesOrder = CaseSalesOrder::get() + ->addSelect('status_id', 'case_id') + ->addWhere('id', '=', $salesOrderId) + ->execute() + ->first(); + $salesOrderStatusId = CRM_Utils_Request::retrieve('sales_order_status_id', 'Integer'); if (empty($salesOrderStatusId)) { - $salesOrderStatusId = CaseSalesOrder::get() - ->addSelect('status_id') - ->addWhere('id', '=', $salesOrderId) - ->execute() - ->first()['status_id']; + $salesOrder = $salesOrder['status_id']; } $transaction = CRM_Core_Transaction::create(); @@ -55,6 +57,13 @@ public function run($op, $objectName, $objectId, &$objectRef) { ->addValue('invoicing_status_id', $invoicingStatusID) ->addValue('payment_status_id', $paymentStatusID) ->execute(); + + $caseId = $salesOrder['case_id']; + if (empty($caseId)) { + return; + } + $calculator = new CRM_Civicase_Service_CaseSalesOrderOpportunityCalculator($caseId); + $calculator->updateOpportunityFinancialDetails(); } catch (\Throwable $th) { $transaction->rollback(); diff --git a/CRM/Civicase/Service/AbstractBaseSalesOrderCalculator.php b/CRM/Civicase/Service/AbstractBaseSalesOrderCalculator.php new file mode 100644 index 000000000..1e9b74b2c --- /dev/null +++ b/CRM/Civicase/Service/AbstractBaseSalesOrderCalculator.php @@ -0,0 +1,94 @@ +paymentStatusOptionValues = $this->getOptionValues('case_sales_order_payment_status'); + $this->invoicingStatusOptionValues = $this->getOptionValues('case_sales_order_invoicing_status'); + } + + /** + * Gets option values by option group name. + * + * @param string $name + * Option group name. + * + * @return array + * Option values. + * + * @throws API_Exception + * @throws \Civi\API\Exception\UnauthorizedException + */ + protected function getOptionValues($name) { + return OptionValue::get(FALSE) + ->addSelect('*') + ->addWhere('option_group_id:name', '=', $name) + ->execute() + ->getArrayCopy(); + } + + /** + * Gets status (option values' value) from the given options. + * + * @param string $needle + * Search value. + * @param array $options + * Option value. + * + * @return string + * Option values' value. + */ + protected function getValueFromOptionValues($needle, $options) { + $key = array_search($needle, array_column($options, 'name')); + + return $options[$key]['value']; + } + + /** + * Gets status (option values' label) from the given options. + * + * @param string $needle + * Search value. + * @param array $options + * Option value. + * + * @return string + * Option values' value. + */ + protected function getLabelFromOptionValues($needle, $options) { + $key = array_search($needle, array_column($options, 'name')); + + return $options[$key]['label']; + } + +} diff --git a/CRM/Civicase/Service/CaseSalesOrderContributionCalculator.php b/CRM/Civicase/Service/CaseSalesOrderContributionCalculator.php index 1e3a99ea8..2e44c8470 100644 --- a/CRM/Civicase/Service/CaseSalesOrderContributionCalculator.php +++ b/CRM/Civicase/Service/CaseSalesOrderContributionCalculator.php @@ -2,7 +2,6 @@ use Civi\Api4\CaseSalesOrder; use Civi\Api4\Contribution; -use Civi\Api4\OptionValue; /** * Case Sale Order Contribution Service. @@ -10,7 +9,7 @@ * This class provides calculations for payment and invoices that * attached to the sale order. */ -class CRM_Civicase_Service_CaseSalesOrderContributionCalculator { +class CRM_Civicase_Service_CaseSalesOrderContributionCalculator extends CRM_Civicase_Service_AbstractBaseSalesOrderCalculator { /** * Case Sales Order object. @@ -18,18 +17,7 @@ class CRM_Civicase_Service_CaseSalesOrderContributionCalculator { * @var array|null */ private ?array $salesOrder; - /** - * Case Sales Order payment status option values. - * - * @var array - */ - private array $paymentStatusOptionValues; - /** - * Case Sales Order Invoicing status option values. - * - * @var array - */ - private array $invoicingStatusOptionValues; + /** * List of contributions that links to the sales order. * @@ -60,12 +48,12 @@ class CRM_Civicase_Service_CaseSalesOrderContributionCalculator { * @throws \Civi\API\Exception\UnauthorizedException */ public function __construct($salesOrderId) { + parent::__construct(); $this->salesOrder = $this->getSalesOrder($salesOrderId); - $this->paymentStatusOptionValues = $this->getOptionValues('case_sales_order_payment_status'); - $this->invoicingStatusOptionValues = $this->getOptionValues('case_sales_order_invoicing_status'); $this->contributions = $this->getContributions(); $this->totalInvoicedAmount = $this->getTotalInvoicedAmount(); $this->totalPaymentsAmount = $this->getTotalPaymentsAmount(); + } /** @@ -89,6 +77,13 @@ public function calculateTotalPaidAmount(): float { return $this->getTotalPaymentsAmount(); } + /** + * Gets SalesOrder Total amount after tax. + */ + public function getQuotedAmount(): float { + return $this->salesOrder['total_after_tax']; + } + /** * Calculates invoicing status. * @@ -97,15 +92,15 @@ public function calculateTotalPaidAmount(): float { */ public function calculateInvoicingStatus() { if (empty($this->salesOrder) || empty($this->contributions)) { - return $this->getStatus('no_invoices', $this->invoicingStatusOptionValues); + return $this->getValueFromOptionValues(parent::INVOICING_STATUS_NO_INVOICES, $this->invoicingStatusOptionValues); } $quotationTotalAmount = $this->salesOrder['total_after_tax']; if ($this->totalInvoicedAmount < $quotationTotalAmount) { - return $this->getStatus('partially_invoiced', $this->invoicingStatusOptionValues); + return $this->getValueFromOptionValues(parent::INVOICING_STATUS_PARTIALLY_INVOICED, $this->invoicingStatusOptionValues); } - return $this->getStatus('fully_invoiced', $this->invoicingStatusOptionValues); + return $this->getValueFromOptionValues(parent::INVOICING_STATUS_FULLY_INVOICED, $this->invoicingStatusOptionValues); } /** @@ -117,14 +112,18 @@ public function calculateInvoicingStatus() { public function calculatePaymentStatus() { if (empty($this->salesOrder) || empty($this->contributions) || !($this->totalPaymentsAmount > 0)) { - return $this->getStatus('no_payments', $this->paymentStatusOptionValues); + return $this->getValueFromOptionValues(parent::PAYMENT_STATUS_NO_PAYMENTS, $this->paymentStatusOptionValues); } if ($this->totalPaymentsAmount < $this->totalInvoicedAmount) { - return $this->getStatus('partially_paid', $this->paymentStatusOptionValues); + return $this->getValueFromOptionValues(parent::PAYMENT_STATUS_PARTIALLY_PAID, $this->paymentStatusOptionValues); } - return $this->getStatus('fully_paid', $this->paymentStatusOptionValues); + if ($this->totalPaymentsAmount > $this->totalInvoicedAmount) { + return $this->getValueFromOptionValues(parent::PAYMENT_STATUS_OVERPAID, $this->paymentStatusOptionValues); + } + + return $this->getValueFromOptionValues(parent::PAYMENT_STATUS_FULLY_PAID, $this->paymentStatusOptionValues); } /** @@ -166,26 +165,6 @@ private function getSalesOrder($saleOrderId) { ->first(); } - /** - * Gets option values by option group name. - * - * @param string $name - * Option group name. - * - * @return array - * Option values. - * - * @throws API_Exception - * @throws \Civi\API\Exception\UnauthorizedException - */ - private function getOptionValues($name) { - return OptionValue::get() - ->addSelect('*') - ->addWhere('option_group_id:name', '=', $name) - ->execute() - ->getArrayCopy(); - } - /** * Gets total invoiced amount. * @@ -223,21 +202,4 @@ private function getTotalPaymentsAmount(): float { return $totalPaymentsAmount; } - /** - * Gets status (option values' value) from the given options. - * - * @param string $needle - * Search value. - * @param array $options - * Option value. - * - * @return string - * Option values' value. - */ - private function getStatus($needle, $options) { - $key = array_search($needle, array_column($options, 'name')); - - return $options[$key]['value']; - } - } diff --git a/CRM/Civicase/Service/CaseSalesOrderOpportunityCalculator.php b/CRM/Civicase/Service/CaseSalesOrderOpportunityCalculator.php new file mode 100644 index 000000000..b73923908 --- /dev/null +++ b/CRM/Civicase/Service/CaseSalesOrderOpportunityCalculator.php @@ -0,0 +1,171 @@ +caseId = $caseId; + $contributions = $this->getContributions($caseId); + $this->calculateOpportunityFinancialAmount($contributions); + } + + /** + * Calculates total invoiced amount. + * + * @return float + * Total invoiced amount. + */ + public function calculateTotalInvoicedAmount(): float { + return $this->totalInvoicedAmount; + } + + /** + * Calculates total paid amount. + * + * @return float + * Total paid amounts. + */ + public function calculateTotalPaidAmount(): float { + return $this->totalPaidAmount; + } + + /** + * Calculates total quoted amount. + * + * @return float + * Total paid amounts. + */ + public function calculateTotalQuotedAmount(): float { + return $this->totalQuotedAmount; + } + + /** + * Calculates opportunity invoicing status. + * + * @return string + * Invoicing status option value's value + */ + public function calculateInvoicingStatus() { + if (!($this->totalInvoicedAmount > 0)) { + return $this->getLabelFromOptionValues(parent::INVOICING_STATUS_NO_INVOICES, $this->invoicingStatusOptionValues); + } + + if ($this->totalInvoicedAmount < $this->totalQuotedAmount) { + return $this->getLabelFromOptionValues(parent::INVOICING_STATUS_PARTIALLY_INVOICED, $this->invoicingStatusOptionValues); + } + + return $this->getLabelFromOptionValues(parent::INVOICING_STATUS_FULLY_INVOICED, $this->invoicingStatusOptionValues); + } + + /** + * Calculates opportunity payment status. + * + * @return string + * Payment status option value's value + */ + public function calculatePaymentStatus() { + if (!($this->totalPaidAmount > 0)) { + return $this->getLabelFromOptionValues(parent::PAYMENT_STATUS_NO_PAYMENTS, $this->paymentStatusOptionValues); + } + + if ($this->totalPaidAmount < $this->totalInvoicedAmount) { + return $this->getLabelFromOptionValues(parent::PAYMENT_STATUS_PARTIALLY_PAID, $this->paymentStatusOptionValues); + } + + if ($this->totalPaidAmount > $this->totalInvoicedAmount) { + return $this->getLabelFromOptionValues(parent::PAYMENT_STATUS_OVERPAID, $this->paymentStatusOptionValues); + + } + + return $this->getLabelFromOptionValues(parent::PAYMENT_STATUS_FULLY_PAID, $this->paymentStatusOptionValues); + } + + /** + * Updates opportunity financial details. + */ + public function updateOpportunityFinancialDetails(): void { + CiviCase::update() + ->addValue('Case_Opportunity_Details.Total_Amount_Quoted', $this->calculateTotalQuotedAmount()) + ->addValue('Case_Opportunity_Details.Total_Amount_Invoiced', $this->calculateTotalQuotedAmount()) + ->addValue('Case_Opportunity_Details.Invoicing_Status', $this->calculateInvoicingStatus()) + ->addValue('Case_Opportunity_Details.Total_Amounts_Paid', $this->calculateTotalPaidAmount()) + ->addValue('Case_Opportunity_Details.Payments_Status', $this->calculatePaymentStatus()) + ->addWhere('id', '=', $this->caseId) + ->execute(); + } + + /** + * Calculates opportunity financial amounts. + * + * @param array $contributions + * List of contributions that link to the opportunity. + */ + private function calculateOpportunityFinancialAmount($contributions) { + $totalQuotedAmount = 0; + $totalInvoicedAmount = 0; + $totalPaidAmount = 0; + foreach ($contributions as $contribution) { + $salesOrderId = $contribution['Opportunity_Details.Quotation']; + $caseSaleOrderContributionService = new CRM_Civicase_Service_CaseSalesOrderContributionCalculator($salesOrderId); + + $totalQuotedAmount += $caseSaleOrderContributionService->getQuotedAmount(); + $totalPaidAmount += $caseSaleOrderContributionService->calculateTotalPaidAmount(); + $totalInvoicedAmount += $caseSaleOrderContributionService->calculateTotalInvoicedAmount(); + } + + $this->totalQuotedAmount = $totalQuotedAmount; + $this->totalPaidAmount = $totalPaidAmount; + $this->totalInvoicedAmount = $totalInvoicedAmount; + } + + /** + * Gets Contributions by case Id. + * + * @param int $caseId + * List of contributions that link to the opportunity. + */ + private function getContributions($caseId) { + return Contribution::get(FALSE) + ->addSelect('*', 'Opportunity_Details.Quotation') + ->addWhere('Opportunity_Details.Case_Opportunity', '=', $caseId) + ->execute() + ->getArrayCopy(); + } + +} diff --git a/Civi/Api4/Action/CaseSalesOrder/SalesOrderSaveAction.php b/Civi/Api4/Action/CaseSalesOrder/SalesOrderSaveAction.php index 1f766af03..c5918c126 100644 --- a/Civi/Api4/Action/CaseSalesOrder/SalesOrderSaveAction.php +++ b/Civi/Api4/Action/CaseSalesOrder/SalesOrderSaveAction.php @@ -2,6 +2,7 @@ namespace Civi\Api4\Action\CaseSalesOrder; +use Civi\Api4\CaseSalesOrder; use Civi\Api4\CaseSalesOrderLine; use Civi\Api4\Generic\AbstractSaveAction; use Civi\Api4\Generic\Result; @@ -53,6 +54,10 @@ protected function writeRecord($items) { $salesOrder['payment_status_id'] = $caseSaleOrderContributionService->calculateInvoicingStatus(); $salesOrder['invoicing_status_id'] = $caseSaleOrderContributionService->calculatePaymentStatus(); + if (!is_null($saleOrderId)) { + $this->updateOpportunityDetails($saleOrderId); + } + $salesOrders = $this->writeObjects([$salesOrder]); $result = array_pop($salesOrders); @@ -103,4 +108,25 @@ public function removeStaleLineItems(array $salesOrder) { ->execute(); } + /** + * Updates sales order's case opportunity details. + * + * @param int $salesOrderId + * Sales Order Id. + */ + private function updateOpportunityDetails($salesOrderId): void { + $caseSalesOrder = CaseSalesOrder::get() + ->addSelect('case_id') + ->addWhere('id', '=', $salesOrderId) + ->execute() + ->first(); + + if (empty($caseSalesOrder)) { + return; + } + + $caseSaleOrderContributionService = new \CRM_Civicase_Service_CaseSalesOrderOpportunityCalculator($caseSalesOrder['case_id']); + $caseSaleOrderContributionService->updateOpportunityFinancialDetails(); + } + }