From d8f0ffc42f3520c1bf61de7e4e027d2fdf62c66d Mon Sep 17 00:00:00 2001 From: Gytautas Date: Thu, 7 Mar 2024 16:46:43 +0200 Subject: [PATCH 01/20] initial opc compatibility --- dpdbaltics.php | 38 ++++++++++++++++++++----------------- views/js/front/order-opc.js | 18 ++++++++++++++++++ 2 files changed, 39 insertions(+), 17 deletions(-) diff --git a/dpdbaltics.php b/dpdbaltics.php index e22b3855..d128d0b8 100644 --- a/dpdbaltics.php +++ b/dpdbaltics.php @@ -168,7 +168,7 @@ public function getModuleContainer($id = false) public function hookActionFrontControllerSetMedia() { //TODO fillup this array when more modules are compatible with OPC - $onePageCheckoutControllers = ['supercheckout']; + $onePageCheckoutControllers = ['supercheckout', 'onepagecheckoutps']; $applicableControlelrs = ['order', 'order-opc', 'ShipmentReturn', 'supercheckout']; $currentController = !empty($this->context->controller->php_self) ? $this->context->controller->php_self : Tools::getValue('controller'); @@ -183,7 +183,9 @@ public function hookActionFrontControllerSetMedia() ); } - if (in_array($currentController, $onePageCheckoutControllers, true)) { + //todo for onepagecheckoutps module these are needed to be included as it handles phone number + // onepageps module controller is like in normal prestashop OrderController and some test are needed with normal flow +// if (in_array($currentController, $onePageCheckoutControllers, true)) { $this->context->controller->addJqueryPlugin('chosen'); $this->context->controller->registerJavascript( @@ -204,7 +206,7 @@ public function hookActionFrontControllerSetMedia() ] ); - } +// } /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); @@ -348,21 +350,23 @@ public function hookActionValidateStepComplete(&$params) return; } } - if (!Tools::getValue('dpd-phone')) { - $this->context->controller->errors[] = - $this->l('In order to use DPD Carrier you need to enter phone number'); - $params['completed'] = false; - - return; - } - if (!Tools::getValue('dpd-phone-area')) { - $this->context->controller->errors[] = - $this->l('In order to use DPD Carrier you need to enter phone area'); - $params['completed'] = false; - - return; - } + // todo thecheckout module triggers this hook when submiting the payment form +// if (!Tools::getValue('dpd-phone')) { +// $this->context->controller->errors[] = +// $this->l('In order to use DPD Carrier you need to enter phone number'); +// $params['completed'] = false; +// +// return; +// } +// +// if (!Tools::getValue('dpd-phone-area')) { +// $this->context->controller->errors[] = +// $this->l('In order to use DPD Carrier you need to enter phone area'); +// $params['completed'] = false; +// +// return; +// } /** @var CarrierPhoneService $carrierPhoneService */ $carrierPhoneService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.carrier_phone_service'); diff --git a/views/js/front/order-opc.js b/views/js/front/order-opc.js index 33065923..2b05e7be 100644 --- a/views/js/front/order-opc.js +++ b/views/js/front/order-opc.js @@ -24,6 +24,18 @@ $(document).ready(function (){ handlePhoneNumber($(this)); }); + // todo need event listener to trigger the checkout form and handle phone number + $('body').on('click', '#confirm_order', function(e) { + e.preventDefault(); + + if ($('.dpd-phone-block') !== undefined) { + if(!handlePhoneNumber($('.dpd-phone-block'))) { + return; + } + } + + }); + $(document).on('click','.payment_module a', function (e){ e.preventDefault(); @@ -82,3 +94,9 @@ function DPDdisplayMessageOpc(parent, template) { var $messageContainer = parent.find('.dpd-message-container'); $messageContainer.html(template); } + +// Module "onepagecheckoutps" compatibility +$(document).on('opc-load-review:completed', function() { + $('.delivery-option.selected .carrier-extra-content').show(); +}); + From 8e16f8c2d647d71a1ba9f128383237627769b97b Mon Sep 17 00:00:00 2001 From: Gytautas Date: Thu, 28 Mar 2024 16:53:47 +0200 Subject: [PATCH 02/20] fixed carrier parameter on ajax call --- views/js/front/pudo-search.js | 17 +++++++++-------- 1 file changed, 9 insertions(+), 8 deletions(-) diff --git a/views/js/front/pudo-search.js b/views/js/front/pudo-search.js index ee43983b..4aaed251 100644 --- a/views/js/front/pudo-search.js +++ b/views/js/front/pudo-search.js @@ -36,7 +36,11 @@ $(document).ready(function () { $(document).on('keyup', 'input[name="dpd-street"]', function () { var city = $('select[name="dpd-city"]').val(); var street = $('input[name="dpd-street"]').val(); - updateParcelBlock(city, street); + var selectedCarrier = $('select[name="dpd-city"]').closest('.dpd-pudo-container'); + + var idCarrier = selectedCarrier.attr('data-id'); + + updateParcelBlock(city, street, idCarrier); }); if ($('.dpd-checkout-pickup-container').is(':visible')) { @@ -75,12 +79,7 @@ $( document ).ajaxComplete(function( event, request, settings ) { }); function updateStreetSelect(city) { - - var $this = $(this); - var $pudoId = $this.data('id'); - var $container = $this.closest('.dpd-pudo-container'); - var $submitInput = $container.find('input[name="dpd-pudo-id"]'); - var $idReference = $container.data('id'); + var $container = $(this).closest('.dpd-pudo-container'); $.ajax(dpdHookAjaxUrl, { type: 'POST', @@ -152,13 +151,15 @@ function saveSelectedStreet(city, street) { }); } -function updateParcelBlock(city, street) { +function updateParcelBlock(city, street, idCarrier) { + $.ajax(dpdHookAjaxUrl, { type: 'POST', data: { 'ajax': 1, 'city': city, 'street': street, + 'id_carrier': idCarrier, 'action': 'updateParcelBlock', 'token': typeof prestashop !== 'undefined' ? prestashop.static_token : '' }, From d8241640d8800759ddc0554a0dee23f6c78b4e13 Mon Sep 17 00:00:00 2001 From: Gytautas Date: Thu, 28 Mar 2024 16:55:31 +0200 Subject: [PATCH 03/20] whitespace removed --- views/js/front/pudo-search.js | 1 - 1 file changed, 1 deletion(-) diff --git a/views/js/front/pudo-search.js b/views/js/front/pudo-search.js index 4aaed251..ef907e65 100644 --- a/views/js/front/pudo-search.js +++ b/views/js/front/pudo-search.js @@ -152,7 +152,6 @@ function saveSelectedStreet(city, street) { } function updateParcelBlock(city, street, idCarrier) { - $.ajax(dpdHookAjaxUrl, { type: 'POST', data: { From 0c61096bc0aae8a35e18f6e5017d5e0f9b9b7b84 Mon Sep 17 00:00:00 2001 From: Gytautas Date: Thu, 18 Apr 2024 10:39:20 +0300 Subject: [PATCH 04/20] DGS-322 getErrLog on null fix --- dpdbaltics.php | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/dpdbaltics.php b/dpdbaltics.php index 9043978f..dbe0b549 100644 --- a/dpdbaltics.php +++ b/dpdbaltics.php @@ -1100,7 +1100,7 @@ public function printLabel($idShipment) if ($parcelPrintResponse->getStatus() === Config::API_SUCCESS_STATUS) { $this->updateOrderCarrier($idShipment); - return; + return $parcelPrintResponse; } return $parcelPrintResponse; @@ -1117,7 +1117,7 @@ public function printMultipleLabels($shipmentIds) foreach ($shipmentIds as $shipmentId) { $this->updateOrderCarrier($shipmentId); } - return; + return $parcelPrintResponse; } return $parcelPrintResponse; @@ -1358,7 +1358,7 @@ private function handleLabelPrintService() return; } - if (!empty($parcelPrintResponse->getErrLog())) { + if (isset($parcelPrintResponse) && !empty($parcelPrintResponse->getErrLog())) { Context::getContext()->controller->errors[] = $parcelPrintResponse->getErrLog(); } @@ -1383,7 +1383,7 @@ private function handleLabelPrintService() return; } - if (!empty($parcelPrintResponse->getErrLog())) { + if (isset($parcelPrintResponse) && !empty($parcelPrintResponse->getErrLog())) { Context::getContext()->controller->errors[] = $parcelPrintResponse->getErrLog(); } From 9664a7dbe22d701086804493b166857b907540e2 Mon Sep 17 00:00:00 2001 From: Gytautas Date: Fri, 3 May 2024 16:30:13 +0300 Subject: [PATCH 05/20] total paid fix --- src/Service/ShipmentService.php | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/src/Service/ShipmentService.php b/src/Service/ShipmentService.php index ffe9dcc6..fdcac7b6 100644 --- a/src/Service/ShipmentService.php +++ b/src/Service/ShipmentService.php @@ -221,16 +221,14 @@ public function createShipmentFromOrder(Order $order) private function createNonDistributedShipment(Order $order, $idProduct, $isTestMode) { $products = $order->getProducts(); - $parcelPrice = 0; + $parcelPrice = $order->getTotalPaid(); $parcelWeight = 0; - $orderShippingCost = $order->total_shipping_tax_incl ?: 0; + foreach ($products as $product) { - $parcelPrice += $this->calculateProductsPrice($product); $parcelWeight += $product['weight'] * $product['product_quantity']; } $parcelPrice = $this->calculateParcelPriceWithOrderDiscount($order, $parcelPrice); - $parcelPrice += $orderShippingCost; $shipment = $this->createShipment($order, $idProduct, $isTestMode, 1, $parcelWeight, $parcelPrice); if (!$shipment->id) { From 02a69ab70ef087173bfc11e2e1601037ccd11566 Mon Sep 17 00:00:00 2001 From: Gytautas Date: Mon, 3 Jun 2024 09:12:02 +0300 Subject: [PATCH 06/20] opc compatibility improvements --- dpdbaltics.php | 68 ++++++++++++++++++++----------------- views/js/front/order-opc.js | 4 +-- 2 files changed, 38 insertions(+), 34 deletions(-) diff --git a/dpdbaltics.php b/dpdbaltics.php index d128d0b8..6c97f8ff 100644 --- a/dpdbaltics.php +++ b/dpdbaltics.php @@ -168,7 +168,7 @@ public function getModuleContainer($id = false) public function hookActionFrontControllerSetMedia() { //TODO fillup this array when more modules are compatible with OPC - $onePageCheckoutControllers = ['supercheckout', 'onepagecheckoutps']; + $onePageCheckoutControllers = ['supercheckout', 'onepagecheckoutps', 'thecheckout']; $applicableControlelrs = ['order', 'order-opc', 'ShipmentReturn', 'supercheckout']; $currentController = !empty($this->context->controller->php_self) ? $this->context->controller->php_self : Tools::getValue('controller'); @@ -186,25 +186,25 @@ public function hookActionFrontControllerSetMedia() //todo for onepagecheckoutps module these are needed to be included as it handles phone number // onepageps module controller is like in normal prestashop OrderController and some test are needed with normal flow // if (in_array($currentController, $onePageCheckoutControllers, true)) { - $this->context->controller->addJqueryPlugin('chosen'); + $this->context->controller->addJqueryPlugin('chosen'); - $this->context->controller->registerJavascript( - 'dpdbaltics-opc', - 'modules/' . $this->name . '/views/js/front/order-opc.js', - [ + $this->context->controller->registerJavascript( + 'dpdbaltics-opc', + 'modules/' . $this->name . '/views/js/front/order-opc.js', + [ 'position' => 'bottom', 'priority' => 130 ] - ); + ); - $this->context->controller->registerJavascript( - 'dpdbaltics-supercheckout', - 'modules/' . $this->name . '/views/js/front/modules/supercheckout.js', - [ + $this->context->controller->registerJavascript( + 'dpdbaltics-supercheckout', + 'modules/' . $this->name . '/views/js/front/modules/supercheckout.js', + [ 'position' => 'bottom', 'priority' => 130 ] - ); + ); // } @@ -280,7 +280,8 @@ public function hookActionFrontControllerSetMedia() $googleApiService = $this->getModuleContainer('invertus.dpdbaltics.service.google_api_service'); $this->context->controller->registerJavascript( 'dpdbaltics-google-api', - $googleApiService->getFormattedGoogleMapsUrl(), [ + $googleApiService->getFormattedGoogleMapsUrl(), + [ 'server' => 'remote' ] ); @@ -351,22 +352,26 @@ public function hookActionValidateStepComplete(&$params) } } - // todo thecheckout module triggers this hook when submiting the payment form -// if (!Tools::getValue('dpd-phone')) { -// $this->context->controller->errors[] = -// $this->l('In order to use DPD Carrier you need to enter phone number'); -// $params['completed'] = false; -// -// return; -// } -// -// if (!Tools::getValue('dpd-phone-area')) { -// $this->context->controller->errors[] = -// $this->l('In order to use DPD Carrier you need to enter phone area'); -// $params['completed'] = false; -// -// return; -// } + //NOTE: thecheckout triggers this hook without phone parameters the phone is saved with ajax request + if (Tools::getValue('module') === 'thecheckout') { + return; + } + + if (!Tools::getValue('dpd-phone')) { + $this->context->controller->errors[] = + $this->l('In order to use DPD Carrier you need to enter phone number'); + $params['completed'] = false; + + return; + } + + if (!Tools::getValue('dpd-phone-area')) { + $this->context->controller->errors[] = + $this->l('In order to use DPD Carrier you need to enter phone area'); + $params['completed'] = false; + + return; + } /** @var CarrierPhoneService $carrierPhoneService */ $carrierPhoneService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.carrier_phone_service'); @@ -511,7 +516,7 @@ public function getOrderShippingCostExternal($cart) $parcelDistribution = \Configuration::get(Config::PARCEL_DISTRIBUTION); $maxAllowedWeight = Config::getDefaultServiceWeights($countryCode, $serviceCarrier['product_reference']); - if (!$cartWeightValidator->validate($cart, $parcelDistribution ,$maxAllowedWeight)) { + if (!$cartWeightValidator->validate($cart, $parcelDistribution, $maxAllowedWeight)) { return false; } @@ -1154,7 +1159,6 @@ private function updateOrderCarrier($shipmentId) public function hookDisplayOrderDetail($params) { - $isReturnServiceEnabled = Configuration::get(Config::PARCEL_RETURN); if (!$isReturnServiceEnabled) { return; @@ -1283,7 +1287,7 @@ public function hookActionOrderGridDefinitionModifier(array $params) $definition = $params['definition']; if (!(bool) Configuration::get(Config::HIDE_ORDERS_LABEL_PRINT_BUTTON)) { - $definition->getColumns() + $definition->getColumns() ->addAfter( 'date_add', (new ActionColumn('dpd_print_label')) diff --git a/views/js/front/order-opc.js b/views/js/front/order-opc.js index 2b05e7be..fe3065d3 100644 --- a/views/js/front/order-opc.js +++ b/views/js/front/order-opc.js @@ -17,14 +17,12 @@ * @license https://opensource.org/licenses/AFL-3.0 Academic Free License version 3.0 */ var isPudoPointSelected = false; - $(document).ready(function (){ $(document).on('change', '.dpd-phone-block', function() { handlePhoneNumber($(this)); }); - // todo need event listener to trigger the checkout form and handle phone number $('body').on('click', '#confirm_order', function(e) { e.preventDefault(); @@ -98,5 +96,7 @@ function DPDdisplayMessageOpc(parent, template) { // Module "onepagecheckoutps" compatibility $(document).on('opc-load-review:completed', function() { $('.delivery-option.selected .carrier-extra-content').show(); + handlePhoneNumber($('.dpd-phone-block')); }); + From 3d49649b1e0ddf692a739dcf0ce51b99f61f5040 Mon Sep 17 00:00:00 2001 From: Gytautas Date: Wed, 3 Jul 2024 10:19:05 +0300 Subject: [PATCH 07/20] opc improvements and added some styles --- config/service.yml | 1 + config/validator.yml | 3 ++ dpdbaltics.php | 14 +++++- src/Config/Config.php | 3 ++ src/Service/CarrierPhoneService.php | 10 ++++- .../OpcModuleCompatibilityValidator.php | 44 +++++++++++++++++++ views/css/front/onepagecheckout.css | 22 ++++++++++ .../hook/front/carrier-phone-number.tpl | 8 ++-- 8 files changed, 99 insertions(+), 6 deletions(-) create mode 100644 src/Validate/Compatibility/OpcModuleCompatibilityValidator.php create mode 100644 views/css/front/onepagecheckout.css diff --git a/config/service.yml b/config/service.yml index e7a9c2ee..1271387d 100644 --- a/config/service.yml +++ b/config/service.yml @@ -18,6 +18,7 @@ services: - '@invertus.dpdbaltics.orm.entity_manager' - '@invertus.dpdbaltics.repository.phone_prefix_repository' - '@invertus.dpdbaltics.repository.order_repository' + - '@invertus.dpdbaltics.validator.opc_module_compatibility_validator' invertus.dpdbaltics.service.carrier.update_carrier_service: class: 'Invertus\dpdBaltics\Service\Carrier\UpdateCarrierService' diff --git a/config/validator.yml b/config/validator.yml index a254ce24..190c529f 100644 --- a/config/validator.yml +++ b/config/validator.yml @@ -41,3 +41,6 @@ services: arguments: - '@dpdbaltics' - '@invertus.dpdbaltics.repository.phone_repository' + + invertus.dpdbaltics.validator.opc_module_compatibility_validator: + class: 'Invertus\dpdBaltics\Validate\Compatibility\OpcModuleCompatibilityValidator' diff --git a/dpdbaltics.php b/dpdbaltics.php index f661a91e..2c13039f 100644 --- a/dpdbaltics.php +++ b/dpdbaltics.php @@ -182,7 +182,10 @@ public function hookActionFrontControllerSetMedia() ); } - if (in_array($currentController, $onePageCheckoutControllers, true) || Tools::getValue('module') === 'thecheckout') { + /** @var \Invertus\dpdBaltics\Validate\Compatibility\OpcModuleCompatibilityValidator $opcModuleCompatibilityValidator */ + $opcModuleCompatibilityValidator = $this->getModuleContainer('invertus.dpdbaltics.validator.opc_module_compatibility_validator'); + + if (in_array($currentController, $onePageCheckoutControllers, true) || $opcModuleCompatibilityValidator->isOpcModuleInUse()) { $this->context->controller->addJqueryPlugin('chosen'); $this->context->controller->registerJavascript( @@ -202,6 +205,15 @@ public function hookActionFrontControllerSetMedia() 'priority' => 130 ] ); + + $this->context->controller->registerStylesheet( + 'dpdbaltics-opc', + 'modules/' . $this->name . '/views/css/front/onepagecheckout.css', + [ + 'position' => 'bottom', + 'priority' => 130 + ] + ); } /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ diff --git a/src/Config/Config.php b/src/Config/Config.php index fe9a188a..c50f5c1d 100644 --- a/src/Config/Config.php +++ b/src/Config/Config.php @@ -257,6 +257,9 @@ class Config const COUNTRY_ISO_CODES_WITH_MIXED_CHARACTERS = ['IE', 'GB', 'NL']; + // NOTE: Add OPC module tech name if payment option is compatible + public const DPD_OPC_MODULE_LIST = ['onepagecheckoutps', 'supercheckout', 'thecheckout']; + const PRODUCT_NAME_B2B = [ 'LT' => 'Pristatymas privatiems asmenims', 'EE' => 'DPD kuller', diff --git a/src/Service/CarrierPhoneService.php b/src/Service/CarrierPhoneService.php index b14a4f9a..5febbb9b 100644 --- a/src/Service/CarrierPhoneService.php +++ b/src/Service/CarrierPhoneService.php @@ -22,6 +22,7 @@ use Invertus\dpdBaltics\Repository\PhonePrefixRepository; use Invertus\dpdBaltics\Config\Config; use DPDOrderPhone; +use Invertus\dpdBaltics\Validate\Compatibility\OpcModuleCompatibilityValidator; class CarrierPhoneService { @@ -45,19 +46,25 @@ class CarrierPhoneService * @var OrderRepository */ private $orderRepository; + /** + * @var OpcModuleCompatibilityValidator + */ + private $opcModuleCompatibilityValidator; public function __construct( DPDBaltics $module, Context $context, EntityManager $entityManager, PhonePrefixRepository $phonePrefixRepository, - OrderRepository $orderRepository + OrderRepository $orderRepository, + OpcModuleCompatibilityValidator $opcModuleCompatibilityValidator ) { $this->module = $module; $this->context = $context; $this->entityManager = $entityManager; $this->phonePrefixRepository = $phonePrefixRepository; $this->orderRepository = $orderRepository; + $this->opcModuleCompatibilityValidator = $opcModuleCompatibilityValidator; } public function getCarrierPhoneTemplate($cartId, $carrierReference) @@ -94,6 +101,7 @@ public function getCarrierPhoneTemplate($cartId, $carrierReference) 'dpdPhoneArea' => $phoneData['mobile_phone_code_list'], 'contextPrefix' => Config::PHONE_CODE_PREFIX . $phonePrefix, 'isAbove177' => Config::isPrestashopVersionAbove177(), + 'isOpcCheckout' => $this->opcModuleCompatibilityValidator->isOpcModuleInUse(), ] ); diff --git a/src/Validate/Compatibility/OpcModuleCompatibilityValidator.php b/src/Validate/Compatibility/OpcModuleCompatibilityValidator.php new file mode 100644 index 00000000..dbf3d191 --- /dev/null +++ b/src/Validate/Compatibility/OpcModuleCompatibilityValidator.php @@ -0,0 +1,44 @@ + + * @copyright Since 2007 PrestaShop SA and Contributors + * @license https://opensource.org/licenses/AFL-3.0 Academic Free License version 3.0 + */ + + +namespace Invertus\dpdBaltics\Validate\Compatibility; + +use DPDBaltics; +use Invertus\dpdBaltics\Config\Config; +use Invertus\dpdBaltics\Exception\DpdCarrierException; +use Invertus\dpdBaltics\Repository\PhoneRepository; +use Invertus\dpdBaltics\Util\NumberUtility; +use Invertus\dpdBaltics\Util\StringUtility; +use Module; + +class OpcModuleCompatibilityValidator +{ + public function isOpcModuleInUse(): bool + { + foreach (Config::DPD_OPC_MODULE_LIST as $opcModule){ + if (Module::isInstalled($opcModule) && Module::isEnabled($opcModule)) { + return true; + } + } + + return false; + } +} \ No newline at end of file diff --git a/views/css/front/onepagecheckout.css b/views/css/front/onepagecheckout.css new file mode 100644 index 00000000..89be97b5 --- /dev/null +++ b/views/css/front/onepagecheckout.css @@ -0,0 +1,22 @@ +#phone-block-wrapper { + display: flex; + flex-direction: column; +} + +.points-container .list-inline .row { + margin-left: 15px; + margin-right: 15px; +} + +.panel-body .points-container .dpd-services-block { + width: 100%; +} + +.panel-body .points-container { + width: 340px; + margin-left: 18px; +} + +.dpd-services-block .list-inline-item { + min-width: 370px; +} \ No newline at end of file diff --git a/views/templates/hook/front/carrier-phone-number.tpl b/views/templates/hook/front/carrier-phone-number.tpl index 9847fb21..9567e4fa 100644 --- a/views/templates/hook/front/carrier-phone-number.tpl +++ b/views/templates/hook/front/carrier-phone-number.tpl @@ -24,9 +24,9 @@ {include file='module:dpdbaltics/views/templates/hook/front/partials/dpd-message.tpl' messageType='error'} {/if}
-
-

{l s='This a phone number that will be used for deliveries' mod='dpdbaltics'}

-
+
+

{l s='This a phone number that will be used for deliveriessss' mod='dpdbaltics'}

+
{l s='Phone' mod='dpdbaltics'}
From 2ccea3e9e5640475105eca1ce53d8f6c8b15c4a3 Mon Sep 17 00:00:00 2001 From: GytisZum Date: Wed, 3 Jul 2024 07:19:24 +0000 Subject: [PATCH 08/20] Adding auto indexes --- src/Validate/Compatibility/index.php | 11 +++++++++++ 1 file changed, 11 insertions(+) create mode 100644 src/Validate/Compatibility/index.php diff --git a/src/Validate/Compatibility/index.php b/src/Validate/Compatibility/index.php new file mode 100644 index 00000000..15aba820 --- /dev/null +++ b/src/Validate/Compatibility/index.php @@ -0,0 +1,11 @@ + Date: Wed, 3 Jul 2024 07:20:03 +0000 Subject: [PATCH 09/20] Automatic license addition applying --- views/css/front/onepagecheckout.css | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) diff --git a/views/css/front/onepagecheckout.css b/views/css/front/onepagecheckout.css index 89be97b5..8e61beba 100644 --- a/views/css/front/onepagecheckout.css +++ b/views/css/front/onepagecheckout.css @@ -1,3 +1,21 @@ +/** + * Copyright since 2007 PrestaShop SA and Contributors + * PrestaShop is an International Registered Trademark & Property of PrestaShop SA + * + * NOTICE OF LICENSE + * + * This source file is subject to the Academic Free License version 3.0 + * that is bundled with this package in the file LICENSE.md. + * It is also available through the world-wide-web at this URL: + * https://opensource.org/licenses/AFL-3.0 + * If you did not receive a copy of the license and are unable to + * obtain it through the world-wide-web, please send an email + * to license@prestashop.com so we can send you a copy immediately. + * + * @author PrestaShop SA and Contributors + * @copyright Since 2007 PrestaShop SA and Contributors + * @license https://opensource.org/licenses/AFL-3.0 Academic Free License version 3.0 + */ #phone-block-wrapper { display: flex; flex-direction: column; From 583b64ff8c0f8fc16553eb6818de268d4745aa3f Mon Sep 17 00:00:00 2001 From: Justas Vaitkus Date: Wed, 3 Jul 2024 18:43:09 +0300 Subject: [PATCH 10/20] styling and layout for checkout --- views/css/front/onepagecheckout.css | 23 +++++++++++++++++++ views/css/front/order-input.css | 11 +++++---- views/css/front/pudo-shipment.css | 6 ++++- .../hook/front/carrier-delivery-time.tpl | 4 ++-- .../hook/front/carrier-phone-number.tpl | 10 ++++---- 5 files changed, 41 insertions(+), 13 deletions(-) diff --git a/views/css/front/onepagecheckout.css b/views/css/front/onepagecheckout.css index 8e61beba..86f6f2f0 100644 --- a/views/css/front/onepagecheckout.css +++ b/views/css/front/onepagecheckout.css @@ -37,4 +37,27 @@ .dpd-services-block .list-inline-item { min-width: 370px; +} + +.carrier-extra-content { + display: flex; + flex-direction: column; + padding: 10px 5px; +} + +.dpd-phone-block { + padding-bottom: 0; +} + +.chosen-container { + width: 100%; +} + +.carrier-extra-content .chosen-select { + min-height: 49px; + height: 100%; +} + +.dpd-checkout-delivery-time--container { + width: 100%; } \ No newline at end of file diff --git a/views/css/front/order-input.css b/views/css/front/order-input.css index 9cc1ef45..3e4d9902 100644 --- a/views/css/front/order-input.css +++ b/views/css/front/order-input.css @@ -19,7 +19,7 @@ This file is generated by `grunt build`, do not edit it by hand. -moz-user-select: none; -ms-user-select: none; user-select: none; - width: 100% !important; + width: 100%; max-width: 100%; } @@ -84,7 +84,7 @@ This file is generated by `grunt build`, do not edit it by hand. margin: 1px 0; padding: 4px 20px 4px 5px; width: 100%; - height: auto; + height: 49px; outline: 0; border: 1px solid #aaa; background: url("../../img/dpd-chosen-front.png") no-repeat 100% -20px; @@ -178,11 +178,12 @@ This file is generated by `grunt build`, do not edit it by hand. margin-right: 20px; text-overflow: ellipsis; white-space: nowrap; - padding: 13px 0; + padding: 16px 0; } .chosen-container-single a.chosen-single { - height: auto; + min-height: 49px; + height: 100%; } .chosen-container-single .chosen-single div { @@ -203,7 +204,7 @@ div.form-control-chosen:focus-within { .css-dpd-phone-prefix .chosen-single:before { content: '+'; position: absolute; - transform: translate(-12px, 12px); + transform: translate(-12px, 16px); } .dpd-input-wrapper .form-control, diff --git a/views/css/front/pudo-shipment.css b/views/css/front/pudo-shipment.css index 30d25df0..be2c6e7e 100644 --- a/views/css/front/pudo-shipment.css +++ b/views/css/front/pudo-shipment.css @@ -123,7 +123,7 @@ .phone-block-hr, .delivery-time-hr { - padding-bottom: 2rem; + padding-bottom: 0; } @@ -244,3 +244,7 @@ a.chosen-single:hover { width: 100%; } +.dpd-delivery-time-block { + margin-top: 1rem; +} + diff --git a/views/templates/hook/front/carrier-delivery-time.tpl b/views/templates/hook/front/carrier-delivery-time.tpl index f166f8dd..ed02d79a 100644 --- a/views/templates/hook/front/carrier-delivery-time.tpl +++ b/views/templates/hook/front/carrier-delivery-time.tpl @@ -18,9 +18,9 @@ *}
-
+

{l s='Desirable delivery time' mod='dpdbaltics'}

-
+
{html_options options=$dpdPhoneArea selected=$contextPrefix} @@ -37,7 +37,7 @@
-
+
{l s='Phone' mod='dpdbaltics'}
From 4a9116df96cd84fa25d7f528ee196fbc70e733ba Mon Sep 17 00:00:00 2001 From: justelis22 Date: Wed, 3 Jul 2024 15:46:14 +0000 Subject: [PATCH 11/20] Automatic license addition applying --- src/Validate/Compatibility/index.php | 19 +++++++++++++++++++ 1 file changed, 19 insertions(+) diff --git a/src/Validate/Compatibility/index.php b/src/Validate/Compatibility/index.php index 15aba820..74851d77 100644 --- a/src/Validate/Compatibility/index.php +++ b/src/Validate/Compatibility/index.php @@ -1,4 +1,23 @@ + * @copyright Since 2007 PrestaShop SA and Contributors + * @license https://opensource.org/licenses/AFL-3.0 Academic Free License version 3.0 + */ + header("Expires: Mon, 26 Jul 1997 05:00:00 GMT"); header("Last-Modified: ".gmdate("D, d M Y H:i:s")." GMT"); From a9b70ed5d28be13af5cf12a33e015198dee3b3d9 Mon Sep 17 00:00:00 2001 From: Gytautas Date: Thu, 4 Jul 2024 11:48:35 +0300 Subject: [PATCH 12/20] onepagecheckout ps module compatibility fixes --- dpdbaltics.php | 1411 +-------------------------- views/css/front/onepagecheckout.css | 2 +- views/js/front/order-opc.js | 21 +- 3 files changed, 22 insertions(+), 1412 deletions(-) diff --git a/dpdbaltics.php b/dpdbaltics.php index 2c13039f..7b3be301 100644 --- a/dpdbaltics.php +++ b/dpdbaltics.php @@ -1,1410 +1 @@ - - * @copyright Since 2007 PrestaShop SA and Contributors - * @license https://opensource.org/licenses/AFL-3.0 Academic Free License version 3.0 - */ - -use Invertus\dpdBaltics\Grid\Row\PrintAccessibilityChecker; -use Invertus\dpdBaltics\Builder\Template\Front\CarrierOptionsBuilder; -use Invertus\dpdBaltics\Config\Config; -use Invertus\dpdBaltics\Controller\AbstractAdminController; -use Invertus\dpdBaltics\Grid\LinkRowActionCustom; -use Invertus\dpdBaltics\Grid\SubmitBulkActionCustom; -use invertus\dpdbaltics\install\installer; -use Invertus\dpdBaltics\Logger\Logger; -use Invertus\dpdBaltics\OnBoard\Service\OnBoardService; -use Invertus\dpdBaltics\Repository\AddressTemplateRepository; -use Invertus\dpdBaltics\Repository\OrderRepository; -use Invertus\dpdBaltics\Repository\ParcelShopRepository; -use Invertus\dpdBaltics\Repository\PhonePrefixRepository; -use Invertus\dpdBaltics\Repository\PriceRuleRepository; -use Invertus\dpdBaltics\Repository\ProductRepository; -use Invertus\dpdBaltics\Repository\PudoRepository; -use Invertus\dpdBaltics\Repository\ReceiverAddressRepository; -use Invertus\dpdBaltics\Repository\ShipmentRepository; -use Invertus\dpdBaltics\Repository\ZoneRepository; -use Invertus\dpdBaltics\Service\API\LabelApiService; -use Invertus\dpdBaltics\Service\API\ParcelShopSearchApiService; -use Invertus\dpdBaltics\Service\CarrierPhoneService; -use Invertus\dpdBaltics\Service\Exception\ExceptionService; -use Invertus\dpdBaltics\Service\GoogleApiService; -use Invertus\dpdBaltics\Service\Label\LabelPositionService; -use Invertus\dpdBaltics\Service\OrderService; -use Invertus\dpdBaltics\Service\Parcel\ParcelShopService; -use Invertus\dpdBaltics\Service\Payment\PaymentService; -use Invertus\dpdBaltics\Service\PriceRuleService; -use Invertus\dpdBaltics\Service\PudoService; -use Invertus\dpdBaltics\Service\ShipmentService; -use Invertus\dpdBaltics\Service\ShippingPriceCalculationService; -use Invertus\dpdBaltics\Service\TabService; -use Invertus\dpdBaltics\Service\TrackingService; -use Invertus\dpdBaltics\Util\CountryUtility; -use Invertus\dpdBaltics\Validate\Carrier\PudoValidate; -use Invertus\dpdBalticsApi\Api\DTO\Response\ParcelPrintResponse; -use Invertus\dpdBalticsApi\Api\DTO\Response\ParcelShopSearchResponse; -use Invertus\dpdBalticsApi\Exception\DPDBalticsAPIException; -use Invertus\dpdBalticsApi\Factory\SerializerFactory; -use PrestaShop\PrestaShop\Core\Grid\Action\Bulk\Type\SubmitBulkAction; -use PrestaShop\PrestaShop\Core\Grid\Action\Row\RowActionCollection; -use PrestaShop\PrestaShop\Core\Grid\Action\Row\Type\LinkRowAction; -use PrestaShop\PrestaShop\Core\Grid\Column\Type\Common\ActionColumn; -use Symfony\Component\Config\ConfigCache; -use Symfony\Component\Config\FileLocator; -use Symfony\Component\Console\Application; -use Symfony\Component\DependencyInjection\ContainerBuilder; -use Symfony\Component\DependencyInjection\Dumper\PhpDumper; -use Symfony\Component\DependencyInjection\Loader\YamlFileLoader; - -class DPDBaltics extends CarrierModule -{ - /** - * Symfony DI Container - **/ - private $moduleContainer; - - /** - * Prestashop fills this property automatically with selected carrier ID in FO checkout - * - * @var int $id_carrier - */ - public $id_carrier; - - - public function __construct() - { - $this->name = 'dpdbaltics'; - $this->displayName = $this->l('DPDBaltics'); - $this->author = 'Invertus'; - $this->tab = 'shipping_logistics'; - $this->description = 'DPD Baltics shipping integration'; - $this->version = '3.2.19'; - $this->ps_versions_compliancy = ['min' => '1.7.1.0', 'max' => _PS_VERSION_]; - $this->need_instance = 0; - parent::__construct(); - - $this->autoLoad(); - $this->compile(); - } - - public function install() - { - if (!parent::install()) { - return false; - } - - /** @var \Invertus\dpdBaltics\Infrastructure\Bootstrap\Install\Installer $installer */ - $installer = $this->getService(\Invertus\dpdBaltics\Infrastructure\Bootstrap\Install\Installer::class); - - try { - $installer->init(); - - return true; - } catch (\Invertus\dpdBaltics\Infrastructure\Bootstrap\Exception\CouldNotInstallModule $exception) { - $this->_errors[] = $exception->getMessage(); - - return false; - } - } - - public function uninstall() - { - $uninstall = parent::uninstall(); - - if (!$uninstall) { - return false; - } - - /** @var \Invertus\dpdBaltics\Infrastructure\Bootstrap\Uninstall\Uninstaller $uninstaller */ - $uninstaller = $this->getService(\Invertus\dpdBaltics\Infrastructure\Bootstrap\Uninstall\Uninstaller::class); - - try { - $uninstaller->init(); - - return true; - } catch (\Invertus\dpdBaltics\Infrastructure\Bootstrap\Exception\CouldNotUninstallModule $exception) { - $this->_errors[] = $exception->getMessage(); - - return false; - } - } - - public function getContent() - { - Tools::redirectAdmin($this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_SETTINGS_CONTROLLER)); - } - - public function getService($serviceName) - { - return $this->getModuleContainer()->get($serviceName); - } - - /** - * @return mixed - */ - public function getModuleContainer($id = false) - { - if ($id) { - return $this->moduleContainer->get($id); - } - - return $this->moduleContainer; - } - - public function hookActionFrontControllerSetMedia() - { - //TODO fillup this array when more modules are compatible with OPC - $onePageCheckoutControllers = ['supercheckout', 'onepagecheckoutps', 'thecheckout']; - $applicableControlelrs = ['order', 'order-opc', 'ShipmentReturn', 'supercheckout']; - $currentController = !empty($this->context->controller->php_self) ? $this->context->controller->php_self : Tools::getValue('controller'); - - if ('product' === $currentController) { - $this->context->controller->registerStylesheet( - 'dpdbaltics-product-carriers.', - 'modules/' . $this->name . '/views/css/front/product-carriers.css', - [ - 'media' => 'all', - 'position' => 150 - ] - ); - } - - /** @var \Invertus\dpdBaltics\Validate\Compatibility\OpcModuleCompatibilityValidator $opcModuleCompatibilityValidator */ - $opcModuleCompatibilityValidator = $this->getModuleContainer('invertus.dpdbaltics.validator.opc_module_compatibility_validator'); - - if (in_array($currentController, $onePageCheckoutControllers, true) || $opcModuleCompatibilityValidator->isOpcModuleInUse()) { - $this->context->controller->addJqueryPlugin('chosen'); - - $this->context->controller->registerJavascript( - 'dpdbaltics-opc', - 'modules/' . $this->name . '/views/js/front/order-opc.js', - [ - 'position' => 'bottom', - 'priority' => 130 - ] - ); - - $this->context->controller->registerJavascript( - 'dpdbaltics-supercheckout', - 'modules/' . $this->name . '/views/js/front/modules/supercheckout.js', - [ - 'position' => 'bottom', - 'priority' => 130 - ] - ); - - $this->context->controller->registerStylesheet( - 'dpdbaltics-opc', - 'modules/' . $this->name . '/views/css/front/onepagecheckout.css', - [ - 'position' => 'bottom', - 'priority' => 130 - ] - ); - } - - /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ - $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); - $webServiceCountryCode = Configuration::get(Config::WEB_SERVICE_COUNTRY); - $carrierIds = []; - $baseUrl = $this->context->shop->getBaseURL(true, false); - - if ($webServiceCountryCode === Config::LATVIA_ISO_CODE || $currentController === 'supercheckout') { - /** @var ProductRepository $productRepo */ - $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); - - $dpdProductReferences = $productRepo->getAllActiveDpdProductReferences(); - - foreach ($dpdProductReferences as $reference) { - $carrier = Carrier::getCarrierByReference($reference['id_reference']); - if (Validate::isLoadedObject($carrier)) { - $carrierIds[] = $carrier->id; - } - } - } - - Media::addJsDef([ - 'lapinas_img' => $baseUrl . $this->getPathUri() . 'views/img/lapinas.png', - 'lapinas_text' => $this->l('Sustainable'), - 'dpd_carrier_ids' => $carrierIds - ]); - if (in_array($currentController, $applicableControlelrs, true)) { - Media::addJsDef([ - 'select_an_option_translatable' => $this->l('Select an Option'), - 'select_an_option_multiple_translatable' => $this->l('Select Some Options'), - 'no_results_translatable' => $this->l('No results match'), - ]); - $this->context->controller->addJS($this->getPathUri() . 'views/js/front/order.js'); - $this->context->controller->addJS($this->getPathUri() . 'views/js/front/order-input.js'); - $this->context->controller->addJS($this->getPathUri() . 'views/js/front/sustainable-logo.js'); - $this->context->controller->addCSS($this->getPathUri() . 'views/css/front/order-input.css'); - $this->context->controller->addCSS($this->getPathUri() . 'views/css/front/sustainable-logo.css'); - /** @var PaymentService $paymentService */ - $paymentService = $this->getModuleContainer('invertus.dpdbaltics.service.payment.payment_service'); - $isPickupMap = Configuration::get(\Invertus\dpdBaltics\Config\Config::PICKUP_MAP); - $cart = Context::getContext()->cart; - $paymentService->filterPaymentMethods($cart); - $paymentService->filterPaymentMethodsByCod($cart); - - /** @var ProductRepository $productRepo */ - $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); - Media::addJsDef([ - 'pudoCarriers' => json_encode($productRepo->getPudoProducts()), - 'currentController' => $currentController, - 'is_pickup_map' => $isPickupMap, - 'id_language' => $this->context->language->id, - 'id_shop' => $this->context->shop->id, - 'dpdAjaxLoaderPath' => $this->getPathUri() . 'views/img/ajax-loader-big.gif', - 'dpdPickupMarkerPath' => $this->getPathUri() . 'views/img/dpd-pick-up.png', - 'dpdLockerMarkerPath' => $this->getPathUri() . 'views/img/locker.png', - 'dpdHookAjaxUrl' => $this->context->link->getModuleLink($this->name, 'Ajax'), - 'pudoSelectSuccess' => $this->l('Pick-up point selected'), - 'dpd_carrier_ids' => $carrierIds - ]); - - $this->context->controller->registerStylesheet( - 'dpdbaltics-pudo-shipment', - 'modules/' . $this->name . '/views/css/front/' . 'pudo-shipment.css', - [ - 'media' => 'all', - 'position' => 150 - ] - ); - if ($isPickupMap) { - /** @var GoogleApiService $googleApiService */ - $googleApiService = $this->getModuleContainer('invertus.dpdbaltics.service.google_api_service'); - $this->context->controller->registerJavascript( - 'dpdbaltics-google-api', - $googleApiService->getFormattedGoogleMapsUrl(), - [ - 'server' => 'remote' - ] - ); - } - $this->context->controller->registerJavascript( - 'dpdbaltics-pudo', - 'modules/' . $this->name . '/views/js/front/pudo.js', - [ - 'position' => 'bottom', - 'priority' => 130 - ] - ); - $this->context->controller->registerJavascript( - 'dpdbaltics-pudo-search', - 'modules/' . $this->name . '/views/js/front/pudo-search.js', - [ - 'position' => 'bottom', - 'priority' => 130 - ] - ); - } - } - - public function hookActionValidateStepComplete(&$params) - { - if ('delivery' !== $params['step_name']) { - return; - } - - /** @var Cart $cart */ - $cart = $params['cart']; - $carrier = new Carrier($cart->id_carrier); - $idShop = $this->context->shop->id; - - /** @var Invertus\dpdBaltics\Repository\CarrierRepository $carrierRepo */ - /** @var Invertus\dpdBaltics\Repository\ProductRepository $productRepo */ - $carrierRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.carrier_repository'); - $productRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.product_repository'); - - $carrierReference = $carrier->id_reference; - $dpdCarriers = $carrierRepo->getDpdCarriers($idShop); - $isDpdCarrier = false; - foreach ($dpdCarriers as $dpdCarrier) { - if ($carrierReference == $dpdCarrier['id_reference']) { - $isDpdCarrier = true; - $productId = $productRepo->getProductIdByCarrierReference($carrier->id_reference); - $product = new DPDProduct($productId); - $isSameDayDelivery = $product->product_reference === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY; - break; - } - } - if (!$isDpdCarrier) { - return true; - } - - if ($isSameDayDelivery) { - /** @var PudoRepository $pudoRepo */ - $pudoRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.pudo_repository'); - $pudoId = $pudoRepo->getIdByCart($cart->id); - $selectedPudo = new DPDPudo($pudoId); - if ($selectedPudo->city !== Config::SAME_DAY_DELIVERY_CITY) { - $this->context->controller->errors[] = - $this->l('This carrier can\'t deliver to your selected city'); - $params['completed'] = false; - $selectedPudo->delete(); - - return; - } - } - - //NOTE: thecheckout triggers this hook without phone parameters the phone is saved with ajax request - if (Tools::getValue('module') === 'thecheckout') { - return; - } - - if (!Tools::getValue('dpd-phone')) { - $this->context->controller->errors[] = - $this->l('In order to use DPD Carrier you need to enter phone number'); - $params['completed'] = false; - - return; - } - - if (!Tools::getValue('dpd-phone-area')) { - $this->context->controller->errors[] = - $this->l('In order to use DPD Carrier you need to enter phone area'); - $params['completed'] = false; - - return; - } - - /** @var CarrierPhoneService $carrierPhoneService */ - $carrierPhoneService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.carrier_phone_service'); - - try { - $carrierPhoneService->saveCarrierPhone( - $this->context->cart->id, - Tools::getValue('dpd-phone'), - Tools::getValue('dpd-phone-area') - ); - } catch (Exception $exception) { - if ($exception->getCode() === Config::ERROR_COULD_NOT_SAVE_PHONE_NUMBER) { - $this->context->controller->errors[] = $this->l('Phone data is not saved'); - $params['completed'] = false; - } - } - /** @var \Invertus\dpdBaltics\Service\OrderDeliveryTimeService $orderDeliveryService */ - $orderDeliveryService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.order_delivery_time_service'); - - $deliveryTime = Tools::getValue('dpd-delivery-time'); - if ($deliveryTime) { - if (!$orderDeliveryService->saveDeliveryTime( - $this->context->cart->id, - $deliveryTime - )) { - $this->context->controller->errors[] = $this->l('Delivery time data is not saved'); - $params['completed'] = false; - }; - } - - /** @var Cart $cart */ - $cart = $params['cart']; - $carrier = new Carrier($cart->id_carrier); - /** @var PudoValidate $pudoValidator */ - $pudoValidator = $this->getModuleContainer('invertus.dpdbaltics.validate.carrier.pudo_validate'); - if (!$pudoValidator->validatePickupPoints($cart->id, $carrier->id_reference)) { - $carrier = new Carrier($cart->id_carrier, $this->context->language->id); - $this->context->controller->errors[] = - sprintf($this->l('Please select pickup point for carrier: %s.'), $carrier->name); - - $params['completed'] = false; - } - } - - public function hookDisplayProductPriceBlock($params) - { - if ($params['type'] == 'after_price') { - /** @var CarrierOptionsBuilder $carrierOptionsBuilder */ - $carrierOptionsBuilder = $this->getModuleContainer()->get('invertus.dpdbaltics.builder.template.front.carrier_options_builder'); - - return $carrierOptionsBuilder->renderCarrierOptionsInProductPage(); - } - } - - public function hookDisplayBackOfficeTop() - { - if ($this->context->controller instanceof AbstractAdminController && - Configuration::get(Config::ON_BOARD_TURNED_ON) && - Configuration::get(Config::ON_BOARD_STEP) - ) { - /** @var OnBoardService $onBoardService */ - $onBoardService = $this->getModuleContainer('invertus.dpdbaltics.on_board.service.on_board_service'); - return $onBoardService->makeStepActionWithTemplateReturn(); - } - } - - public function getOrderShippingCost($cart, $shippingCost) - { - return $this->getOrderShippingCostExternal($cart); - } - - /** - * @param $cart Cart - * @return bool|float|string - * @throws PrestaShopDatabaseException - * @throws PrestaShopException - * @throws SmartyException - */ - public function getOrderShippingCostExternal($cart) - { - // This method is still called when module is disabled so we need to do a manual check here - if (!$this->active) { - return false; - } - - if ($this->context->controller->ajax && Tools::getValue('id_address_delivery')) { - $cart->id_address_delivery = (int)Tools::getValue('id_address_delivery'); - } - - /** @var ZoneRepository $zoneRepository */ - /** @var ProductRepository $productRepo */ - /** @var \Invertus\dpdBaltics\Service\Product\ProductAvailabilityService $productAvailabilityService */ - /** @var \Invertus\dpdBaltics\Validate\Weight\CartWeightValidator $cartWeightValidator */ - /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ - $zoneRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.zone_repository'); - $productRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.product_repository'); - $productAvailabilityService = $this->getModuleContainer('invertus.dpdbaltics.service.product.product_availability_service'); - $cartWeightValidator = $this->getModuleContainer('invertus.dpdbaltics.validate.weight.cart_weight_validator'); - $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); - - $deliveryAddress = new Address($cart->id_address_delivery); - - if (empty($zoneRepository->findZoneInRangeByAddress($deliveryAddress))) { - return false; - } - - $carrier = new Carrier($this->id_carrier); - - if (!$productAvailabilityService->checkIfCarrierIsAvailable($carrier->id_reference)) { - return false; - } - - if (!$productRepo->checkIfCarrierIsAvailableInCountry((int) $carrier->id_reference, (int) $deliveryAddress->id_country) - ) { - return false; - } - - try { - $isCarrierAvailableInShop = $productRepo->checkIfCarrierIsAvailableInShop($carrier->id_reference, $this->context->shop->id); - if (empty($isCarrierAvailableInShop)) { - return false; - } - - $serviceCarrier = $productRepo->findProductByCarrierReference($carrier->id_reference); - } catch (Exception $e) { - $tplVars = [ - 'errorMessage' => $this->l('Something went wrong while collecting DPD carrier data'), - ]; - $this->context->smarty->assign($tplVars); - - return $this->context->smarty->fetch( - $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' - ); - } - - if ((bool)$serviceCarrier['is_home_collection']) { - return false; - } - - $countryCode = $currentCountryProvider->getCurrentCountryIsoCode($cart); - - $parcelDistribution = \Configuration::get(Config::PARCEL_DISTRIBUTION); - $maxAllowedWeight = Config::getDefaultServiceWeights($countryCode, $serviceCarrier['product_reference']); - - if (!$cartWeightValidator->validate($cart, $parcelDistribution, $maxAllowedWeight)) { - return false; - } - - if ($serviceCarrier['product_reference'] === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY) { - $isSameDayAvailable = \Invertus\dpdBaltics\Util\ProductUtility::validateSameDayDelivery( - $countryCode, - $deliveryAddress->city - ); - - if (!$isSameDayAvailable) { - return false; - } - } - - /** @var ShippingPriceCalculationService $shippingPriceCalculationService */ - $shippingPriceCalculationService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.shipping_price_calculation_service'); - - return $shippingPriceCalculationService->calculate($cart, $carrier, $deliveryAddress); - } - - public function hookDisplayCarrierExtraContent($params) - { - /** @var Cart $cart */ - $cart = $params['cart']; - $carrier = new Carrier($params['carrier']['id']); - - /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ - $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); - $countryCode = $currentCountryProvider->getCurrentCountryIsoCode($cart); - - $deliveryAddress = new Address($cart->id_address_delivery); - - /** @var CarrierPhoneService $carrierPhoneService */ - /** @var \Invertus\dpdBaltics\Presenter\DeliveryTimePresenter $deliveryTimePresenter */ - /** @var ProductRepository $productRepo */ - $carrierPhoneService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.carrier_phone_service'); - $deliveryTimePresenter = $this->getModuleContainer()->get('invertus.dpdbaltics.presenter.delivery_time_presenter'); - $productRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.product_repository'); - - $productId = $productRepo->getProductIdByCarrierReference($carrier->id_reference); - $dpdProduct = new DPDProduct($productId); - $return = ''; - if ($dpdProduct->getProductReference() === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY) { - /** @var \Invertus\dpdBaltics\Presenter\SameDayDeliveryMessagePresenter $sameDayDeliveryPresenter */ - $sameDayDeliveryPresenter = $this->getModuleContainer()->get('invertus.dpdbaltics.presenter.same_day_delivery_message_presenter'); - $return .= $sameDayDeliveryPresenter->getSameDayDeliveryMessageTemplate(); - } - $return .= $carrierPhoneService->getCarrierPhoneTemplate($this->context->cart->id, $carrier->id_reference); - if ($dpdProduct->getProductReference() === Config::PRODUCT_TYPE_B2B || - $dpdProduct->getProductReference() === Config::PRODUCT_TYPE_B2B_COD - ) { - $return .= $deliveryTimePresenter->getDeliveryTimeTemplate($countryCode, $deliveryAddress->city); - } - - /** @var ProductRepository $productRep */ - $productRep = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); - $isPudo = $productRep->isProductPudo($carrier->id_reference); - if ($isPudo) { - /** @var PudoRepository $pudoRepo */ - /** @var ParcelShopRepository $parcelShopRepo */ - /** @var ProductRepository $productRepo */ - $pudoRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.pudo_repository'); - $parcelShopRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.parcel_shop_repository'); - $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); - $product = $productRepo->findProductByCarrierReference($carrier->id_reference); - $isSameDayDelivery = ($product['product_reference'] === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY); - - $pudoId = $pudoRepo->getIdByCart($cart->id); - $selectedPudo = new DPDPudo($pudoId); - - /** @var ParcelShopService $parcelShopService */ - $parcelShopService= $this->getModuleContainer('invertus.dpdbaltics.service.parcel.parcel_shop_service'); - - $selectedCity = null; - $selectedStreet = null; - - try { - if (Validate::isLoadedObject($selectedPudo) && !$isSameDayDelivery) { - $selectedCity = $selectedPudo->city; - $selectedStreet = $selectedPudo->street; - $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity($countryCode, $selectedCity); - $parcelShops = $parcelShopService->moveSelectedShopToFirst($parcelShops, $selectedStreet); - } else { - $selectedCity = $deliveryAddress->city; - $selectedStreet = $deliveryAddress->address1; - $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity($countryCode, $selectedCity); - $parcelShops = $parcelShopService->moveSelectedShopToFirst($parcelShops, $selectedStreet); - if (!$parcelShops) { - $selectedCity = null; - } - } - } catch (DPDBalticsAPIException $e) { - /** @var ExceptionService $exceptionService */ - $exceptionService = $this->getModuleContainer('invertus.dpdbaltics.service.exception.exception_service'); - $tplVars = [ - 'errorMessage' => $exceptionService->getErrorMessageForException( - $e, - $exceptionService->getAPIErrorMessages() - ) - ]; - $this->context->smarty->assign($tplVars); - - return $this->context->smarty->fetch( - $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' - ); - } catch (Exception $e) { - $tplVars = [ - 'errorMessage' => $this->l("Something went wrong. We couldn't find parcel shops."), - ]; - $this->context->smarty->assign($tplVars); - - return $this->context->smarty->fetch( - $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' - ); - } - - /** @var PudoService $pudoService */ - $pudoService = $this->getModuleContainer('invertus.dpdbaltics.service.pudo_service'); - - $pudoServices = $pudoService->setPudoServiceTypes($parcelShops); - $pudoServices = $pudoService->formatPudoServicesWorkHours($pudoServices); - - if (isset($parcelShops[0])) { - $coordinates = [ - 'lat' => $parcelShops[0]->getLatitude(), - 'lng' => $parcelShops[0]->getLongitude(), - ]; - $this->context->smarty->assign( - [ - 'coordinates' => $coordinates - ] - ); - } - - if ($isSameDayDelivery) { - $cityList['Rīga'] = 'Rīga'; - } else { - $cityList = $parcelShopRepo->getAllCitiesByCountryCode($countryCode); - } - - if (!in_array($selectedCity, $cityList) && isset($parcelShops[0])) { - $selectedCity = $parcelShops[0]->getCity(); - } - - if (!$selectedCity && Tools::getValue('controller') !== 'order') { - $tplVars = [ - 'displayMessage' => true, - 'messages' => [$this->l("Your delivery address city is not in a list of pickup cities, please select closest pickup point city below manually")], - 'messageType_pudo' => 'danger' - - ]; - $this->context->smarty->assign($tplVars); - } - - $streetList = $parcelShopRepo->getAllAddressesByCountryCodeAndCity($countryCode, $selectedCity); - $this->context->smarty->assign( - [ - 'currentController' => Tools::getValue('controller'), - 'carrierId' => $carrier->id, - 'pickUpMap' => Configuration::get(Config::PICKUP_MAP), - 'pudoId' => $pudoId, - 'pudoServices' => $pudoServices, - 'dpd_pickup_logo' => $this->getPathUri() . 'views/img/pickup.png', - 'dpd_locker_logo' => $this->getPathUri() . 'views/img/locker.png', - 'delivery_address' => $deliveryAddress, - 'saved_pudo_id' => $selectedPudo->pudo_id, - 'is_pudo' => (bool)$isPudo, - 'city_list' => $cityList, - 'selected_city' => $selectedCity, - 'show_shop_list' => Configuration::get(Config::PARCEL_SHOP_DISPLAY), - 'street_list' => $streetList, - 'selected_street' => $selectedStreet, - ] - ); - - $return .= $this->context->smarty->fetch( - $this->getLocalPath() . '/views/templates/hook/front/pudo-points.tpl' - ); - } - - return $return; - } - - /** - * Includes Vendor Autoload. - */ - private function autoLoad() - { - require_once $this->getLocalPath() . 'vendor/autoload.php'; - } - - private function compile() - { - $containerBuilder = new ContainerBuilder(); - $locator = new FileLocator($this->getLocalPath() . 'config'); - $loader = new YamlFileLoader($containerBuilder, $locator); - $loader->load('config.yml'); - $containerBuilder->compile(); - - $this->moduleContainer = $containerBuilder; - } - - public function hookActionAdminControllerSetMedia($params) - { - $currentController = Tools::getValue('controller'); - - if (Config::isPrestashopVersionBelow174()) { - /** @var \Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs $moduleTabs */ - $moduleTabs = $this->getService(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::class); - $visibleClasses = $moduleTabs->getTabsClassNames(false); - - if (in_array($currentController, $visibleClasses, true)) { - Media::addJsDef(['visibleTabs' => $moduleTabs->getTabsClassNames(true)]); - $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/tabsHandlerBelowPs174.js'); - } - } - - if ('AdminOrders' === $currentController) { - $this->handleLabelPrintService(); - - $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/order-list.css'); - $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/order_list.js'); - Media::addJsDef( - [ - 'dpdHookAjaxShipmentController' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_AJAX_SHIPMENTS_CONTROLLER), - 'shipmentIsBeingPrintedMessage' => $this->context->smarty->fetch($this->getLocalPath() . 'views/templates/admin/partials/spinner.tpl') . - $this->l('Your labels are being saved please stay on the page'), - 'noOrdersSelectedMessage' => $this->l('No orders were selected'), - 'downloadSelectedLabelsButton' => $this->context->smarty->fetch($this->getLocalPath() . 'views/templates/admin/partials/download-selected-labels-button.tpl') - ] - ); - - $orderId = Tools::getValue('id_order'); - $shipment = $this->getShipment($orderId); - $baseUrl = $this->context->shop->getBaseURL(true, false); - $isAbove177 = Config::isPrestashopVersionAbove177(); - $baseUrlAdmin = $isAbove177 ? $this->context->link->getAdminBaseLink() : null; - /** @var \Invertus\dpdBaltics\Service\Label\LabelUrlFormatter $labelUrlService */ - $labelUrlService = $isAbove177 ? $this->getModuleContainer('invertus.dpdbaltics.service.label.label_url_formatter') : null; - - Media::addJsDef( - [ - 'print_url' => $labelUrlService && $isAbove177 ? $baseUrl.$labelUrlService->formatJsLabelPrintUrl() : null, - 'print_and_save_label_url' => $labelUrlService && $isAbove177 ? $baseUrl.$labelUrlService->formatJsLabelSaveAndPrintUrl() : null, - 'shipment' => $shipment, - 'id_order' => $orderId, - 'is_label_download_option' => Configuration::get(Config::LABEL_PRINT_OPTION) === 'download', - 'is_ps_above_177' => Config::isPrestashopVersionAbove177(), - 'loader_url' => $isAbove177 ? "{$baseUrlAdmin}modules/dpdbaltics/views/templates/admin/loader/loader.html" : null - ] - ); - } - - if ('AdminOrders' === $currentController && - (Tools::isSubmit('vieworder') || Tools::getValue('action') === 'vieworder') - ) { - $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/order_expand_form.js'); - $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/shipment.js'); - Media::addJsDef([ - 'expandText' => $this->l('Expand'), - 'collapseText' => $this->l('Collapse'), - 'dpdAjaxShipmentsUrl' => - $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_AJAX_SHIPMENTS_CONTROLLER), - 'dpdMessages' => [ - 'invalidProductQuantity' => $this->l('Invalid product quantity entered'), - 'invalidShipment' => $this->l('Invalid shipment selected'), - 'parcelsLimitReached' => $this->l('Parcels limit reached in shipment'), - 'successProductMove' => $this->l('Product moved successfully'), - 'successCreation' => $this->l('Successful creation'), - 'unexpectedError' => $this->l('Unexpected error appeared.'), - 'invalidPrintoutFormat' => $this->l('Invalid printout format selected.'), - 'cannotOpenWindow' => $this->l('Cannot print label, your browser may be blocking it.'), - 'dpdRecipientAddressError' => $this->l('Please fill required fields') - ], - 'id_language' => $this->context->language->id, - 'id_shop' => $this->context->shop->id, - 'id_cart' => $this->context->cart->id, - 'currentController' => $currentController, - - ]); - - $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/carrier_phone.js'); - $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/custom_select.js'); - $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/label_position.js'); - $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/pudo.js'); - $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/customSelect/custom-select.css'); - $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/admin-orders-controller.css'); - $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/quantity-max-value-tip.css'); - } - if ('AdminOrders' === $currentController && - (Tools::isSubmit('addorder') || Tools::getValue('action') === 'addorder') - ) { - /** @var ProductRepository $productRepo */ - $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); - - Media::addJsDef([ - 'dpdFrontController' => false, - 'pudoCarriers' => json_encode($productRepo->getPudoProducts()), - 'currentController' => $currentController, - 'dpdAjaxShipmentsUrl' => - $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_AJAX_SHIPMENTS_CONTROLLER), - 'ignoreAdminController' => true, - 'dpdAjaxPudoUrl' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_PUDO_AJAX_CONTROLLER), - 'id_shop' => $this->context->shop->id, - ]); - - $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/pudo_list.js'); - $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/carrier_phone.js'); - $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/pudo.js'); - - $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/admin-orders-controller.css'); - - return; - } - } - - public function hookDisplayAdminOrder(array $params) - { - return Config::isPrestashopVersionAbove177() ? false : $this->displayInAdminOrderPage($params); - } - - public function hookDisplayAdminOrderTabContent(array $params) - { - return !Config::isPrestashopVersionAbove177() ? false : $this->displayInAdminOrderPage($params); - } - - private function displayInAdminOrderPage(array $params) - { - $order = new Order($params['id_order']); - $cart = Cart::getCartByOrderId($params['id_order']); - - /** @var ProductRepository $productRepo */ - $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); - $carrier = new Carrier($order->id_carrier); - if (!$productRepo->getProductIdByCarrierReference($carrier->id_reference)) { - return; - } - - $shipment = $this->getShipment($order->id); - $dpdCodWarning = false; - - /** @var OrderService $orderService */ - $orderService = $this->getModuleContainer('invertus.dpdbaltics.service.order_service'); - $orderDetails = $orderService->getOrderDetails($order, $cart->id_lang); - - $customAddresses = []; - - /** @var ReceiverAddressRepository $receiverAddressRepository */ - $receiverAddressRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.receiver_address_repository'); - - $customOrderAddressesIds = $receiverAddressRepository->getAddressIdByOrderId($order->id); - foreach ($customOrderAddressesIds as $customOrderAddressId) { - $customAddress = new Address($customOrderAddressId); - - $customAddresses[] = [ - 'id_address' => $customAddress->id, - 'alias' => $customAddress->alias - ]; - } - $combinedCustomerAddresses = array_merge($orderDetails['customer']['addresses'], $customAddresses); - - /** @var PhonePrefixRepository $phonePrefixRepository */ - $phonePrefixRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.phone_prefix_repository'); - - $products = $cart->getProducts(); - - /** @var LabelPositionService $labelPositionService */ - $labelPositionService = $this->getModuleContainer('invertus.dpdbaltics.service.label.label_position_service'); - $labelPositionService->assignLabelPositions($shipment->id); - $labelPositionService->assignLabelFormat($shipment->id); - - /** @var PaymentService $paymentService */ - $paymentService = $this->getModuleContainer('invertus.dpdbaltics.service.payment.payment_service'); - try { - $isCodPayment = $paymentService->isOrderPaymentCod($order->module); - } catch (Exception $e) { - $tplVars = [ - 'errorMessage' => $this->l('Something went wrong checking payment method'), - ]; - $this->context->smarty->assign($tplVars); - - return $this->context->smarty->fetch( - $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' - ); - } - - /** @var PudoRepository $pudoRepo */ - $pudoRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.pudo_repository'); - $selectedProduct = new DPDProduct($shipment->id_service); - $isPudo = $selectedProduct->is_pudo; - $pudoId = $pudoRepo->getIdByCart($order->id_cart); - - $selectedPudo = new DPDPudo($pudoId); - - /** @var ParcelShopService $parcelShopService */ - /** @var ParcelShopRepository $parcelShopRepo */ - $parcelShopService= $this->getModuleContainer('invertus.dpdbaltics.service.parcel.parcel_shop_service'); - $parcelShopRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.parcel_shop_repository'); - - /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ - $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); - $countryCode = $currentCountryProvider->getCurrentCountryIsoCode($cart); - - $selectedCity = null; - try { - if ($pudoId) { - $selectedCity = $selectedPudo->city; - $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity( - $countryCode, - $selectedCity - ); - } else { - $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity( - $countryCode, - $orderDetails['order_address']['city'] - ); - } - } catch (Exception $e) { - $tplVars = [ - 'errorMessage' => $this->l('Fatal error while searching for parcel shops: ') . $e->getMessage(), - ]; - $this->context->smarty->assign($tplVars); - - return $this->context->smarty->fetch( - $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' - ); - } - - /** @var null|\Invertus\dpdBalticsApi\Api\DTO\Object\ParcelShop $selectedPudoService */ - $selectedPudoService = null; - $hasParcelShops = false; - if ($parcelShops) { - if ($selectedPudo->pudo_id) { - $selectedPudoService = $parcelShopService->getParcelShopByShopId($selectedPudo->pudo_id)[0]; - } else { - $selectedPudoService = $parcelShops[0]; - } - $hasParcelShops = true; - } - - if ($selectedPudoService) { - /** @var PudoService $pudoService */ - $pudoService = $this->getModuleContainer('invertus.dpdbaltics.service.pudo_service'); - - $selectedPudoService->setOpeningHours( - $pudoService->formatPudoServiceWorkHours($selectedPudoService->getOpeningHours()) - ); - } - - /** @var ProductRepository $productRepository */ - $productRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); - $dpdProducts = $productRepository->getAllProducts(); - - $dpdCarrierOptions = []; - - /** @var DPDProduct $dpdProduct */ - foreach ($dpdProducts as $dpdProduct) { - $dpdCarrierOptions[] = [ - 'id_dpd_product' => $dpdProduct->id_dpd_product, - 'name' => $dpdProduct->name, - 'available' => - $dpdProduct->active && - (int) $dpdProduct->is_cod === (int) $isCodPayment && - (!$dpdProduct->is_pudo && $hasParcelShops) - ]; - } - - $cityList = $parcelShopRepo->getAllCitiesByCountryCode($countryCode); - - if (\Invertus\dpdBaltics\Config\Config::productHasDeliveryTime($selectedProduct->product_reference)) { - /** @var \Invertus\dpdBaltics\Repository\OrderDeliveryTimeRepository $orderDeliveryTimeRepo */ - $orderDeliveryTimeRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.order_delivery_time_repository'); - $orderDeliveryTimeId = $orderDeliveryTimeRepo->getOrderDeliveryIdByCartId($cart->id); - if ($orderDeliveryTimeId) { - $orderDeliveryTime = new DPDOrderDeliveryTime($orderDeliveryTimeId); - $this->context->smarty->assign([ - 'orderDeliveryTime' => $orderDeliveryTime->delivery_time, - 'deliveryTimes' => \Invertus\dpdBaltics\Config\Config::getDeliveryTimes($countryCode) - ]); - } - } - - $href = $this->context->link->getModuleLink( - $this->name, - 'ShipmentReturn', - [ - 'id_order' => $order->id, - 'dpd-return-submit' => '' - ] - ); - - $tplVars = [ - 'dpdLogoUrl' => $this->getPathUri() . 'views/img/DPDLogo.gif', - 'shipment' => $shipment, - 'isAbove177' => Config::isPrestashopVersionAbove177(), - 'testOrder' => $shipment->is_test, - 'total_products' => 1, - 'contractPageLink' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_PRODUCTS_CONTROLLER), - 'dpdCodWarning' => $dpdCodWarning, - 'testMode' => Configuration::get(Config::SHIPMENT_TEST_MODE), - 'printLabelOption' => Configuration::get(Config::LABEL_PRINT_OPTION), - 'defaultLabelFormat' => Configuration::get(Config::DEFAULT_LABEL_FORMAT), - 'combinedAddresses' => $combinedCustomerAddresses, - 'orderDetails' => $orderDetails, - 'mobilePhoneCodeList' => $phonePrefixRepository->getCallPrefixes(), - 'products' => $products, - 'dpdProducts' => $dpdCarrierOptions, - 'isCodPayment' => $isCodPayment, - 'is_pudo' => (bool)$isPudo, - 'selectedPudo' => $selectedPudoService, - 'city_list' => $cityList, - 'selected_city' => $selectedCity, - 'has_parcel_shops' => $hasParcelShops, - 'receiverAddressCountries' => Country::getCountries($this->context->language->id, true), - 'documentReturnEnabled' => Configuration::get(Config::DOCUMENT_RETURN), - 'href' => $href, - 'adminLabelLink' => $this->context->link->getAdminLink( - 'AdminDPDBalticsAjaxShipments', - true, - [], - ['action' => 'print-return'] - ), - 'isAutomated' => Configuration::get(Config::AUTOMATED_PARCEL_RETURN), - ]; - - $this->context->smarty->assign($tplVars); - - return $this->context->smarty->fetch( - $this->getLocalPath() . 'views/templates/hook/admin/admin-order.tpl' - ); - } - - public function hookActionValidateOrder($params) - { - $carrier = new Carrier($params['order']->id_carrier); - if ($carrier->external_module_name !== $this->name) { - return; - } - - $isAdminOrderPage = 'AdminOrders' === Tools::getValue('controller') || Config::isPrestashopVersionAbove177(); - $isAdminNewOrderForm = Tools::isSubmit('addorder') || Tools::isSubmit('cart_summary'); - - if ($isAdminOrderPage && $isAdminNewOrderForm) { - $dpdPhone = Tools::getValue('dpd-phone'); - $dpdPhoneArea = Tools::getValue('dpd-phone-area'); - - /** @var \Invertus\dpdBaltics\Service\OrderDeliveryTimeService $orderDeliveryService */ - $carrierPhoneService = $this->getModuleContainer('invertus.dpdbaltics.service.carrier_phone_service'); - - if (!empty($dpdPhone) && !empty($dpdPhoneArea)) { - try { - $carrierPhoneService->saveCarrierPhone( - $this->context->cart->id, - $dpdPhone, - $dpdPhoneArea - ); - } catch (Exception $exception) { - if ($exception->getCode() === Config::ERROR_COULD_NOT_SAVE_PHONE_NUMBER) { - $error = $this->l('Phone data is not saved'); - die($error); - } - } - } - - /** @var CarrierPhoneService $carrierPhoneService */ - $orderDeliveryService = $this->getModuleContainer('invertus.dpdbaltics.service.order_delivery_time_service'); - $deliveryTime = Tools::getValue('dpd-delivery-time'); - if ($deliveryTime !== null) { - if (!$orderDeliveryService->saveDeliveryTime( - $this->context->cart->id, - $deliveryTime - )) { - $error = $this->l('Delivery time is not saved'); - die($error); - }; - } - } - - /** @var ShipmentService $shipmentService */ - $shipmentService = $this->getModuleContainer('invertus.dpdbaltics.service.shipment_service'); - $shipmentService->createShipmentFromOrder($params['order']); - } - - public function printLabel($idShipment) - { - /** @var \Invertus\dpdBaltics\Service\LabelPrintingService $labelPrintingService */ - $labelPrintingService = $this->getModuleContainer('invertus.dpdbaltics.service.label_printing_service'); - - $parcelPrintResponse = $labelPrintingService->printOne($idShipment); - - if ($parcelPrintResponse->getStatus() === Config::API_SUCCESS_STATUS) { - $this->updateOrderCarrier($idShipment); - return; - } - - return $parcelPrintResponse; - } - - public function printMultipleLabels($shipmentIds) - { - /** @var \Invertus\dpdBaltics\Service\LabelPrintingService $labelPrintingService */ - $labelPrintingService = $this->getModuleContainer('invertus.dpdbaltics.service.label_printing_service'); - - $parcelPrintResponse = $labelPrintingService->printMultiple($shipmentIds); - - if ($parcelPrintResponse->getStatus() === Config::API_SUCCESS_STATUS) { - foreach ($shipmentIds as $shipmentId) { - $this->updateOrderCarrier($shipmentId); - } - return; - } - - return $parcelPrintResponse; - } - - private function updateOrderCarrier($shipmentId) - { - $shipment = new DPDShipment($shipmentId); - /** @var OrderRepository $orderRepo */ - /** @var TrackingService $trackingService */ - $orderRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.order_repository'); - $trackingService = $this->getModuleContainer('invertus.dpdbaltics.service.tracking_service'); - $orderCarrierId = $orderRepo->getOrderCarrierId($shipment->id_order); - - $orderCarrier = new OrderCarrier($orderCarrierId); - $orderCarrier->tracking_number = $trackingService->getTrackingNumber($shipment->pl_number); - - try { - $orderCarrier->update(); - } catch (Exception $e) { - Context::getContext()->controller->errors[] = - $this->l('Failed to save tracking number: ') . $e->getMessage(); - return; - } - - $shipment->printed_label = 1; - $shipment->date_print = date('Y-m-d H:i:s'); - $shipment->update(); - } - - public function hookDisplayOrderDetail($params) - { - $isReturnServiceEnabled = Configuration::get(Config::PARCEL_RETURN); - if (!$isReturnServiceEnabled) { - return; - } - - if (CountryUtility::isEstonia()) { - return; - } - - /** @var ShipmentRepository $shipmentRepo */ - $shipmentRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.shipment_repository'); - $shipmentId = $shipmentRepo->getIdByOrderId($params['order']->id); - - $orderState = new OrderState($params['order']->current_state); - if (!$orderState->delivery) { - return; - } - /** @var AddressTemplateRepository $addressTemplateRepo */ - $addressTemplateRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.address_template_repository'); - $returnAddressTemplates = $addressTemplateRepo->getReturnServiceAddressTemplates(); - - $shipment = new DPDShipment($shipmentId); - - if (!$returnAddressTemplates) { - return; - } - - $showTemplates = false; - if (sizeof($returnAddressTemplates) > 1 && !$shipment->return_pl_number) { - $showTemplates = true; - } - - if (isset($this->context->cookie->dpd_error)) { - $this->context->controller->errors[] = json_decode($this->context->cookie->dpd_error); - unset($this->context->cookie->dpd_error); - } - $href = $this->context->link->getModuleLink( - $this->name, - 'ShipmentReturn', - [ - 'id_order' => $params['order']->id, - 'dpd-return-submit' => '' - ] - ); - - $this->context->smarty->assign( - [ - 'href' => $href, - 'return_template_ids' => $returnAddressTemplates, - 'show_template' => $showTemplates, - ] - ); - $html = $this->context->smarty->fetch( - $this->getLocalPath() . 'views/templates/hook/front/order-detail.tpl' - ); - - return $html; - } - - public function hookActionAdminOrdersListingFieldsModifier($params) - { - if ((bool) Configuration::get(Config::HIDE_ORDERS_LABEL_PRINT_BUTTON)) { - return false; - } - - if (isset($params['select'])) { - $params['select'] .= ' ,ds.`id_order` AS id_order_shipment '; - } - - if (isset($params['join'])) { - $params['join'] .= ' LEFT JOIN `' . _DB_PREFIX_ . 'dpd_shipment` ds ON ds.`id_order` = a.`id_order` '; - } - - $params['fields']['id_order_shipment'] = [ - 'title' => $this->l('DPD Label'), - 'align' => 'text-center', - 'class' => 'fixed-width-xs', - 'orderby' => false, - 'search' => false, - 'remove_onclick' => true, - 'callback_object' => 'dpdbaltics', - 'callback' => 'returnOrderListIcon' - ]; - } - - /** - * Callback function, it has to be static so can't call $this, so have to reload dpdBaltics module inside the function - * @param $idOrder - * @return string - * @throws Exception - */ - public static function returnOrderListIcon($orderId) - { - $dpdBaltics = Module::getInstanceByName('dpdbaltics'); - - $dpdBaltics->context->smarty->assign('idOrder', $orderId); - - $dpdBaltics->context->smarty->assign( - 'message', - $dpdBaltics->l('Print label(s) from DPD system. Once label is saved you won\'t be able to modify contents of shipments') - ); - $icon = $dpdBaltics->context->smarty->fetch( - $dpdBaltics->getLocalPath() . 'views/templates/hook/admin/order-list-save-label-icon.tpl' - ); - - - $dpdBaltics->context->smarty->assign('icon', $icon); - - return $dpdBaltics->context->smarty->fetch($dpdBaltics->getLocalPath() . 'views/templates/hook/admin/order-list-icon-container.tpl'); - } - - - public function hookDisplayAdminListBefore() - { - if ($this->context->controller instanceof AdminOrdersControllerCore) { - return $this->context->smarty->fetch($this->getLocalPath() . 'views/templates/hook/admin/admin-orders-header-hook.tpl'); - } - } - - public function hookActionOrderGridDefinitionModifier(array $params) - { - if (!Config::isPrestashopVersionAbove177()) { - return false; - } - - $definition = $params['definition']; - - if (!(bool) Configuration::get(Config::HIDE_ORDERS_LABEL_PRINT_BUTTON)) { - $definition->getColumns() - ->addAfter( - 'date_add', - (new ActionColumn('dpd_print_label')) - ->setName($this->l('Dpd print label')) - ->setOptions([ - 'actions' => $this->getGridAction() - ]) - ); - } - - $definition->getBulkActions() - ->add( - (new SubmitBulkActionCustom('print_multiple_labels')) - ->setName($this->l('Print multiple labels')) - ->setOptions([ - 'submit_route' => 'dpdbaltics_save_and_download_printed_labels_order_list_multiple', - ]) - ) - ; - } - - /** - * @return RowActionCollection - */ - private function getGridAction() - { - return (new RowActionCollection()) - ->add( - (new LinkRowActionCustom('print_delivery_slip')) - ->setName($this->l('Print label(s) from DPD system. Once label is saved you won\'t be able to modify contents of shipments')) - ->setIcon('print') - ->setOptions([ - 'route' => 'dpdbaltics_save_and_download_printed_label_order_list', - 'route_param_name' => 'orderId', - 'route_param_field' => 'id_order', - 'is_label_download' => Configuration::get(Config::LABEL_PRINT_OPTION) === 'download', - 'confirm_message' => $this->l('Would you like to print shipping label?'), - 'accessibility_checker' => $this->getModuleContainer()->get('invertus.dpdbaltics.grid.row.print_accessibility_checker'), - ]) - ); - } - - private function getShipment($idOrder) - { - if (!$idOrder) { - return false; - } - /** @var ShipmentRepository $shipmentRepository */ - $shipmentRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.shipment_repository'); - $shipmentId = $shipmentRepository->getIdByOrderId($idOrder); - $shipment = new DPDShipment($shipmentId); - - if (!Validate::isLoadedObject($shipment)) { - return false; - } - - return $shipment; - } - - private function handleLabelPrintService() - { - if (Tools::isSubmit('print_label')) { - $idShipment = Tools::getValue('id_dpd_shipment'); - - try { - $parcelPrintResponse = $this->printLabel($idShipment); - } catch (DPDBalticsAPIException $e) { - /** @var ExceptionService $exceptionService */ - $exceptionService = $this->getModuleContainer('invertus.dpdbaltics.service.exception.exception_service'); - Context::getContext()->controller->errors[] = $exceptionService->getErrorMessageForException( - $e, - $exceptionService->getAPIErrorMessages() - ); - return; - } catch (Exception $e) { - Context::getContext()->controller->errors[] = $this->l('Failed to print label: ') . $e->getMessage(); - return; - } - - if (!empty($parcelPrintResponse->getErrLog())) { - Context::getContext()->controller->errors[] = $parcelPrintResponse->getErrLog(); - } - - exit; - } - - if (Tools::isSubmit('print_multiple_labels')) { - $shipmentIds = json_decode(Tools::getValue('shipment_ids')); - - try { - $parcelPrintResponse = $this->printMultipleLabels($shipmentIds); - } catch (DPDBalticsAPIException $e) { - /** @var ExceptionService $exceptionService */ - $exceptionService = $this->getModuleContainer('invertus.dpdbaltics.service.exception.exception_service'); - Context::getContext()->controller->errors[] = $exceptionService->getErrorMessageForException( - $e, - $exceptionService->getAPIErrorMessages() - ); - return; - } catch (Exception $e) { - Context::getContext()->controller->errors[] = $this->l('Failed to print label: ') . $e->getMessage(); - return; - } - - if (!empty($parcelPrintResponse->getErrLog())) { - Context::getContext()->controller->errors[] = $parcelPrintResponse->getErrLog(); - } - - exit; - } - } -} + * @copyright Since 2007 PrestaShop SA and Contributors * @license https://opensource.org/licenses/AFL-3.0 Academic Free License version 3.0 */ use Invertus\dpdBaltics\Grid\Row\PrintAccessibilityChecker; use Invertus\dpdBaltics\Builder\Template\Front\CarrierOptionsBuilder; use Invertus\dpdBaltics\Config\Config; use Invertus\dpdBaltics\Controller\AbstractAdminController; use Invertus\dpdBaltics\Grid\LinkRowActionCustom; use Invertus\dpdBaltics\Grid\SubmitBulkActionCustom; use invertus\dpdbaltics\install\installer; use Invertus\dpdBaltics\Logger\Logger; use Invertus\dpdBaltics\OnBoard\Service\OnBoardService; use Invertus\dpdBaltics\Repository\AddressTemplateRepository; use Invertus\dpdBaltics\Repository\OrderRepository; use Invertus\dpdBaltics\Repository\ParcelShopRepository; use Invertus\dpdBaltics\Repository\PhonePrefixRepository; use Invertus\dpdBaltics\Repository\PriceRuleRepository; use Invertus\dpdBaltics\Repository\ProductRepository; use Invertus\dpdBaltics\Repository\PudoRepository; use Invertus\dpdBaltics\Repository\ReceiverAddressRepository; use Invertus\dpdBaltics\Repository\ShipmentRepository; use Invertus\dpdBaltics\Repository\ZoneRepository; use Invertus\dpdBaltics\Service\API\LabelApiService; use Invertus\dpdBaltics\Service\API\ParcelShopSearchApiService; use Invertus\dpdBaltics\Service\CarrierPhoneService; use Invertus\dpdBaltics\Service\Exception\ExceptionService; use Invertus\dpdBaltics\Service\GoogleApiService; use Invertus\dpdBaltics\Service\Label\LabelPositionService; use Invertus\dpdBaltics\Service\OrderService; use Invertus\dpdBaltics\Service\Parcel\ParcelShopService; use Invertus\dpdBaltics\Service\Payment\PaymentService; use Invertus\dpdBaltics\Service\PriceRuleService; use Invertus\dpdBaltics\Service\PudoService; use Invertus\dpdBaltics\Service\ShipmentService; use Invertus\dpdBaltics\Service\ShippingPriceCalculationService; use Invertus\dpdBaltics\Service\TabService; use Invertus\dpdBaltics\Service\TrackingService; use Invertus\dpdBaltics\Util\CountryUtility; use Invertus\dpdBaltics\Validate\Carrier\PudoValidate; use Invertus\dpdBalticsApi\Api\DTO\Response\ParcelPrintResponse; use Invertus\dpdBalticsApi\Api\DTO\Response\ParcelShopSearchResponse; use Invertus\dpdBalticsApi\Exception\DPDBalticsAPIException; use Invertus\dpdBalticsApi\Factory\SerializerFactory; use PrestaShop\PrestaShop\Core\Grid\Action\Bulk\Type\SubmitBulkAction; use PrestaShop\PrestaShop\Core\Grid\Action\Row\RowActionCollection; use PrestaShop\PrestaShop\Core\Grid\Action\Row\Type\LinkRowAction; use PrestaShop\PrestaShop\Core\Grid\Column\Type\Common\ActionColumn; use Symfony\Component\Config\ConfigCache; use Symfony\Component\Config\FileLocator; use Symfony\Component\Console\Application; use Symfony\Component\DependencyInjection\ContainerBuilder; use Symfony\Component\DependencyInjection\Dumper\PhpDumper; use Symfony\Component\DependencyInjection\Loader\YamlFileLoader; class DPDBaltics extends CarrierModule { /** * Symfony DI Container **/ private $moduleContainer; /** * Prestashop fills this property automatically with selected carrier ID in FO checkout * * @var int $id_carrier */ public $id_carrier; public function __construct() { $this->name = 'dpdbaltics'; $this->displayName = $this->l('DPDBaltics'); $this->author = 'Invertus'; $this->tab = 'shipping_logistics'; $this->description = 'DPD Baltics shipping integration'; $this->version = '3.2.19'; $this->ps_versions_compliancy = ['min' => '1.7.1.0', 'max' => _PS_VERSION_]; $this->need_instance = 0; parent::__construct(); $this->autoLoad(); $this->compile(); } public function install() { if (!parent::install()) { return false; } /** @var \Invertus\dpdBaltics\Infrastructure\Bootstrap\Install\Installer $installer */ $installer = $this->getService(\Invertus\dpdBaltics\Infrastructure\Bootstrap\Install\Installer::class); try { $installer->init(); return true; } catch (\Invertus\dpdBaltics\Infrastructure\Bootstrap\Exception\CouldNotInstallModule $exception) { $this->_errors[] = $exception->getMessage(); return false; } } public function uninstall() { $uninstall = parent::uninstall(); if (!$uninstall) { return false; } /** @var \Invertus\dpdBaltics\Infrastructure\Bootstrap\Uninstall\Uninstaller $uninstaller */ $uninstaller = $this->getService(\Invertus\dpdBaltics\Infrastructure\Bootstrap\Uninstall\Uninstaller::class); try { $uninstaller->init(); return true; } catch (\Invertus\dpdBaltics\Infrastructure\Bootstrap\Exception\CouldNotUninstallModule $exception) { $this->_errors[] = $exception->getMessage(); return false; } } public function getContent() { Tools::redirectAdmin($this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_SETTINGS_CONTROLLER)); } public function getService($serviceName) { return $this->getModuleContainer()->get($serviceName); } /** * @return mixed */ public function getModuleContainer($id = false) { if ($id) { return $this->moduleContainer->get($id); } return $this->moduleContainer; } public function hookActionFrontControllerSetMedia() { //TODO fillup this array when more modules are compatible with OPC $onePageCheckoutControllers = ['supercheckout', 'onepagecheckoutps', 'thecheckout']; $applicableControlelrs = ['order', 'order-opc', 'ShipmentReturn', 'supercheckout']; $currentController = !empty($this->context->controller->php_self) ? $this->context->controller->php_self : Tools::getValue('controller'); if ('product' === $currentController) { $this->context->controller->registerStylesheet( 'dpdbaltics-product-carriers.', 'modules/' . $this->name . '/views/css/front/product-carriers.css', [ 'media' => 'all', 'position' => 150 ] ); } /** @var \Invertus\dpdBaltics\Validate\Compatibility\OpcModuleCompatibilityValidator $opcModuleCompatibilityValidator */ $opcModuleCompatibilityValidator = $this->getModuleContainer('invertus.dpdbaltics.validator.opc_module_compatibility_validator'); if (in_array($currentController, $onePageCheckoutControllers, true) || $opcModuleCompatibilityValidator->isOpcModuleInUse()) { $this->context->controller->addJqueryPlugin('chosen'); $this->context->controller->registerJavascript( 'dpdbaltics-opc', 'modules/' . $this->name . '/views/js/front/order-opc.js', [ 'position' => 'bottom', 'priority' => 130 ] ); $this->context->controller->registerJavascript( 'dpdbaltics-supercheckout', 'modules/' . $this->name . '/views/js/front/modules/supercheckout.js', [ 'position' => 'bottom', 'priority' => 130 ] ); $this->context->controller->registerStylesheet( 'dpdbaltics-opc', 'modules/' . $this->name . '/views/css/front/onepagecheckout.css', [ 'position' => 'bottom', 'priority' => 130 ] ); } /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); $webServiceCountryCode = Configuration::get(Config::WEB_SERVICE_COUNTRY); $carrierIds = []; $baseUrl = $this->context->shop->getBaseURL(true, false); if ($webServiceCountryCode === Config::LATVIA_ISO_CODE || $currentController === 'supercheckout') { /** @var ProductRepository $productRepo */ $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); $dpdProductReferences = $productRepo->getAllActiveDpdProductReferences(); foreach ($dpdProductReferences as $reference) { $carrier = Carrier::getCarrierByReference($reference['id_reference']); if (Validate::isLoadedObject($carrier)) { $carrierIds[] = $carrier->id; } } } Media::addJsDef([ 'lapinas_img' => $baseUrl . $this->getPathUri() . 'views/img/lapinas.png', 'lapinas_text' => $this->l('Sustainable'), 'dpd_carrier_ids' => $carrierIds ]); if (in_array($currentController, $applicableControlelrs, true)) { Media::addJsDef([ 'select_an_option_translatable' => $this->l('Select an Option'), 'select_an_option_multiple_translatable' => $this->l('Select Some Options'), 'no_results_translatable' => $this->l('No results match'), ]); $this->context->controller->addJS($this->getPathUri() . 'views/js/front/order.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/front/order-input.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/front/sustainable-logo.js'); $this->context->controller->addCSS($this->getPathUri() . 'views/css/front/order-input.css'); $this->context->controller->addCSS($this->getPathUri() . 'views/css/front/sustainable-logo.css'); /** @var PaymentService $paymentService */ $paymentService = $this->getModuleContainer('invertus.dpdbaltics.service.payment.payment_service'); $isPickupMap = Configuration::get(\Invertus\dpdBaltics\Config\Config::PICKUP_MAP); $cart = Context::getContext()->cart; $paymentService->filterPaymentMethods($cart); $paymentService->filterPaymentMethodsByCod($cart); /** @var ProductRepository $productRepo */ $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); Media::addJsDef([ 'pudoCarriers' => json_encode($productRepo->getPudoProducts()), 'currentController' => $currentController, 'is_pickup_map' => $isPickupMap, 'id_language' => $this->context->language->id, 'id_shop' => $this->context->shop->id, 'dpdAjaxLoaderPath' => $this->getPathUri() . 'views/img/ajax-loader-big.gif', 'dpdPickupMarkerPath' => $this->getPathUri() . 'views/img/dpd-pick-up.png', 'dpdLockerMarkerPath' => $this->getPathUri() . 'views/img/locker.png', 'dpdHookAjaxUrl' => $this->context->link->getModuleLink($this->name, 'Ajax'), 'pudoSelectSuccess' => $this->l('Pick-up point selected'), 'dpd_carrier_ids' => $carrierIds ]); $this->context->controller->registerStylesheet( 'dpdbaltics-pudo-shipment', 'modules/' . $this->name . '/views/css/front/' . 'pudo-shipment.css', [ 'media' => 'all', 'position' => 150 ] ); if ($isPickupMap) { /** @var GoogleApiService $googleApiService */ $googleApiService = $this->getModuleContainer('invertus.dpdbaltics.service.google_api_service'); $this->context->controller->registerJavascript( 'dpdbaltics-google-api', $googleApiService->getFormattedGoogleMapsUrl(), [ 'server' => 'remote' ] ); } $this->context->controller->registerJavascript( 'dpdbaltics-pudo', 'modules/' . $this->name . '/views/js/front/pudo.js', [ 'position' => 'bottom', 'priority' => 130 ] ); $this->context->controller->registerJavascript( 'dpdbaltics-pudo-search', 'modules/' . $this->name . '/views/js/front/pudo-search.js', [ 'position' => 'bottom', 'priority' => 130 ] ); } } public function hookActionValidateStepComplete(&$params) { if ('delivery' !== $params['step_name']) { return; } /** @var Cart $cart */ $cart = $params['cart']; $carrier = new Carrier($cart->id_carrier); $idShop = $this->context->shop->id; /** @var Invertus\dpdBaltics\Repository\CarrierRepository $carrierRepo */ /** @var Invertus\dpdBaltics\Repository\ProductRepository $productRepo */ $carrierRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.carrier_repository'); $productRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.product_repository'); $carrierReference = $carrier->id_reference; $dpdCarriers = $carrierRepo->getDpdCarriers($idShop); $isDpdCarrier = false; foreach ($dpdCarriers as $dpdCarrier) { if ($carrierReference == $dpdCarrier['id_reference']) { $isDpdCarrier = true; $productId = $productRepo->getProductIdByCarrierReference($carrier->id_reference); $product = new DPDProduct($productId); $isSameDayDelivery = $product->product_reference === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY; break; } } if (!$isDpdCarrier) { return true; } if ($isSameDayDelivery) { /** @var PudoRepository $pudoRepo */ $pudoRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.pudo_repository'); $pudoId = $pudoRepo->getIdByCart($cart->id); $selectedPudo = new DPDPudo($pudoId); if ($selectedPudo->city !== Config::SAME_DAY_DELIVERY_CITY) { $this->context->controller->errors[] = $this->l('This carrier can\'t deliver to your selected city'); $params['completed'] = false; $selectedPudo->delete(); return; } } //NOTE: thecheckout triggers this hook without phone parameters the phone is saved with ajax request if (Tools::getValue('module') === 'thecheckout') { return; } if (!Tools::getValue('dpd-phone')) { $this->context->controller->errors[] = $this->l('In order to use DPD Carrier you need to enter phone number'); $params['completed'] = false; return; } if (!Tools::getValue('dpd-phone-area')) { $this->context->controller->errors[] = $this->l('In order to use DPD Carrier you need to enter phone area'); $params['completed'] = false; return; } /** @var CarrierPhoneService $carrierPhoneService */ $carrierPhoneService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.carrier_phone_service'); try { $carrierPhoneService->saveCarrierPhone( $this->context->cart->id, Tools::getValue('dpd-phone'), Tools::getValue('dpd-phone-area') ); } catch (Exception $exception) { if ($exception->getCode() === Config::ERROR_COULD_NOT_SAVE_PHONE_NUMBER) { $this->context->controller->errors[] = $this->l('Phone data is not saved'); $params['completed'] = false; } } /** @var \Invertus\dpdBaltics\Service\OrderDeliveryTimeService $orderDeliveryService */ $orderDeliveryService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.order_delivery_time_service'); $deliveryTime = Tools::getValue('dpd-delivery-time'); if ($deliveryTime) { if (!$orderDeliveryService->saveDeliveryTime( $this->context->cart->id, $deliveryTime )) { $this->context->controller->errors[] = $this->l('Delivery time data is not saved'); $params['completed'] = false; }; } /** @var Cart $cart */ $cart = $params['cart']; $carrier = new Carrier($cart->id_carrier); /** @var PudoValidate $pudoValidator */ $pudoValidator = $this->getModuleContainer('invertus.dpdbaltics.validate.carrier.pudo_validate'); if (!$pudoValidator->validatePickupPoints($cart->id, $carrier->id_reference)) { $carrier = new Carrier($cart->id_carrier, $this->context->language->id); $this->context->controller->errors[] = sprintf($this->l('Please select pickup point for carrier: %s.'), $carrier->name); $params['completed'] = false; } } public function hookDisplayProductPriceBlock($params) { if ($params['type'] == 'after_price') { /** @var CarrierOptionsBuilder $carrierOptionsBuilder */ $carrierOptionsBuilder = $this->getModuleContainer()->get('invertus.dpdbaltics.builder.template.front.carrier_options_builder'); return $carrierOptionsBuilder->renderCarrierOptionsInProductPage(); } } public function hookDisplayBackOfficeTop() { if ($this->context->controller instanceof AbstractAdminController && Configuration::get(Config::ON_BOARD_TURNED_ON) && Configuration::get(Config::ON_BOARD_STEP) ) { /** @var OnBoardService $onBoardService */ $onBoardService = $this->getModuleContainer('invertus.dpdbaltics.on_board.service.on_board_service'); return $onBoardService->makeStepActionWithTemplateReturn(); } } public function getOrderShippingCost($cart, $shippingCost) { return $this->getOrderShippingCostExternal($cart); } /** * @param $cart Cart * @return bool|float|string * @throws PrestaShopDatabaseException * @throws PrestaShopException * @throws SmartyException */ public function getOrderShippingCostExternal($cart) { // This method is still called when module is disabled so we need to do a manual check here if (!$this->active) { return false; } if ($this->context->controller->ajax && Tools::getValue('id_address_delivery')) { $cart->id_address_delivery = (int)Tools::getValue('id_address_delivery'); } /** @var ZoneRepository $zoneRepository */ /** @var ProductRepository $productRepo */ /** @var \Invertus\dpdBaltics\Service\Product\ProductAvailabilityService $productAvailabilityService */ /** @var \Invertus\dpdBaltics\Validate\Weight\CartWeightValidator $cartWeightValidator */ /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ $zoneRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.zone_repository'); $productRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.product_repository'); $productAvailabilityService = $this->getModuleContainer('invertus.dpdbaltics.service.product.product_availability_service'); $cartWeightValidator = $this->getModuleContainer('invertus.dpdbaltics.validate.weight.cart_weight_validator'); $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); $deliveryAddress = new Address($cart->id_address_delivery); if (empty($zoneRepository->findZoneInRangeByAddress($deliveryAddress))) { return false; } $carrier = new Carrier($this->id_carrier); if (!$productAvailabilityService->checkIfCarrierIsAvailable($carrier->id_reference)) { return false; } if (!$productRepo->checkIfCarrierIsAvailableInCountry((int) $carrier->id_reference, (int) $deliveryAddress->id_country) ) { return false; } try { $isCarrierAvailableInShop = $productRepo->checkIfCarrierIsAvailableInShop($carrier->id_reference, $this->context->shop->id); if (empty($isCarrierAvailableInShop)) { return false; } $serviceCarrier = $productRepo->findProductByCarrierReference($carrier->id_reference); } catch (Exception $e) { $tplVars = [ 'errorMessage' => $this->l('Something went wrong while collecting DPD carrier data'), ]; $this->context->smarty->assign($tplVars); return $this->context->smarty->fetch( $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' ); } if ((bool)$serviceCarrier['is_home_collection']) { return false; } $countryCode = $currentCountryProvider->getCurrentCountryIsoCode($cart); $parcelDistribution = \Configuration::get(Config::PARCEL_DISTRIBUTION); $maxAllowedWeight = Config::getDefaultServiceWeights($countryCode, $serviceCarrier['product_reference']); if (!$cartWeightValidator->validate($cart, $parcelDistribution, $maxAllowedWeight)) { return false; } if ($serviceCarrier['product_reference'] === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY) { $isSameDayAvailable = \Invertus\dpdBaltics\Util\ProductUtility::validateSameDayDelivery( $countryCode, $deliveryAddress->city ); if (!$isSameDayAvailable) { return false; } } /** @var ShippingPriceCalculationService $shippingPriceCalculationService */ $shippingPriceCalculationService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.shipping_price_calculation_service'); return $shippingPriceCalculationService->calculate($cart, $carrier, $deliveryAddress); } public function hookDisplayCarrierExtraContent($params) { /** @var Cart $cart */ $cart = $params['cart']; $carrier = new Carrier($params['carrier']['id']); /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); $countryCode = $currentCountryProvider->getCurrentCountryIsoCode($cart); $deliveryAddress = new Address($cart->id_address_delivery); /** @var CarrierPhoneService $carrierPhoneService */ /** @var \Invertus\dpdBaltics\Presenter\DeliveryTimePresenter $deliveryTimePresenter */ /** @var ProductRepository $productRepo */ $carrierPhoneService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.carrier_phone_service'); $deliveryTimePresenter = $this->getModuleContainer()->get('invertus.dpdbaltics.presenter.delivery_time_presenter'); $productRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.product_repository'); $productId = $productRepo->getProductIdByCarrierReference($carrier->id_reference); $dpdProduct = new DPDProduct($productId); $return = ''; if ($dpdProduct->getProductReference() === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY) { /** @var \Invertus\dpdBaltics\Presenter\SameDayDeliveryMessagePresenter $sameDayDeliveryPresenter */ $sameDayDeliveryPresenter = $this->getModuleContainer()->get('invertus.dpdbaltics.presenter.same_day_delivery_message_presenter'); $return .= $sameDayDeliveryPresenter->getSameDayDeliveryMessageTemplate(); } $return .= $carrierPhoneService->getCarrierPhoneTemplate($this->context->cart->id, $carrier->id_reference); if ($dpdProduct->getProductReference() === Config::PRODUCT_TYPE_B2B || $dpdProduct->getProductReference() === Config::PRODUCT_TYPE_B2B_COD ) { $return .= $deliveryTimePresenter->getDeliveryTimeTemplate($countryCode, $deliveryAddress->city); } /** @var ProductRepository $productRep */ $productRep = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); $isPudo = $productRep->isProductPudo($carrier->id_reference); if ($isPudo) { /** @var PudoRepository $pudoRepo */ /** @var ParcelShopRepository $parcelShopRepo */ /** @var ProductRepository $productRepo */ $pudoRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.pudo_repository'); $parcelShopRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.parcel_shop_repository'); $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); $product = $productRepo->findProductByCarrierReference($carrier->id_reference); $isSameDayDelivery = ($product['product_reference'] === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY); $pudoId = $pudoRepo->getIdByCart($cart->id); $selectedPudo = new DPDPudo($pudoId); /** @var ParcelShopService $parcelShopService */ $parcelShopService= $this->getModuleContainer('invertus.dpdbaltics.service.parcel.parcel_shop_service'); $selectedCity = null; $selectedStreet = null; try { if (Validate::isLoadedObject($selectedPudo) && !$isSameDayDelivery) { $selectedCity = $selectedPudo->city; $selectedStreet = $selectedPudo->street; $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity($countryCode, $selectedCity); $parcelShops = $parcelShopService->moveSelectedShopToFirst($parcelShops, $selectedStreet); } else { $selectedCity = $deliveryAddress->city; $selectedStreet = $deliveryAddress->address1; $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity($countryCode, $selectedCity); $parcelShops = $parcelShopService->moveSelectedShopToFirst($parcelShops, $selectedStreet); if (!$parcelShops) { $selectedCity = null; } } } catch (DPDBalticsAPIException $e) { /** @var ExceptionService $exceptionService */ $exceptionService = $this->getModuleContainer('invertus.dpdbaltics.service.exception.exception_service'); $tplVars = [ 'errorMessage' => $exceptionService->getErrorMessageForException( $e, $exceptionService->getAPIErrorMessages() ) ]; $this->context->smarty->assign($tplVars); return $this->context->smarty->fetch( $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' ); } catch (Exception $e) { $tplVars = [ 'errorMessage' => $this->l("Something went wrong. We couldn't find parcel shops."), ]; $this->context->smarty->assign($tplVars); return $this->context->smarty->fetch( $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' ); } /** @var PudoService $pudoService */ $pudoService = $this->getModuleContainer('invertus.dpdbaltics.service.pudo_service'); $pudoServices = $pudoService->setPudoServiceTypes($parcelShops); $pudoServices = $pudoService->formatPudoServicesWorkHours($pudoServices); if (isset($parcelShops[0])) { $coordinates = [ 'lat' => $parcelShops[0]->getLatitude(), 'lng' => $parcelShops[0]->getLongitude(), ]; $this->context->smarty->assign( [ 'coordinates' => $coordinates ] ); } if ($isSameDayDelivery) { $cityList['Rīga'] = 'Rīga'; } else { $cityList = $parcelShopRepo->getAllCitiesByCountryCode($countryCode); } if (!in_array($selectedCity, $cityList) && isset($parcelShops[0])) { $selectedCity = $parcelShops[0]->getCity(); } if (!$selectedCity && Tools::getValue('controller') !== 'order') { $tplVars = [ 'displayMessage' => true, 'messages' => [$this->l("Your delivery address city is not in a list of pickup cities, please select closest pickup point city below manually")], 'messageType_pudo' => 'danger' ]; $this->context->smarty->assign($tplVars); } $streetList = $parcelShopRepo->getAllAddressesByCountryCodeAndCity($countryCode, $selectedCity); $this->context->smarty->assign( [ 'currentController' => Tools::getValue('controller'), 'carrierId' => $carrier->id, 'pickUpMap' => Configuration::get(Config::PICKUP_MAP), 'pudoId' => $pudoId, 'pudoServices' => $pudoServices, 'dpd_pickup_logo' => $this->getPathUri() . 'views/img/pickup.png', 'dpd_locker_logo' => $this->getPathUri() . 'views/img/locker.png', 'delivery_address' => $deliveryAddress, 'saved_pudo_id' => $selectedPudo->pudo_id, 'is_pudo' => (bool)$isPudo, 'city_list' => $cityList, 'selected_city' => $selectedCity, 'show_shop_list' => Configuration::get(Config::PARCEL_SHOP_DISPLAY), 'street_list' => $streetList, 'selected_street' => $selectedStreet, ] ); $return .= $this->context->smarty->fetch( $this->getLocalPath() . '/views/templates/hook/front/pudo-points.tpl' ); } return $return; } /** * Includes Vendor Autoload. */ private function autoLoad() { require_once $this->getLocalPath() . 'vendor/autoload.php'; } private function compile() { $containerBuilder = new ContainerBuilder(); $locator = new FileLocator($this->getLocalPath() . 'config'); $loader = new YamlFileLoader($containerBuilder, $locator); $loader->load('config.yml'); $containerBuilder->compile(); $this->moduleContainer = $containerBuilder; } public function hookActionAdminControllerSetMedia($params) { $currentController = Tools::getValue('controller'); if (Config::isPrestashopVersionBelow174()) { /** @var \Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs $moduleTabs */ $moduleTabs = $this->getService(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::class); $visibleClasses = $moduleTabs->getTabsClassNames(false); if (in_array($currentController, $visibleClasses, true)) { Media::addJsDef(['visibleTabs' => $moduleTabs->getTabsClassNames(true)]); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/tabsHandlerBelowPs174.js'); } } if ('AdminOrders' === $currentController) { $this->handleLabelPrintService(); $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/order-list.css'); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/order_list.js'); Media::addJsDef( [ 'dpdHookAjaxShipmentController' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_AJAX_SHIPMENTS_CONTROLLER), 'shipmentIsBeingPrintedMessage' => $this->context->smarty->fetch($this->getLocalPath() . 'views/templates/admin/partials/spinner.tpl') . $this->l('Your labels are being saved please stay on the page'), 'noOrdersSelectedMessage' => $this->l('No orders were selected'), 'downloadSelectedLabelsButton' => $this->context->smarty->fetch($this->getLocalPath() . 'views/templates/admin/partials/download-selected-labels-button.tpl') ] ); $orderId = Tools::getValue('id_order'); $shipment = $this->getShipment($orderId); $baseUrl = $this->context->shop->getBaseURL(true, false); $isAbove177 = Config::isPrestashopVersionAbove177(); $baseUrlAdmin = $isAbove177 ? $this->context->link->getAdminBaseLink() : null; /** @var \Invertus\dpdBaltics\Service\Label\LabelUrlFormatter $labelUrlService */ $labelUrlService = $isAbove177 ? $this->getModuleContainer('invertus.dpdbaltics.service.label.label_url_formatter') : null; Media::addJsDef( [ 'print_url' => $labelUrlService && $isAbove177 ? $baseUrl.$labelUrlService->formatJsLabelPrintUrl() : null, 'print_and_save_label_url' => $labelUrlService && $isAbove177 ? $baseUrl.$labelUrlService->formatJsLabelSaveAndPrintUrl() : null, 'shipment' => $shipment, 'id_order' => $orderId, 'is_label_download_option' => Configuration::get(Config::LABEL_PRINT_OPTION) === 'download', 'is_ps_above_177' => Config::isPrestashopVersionAbove177(), 'loader_url' => $isAbove177 ? "{$baseUrlAdmin}modules/dpdbaltics/views/templates/admin/loader/loader.html" : null ] ); } if ('AdminOrders' === $currentController && (Tools::isSubmit('vieworder') || Tools::getValue('action') === 'vieworder') ) { $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/order_expand_form.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/shipment.js'); Media::addJsDef([ 'expandText' => $this->l('Expand'), 'collapseText' => $this->l('Collapse'), 'dpdAjaxShipmentsUrl' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_AJAX_SHIPMENTS_CONTROLLER), 'dpdMessages' => [ 'invalidProductQuantity' => $this->l('Invalid product quantity entered'), 'invalidShipment' => $this->l('Invalid shipment selected'), 'parcelsLimitReached' => $this->l('Parcels limit reached in shipment'), 'successProductMove' => $this->l('Product moved successfully'), 'successCreation' => $this->l('Successful creation'), 'unexpectedError' => $this->l('Unexpected error appeared.'), 'invalidPrintoutFormat' => $this->l('Invalid printout format selected.'), 'cannotOpenWindow' => $this->l('Cannot print label, your browser may be blocking it.'), 'dpdRecipientAddressError' => $this->l('Please fill required fields') ], 'id_language' => $this->context->language->id, 'id_shop' => $this->context->shop->id, 'id_cart' => $this->context->cart->id, 'currentController' => $currentController, ]); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/carrier_phone.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/custom_select.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/label_position.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/pudo.js'); $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/customSelect/custom-select.css'); $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/admin-orders-controller.css'); $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/quantity-max-value-tip.css'); } if ('AdminOrders' === $currentController && (Tools::isSubmit('addorder') || Tools::getValue('action') === 'addorder') ) { /** @var ProductRepository $productRepo */ $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); Media::addJsDef([ 'dpdFrontController' => false, 'pudoCarriers' => json_encode($productRepo->getPudoProducts()), 'currentController' => $currentController, 'dpdAjaxShipmentsUrl' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_AJAX_SHIPMENTS_CONTROLLER), 'ignoreAdminController' => true, 'dpdAjaxPudoUrl' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_PUDO_AJAX_CONTROLLER), 'id_shop' => $this->context->shop->id, ]); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/pudo_list.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/carrier_phone.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/pudo.js'); $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/admin-orders-controller.css'); return; } } public function hookDisplayAdminOrder(array $params) { return Config::isPrestashopVersionAbove177() ? false : $this->displayInAdminOrderPage($params); } public function hookDisplayAdminOrderTabContent(array $params) { return !Config::isPrestashopVersionAbove177() ? false : $this->displayInAdminOrderPage($params); } private function displayInAdminOrderPage(array $params) { $order = new Order($params['id_order']); $cart = Cart::getCartByOrderId($params['id_order']); /** @var ProductRepository $productRepo */ $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); $carrier = new Carrier($order->id_carrier); if (!$productRepo->getProductIdByCarrierReference($carrier->id_reference)) { return; } $shipment = $this->getShipment($order->id); $dpdCodWarning = false; /** @var OrderService $orderService */ $orderService = $this->getModuleContainer('invertus.dpdbaltics.service.order_service'); $orderDetails = $orderService->getOrderDetails($order, $cart->id_lang); $customAddresses = []; /** @var ReceiverAddressRepository $receiverAddressRepository */ $receiverAddressRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.receiver_address_repository'); $customOrderAddressesIds = $receiverAddressRepository->getAddressIdByOrderId($order->id); foreach ($customOrderAddressesIds as $customOrderAddressId) { $customAddress = new Address($customOrderAddressId); $customAddresses[] = [ 'id_address' => $customAddress->id, 'alias' => $customAddress->alias ]; } $combinedCustomerAddresses = array_merge($orderDetails['customer']['addresses'], $customAddresses); /** @var PhonePrefixRepository $phonePrefixRepository */ $phonePrefixRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.phone_prefix_repository'); $products = $cart->getProducts(); /** @var LabelPositionService $labelPositionService */ $labelPositionService = $this->getModuleContainer('invertus.dpdbaltics.service.label.label_position_service'); $labelPositionService->assignLabelPositions($shipment->id); $labelPositionService->assignLabelFormat($shipment->id); /** @var PaymentService $paymentService */ $paymentService = $this->getModuleContainer('invertus.dpdbaltics.service.payment.payment_service'); try { $isCodPayment = $paymentService->isOrderPaymentCod($order->module); } catch (Exception $e) { $tplVars = [ 'errorMessage' => $this->l('Something went wrong checking payment method'), ]; $this->context->smarty->assign($tplVars); return $this->context->smarty->fetch( $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' ); } /** @var PudoRepository $pudoRepo */ $pudoRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.pudo_repository'); $selectedProduct = new DPDProduct($shipment->id_service); $isPudo = $selectedProduct->is_pudo; $pudoId = $pudoRepo->getIdByCart($order->id_cart); $selectedPudo = new DPDPudo($pudoId); /** @var ParcelShopService $parcelShopService */ /** @var ParcelShopRepository $parcelShopRepo */ $parcelShopService= $this->getModuleContainer('invertus.dpdbaltics.service.parcel.parcel_shop_service'); $parcelShopRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.parcel_shop_repository'); /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); $countryCode = $currentCountryProvider->getCurrentCountryIsoCode($cart); $selectedCity = null; try { if ($pudoId) { $selectedCity = $selectedPudo->city; $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity( $countryCode, $selectedCity ); } else { $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity( $countryCode, $orderDetails['order_address']['city'] ); } } catch (Exception $e) { $tplVars = [ 'errorMessage' => $this->l('Fatal error while searching for parcel shops: ') . $e->getMessage(), ]; $this->context->smarty->assign($tplVars); return $this->context->smarty->fetch( $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' ); } /** @var null|\Invertus\dpdBalticsApi\Api\DTO\Object\ParcelShop $selectedPudoService */ $selectedPudoService = null; $hasParcelShops = false; if ($parcelShops) { if ($selectedPudo->pudo_id) { $selectedPudoService = $parcelShopService->getParcelShopByShopId($selectedPudo->pudo_id)[0]; } else { $selectedPudoService = $parcelShops[0]; } $hasParcelShops = true; } if ($selectedPudoService) { /** @var PudoService $pudoService */ $pudoService = $this->getModuleContainer('invertus.dpdbaltics.service.pudo_service'); $selectedPudoService->setOpeningHours( $pudoService->formatPudoServiceWorkHours($selectedPudoService->getOpeningHours()) ); } /** @var ProductRepository $productRepository */ $productRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); $dpdProducts = $productRepository->getAllProducts(); $dpdCarrierOptions = []; /** @var DPDProduct $dpdProduct */ foreach ($dpdProducts as $dpdProduct) { $dpdCarrierOptions[] = [ 'id_dpd_product' => $dpdProduct->id_dpd_product, 'name' => $dpdProduct->name, 'available' => $dpdProduct->active && (int) $dpdProduct->is_cod === (int) $isCodPayment && (!$dpdProduct->is_pudo && $hasParcelShops) ]; } $cityList = $parcelShopRepo->getAllCitiesByCountryCode($countryCode); if (\Invertus\dpdBaltics\Config\Config::productHasDeliveryTime($selectedProduct->product_reference)) { /** @var \Invertus\dpdBaltics\Repository\OrderDeliveryTimeRepository $orderDeliveryTimeRepo */ $orderDeliveryTimeRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.order_delivery_time_repository'); $orderDeliveryTimeId = $orderDeliveryTimeRepo->getOrderDeliveryIdByCartId($cart->id); if ($orderDeliveryTimeId) { $orderDeliveryTime = new DPDOrderDeliveryTime($orderDeliveryTimeId); $this->context->smarty->assign([ 'orderDeliveryTime' => $orderDeliveryTime->delivery_time, 'deliveryTimes' => \Invertus\dpdBaltics\Config\Config::getDeliveryTimes($countryCode) ]); } } $href = $this->context->link->getModuleLink( $this->name, 'ShipmentReturn', [ 'id_order' => $order->id, 'dpd-return-submit' => '' ] ); $tplVars = [ 'dpdLogoUrl' => $this->getPathUri() . 'views/img/DPDLogo.gif', 'shipment' => $shipment, 'isAbove177' => Config::isPrestashopVersionAbove177(), 'testOrder' => $shipment->is_test, 'total_products' => 1, 'contractPageLink' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_PRODUCTS_CONTROLLER), 'dpdCodWarning' => $dpdCodWarning, 'testMode' => Configuration::get(Config::SHIPMENT_TEST_MODE), 'printLabelOption' => Configuration::get(Config::LABEL_PRINT_OPTION), 'defaultLabelFormat' => Configuration::get(Config::DEFAULT_LABEL_FORMAT), 'combinedAddresses' => $combinedCustomerAddresses, 'orderDetails' => $orderDetails, 'mobilePhoneCodeList' => $phonePrefixRepository->getCallPrefixes(), 'products' => $products, 'dpdProducts' => $dpdCarrierOptions, 'isCodPayment' => $isCodPayment, 'is_pudo' => (bool)$isPudo, 'selectedPudo' => $selectedPudoService, 'city_list' => $cityList, 'selected_city' => $selectedCity, 'has_parcel_shops' => $hasParcelShops, 'receiverAddressCountries' => Country::getCountries($this->context->language->id, true), 'documentReturnEnabled' => Configuration::get(Config::DOCUMENT_RETURN), 'href' => $href, 'adminLabelLink' => $this->context->link->getAdminLink( 'AdminDPDBalticsAjaxShipments', true, [], ['action' => 'print-return'] ), 'isAutomated' => Configuration::get(Config::AUTOMATED_PARCEL_RETURN), ]; $this->context->smarty->assign($tplVars); return $this->context->smarty->fetch( $this->getLocalPath() . 'views/templates/hook/admin/admin-order.tpl' ); } public function hookActionValidateOrder($params) { $carrier = new Carrier($params['order']->id_carrier); if ($carrier->external_module_name !== $this->name) { return; } $isAdminOrderPage = 'AdminOrders' === Tools::getValue('controller') || Config::isPrestashopVersionAbove177(); $isAdminNewOrderForm = Tools::isSubmit('addorder') || Tools::isSubmit('cart_summary'); if ($isAdminOrderPage && $isAdminNewOrderForm) { $dpdPhone = Tools::getValue('dpd-phone'); $dpdPhoneArea = Tools::getValue('dpd-phone-area'); /** @var \Invertus\dpdBaltics\Service\OrderDeliveryTimeService $orderDeliveryService */ $carrierPhoneService = $this->getModuleContainer('invertus.dpdbaltics.service.carrier_phone_service'); if (!empty($dpdPhone) && !empty($dpdPhoneArea)) { try { $carrierPhoneService->saveCarrierPhone( $this->context->cart->id, $dpdPhone, $dpdPhoneArea ); } catch (Exception $exception) { if ($exception->getCode() === Config::ERROR_COULD_NOT_SAVE_PHONE_NUMBER) { $error = $this->l('Phone data is not saved'); die($error); } } } /** @var CarrierPhoneService $carrierPhoneService */ $orderDeliveryService = $this->getModuleContainer('invertus.dpdbaltics.service.order_delivery_time_service'); $deliveryTime = Tools::getValue('dpd-delivery-time'); if ($deliveryTime !== null) { if (!$orderDeliveryService->saveDeliveryTime( $this->context->cart->id, $deliveryTime )) { $error = $this->l('Delivery time is not saved'); die($error); }; } } /** @var ShipmentService $shipmentService */ $shipmentService = $this->getModuleContainer('invertus.dpdbaltics.service.shipment_service'); $shipmentService->createShipmentFromOrder($params['order']); } public function printLabel($idShipment) { /** @var \Invertus\dpdBaltics\Service\LabelPrintingService $labelPrintingService */ $labelPrintingService = $this->getModuleContainer('invertus.dpdbaltics.service.label_printing_service'); $parcelPrintResponse = $labelPrintingService->printOne($idShipment); if ($parcelPrintResponse->getStatus() === Config::API_SUCCESS_STATUS) { $this->updateOrderCarrier($idShipment); return; } return $parcelPrintResponse; } public function printMultipleLabels($shipmentIds) { /** @var \Invertus\dpdBaltics\Service\LabelPrintingService $labelPrintingService */ $labelPrintingService = $this->getModuleContainer('invertus.dpdbaltics.service.label_printing_service'); $parcelPrintResponse = $labelPrintingService->printMultiple($shipmentIds); if ($parcelPrintResponse->getStatus() === Config::API_SUCCESS_STATUS) { foreach ($shipmentIds as $shipmentId) { $this->updateOrderCarrier($shipmentId); } return; } return $parcelPrintResponse; } private function updateOrderCarrier($shipmentId) { $shipment = new DPDShipment($shipmentId); /** @var OrderRepository $orderRepo */ /** @var TrackingService $trackingService */ $orderRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.order_repository'); $trackingService = $this->getModuleContainer('invertus.dpdbaltics.service.tracking_service'); $orderCarrierId = $orderRepo->getOrderCarrierId($shipment->id_order); $orderCarrier = new OrderCarrier($orderCarrierId); $orderCarrier->tracking_number = $trackingService->getTrackingNumber($shipment->pl_number); try { $orderCarrier->update(); } catch (Exception $e) { Context::getContext()->controller->errors[] = $this->l('Failed to save tracking number: ') . $e->getMessage(); return; } $shipment->printed_label = 1; $shipment->date_print = date('Y-m-d H:i:s'); $shipment->update(); } public function hookDisplayOrderDetail($params) { $isReturnServiceEnabled = Configuration::get(Config::PARCEL_RETURN); if (!$isReturnServiceEnabled) { return; } if (CountryUtility::isEstonia()) { return; } /** @var ShipmentRepository $shipmentRepo */ $shipmentRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.shipment_repository'); $shipmentId = $shipmentRepo->getIdByOrderId($params['order']->id); $orderState = new OrderState($params['order']->current_state); if (!$orderState->delivery) { return; } /** @var AddressTemplateRepository $addressTemplateRepo */ $addressTemplateRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.address_template_repository'); $returnAddressTemplates = $addressTemplateRepo->getReturnServiceAddressTemplates(); $shipment = new DPDShipment($shipmentId); if (!$returnAddressTemplates) { return; } $showTemplates = false; if (sizeof($returnAddressTemplates) > 1 && !$shipment->return_pl_number) { $showTemplates = true; } if (isset($this->context->cookie->dpd_error)) { $this->context->controller->errors[] = json_decode($this->context->cookie->dpd_error); unset($this->context->cookie->dpd_error); } $href = $this->context->link->getModuleLink( $this->name, 'ShipmentReturn', [ 'id_order' => $params['order']->id, 'dpd-return-submit' => '' ] ); $this->context->smarty->assign( [ 'href' => $href, 'return_template_ids' => $returnAddressTemplates, 'show_template' => $showTemplates, ] ); $html = $this->context->smarty->fetch( $this->getLocalPath() . 'views/templates/hook/front/order-detail.tpl' ); return $html; } public function hookActionAdminOrdersListingFieldsModifier($params) { if ((bool) Configuration::get(Config::HIDE_ORDERS_LABEL_PRINT_BUTTON)) { return false; } if (isset($params['select'])) { $params['select'] .= ' ,ds.`id_order` AS id_order_shipment '; } if (isset($params['join'])) { $params['join'] .= ' LEFT JOIN `' . _DB_PREFIX_ . 'dpd_shipment` ds ON ds.`id_order` = a.`id_order` '; } $params['fields']['id_order_shipment'] = [ 'title' => $this->l('DPD Label'), 'align' => 'text-center', 'class' => 'fixed-width-xs', 'orderby' => false, 'search' => false, 'remove_onclick' => true, 'callback_object' => 'dpdbaltics', 'callback' => 'returnOrderListIcon' ]; } /** * Callback function, it has to be static so can't call $this, so have to reload dpdBaltics module inside the function * @param $idOrder * @return string * @throws Exception */ public static function returnOrderListIcon($orderId) { $dpdBaltics = Module::getInstanceByName('dpdbaltics'); $dpdBaltics->context->smarty->assign('idOrder', $orderId); $dpdBaltics->context->smarty->assign( 'message', $dpdBaltics->l('Print label(s) from DPD system. Once label is saved you won\'t be able to modify contents of shipments') ); $icon = $dpdBaltics->context->smarty->fetch( $dpdBaltics->getLocalPath() . 'views/templates/hook/admin/order-list-save-label-icon.tpl' ); $dpdBaltics->context->smarty->assign('icon', $icon); return $dpdBaltics->context->smarty->fetch($dpdBaltics->getLocalPath() . 'views/templates/hook/admin/order-list-icon-container.tpl'); } public function hookDisplayAdminListBefore() { if ($this->context->controller instanceof AdminOrdersControllerCore) { return $this->context->smarty->fetch($this->getLocalPath() . 'views/templates/hook/admin/admin-orders-header-hook.tpl'); } } public function hookActionOrderGridDefinitionModifier(array $params) { if (!Config::isPrestashopVersionAbove177()) { return false; } $definition = $params['definition']; if (!(bool) Configuration::get(Config::HIDE_ORDERS_LABEL_PRINT_BUTTON)) { $definition->getColumns() ->addAfter( 'date_add', (new ActionColumn('dpd_print_label')) ->setName($this->l('Dpd print label')) ->setOptions([ 'actions' => $this->getGridAction() ]) ); } $definition->getBulkActions() ->add( (new SubmitBulkActionCustom('print_multiple_labels')) ->setName($this->l('Print multiple labels')) ->setOptions([ 'submit_route' => 'dpdbaltics_save_and_download_printed_labels_order_list_multiple', ]) ) ; } /** * @return RowActionCollection */ private function getGridAction() { return (new RowActionCollection()) ->add( (new LinkRowActionCustom('print_delivery_slip')) ->setName($this->l('Print label(s) from DPD system. Once label is saved you won\'t be able to modify contents of shipments')) ->setIcon('print') ->setOptions([ 'route' => 'dpdbaltics_save_and_download_printed_label_order_list', 'route_param_name' => 'orderId', 'route_param_field' => 'id_order', 'is_label_download' => Configuration::get(Config::LABEL_PRINT_OPTION) === 'download', 'confirm_message' => $this->l('Would you like to print shipping label?'), 'accessibility_checker' => $this->getModuleContainer()->get('invertus.dpdbaltics.grid.row.print_accessibility_checker'), ]) ); } private function getShipment($idOrder) { if (!$idOrder) { return false; } /** @var ShipmentRepository $shipmentRepository */ $shipmentRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.shipment_repository'); $shipmentId = $shipmentRepository->getIdByOrderId($idOrder); $shipment = new DPDShipment($shipmentId); if (!Validate::isLoadedObject($shipment)) { return false; } return $shipment; } private function handleLabelPrintService() { if (Tools::isSubmit('print_label')) { $idShipment = Tools::getValue('id_dpd_shipment'); try { $parcelPrintResponse = $this->printLabel($idShipment); } catch (DPDBalticsAPIException $e) { /** @var ExceptionService $exceptionService */ $exceptionService = $this->getModuleContainer('invertus.dpdbaltics.service.exception.exception_service'); Context::getContext()->controller->errors[] = $exceptionService->getErrorMessageForException( $e, $exceptionService->getAPIErrorMessages() ); return; } catch (Exception $e) { Context::getContext()->controller->errors[] = $this->l('Failed to print label: ') . $e->getMessage(); return; } if (!empty($parcelPrintResponse->getErrLog())) { Context::getContext()->controller->errors[] = $parcelPrintResponse->getErrLog(); } exit; } if (Tools::isSubmit('print_multiple_labels')) { $shipmentIds = json_decode(Tools::getValue('shipment_ids')); try { $parcelPrintResponse = $this->printMultipleLabels($shipmentIds); } catch (DPDBalticsAPIException $e) { /** @var ExceptionService $exceptionService */ $exceptionService = $this->getModuleContainer('invertus.dpdbaltics.service.exception.exception_service'); Context::getContext()->controller->errors[] = $exceptionService->getErrorMessageForException( $e, $exceptionService->getAPIErrorMessages() ); return; } catch (Exception $e) { Context::getContext()->controller->errors[] = $this->l('Failed to print label: ') . $e->getMessage(); return; } if (!empty($parcelPrintResponse->getErrLog())) { Context::getContext()->controller->errors[] = $parcelPrintResponse->getErrLog(); } exit; } } } \ No newline at end of file diff --git a/views/css/front/onepagecheckout.css b/views/css/front/onepagecheckout.css index 86f6f2f0..16b23208 100644 --- a/views/css/front/onepagecheckout.css +++ b/views/css/front/onepagecheckout.css @@ -50,7 +50,7 @@ } .chosen-container { - width: 100%; + width: 100% !important; } .carrier-extra-content .chosen-select { diff --git a/views/js/front/order-opc.js b/views/js/front/order-opc.js index fe3065d3..ceb26f2e 100644 --- a/views/js/front/order-opc.js +++ b/views/js/front/order-opc.js @@ -17,6 +17,12 @@ * @license https://opensource.org/licenses/AFL-3.0 Academic Free License version 3.0 */ var isPudoPointSelected = false; +var phoneQuerySelectors = [ + '#checkout #delivery-address [name="phone"]', + '.address-fields input[name="phone"]', + '#checkout [name="delivery_phone_mobile"]' +]; + $(document).ready(function (){ $(document).on('change', '.dpd-phone-block', function() { @@ -51,9 +57,19 @@ function handlePhoneNumber(selector) if (!$('.dpd-phone-block')) { return true; } + var phone = selector.find('input[name="dpd-phone"]').val(); var phoneArea = selector.find('select[name="dpd-phone-area"] option:selected').val(); + // if empty then take it from address-fields input[name="phone"] + if (phone === '') { + phoneQuerySelectors.forEach(function (phoneSelector) { + if ($(phoneSelector).val()) { + phone = $(phoneSelector).val(); + } + }); + } + if (currentController === 'supercheckout') { saveSelectedPhoneNumberSuperCheckout(phone, phoneArea) } else { @@ -99,4 +115,7 @@ $(document).on('opc-load-review:completed', function() { handlePhoneNumber($('.dpd-phone-block')); }); - +// Then address is being modified +$(document).on('thecheckout_Address_Modified', function () { + handlePhoneNumber($('.dpd-phone-block')); +}); \ No newline at end of file From 453f0c6dcb6188837ad615bb89fcda53091870f8 Mon Sep 17 00:00:00 2001 From: Gytautas Date: Thu, 4 Jul 2024 11:51:19 +0300 Subject: [PATCH 13/20] comment edited --- views/js/front/order-opc.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/views/js/front/order-opc.js b/views/js/front/order-opc.js index ceb26f2e..974464bf 100644 --- a/views/js/front/order-opc.js +++ b/views/js/front/order-opc.js @@ -61,7 +61,7 @@ function handlePhoneNumber(selector) var phone = selector.find('input[name="dpd-phone"]').val(); var phoneArea = selector.find('select[name="dpd-phone-area"] option:selected').val(); - // if empty then take it from address-fields input[name="phone"] + // if empty then take it from opc phone field if (phone === '') { phoneQuerySelectors.forEach(function (phoneSelector) { if ($(phoneSelector).val()) { From f8b3463faad96836923e0e341f998df0d1a9ef3a Mon Sep 17 00:00:00 2001 From: Gytautas Date: Fri, 5 Jul 2024 09:44:53 +0300 Subject: [PATCH 14/20] fixed onepagecheckout ps dropdown --- dpdbaltics.php | 1417 ++++++++++++++++++++++++++++++++++- views/js/front/order-opc.js | 3 +- views/js/front/pudo.js | 36 +- 3 files changed, 1435 insertions(+), 21 deletions(-) diff --git a/dpdbaltics.php b/dpdbaltics.php index 7b3be301..d7a01429 100644 --- a/dpdbaltics.php +++ b/dpdbaltics.php @@ -1 +1,1416 @@ - * @copyright Since 2007 PrestaShop SA and Contributors * @license https://opensource.org/licenses/AFL-3.0 Academic Free License version 3.0 */ use Invertus\dpdBaltics\Grid\Row\PrintAccessibilityChecker; use Invertus\dpdBaltics\Builder\Template\Front\CarrierOptionsBuilder; use Invertus\dpdBaltics\Config\Config; use Invertus\dpdBaltics\Controller\AbstractAdminController; use Invertus\dpdBaltics\Grid\LinkRowActionCustom; use Invertus\dpdBaltics\Grid\SubmitBulkActionCustom; use invertus\dpdbaltics\install\installer; use Invertus\dpdBaltics\Logger\Logger; use Invertus\dpdBaltics\OnBoard\Service\OnBoardService; use Invertus\dpdBaltics\Repository\AddressTemplateRepository; use Invertus\dpdBaltics\Repository\OrderRepository; use Invertus\dpdBaltics\Repository\ParcelShopRepository; use Invertus\dpdBaltics\Repository\PhonePrefixRepository; use Invertus\dpdBaltics\Repository\PriceRuleRepository; use Invertus\dpdBaltics\Repository\ProductRepository; use Invertus\dpdBaltics\Repository\PudoRepository; use Invertus\dpdBaltics\Repository\ReceiverAddressRepository; use Invertus\dpdBaltics\Repository\ShipmentRepository; use Invertus\dpdBaltics\Repository\ZoneRepository; use Invertus\dpdBaltics\Service\API\LabelApiService; use Invertus\dpdBaltics\Service\API\ParcelShopSearchApiService; use Invertus\dpdBaltics\Service\CarrierPhoneService; use Invertus\dpdBaltics\Service\Exception\ExceptionService; use Invertus\dpdBaltics\Service\GoogleApiService; use Invertus\dpdBaltics\Service\Label\LabelPositionService; use Invertus\dpdBaltics\Service\OrderService; use Invertus\dpdBaltics\Service\Parcel\ParcelShopService; use Invertus\dpdBaltics\Service\Payment\PaymentService; use Invertus\dpdBaltics\Service\PriceRuleService; use Invertus\dpdBaltics\Service\PudoService; use Invertus\dpdBaltics\Service\ShipmentService; use Invertus\dpdBaltics\Service\ShippingPriceCalculationService; use Invertus\dpdBaltics\Service\TabService; use Invertus\dpdBaltics\Service\TrackingService; use Invertus\dpdBaltics\Util\CountryUtility; use Invertus\dpdBaltics\Validate\Carrier\PudoValidate; use Invertus\dpdBalticsApi\Api\DTO\Response\ParcelPrintResponse; use Invertus\dpdBalticsApi\Api\DTO\Response\ParcelShopSearchResponse; use Invertus\dpdBalticsApi\Exception\DPDBalticsAPIException; use Invertus\dpdBalticsApi\Factory\SerializerFactory; use PrestaShop\PrestaShop\Core\Grid\Action\Bulk\Type\SubmitBulkAction; use PrestaShop\PrestaShop\Core\Grid\Action\Row\RowActionCollection; use PrestaShop\PrestaShop\Core\Grid\Action\Row\Type\LinkRowAction; use PrestaShop\PrestaShop\Core\Grid\Column\Type\Common\ActionColumn; use Symfony\Component\Config\ConfigCache; use Symfony\Component\Config\FileLocator; use Symfony\Component\Console\Application; use Symfony\Component\DependencyInjection\ContainerBuilder; use Symfony\Component\DependencyInjection\Dumper\PhpDumper; use Symfony\Component\DependencyInjection\Loader\YamlFileLoader; class DPDBaltics extends CarrierModule { /** * Symfony DI Container **/ private $moduleContainer; /** * Prestashop fills this property automatically with selected carrier ID in FO checkout * * @var int $id_carrier */ public $id_carrier; public function __construct() { $this->name = 'dpdbaltics'; $this->displayName = $this->l('DPDBaltics'); $this->author = 'Invertus'; $this->tab = 'shipping_logistics'; $this->description = 'DPD Baltics shipping integration'; $this->version = '3.2.19'; $this->ps_versions_compliancy = ['min' => '1.7.1.0', 'max' => _PS_VERSION_]; $this->need_instance = 0; parent::__construct(); $this->autoLoad(); $this->compile(); } public function install() { if (!parent::install()) { return false; } /** @var \Invertus\dpdBaltics\Infrastructure\Bootstrap\Install\Installer $installer */ $installer = $this->getService(\Invertus\dpdBaltics\Infrastructure\Bootstrap\Install\Installer::class); try { $installer->init(); return true; } catch (\Invertus\dpdBaltics\Infrastructure\Bootstrap\Exception\CouldNotInstallModule $exception) { $this->_errors[] = $exception->getMessage(); return false; } } public function uninstall() { $uninstall = parent::uninstall(); if (!$uninstall) { return false; } /** @var \Invertus\dpdBaltics\Infrastructure\Bootstrap\Uninstall\Uninstaller $uninstaller */ $uninstaller = $this->getService(\Invertus\dpdBaltics\Infrastructure\Bootstrap\Uninstall\Uninstaller::class); try { $uninstaller->init(); return true; } catch (\Invertus\dpdBaltics\Infrastructure\Bootstrap\Exception\CouldNotUninstallModule $exception) { $this->_errors[] = $exception->getMessage(); return false; } } public function getContent() { Tools::redirectAdmin($this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_SETTINGS_CONTROLLER)); } public function getService($serviceName) { return $this->getModuleContainer()->get($serviceName); } /** * @return mixed */ public function getModuleContainer($id = false) { if ($id) { return $this->moduleContainer->get($id); } return $this->moduleContainer; } public function hookActionFrontControllerSetMedia() { //TODO fillup this array when more modules are compatible with OPC $onePageCheckoutControllers = ['supercheckout', 'onepagecheckoutps', 'thecheckout']; $applicableControlelrs = ['order', 'order-opc', 'ShipmentReturn', 'supercheckout']; $currentController = !empty($this->context->controller->php_self) ? $this->context->controller->php_self : Tools::getValue('controller'); if ('product' === $currentController) { $this->context->controller->registerStylesheet( 'dpdbaltics-product-carriers.', 'modules/' . $this->name . '/views/css/front/product-carriers.css', [ 'media' => 'all', 'position' => 150 ] ); } /** @var \Invertus\dpdBaltics\Validate\Compatibility\OpcModuleCompatibilityValidator $opcModuleCompatibilityValidator */ $opcModuleCompatibilityValidator = $this->getModuleContainer('invertus.dpdbaltics.validator.opc_module_compatibility_validator'); if (in_array($currentController, $onePageCheckoutControllers, true) || $opcModuleCompatibilityValidator->isOpcModuleInUse()) { $this->context->controller->addJqueryPlugin('chosen'); $this->context->controller->registerJavascript( 'dpdbaltics-opc', 'modules/' . $this->name . '/views/js/front/order-opc.js', [ 'position' => 'bottom', 'priority' => 130 ] ); $this->context->controller->registerJavascript( 'dpdbaltics-supercheckout', 'modules/' . $this->name . '/views/js/front/modules/supercheckout.js', [ 'position' => 'bottom', 'priority' => 130 ] ); $this->context->controller->registerStylesheet( 'dpdbaltics-opc', 'modules/' . $this->name . '/views/css/front/onepagecheckout.css', [ 'position' => 'bottom', 'priority' => 130 ] ); } /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); $webServiceCountryCode = Configuration::get(Config::WEB_SERVICE_COUNTRY); $carrierIds = []; $baseUrl = $this->context->shop->getBaseURL(true, false); if ($webServiceCountryCode === Config::LATVIA_ISO_CODE || $currentController === 'supercheckout') { /** @var ProductRepository $productRepo */ $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); $dpdProductReferences = $productRepo->getAllActiveDpdProductReferences(); foreach ($dpdProductReferences as $reference) { $carrier = Carrier::getCarrierByReference($reference['id_reference']); if (Validate::isLoadedObject($carrier)) { $carrierIds[] = $carrier->id; } } } Media::addJsDef([ 'lapinas_img' => $baseUrl . $this->getPathUri() . 'views/img/lapinas.png', 'lapinas_text' => $this->l('Sustainable'), 'dpd_carrier_ids' => $carrierIds ]); if (in_array($currentController, $applicableControlelrs, true)) { Media::addJsDef([ 'select_an_option_translatable' => $this->l('Select an Option'), 'select_an_option_multiple_translatable' => $this->l('Select Some Options'), 'no_results_translatable' => $this->l('No results match'), ]); $this->context->controller->addJS($this->getPathUri() . 'views/js/front/order.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/front/order-input.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/front/sustainable-logo.js'); $this->context->controller->addCSS($this->getPathUri() . 'views/css/front/order-input.css'); $this->context->controller->addCSS($this->getPathUri() . 'views/css/front/sustainable-logo.css'); /** @var PaymentService $paymentService */ $paymentService = $this->getModuleContainer('invertus.dpdbaltics.service.payment.payment_service'); $isPickupMap = Configuration::get(\Invertus\dpdBaltics\Config\Config::PICKUP_MAP); $cart = Context::getContext()->cart; $paymentService->filterPaymentMethods($cart); $paymentService->filterPaymentMethodsByCod($cart); /** @var ProductRepository $productRepo */ $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); Media::addJsDef([ 'pudoCarriers' => json_encode($productRepo->getPudoProducts()), 'currentController' => $currentController, 'is_pickup_map' => $isPickupMap, 'id_language' => $this->context->language->id, 'id_shop' => $this->context->shop->id, 'dpdAjaxLoaderPath' => $this->getPathUri() . 'views/img/ajax-loader-big.gif', 'dpdPickupMarkerPath' => $this->getPathUri() . 'views/img/dpd-pick-up.png', 'dpdLockerMarkerPath' => $this->getPathUri() . 'views/img/locker.png', 'dpdHookAjaxUrl' => $this->context->link->getModuleLink($this->name, 'Ajax'), 'pudoSelectSuccess' => $this->l('Pick-up point selected'), 'dpd_carrier_ids' => $carrierIds ]); $this->context->controller->registerStylesheet( 'dpdbaltics-pudo-shipment', 'modules/' . $this->name . '/views/css/front/' . 'pudo-shipment.css', [ 'media' => 'all', 'position' => 150 ] ); if ($isPickupMap) { /** @var GoogleApiService $googleApiService */ $googleApiService = $this->getModuleContainer('invertus.dpdbaltics.service.google_api_service'); $this->context->controller->registerJavascript( 'dpdbaltics-google-api', $googleApiService->getFormattedGoogleMapsUrl(), [ 'server' => 'remote' ] ); } $this->context->controller->registerJavascript( 'dpdbaltics-pudo', 'modules/' . $this->name . '/views/js/front/pudo.js', [ 'position' => 'bottom', 'priority' => 130 ] ); $this->context->controller->registerJavascript( 'dpdbaltics-pudo-search', 'modules/' . $this->name . '/views/js/front/pudo-search.js', [ 'position' => 'bottom', 'priority' => 130 ] ); } } public function hookActionValidateStepComplete(&$params) { if ('delivery' !== $params['step_name']) { return; } /** @var Cart $cart */ $cart = $params['cart']; $carrier = new Carrier($cart->id_carrier); $idShop = $this->context->shop->id; /** @var Invertus\dpdBaltics\Repository\CarrierRepository $carrierRepo */ /** @var Invertus\dpdBaltics\Repository\ProductRepository $productRepo */ $carrierRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.carrier_repository'); $productRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.product_repository'); $carrierReference = $carrier->id_reference; $dpdCarriers = $carrierRepo->getDpdCarriers($idShop); $isDpdCarrier = false; foreach ($dpdCarriers as $dpdCarrier) { if ($carrierReference == $dpdCarrier['id_reference']) { $isDpdCarrier = true; $productId = $productRepo->getProductIdByCarrierReference($carrier->id_reference); $product = new DPDProduct($productId); $isSameDayDelivery = $product->product_reference === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY; break; } } if (!$isDpdCarrier) { return true; } if ($isSameDayDelivery) { /** @var PudoRepository $pudoRepo */ $pudoRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.pudo_repository'); $pudoId = $pudoRepo->getIdByCart($cart->id); $selectedPudo = new DPDPudo($pudoId); if ($selectedPudo->city !== Config::SAME_DAY_DELIVERY_CITY) { $this->context->controller->errors[] = $this->l('This carrier can\'t deliver to your selected city'); $params['completed'] = false; $selectedPudo->delete(); return; } } //NOTE: thecheckout triggers this hook without phone parameters the phone is saved with ajax request if (Tools::getValue('module') === 'thecheckout') { return; } if (!Tools::getValue('dpd-phone')) { $this->context->controller->errors[] = $this->l('In order to use DPD Carrier you need to enter phone number'); $params['completed'] = false; return; } if (!Tools::getValue('dpd-phone-area')) { $this->context->controller->errors[] = $this->l('In order to use DPD Carrier you need to enter phone area'); $params['completed'] = false; return; } /** @var CarrierPhoneService $carrierPhoneService */ $carrierPhoneService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.carrier_phone_service'); try { $carrierPhoneService->saveCarrierPhone( $this->context->cart->id, Tools::getValue('dpd-phone'), Tools::getValue('dpd-phone-area') ); } catch (Exception $exception) { if ($exception->getCode() === Config::ERROR_COULD_NOT_SAVE_PHONE_NUMBER) { $this->context->controller->errors[] = $this->l('Phone data is not saved'); $params['completed'] = false; } } /** @var \Invertus\dpdBaltics\Service\OrderDeliveryTimeService $orderDeliveryService */ $orderDeliveryService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.order_delivery_time_service'); $deliveryTime = Tools::getValue('dpd-delivery-time'); if ($deliveryTime) { if (!$orderDeliveryService->saveDeliveryTime( $this->context->cart->id, $deliveryTime )) { $this->context->controller->errors[] = $this->l('Delivery time data is not saved'); $params['completed'] = false; }; } /** @var Cart $cart */ $cart = $params['cart']; $carrier = new Carrier($cart->id_carrier); /** @var PudoValidate $pudoValidator */ $pudoValidator = $this->getModuleContainer('invertus.dpdbaltics.validate.carrier.pudo_validate'); if (!$pudoValidator->validatePickupPoints($cart->id, $carrier->id_reference)) { $carrier = new Carrier($cart->id_carrier, $this->context->language->id); $this->context->controller->errors[] = sprintf($this->l('Please select pickup point for carrier: %s.'), $carrier->name); $params['completed'] = false; } } public function hookDisplayProductPriceBlock($params) { if ($params['type'] == 'after_price') { /** @var CarrierOptionsBuilder $carrierOptionsBuilder */ $carrierOptionsBuilder = $this->getModuleContainer()->get('invertus.dpdbaltics.builder.template.front.carrier_options_builder'); return $carrierOptionsBuilder->renderCarrierOptionsInProductPage(); } } public function hookDisplayBackOfficeTop() { if ($this->context->controller instanceof AbstractAdminController && Configuration::get(Config::ON_BOARD_TURNED_ON) && Configuration::get(Config::ON_BOARD_STEP) ) { /** @var OnBoardService $onBoardService */ $onBoardService = $this->getModuleContainer('invertus.dpdbaltics.on_board.service.on_board_service'); return $onBoardService->makeStepActionWithTemplateReturn(); } } public function getOrderShippingCost($cart, $shippingCost) { return $this->getOrderShippingCostExternal($cart); } /** * @param $cart Cart * @return bool|float|string * @throws PrestaShopDatabaseException * @throws PrestaShopException * @throws SmartyException */ public function getOrderShippingCostExternal($cart) { // This method is still called when module is disabled so we need to do a manual check here if (!$this->active) { return false; } if ($this->context->controller->ajax && Tools::getValue('id_address_delivery')) { $cart->id_address_delivery = (int)Tools::getValue('id_address_delivery'); } /** @var ZoneRepository $zoneRepository */ /** @var ProductRepository $productRepo */ /** @var \Invertus\dpdBaltics\Service\Product\ProductAvailabilityService $productAvailabilityService */ /** @var \Invertus\dpdBaltics\Validate\Weight\CartWeightValidator $cartWeightValidator */ /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ $zoneRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.zone_repository'); $productRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.product_repository'); $productAvailabilityService = $this->getModuleContainer('invertus.dpdbaltics.service.product.product_availability_service'); $cartWeightValidator = $this->getModuleContainer('invertus.dpdbaltics.validate.weight.cart_weight_validator'); $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); $deliveryAddress = new Address($cart->id_address_delivery); if (empty($zoneRepository->findZoneInRangeByAddress($deliveryAddress))) { return false; } $carrier = new Carrier($this->id_carrier); if (!$productAvailabilityService->checkIfCarrierIsAvailable($carrier->id_reference)) { return false; } if (!$productRepo->checkIfCarrierIsAvailableInCountry((int) $carrier->id_reference, (int) $deliveryAddress->id_country) ) { return false; } try { $isCarrierAvailableInShop = $productRepo->checkIfCarrierIsAvailableInShop($carrier->id_reference, $this->context->shop->id); if (empty($isCarrierAvailableInShop)) { return false; } $serviceCarrier = $productRepo->findProductByCarrierReference($carrier->id_reference); } catch (Exception $e) { $tplVars = [ 'errorMessage' => $this->l('Something went wrong while collecting DPD carrier data'), ]; $this->context->smarty->assign($tplVars); return $this->context->smarty->fetch( $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' ); } if ((bool)$serviceCarrier['is_home_collection']) { return false; } $countryCode = $currentCountryProvider->getCurrentCountryIsoCode($cart); $parcelDistribution = \Configuration::get(Config::PARCEL_DISTRIBUTION); $maxAllowedWeight = Config::getDefaultServiceWeights($countryCode, $serviceCarrier['product_reference']); if (!$cartWeightValidator->validate($cart, $parcelDistribution, $maxAllowedWeight)) { return false; } if ($serviceCarrier['product_reference'] === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY) { $isSameDayAvailable = \Invertus\dpdBaltics\Util\ProductUtility::validateSameDayDelivery( $countryCode, $deliveryAddress->city ); if (!$isSameDayAvailable) { return false; } } /** @var ShippingPriceCalculationService $shippingPriceCalculationService */ $shippingPriceCalculationService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.shipping_price_calculation_service'); return $shippingPriceCalculationService->calculate($cart, $carrier, $deliveryAddress); } public function hookDisplayCarrierExtraContent($params) { /** @var Cart $cart */ $cart = $params['cart']; $carrier = new Carrier($params['carrier']['id']); /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); $countryCode = $currentCountryProvider->getCurrentCountryIsoCode($cart); $deliveryAddress = new Address($cart->id_address_delivery); /** @var CarrierPhoneService $carrierPhoneService */ /** @var \Invertus\dpdBaltics\Presenter\DeliveryTimePresenter $deliveryTimePresenter */ /** @var ProductRepository $productRepo */ $carrierPhoneService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.carrier_phone_service'); $deliveryTimePresenter = $this->getModuleContainer()->get('invertus.dpdbaltics.presenter.delivery_time_presenter'); $productRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.product_repository'); $productId = $productRepo->getProductIdByCarrierReference($carrier->id_reference); $dpdProduct = new DPDProduct($productId); $return = ''; if ($dpdProduct->getProductReference() === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY) { /** @var \Invertus\dpdBaltics\Presenter\SameDayDeliveryMessagePresenter $sameDayDeliveryPresenter */ $sameDayDeliveryPresenter = $this->getModuleContainer()->get('invertus.dpdbaltics.presenter.same_day_delivery_message_presenter'); $return .= $sameDayDeliveryPresenter->getSameDayDeliveryMessageTemplate(); } $return .= $carrierPhoneService->getCarrierPhoneTemplate($this->context->cart->id, $carrier->id_reference); if ($dpdProduct->getProductReference() === Config::PRODUCT_TYPE_B2B || $dpdProduct->getProductReference() === Config::PRODUCT_TYPE_B2B_COD ) { $return .= $deliveryTimePresenter->getDeliveryTimeTemplate($countryCode, $deliveryAddress->city); } /** @var ProductRepository $productRep */ $productRep = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); $isPudo = $productRep->isProductPudo($carrier->id_reference); if ($isPudo) { /** @var PudoRepository $pudoRepo */ /** @var ParcelShopRepository $parcelShopRepo */ /** @var ProductRepository $productRepo */ $pudoRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.pudo_repository'); $parcelShopRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.parcel_shop_repository'); $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); $product = $productRepo->findProductByCarrierReference($carrier->id_reference); $isSameDayDelivery = ($product['product_reference'] === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY); $pudoId = $pudoRepo->getIdByCart($cart->id); $selectedPudo = new DPDPudo($pudoId); /** @var ParcelShopService $parcelShopService */ $parcelShopService= $this->getModuleContainer('invertus.dpdbaltics.service.parcel.parcel_shop_service'); $selectedCity = null; $selectedStreet = null; try { if (Validate::isLoadedObject($selectedPudo) && !$isSameDayDelivery) { $selectedCity = $selectedPudo->city; $selectedStreet = $selectedPudo->street; $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity($countryCode, $selectedCity); $parcelShops = $parcelShopService->moveSelectedShopToFirst($parcelShops, $selectedStreet); } else { $selectedCity = $deliveryAddress->city; $selectedStreet = $deliveryAddress->address1; $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity($countryCode, $selectedCity); $parcelShops = $parcelShopService->moveSelectedShopToFirst($parcelShops, $selectedStreet); if (!$parcelShops) { $selectedCity = null; } } } catch (DPDBalticsAPIException $e) { /** @var ExceptionService $exceptionService */ $exceptionService = $this->getModuleContainer('invertus.dpdbaltics.service.exception.exception_service'); $tplVars = [ 'errorMessage' => $exceptionService->getErrorMessageForException( $e, $exceptionService->getAPIErrorMessages() ) ]; $this->context->smarty->assign($tplVars); return $this->context->smarty->fetch( $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' ); } catch (Exception $e) { $tplVars = [ 'errorMessage' => $this->l("Something went wrong. We couldn't find parcel shops."), ]; $this->context->smarty->assign($tplVars); return $this->context->smarty->fetch( $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' ); } /** @var PudoService $pudoService */ $pudoService = $this->getModuleContainer('invertus.dpdbaltics.service.pudo_service'); $pudoServices = $pudoService->setPudoServiceTypes($parcelShops); $pudoServices = $pudoService->formatPudoServicesWorkHours($pudoServices); if (isset($parcelShops[0])) { $coordinates = [ 'lat' => $parcelShops[0]->getLatitude(), 'lng' => $parcelShops[0]->getLongitude(), ]; $this->context->smarty->assign( [ 'coordinates' => $coordinates ] ); } if ($isSameDayDelivery) { $cityList['Rīga'] = 'Rīga'; } else { $cityList = $parcelShopRepo->getAllCitiesByCountryCode($countryCode); } if (!in_array($selectedCity, $cityList) && isset($parcelShops[0])) { $selectedCity = $parcelShops[0]->getCity(); } if (!$selectedCity && Tools::getValue('controller') !== 'order') { $tplVars = [ 'displayMessage' => true, 'messages' => [$this->l("Your delivery address city is not in a list of pickup cities, please select closest pickup point city below manually")], 'messageType_pudo' => 'danger' ]; $this->context->smarty->assign($tplVars); } $streetList = $parcelShopRepo->getAllAddressesByCountryCodeAndCity($countryCode, $selectedCity); $this->context->smarty->assign( [ 'currentController' => Tools::getValue('controller'), 'carrierId' => $carrier->id, 'pickUpMap' => Configuration::get(Config::PICKUP_MAP), 'pudoId' => $pudoId, 'pudoServices' => $pudoServices, 'dpd_pickup_logo' => $this->getPathUri() . 'views/img/pickup.png', 'dpd_locker_logo' => $this->getPathUri() . 'views/img/locker.png', 'delivery_address' => $deliveryAddress, 'saved_pudo_id' => $selectedPudo->pudo_id, 'is_pudo' => (bool)$isPudo, 'city_list' => $cityList, 'selected_city' => $selectedCity, 'show_shop_list' => Configuration::get(Config::PARCEL_SHOP_DISPLAY), 'street_list' => $streetList, 'selected_street' => $selectedStreet, ] ); $return .= $this->context->smarty->fetch( $this->getLocalPath() . '/views/templates/hook/front/pudo-points.tpl' ); } return $return; } /** * Includes Vendor Autoload. */ private function autoLoad() { require_once $this->getLocalPath() . 'vendor/autoload.php'; } private function compile() { $containerBuilder = new ContainerBuilder(); $locator = new FileLocator($this->getLocalPath() . 'config'); $loader = new YamlFileLoader($containerBuilder, $locator); $loader->load('config.yml'); $containerBuilder->compile(); $this->moduleContainer = $containerBuilder; } public function hookActionAdminControllerSetMedia($params) { $currentController = Tools::getValue('controller'); if (Config::isPrestashopVersionBelow174()) { /** @var \Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs $moduleTabs */ $moduleTabs = $this->getService(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::class); $visibleClasses = $moduleTabs->getTabsClassNames(false); if (in_array($currentController, $visibleClasses, true)) { Media::addJsDef(['visibleTabs' => $moduleTabs->getTabsClassNames(true)]); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/tabsHandlerBelowPs174.js'); } } if ('AdminOrders' === $currentController) { $this->handleLabelPrintService(); $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/order-list.css'); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/order_list.js'); Media::addJsDef( [ 'dpdHookAjaxShipmentController' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_AJAX_SHIPMENTS_CONTROLLER), 'shipmentIsBeingPrintedMessage' => $this->context->smarty->fetch($this->getLocalPath() . 'views/templates/admin/partials/spinner.tpl') . $this->l('Your labels are being saved please stay on the page'), 'noOrdersSelectedMessage' => $this->l('No orders were selected'), 'downloadSelectedLabelsButton' => $this->context->smarty->fetch($this->getLocalPath() . 'views/templates/admin/partials/download-selected-labels-button.tpl') ] ); $orderId = Tools::getValue('id_order'); $shipment = $this->getShipment($orderId); $baseUrl = $this->context->shop->getBaseURL(true, false); $isAbove177 = Config::isPrestashopVersionAbove177(); $baseUrlAdmin = $isAbove177 ? $this->context->link->getAdminBaseLink() : null; /** @var \Invertus\dpdBaltics\Service\Label\LabelUrlFormatter $labelUrlService */ $labelUrlService = $isAbove177 ? $this->getModuleContainer('invertus.dpdbaltics.service.label.label_url_formatter') : null; Media::addJsDef( [ 'print_url' => $labelUrlService && $isAbove177 ? $baseUrl.$labelUrlService->formatJsLabelPrintUrl() : null, 'print_and_save_label_url' => $labelUrlService && $isAbove177 ? $baseUrl.$labelUrlService->formatJsLabelSaveAndPrintUrl() : null, 'shipment' => $shipment, 'id_order' => $orderId, 'is_label_download_option' => Configuration::get(Config::LABEL_PRINT_OPTION) === 'download', 'is_ps_above_177' => Config::isPrestashopVersionAbove177(), 'loader_url' => $isAbove177 ? "{$baseUrlAdmin}modules/dpdbaltics/views/templates/admin/loader/loader.html" : null ] ); } if ('AdminOrders' === $currentController && (Tools::isSubmit('vieworder') || Tools::getValue('action') === 'vieworder') ) { $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/order_expand_form.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/shipment.js'); Media::addJsDef([ 'expandText' => $this->l('Expand'), 'collapseText' => $this->l('Collapse'), 'dpdAjaxShipmentsUrl' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_AJAX_SHIPMENTS_CONTROLLER), 'dpdMessages' => [ 'invalidProductQuantity' => $this->l('Invalid product quantity entered'), 'invalidShipment' => $this->l('Invalid shipment selected'), 'parcelsLimitReached' => $this->l('Parcels limit reached in shipment'), 'successProductMove' => $this->l('Product moved successfully'), 'successCreation' => $this->l('Successful creation'), 'unexpectedError' => $this->l('Unexpected error appeared.'), 'invalidPrintoutFormat' => $this->l('Invalid printout format selected.'), 'cannotOpenWindow' => $this->l('Cannot print label, your browser may be blocking it.'), 'dpdRecipientAddressError' => $this->l('Please fill required fields') ], 'id_language' => $this->context->language->id, 'id_shop' => $this->context->shop->id, 'id_cart' => $this->context->cart->id, 'currentController' => $currentController, ]); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/carrier_phone.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/custom_select.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/label_position.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/pudo.js'); $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/customSelect/custom-select.css'); $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/admin-orders-controller.css'); $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/quantity-max-value-tip.css'); } if ('AdminOrders' === $currentController && (Tools::isSubmit('addorder') || Tools::getValue('action') === 'addorder') ) { /** @var ProductRepository $productRepo */ $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); Media::addJsDef([ 'dpdFrontController' => false, 'pudoCarriers' => json_encode($productRepo->getPudoProducts()), 'currentController' => $currentController, 'dpdAjaxShipmentsUrl' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_AJAX_SHIPMENTS_CONTROLLER), 'ignoreAdminController' => true, 'dpdAjaxPudoUrl' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_PUDO_AJAX_CONTROLLER), 'id_shop' => $this->context->shop->id, ]); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/pudo_list.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/carrier_phone.js'); $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/pudo.js'); $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/admin-orders-controller.css'); return; } } public function hookDisplayAdminOrder(array $params) { return Config::isPrestashopVersionAbove177() ? false : $this->displayInAdminOrderPage($params); } public function hookDisplayAdminOrderTabContent(array $params) { return !Config::isPrestashopVersionAbove177() ? false : $this->displayInAdminOrderPage($params); } private function displayInAdminOrderPage(array $params) { $order = new Order($params['id_order']); $cart = Cart::getCartByOrderId($params['id_order']); /** @var ProductRepository $productRepo */ $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); $carrier = new Carrier($order->id_carrier); if (!$productRepo->getProductIdByCarrierReference($carrier->id_reference)) { return; } $shipment = $this->getShipment($order->id); $dpdCodWarning = false; /** @var OrderService $orderService */ $orderService = $this->getModuleContainer('invertus.dpdbaltics.service.order_service'); $orderDetails = $orderService->getOrderDetails($order, $cart->id_lang); $customAddresses = []; /** @var ReceiverAddressRepository $receiverAddressRepository */ $receiverAddressRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.receiver_address_repository'); $customOrderAddressesIds = $receiverAddressRepository->getAddressIdByOrderId($order->id); foreach ($customOrderAddressesIds as $customOrderAddressId) { $customAddress = new Address($customOrderAddressId); $customAddresses[] = [ 'id_address' => $customAddress->id, 'alias' => $customAddress->alias ]; } $combinedCustomerAddresses = array_merge($orderDetails['customer']['addresses'], $customAddresses); /** @var PhonePrefixRepository $phonePrefixRepository */ $phonePrefixRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.phone_prefix_repository'); $products = $cart->getProducts(); /** @var LabelPositionService $labelPositionService */ $labelPositionService = $this->getModuleContainer('invertus.dpdbaltics.service.label.label_position_service'); $labelPositionService->assignLabelPositions($shipment->id); $labelPositionService->assignLabelFormat($shipment->id); /** @var PaymentService $paymentService */ $paymentService = $this->getModuleContainer('invertus.dpdbaltics.service.payment.payment_service'); try { $isCodPayment = $paymentService->isOrderPaymentCod($order->module); } catch (Exception $e) { $tplVars = [ 'errorMessage' => $this->l('Something went wrong checking payment method'), ]; $this->context->smarty->assign($tplVars); return $this->context->smarty->fetch( $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' ); } /** @var PudoRepository $pudoRepo */ $pudoRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.pudo_repository'); $selectedProduct = new DPDProduct($shipment->id_service); $isPudo = $selectedProduct->is_pudo; $pudoId = $pudoRepo->getIdByCart($order->id_cart); $selectedPudo = new DPDPudo($pudoId); /** @var ParcelShopService $parcelShopService */ /** @var ParcelShopRepository $parcelShopRepo */ $parcelShopService= $this->getModuleContainer('invertus.dpdbaltics.service.parcel.parcel_shop_service'); $parcelShopRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.parcel_shop_repository'); /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); $countryCode = $currentCountryProvider->getCurrentCountryIsoCode($cart); $selectedCity = null; try { if ($pudoId) { $selectedCity = $selectedPudo->city; $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity( $countryCode, $selectedCity ); } else { $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity( $countryCode, $orderDetails['order_address']['city'] ); } } catch (Exception $e) { $tplVars = [ 'errorMessage' => $this->l('Fatal error while searching for parcel shops: ') . $e->getMessage(), ]; $this->context->smarty->assign($tplVars); return $this->context->smarty->fetch( $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' ); } /** @var null|\Invertus\dpdBalticsApi\Api\DTO\Object\ParcelShop $selectedPudoService */ $selectedPudoService = null; $hasParcelShops = false; if ($parcelShops) { if ($selectedPudo->pudo_id) { $selectedPudoService = $parcelShopService->getParcelShopByShopId($selectedPudo->pudo_id)[0]; } else { $selectedPudoService = $parcelShops[0]; } $hasParcelShops = true; } if ($selectedPudoService) { /** @var PudoService $pudoService */ $pudoService = $this->getModuleContainer('invertus.dpdbaltics.service.pudo_service'); $selectedPudoService->setOpeningHours( $pudoService->formatPudoServiceWorkHours($selectedPudoService->getOpeningHours()) ); } /** @var ProductRepository $productRepository */ $productRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); $dpdProducts = $productRepository->getAllProducts(); $dpdCarrierOptions = []; /** @var DPDProduct $dpdProduct */ foreach ($dpdProducts as $dpdProduct) { $dpdCarrierOptions[] = [ 'id_dpd_product' => $dpdProduct->id_dpd_product, 'name' => $dpdProduct->name, 'available' => $dpdProduct->active && (int) $dpdProduct->is_cod === (int) $isCodPayment && (!$dpdProduct->is_pudo && $hasParcelShops) ]; } $cityList = $parcelShopRepo->getAllCitiesByCountryCode($countryCode); if (\Invertus\dpdBaltics\Config\Config::productHasDeliveryTime($selectedProduct->product_reference)) { /** @var \Invertus\dpdBaltics\Repository\OrderDeliveryTimeRepository $orderDeliveryTimeRepo */ $orderDeliveryTimeRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.order_delivery_time_repository'); $orderDeliveryTimeId = $orderDeliveryTimeRepo->getOrderDeliveryIdByCartId($cart->id); if ($orderDeliveryTimeId) { $orderDeliveryTime = new DPDOrderDeliveryTime($orderDeliveryTimeId); $this->context->smarty->assign([ 'orderDeliveryTime' => $orderDeliveryTime->delivery_time, 'deliveryTimes' => \Invertus\dpdBaltics\Config\Config::getDeliveryTimes($countryCode) ]); } } $href = $this->context->link->getModuleLink( $this->name, 'ShipmentReturn', [ 'id_order' => $order->id, 'dpd-return-submit' => '' ] ); $tplVars = [ 'dpdLogoUrl' => $this->getPathUri() . 'views/img/DPDLogo.gif', 'shipment' => $shipment, 'isAbove177' => Config::isPrestashopVersionAbove177(), 'testOrder' => $shipment->is_test, 'total_products' => 1, 'contractPageLink' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_PRODUCTS_CONTROLLER), 'dpdCodWarning' => $dpdCodWarning, 'testMode' => Configuration::get(Config::SHIPMENT_TEST_MODE), 'printLabelOption' => Configuration::get(Config::LABEL_PRINT_OPTION), 'defaultLabelFormat' => Configuration::get(Config::DEFAULT_LABEL_FORMAT), 'combinedAddresses' => $combinedCustomerAddresses, 'orderDetails' => $orderDetails, 'mobilePhoneCodeList' => $phonePrefixRepository->getCallPrefixes(), 'products' => $products, 'dpdProducts' => $dpdCarrierOptions, 'isCodPayment' => $isCodPayment, 'is_pudo' => (bool)$isPudo, 'selectedPudo' => $selectedPudoService, 'city_list' => $cityList, 'selected_city' => $selectedCity, 'has_parcel_shops' => $hasParcelShops, 'receiverAddressCountries' => Country::getCountries($this->context->language->id, true), 'documentReturnEnabled' => Configuration::get(Config::DOCUMENT_RETURN), 'href' => $href, 'adminLabelLink' => $this->context->link->getAdminLink( 'AdminDPDBalticsAjaxShipments', true, [], ['action' => 'print-return'] ), 'isAutomated' => Configuration::get(Config::AUTOMATED_PARCEL_RETURN), ]; $this->context->smarty->assign($tplVars); return $this->context->smarty->fetch( $this->getLocalPath() . 'views/templates/hook/admin/admin-order.tpl' ); } public function hookActionValidateOrder($params) { $carrier = new Carrier($params['order']->id_carrier); if ($carrier->external_module_name !== $this->name) { return; } $isAdminOrderPage = 'AdminOrders' === Tools::getValue('controller') || Config::isPrestashopVersionAbove177(); $isAdminNewOrderForm = Tools::isSubmit('addorder') || Tools::isSubmit('cart_summary'); if ($isAdminOrderPage && $isAdminNewOrderForm) { $dpdPhone = Tools::getValue('dpd-phone'); $dpdPhoneArea = Tools::getValue('dpd-phone-area'); /** @var \Invertus\dpdBaltics\Service\OrderDeliveryTimeService $orderDeliveryService */ $carrierPhoneService = $this->getModuleContainer('invertus.dpdbaltics.service.carrier_phone_service'); if (!empty($dpdPhone) && !empty($dpdPhoneArea)) { try { $carrierPhoneService->saveCarrierPhone( $this->context->cart->id, $dpdPhone, $dpdPhoneArea ); } catch (Exception $exception) { if ($exception->getCode() === Config::ERROR_COULD_NOT_SAVE_PHONE_NUMBER) { $error = $this->l('Phone data is not saved'); die($error); } } } /** @var CarrierPhoneService $carrierPhoneService */ $orderDeliveryService = $this->getModuleContainer('invertus.dpdbaltics.service.order_delivery_time_service'); $deliveryTime = Tools::getValue('dpd-delivery-time'); if ($deliveryTime !== null) { if (!$orderDeliveryService->saveDeliveryTime( $this->context->cart->id, $deliveryTime )) { $error = $this->l('Delivery time is not saved'); die($error); }; } } /** @var ShipmentService $shipmentService */ $shipmentService = $this->getModuleContainer('invertus.dpdbaltics.service.shipment_service'); $shipmentService->createShipmentFromOrder($params['order']); } public function printLabel($idShipment) { /** @var \Invertus\dpdBaltics\Service\LabelPrintingService $labelPrintingService */ $labelPrintingService = $this->getModuleContainer('invertus.dpdbaltics.service.label_printing_service'); $parcelPrintResponse = $labelPrintingService->printOne($idShipment); if ($parcelPrintResponse->getStatus() === Config::API_SUCCESS_STATUS) { $this->updateOrderCarrier($idShipment); return; } return $parcelPrintResponse; } public function printMultipleLabels($shipmentIds) { /** @var \Invertus\dpdBaltics\Service\LabelPrintingService $labelPrintingService */ $labelPrintingService = $this->getModuleContainer('invertus.dpdbaltics.service.label_printing_service'); $parcelPrintResponse = $labelPrintingService->printMultiple($shipmentIds); if ($parcelPrintResponse->getStatus() === Config::API_SUCCESS_STATUS) { foreach ($shipmentIds as $shipmentId) { $this->updateOrderCarrier($shipmentId); } return; } return $parcelPrintResponse; } private function updateOrderCarrier($shipmentId) { $shipment = new DPDShipment($shipmentId); /** @var OrderRepository $orderRepo */ /** @var TrackingService $trackingService */ $orderRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.order_repository'); $trackingService = $this->getModuleContainer('invertus.dpdbaltics.service.tracking_service'); $orderCarrierId = $orderRepo->getOrderCarrierId($shipment->id_order); $orderCarrier = new OrderCarrier($orderCarrierId); $orderCarrier->tracking_number = $trackingService->getTrackingNumber($shipment->pl_number); try { $orderCarrier->update(); } catch (Exception $e) { Context::getContext()->controller->errors[] = $this->l('Failed to save tracking number: ') . $e->getMessage(); return; } $shipment->printed_label = 1; $shipment->date_print = date('Y-m-d H:i:s'); $shipment->update(); } public function hookDisplayOrderDetail($params) { $isReturnServiceEnabled = Configuration::get(Config::PARCEL_RETURN); if (!$isReturnServiceEnabled) { return; } if (CountryUtility::isEstonia()) { return; } /** @var ShipmentRepository $shipmentRepo */ $shipmentRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.shipment_repository'); $shipmentId = $shipmentRepo->getIdByOrderId($params['order']->id); $orderState = new OrderState($params['order']->current_state); if (!$orderState->delivery) { return; } /** @var AddressTemplateRepository $addressTemplateRepo */ $addressTemplateRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.address_template_repository'); $returnAddressTemplates = $addressTemplateRepo->getReturnServiceAddressTemplates(); $shipment = new DPDShipment($shipmentId); if (!$returnAddressTemplates) { return; } $showTemplates = false; if (sizeof($returnAddressTemplates) > 1 && !$shipment->return_pl_number) { $showTemplates = true; } if (isset($this->context->cookie->dpd_error)) { $this->context->controller->errors[] = json_decode($this->context->cookie->dpd_error); unset($this->context->cookie->dpd_error); } $href = $this->context->link->getModuleLink( $this->name, 'ShipmentReturn', [ 'id_order' => $params['order']->id, 'dpd-return-submit' => '' ] ); $this->context->smarty->assign( [ 'href' => $href, 'return_template_ids' => $returnAddressTemplates, 'show_template' => $showTemplates, ] ); $html = $this->context->smarty->fetch( $this->getLocalPath() . 'views/templates/hook/front/order-detail.tpl' ); return $html; } public function hookActionAdminOrdersListingFieldsModifier($params) { if ((bool) Configuration::get(Config::HIDE_ORDERS_LABEL_PRINT_BUTTON)) { return false; } if (isset($params['select'])) { $params['select'] .= ' ,ds.`id_order` AS id_order_shipment '; } if (isset($params['join'])) { $params['join'] .= ' LEFT JOIN `' . _DB_PREFIX_ . 'dpd_shipment` ds ON ds.`id_order` = a.`id_order` '; } $params['fields']['id_order_shipment'] = [ 'title' => $this->l('DPD Label'), 'align' => 'text-center', 'class' => 'fixed-width-xs', 'orderby' => false, 'search' => false, 'remove_onclick' => true, 'callback_object' => 'dpdbaltics', 'callback' => 'returnOrderListIcon' ]; } /** * Callback function, it has to be static so can't call $this, so have to reload dpdBaltics module inside the function * @param $idOrder * @return string * @throws Exception */ public static function returnOrderListIcon($orderId) { $dpdBaltics = Module::getInstanceByName('dpdbaltics'); $dpdBaltics->context->smarty->assign('idOrder', $orderId); $dpdBaltics->context->smarty->assign( 'message', $dpdBaltics->l('Print label(s) from DPD system. Once label is saved you won\'t be able to modify contents of shipments') ); $icon = $dpdBaltics->context->smarty->fetch( $dpdBaltics->getLocalPath() . 'views/templates/hook/admin/order-list-save-label-icon.tpl' ); $dpdBaltics->context->smarty->assign('icon', $icon); return $dpdBaltics->context->smarty->fetch($dpdBaltics->getLocalPath() . 'views/templates/hook/admin/order-list-icon-container.tpl'); } public function hookDisplayAdminListBefore() { if ($this->context->controller instanceof AdminOrdersControllerCore) { return $this->context->smarty->fetch($this->getLocalPath() . 'views/templates/hook/admin/admin-orders-header-hook.tpl'); } } public function hookActionOrderGridDefinitionModifier(array $params) { if (!Config::isPrestashopVersionAbove177()) { return false; } $definition = $params['definition']; if (!(bool) Configuration::get(Config::HIDE_ORDERS_LABEL_PRINT_BUTTON)) { $definition->getColumns() ->addAfter( 'date_add', (new ActionColumn('dpd_print_label')) ->setName($this->l('Dpd print label')) ->setOptions([ 'actions' => $this->getGridAction() ]) ); } $definition->getBulkActions() ->add( (new SubmitBulkActionCustom('print_multiple_labels')) ->setName($this->l('Print multiple labels')) ->setOptions([ 'submit_route' => 'dpdbaltics_save_and_download_printed_labels_order_list_multiple', ]) ) ; } /** * @return RowActionCollection */ private function getGridAction() { return (new RowActionCollection()) ->add( (new LinkRowActionCustom('print_delivery_slip')) ->setName($this->l('Print label(s) from DPD system. Once label is saved you won\'t be able to modify contents of shipments')) ->setIcon('print') ->setOptions([ 'route' => 'dpdbaltics_save_and_download_printed_label_order_list', 'route_param_name' => 'orderId', 'route_param_field' => 'id_order', 'is_label_download' => Configuration::get(Config::LABEL_PRINT_OPTION) === 'download', 'confirm_message' => $this->l('Would you like to print shipping label?'), 'accessibility_checker' => $this->getModuleContainer()->get('invertus.dpdbaltics.grid.row.print_accessibility_checker'), ]) ); } private function getShipment($idOrder) { if (!$idOrder) { return false; } /** @var ShipmentRepository $shipmentRepository */ $shipmentRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.shipment_repository'); $shipmentId = $shipmentRepository->getIdByOrderId($idOrder); $shipment = new DPDShipment($shipmentId); if (!Validate::isLoadedObject($shipment)) { return false; } return $shipment; } private function handleLabelPrintService() { if (Tools::isSubmit('print_label')) { $idShipment = Tools::getValue('id_dpd_shipment'); try { $parcelPrintResponse = $this->printLabel($idShipment); } catch (DPDBalticsAPIException $e) { /** @var ExceptionService $exceptionService */ $exceptionService = $this->getModuleContainer('invertus.dpdbaltics.service.exception.exception_service'); Context::getContext()->controller->errors[] = $exceptionService->getErrorMessageForException( $e, $exceptionService->getAPIErrorMessages() ); return; } catch (Exception $e) { Context::getContext()->controller->errors[] = $this->l('Failed to print label: ') . $e->getMessage(); return; } if (!empty($parcelPrintResponse->getErrLog())) { Context::getContext()->controller->errors[] = $parcelPrintResponse->getErrLog(); } exit; } if (Tools::isSubmit('print_multiple_labels')) { $shipmentIds = json_decode(Tools::getValue('shipment_ids')); try { $parcelPrintResponse = $this->printMultipleLabels($shipmentIds); } catch (DPDBalticsAPIException $e) { /** @var ExceptionService $exceptionService */ $exceptionService = $this->getModuleContainer('invertus.dpdbaltics.service.exception.exception_service'); Context::getContext()->controller->errors[] = $exceptionService->getErrorMessageForException( $e, $exceptionService->getAPIErrorMessages() ); return; } catch (Exception $e) { Context::getContext()->controller->errors[] = $this->l('Failed to print label: ') . $e->getMessage(); return; } if (!empty($parcelPrintResponse->getErrLog())) { Context::getContext()->controller->errors[] = $parcelPrintResponse->getErrLog(); } exit; } } } \ No newline at end of file + + * @copyright Since 2007 PrestaShop SA and Contributors + * @license https://opensource.org/licenses/AFL-3.0 Academic Free License version 3.0 + */ + +use Invertus\dpdBaltics\Grid\Row\PrintAccessibilityChecker; +use Invertus\dpdBaltics\Builder\Template\Front\CarrierOptionsBuilder; +use Invertus\dpdBaltics\Config\Config; +use Invertus\dpdBaltics\Controller\AbstractAdminController; +use Invertus\dpdBaltics\Grid\LinkRowActionCustom; +use Invertus\dpdBaltics\Grid\SubmitBulkActionCustom; +use invertus\dpdbaltics\install\installer; +use Invertus\dpdBaltics\Logger\Logger; +use Invertus\dpdBaltics\OnBoard\Service\OnBoardService; +use Invertus\dpdBaltics\Repository\AddressTemplateRepository; +use Invertus\dpdBaltics\Repository\OrderRepository; +use Invertus\dpdBaltics\Repository\ParcelShopRepository; +use Invertus\dpdBaltics\Repository\PhonePrefixRepository; +use Invertus\dpdBaltics\Repository\PriceRuleRepository; +use Invertus\dpdBaltics\Repository\ProductRepository; +use Invertus\dpdBaltics\Repository\PudoRepository; +use Invertus\dpdBaltics\Repository\ReceiverAddressRepository; +use Invertus\dpdBaltics\Repository\ShipmentRepository; +use Invertus\dpdBaltics\Repository\ZoneRepository; +use Invertus\dpdBaltics\Service\API\LabelApiService; +use Invertus\dpdBaltics\Service\API\ParcelShopSearchApiService; +use Invertus\dpdBaltics\Service\CarrierPhoneService; +use Invertus\dpdBaltics\Service\Exception\ExceptionService; +use Invertus\dpdBaltics\Service\GoogleApiService; +use Invertus\dpdBaltics\Service\Label\LabelPositionService; +use Invertus\dpdBaltics\Service\OrderService; +use Invertus\dpdBaltics\Service\Parcel\ParcelShopService; +use Invertus\dpdBaltics\Service\Payment\PaymentService; +use Invertus\dpdBaltics\Service\PriceRuleService; +use Invertus\dpdBaltics\Service\PudoService; +use Invertus\dpdBaltics\Service\ShipmentService; +use Invertus\dpdBaltics\Service\ShippingPriceCalculationService; +use Invertus\dpdBaltics\Service\TabService; +use Invertus\dpdBaltics\Service\TrackingService; +use Invertus\dpdBaltics\Util\CountryUtility; +use Invertus\dpdBaltics\Validate\Carrier\PudoValidate; +use Invertus\dpdBalticsApi\Api\DTO\Response\ParcelPrintResponse; +use Invertus\dpdBalticsApi\Api\DTO\Response\ParcelShopSearchResponse; +use Invertus\dpdBalticsApi\Exception\DPDBalticsAPIException; +use Invertus\dpdBalticsApi\Factory\SerializerFactory; +use PrestaShop\PrestaShop\Core\Grid\Action\Bulk\Type\SubmitBulkAction; +use PrestaShop\PrestaShop\Core\Grid\Action\Row\RowActionCollection; +use PrestaShop\PrestaShop\Core\Grid\Action\Row\Type\LinkRowAction; +use PrestaShop\PrestaShop\Core\Grid\Column\Type\Common\ActionColumn; +use Symfony\Component\Config\ConfigCache; +use Symfony\Component\Config\FileLocator; +use Symfony\Component\Console\Application; +use Symfony\Component\DependencyInjection\ContainerBuilder; +use Symfony\Component\DependencyInjection\Dumper\PhpDumper; +use Symfony\Component\DependencyInjection\Loader\YamlFileLoader; + +class DPDBaltics extends CarrierModule +{ + /** + * Symfony DI Container + **/ + private $moduleContainer; + + /** + * Prestashop fills this property automatically with selected carrier ID in FO checkout + * + * @var int $id_carrier + */ + public $id_carrier; + + + public function __construct() + { + $this->name = 'dpdbaltics'; + $this->displayName = $this->l('DPDBaltics'); + $this->author = 'Invertus'; + $this->tab = 'shipping_logistics'; + $this->description = 'DPD Baltics shipping integration'; + $this->version = '3.2.19'; + $this->ps_versions_compliancy = ['min' => '1.7.1.0', 'max' => _PS_VERSION_]; + $this->need_instance = 0; + parent::__construct(); + + $this->autoLoad(); + $this->compile(); + } + + public function install() + { + if (!parent::install()) { + return false; + } + + /** @var \Invertus\dpdBaltics\Infrastructure\Bootstrap\Install\Installer $installer */ + $installer = $this->getService(\Invertus\dpdBaltics\Infrastructure\Bootstrap\Install\Installer::class); + + try { + $installer->init(); + + return true; + } catch (\Invertus\dpdBaltics\Infrastructure\Bootstrap\Exception\CouldNotInstallModule $exception) { + $this->_errors[] = $exception->getMessage(); + + return false; + } + } + + public function uninstall() + { + $uninstall = parent::uninstall(); + + if (!$uninstall) { + return false; + } + + /** @var \Invertus\dpdBaltics\Infrastructure\Bootstrap\Uninstall\Uninstaller $uninstaller */ + $uninstaller = $this->getService(\Invertus\dpdBaltics\Infrastructure\Bootstrap\Uninstall\Uninstaller::class); + + try { + $uninstaller->init(); + + return true; + } catch (\Invertus\dpdBaltics\Infrastructure\Bootstrap\Exception\CouldNotUninstallModule $exception) { + $this->_errors[] = $exception->getMessage(); + + return false; + } + } + + public function getContent() + { + Tools::redirectAdmin($this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_SETTINGS_CONTROLLER)); + } + + public function getService($serviceName) + { + return $this->getModuleContainer()->get($serviceName); + } + + /** + * @return mixed + */ + public function getModuleContainer($id = false) + { + if ($id) { + return $this->moduleContainer->get($id); + } + + return $this->moduleContainer; + } + + public function hookActionFrontControllerSetMedia() + { + //TODO fillup this array when more modules are compatible with OPC + $onePageCheckoutControllers = ['supercheckout', 'onepagecheckoutps', 'thecheckout']; + $applicableControlelrs = ['order', 'order-opc', 'ShipmentReturn', 'supercheckout']; + $currentController = !empty($this->context->controller->php_self) ? $this->context->controller->php_self : Tools::getValue('controller'); + + if ('product' === $currentController) { + $this->context->controller->registerStylesheet( + 'dpdbaltics-product-carriers.', + 'modules/' . $this->name . '/views/css/front/product-carriers.css', + [ + 'media' => 'all', + 'position' => 150 + ] + ); + } + + /** @var \Invertus\dpdBaltics\Validate\Compatibility\OpcModuleCompatibilityValidator $opcModuleCompatibilityValidator */ + $opcModuleCompatibilityValidator = $this->getModuleContainer('invertus.dpdbaltics.validator.opc_module_compatibility_validator'); + + if (in_array($currentController, $onePageCheckoutControllers, true) || $opcModuleCompatibilityValidator->isOpcModuleInUse()) { + $this->context->controller->addJqueryPlugin('chosen'); + + $this->context->controller->registerJavascript( + 'dpdbaltics-opc', + 'modules/' . $this->name . '/views/js/front/order-opc.js', + [ + 'position' => 'bottom', + 'priority' => 130 + ] + ); + + $this->context->controller->registerJavascript( + 'dpdbaltics-supercheckout', + 'modules/' . $this->name . '/views/js/front/modules/supercheckout.js', + [ + 'position' => 'bottom', + 'priority' => 130 + ] + ); + + $this->context->controller->registerStylesheet( + 'dpdbaltics-opc', + 'modules/' . $this->name . '/views/css/front/onepagecheckout.css', + [ + 'position' => 'bottom', + 'priority' => 130 + ] + ); + + Media::addJsDef([ + 'dpdbaltics' => [ + 'isOnePageCheckout' => 1 + ] + ]); + } + + /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ + $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); + $webServiceCountryCode = Configuration::get(Config::WEB_SERVICE_COUNTRY); + $carrierIds = []; + $baseUrl = $this->context->shop->getBaseURL(true, false); + + if ($webServiceCountryCode === Config::LATVIA_ISO_CODE || $currentController === 'supercheckout') { + /** @var ProductRepository $productRepo */ + $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); + + $dpdProductReferences = $productRepo->getAllActiveDpdProductReferences(); + + foreach ($dpdProductReferences as $reference) { + $carrier = Carrier::getCarrierByReference($reference['id_reference']); + if (Validate::isLoadedObject($carrier)) { + $carrierIds[] = $carrier->id; + } + } + } + + Media::addJsDef([ + 'lapinas_img' => $baseUrl . $this->getPathUri() . 'views/img/lapinas.png', + 'lapinas_text' => $this->l('Sustainable'), + 'dpd_carrier_ids' => $carrierIds + ]); + if (in_array($currentController, $applicableControlelrs, true)) { + Media::addJsDef([ + 'select_an_option_translatable' => $this->l('Select an Option'), + 'select_an_option_multiple_translatable' => $this->l('Select Some Options'), + 'no_results_translatable' => $this->l('No results match'), + ]); + $this->context->controller->addJS($this->getPathUri() . 'views/js/front/order.js'); + $this->context->controller->addJS($this->getPathUri() . 'views/js/front/order-input.js'); + $this->context->controller->addJS($this->getPathUri() . 'views/js/front/sustainable-logo.js'); + $this->context->controller->addCSS($this->getPathUri() . 'views/css/front/order-input.css'); + $this->context->controller->addCSS($this->getPathUri() . 'views/css/front/sustainable-logo.css'); + /** @var PaymentService $paymentService */ + $paymentService = $this->getModuleContainer('invertus.dpdbaltics.service.payment.payment_service'); + $isPickupMap = Configuration::get(\Invertus\dpdBaltics\Config\Config::PICKUP_MAP); + $cart = Context::getContext()->cart; + $paymentService->filterPaymentMethods($cart); + $paymentService->filterPaymentMethodsByCod($cart); + + /** @var ProductRepository $productRepo */ + $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); + Media::addJsDef([ + 'pudoCarriers' => json_encode($productRepo->getPudoProducts()), + 'currentController' => $currentController, + 'is_pickup_map' => $isPickupMap, + 'id_language' => $this->context->language->id, + 'id_shop' => $this->context->shop->id, + 'dpdAjaxLoaderPath' => $this->getPathUri() . 'views/img/ajax-loader-big.gif', + 'dpdPickupMarkerPath' => $this->getPathUri() . 'views/img/dpd-pick-up.png', + 'dpdLockerMarkerPath' => $this->getPathUri() . 'views/img/locker.png', + 'dpdHookAjaxUrl' => $this->context->link->getModuleLink($this->name, 'Ajax'), + 'pudoSelectSuccess' => $this->l('Pick-up point selected'), + 'dpd_carrier_ids' => $carrierIds, + ]); + + $this->context->controller->registerStylesheet( + 'dpdbaltics-pudo-shipment', + 'modules/' . $this->name . '/views/css/front/' . 'pudo-shipment.css', + [ + 'media' => 'all', + 'position' => 150 + ] + ); + if ($isPickupMap) { + /** @var GoogleApiService $googleApiService */ + $googleApiService = $this->getModuleContainer('invertus.dpdbaltics.service.google_api_service'); + $this->context->controller->registerJavascript( + 'dpdbaltics-google-api', + $googleApiService->getFormattedGoogleMapsUrl(), + [ + 'server' => 'remote' + ] + ); + } + $this->context->controller->registerJavascript( + 'dpdbaltics-pudo', + 'modules/' . $this->name . '/views/js/front/pudo.js', + [ + 'position' => 'bottom', + 'priority' => 130 + ] + ); + $this->context->controller->registerJavascript( + 'dpdbaltics-pudo-search', + 'modules/' . $this->name . '/views/js/front/pudo-search.js', + [ + 'position' => 'bottom', + 'priority' => 130 + ] + ); + } + } + + public function hookActionValidateStepComplete(&$params) + { + if ('delivery' !== $params['step_name']) { + return; + } + + /** @var Cart $cart */ + $cart = $params['cart']; + $carrier = new Carrier($cart->id_carrier); + $idShop = $this->context->shop->id; + + /** @var Invertus\dpdBaltics\Repository\CarrierRepository $carrierRepo */ + /** @var Invertus\dpdBaltics\Repository\ProductRepository $productRepo */ + $carrierRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.carrier_repository'); + $productRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.product_repository'); + + $carrierReference = $carrier->id_reference; + $dpdCarriers = $carrierRepo->getDpdCarriers($idShop); + $isDpdCarrier = false; + foreach ($dpdCarriers as $dpdCarrier) { + if ($carrierReference == $dpdCarrier['id_reference']) { + $isDpdCarrier = true; + $productId = $productRepo->getProductIdByCarrierReference($carrier->id_reference); + $product = new DPDProduct($productId); + $isSameDayDelivery = $product->product_reference === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY; + break; + } + } + if (!$isDpdCarrier) { + return true; + } + + if ($isSameDayDelivery) { + /** @var PudoRepository $pudoRepo */ + $pudoRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.pudo_repository'); + $pudoId = $pudoRepo->getIdByCart($cart->id); + $selectedPudo = new DPDPudo($pudoId); + if ($selectedPudo->city !== Config::SAME_DAY_DELIVERY_CITY) { + $this->context->controller->errors[] = + $this->l('This carrier can\'t deliver to your selected city'); + $params['completed'] = false; + $selectedPudo->delete(); + + return; + } + } + + //NOTE: thecheckout triggers this hook without phone parameters the phone is saved with ajax request + if (Tools::getValue('module') === 'thecheckout') { + return; + } + + if (!Tools::getValue('dpd-phone')) { + $this->context->controller->errors[] = + $this->l('In order to use DPD Carrier you need to enter phone number'); + $params['completed'] = false; + + return; + } + + if (!Tools::getValue('dpd-phone-area')) { + $this->context->controller->errors[] = + $this->l('In order to use DPD Carrier you need to enter phone area'); + $params['completed'] = false; + + return; + } + + /** @var CarrierPhoneService $carrierPhoneService */ + $carrierPhoneService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.carrier_phone_service'); + + try { + $carrierPhoneService->saveCarrierPhone( + $this->context->cart->id, + Tools::getValue('dpd-phone'), + Tools::getValue('dpd-phone-area') + ); + } catch (Exception $exception) { + if ($exception->getCode() === Config::ERROR_COULD_NOT_SAVE_PHONE_NUMBER) { + $this->context->controller->errors[] = $this->l('Phone data is not saved'); + $params['completed'] = false; + } + } + /** @var \Invertus\dpdBaltics\Service\OrderDeliveryTimeService $orderDeliveryService */ + $orderDeliveryService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.order_delivery_time_service'); + + $deliveryTime = Tools::getValue('dpd-delivery-time'); + if ($deliveryTime) { + if (!$orderDeliveryService->saveDeliveryTime( + $this->context->cart->id, + $deliveryTime + )) { + $this->context->controller->errors[] = $this->l('Delivery time data is not saved'); + $params['completed'] = false; + }; + } + + /** @var Cart $cart */ + $cart = $params['cart']; + $carrier = new Carrier($cart->id_carrier); + /** @var PudoValidate $pudoValidator */ + $pudoValidator = $this->getModuleContainer('invertus.dpdbaltics.validate.carrier.pudo_validate'); + if (!$pudoValidator->validatePickupPoints($cart->id, $carrier->id_reference)) { + $carrier = new Carrier($cart->id_carrier, $this->context->language->id); + $this->context->controller->errors[] = + sprintf($this->l('Please select pickup point for carrier: %s.'), $carrier->name); + + $params['completed'] = false; + } + } + + public function hookDisplayProductPriceBlock($params) + { + if ($params['type'] == 'after_price') { + /** @var CarrierOptionsBuilder $carrierOptionsBuilder */ + $carrierOptionsBuilder = $this->getModuleContainer()->get('invertus.dpdbaltics.builder.template.front.carrier_options_builder'); + + return $carrierOptionsBuilder->renderCarrierOptionsInProductPage(); + } + } + + public function hookDisplayBackOfficeTop() + { + if ($this->context->controller instanceof AbstractAdminController && + Configuration::get(Config::ON_BOARD_TURNED_ON) && + Configuration::get(Config::ON_BOARD_STEP) + ) { + /** @var OnBoardService $onBoardService */ + $onBoardService = $this->getModuleContainer('invertus.dpdbaltics.on_board.service.on_board_service'); + return $onBoardService->makeStepActionWithTemplateReturn(); + } + } + + public function getOrderShippingCost($cart, $shippingCost) + { + return $this->getOrderShippingCostExternal($cart); + } + + /** + * @param $cart Cart + * @return bool|float|string + * @throws PrestaShopDatabaseException + * @throws PrestaShopException + * @throws SmartyException + */ + public function getOrderShippingCostExternal($cart) + { + // This method is still called when module is disabled so we need to do a manual check here + if (!$this->active) { + return false; + } + + if ($this->context->controller->ajax && Tools::getValue('id_address_delivery')) { + $cart->id_address_delivery = (int)Tools::getValue('id_address_delivery'); + } + + /** @var ZoneRepository $zoneRepository */ + /** @var ProductRepository $productRepo */ + /** @var \Invertus\dpdBaltics\Service\Product\ProductAvailabilityService $productAvailabilityService */ + /** @var \Invertus\dpdBaltics\Validate\Weight\CartWeightValidator $cartWeightValidator */ + /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ + $zoneRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.zone_repository'); + $productRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.product_repository'); + $productAvailabilityService = $this->getModuleContainer('invertus.dpdbaltics.service.product.product_availability_service'); + $cartWeightValidator = $this->getModuleContainer('invertus.dpdbaltics.validate.weight.cart_weight_validator'); + $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); + + $deliveryAddress = new Address($cart->id_address_delivery); + + if (empty($zoneRepository->findZoneInRangeByAddress($deliveryAddress))) { + return false; + } + + $carrier = new Carrier($this->id_carrier); + + if (!$productAvailabilityService->checkIfCarrierIsAvailable($carrier->id_reference)) { + return false; + } + + if (!$productRepo->checkIfCarrierIsAvailableInCountry((int) $carrier->id_reference, (int) $deliveryAddress->id_country) + ) { + return false; + } + + try { + $isCarrierAvailableInShop = $productRepo->checkIfCarrierIsAvailableInShop($carrier->id_reference, $this->context->shop->id); + if (empty($isCarrierAvailableInShop)) { + return false; + } + + $serviceCarrier = $productRepo->findProductByCarrierReference($carrier->id_reference); + } catch (Exception $e) { + $tplVars = [ + 'errorMessage' => $this->l('Something went wrong while collecting DPD carrier data'), + ]; + $this->context->smarty->assign($tplVars); + + return $this->context->smarty->fetch( + $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' + ); + } + + if ((bool)$serviceCarrier['is_home_collection']) { + return false; + } + + $countryCode = $currentCountryProvider->getCurrentCountryIsoCode($cart); + + $parcelDistribution = \Configuration::get(Config::PARCEL_DISTRIBUTION); + $maxAllowedWeight = Config::getDefaultServiceWeights($countryCode, $serviceCarrier['product_reference']); + + if (!$cartWeightValidator->validate($cart, $parcelDistribution, $maxAllowedWeight)) { + return false; + } + + if ($serviceCarrier['product_reference'] === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY) { + $isSameDayAvailable = \Invertus\dpdBaltics\Util\ProductUtility::validateSameDayDelivery( + $countryCode, + $deliveryAddress->city + ); + + if (!$isSameDayAvailable) { + return false; + } + } + + /** @var ShippingPriceCalculationService $shippingPriceCalculationService */ + $shippingPriceCalculationService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.shipping_price_calculation_service'); + + return $shippingPriceCalculationService->calculate($cart, $carrier, $deliveryAddress); + } + + public function hookDisplayCarrierExtraContent($params) + { + /** @var Cart $cart */ + $cart = $params['cart']; + $carrier = new Carrier($params['carrier']['id']); + + /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ + $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); + $countryCode = $currentCountryProvider->getCurrentCountryIsoCode($cart); + + $deliveryAddress = new Address($cart->id_address_delivery); + + /** @var CarrierPhoneService $carrierPhoneService */ + /** @var \Invertus\dpdBaltics\Presenter\DeliveryTimePresenter $deliveryTimePresenter */ + /** @var ProductRepository $productRepo */ + $carrierPhoneService = $this->getModuleContainer()->get('invertus.dpdbaltics.service.carrier_phone_service'); + $deliveryTimePresenter = $this->getModuleContainer()->get('invertus.dpdbaltics.presenter.delivery_time_presenter'); + $productRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.product_repository'); + + $productId = $productRepo->getProductIdByCarrierReference($carrier->id_reference); + $dpdProduct = new DPDProduct($productId); + $return = ''; + if ($dpdProduct->getProductReference() === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY) { + /** @var \Invertus\dpdBaltics\Presenter\SameDayDeliveryMessagePresenter $sameDayDeliveryPresenter */ + $sameDayDeliveryPresenter = $this->getModuleContainer()->get('invertus.dpdbaltics.presenter.same_day_delivery_message_presenter'); + $return .= $sameDayDeliveryPresenter->getSameDayDeliveryMessageTemplate(); + } + $return .= $carrierPhoneService->getCarrierPhoneTemplate($this->context->cart->id, $carrier->id_reference); + if ($dpdProduct->getProductReference() === Config::PRODUCT_TYPE_B2B || + $dpdProduct->getProductReference() === Config::PRODUCT_TYPE_B2B_COD + ) { + $return .= $deliveryTimePresenter->getDeliveryTimeTemplate($countryCode, $deliveryAddress->city); + } + + /** @var ProductRepository $productRep */ + $productRep = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); + $isPudo = $productRep->isProductPudo($carrier->id_reference); + if ($isPudo) { + /** @var PudoRepository $pudoRepo */ + /** @var ParcelShopRepository $parcelShopRepo */ + /** @var ProductRepository $productRepo */ + $pudoRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.pudo_repository'); + $parcelShopRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.parcel_shop_repository'); + $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); + $product = $productRepo->findProductByCarrierReference($carrier->id_reference); + $isSameDayDelivery = ($product['product_reference'] === Config::PRODUCT_TYPE_SAME_DAY_DELIVERY); + + $pudoId = $pudoRepo->getIdByCart($cart->id); + $selectedPudo = new DPDPudo($pudoId); + + /** @var ParcelShopService $parcelShopService */ + $parcelShopService= $this->getModuleContainer('invertus.dpdbaltics.service.parcel.parcel_shop_service'); + + $selectedCity = null; + $selectedStreet = null; + + try { + if (Validate::isLoadedObject($selectedPudo) && !$isSameDayDelivery) { + $selectedCity = $selectedPudo->city; + $selectedStreet = $selectedPudo->street; + $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity($countryCode, $selectedCity); + $parcelShops = $parcelShopService->moveSelectedShopToFirst($parcelShops, $selectedStreet); + } else { + $selectedCity = $deliveryAddress->city; + $selectedStreet = $deliveryAddress->address1; + $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity($countryCode, $selectedCity); + $parcelShops = $parcelShopService->moveSelectedShopToFirst($parcelShops, $selectedStreet); + if (!$parcelShops) { + $selectedCity = null; + } + } + } catch (DPDBalticsAPIException $e) { + /** @var ExceptionService $exceptionService */ + $exceptionService = $this->getModuleContainer('invertus.dpdbaltics.service.exception.exception_service'); + $tplVars = [ + 'errorMessage' => $exceptionService->getErrorMessageForException( + $e, + $exceptionService->getAPIErrorMessages() + ) + ]; + $this->context->smarty->assign($tplVars); + + return $this->context->smarty->fetch( + $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' + ); + } catch (Exception $e) { + $tplVars = [ + 'errorMessage' => $this->l("Something went wrong. We couldn't find parcel shops."), + ]; + $this->context->smarty->assign($tplVars); + + return $this->context->smarty->fetch( + $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' + ); + } + + /** @var PudoService $pudoService */ + $pudoService = $this->getModuleContainer('invertus.dpdbaltics.service.pudo_service'); + + $pudoServices = $pudoService->setPudoServiceTypes($parcelShops); + $pudoServices = $pudoService->formatPudoServicesWorkHours($pudoServices); + + if (isset($parcelShops[0])) { + $coordinates = [ + 'lat' => $parcelShops[0]->getLatitude(), + 'lng' => $parcelShops[0]->getLongitude(), + ]; + $this->context->smarty->assign( + [ + 'coordinates' => $coordinates + ] + ); + } + + if ($isSameDayDelivery) { + $cityList['Rīga'] = 'Rīga'; + } else { + $cityList = $parcelShopRepo->getAllCitiesByCountryCode($countryCode); + } + + if (!in_array($selectedCity, $cityList) && isset($parcelShops[0])) { + $selectedCity = $parcelShops[0]->getCity(); + } + + if (!$selectedCity && Tools::getValue('controller') !== 'order') { + $tplVars = [ + 'displayMessage' => true, + 'messages' => [$this->l("Your delivery address city is not in a list of pickup cities, please select closest pickup point city below manually")], + 'messageType_pudo' => 'danger' + + ]; + $this->context->smarty->assign($tplVars); + } + + $streetList = $parcelShopRepo->getAllAddressesByCountryCodeAndCity($countryCode, $selectedCity); + $this->context->smarty->assign( + [ + 'currentController' => Tools::getValue('controller'), + 'carrierId' => $carrier->id, + 'pickUpMap' => Configuration::get(Config::PICKUP_MAP), + 'pudoId' => $pudoId, + 'pudoServices' => $pudoServices, + 'dpd_pickup_logo' => $this->getPathUri() . 'views/img/pickup.png', + 'dpd_locker_logo' => $this->getPathUri() . 'views/img/locker.png', + 'delivery_address' => $deliveryAddress, + 'saved_pudo_id' => $selectedPudo->pudo_id, + 'is_pudo' => (bool)$isPudo, + 'city_list' => $cityList, + 'selected_city' => $selectedCity, + 'show_shop_list' => Configuration::get(Config::PARCEL_SHOP_DISPLAY), + 'street_list' => $streetList, + 'selected_street' => $selectedStreet, + ] + ); + + $return .= $this->context->smarty->fetch( + $this->getLocalPath() . '/views/templates/hook/front/pudo-points.tpl' + ); + } + + return $return; + } + + /** + * Includes Vendor Autoload. + */ + private function autoLoad() + { + require_once $this->getLocalPath() . 'vendor/autoload.php'; + } + + private function compile() + { + $containerBuilder = new ContainerBuilder(); + $locator = new FileLocator($this->getLocalPath() . 'config'); + $loader = new YamlFileLoader($containerBuilder, $locator); + $loader->load('config.yml'); + $containerBuilder->compile(); + + $this->moduleContainer = $containerBuilder; + } + + public function hookActionAdminControllerSetMedia($params) + { + $currentController = Tools::getValue('controller'); + + if (Config::isPrestashopVersionBelow174()) { + /** @var \Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs $moduleTabs */ + $moduleTabs = $this->getService(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::class); + $visibleClasses = $moduleTabs->getTabsClassNames(false); + + if (in_array($currentController, $visibleClasses, true)) { + Media::addJsDef(['visibleTabs' => $moduleTabs->getTabsClassNames(true)]); + $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/tabsHandlerBelowPs174.js'); + } + } + + if ('AdminOrders' === $currentController) { + $this->handleLabelPrintService(); + + $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/order-list.css'); + $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/order_list.js'); + Media::addJsDef( + [ + 'dpdHookAjaxShipmentController' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_AJAX_SHIPMENTS_CONTROLLER), + 'shipmentIsBeingPrintedMessage' => $this->context->smarty->fetch($this->getLocalPath() . 'views/templates/admin/partials/spinner.tpl') . + $this->l('Your labels are being saved please stay on the page'), + 'noOrdersSelectedMessage' => $this->l('No orders were selected'), + 'downloadSelectedLabelsButton' => $this->context->smarty->fetch($this->getLocalPath() . 'views/templates/admin/partials/download-selected-labels-button.tpl') + ] + ); + + $orderId = Tools::getValue('id_order'); + $shipment = $this->getShipment($orderId); + $baseUrl = $this->context->shop->getBaseURL(true, false); + $isAbove177 = Config::isPrestashopVersionAbove177(); + $baseUrlAdmin = $isAbove177 ? $this->context->link->getAdminBaseLink() : null; + /** @var \Invertus\dpdBaltics\Service\Label\LabelUrlFormatter $labelUrlService */ + $labelUrlService = $isAbove177 ? $this->getModuleContainer('invertus.dpdbaltics.service.label.label_url_formatter') : null; + + Media::addJsDef( + [ + 'print_url' => $labelUrlService && $isAbove177 ? $baseUrl.$labelUrlService->formatJsLabelPrintUrl() : null, + 'print_and_save_label_url' => $labelUrlService && $isAbove177 ? $baseUrl.$labelUrlService->formatJsLabelSaveAndPrintUrl() : null, + 'shipment' => $shipment, + 'id_order' => $orderId, + 'is_label_download_option' => Configuration::get(Config::LABEL_PRINT_OPTION) === 'download', + 'is_ps_above_177' => Config::isPrestashopVersionAbove177(), + 'loader_url' => $isAbove177 ? "{$baseUrlAdmin}modules/dpdbaltics/views/templates/admin/loader/loader.html" : null + ] + ); + } + + if ('AdminOrders' === $currentController && + (Tools::isSubmit('vieworder') || Tools::getValue('action') === 'vieworder') + ) { + $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/order_expand_form.js'); + $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/shipment.js'); + Media::addJsDef([ + 'expandText' => $this->l('Expand'), + 'collapseText' => $this->l('Collapse'), + 'dpdAjaxShipmentsUrl' => + $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_AJAX_SHIPMENTS_CONTROLLER), + 'dpdMessages' => [ + 'invalidProductQuantity' => $this->l('Invalid product quantity entered'), + 'invalidShipment' => $this->l('Invalid shipment selected'), + 'parcelsLimitReached' => $this->l('Parcels limit reached in shipment'), + 'successProductMove' => $this->l('Product moved successfully'), + 'successCreation' => $this->l('Successful creation'), + 'unexpectedError' => $this->l('Unexpected error appeared.'), + 'invalidPrintoutFormat' => $this->l('Invalid printout format selected.'), + 'cannotOpenWindow' => $this->l('Cannot print label, your browser may be blocking it.'), + 'dpdRecipientAddressError' => $this->l('Please fill required fields') + ], + 'id_language' => $this->context->language->id, + 'id_shop' => $this->context->shop->id, + 'id_cart' => $this->context->cart->id, + 'currentController' => $currentController, + + ]); + + $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/carrier_phone.js'); + $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/custom_select.js'); + $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/label_position.js'); + $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/pudo.js'); + $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/customSelect/custom-select.css'); + $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/admin-orders-controller.css'); + $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/quantity-max-value-tip.css'); + } + if ('AdminOrders' === $currentController && + (Tools::isSubmit('addorder') || Tools::getValue('action') === 'addorder') + ) { + /** @var ProductRepository $productRepo */ + $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); + + Media::addJsDef([ + 'dpdFrontController' => false, + 'pudoCarriers' => json_encode($productRepo->getPudoProducts()), + 'currentController' => $currentController, + 'dpdAjaxShipmentsUrl' => + $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_AJAX_SHIPMENTS_CONTROLLER), + 'ignoreAdminController' => true, + 'dpdAjaxPudoUrl' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_PUDO_AJAX_CONTROLLER), + 'id_shop' => $this->context->shop->id, + ]); + + $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/pudo_list.js'); + $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/carrier_phone.js'); + $this->context->controller->addJS($this->getPathUri() . 'views/js/admin/pudo.js'); + + $this->context->controller->addCSS($this->getPathUri() . 'views/css/admin/order/admin-orders-controller.css'); + + return; + } + } + + public function hookDisplayAdminOrder(array $params) + { + return Config::isPrestashopVersionAbove177() ? false : $this->displayInAdminOrderPage($params); + } + + public function hookDisplayAdminOrderTabContent(array $params) + { + return !Config::isPrestashopVersionAbove177() ? false : $this->displayInAdminOrderPage($params); + } + + private function displayInAdminOrderPage(array $params) + { + $order = new Order($params['id_order']); + $cart = Cart::getCartByOrderId($params['id_order']); + + /** @var ProductRepository $productRepo */ + $productRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); + $carrier = new Carrier($order->id_carrier); + if (!$productRepo->getProductIdByCarrierReference($carrier->id_reference)) { + return; + } + + $shipment = $this->getShipment($order->id); + $dpdCodWarning = false; + + /** @var OrderService $orderService */ + $orderService = $this->getModuleContainer('invertus.dpdbaltics.service.order_service'); + $orderDetails = $orderService->getOrderDetails($order, $cart->id_lang); + + $customAddresses = []; + + /** @var ReceiverAddressRepository $receiverAddressRepository */ + $receiverAddressRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.receiver_address_repository'); + + $customOrderAddressesIds = $receiverAddressRepository->getAddressIdByOrderId($order->id); + foreach ($customOrderAddressesIds as $customOrderAddressId) { + $customAddress = new Address($customOrderAddressId); + + $customAddresses[] = [ + 'id_address' => $customAddress->id, + 'alias' => $customAddress->alias + ]; + } + $combinedCustomerAddresses = array_merge($orderDetails['customer']['addresses'], $customAddresses); + + /** @var PhonePrefixRepository $phonePrefixRepository */ + $phonePrefixRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.phone_prefix_repository'); + + $products = $cart->getProducts(); + + /** @var LabelPositionService $labelPositionService */ + $labelPositionService = $this->getModuleContainer('invertus.dpdbaltics.service.label.label_position_service'); + $labelPositionService->assignLabelPositions($shipment->id); + $labelPositionService->assignLabelFormat($shipment->id); + + /** @var PaymentService $paymentService */ + $paymentService = $this->getModuleContainer('invertus.dpdbaltics.service.payment.payment_service'); + try { + $isCodPayment = $paymentService->isOrderPaymentCod($order->module); + } catch (Exception $e) { + $tplVars = [ + 'errorMessage' => $this->l('Something went wrong checking payment method'), + ]; + $this->context->smarty->assign($tplVars); + + return $this->context->smarty->fetch( + $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' + ); + } + + /** @var PudoRepository $pudoRepo */ + $pudoRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.pudo_repository'); + $selectedProduct = new DPDProduct($shipment->id_service); + $isPudo = $selectedProduct->is_pudo; + $pudoId = $pudoRepo->getIdByCart($order->id_cart); + + $selectedPudo = new DPDPudo($pudoId); + + /** @var ParcelShopService $parcelShopService */ + /** @var ParcelShopRepository $parcelShopRepo */ + $parcelShopService= $this->getModuleContainer('invertus.dpdbaltics.service.parcel.parcel_shop_service'); + $parcelShopRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.parcel_shop_repository'); + + /** @var \Invertus\dpdBaltics\Provider\CurrentCountryProvider $currentCountryProvider */ + $currentCountryProvider = $this->getModuleContainer('invertus.dpdbaltics.provider.current_country_provider'); + $countryCode = $currentCountryProvider->getCurrentCountryIsoCode($cart); + + $selectedCity = null; + try { + if ($pudoId) { + $selectedCity = $selectedPudo->city; + $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity( + $countryCode, + $selectedCity + ); + } else { + $parcelShops = $parcelShopService->getParcelShopsByCountryAndCity( + $countryCode, + $orderDetails['order_address']['city'] + ); + } + } catch (Exception $e) { + $tplVars = [ + 'errorMessage' => $this->l('Fatal error while searching for parcel shops: ') . $e->getMessage(), + ]; + $this->context->smarty->assign($tplVars); + + return $this->context->smarty->fetch( + $this->getLocalPath() . 'views/templates/admin/dpd-shipment-fatal-error.tpl' + ); + } + + /** @var null|\Invertus\dpdBalticsApi\Api\DTO\Object\ParcelShop $selectedPudoService */ + $selectedPudoService = null; + $hasParcelShops = false; + if ($parcelShops) { + if ($selectedPudo->pudo_id) { + $selectedPudoService = $parcelShopService->getParcelShopByShopId($selectedPudo->pudo_id)[0]; + } else { + $selectedPudoService = $parcelShops[0]; + } + $hasParcelShops = true; + } + + if ($selectedPudoService) { + /** @var PudoService $pudoService */ + $pudoService = $this->getModuleContainer('invertus.dpdbaltics.service.pudo_service'); + + $selectedPudoService->setOpeningHours( + $pudoService->formatPudoServiceWorkHours($selectedPudoService->getOpeningHours()) + ); + } + + /** @var ProductRepository $productRepository */ + $productRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.product_repository'); + $dpdProducts = $productRepository->getAllProducts(); + + $dpdCarrierOptions = []; + + /** @var DPDProduct $dpdProduct */ + foreach ($dpdProducts as $dpdProduct) { + $dpdCarrierOptions[] = [ + 'id_dpd_product' => $dpdProduct->id_dpd_product, + 'name' => $dpdProduct->name, + 'available' => + $dpdProduct->active && + (int) $dpdProduct->is_cod === (int) $isCodPayment && + (!$dpdProduct->is_pudo && $hasParcelShops) + ]; + } + + $cityList = $parcelShopRepo->getAllCitiesByCountryCode($countryCode); + + if (\Invertus\dpdBaltics\Config\Config::productHasDeliveryTime($selectedProduct->product_reference)) { + /** @var \Invertus\dpdBaltics\Repository\OrderDeliveryTimeRepository $orderDeliveryTimeRepo */ + $orderDeliveryTimeRepo = $this->getModuleContainer()->get('invertus.dpdbaltics.repository.order_delivery_time_repository'); + $orderDeliveryTimeId = $orderDeliveryTimeRepo->getOrderDeliveryIdByCartId($cart->id); + if ($orderDeliveryTimeId) { + $orderDeliveryTime = new DPDOrderDeliveryTime($orderDeliveryTimeId); + $this->context->smarty->assign([ + 'orderDeliveryTime' => $orderDeliveryTime->delivery_time, + 'deliveryTimes' => \Invertus\dpdBaltics\Config\Config::getDeliveryTimes($countryCode) + ]); + } + } + + $href = $this->context->link->getModuleLink( + $this->name, + 'ShipmentReturn', + [ + 'id_order' => $order->id, + 'dpd-return-submit' => '' + ] + ); + + $tplVars = [ + 'dpdLogoUrl' => $this->getPathUri() . 'views/img/DPDLogo.gif', + 'shipment' => $shipment, + 'isAbove177' => Config::isPrestashopVersionAbove177(), + 'testOrder' => $shipment->is_test, + 'total_products' => 1, + 'contractPageLink' => $this->context->link->getAdminLink(\Invertus\dpdBaltics\Infrastructure\Bootstrap\ModuleTabs::ADMIN_PRODUCTS_CONTROLLER), + 'dpdCodWarning' => $dpdCodWarning, + 'testMode' => Configuration::get(Config::SHIPMENT_TEST_MODE), + 'printLabelOption' => Configuration::get(Config::LABEL_PRINT_OPTION), + 'defaultLabelFormat' => Configuration::get(Config::DEFAULT_LABEL_FORMAT), + 'combinedAddresses' => $combinedCustomerAddresses, + 'orderDetails' => $orderDetails, + 'mobilePhoneCodeList' => $phonePrefixRepository->getCallPrefixes(), + 'products' => $products, + 'dpdProducts' => $dpdCarrierOptions, + 'isCodPayment' => $isCodPayment, + 'is_pudo' => (bool)$isPudo, + 'selectedPudo' => $selectedPudoService, + 'city_list' => $cityList, + 'selected_city' => $selectedCity, + 'has_parcel_shops' => $hasParcelShops, + 'receiverAddressCountries' => Country::getCountries($this->context->language->id, true), + 'documentReturnEnabled' => Configuration::get(Config::DOCUMENT_RETURN), + 'href' => $href, + 'adminLabelLink' => $this->context->link->getAdminLink( + 'AdminDPDBalticsAjaxShipments', + true, + [], + ['action' => 'print-return'] + ), + 'isAutomated' => Configuration::get(Config::AUTOMATED_PARCEL_RETURN), + ]; + + $this->context->smarty->assign($tplVars); + + return $this->context->smarty->fetch( + $this->getLocalPath() . 'views/templates/hook/admin/admin-order.tpl' + ); + } + + public function hookActionValidateOrder($params) + { + $carrier = new Carrier($params['order']->id_carrier); + if ($carrier->external_module_name !== $this->name) { + return; + } + + $isAdminOrderPage = 'AdminOrders' === Tools::getValue('controller') || Config::isPrestashopVersionAbove177(); + $isAdminNewOrderForm = Tools::isSubmit('addorder') || Tools::isSubmit('cart_summary'); + + if ($isAdminOrderPage && $isAdminNewOrderForm) { + $dpdPhone = Tools::getValue('dpd-phone'); + $dpdPhoneArea = Tools::getValue('dpd-phone-area'); + + /** @var \Invertus\dpdBaltics\Service\OrderDeliveryTimeService $orderDeliveryService */ + $carrierPhoneService = $this->getModuleContainer('invertus.dpdbaltics.service.carrier_phone_service'); + + if (!empty($dpdPhone) && !empty($dpdPhoneArea)) { + try { + $carrierPhoneService->saveCarrierPhone( + $this->context->cart->id, + $dpdPhone, + $dpdPhoneArea + ); + } catch (Exception $exception) { + if ($exception->getCode() === Config::ERROR_COULD_NOT_SAVE_PHONE_NUMBER) { + $error = $this->l('Phone data is not saved'); + die($error); + } + } + } + + /** @var CarrierPhoneService $carrierPhoneService */ + $orderDeliveryService = $this->getModuleContainer('invertus.dpdbaltics.service.order_delivery_time_service'); + $deliveryTime = Tools::getValue('dpd-delivery-time'); + if ($deliveryTime !== null) { + if (!$orderDeliveryService->saveDeliveryTime( + $this->context->cart->id, + $deliveryTime + )) { + $error = $this->l('Delivery time is not saved'); + die($error); + }; + } + } + + /** @var ShipmentService $shipmentService */ + $shipmentService = $this->getModuleContainer('invertus.dpdbaltics.service.shipment_service'); + $shipmentService->createShipmentFromOrder($params['order']); + } + + public function printLabel($idShipment) + { + /** @var \Invertus\dpdBaltics\Service\LabelPrintingService $labelPrintingService */ + $labelPrintingService = $this->getModuleContainer('invertus.dpdbaltics.service.label_printing_service'); + + $parcelPrintResponse = $labelPrintingService->printOne($idShipment); + + if ($parcelPrintResponse->getStatus() === Config::API_SUCCESS_STATUS) { + $this->updateOrderCarrier($idShipment); + return; + } + + return $parcelPrintResponse; + } + + public function printMultipleLabels($shipmentIds) + { + /** @var \Invertus\dpdBaltics\Service\LabelPrintingService $labelPrintingService */ + $labelPrintingService = $this->getModuleContainer('invertus.dpdbaltics.service.label_printing_service'); + + $parcelPrintResponse = $labelPrintingService->printMultiple($shipmentIds); + + if ($parcelPrintResponse->getStatus() === Config::API_SUCCESS_STATUS) { + foreach ($shipmentIds as $shipmentId) { + $this->updateOrderCarrier($shipmentId); + } + return; + } + + return $parcelPrintResponse; + } + + private function updateOrderCarrier($shipmentId) + { + $shipment = new DPDShipment($shipmentId); + /** @var OrderRepository $orderRepo */ + /** @var TrackingService $trackingService */ + $orderRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.order_repository'); + $trackingService = $this->getModuleContainer('invertus.dpdbaltics.service.tracking_service'); + $orderCarrierId = $orderRepo->getOrderCarrierId($shipment->id_order); + + $orderCarrier = new OrderCarrier($orderCarrierId); + $orderCarrier->tracking_number = $trackingService->getTrackingNumber($shipment->pl_number); + + try { + $orderCarrier->update(); + } catch (Exception $e) { + Context::getContext()->controller->errors[] = + $this->l('Failed to save tracking number: ') . $e->getMessage(); + return; + } + + $shipment->printed_label = 1; + $shipment->date_print = date('Y-m-d H:i:s'); + $shipment->update(); + } + + public function hookDisplayOrderDetail($params) + { + $isReturnServiceEnabled = Configuration::get(Config::PARCEL_RETURN); + if (!$isReturnServiceEnabled) { + return; + } + + if (CountryUtility::isEstonia()) { + return; + } + + /** @var ShipmentRepository $shipmentRepo */ + $shipmentRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.shipment_repository'); + $shipmentId = $shipmentRepo->getIdByOrderId($params['order']->id); + + $orderState = new OrderState($params['order']->current_state); + if (!$orderState->delivery) { + return; + } + /** @var AddressTemplateRepository $addressTemplateRepo */ + $addressTemplateRepo = $this->getModuleContainer('invertus.dpdbaltics.repository.address_template_repository'); + $returnAddressTemplates = $addressTemplateRepo->getReturnServiceAddressTemplates(); + + $shipment = new DPDShipment($shipmentId); + + if (!$returnAddressTemplates) { + return; + } + + $showTemplates = false; + if (sizeof($returnAddressTemplates) > 1 && !$shipment->return_pl_number) { + $showTemplates = true; + } + + if (isset($this->context->cookie->dpd_error)) { + $this->context->controller->errors[] = json_decode($this->context->cookie->dpd_error); + unset($this->context->cookie->dpd_error); + } + $href = $this->context->link->getModuleLink( + $this->name, + 'ShipmentReturn', + [ + 'id_order' => $params['order']->id, + 'dpd-return-submit' => '' + ] + ); + + $this->context->smarty->assign( + [ + 'href' => $href, + 'return_template_ids' => $returnAddressTemplates, + 'show_template' => $showTemplates, + ] + ); + $html = $this->context->smarty->fetch( + $this->getLocalPath() . 'views/templates/hook/front/order-detail.tpl' + ); + + return $html; + } + + public function hookActionAdminOrdersListingFieldsModifier($params) + { + if ((bool) Configuration::get(Config::HIDE_ORDERS_LABEL_PRINT_BUTTON)) { + return false; + } + + if (isset($params['select'])) { + $params['select'] .= ' ,ds.`id_order` AS id_order_shipment '; + } + + if (isset($params['join'])) { + $params['join'] .= ' LEFT JOIN `' . _DB_PREFIX_ . 'dpd_shipment` ds ON ds.`id_order` = a.`id_order` '; + } + + $params['fields']['id_order_shipment'] = [ + 'title' => $this->l('DPD Label'), + 'align' => 'text-center', + 'class' => 'fixed-width-xs', + 'orderby' => false, + 'search' => false, + 'remove_onclick' => true, + 'callback_object' => 'dpdbaltics', + 'callback' => 'returnOrderListIcon' + ]; + } + + /** + * Callback function, it has to be static so can't call $this, so have to reload dpdBaltics module inside the function + * @param $idOrder + * @return string + * @throws Exception + */ + public static function returnOrderListIcon($orderId) + { + $dpdBaltics = Module::getInstanceByName('dpdbaltics'); + + $dpdBaltics->context->smarty->assign('idOrder', $orderId); + + $dpdBaltics->context->smarty->assign( + 'message', + $dpdBaltics->l('Print label(s) from DPD system. Once label is saved you won\'t be able to modify contents of shipments') + ); + $icon = $dpdBaltics->context->smarty->fetch( + $dpdBaltics->getLocalPath() . 'views/templates/hook/admin/order-list-save-label-icon.tpl' + ); + + + $dpdBaltics->context->smarty->assign('icon', $icon); + + return $dpdBaltics->context->smarty->fetch($dpdBaltics->getLocalPath() . 'views/templates/hook/admin/order-list-icon-container.tpl'); + } + + + public function hookDisplayAdminListBefore() + { + if ($this->context->controller instanceof AdminOrdersControllerCore) { + return $this->context->smarty->fetch($this->getLocalPath() . 'views/templates/hook/admin/admin-orders-header-hook.tpl'); + } + } + + public function hookActionOrderGridDefinitionModifier(array $params) + { + if (!Config::isPrestashopVersionAbove177()) { + return false; + } + + $definition = $params['definition']; + + if (!(bool) Configuration::get(Config::HIDE_ORDERS_LABEL_PRINT_BUTTON)) { + $definition->getColumns() + ->addAfter( + 'date_add', + (new ActionColumn('dpd_print_label')) + ->setName($this->l('Dpd print label')) + ->setOptions([ + 'actions' => $this->getGridAction() + ]) + ); + } + + $definition->getBulkActions() + ->add( + (new SubmitBulkActionCustom('print_multiple_labels')) + ->setName($this->l('Print multiple labels')) + ->setOptions([ + 'submit_route' => 'dpdbaltics_save_and_download_printed_labels_order_list_multiple', + ]) + ) + ; + } + + /** + * @return RowActionCollection + */ + private function getGridAction() + { + return (new RowActionCollection()) + ->add( + (new LinkRowActionCustom('print_delivery_slip')) + ->setName($this->l('Print label(s) from DPD system. Once label is saved you won\'t be able to modify contents of shipments')) + ->setIcon('print') + ->setOptions([ + 'route' => 'dpdbaltics_save_and_download_printed_label_order_list', + 'route_param_name' => 'orderId', + 'route_param_field' => 'id_order', + 'is_label_download' => Configuration::get(Config::LABEL_PRINT_OPTION) === 'download', + 'confirm_message' => $this->l('Would you like to print shipping label?'), + 'accessibility_checker' => $this->getModuleContainer()->get('invertus.dpdbaltics.grid.row.print_accessibility_checker'), + ]) + ); + } + + private function getShipment($idOrder) + { + if (!$idOrder) { + return false; + } + /** @var ShipmentRepository $shipmentRepository */ + $shipmentRepository = $this->getModuleContainer('invertus.dpdbaltics.repository.shipment_repository'); + $shipmentId = $shipmentRepository->getIdByOrderId($idOrder); + $shipment = new DPDShipment($shipmentId); + + if (!Validate::isLoadedObject($shipment)) { + return false; + } + + return $shipment; + } + + private function handleLabelPrintService() + { + if (Tools::isSubmit('print_label')) { + $idShipment = Tools::getValue('id_dpd_shipment'); + + try { + $parcelPrintResponse = $this->printLabel($idShipment); + } catch (DPDBalticsAPIException $e) { + /** @var ExceptionService $exceptionService */ + $exceptionService = $this->getModuleContainer('invertus.dpdbaltics.service.exception.exception_service'); + Context::getContext()->controller->errors[] = $exceptionService->getErrorMessageForException( + $e, + $exceptionService->getAPIErrorMessages() + ); + return; + } catch (Exception $e) { + Context::getContext()->controller->errors[] = $this->l('Failed to print label: ') . $e->getMessage(); + return; + } + + if (!empty($parcelPrintResponse->getErrLog())) { + Context::getContext()->controller->errors[] = $parcelPrintResponse->getErrLog(); + } + + exit; + } + + if (Tools::isSubmit('print_multiple_labels')) { + $shipmentIds = json_decode(Tools::getValue('shipment_ids')); + + try { + $parcelPrintResponse = $this->printMultipleLabels($shipmentIds); + } catch (DPDBalticsAPIException $e) { + /** @var ExceptionService $exceptionService */ + $exceptionService = $this->getModuleContainer('invertus.dpdbaltics.service.exception.exception_service'); + Context::getContext()->controller->errors[] = $exceptionService->getErrorMessageForException( + $e, + $exceptionService->getAPIErrorMessages() + ); + return; + } catch (Exception $e) { + Context::getContext()->controller->errors[] = $this->l('Failed to print label: ') . $e->getMessage(); + return; + } + + if (!empty($parcelPrintResponse->getErrLog())) { + Context::getContext()->controller->errors[] = $parcelPrintResponse->getErrLog(); + } + + exit; + } + } +} diff --git a/views/js/front/order-opc.js b/views/js/front/order-opc.js index 974464bf..5589e21c 100644 --- a/views/js/front/order-opc.js +++ b/views/js/front/order-opc.js @@ -111,11 +111,10 @@ function DPDdisplayMessageOpc(parent, template) { // Module "onepagecheckoutps" compatibility $(document).on('opc-load-review:completed', function() { - $('.delivery-option.selected .carrier-extra-content').show(); handlePhoneNumber($('.dpd-phone-block')); }); // Then address is being modified $(document).on('thecheckout_Address_Modified', function () { handlePhoneNumber($('.dpd-phone-block')); -}); \ No newline at end of file +}); diff --git a/views/js/front/pudo.js b/views/js/front/pudo.js index e1137f44..4771eb1a 100644 --- a/views/js/front/pudo.js +++ b/views/js/front/pudo.js @@ -1,20 +1,20 @@ -/** - * Copyright since 2007 PrestaShop SA and Contributors - * PrestaShop is an International Registered Trademark & Property of PrestaShop SA - * - * NOTICE OF LICENSE - * - * This source file is subject to the Academic Free License version 3.0 - * that is bundled with this package in the file LICENSE.md. - * It is also available through the world-wide-web at this URL: - * https://opensource.org/licenses/AFL-3.0 - * If you did not receive a copy of the license and are unable to - * obtain it through the world-wide-web, please send an email - * to license@prestashop.com so we can send you a copy immediately. - * - * @author PrestaShop SA and Contributors - * @copyright Since 2007 PrestaShop SA and Contributors - * @license https://opensource.org/licenses/AFL-3.0 Academic Free License version 3.0 +/** + * Copyright since 2007 PrestaShop SA and Contributors + * PrestaShop is an International Registered Trademark & Property of PrestaShop SA + * + * NOTICE OF LICENSE + * + * This source file is subject to the Academic Free License version 3.0 + * that is bundled with this package in the file LICENSE.md. + * It is also available through the world-wide-web at this URL: + * https://opensource.org/licenses/AFL-3.0 + * If you did not receive a copy of the license and are unable to + * obtain it through the world-wide-web, please send an email + * to license@prestashop.com so we can send you a copy immediately. + * + * @author PrestaShop SA and Contributors + * @copyright Since 2007 PrestaShop SA and Contributors + * @license https://opensource.org/licenses/AFL-3.0 Academic Free License version 3.0 */ /** @@ -108,7 +108,7 @@ $(document).ready(function () { } //Js to open extra content failed from theme, doing it manually. - if (params.deliveryOption.length > 0 && !$(params.deliveryOption).next('.carrier-extra-content').is(':visible')) { + if (params.deliveryOption.length > 0 && !$(params.deliveryOption).next('.carrier-extra-content').is(':visible') && !dpdbaltics.isOnePageCheckout) { $('.carrier-extra-content').hide(); $(params.deliveryOption).next('.carrier-extra-content').slideDown(); } From 235508651db150bef5e9a7065578478d0d928de2 Mon Sep 17 00:00:00 2001 From: GytisZum Date: Fri, 5 Jul 2024 06:45:54 +0000 Subject: [PATCH 15/20] Automatic license addition applying --- views/js/front/pudo.js | 34 +++++++++++++++++----------------- 1 file changed, 17 insertions(+), 17 deletions(-) diff --git a/views/js/front/pudo.js b/views/js/front/pudo.js index 4771eb1a..b7d0debf 100644 --- a/views/js/front/pudo.js +++ b/views/js/front/pudo.js @@ -1,20 +1,20 @@ -/** - * Copyright since 2007 PrestaShop SA and Contributors - * PrestaShop is an International Registered Trademark & Property of PrestaShop SA - * - * NOTICE OF LICENSE - * - * This source file is subject to the Academic Free License version 3.0 - * that is bundled with this package in the file LICENSE.md. - * It is also available through the world-wide-web at this URL: - * https://opensource.org/licenses/AFL-3.0 - * If you did not receive a copy of the license and are unable to - * obtain it through the world-wide-web, please send an email - * to license@prestashop.com so we can send you a copy immediately. - * - * @author PrestaShop SA and Contributors - * @copyright Since 2007 PrestaShop SA and Contributors - * @license https://opensource.org/licenses/AFL-3.0 Academic Free License version 3.0 +/** + * Copyright since 2007 PrestaShop SA and Contributors + * PrestaShop is an International Registered Trademark & Property of PrestaShop SA + * + * NOTICE OF LICENSE + * + * This source file is subject to the Academic Free License version 3.0 + * that is bundled with this package in the file LICENSE.md. + * It is also available through the world-wide-web at this URL: + * https://opensource.org/licenses/AFL-3.0 + * If you did not receive a copy of the license and are unable to + * obtain it through the world-wide-web, please send an email + * to license@prestashop.com so we can send you a copy immediately. + * + * @author PrestaShop SA and Contributors + * @copyright Since 2007 PrestaShop SA and Contributors + * @license https://opensource.org/licenses/AFL-3.0 Academic Free License version 3.0 */ /** From 1aab04723d8e91e33cdac55122292f85fadd7d08 Mon Sep 17 00:00:00 2001 From: Gytautas Date: Fri, 5 Jul 2024 09:46:47 +0300 Subject: [PATCH 16/20] changed hardcoded value --- dpdbaltics.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/dpdbaltics.php b/dpdbaltics.php index d7a01429..12f757b1 100644 --- a/dpdbaltics.php +++ b/dpdbaltics.php @@ -217,7 +217,7 @@ public function hookActionFrontControllerSetMedia() Media::addJsDef([ 'dpdbaltics' => [ - 'isOnePageCheckout' => 1 + 'isOnePageCheckout' => $opcModuleCompatibilityValidator->isOpcModuleInUse() ] ]); } From 490369062f8a9d46a9e69fdcefd4e80d96b5f156 Mon Sep 17 00:00:00 2001 From: Justas Vaitkus Date: Fri, 5 Jul 2024 12:05:42 +0300 Subject: [PATCH 17/20] css changes to change input order --- views/css/front/onepagecheckout.css | 6 ++++++ views/templates/hook/front/partials/pudo-search-block.tpl | 6 +++--- 2 files changed, 9 insertions(+), 3 deletions(-) diff --git a/views/css/front/onepagecheckout.css b/views/css/front/onepagecheckout.css index 16b23208..5de29d0c 100644 --- a/views/css/front/onepagecheckout.css +++ b/views/css/front/onepagecheckout.css @@ -51,6 +51,7 @@ .chosen-container { width: 100% !important; + max-width: 290px; } .carrier-extra-content .chosen-select { @@ -60,4 +61,9 @@ .dpd-checkout-delivery-time--container { width: 100%; +} + +.search-block-container { + padding-left: 15px; + padding-right: 15px; } \ No newline at end of file diff --git a/views/templates/hook/front/partials/pudo-search-block.tpl b/views/templates/hook/front/partials/pudo-search-block.tpl index c230e3dc..3ada10d1 100644 --- a/views/templates/hook/front/partials/pudo-search-block.tpl +++ b/views/templates/hook/front/partials/pudo-search-block.tpl @@ -19,10 +19,10 @@
{if count($city_list) > 1} -
+
{/if} {if !empty($city_list)} @@ -40,7 +40,7 @@
-
+
{include file='module:dpdbaltics/views/templates/hook/front/partials/pudo-search-street.tpl'}
From 28a072811db8b54ab125e32bc799b73ab408e321 Mon Sep 17 00:00:00 2001 From: Gytautas Date: Fri, 5 Jul 2024 14:13:31 +0300 Subject: [PATCH 18/20] version bump and changelog added --- CHANGELOG.md | 5 ++++- dpdbaltics.php | 2 +- 2 files changed, 5 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index a1aeae13..d821b8e5 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -164,4 +164,7 @@ - Carrier availability in country - Phone input selections sorted by active countries in shop - Work hours pop up fix -- Numeric post code improvements \ No newline at end of file +- Numeric post code improvements + +## [3.2.20] +- One page checkout compatibility improvements \ No newline at end of file diff --git a/dpdbaltics.php b/dpdbaltics.php index 12f757b1..812579f8 100644 --- a/dpdbaltics.php +++ b/dpdbaltics.php @@ -91,7 +91,7 @@ public function __construct() $this->author = 'Invertus'; $this->tab = 'shipping_logistics'; $this->description = 'DPD Baltics shipping integration'; - $this->version = '3.2.19'; + $this->version = '3.2.20'; $this->ps_versions_compliancy = ['min' => '1.7.1.0', 'max' => _PS_VERSION_]; $this->need_instance = 0; parent::__construct(); From 451b3594e5e1d72b9d92d114d443fbd5d2f7206e Mon Sep 17 00:00:00 2001 From: Gytautas Date: Fri, 2 Aug 2024 10:22:43 +0300 Subject: [PATCH 19/20] changelog updated --- CHANGELOG.md | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index d821b8e5..d1f964d3 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -167,4 +167,6 @@ - Numeric post code improvements ## [3.2.20] -- One page checkout compatibility improvements \ No newline at end of file +- One page checkout compatibility improvements +- Multiple payment transaction and total price fix +- Carriers error handling improvements \ No newline at end of file From 4de97d52643ca2b542ea4eb017543355a0dee496 Mon Sep 17 00:00:00 2001 From: Gytautas Date: Thu, 26 Sep 2024 15:11:09 +0300 Subject: [PATCH 20/20] strlen and substr are not suitable for multibyte strings, which can lead to issues when dealing with certain character sets like UTF-8 --- src/Util/StringUtility.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Util/StringUtility.php b/src/Util/StringUtility.php index 72dce3d7..a949769e 100644 --- a/src/Util/StringUtility.php +++ b/src/Util/StringUtility.php @@ -34,7 +34,7 @@ public static function toLowerCase($string) } public static function trimString($message) { - return strlen($message) > self::REMARK_LIMIT ? substr($message,0,self::REMARK_LIMIT)."..." : $message; + return mb_strlen($message) > self::REMARK_LIMIT ? mb_substr($message, 0, self::REMARK_LIMIT) . "..." : $message; } public static function removeSpecialCharacters($string) {